|
'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> |
|
} |
|
|