2020-03-13 23:46:14 +00:00
|
|
|
/* eslint prefer-const: 0*/
|
|
|
|
|
2020-03-09 23:24:57 +00:00
|
|
|
// Note: this is copied from https://github.com/markdown-it/markdown-it/blob/master/lib/renderer.js
|
|
|
|
// Markdown-it assigns a special meaning to code returned from highlight() when it starts with PRE or not.
|
|
|
|
// If it starts with PRE, the highlited code is returned as-is. If it does not, it is wrapped in <PRE><CODE>
|
|
|
|
// This is a bit of a hack and magic behaviour, and it prevents us from returning a DIV from the highlight
|
|
|
|
// function.
|
|
|
|
// So we modify the code below to allow highlight() to return an object that tells how to render
|
|
|
|
// the code.
|
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2020-11-12 19:13:28 +00:00
|
|
|
function plugin(markdownIt: any) {
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2023-06-30 10:22:47 +01:00
|
|
|
markdownIt.renderer.rules.fence = function(tokens: any[], idx: number, options: any, _env: any, slf: any) {
|
2020-03-13 23:46:14 +00:00
|
|
|
let token = tokens[idx],
|
2020-03-09 23:24:57 +00:00
|
|
|
info = token.info ? markdownIt.utils.unescapeAll(token.info).trim() : '',
|
|
|
|
langName = '',
|
|
|
|
highlighted, i, tmpAttrs, tmpToken;
|
|
|
|
|
|
|
|
if (info) {
|
|
|
|
langName = info.split(/\s+/g)[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.highlight) {
|
|
|
|
highlighted = options.highlight(token.content, langName) || markdownIt.utils.escapeHtml(token.content);
|
|
|
|
} else {
|
|
|
|
highlighted = markdownIt.utils.escapeHtml(token.content);
|
|
|
|
}
|
|
|
|
|
|
|
|
const wrapCode = highlighted && highlighted.wrapCode !== false;
|
|
|
|
highlighted = typeof highlighted !== 'string' ? highlighted.html : highlighted;
|
|
|
|
|
|
|
|
if (highlighted.indexOf('<pre') === 0 || !wrapCode) {
|
|
|
|
return `${highlighted}\n`;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If language exists, inject class gently, without modifying original token.
|
|
|
|
// May be, one day we will add .clone() for token and simplify this part, but
|
|
|
|
// now we prefer to keep things local.
|
|
|
|
if (info) {
|
2021-01-23 15:51:19 +00:00
|
|
|
i = token.attrIndex('class');
|
2020-03-09 23:24:57 +00:00
|
|
|
tmpAttrs = token.attrs ? token.attrs.slice() : [];
|
|
|
|
|
|
|
|
if (i < 0) {
|
|
|
|
tmpAttrs.push(['class', options.langPrefix + langName]);
|
|
|
|
} else {
|
|
|
|
tmpAttrs[i][1] += ` ${options.langPrefix}${langName}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fake token just to render attributes
|
|
|
|
tmpToken = {
|
|
|
|
attrs: tmpAttrs,
|
|
|
|
};
|
|
|
|
|
2021-01-23 15:51:19 +00:00
|
|
|
return `<pre><code${slf.renderAttrs(tmpToken)}>${
|
2020-03-09 23:24:57 +00:00
|
|
|
highlighted
|
|
|
|
}</code></pre>\n`;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-01-23 15:51:19 +00:00
|
|
|
return `<pre><code${slf.renderAttrs(token)}>${
|
2020-03-09 23:24:57 +00:00
|
|
|
highlighted
|
|
|
|
}</code></pre>\n`;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-10-21 00:23:55 +01:00
|
|
|
export default {
|
|
|
|
plugin,
|
|
|
|
};
|
|
|
|
|