mirror of
https://github.com/reactjs/react.dev.git
synced 2026-02-22 03:42:14 +00:00
* Skip smartypants on TerminalBlock * Improve TerminalBlock HTML tags * Remove unnecessary TerminalBlock escapes from docs * Bump DISK_CACHE_BREAKER
88 lines
2.4 KiB
JavaScript
88 lines
2.4 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
/*!
|
|
* Based on 'silvenon/remark-smartypants'
|
|
* https://github.com/silvenon/remark-smartypants/pull/80
|
|
*/
|
|
|
|
const visit = require('unist-util-visit');
|
|
const retext = require('retext');
|
|
const smartypants = require('retext-smartypants');
|
|
|
|
function check(node, parent) {
|
|
if (node.data?.skipSmartyPants) return false;
|
|
if (parent.tagName === 'script') return false;
|
|
if (parent.tagName === 'style') return false;
|
|
return true;
|
|
}
|
|
|
|
function markSkip(node) {
|
|
if (!node) return;
|
|
node.data ??= {};
|
|
node.data.skipSmartyPants = true;
|
|
if (Array.isArray(node.children)) {
|
|
for (const child of node.children) {
|
|
markSkip(child);
|
|
}
|
|
}
|
|
}
|
|
|
|
module.exports = function (options) {
|
|
const processor = retext().use(smartypants, {
|
|
...options,
|
|
// Do not replace ellipses, dashes, backticks because they change string
|
|
// length, and we couldn't guarantee right splice of text in second visit of
|
|
// tree
|
|
ellipses: false,
|
|
dashes: false,
|
|
backticks: false,
|
|
});
|
|
|
|
const processor2 = retext().use(smartypants, {
|
|
...options,
|
|
// Do not replace quotes because they are already replaced in the first
|
|
// processor
|
|
quotes: false,
|
|
});
|
|
|
|
function transformer(tree) {
|
|
let allText = '';
|
|
let startIndex = 0;
|
|
const textOrInlineCodeNodes = [];
|
|
|
|
visit(tree, 'mdxJsxFlowElement', (node) => {
|
|
if (['TerminalBlock'].includes(node.name)) {
|
|
markSkip(node); // Mark all children to skip smarty pants
|
|
}
|
|
});
|
|
|
|
visit(tree, ['text', 'inlineCode'], (node, _, parent) => {
|
|
if (check(node, parent)) {
|
|
if (node.type === 'text') allText += node.value;
|
|
// for the case when inlineCode contains just one part of quote: `foo'bar`
|
|
else allText += 'A'.repeat(node.value.length);
|
|
textOrInlineCodeNodes.push(node);
|
|
}
|
|
});
|
|
|
|
// Concat all text into one string, to properly replace quotes around non-"text" nodes
|
|
allText = String(processor.processSync(allText));
|
|
|
|
for (const node of textOrInlineCodeNodes) {
|
|
const endIndex = startIndex + node.value.length;
|
|
if (node.type === 'text') {
|
|
const processedText = allText.slice(startIndex, endIndex);
|
|
node.value = String(processor2.processSync(processedText));
|
|
}
|
|
startIndex = endIndex;
|
|
}
|
|
}
|
|
|
|
return transformer;
|
|
};
|