import ReactMarkdown from 'react-markdown' import ReactEcharts from 'echarts-for-react' import 'katex/dist/katex.min.css' import RemarkMath from 'remark-math' import RemarkBreaks from 'remark-breaks' import RehypeKatex from 'rehype-katex' import RemarkGfm from 'remark-gfm' import RehypeRaw from 'rehype-raw' import SyntaxHighlighter from 'react-syntax-highlighter' import { atelierHeathDark, atelierHeathLight, } from 'react-syntax-highlighter/dist/esm/styles/hljs' import { Component, memo, useMemo, useRef, useState } from 'react' import { flow } from 'lodash-es' import ActionButton from '@/app/components/base/action-button' import CopyIcon from '@/app/components/base/copy-icon' import SVGBtn from '@/app/components/base/svg' import Flowchart from '@/app/components/base/mermaid' import ImageGallery from '@/app/components/base/image-gallery' import { useChatContext } from '@/app/components/base/chat/chat/context' import VideoGallery from '@/app/components/base/video-gallery' import AudioGallery from '@/app/components/base/audio-gallery' import MarkdownButton from '@/app/components/base/markdown-blocks/button' import MarkdownForm from '@/app/components/base/markdown-blocks/form' import ThinkBlock from '@/app/components/base/markdown-blocks/think-block' import { Theme } from '@/types/app' import useTheme from '@/hooks/use-theme' import cn from '@/utils/classnames' // Available language https://github.com/react-syntax-highlighter/react-syntax-highlighter/blob/master/AVAILABLE_LANGUAGES_HLJS.MD const capitalizationLanguageNameMap: Record = { sql: 'SQL', javascript: 'JavaScript', java: 'Java', typescript: 'TypeScript', vbscript: 'VBScript', css: 'CSS', html: 'HTML', xml: 'XML', php: 'PHP', python: 'Python', yaml: 'Yaml', mermaid: 'Mermaid', markdown: 'MarkDown', makefile: 'MakeFile', echarts: 'ECharts', shell: 'Shell', powershell: 'PowerShell', json: 'JSON', latex: 'Latex', svg: 'SVG', } const getCorrectCapitalizationLanguageName = (language: string) => { if (!language) return 'Plain' if (language in capitalizationLanguageNameMap) return capitalizationLanguageNameMap[language] return language.charAt(0).toUpperCase() + language.substring(1) } const preprocessLaTeX = (content: string) => { if (typeof content !== 'string') return content return flow([ (str: string) => str.replace(/\\\[(.*?)\\\]/g, (_, equation) => `$$${equation}$$`), (str: string) => str.replace(/\\\[(.*?)\\\]/gs, (_, equation) => `$$${equation}$$`), (str: string) => str.replace(/\\\((.*?)\\\)/g, (_, equation) => `$$${equation}$$`), (str: string) => str.replace(/(^|[^\\])\$(.+?)\$/g, (_, prefix, equation) => `${prefix}$${equation}$`), ])(content) } const preprocessThinkTag = (content: string) => { return flow([ (str: string) => str.replace('\n', '
\n'), (str: string) => str.replace('\n', '\n[ENDTHINKFLAG]
'), ])(content) } export function PreCode(props: { children: any }) { const ref = useRef(null) return (
      
      {props.children}
    
) } // **Add code block // Avoid error #185 (Maximum update depth exceeded. // This can happen when a component repeatedly calls setState inside componentWillUpdate or componentDidUpdate. // React limits the number of nested updates to prevent infinite loops.) // Reference A: https://reactjs.org/docs/error-decoder.html?invariant=185 // Reference B1: https://react.dev/reference/react/memo // Reference B2: https://react.dev/reference/react/useMemo // **** // The original error that occurred in the streaming response during the conversation: // Error: Minified React error 185; // visit https://reactjs.org/docs/error-decoder.html?invariant=185 for the full message // or use the non-minified dev environment for full errors and additional helpful warnings. const CodeBlock: any = memo(({ inline, className, children, ...props }: any) => { const { theme } = useTheme() const [isSVG, setIsSVG] = useState(true) const match = /language-(\w+)/.exec(className || '') const language = match?.[1] const languageShowName = getCorrectCapitalizationLanguageName(language || '') const chartData = useMemo(() => { if (language === 'echarts') { try { return JSON.parse(String(children).replace(/\n$/, '')) } catch (error) { } } return JSON.parse('{"title":{"text":"ECharts error - Wrong JSON format."}}') }, [language, children]) const renderCodeContent = useMemo(() => { const content = String(children).replace(/\n$/, '') if (language === 'mermaid' && isSVG) { return } else if (language === 'echarts') { return (
) } // Attention: SVGRenderer has xss vulnerability // else if (language === 'svg' && isSVG) { // return ( // // // // ) // } else { return ( {content} ) } }, [language, match, props, children, chartData, isSVG]) if (inline || !match) return {children} return (
{languageShowName}
{(['mermaid', 'svg']).includes(language!) && }
{renderCodeContent}
) }) CodeBlock.displayName = 'CodeBlock' const VideoBlock: any = memo(({ node }: any) => { const srcs = node.children.filter((child: any) => 'properties' in child).map((child: any) => (child as any).properties.src) if (srcs.length === 0) return null return }) VideoBlock.displayName = 'VideoBlock' const AudioBlock: any = memo(({ node }: any) => { const srcs = node.children.filter((child: any) => 'properties' in child).map((child: any) => (child as any).properties.src) if (srcs.length === 0) return null return }) AudioBlock.displayName = 'AudioBlock' const ScriptBlock = memo(({ node }: any) => { const scriptContent = node.children[0]?.value || '' return `` }) ScriptBlock.displayName = 'ScriptBlock' const Paragraph = (paragraph: any) => { const { node }: any = paragraph const children_node = node.children if (children_node && children_node[0] && 'tagName' in children_node[0] && children_node[0].tagName === 'img') { return ( <> { Array.isArray(paragraph.children) ?

{paragraph.children.slice(1)}

: null } ) } return

{paragraph.children}

} const Img = ({ src }: any) => { return () } const Link = ({ node, ...props }: any) => { if (node.properties?.href && node.properties.href?.toString().startsWith('abbr')) { // eslint-disable-next-line react-hooks/rules-of-hooks const { onSend } = useChatContext() const hidden_text = decodeURIComponent(node.properties.href.toString().split('abbr:')[1]) return onSend?.(hidden_text)} title={node.children[0]?.value}>{node.children[0]?.value} } else { return {node.children[0] ? node.children[0]?.value : 'Download'} } } function escapeSVGTags(htmlString: string): string { return htmlString.replace(/()([\s\S]*?)(<\/svg>)/gi, (match: string, openTag: string, innerContent: string, closeTag: string): string => { return openTag.replace(//g, '>') + innerContent.replace(//g, '>') + closeTag.replace(//g, '>') }) } export function Markdown(props: { content: string; className?: string; customDisallowedElements?: string[] }) { const latexContent = flow([ preprocessThinkTag, preprocessLaTeX, ])(escapeSVGTags(props.content)) return (
{ return (tree) => { const iterate = (node: any) => { if (node.type === 'element' && node.properties?.ref) delete node.properties.ref if (node.type === 'element' && !/^[a-z][a-z0-9]*$/i.test(node.tagName)) { node.type = 'text' node.value = `<${node.tagName}` } if (node.children) node.children.forEach(iterate) } tree.children.forEach(iterate) } }, ]} disallowedElements={['iframe', 'head', 'html', 'meta', 'link', 'style', 'body', ...(props.customDisallowedElements || [])]} components={{ code: CodeBlock, img: Img, video: VideoBlock, audio: AudioBlock, a: Link, p: Paragraph, button: MarkdownButton, form: MarkdownForm, script: ScriptBlock as any, details: ThinkBlock, }} > {/* Markdown detect has problem. */} {latexContent}
) } // **Add an ECharts runtime error handler // Avoid error #7832 (Crash when ECharts accesses undefined objects) // This can happen when a component attempts to access an undefined object that references an unregistered map, causing the program to crash. export default class ErrorBoundary extends Component { constructor(props: any) { super(props) this.state = { hasError: false } } componentDidCatch(error: any, errorInfo: any) { this.setState({ hasError: true }) console.error(error, errorInfo) } render() { // eslint-disable-next-line ts/ban-ts-comment // @ts-expect-error if (this.state.hasError) return
Oops! An error occurred. This could be due to an ECharts runtime error or invalid SVG content.
(see the browser console for more information)
// eslint-disable-next-line ts/ban-ts-comment // @ts-expect-error return this.props.children } }