mirror of
https://github.com/stackblitz/bolt.new
synced 2024-11-28 06:52:47 +00:00
67 lines
1.9 KiB
TypeScript
67 lines
1.9 KiB
TypeScript
|
import { memo } from 'react';
|
||
|
import ReactMarkdown from 'react-markdown';
|
||
|
import type { BundledLanguage } from 'shiki';
|
||
|
import { createScopedLogger } from '~/utils/logger';
|
||
|
import { rehypePlugins, remarkPlugins } from '~/utils/markdown';
|
||
|
import { Artifact } from './Artifact';
|
||
|
import { CodeBlock } from './CodeBlock';
|
||
|
import styles from './Markdown.module.scss';
|
||
|
|
||
|
const logger = createScopedLogger('MarkdownComponent');
|
||
|
|
||
|
interface MarkdownProps {
|
||
|
children: string;
|
||
|
}
|
||
|
|
||
|
export const Markdown = memo(({ children }: MarkdownProps) => {
|
||
|
logger.trace('Render');
|
||
|
|
||
|
return (
|
||
|
<ReactMarkdown
|
||
|
className={styles.MarkdownContent}
|
||
|
components={{
|
||
|
div: ({ className, children, node, ...props }) => {
|
||
|
if (className?.includes('__boltArtifact__')) {
|
||
|
const messageId = node?.properties.dataMessageId as string;
|
||
|
|
||
|
if (!messageId) {
|
||
|
logger.warn(`Invalud message id ${messageId}`);
|
||
|
}
|
||
|
|
||
|
return <Artifact messageId={messageId} />;
|
||
|
}
|
||
|
|
||
|
return (
|
||
|
<div className={className} {...props}>
|
||
|
{children}
|
||
|
</div>
|
||
|
);
|
||
|
},
|
||
|
pre: (props) => {
|
||
|
const { children, node, ...rest } = props;
|
||
|
|
||
|
const [firstChild] = node?.children ?? [];
|
||
|
|
||
|
if (
|
||
|
firstChild &&
|
||
|
firstChild.type === 'element' &&
|
||
|
firstChild.tagName === 'code' &&
|
||
|
firstChild.children[0].type === 'text'
|
||
|
) {
|
||
|
const { className, ...rest } = firstChild.properties;
|
||
|
const [, language = 'plaintext'] = /language-(\w+)/.exec(String(className) || '') ?? [];
|
||
|
|
||
|
return <CodeBlock code={firstChild.children[0].value} language={language as BundledLanguage} {...rest} />;
|
||
|
}
|
||
|
|
||
|
return <pre {...rest}>{children}</pre>;
|
||
|
},
|
||
|
}}
|
||
|
remarkPlugins={remarkPlugins}
|
||
|
rehypePlugins={rehypePlugins}
|
||
|
>
|
||
|
{children}
|
||
|
</ReactMarkdown>
|
||
|
);
|
||
|
});
|