2022-08-01 21:25:50 +00:00
|
|
|
import { compile as mdxCompile, nodeTypes } from '@mdx-js/mdx';
|
2022-07-21 20:43:58 +00:00
|
|
|
import mdxPlugin, { Options as MdxRollupPluginOptions } from '@mdx-js/rollup';
|
2022-08-05 23:58:09 +00:00
|
|
|
import type { AstroConfig, AstroIntegration } from 'astro';
|
2022-07-20 14:56:32 +00:00
|
|
|
import { parse as parseESM } from 'es-module-lexer';
|
2022-07-21 20:46:16 +00:00
|
|
|
import rehypeRaw from 'rehype-raw';
|
2022-07-21 01:36:26 +00:00
|
|
|
import remarkGfm from 'remark-gfm';
|
2022-07-21 20:46:16 +00:00
|
|
|
import remarkShikiTwoslash from 'remark-shiki-twoslash';
|
2022-07-21 01:36:26 +00:00
|
|
|
import remarkSmartypants from 'remark-smartypants';
|
2022-08-01 21:25:50 +00:00
|
|
|
import { VFile } from 'vfile';
|
2022-07-29 15:24:57 +00:00
|
|
|
import type { Plugin as VitePlugin } from 'vite';
|
2022-08-05 23:58:09 +00:00
|
|
|
import { rehypeApplyFrontmatterExport, remarkInitializeAstroData } from './astro-data-utils.js';
|
2022-08-01 21:25:50 +00:00
|
|
|
import rehypeCollectHeadings from './rehype-collect-headings.js';
|
2022-07-21 20:43:58 +00:00
|
|
|
import remarkPrism from './remark-prism.js';
|
2022-08-05 23:55:38 +00:00
|
|
|
import { getFileInfo, parseFrontmatter } from './utils.js';
|
2022-06-30 18:09:09 +00:00
|
|
|
|
2022-07-20 18:14:23 +00:00
|
|
|
type WithExtends<T> = T | { extends: T };
|
|
|
|
|
|
|
|
type MdxOptions = {
|
|
|
|
remarkPlugins?: WithExtends<MdxRollupPluginOptions['remarkPlugins']>;
|
|
|
|
rehypePlugins?: WithExtends<MdxRollupPluginOptions['rehypePlugins']>;
|
2022-07-21 01:36:26 +00:00
|
|
|
};
|
2022-07-20 18:14:23 +00:00
|
|
|
|
2022-08-06 03:51:38 +00:00
|
|
|
const DEFAULT_REMARK_PLUGINS: MdxRollupPluginOptions['remarkPlugins'] = [
|
|
|
|
remarkGfm,
|
|
|
|
remarkSmartypants,
|
|
|
|
];
|
|
|
|
const DEFAULT_REHYPE_PLUGINS: MdxRollupPluginOptions['rehypePlugins'] = [];
|
2022-07-20 18:14:23 +00:00
|
|
|
|
2022-07-21 20:46:16 +00:00
|
|
|
function handleExtends<T>(config: WithExtends<T[] | undefined>, defaults: T[] = []): T[] {
|
2022-07-20 18:14:23 +00:00
|
|
|
if (Array.isArray(config)) return config;
|
|
|
|
|
|
|
|
return [...defaults, ...(config?.extends ?? [])];
|
|
|
|
}
|
|
|
|
|
2022-08-05 23:55:38 +00:00
|
|
|
function getRemarkPlugins(
|
|
|
|
mdxOptions: MdxOptions,
|
|
|
|
config: AstroConfig
|
|
|
|
): MdxRollupPluginOptions['remarkPlugins'] {
|
|
|
|
let remarkPlugins = [
|
|
|
|
// Initialize vfile.data.astroExports before all plugins are run
|
|
|
|
remarkInitializeAstroData,
|
|
|
|
...handleExtends(mdxOptions.remarkPlugins, DEFAULT_REMARK_PLUGINS),
|
|
|
|
];
|
|
|
|
if (config.markdown.syntaxHighlight === 'shiki') {
|
|
|
|
// Default export still requires ".default" chaining for some reason
|
|
|
|
// Workarounds tried:
|
|
|
|
// - "import * as remarkShikiTwoslash"
|
|
|
|
// - "import { default as remarkShikiTwoslash }"
|
|
|
|
const shikiTwoslash = (remarkShikiTwoslash as any).default ?? remarkShikiTwoslash;
|
|
|
|
remarkPlugins.push([shikiTwoslash, config.markdown.shikiConfig]);
|
|
|
|
}
|
|
|
|
if (config.markdown.syntaxHighlight === 'prism') {
|
|
|
|
remarkPlugins.push(remarkPrism);
|
|
|
|
}
|
|
|
|
return remarkPlugins;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getRehypePlugins(
|
|
|
|
mdxOptions: MdxOptions,
|
|
|
|
config: AstroConfig
|
|
|
|
): MdxRollupPluginOptions['rehypePlugins'] {
|
|
|
|
let rehypePlugins = handleExtends(mdxOptions.rehypePlugins, DEFAULT_REHYPE_PLUGINS);
|
|
|
|
|
|
|
|
if (config.markdown.syntaxHighlight === 'shiki' || config.markdown.syntaxHighlight === 'prism') {
|
2022-08-10 21:10:06 +00:00
|
|
|
rehypePlugins.unshift([rehypeRaw, { passThrough: nodeTypes }]);
|
2022-08-05 23:55:38 +00:00
|
|
|
}
|
2022-08-06 03:51:38 +00:00
|
|
|
// getHeadings() is guaranteed by TS, so we can't allow user to override
|
2022-08-10 21:10:06 +00:00
|
|
|
rehypePlugins.unshift(rehypeCollectHeadings);
|
2022-08-05 23:55:38 +00:00
|
|
|
|
|
|
|
return rehypePlugins;
|
|
|
|
}
|
|
|
|
|
2022-07-20 18:14:23 +00:00
|
|
|
export default function mdx(mdxOptions: MdxOptions = {}): AstroIntegration {
|
2022-06-30 18:09:09 +00:00
|
|
|
return {
|
2022-06-30 18:11:12 +00:00
|
|
|
name: '@astrojs/mdx',
|
|
|
|
hooks: {
|
2022-07-20 14:56:32 +00:00
|
|
|
'astro:config:setup': ({ updateConfig, config, addPageExtension, command }: any) => {
|
2022-06-30 18:11:12 +00:00
|
|
|
addPageExtension('.mdx');
|
2022-07-21 20:43:58 +00:00
|
|
|
|
2022-08-01 21:23:56 +00:00
|
|
|
const mdxPluginOpts: MdxRollupPluginOptions = {
|
2022-08-05 23:55:38 +00:00
|
|
|
remarkPlugins: getRemarkPlugins(mdxOptions, config),
|
|
|
|
rehypePlugins: getRehypePlugins(mdxOptions, config),
|
2022-07-29 15:22:57 +00:00
|
|
|
jsx: true,
|
|
|
|
jsxImportSource: 'astro',
|
|
|
|
// Note: disable `.md` support
|
|
|
|
format: 'mdx',
|
|
|
|
mdExtensions: [],
|
2022-08-01 21:23:56 +00:00
|
|
|
};
|
2022-07-29 15:22:57 +00:00
|
|
|
|
2022-06-30 18:11:12 +00:00
|
|
|
updateConfig({
|
|
|
|
vite: {
|
|
|
|
plugins: [
|
|
|
|
{
|
|
|
|
enforce: 'pre',
|
2022-08-01 21:23:56 +00:00
|
|
|
...mdxPlugin(mdxPluginOpts),
|
|
|
|
// Override transform to alter code before MDX compilation
|
|
|
|
// ex. inject layouts
|
|
|
|
async transform(code, id) {
|
|
|
|
if (!id.endsWith('mdx')) return;
|
2022-07-29 15:22:57 +00:00
|
|
|
|
2022-08-05 23:55:38 +00:00
|
|
|
let { data: frontmatter, content: pageContent } = parseFrontmatter(code, id);
|
|
|
|
if (frontmatter.layout) {
|
|
|
|
const { layout, ...contentProp } = frontmatter;
|
|
|
|
pageContent += `\n\nexport default async function({ children }) {\nconst Layout = (await import(${JSON.stringify(
|
|
|
|
frontmatter.layout
|
|
|
|
)})).default;\nconst frontmatter=${JSON.stringify(
|
|
|
|
contentProp
|
|
|
|
)};\nreturn <Layout frontmatter={frontmatter} content={frontmatter} headings={getHeadings()}>{children}</Layout> }`;
|
2022-07-29 15:22:57 +00:00
|
|
|
}
|
2022-08-01 21:23:56 +00:00
|
|
|
|
2022-08-05 23:55:38 +00:00
|
|
|
const compiled = await mdxCompile(new VFile({ value: pageContent, path: id }), {
|
|
|
|
...mdxPluginOpts,
|
|
|
|
rehypePlugins: [
|
|
|
|
...(mdxPluginOpts.rehypePlugins ?? []),
|
2022-08-08 22:33:35 +00:00
|
|
|
() => rehypeApplyFrontmatterExport(frontmatter),
|
2022-08-05 23:55:38 +00:00
|
|
|
],
|
|
|
|
});
|
2022-08-01 21:23:56 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
code: String(compiled.value),
|
|
|
|
map: compiled.map,
|
|
|
|
};
|
2022-07-29 15:24:57 +00:00
|
|
|
},
|
2022-06-30 18:11:12 +00:00
|
|
|
},
|
2022-07-20 14:56:32 +00:00
|
|
|
{
|
2022-08-01 21:23:56 +00:00
|
|
|
name: '@astrojs/mdx-postprocess',
|
|
|
|
// These transforms must happen *after* JSX runtime transformations
|
2022-07-29 15:22:57 +00:00
|
|
|
transform(code, id) {
|
2022-06-30 18:11:12 +00:00
|
|
|
if (!id.endsWith('.mdx')) return;
|
2022-07-20 14:56:32 +00:00
|
|
|
const [, moduleExports] = parseESM(code);
|
|
|
|
|
2022-07-28 14:58:44 +00:00
|
|
|
const { fileUrl, fileId } = getFileInfo(id, config);
|
2022-07-20 14:56:32 +00:00
|
|
|
if (!moduleExports.includes('url')) {
|
|
|
|
code += `\nexport const url = ${JSON.stringify(fileUrl)};`;
|
|
|
|
}
|
2022-07-28 14:58:44 +00:00
|
|
|
if (!moduleExports.includes('file')) {
|
|
|
|
code += `\nexport const file = ${JSON.stringify(fileId)};`;
|
|
|
|
}
|
|
|
|
|
2022-07-20 14:56:32 +00:00
|
|
|
if (command === 'dev') {
|
|
|
|
// TODO: decline HMR updates until we have a stable approach
|
|
|
|
code += `\nif (import.meta.hot) {
|
2022-06-30 18:09:09 +00:00
|
|
|
import.meta.hot.decline();
|
2022-07-20 14:58:33 +00:00
|
|
|
}`;
|
2022-07-20 14:56:32 +00:00
|
|
|
}
|
|
|
|
return code;
|
2022-06-30 18:11:12 +00:00
|
|
|
},
|
|
|
|
},
|
2022-07-29 15:22:57 +00:00
|
|
|
] as VitePlugin[],
|
2022-06-30 18:11:12 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
},
|
|
|
|
};
|
2022-06-30 18:09:09 +00:00
|
|
|
}
|