@diplodoc/transform
Version:
A simple transformer of text in YFM (Yandex Flavored Markdown) to HTML
142 lines (113 loc) • 4.22 kB
text/typescript
import type {Delimiter} from 'markdown-it/lib/rules_inline/state_inline';
import type StateInline from 'markdown-it/lib/rules_inline/state_inline';
import type {MarkdownItPluginCb} from './typings';
const monospace: MarkdownItPluginCb = (md) => {
function postProcess(state: StateInline, delimiters: Delimiter[]) {
let token;
const loneMarkers = [];
for (let i = 0; i < delimiters.length; i++) {
const startDelim = delimiters[i];
if (startDelim.marker !== 0x23 /* # */) {
continue;
}
if (startDelim.end === -1) {
continue;
}
const endDelim = delimiters[startDelim.end];
token = state.tokens[startDelim.token];
token.type = 'monospace_open';
token.tag = 'samp';
token.nesting = 1;
token.markup = '##';
token.content = '';
token = state.tokens[endDelim.token];
token.type = 'monospace_close';
token.tag = 'samp';
token.nesting = -1;
token.markup = '##';
token.content = '';
if (
state.tokens[endDelim.token - 1].type === 'text' &&
state.tokens[endDelim.token - 1].content === '#'
) {
loneMarkers.push(endDelim.token - 1);
}
}
// If a marker sequence has an odd number of characters, it's splitted
// like this: `#####` -> `#` + `##` + `##`, leaving one marker at the
// start of the sequence.
//
// So, we have to move all those markers after subsequent s_close tags.
//
while (loneMarkers.length) {
const currentMarker = loneMarkers.pop();
if (typeof currentMarker === 'number') {
let nextMarker = currentMarker + 1;
while (
nextMarker < state.tokens.length &&
state.tokens[nextMarker].type === 'monospace_close'
) {
nextMarker++;
}
nextMarker--;
if (currentMarker !== nextMarker) {
token = state.tokens[nextMarker];
state.tokens[nextMarker] = state.tokens[currentMarker];
state.tokens[currentMarker] = token;
}
}
}
}
md.inline.ruler.before('emphasis', 'monospace', (state, silent) => {
let token;
const start = state.pos;
const marker = state.src.charCodeAt(start);
if (silent) {
return false;
}
if (marker !== 0x23 /* # */) {
return false;
}
const scanned = state.scanDelims(state.pos, true);
let len = scanned.length;
const ch = String.fromCharCode(marker);
if (len < 2) {
return false;
}
if (len % 2) {
token = state.push('text', '', 0);
token.content = ch;
len--;
}
for (let i = 0; i < len; i += 2) {
token = state.push('text', '', 0);
token.content = ch + ch;
if (!scanned.can_open && !scanned.can_close) {
continue;
}
state.delimiters.push({
marker: marker,
length: 0, // disable "rule of 3" length checks meant for emphasis
token: state.tokens.length - 1,
end: -1,
open: scanned.can_open,
close: scanned.can_close,
});
}
state.pos += scanned.length;
return true;
});
md.inline.ruler2.before('emphasis', 'monospace', (state) => {
const tokensMeta = state.tokens_meta;
const max = (state.tokens_meta || []).length;
postProcess(state, state.delimiters);
for (let curr = 0; curr < max; curr++) {
const currentToken = tokensMeta[curr];
if (currentToken && currentToken.delimiters) {
postProcess(state, currentToken.delimiters);
}
}
return true;
});
};
export = monospace;