Spaces:
Build error
Build error
'use client' | |
import { | |
Children, | |
createContext, | |
useContext, | |
useEffect, | |
useRef, | |
useState, | |
} from 'react' | |
import { Tab } from '@headlessui/react' | |
import { Tag } from './tag' | |
import classNames from '@/utils/classnames' | |
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> | |
} | |