mirror of
				https://github.com/langgenius/dify.git
				synced 2025-10-26 00:18:44 +00:00 
			
		
		
		
	 7709d9df20
			
		
	
	
		7709d9df20
		
			
		
	
	
	
	
		
			
			Co-authored-by: NFish <douxc512@gmail.com> Co-authored-by: zxhlyh <jasonapring2015@outlook.com> Co-authored-by: twwu <twwu@dify.ai> Co-authored-by: jZonG <jzongcode@gmail.com>
		
			
				
	
	
		
			306 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			306 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| 'use client'
 | |
| import {
 | |
|   Children,
 | |
|   createContext,
 | |
|   useContext,
 | |
|   useEffect,
 | |
|   useRef,
 | |
|   useState,
 | |
| } from 'react'
 | |
| import { Tab, TabList, TabPanel, TabPanels } from '@headlessui/react'
 | |
| import { Tag } from './tag'
 | |
| import classNames from '@/utils/classnames'
 | |
| import { writeTextToClipboard } from '@/utils/clipboard'
 | |
| 
 | |
| const languageNames = {
 | |
|   js: 'JavaScript',
 | |
|   ts: 'TypeScript',
 | |
|   javascript: 'JavaScript',
 | |
|   typescript: 'TypeScript',
 | |
|   php: 'PHP',
 | |
|   python: 'Python',
 | |
|   ruby: 'Ruby',
 | |
|   go: 'Go',
 | |
| } as { [key: string]: string }
 | |
| 
 | |
| type IChildrenProps = {
 | |
|   children: React.ReactNode
 | |
|   [key: string]: any
 | |
| }
 | |
| 
 | |
| function getPanelTitle({ className }: { className: string }) {
 | |
|   const language = className.split('-')[1]
 | |
|   return languageNames[language] ?? 'Code'
 | |
| }
 | |
| 
 | |
| function ClipboardIcon(props: any) {
 | |
|   return (
 | |
|     <svg viewBox="0 0 20 20" aria-hidden="true" {...props}>
 | |
|       <path
 | |
|         strokeWidth="0"
 | |
|         d="M5.5 13.5v-5a2 2 0 0 1 2-2l.447-.894A2 2 0 0 1 9.737 4.5h.527a2 2 0 0 1 1.789 1.106l.447.894a2 2 0 0 1 2 2v5a2 2 0 0 1-2 2h-5a2 2 0 0 1-2-2Z"
 | |
|       />
 | |
|       <path
 | |
|         fill="none"
 | |
|         strokeLinejoin="round"
 | |
|         d="M12.5 6.5a2 2 0 0 1 2 2v5a2 2 0 0 1-2 2h-5a2 2 0 0 1-2-2v-5a2 2 0 0 1 2-2m5 0-.447-.894a2 2 0 0 0-1.79-1.106h-.527a2 2 0 0 0-1.789 1.106L7.5 6.5m5 0-1 1h-3l-1-1"
 | |
|       />
 | |
|     </svg>
 | |
|   )
 | |
| }
 | |
| 
 | |
| function CopyButton({ code }: { code: string }) {
 | |
|   const [copyCount, setCopyCount] = useState(0)
 | |
|   const copied = copyCount > 0
 | |
| 
 | |
|   useEffect(() => {
 | |
|     if (copyCount > 0) {
 | |
|       const timeout = setTimeout(() => setCopyCount(0), 1000)
 | |
|       return () => {
 | |
|         clearTimeout(timeout)
 | |
|       }
 | |
|     }
 | |
|   }, [copyCount])
 | |
| 
 | |
|   return (
 | |
|     <button
 | |
|       type="button"
 | |
|       className={classNames(
 | |
|         'group/button absolute top-3.5 right-4 overflow-hidden rounded-full py-1 pl-2 pr-3 text-2xs font-medium opacity-0 backdrop-blur transition focus:opacity-100 group-hover:opacity-100',
 | |
|         copied
 | |
|           ? 'bg-emerald-400/10 ring-1 ring-inset ring-emerald-400/20'
 | |
|           : 'bg-white/5 hover:bg-white/7.5 dark:bg-white/2.5 dark:hover:bg-white/5',
 | |
|       )}
 | |
|       onClick={() => {
 | |
|         writeTextToClipboard(code).then(() => {
 | |
|           setCopyCount(count => count + 1)
 | |
|         })
 | |
|       }}
 | |
|     >
 | |
|       <span
 | |
|         aria-hidden={copied}
 | |
|         className={classNames(
 | |
|           'pointer-events-none flex items-center gap-0.5 text-zinc-400 transition duration-300',
 | |
|           copied && '-translate-y-1.5 opacity-0',
 | |
|         )}
 | |
|       >
 | |
|         <ClipboardIcon className="h-5 w-5 fill-zinc-500/20 stroke-zinc-500 transition-colors group-hover/button:stroke-zinc-400" />
 | |
|         Copy
 | |
|       </span>
 | |
|       <span
 | |
|         aria-hidden={!copied}
 | |
|         className={classNames(
 | |
|           'pointer-events-none absolute inset-0 flex items-center justify-center text-emerald-400 transition duration-300',
 | |
|           !copied && 'translate-y-1.5 opacity-0',
 | |
|         )}
 | |
|       >
 | |
|         Copied!
 | |
|       </span>
 | |
|     </button>
 | |
|   )
 | |
| }
 | |
