Files
FastGPT/packages/web/components/common/Markdown/index.tsx
T
Archer a499d05a02 V4.14.0 features (#5850)
* feat: migrate chat files to s3 (#5802)

* feat: migrate chat files to s3

* feat: add delete jobs for deleting s3 files

* chore: improvements

* fix: lockfile

* fix: imports

* feat: add ttl for those uploaded files but not send yet

* feat: init bullmq worker

* fix: s3 key

* perf: s3 internal url

* remove env

* fix: re-sign a new url

* fix: re-sign a new url

* perf: s3 code

---------

Co-authored-by: archer <545436317@qq.com>

* update pacakge

* feat: add more file type for uploading (#5807)

* fix: re-sign a new url

* wip: file selector

* feat: add more file type for uploading

* feat: migrate chat files to s3 (#5802)

* feat: migrate chat files to s3

* feat: add delete jobs for deleting s3 files

* chore: improvements

* fix: lockfile

* fix: imports

* feat: add ttl for those uploaded files but not send yet

* feat: init bullmq worker

* fix: s3 key

* perf: s3 internal url

* remove env

* fix: re-sign a new url

* fix: re-sign a new url

* perf: s3 code

---------

Co-authored-by: archer <545436317@qq.com>

* fix: limit minmax available file upload number

* perf: file select modal code

* fix: fileselect refresh

* fix: ts

---------

Co-authored-by: archer <545436317@qq.com>

* bugfix: chat page (#5809)

* fix: upload avatar

* fix: chat page username display issue and setting button visibility

* doc

* Markdown match base64 performance

* feat: improve global variables(time, file, dataset) (#5804)

* feat: improve global variables(time, file, dataset)

* feat: optimize code

* perf: time variables code

* fix: model, file

* fix: hide file upload

* fix: ts

* hide dataset select

---------

Co-authored-by: archer <545436317@qq.com>

* perf: insert training queue

* perf: s3 upload error i18n

* fix: share page s3

* fix: timeselector ui error

* var update node

* Timepicker ui

* feat: plugin support password

* fix: password disabled UX

* fix: button size

* fix: no model cache for chat page (#5820)

* rename function

* fix: workflow bug

* fix: interactive loop

* fix test

* perf: common textare no richtext

* move system plugin config (#5803) (#5813)

* move system plugin config (#5803)

* move system plugin config

* extract tag bar

* filter

* tool detail temp

* marketplace

* params

* fix

* type

* search

* tags render

* status

* ui

* code

* connect to backend (#5815)

* feat: marketplace apis & type definitions (#5817)

* chore: marketplace init

* chore: marketplace list api type

* chore: detail api

* marketplace & import

* feat: marketplace ui (#5826)

* temp

* marketplace

* import

* feat: detail return readme

* chore: cache data expire 10 mins

* chore: update docs

* feat: marketplace ui

---------

Co-authored-by: heheer <zhiyu44@qq.com>

* feat: marketplace (#5830)

* temp

* marketplace

* chore: tool list tag filter

* chore: adjust

---------

Co-authored-by: heheer <zhiyu44@qq.com>

* tool detail drawer

* remove tag filter

* fix

* fix

* fix build

* update pnpm-lock

* fix type

* perf code

* marketplace router

* fix build

* navbar icon

* fix ui

* fix init

* docs: marketplace/plugin (#5832)

* temp

* marketplace

* docs(plugin): system tool docs

---------

Co-authored-by: heheer <zhiyu44@qq.com>

* default url

* feat: i18n/ docker build (#5833)

* chore: docker build

* feat: i18n selector

* fix

* fix

* fix: i18n parse

* fix: i18n parse

---------

Co-authored-by: heheer <heheer@sealos.io>
Co-authored-by: Finley Ge <32237950+FinleyGe@users.noreply.github.com>
Co-authored-by: heheer <zhiyu44@qq.com>

* marketplace url

* update action

* market place code

* market place code

* title

* fix: nextconfig

* fix: copilot review

* Remove bypassable regex-based XSS sanitization from marketplace search (#5835)

* Initial plan

* Remove problematic regex-based XSS sanitization from search inputs

Co-authored-by: c121914yu <50446880+c121914yu@users.noreply.github.com>

---------

Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com>
Co-authored-by: c121914yu <50446880+c121914yu@users.noreply.github.com>

* feat: tool tag openapi

* api check

* fix: tsc

* fix: ts

* fix: lock

* sdk version

* ts

* sdk version

* remove invalid tip

* perf: export data add timezone

* perf: admin plugin api move

* perf: tool code

* move tag code

* perf: marketplace and team plugin code

* remove workflow invalid request

* rename global tool code

* rename global tool code

* rename api

* fix some bugs (#5841)

* fix some bugs

* fix

* perf: Tag filter

* fix: ts

* fix: ts

---------

Co-authored-by: archer <545436317@qq.com>

* perf: Concat function

* fix: workflow snapshot push

* fix: ts type

* fix: login to config/*

* fix: ts

* fix: model avatar (#5848)

* fix: model avatar

* fix: ts

* fix: avatar migration to s3

* update lock

* fix: avatar redirect

---------

Co-authored-by: archer <545436317@qq.com>

* fix tool detail (#5847)

* fix tool detail

* init script

* fix build

* perf: plugin detail modal

* change tooltags to tags

* fix icon

---------

Co-authored-by: archer <545436317@qq.com>

* fix tag filter scroll (#5852)

* fix create app plugin & import info (#5853)

* tag size

* rename toolkit

* download url

* import plugin status (#5854)

* init doc

* fix: init shell

---------

Co-authored-by: 伍闲犬 <whoeverimf5@gmail.com>
Co-authored-by: Zeng Qingwen <143274079+fishwww-ww@users.noreply.github.com>
Co-authored-by: heheer <heheer@sealos.io>
Co-authored-by: Finley Ge <32237950+FinleyGe@users.noreply.github.com>
Co-authored-by: heheer <zhiyu44@qq.com>
Co-authored-by: Copilot <198982749+Copilot@users.noreply.github.com>
2025-11-04 16:58:12 +08:00

165 lines
4.2 KiB
TypeScript

import React, { useMemo } from 'react';
import ReactMarkdown from 'react-markdown';
import RemarkGfm from 'remark-gfm';
import RehypeExternalLinks from 'rehype-external-links';
import { Box, Code as ChakraCode, Image, Link } from '@chakra-ui/react';
type MarkdownProps = {
source: string;
className?: string;
};
/**
* 简化版 Markdown 组件
* 用于渲染基础的 Markdown 内容(README 等)
*/
const Markdown = ({ source, className }: MarkdownProps) => {
const components = useMemo(
() => ({
// 图片
img: ({ src, alt }: any) => (
<Image src={src} alt={alt} maxW="100%" my={2} borderRadius="md" loading="lazy" />
),
// 链接
a: ({ href, children }: any) => (
<Link href={href} color="primary.600" textDecoration="underline" isExternal>
{children}
</Link>
),
// 行内代码
code: ({ children, className }: any) => {
// 如果有 className,说明是代码块,保留原样
if (className) {
return (
<Box
as="pre"
p={3}
bg="myGray.100"
borderRadius="md"
overflow="auto"
fontSize="xs"
fontFamily="monospace"
>
<code className={className}>{children}</code>
</Box>
);
}
// 行内代码
return (
<ChakraCode
px={1}
py={0.5}
bg="myGray.100"
borderRadius="sm"
fontSize="xs"
fontFamily="monospace"
>
{children}
</ChakraCode>
);
},
// 标题
h1: ({ children }: any) => (
<Box as="h1" fontSize="xl" fontWeight="bold" mt={4} mb={2}>
{children}
</Box>
),
h2: ({ children }: any) => (
<Box as="h2" fontSize="lg" fontWeight="bold" mt={3} mb={2}>
{children}
</Box>
),
h3: ({ children }: any) => (
<Box as="h3" fontSize="md" fontWeight="semibold" mt={2} mb={1}>
{children}
</Box>
),
// 段落
p: ({ children }: any) => (
<Box as="p" mb={2} lineHeight={1.6}>
{children}
</Box>
),
// 列表
ul: ({ children }: any) => (
<Box as="ul" pl={4} mb={2}>
{children}
</Box>
),
ol: ({ children }: any) => (
<Box as="ol" pl={4} mb={2}>
{children}
</Box>
),
li: ({ children }: any) => (
<Box as="li" mb={1}>
{children}
</Box>
),
// 引用
blockquote: ({ children }: any) => (
<Box
as="blockquote"
borderLeft="4px solid"
borderColor="myGray.300"
pl={4}
py={2}
my={2}
fontStyle="italic"
color="myGray.600"
>
{children}
</Box>
),
// 水平线
hr: () => <Box as="hr" my={4} borderColor="myGray.200" />,
// 表格
table: ({ children }: any) => (
<Box overflowX="auto" my={2}>
<Box as="table" w="100%" border="1px solid" borderColor="myGray.200" borderRadius="md">
{children}
</Box>
</Box>
),
thead: ({ children }: any) => (
<Box as="thead" bg="myGray.50">
{children}
</Box>
),
tbody: ({ children }: any) => <Box as="tbody">{children}</Box>,
tr: ({ children }: any) => (
<Box as="tr" borderBottom="1px solid" borderColor="myGray.200">
{children}
</Box>
),
th: ({ children }: any) => (
<Box as="th" px={3} py={2} textAlign="left" fontWeight="semibold" fontSize="sm">
{children}
</Box>
),
td: ({ children }: any) => (
<Box as="td" px={3} py={2} fontSize="sm">
{children}
</Box>
)
}),
[]
);
return (
<Box className={className}>
<ReactMarkdown
remarkPlugins={[RemarkGfm]}
rehypePlugins={[
[RehypeExternalLinks, { target: '_blank', rel: ['noopener', 'noreferrer'] }]
]}
components={components}
>
{source}
</ReactMarkdown>
</Box>
);
};
export default React.memo(Markdown);