mirror of
https://github.com/dair-ai/Prompt-Engineering-Guide
synced 2024-11-08 07:10:41 +00:00
83 lines
2.5 KiB
TypeScript
83 lines
2.5 KiB
TypeScript
import cn from 'clsx'
|
|
import type { ComponentProps, ReactElement } from 'react'
|
|
import { useCallback, useRef } from 'react'
|
|
import { WordWrapIcon } from './word-wrap'
|
|
import { Button } from './button'
|
|
import { CopyToClipboard } from './copy-to-clipboard'
|
|
import React from 'react'
|
|
|
|
|
|
export const Pre = ({
|
|
children,
|
|
className,
|
|
hasCopyCode = true,
|
|
filename,
|
|
...props
|
|
}: ComponentProps<'pre'> & {
|
|
filename?: string
|
|
hasCopyCode?: boolean
|
|
}): ReactElement => {
|
|
const preRef = useRef<HTMLPreElement | null>(null);
|
|
|
|
const toggleWordWrap = useCallback(() => {
|
|
const htmlDataset = document.documentElement.dataset;
|
|
const hasWordWrap = 'nextraWordWrap' in htmlDataset;
|
|
if (hasWordWrap) {
|
|
delete htmlDataset.nextraWordWrap;
|
|
} else {
|
|
htmlDataset.nextraWordWrap = '';
|
|
}
|
|
}, []);
|
|
|
|
const renderChildren = () => {
|
|
if (React.isValidElement(children) && children.type === 'code') {
|
|
return children.props.children;
|
|
}
|
|
return children;
|
|
};
|
|
|
|
return (
|
|
<div className="nextra-code-block nx-relative nx-mt-6 first:nx-mt-0">
|
|
{filename && (
|
|
<div className="nx-absolute nx-top-0 nx-z-[1] nx-w-full nx-truncate nx-rounded-t-xl nx-bg-primary-700/5 nx-py-2 nx-px-4 nx-text-xs nx-text-gray-700 dark:nx-bg-primary-300/10 dark:nx-text-gray-200">
|
|
{filename}
|
|
</div>
|
|
)}
|
|
<pre
|
|
className={cn(
|
|
'nx-bg-primary-700/5 nx-mb-4 nx-overflow-x-auto nx-rounded-xl nx-subpixel-antialiased dark:nx-bg-primary-300/10 nx-text-[.9em]',
|
|
'contrast-more:nx-border contrast-more:nx-border-primary-900/20 contrast-more:nx-contrast-150 contrast-more:dark:nx-border-primary-100/40',
|
|
filename ? 'nx-pt-12 nx-pb-4' : 'nx-py-4',
|
|
className
|
|
)}
|
|
ref={preRef}
|
|
{...props}
|
|
>
|
|
{renderChildren()}
|
|
</pre>
|
|
<div
|
|
className={cn(
|
|
'nx-opacity-0 nx-transition [div:hover>&]:nx-opacity-100 focus-within:nx-opacity-100',
|
|
'nx-flex nx-gap-1 nx-absolute nx-m-[11px] nx-right-0',
|
|
filename ? 'nx-top-8' : 'nx-top-0'
|
|
)}
|
|
>
|
|
<Button
|
|
onClick={toggleWordWrap}
|
|
className="md:nx-hidden"
|
|
title="Toggle word wrap elvis"
|
|
>
|
|
<WordWrapIcon className="nx-pointer-events-none nx-h-4 nx-w-4" />
|
|
</Button>
|
|
{hasCopyCode && (
|
|
<CopyToClipboard
|
|
getValue={() =>
|
|
preRef.current?.querySelector('code')?.textContent || ''
|
|
}
|
|
/>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|