| 
 | |
| function CodePanelHeader({ tag, label }: { tag: string; label: string }) {
 | |
|   if (!tag && !label)
 | |
|     return null
 | |
| 
 | |
|   return (
 | |
|     <div className="border-b-white/7.5 bg-white/2.5 dark:bg-white/1 flex h-9 items-center gap-2 border-y border-t-transparent bg-zinc-900 px-4 dark:border-b-white/5">
 | |
|       {tag && (
 | |
|         <div className="dark flex">
 | |
|           <Tag variant="small">{tag}</Tag>
 | |
|         </div>
 | |
|       )}
 | |
|       {tag && label && (
 | |
|         <span className="h-0.5 w-0.5 rounded-full bg-zinc-500" />
 | |
|       )}
 | |
|       {label && (
 | |
|         <span className="font-mono text-xs text-zinc-400">{label}</span>
 | |
|       )}
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| type ICodePanelProps = {
 | |
|   children: React.ReactNode
 | |
|   tag?: string
 | |
|   code?: string
 | |
|   label?: string
 | |
|   targetCode?: string
 | |
| }
 | |
| function CodePanel({ tag, label, code, children, targetCode }: ICodePanelProps) {
 | |
|   const child = Children.only(children)
 | |
| 
 | |
|   return (
 | |
|     <div className="dark:bg-white/2.5 group">
 | |
|       <CodePanelHeader
 | |
|         tag={child.props.tag ?? tag}
 | |
|         label={child.props.label ?? label}
 | |
|       />
 | |
|       <div className="relative">
 | |
|         {/* <pre className="p-4 overflow-x-auto text-xs text-white">{children}</pre> */}
 | |
|         {/* <CopyButton code={child.props.code ?? code} /> */}
 | |
|         {/* <CopyButton code={child.props.children.props.children} /> */}
 | |
|         <pre className="overflow-x-auto p-4 text-xs text-white">{targetCode || children}</pre>
 | |
|         <CopyButton code={targetCode || child.props.children.props.children} />
 | |
|       </div>
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| function CodeGroupHeader({ title, children, selectedIndex }: IChildrenProps) {
 | |
|   const hasTabs = Children.count(children) > 1
 | |
| 
 | |
|   if (!title && !hasTabs)
 | |
|     return null
 | |
| 
 | |
|   return (
 | |
|     <div className="flex min-h-[calc(theme(spacing.12)+1px)] flex-wrap items-start gap-x-4 border-b border-zinc-700 bg-zinc-800 px-4 dark:border-zinc-800 dark:bg-transparent">
 | |
|       {title && (
 | |
|         <h3 className="mr-auto pt-3 text-xs font-semibold text-white">
 | |
|           {title}
 | |
|         </h3>
 | |
|       )}
 | |
|       {hasTabs && (
 | |
|         <TabList className="-mb-px flex gap-4 text-xs font-medium">
 | |
|           {Children.map(children, (child, childIndex) => (
 | |
|             <Tab
 | |
|               className={classNames(
 | |
|                 'border-b py-3 transition focus:[&:not(:focus-visible)]:outline-none',
 | |
|                 childIndex === selectedIndex
 | |
|                   ? 'border-emerald-500 text-emerald-400'
 | |
|                   : 'border-transparent text-zinc-400 hover:text-zinc-300',
 | |
|               )}
 | |
|             >
 | |
|               {getPanelTitle(child.props.children.props)}
 | |
|             </Tab>
 | |
|           ))}
 | |
|         </TabList>
 | |
|       )}
 | |
|     </div>
 | |
|   )
 | |
| }
 | |
| 
 | |
| type ICodeGroupPanelsProps = {
 | |
|   children: React.ReactNode
 | |
|   [key: string]: any
 | |
| }
 | |
| function CodeGroupPanels({ children, targetCode, ...props }: ICodeGroupPanelsProps) {
 | |
|   const hasTabs = Children.count(children) > 1
 | |
| 
 | |
|   if (hasTabs) {
 | |
|     return (
 | |
|       <TabPanels>
 | |
|         {Children.map(children, child => (
 | |
|           <TabPanel>
 | |
|             <CodePanel {...props}>{child}</CodePanel>
 | |
|           </TabPanel>
 | |
|         ))}
 | |
|       </TabPanels>
 | |
|     )
 | |
|   }
 | |
| 
 | |
|   return <CodePanel {...props} targetCode={targetCode}>{children}</CodePanel>
 | |
| }
 | |
| 
 | |
| function usePreventLayoutShift() {
 | |
|   const positionRef = useRef<any>()
 | |
|   const rafRef = useRef<any>()
 | |
| 
 | |
|   useEffect(() => {
 | |
|     return () => {
 | |
|       window.cancelAnimationFrame(rafRef.current)
 | |
|     }
 | |
|   }, [])
 | |
| 
 | |
|   return {
 | |
|     positionRef,
 | |
|     preventLayoutShift(callback: () => {}) {
 | |
|       const initialTop = positionRef.current.getBoundingClientRect().top
 | |
| 
 | |
|       callback()
 | |
| 
 | |
|       rafRef.current = window.requestAnimationFrame(() => {
 | |
|         const newTop = positionRef.current.getBoundingClientRect().top
 | |
|         window.scrollBy(0, newTop - initialTop)
 | |
|       })
 | |
|     },
 | |
|   }
 | |
| }
 | |
| 
 | |
| function useTabGroupProps(availableLanguages: string[]) {
 | |
|   const [preferredLanguages, addPreferredLanguage] = useState<any>([])
 | |
|   const [selectedIndex, setSelectedIndex] = useState(0)
 | |
|   const activeLanguage = [...availableLanguages].sort(
 | |
|     (a, z) => preferredLanguages.indexOf(z) - preferredLanguages.indexOf(a),
 | |
|   )[0]
 | |
|   const languageIndex = availableLanguages.indexOf(activeLanguage)
 | |
|   const newSelectedIndex = languageIndex === -1 ? selectedIndex : languageIndex
 | |
|   if (newSelectedIndex !== selectedIndex)
 | |
|     setSelectedIndex(newSelectedIndex)
 | |
| 
 | |
|   const { positionRef, preventLayoutShift } = usePreventLayoutShift()
 | |
| 
 | |
|   return {
 | |
|     as: 'div',
 | |
|     ref: positionRef,
 | |
|     selectedIndex,
 | |
|     onChange: (newSelectedIndex: number) => {
 | |
|       preventLayoutShift(() =>
 | |
|         (addPreferredLanguage(availableLanguages[newSelectedIndex]) as any),
 | |
|       )
 | |
|     },
 | |
|   }
 | |
| }
 | |
| 
 | |
| const CodeGroupContext = createContext(false)
 | |
| 
 | |
| export function CodeGroup({ children, title, inputs, targetCode, ...props }: IChildrenProps) {
 | |
|   const languages = Children.map(children, child =>
 | |
|     getPanelTitle(child.props.children.props),
 | |
|   )
 | |
|   const tabGroupProps = useTabGroupProps(languages)
 | |
|   const hasTabs = Children.count(children) > 1
 | |
|   const Container = hasTabs ? Tab.Group : 'div'
 | |
|   const containerProps = hasTabs ? tabGroupProps : {}
 | |
|   const headerProps = hasTabs
 | |
|     ? { selectedIndex: tabGroupProps.selectedIndex }
 | |
|     : {}
 | |
| 
 | |
|   return (
 | |
|     <CodeGroupContext.Provider value={true}>
 | |
|       <Container
 | |
|         {...containerProps}
 | |
|         className="not-prose my-6 overflow-hidden rounded-2xl bg-zinc-900 shadow-md dark:ring-1 dark:ring-white/10"
 | |
|       >
 | |
|         <CodeGroupHeader title={title} {...headerProps}>
 | |
|           {children}
 | |
|         </CodeGroupHeader>
 | |
|         <CodeGroupPanels {...props} targetCode={targetCode}>{children}</CodeGroupPanels>
 | |
|       </Container>
 | |
|     </CodeGroupContext.Provider>
 | |
|   )
 | |
| }
 | |
| 
 | |
| type IChildProps = {
 | |
|   children: string
 | |
|   [key: string]: any
 | |
| }
 | |
| export function Code({ children, ...props }: IChildProps) {
 | |
|   const isGrouped = useContext(CodeGroupContext)
 | |
| 
 | |
|   if (isGrouped)
 | |
|     return <code {...props} dangerouslySetInnerHTML={{ __html: children }} />
 | |
| 
 | |
|   return <code {...props}>{children}</code>
 | |
| }
 | |
| 
 | |
| export function Pre({ children, ...props }: IChildrenProps) {
 | |
|   const isGrouped = useContext(CodeGroupContext)
 | |
| 
 | |
|   if (isGrouped)
 | |
|     return children
 | |
| 
 | |
|   return <CodeGroup {...props}>{children}</CodeGroup>
 | |
| }
 |