| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 | 
							- 'use client'
 
- import {
 
-   Children,
 
-   createContext,
 
-   useContext,
 
-   useEffect,
 
-   useRef,
 
-   useState,
 
- } from 'react'
 
- import { Tab } from '@headlessui/react'
 
- import classNames from 'classnames'
 
- import { Tag } from './tag'
 
- 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.ReactElement
 
-   [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={() => {
 
-         window.navigator.clipboard.writeText(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="w-5 h-5 transition-colors fill-zinc-500/20 stroke-zinc-500 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="flex h-9 items-center gap-2 border-y border-t-transparent border-b-white/7.5 bg-zinc-900 bg-white/2.5 px-4 dark:border-b-white/5 dark:bg-white/1">
 
-       {tag && (
 
-         <div className="flex dark">
 
-           <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.ReactElement
 
-   tag?: string
 
-   code?: string
 
-   label?: string
 
-   targetCode?: string
 
- }
 
- function CodePanel({ tag, label, code, children, targetCode }: ICodePanelProps) {
 
-   const child = Children.only(children)
 
-   return (
 
-     <div className="group dark:bg-white/2.5">
 
-       <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="p-4 overflow-x-auto 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="pt-3 mr-auto text-xs font-semibold text-white">
 
-           {title}
 
-         </h3>
 
-       )}
 
-       {hasTabs && (
 
-         <Tab.List className="flex gap-4 -mb-px 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>
 
-           ))}
 
-         </Tab.List>
 
-       )}
 
-     </div>
 
-   )
 
- }
 
- type ICodeGroupPanelsProps = {
 
-   children: React.ReactElement
 
-   [key: string]: any
 
- }
 
- function CodeGroupPanels({ children, targetCode, ...props }: ICodeGroupPanelsProps) {
 
-   const hasTabs = Children.count(children) > 1
 
-   if (hasTabs) {
 
-     return (
 
-       <Tab.Panels>
 
-         {Children.map(children, child => (
 
-           <Tab.Panel>
 
-             <CodePanel {...props}>{child}</CodePanel>
 
-           </Tab.Panel>
 
-         ))}
 
-       </Tab.Panels>
 
-     )
 
-   }
 
-   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="my-6 overflow-hidden shadow-md not-prose rounded-2xl bg-zinc-900 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>
 
- }
 
 
  |