gsaivinay's picture
Duplicate from matthoffner/starchat-ui
a28cd69
raw
history blame
4 kB
import { IconCheck, IconClipboard, IconDownload, IconCaretRight } from '@tabler/icons-react';
import { FC, memo, useEffect, useRef, useState } from 'react';
import { Prism as SyntaxHighlighter } from 'react-syntax-highlighter';
import { oneDark } from 'react-syntax-highlighter/dist/cjs/styles/prism';
import { useTranslation } from 'next-i18next';
import {
generateRandomString,
programmingLanguages,
} from '@/utils/app/codeblock';
interface Props {
language: string;
value: string;
}
const IframeContent: FC = () => {
useEffect(() => {
window.addEventListener("message", ({ data: { code, language } }) => {
if (language === 'html') {
document.body.innerHTML = code;
} else if (language === 'javascript') {
eval(code);
}
});
}, []);
return null;
};
export const CodeBlock: FC<Props> = memo(({ language, value }) => {
const { t } = useTranslation('markdown');
const [isCopied, setIsCopied] = useState<Boolean>(false);
const [isRunning, setIsRunning] = useState<Boolean>(false);
const iframeRef = useRef<any>(null);
const toggleRunCode = () => {
setIsRunning(!isRunning);
};
useEffect(() => {
if (isRunning && iframeRef.current) {
iframeRef.current.contentWindow?.postMessage({ code: value, language }, "*");
}
}, [isRunning, value, language]);
const copyToClipboard = () => {
if (!navigator.clipboard || !navigator.clipboard.writeText) {
return;
}
navigator.clipboard.writeText(value).then(() => {
setIsCopied(true);
setTimeout(() => {
setIsCopied(false);
}, 2000);
});
};
const downloadAsFile = () => {
const fileExtension = programmingLanguages[language] || '.file';
const suggestedFileName = `file-${generateRandomString(
3,
true,
)}${fileExtension}`;
const fileName = window.prompt(
t('Enter file name') || '',
suggestedFileName,
);
if (!fileName) {
// user pressed cancel on prompt
return;
}
const blob = new Blob([value], { type: 'text/plain' });
const url = URL.createObjectURL(blob);
const link = document.createElement('a');
link.download = fileName;
link.href = url;
link.style.display = 'none';
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
URL.revokeObjectURL(url);
};
return (
<div className="codeblock relative font-sans text-[16px]">
<div className="flex items-center justify-between py-1.5 px-4">
<span className="text-xs lowercase text-white">{language}</span>
<div className="flex items-center">
<button
className="flex gap-1.5 items-center rounded bg-none p-1 text-xs text-white"
onClick={copyToClipboard}
>
{isCopied ? <IconCheck size={18} /> : <IconClipboard size={18} />}
{isCopied ? t('Copied!') : t('Copy code')}
</button>
{(language === 'html' || language === 'javascript') && (
<button
className="flex gap-1.5 items-center rounded bg-none p-1 text-xs text-white"
onClick={toggleRunCode}
>
<IconCaretRight size={18} />
{isRunning ? t('Stop') : t('Run code')}
</button>
)}
<button
className="flex items-center rounded bg-none p-1 text-xs text-white"
onClick={downloadAsFile}
>
<IconDownload size={18} />
</button>
</div>
</div>
<SyntaxHighlighter
language={language}
style={oneDark}
customStyle={{ margin: 0 }}
>
{value}
</SyntaxHighlighter>
{isRunning && (
<iframe
ref={iframeRef}
title="live code preview"
className="w-full h-64 mt-4 border-2 border-gray-300"
>
<IframeContent />
</iframe>
)}
</div>
);
});
CodeBlock.displayName = 'CodeBlock';