diff --git a/demo/components/Playground.tsx b/demo/components/Playground.tsx index 2b1d096d..b265a2b7 100644 --- a/demo/components/Playground.tsx +++ b/demo/components/Playground.tsx @@ -79,6 +79,7 @@ export type PlaygroundProps = { onChangeSplitModeEnabled?: (splitModeEnabled: boolean) => void; directiveSyntax?: DirectiveSyntaxValue; disabledHTMLBlockModes?: EmbeddingMode[]; + disableMarkdownItAttrs?: boolean; } & Pick & Pick< MarkdownEditorViewProps, @@ -128,6 +129,7 @@ export const Playground = memo((props) => { experimental, directiveSyntax, disabledHTMLBlockModes, + disableMarkdownItAttrs, } = props; const [editorMode, setEditorMode] = useState(initialEditor ?? 'wysiwyg'); const [mdRaw, setMdRaw] = useState(initial || ''); @@ -146,10 +148,11 @@ export const Playground = memo((props) => { breaks={md.breaks} needToSanitizeHtml={sanitizeHtml} plugins={getPlugins({directiveSyntax})} + disableMarkdownItAttrs={disableMarkdownItAttrs} htmlRuntimeConfig={{disabledModes: disabledHTMLBlockModes}} /> ), - [sanitizeHtml, disabledHTMLBlockModes], + [sanitizeHtml, disabledHTMLBlockModes, disableMarkdownItAttrs], ); const logger = useMemo(() => new Logger2().nested({env: 'playground'}), []); @@ -161,6 +164,7 @@ export const Playground = memo((props) => { preset: 'full', wysiwygConfig: { placeholderOptions: placeholderOptions, + disableMarkdownAttrs: disableMarkdownItAttrs, extensions: (builder) => { builder .use(Math, { @@ -250,6 +254,7 @@ export const Playground = memo((props) => { experimental?.beforeEditorModeChange, experimental?.prepareRawMarkup, directiveSyntax, + disableMarkdownItAttrs, ], ); diff --git a/demo/components/PlaygroundMini.tsx b/demo/components/PlaygroundMini.tsx index 66884bb7..845bc6ba 100644 --- a/demo/components/PlaygroundMini.tsx +++ b/demo/components/PlaygroundMini.tsx @@ -24,6 +24,7 @@ export type PlaygroundMiniProps = Pick< | 'onChangeSplitModeEnabled' | 'directiveSyntax' | 'disabledHTMLBlockModes' + | 'disableMarkdownItAttrs' > & {withDefaultInitialContent?: boolean}; export const PlaygroundMini = memo( diff --git a/demo/components/SplitModePreview.tsx b/demo/components/SplitModePreview.tsx index 156128bd..eaa5c1e8 100644 --- a/demo/components/SplitModePreview.tsx +++ b/demo/components/SplitModePreview.tsx @@ -25,7 +25,7 @@ const Preview = withMermaid({runtime: MERMAID_RUNTIME})( ); export type SplitModePreviewProps = { - plugins?: MarkdownIt.PluginSimple[]; + plugins: MarkdownIt.PluginSimple[]; getValue: () => MarkupString; allowHTML?: boolean; breaks?: boolean; @@ -33,6 +33,7 @@ export type SplitModePreviewProps = { linkifyTlds?: string | string[]; needToSanitizeHtml?: boolean; htmlRuntimeConfig?: HTMLRuntimeConfig; + disableMarkdownItAttrs?: boolean; }; export const SplitModePreview: React.FC = (props) => { @@ -45,6 +46,7 @@ export const SplitModePreview: React.FC = (props) => { linkifyTlds, needToSanitizeHtml, htmlRuntimeConfig, + disableMarkdownItAttrs, } = props; const [html, setHtml] = useState(''); const [meta, setMeta] = useState({}); @@ -58,12 +60,17 @@ export const SplitModePreview: React.FC = (props) => { const res = transform(getValue(), { allowHTML, breaks, - plugins, linkify, linkifyTlds, needToSanitizeHtml, linkAttrs: [[ML_ATTR, true]], defaultClassName: colorClassName, + plugins: [ + ...plugins, + ...(disableMarkdownItAttrs + ? [(md: MarkdownIt) => md.core.ruler.disable('curly_attributes')] + : []), + ], }).result; setHtml(res.html); setMeta(res.meta); diff --git a/demo/defaults/args.ts b/demo/defaults/args.ts index 302023c5..18072bf5 100644 --- a/demo/defaults/args.ts +++ b/demo/defaults/args.ts @@ -18,4 +18,5 @@ export const args: Meta['args'] = { height: 'initial', directiveSyntax: 'disabled', disabledHTMLBlockModes: [], + disableMarkdownItAttrs: true, }; diff --git a/demo/stories/playground/Playground.stories.tsx b/demo/stories/playground/Playground.stories.tsx index 44776ad0..b2df7132 100644 --- a/demo/stories/playground/Playground.stories.tsx +++ b/demo/stories/playground/Playground.stories.tsx @@ -4,7 +4,11 @@ import {type PlaygroundProps, Playground as component} from '../../components/Pl import {args} from '../../defaults/args'; import {getInitialMd} from '../../utils/getInitialMd'; -export const Story: StoryObj = {}; +export const Story: StoryObj = { + args: { + disableMarkdownItAttrs: true, + }, +}; Story.storyName = 'Playground'; const meta: Meta = { diff --git a/demo/stories/presets/Preset.tsx b/demo/stories/presets/Preset.tsx index a43854c1..c0e77b67 100644 --- a/demo/stories/presets/Preset.tsx +++ b/demo/stories/presets/Preset.tsx @@ -90,6 +90,7 @@ export const Preset = memo((props) => { splitModeEnabled: true, }, wysiwygConfig: { + disableMarkdownAttrs: true, extensionOptions: { imgSize: { parseInsertedUrlAsImage, diff --git a/package-lock.json b/package-lock.json index 8016e6b0..a61dd6bf 100644 --- a/package-lock.json +++ b/package-lock.json @@ -18,6 +18,7 @@ "@codemirror/search": "~6.5.8", "@codemirror/state": "~6.5.1", "@codemirror/view": "~6.36.2", + "@diplodoc/utils": "^2.1.0", "@gravity-ui/i18n": "^1.7.0", "@gravity-ui/icons": "^2.12.0", "@lezer/highlight": "~1.2.1", @@ -2666,6 +2667,20 @@ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" } }, + "node_modules/@diplodoc/cut-extension/node_modules/@diplodoc/utils": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/@diplodoc/utils/-/utils-1.2.1.tgz", + "integrity": "sha512-2MXzSsm6KkYR8IfftRqnTKdhZSFfdAdjLymVZSHP2Ojzf+hmbl3xBYLLv4hgLfzXvwZWrbLV3dhUw1MVCLpueQ==", + "dev": true, + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0" + }, + "peerDependenciesMeta": { + "react": { + "optional": true + } + } + }, "node_modules/@diplodoc/directive": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/@diplodoc/directive/-/directive-0.3.0.tgz", @@ -2842,20 +2857,6 @@ } } }, - "node_modules/@diplodoc/transform/node_modules/@diplodoc/utils": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/@diplodoc/utils/-/utils-2.0.1.tgz", - "integrity": "sha512-BmEpoWG2fzaBlbS0l7o/nYc4Ww9QXeQGzHM6fTFk6gPV0PRl55aBxMx+60VZn7rDhiKwAQX97yTElBzoznTt4g==", - "dev": true, - "peerDependencies": { - "react": "^16.8.0 || ^17.0.0 || ^18.0.0" - }, - "peerDependenciesMeta": { - "react": { - "optional": true - } - } - }, "node_modules/@diplodoc/transform/node_modules/js-yaml": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", @@ -2869,10 +2870,17 @@ } }, "node_modules/@diplodoc/utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@diplodoc/utils/-/utils-1.0.0.tgz", - "integrity": "sha512-rGDVyqZyJ4GUjuUIYeMG7w6w5mb1dLF/nkloWEyxqZWy/POO4GiHAG83d4wK6U3gTFGTe+BXabQzdIKZwNVCTw==", - "dev": true + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/@diplodoc/utils/-/utils-2.1.0.tgz", + "integrity": "sha512-1XfZSb0gPLqSRGwxlLHcXo4c59bcFomcEaDM5v2S/aFDhgNRfZgDGxWEbHwkIijfBB2rvFWuVgKzON0VDp2uqQ==", + "peerDependencies": { + "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0" + }, + "peerDependenciesMeta": { + "react": { + "optional": true + } + } }, "node_modules/@esbuild/aix-ppc64": { "version": "0.21.5", diff --git a/package.json b/package.json index 1d23c396..e9360be2 100644 --- a/package.json +++ b/package.json @@ -181,6 +181,7 @@ "@codemirror/search": "~6.5.8", "@codemirror/state": "~6.5.1", "@codemirror/view": "~6.36.2", + "@diplodoc/utils": "^2.1.0", "@gravity-ui/i18n": "^1.7.0", "@gravity-ui/icons": "^2.12.0", "@lezer/highlight": "~1.2.1", diff --git a/src/bundle/types.ts b/src/bundle/types.ts index f5577c21..ad8a7e30 100644 --- a/src/bundle/types.ts +++ b/src/bundle/types.ts @@ -178,6 +178,13 @@ export type MarkdownEditorWysiwygConfig = { extensionOptions?: ExtensionsOptions; escapeConfig?: EscapeConfig; placeholderOptions?: WysiwygPlaceholderOptions; + // MAJOR: remove markdown-it-attrs + /** + * Disable the markdown-it-attrs plugin in the markup parser. + * + * Note: The use of the markdown-it-attrs plugin will be removed in the next major version. + */ + disableMarkdownAttrs?: boolean; }; export type MarkdownEditorOptions = { diff --git a/src/bundle/useMarkdownEditor.ts b/src/bundle/useMarkdownEditor.ts index 5bc7078b..1c415ce6 100644 --- a/src/bundle/useMarkdownEditor.ts +++ b/src/bundle/useMarkdownEditor.ts @@ -57,6 +57,7 @@ export function useMarkdownEditor( editor.emit('submit', null); return true; }, + disableMdAttrs: wysiwygConfig.disableMarkdownAttrs, preserveEmptyRows: experimental.preserveEmptyRows, placeholderOptions: wysiwygConfig.placeholderOptions, mdBreaks: md.breaks, diff --git a/src/bundle/wysiwyg-preset.ts b/src/bundle/wysiwyg-preset.ts index 13d4eb01..55cda48a 100644 --- a/src/bundle/wysiwyg-preset.ts +++ b/src/bundle/wysiwyg-preset.ts @@ -40,6 +40,8 @@ export type BundlePresetOptions = ExtensionsOptions & needToSetDimensionsForUploadedImages?: boolean; enableNewImageSizeCalculation?: boolean; directiveSyntax: DirectiveSyntaxContext; + // MAJOR: remove markdown-it-attrs + disableMdAttrs?: boolean; }; declare global { @@ -136,6 +138,7 @@ export const BundlePreset: ExtensionAuto = (builder, opts) }; const yfmOptions: BehaviorPresetOptions & YfmPresetOptions = { ...defaultOptions, + yfmConfigs: {disableAttrs: opts.disableMdAttrs, ...opts.yfmConfigs}, selectionContext: {config: wSelectionMenuConfigByPreset.yfm, ...opts.selectionContext}, commandMenu: {actions: wCommandMenuConfigByPreset.yfm, ...opts.commandMenu}, underline: {underlineKey: f.toPM(A.Underline), ...opts.underline}, diff --git a/src/extensions/yfm/YfmConfigs/YfmConfigsSpecs/index.ts b/src/extensions/yfm/YfmConfigs/YfmConfigsSpecs/index.ts index 5c88c1ac..68c3c0a9 100644 --- a/src/extensions/yfm/YfmConfigs/YfmConfigsSpecs/index.ts +++ b/src/extensions/yfm/YfmConfigs/YfmConfigsSpecs/index.ts @@ -1,7 +1,7 @@ -import attrsPlugin, {type AttrsOptions} from 'markdown-it-attrs'; // eslint-disable-line import/no-extraneous-dependencies +import attrsPlugin, {type AttrsOptions} from 'markdown-it-attrs'; -import type {ExtensionAuto} from '../../../../core'; -import {noop} from '../../../../lodash'; +import type {ExtensionAuto} from '#core'; +import {noop} from 'src/lodash'; const defaultAttrsOpts: AttrsOptions = { allowedAttributes: ['id'], @@ -10,12 +10,17 @@ const defaultAttrsOpts: AttrsOptions = { export type YfmConfigsSpecsOptions = { /** markdown-it-attrs options */ attrs?: AttrsOptions; + /** Disable markdown-it-attrs plugin */ + disableAttrs?: boolean; }; export const YfmConfigsSpecs: ExtensionAuto = (builder, opts) => { const attrsOpts = {...defaultAttrsOpts, ...opts.attrs}; - builder.configureMd((md) => md.use(attrsPlugin, attrsOpts), {text: false}); + // MAJOR: remove markdown-it-attrs + if (!opts.disableAttrs) { + builder.configureMd((md) => md.use(attrsPlugin, attrsOpts), {text: false}); + } // ignore yfm lint token builder.addNode('__yfm_lint', () => ({ diff --git a/src/extensions/yfm/YfmHeading/YfmHeading.test.ts b/src/extensions/yfm/YfmHeading/YfmHeading.test.ts index b9ebe145..a7da4088 100644 --- a/src/extensions/yfm/YfmHeading/YfmHeading.test.ts +++ b/src/extensions/yfm/YfmHeading/YfmHeading.test.ts @@ -1,4 +1,5 @@ import {builders} from 'prosemirror-test-builder'; +import dd from 'ts-dedent'; import {parseDOM} from '../../../../tests/parse-dom'; import {createMarkupChecker} from '../../../../tests/sameMarkup'; @@ -73,21 +74,21 @@ describe('Heading extension', () => { }); it('should parse few headings', () => { - const markup = ` -# h1 {#one} + const markup = dd` + # h1 {#one} -## h2 {#two} + ## h2 {#two} -### h3 {#three} + ### h3 {#three} -#### h4 {#four} + #### h4 {#four} -##### h5 {#five} + ##### h5 {#five} -###### h6 {#six} + ###### h6 {#six} -para -`.trim(); + para + `.trim(); same( markup, @@ -103,6 +104,59 @@ para ); }); + it('should parse headings with id without markdown-it-attrs', () => { + const markup = dd` + # h1 {#one} + + ## h2 {#two} + + ### h3 {#three} + + #### h4 {#four} + + ##### h5 {#five} + + ###### h6 {#six} + + para + `.trim(); + + const { + schema, + markupParser: parser, + serializer, + } = new ExtensionsManager({ + extensions: (builder) => + builder + .use(BaseSchemaSpecs, {}) + .use(YfmConfigsSpecs, {disableAttrs: true}) + .use(YfmHeadingSpecs, {}), + }).buildDeps(); + const {same} = createMarkupChecker({parser, serializer}); + + const {doc, p, h} = builders< + 'doc' | 'p' | 'h' | 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6', + 'b' + >(schema, { + doc: {nodeType: BaseNode.Doc}, + p: {nodeType: BaseNode.Paragraph}, + h: {nodeType: headingNodeName}, + }); + + same( + markup, + doc( + h({[YfmHeadingAttr.Level]: 1, [YfmHeadingAttr.Id]: 'one'}, 'h1'), + h({[YfmHeadingAttr.Level]: 2, [YfmHeadingAttr.Id]: 'two'}, 'h2'), + h({[YfmHeadingAttr.Level]: 3, [YfmHeadingAttr.Id]: 'three'}, 'h3'), + h({[YfmHeadingAttr.Level]: 4, [YfmHeadingAttr.Id]: 'four'}, 'h4'), + h({[YfmHeadingAttr.Level]: 5, [YfmHeadingAttr.Id]: 'five'}, 'h5'), + h({[YfmHeadingAttr.Level]: 6, [YfmHeadingAttr.Id]: 'six'}, 'h6'), + p('para'), + ), + ); + }); + it.each([1, 2, 3, 4, 5, 6])('should parse html - h%s tag', (lvl) => { parseDOM( schema, diff --git a/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/index.ts b/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/index.ts index a7edfd1e..042d30f0 100644 --- a/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/index.ts +++ b/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/index.ts @@ -1,8 +1,8 @@ -import type {Node, NodeSpec} from 'prosemirror-model'; - -import type {ExtensionAuto} from '../../../../core'; +import type {ExtensionAuto} from '#core'; +import type {Node, NodeSpec} from '#pm/model'; import {YfmHeadingAttr, headingNodeName} from './const'; +import {headingAttrsPlugin} from './markdown/heading-attrs'; import {getNodeAttrs} from './utils'; const DEFAULT_PLACEHOLDER = (node: Node) => 'Heading ' + node.attrs[YfmHeadingAttr.Level]; @@ -18,6 +18,7 @@ export type YfmHeadingSpecsOptions = { /** YfmHeading extension needs markdown-it-attrs plugin */ export const YfmHeadingSpecs: ExtensionAuto = (builder, opts) => { + builder.configureMd((md) => md.use(headingAttrsPlugin)); builder.addNode(headingNodeName, () => ({ spec: { attrs: { diff --git a/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/markdown/heading-attrs.ts b/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/markdown/heading-attrs.ts new file mode 100644 index 00000000..53df34d4 --- /dev/null +++ b/src/extensions/yfm/YfmHeading/YfmHeadingSpecs/markdown/heading-attrs.ts @@ -0,0 +1,51 @@ +import {parseMdAttrs} from '@diplodoc/utils'; +import type MarkdownIt from 'markdown-it'; + +export type HeadingAttrsOptions = { + /** Default – `['id']` */ + allowedAttributes?: string[]; +}; + +const defaultAllowedAttributes = ['id']; + +/** + * MarkdownIt plugin for parsing attributes in headings + */ +export const headingAttrsPlugin: MarkdownIt.PluginWithOptions = (md, opts) => { + const allowedAttributes = opts?.allowedAttributes || defaultAllowedAttributes; + + md.core.ruler.push('heading-attrs', (state) => { + const {tokens} = state; + + for (let i = 0; i < tokens.length; i++) { + const token = tokens[i]; + if (token.type !== 'heading_open') continue; + + const lastTextToken = tokens[i + 1]?.children?.at(-1); + if (lastTextToken?.type !== 'text') continue; + + const {content} = lastTextToken; + if (!content.endsWith('}')) continue; + + const idx = content.lastIndexOf('{'); + if (idx === -1) continue; + + const res = parseMdAttrs(md, content, idx, content.length); + if (!res) continue; + + lastTextToken.content = content.slice(0, idx).trimEnd(); + + for (const key of allowedAttributes) { + if (res.attrs[key]) { + if (key === 'class') { + const values = res.attrs[key]; + values.forEach((val) => token.attrJoin(key, val)); + } else { + const value = res.attrs[key][0]; + token.attrSet(key, value); + } + } + } + } + }); +};