diff --git a/src/transform/index.ts b/src/transform/index.ts index 3ca9dfa3..35fbfbaa 100644 --- a/src/transform/index.ts +++ b/src/transform/index.ts @@ -1,9 +1,9 @@ -import type {EnvType, OptionsType, OutputType} from './typings'; +import type {EnvType, OptionsType, OutputType, RootCollectorOptions} from './typings'; import {bold} from 'chalk'; import {log} from './log'; -import liquidSnippet from './liquid'; +import liquidDocument from './liquid'; import initMarkdownIt from './md'; function applyLiquid(input: string, options: OptionsType) { @@ -17,7 +17,7 @@ function applyLiquid(input: string, options: OptionsType) { return disableLiquid || isLiquided ? input - : liquidSnippet(input, vars, path, {conditionsInCode}); + : liquidDocument(input, vars, path, {conditionsInCode}); } function handleError(error: unknown, path?: string): never { @@ -33,8 +33,14 @@ function emitResult(html: string, env: EnvType): OutputType { }; } +type TransformFunction = { + (originInput: string, options?: OptionsType): OutputType; + // eslint-disable-next-line @typescript-eslint/no-explicit-any + collect: (input: string, options: RootCollectorOptions) => string; +}; + // eslint-disable-next-line consistent-return -function transform(originInput: string, options: OptionsType = {}) { +const transform: TransformFunction = (originInput: string, options: OptionsType = {}) => { const input = applyLiquid(originInput, options); const {parse, compile, env} = initMarkdownIt(options); @@ -43,11 +49,37 @@ function transform(originInput: string, options: OptionsType = {}) { } catch (error) { handleError(error, options.path); } -} +}; + +transform.collect = ( + input: string, + // eslint-disable-next-line @typescript-eslint/no-explicit-any + {mdItInitOptions, pluginCollectOptions}: RootCollectorOptions, +) => { + const maybeLiquidedInput = applyLiquid(input, mdItInitOptions); + const {parse} = initMarkdownIt(mdItInitOptions); + + const plugins = mdItInitOptions.plugins ?? []; + + try { + const tokenStream = parse(maybeLiquidedInput); + + return plugins.reduce((collected, plugin) => { + const collectOutput = plugin.collect?.(collected, { + ...pluginCollectOptions, + tokenStream, + }); + + return collectOutput ?? collected; + }, input); + } catch (error) { + handleError(error, mdItInitOptions.path); + } +}; export = transform; -// eslint-disable-next-line @typescript-eslint/no-namespace, no-redeclare -- backward compatibility +// eslint-disable-next-line @typescript-eslint/no-namespace, @typescript-eslint/no-redeclare -- backward compatibility namespace transform { export type Options = OptionsType; export type Output = OutputType; diff --git a/src/transform/typings.ts b/src/transform/typings.ts index 0b67a801..1616fbde 100644 --- a/src/transform/typings.ts +++ b/src/transform/typings.ts @@ -1,5 +1,5 @@ import {LanguageFn} from 'highlight.js'; -import DefaultMarkdownIt from 'markdown-it'; +import DefaultMarkdownIt, {Token} from 'markdown-it'; import DefaultStateCore from 'markdown-it/lib/rules_core/state_core'; import {SanitizeOptions} from './sanitize'; @@ -47,7 +47,7 @@ export interface OptionsType { sanitizeOptions?: SanitizeOptions; needFlatListHeadings?: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any - plugins?: MarkdownItPluginCb[]; + plugins?: ExtendedPluginWithCollect[]; preprocessors?: MarkdownItPreprocessorCb[]; // Preprocessors should modify the input before passing it to MD highlightLangs?: HighlightLangMap; disableRules?: string[]; @@ -96,8 +96,24 @@ export type MarkdownItPluginCb = { (md: MarkdownIt, opts: T & MarkdownItPluginOpts): void; }; +export type IntrinsicCollectOptions = { + tokenStream: Token[]; +}; + +export type ExtendedPluginWithCollect< + PluginRegularOptions extends {} = {}, + PluginCollectOptions = {}, +> = MarkdownItPluginCb & { + collect?: (input: string, options: PluginCollectOptions & IntrinsicCollectOptions) => string; +}; + +export type RootCollectorOptions = { + mdItInitOptions: OptionsType; + pluginCollectOptions: PluginCollectOptions; +}; + export type MarkdownItPreprocessorCb = { - (input: string, opts: T & Partial, md?: MarkdownIt): string; + (input: string, opts: T & Partial, md?: MarkdownIt): string | void; }; export type CssWhiteList = {[property: string]: boolean};