mirror of
https://github.com/facebook/docusaurus.git
synced 2025-08-06 10:20:09 +02:00
* feat(v2): add ability to set custom heading id * Add cli command * Fix slugger * write-heading-ids doc + add in commands/templates * refactor + add tests for writeHeadingIds * polish writeHeadingIds * polish writeHeadingIds * remove i18n goals todo section as the remaining items are quite abstract/useless * fix edge case with 2 md links in heading * extract parseMarkdownHeadingId helper function * refactor using the shared parseMarkdownHeadingId utility fn * change logic of edge case * Handle edge case * Document explicit ids feature Co-authored-by: slorber <lorber.sebastien@gmail.com>
96 lines
2.7 KiB
JavaScript
96 lines
2.7 KiB
JavaScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
const {getOptions} = require('loader-utils');
|
|
const {readFile} = require('fs-extra');
|
|
const mdx = require('@mdx-js/mdx');
|
|
const emoji = require('remark-emoji');
|
|
const matter = require('gray-matter');
|
|
const stringifyObject = require('stringify-object');
|
|
const headings = require('./remark/headings');
|
|
const toc = require('./remark/toc');
|
|
const unwrapMdxCodeBlocks = require('./remark/unwrapMdxCodeBlocks');
|
|
const transformImage = require('./remark/transformImage');
|
|
const transformLinks = require('./remark/transformLinks');
|
|
|
|
const DEFAULT_OPTIONS = {
|
|
rehypePlugins: [],
|
|
remarkPlugins: [unwrapMdxCodeBlocks, emoji, headings, toc],
|
|
};
|
|
|
|
module.exports = async function docusaurusMdxLoader(fileString) {
|
|
const callback = this.async();
|
|
|
|
const {data, content} = matter(fileString);
|
|
const reqOptions = getOptions(this) || {};
|
|
const options = {
|
|
...reqOptions,
|
|
remarkPlugins: [
|
|
...(reqOptions.beforeDefaultRemarkPlugins || []),
|
|
...DEFAULT_OPTIONS.remarkPlugins,
|
|
[
|
|
transformImage,
|
|
{staticDir: reqOptions.staticDir, filePath: this.resourcePath},
|
|
],
|
|
[
|
|
transformLinks,
|
|
{staticDir: reqOptions.staticDir, filePath: this.resourcePath},
|
|
],
|
|
...(reqOptions.remarkPlugins || []),
|
|
],
|
|
rehypePlugins: [
|
|
...(reqOptions.beforeDefaultRehypePlugins || []),
|
|
...DEFAULT_OPTIONS.rehypePlugins,
|
|
|
|
...(reqOptions.rehypePlugins || []),
|
|
],
|
|
filepath: this.resourcePath,
|
|
};
|
|
let result;
|
|
|
|
try {
|
|
result = await mdx(content, options);
|
|
} catch (err) {
|
|
return callback(err);
|
|
}
|
|
|
|
let exportStr = `export const frontMatter = ${stringifyObject(data)};`;
|
|
|
|
// Read metadata for this MDX and export it.
|
|
if (options.metadataPath && typeof options.metadataPath === 'function') {
|
|
const metadataPath = options.metadataPath(this.resourcePath);
|
|
|
|
if (metadataPath) {
|
|
// Add as dependency of this loader result so that we can
|
|
// recompile if metadata is changed.
|
|
this.addDependency(metadataPath);
|
|
const metadata = await readFile(metadataPath, 'utf8');
|
|
exportStr += `\nexport const metadata = ${metadata};`;
|
|
}
|
|
}
|
|
|
|
if (
|
|
options.forbidFrontMatter &&
|
|
typeof options.forbidFrontMatter === 'function'
|
|
) {
|
|
if (
|
|
options.forbidFrontMatter(this.resourcePath) &&
|
|
Object.keys(data).length > 0
|
|
) {
|
|
return callback(new Error(`Front matter is forbidden in this file`));
|
|
}
|
|
}
|
|
const code = `
|
|
import React from 'react';
|
|
import { mdx } from '@mdx-js/react';
|
|
|
|
${exportStr}
|
|
${result}
|
|
`;
|
|
|
|
return callback(null, code);
|
|
};
|