From 510a02286f97fde6573d310c7ac2d554c6f7923d Mon Sep 17 00:00:00 2001
From: lif <1835304752@qq.com>
Date: Thu, 22 Jan 2026 16:33:17 +0800
Subject: [PATCH] fix(web): disable single tilde strikethrough in markdown
rendering (#31400)
Signed-off-by: majiayu000 <1835304752@qq.com>
---
.../markdown/react-markdown-wrapper.spec.tsx | 109 ++++++++++++++++++
.../base/markdown/react-markdown-wrapper.tsx | 2 +-
2 files changed, 110 insertions(+), 1 deletion(-)
create mode 100644 web/app/components/base/markdown/react-markdown-wrapper.spec.tsx
diff --git a/web/app/components/base/markdown/react-markdown-wrapper.spec.tsx b/web/app/components/base/markdown/react-markdown-wrapper.spec.tsx
new file mode 100644
index 0000000000..735222011b
--- /dev/null
+++ b/web/app/components/base/markdown/react-markdown-wrapper.spec.tsx
@@ -0,0 +1,109 @@
+import type { PropsWithChildren, ReactNode } from 'react'
+import { render, screen } from '@testing-library/react'
+import { ReactMarkdownWrapper } from './react-markdown-wrapper'
+
+vi.mock('@/app/components/base/markdown-blocks', () => ({
+ AudioBlock: ({ children }: PropsWithChildren) =>
{children}
,
+ Img: ({ alt }: { alt?: string }) => {alt},
+ Link: ({ children, href }: { children?: ReactNode, href?: string }) => {children},
+ MarkdownButton: ({ children }: PropsWithChildren) => ,
+ MarkdownForm: ({ children }: PropsWithChildren) => ,
+ Paragraph: ({ children }: PropsWithChildren) => {children}
,
+ PluginImg: ({ alt }: { alt?: string }) => {alt},
+ PluginParagraph: ({ children }: PropsWithChildren) => {children}
,
+ ScriptBlock: () => null,
+ ThinkBlock: ({ children }: PropsWithChildren) => {children} ,
+ VideoBlock: ({ children }: PropsWithChildren) => {children}
,
+}))
+
+vi.mock('@/app/components/base/markdown-blocks/code-block', () => ({
+ default: ({ children }: PropsWithChildren) => {children},
+}))
+
+describe('ReactMarkdownWrapper', () => {
+ beforeEach(() => {
+ vi.clearAllMocks()
+ })
+
+ describe('Strikethrough rendering', () => {
+ it('should NOT render single tilde as strikethrough', () => {
+ // Arrange - single tilde should be rendered as literal text
+ const content = 'Range: 0.3~8mm'
+
+ // Act
+ render()
+
+ // Assert - check that ~ is rendered as text, not as strikethrough (del element)
+ // The content should contain the tilde as literal text
+ expect(screen.getByText(/0\.3~8mm/)).toBeInTheDocument()
+ expect(document.querySelector('del')).toBeNull()
+ })
+
+ it('should render double tildes as strikethrough', () => {
+ // Arrange - double tildes should create strikethrough
+ const content = 'This is ~~strikethrough~~ text'
+
+ // Act
+ render()
+
+ // Assert - del element should be present for double tildes
+ const delElement = document.querySelector('del')
+ expect(delElement).not.toBeNull()
+ expect(delElement?.textContent).toBe('strikethrough')
+ })
+
+ it('should handle mixed content with single and double tildes correctly', () => {
+ // Arrange - real-world example from issue #31391
+ const content = 'PCB thickness: 0.3~8mm and ~~removed feature~~ text'
+
+ // Act
+ render()
+
+ // Assert
+ // Only double tildes should create strikethrough
+ const delElements = document.querySelectorAll('del')
+ expect(delElements).toHaveLength(1)
+ expect(delElements[0].textContent).toBe('removed feature')
+
+ // Single tilde should remain as literal text
+ expect(screen.getByText(/0\.3~8mm/)).toBeInTheDocument()
+ })
+ })
+
+ describe('Basic rendering', () => {
+ it('should render plain text content', () => {
+ // Arrange
+ const content = 'Hello World'
+
+ // Act
+ render()
+
+ // Assert
+ expect(screen.getByText('Hello World')).toBeInTheDocument()
+ })
+
+ it('should render bold text', () => {
+ // Arrange
+ const content = '**bold text**'
+
+ // Act
+ render()
+
+ // Assert
+ expect(screen.getByText('bold text')).toBeInTheDocument()
+ expect(document.querySelector('strong')).not.toBeNull()
+ })
+
+ it('should render italic text', () => {
+ // Arrange
+ const content = '*italic text*'
+
+ // Act
+ render()
+
+ // Assert
+ expect(screen.getByText('italic text')).toBeInTheDocument()
+ expect(document.querySelector('em')).not.toBeNull()
+ })
+ })
+})
diff --git a/web/app/components/base/markdown/react-markdown-wrapper.tsx b/web/app/components/base/markdown/react-markdown-wrapper.tsx
index ef735b5e76..ed9e93e8b3 100644
--- a/web/app/components/base/markdown/react-markdown-wrapper.tsx
+++ b/web/app/components/base/markdown/react-markdown-wrapper.tsx
@@ -30,7 +30,7 @@ export const ReactMarkdownWrapper: FC = (props) => {
return (