diff --git a/__tests__/html2/markdown/math/layout.3.html b/__tests__/html2/markdown/math/layout.3.html index cae136727d..cad76722ba 100644 --- a/__tests__/html2/markdown/math/layout.3.html +++ b/__tests__/html2/markdown/math/layout.3.html @@ -22,8 +22,7 @@ ## Basic Math -1. Simple arithmetic: -\(2 + 2 = 4\) +1. Simple arithmetic: \(2 + 2 = 4\) 2. Fractions: \[\frac{1}{2} + \frac{1}{3} = \frac{5}{6}\] @@ -83,11 +82,9 @@ ## Invalid input examples -12. Wrong expression is rendered: -\(2++2\) +12. Wrong expression is rendered: \(2++2\) -13. Inline closing delimeter is required: -\(x^2 +13. Inline closing delimeter is required: \(x^2 14. Katex syntax error: \[\int_0^\infty e^{-x} dx = 1 +}\] diff --git a/packages/bundle/src/markdown/mathExtension/math.ts b/packages/bundle/src/markdown/mathExtension/math.ts index 764f7c1535..01d0479d3d 100644 --- a/packages/bundle/src/markdown/mathExtension/math.ts +++ b/packages/bundle/src/markdown/mathExtension/math.ts @@ -1,20 +1,20 @@ import type { Extension } from 'micromark-util-types'; -import { BACKSLASH, DOLLAR } from './constants'; -import { createTokenizer } from './tokenizer'; +import { BACKSLASH, CLOSE_BRACKET, CLOSE_PAREN, DOLLAR, OPEN_BRACKET, OPEN_PAREN } from './constants'; +import makeConstructTokenizer from './tokenizer'; export default function math(): Extension { - const construct = { + const makeConstruct = (...args: Parameters) => ({ name: 'math', - tokenize: createTokenizer - }; + tokenize: makeConstructTokenizer(...args) + }); return { text: { - [BACKSLASH]: construct + [BACKSLASH]: makeConstruct({ OPEN_CODE: OPEN_PAREN, CLOSE_CODE: CLOSE_PAREN }) }, flow: { - [BACKSLASH]: construct, - [DOLLAR]: construct + [BACKSLASH]: makeConstruct({ OPEN_CODE: OPEN_BRACKET, CLOSE_CODE: CLOSE_BRACKET }), + [DOLLAR]: makeConstruct({ OPEN_CODE: DOLLAR, CLOSE_CODE: DOLLAR }) } } as any; } diff --git a/packages/bundle/src/markdown/mathExtension/tokenizer.ts b/packages/bundle/src/markdown/mathExtension/tokenizer.ts index 90c9441353..a38431e565 100644 --- a/packages/bundle/src/markdown/mathExtension/tokenizer.ts +++ b/packages/bundle/src/markdown/mathExtension/tokenizer.ts @@ -1,4 +1,3 @@ -/* eslint-disable no-magic-numbers */ /* eslint-disable @typescript-eslint/no-use-before-define */ import { BACKSLASH, OPEN_PAREN, CLOSE_PAREN, OPEN_BRACKET, CLOSE_BRACKET, DOLLAR } from './constants'; import { markdownLineEnding } from 'micromark-util-character'; @@ -6,88 +5,71 @@ import { type Code, type Effects, type State } from 'micromark-util-types'; type MathTokenTypes = 'math' | 'mathChunk'; +type OpenCode = typeof OPEN_BRACKET | typeof OPEN_PAREN | typeof DOLLAR; +type CloseCode = typeof CLOSE_BRACKET | typeof CLOSE_PAREN | typeof DOLLAR; + type MathEffects = Omit & { enter(type: MathTokenTypes): void; exit(type: MathTokenTypes): void; }; -export function createTokenizer(effects: MathEffects, ok: State, nok: State) { - let expectedCloseDelimiter: number; - let dollarDelimiterCount = 0; +/** + * Creates a math tokenizer for specified delimiter pair + * @param OPEN_CODE - Opening delimiter code + * @param CLOSE_CODE - Closing delimiter code + */ +export default ({ OPEN_CODE, CLOSE_CODE }: { OPEN_CODE: OpenCode; CLOSE_CODE: CloseCode }) => + function createTokenizer(effects: MathEffects, ok: State, nok: State) { + return start; + + function start(code: Code): State { + if (code === BACKSLASH || (code === DOLLAR && OPEN_CODE === DOLLAR)) { + effects.enter('math'); + effects.enter('mathChunk'); + effects.consume(code); + return openDelimiter; + } + + return nok(code); + } - return start; + function openDelimiter(code: Code): State { + if (code !== OPEN_CODE) { + return nok(code); + } - function start(code: Code): State { - if (code === BACKSLASH || code === DOLLAR) { - effects.enter('math'); - effects.enter('mathChunk'); effects.consume(code); - dollarDelimiterCount = code === DOLLAR ? 1 : 0; - return openDelimiter; + return content; } - return nok(code); - } - - function openDelimiter(code: Code): State { - switch (code) { - case OPEN_PAREN: - expectedCloseDelimiter = CLOSE_PAREN; - break; - case OPEN_BRACKET: - expectedCloseDelimiter = CLOSE_BRACKET; - break; - case DOLLAR: - expectedCloseDelimiter = DOLLAR; - dollarDelimiterCount++; - if (dollarDelimiterCount !== 2) { - return nok(code); - } - break; - default: + function content(code: Code): State { + if (code === null) { return nok(code); - } - effects.consume(code); - return content; - } + } - function content(code: Code): State { - if (code === null) { - return nok(code); - } + if (code === BACKSLASH || (CLOSE_CODE === DOLLAR && code === DOLLAR)) { + effects.consume(code); + return maybeCloseDelimiter; + } - if (code === BACKSLASH || (dollarDelimiterCount && code === DOLLAR)) { effects.consume(code); - code === DOLLAR && dollarDelimiterCount--; - return maybeCloseDelimiter; - } - effects.consume(code); + if (markdownLineEnding(code)) { + effects.exit('mathChunk'); + effects.enter('mathChunk'); + } - if (markdownLineEnding(code)) { - effects.exit('mathChunk'); - effects.enter('mathChunk'); + return content; } - return content; - } - - function maybeCloseDelimiter(code: Code): State { - if (code === expectedCloseDelimiter) { - code === DOLLAR && dollarDelimiterCount--; - if (dollarDelimiterCount !== 0) { - return nok(code); + function maybeCloseDelimiter(code: Code): State { + if (code === CLOSE_CODE) { + effects.consume(code); + effects.exit('mathChunk'); + effects.exit('math'); + return ok; } - effects.consume(code); - effects.exit('mathChunk'); - effects.exit('math'); - - dollarDelimiterCount = 0; - expectedCloseDelimiter = undefined; - return ok; + return content(code); } - - return content(code); - } -} + };