mirror of
https://git.mirrors.martin98.com/https://github.com/langgenius/dify.git
synced 2025-05-29 01:28:22 +08:00

Co-authored-by: luowei <glpat-EjySCyNjWiLqAED-YmwM> Co-authored-by: crazywoola <427733928@qq.com> Co-authored-by: crazywoola <100913391+crazywoola@users.noreply.github.com>
158 lines
5.4 KiB
TypeScript
158 lines
5.4 KiB
TypeScript
import React, { useCallback, useEffect, useRef, useState } from 'react'
|
|
import mermaid from 'mermaid'
|
|
import { usePrevious } from 'ahooks'
|
|
import CryptoJS from 'crypto-js'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { ExclamationTriangleIcon } from '@heroicons/react/24/outline'
|
|
import LoadingAnim from '@/app/components/base/chat/chat/loading-anim'
|
|
import cn from '@/utils/classnames'
|
|
import ImagePreview from '@/app/components/base/image-uploader/image-preview'
|
|
|
|
let mermaidAPI: any
|
|
mermaidAPI = null
|
|
|
|
if (typeof window !== 'undefined')
|
|
mermaidAPI = mermaid.mermaidAPI
|
|
|
|
const style = {
|
|
minWidth: '480px',
|
|
height: 'auto',
|
|
overflow: 'auto',
|
|
}
|
|
|
|
const svgToBase64 = (svgGraph: string) => {
|
|
const svgBytes = new TextEncoder().encode(svgGraph)
|
|
const blob = new Blob([svgBytes], { type: 'image/svg+xml;charset=utf-8' })
|
|
return new Promise((resolve, reject) => {
|
|
const reader = new FileReader()
|
|
reader.onloadend = () => resolve(reader.result)
|
|
reader.onerror = reject
|
|
reader.readAsDataURL(blob)
|
|
})
|
|
}
|
|
|
|
const Flowchart = React.forwardRef((props: {
|
|
PrimitiveCode: string
|
|
}, ref) => {
|
|
const { t } = useTranslation()
|
|
const [svgCode, setSvgCode] = useState(null)
|
|
const [look, setLook] = useState<'classic' | 'handDrawn'>('classic')
|
|
|
|
const chartId = useRef(`flowchart_${CryptoJS.MD5(props.PrimitiveCode).toString()}`)
|
|
const prevPrimitiveCode = usePrevious(props.PrimitiveCode)
|
|
const [isLoading, setIsLoading] = useState(true)
|
|
const timeRef = useRef<NodeJS.Timeout>()
|
|
const [errMsg, setErrMsg] = useState('')
|
|
const [imagePreviewUrl, setImagePreviewUrl] = useState('')
|
|
|
|
const renderFlowchart = useCallback(async (PrimitiveCode: string) => {
|
|
setSvgCode(null)
|
|
setIsLoading(true)
|
|
|
|
try {
|
|
if (typeof window !== 'undefined' && mermaidAPI) {
|
|
const svgGraph = await mermaidAPI.render(chartId.current, PrimitiveCode)
|
|
const base64Svg: any = await svgToBase64(svgGraph.svg)
|
|
setSvgCode(base64Svg)
|
|
setIsLoading(false)
|
|
if (chartId.current && base64Svg)
|
|
localStorage.setItem(chartId.current, base64Svg)
|
|
}
|
|
}
|
|
catch (error) {
|
|
if (prevPrimitiveCode === props.PrimitiveCode) {
|
|
setIsLoading(false)
|
|
setErrMsg((error as Error).message)
|
|
}
|
|
}
|
|
}, [props.PrimitiveCode])
|
|
|
|
useEffect(() => {
|
|
if (typeof window !== 'undefined') {
|
|
mermaid.initialize({
|
|
startOnLoad: true,
|
|
theme: 'neutral',
|
|
look,
|
|
flowchart: {
|
|
htmlLabels: true,
|
|
useMaxWidth: true,
|
|
},
|
|
})
|
|
|
|
localStorage.removeItem(chartId.current)
|
|
renderFlowchart(props.PrimitiveCode)
|
|
}
|
|
}, [look])
|
|
|
|
useEffect(() => {
|
|
const cachedSvg: any = localStorage.getItem(chartId.current)
|
|
|
|
if (cachedSvg) {
|
|
setSvgCode(cachedSvg)
|
|
setIsLoading(false)
|
|
return
|
|
}
|
|
if (timeRef.current)
|
|
clearTimeout(timeRef.current)
|
|
|
|
timeRef.current = setTimeout(() => {
|
|
renderFlowchart(props.PrimitiveCode)
|
|
}, 300)
|
|
}, [props.PrimitiveCode])
|
|
|
|
return (
|
|
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
|
|
// @ts-expect-error
|
|
<div ref={ref}>
|
|
<div className="msh-segmented msh-segmented-sm css-23bs09 css-var-r1">
|
|
<div className="msh-segmented-group">
|
|
<label className="msh-segmented-item flex items-center space-x-1 m-2 w-[200px]">
|
|
<div key='classic'
|
|
className={cn('flex items-center justify-center mb-4 w-[calc((100%-8px)/2)] h-8 rounded-lg border border-components-option-card-option-border bg-components-option-card-option-bg cursor-pointer system-sm-medium text-text-secondary',
|
|
look === 'classic' && 'border-[1.5px] border-components-option-card-option-selected-border bg-components-option-card-option-selected-bg text-text-primary',
|
|
)}
|
|
|
|
onClick={() => setLook('classic')}
|
|
>
|
|
<div className="msh-segmented-item-label">{t('app.mermaid.classic')}</div>
|
|
</div>
|
|
<div key='handDrawn'
|
|
className={cn(
|
|
'flex items-center justify-center mb-4 w-[calc((100%-8px)/2)] h-8 rounded-lg border border-components-option-card-option-border bg-components-option-card-option-bg cursor-pointer system-sm-medium text-text-secondary',
|
|
look === 'handDrawn' && 'border-[1.5px] border-components-option-card-option-selected-border bg-components-option-card-option-selected-bg text-text-primary',
|
|
)}
|
|
onClick={() => setLook('handDrawn')}
|
|
>
|
|
<div className="msh-segmented-item-label">{t('app.mermaid.handDrawn')}</div>
|
|
</div>
|
|
</label>
|
|
</div>
|
|
</div>
|
|
{
|
|
svgCode
|
|
&& <div className="mermaid cursor-pointer" style={style} onClick={() => setImagePreviewUrl(svgCode)}>
|
|
{svgCode && <img src={svgCode} style={{ width: '100%', height: 'auto' }} alt="mermaid_chart" />}
|
|
</div>
|
|
}
|
|
{isLoading
|
|
&& <div className='py-4 px-[26px]'>
|
|
<LoadingAnim type='text'/>
|
|
</div>
|
|
}
|
|
{
|
|
errMsg
|
|
&& <div className='py-4 px-[26px]'>
|
|
<ExclamationTriangleIcon className='w-6 h-6 text-red-500'/>
|
|
|
|
{errMsg}
|
|
</div>
|
|
}
|
|
{
|
|
imagePreviewUrl && (<ImagePreview title='mermaid_chart' url={imagePreviewUrl} onCancel={() => setImagePreviewUrl('')} />)
|
|
}
|
|
</div>
|
|
)
|
|
})
|
|
|
|
export default Flowchart
|