By Titus Wormer
MDX on demand
This guide shows how to use @mdx-js/mdx to compile MDX on the server and run the result on clients. Some frameworks, such as Next.js and Remix, make it easy to split work between servers and clients. Using that it’s possible to for example do most of the work on demand on the server instead of at build time, then pass the resulting data to clients, where they finally use it.
This is similar to what people sometimes use mdx-bundler or next-mdx-remote for, but MDX also supports it.
Quick example
On the server:
import {compile} from '@mdx-js/mdx'
const code = String(await compile('# hi', {
outputFormat: 'function-body',
/* …otherOptions */
}))
// To do: send `code` to the client somehow.
(alias) function compile(vfileCompatible: Readonly<Compatible>, compileOptions?: Readonly<CompileOptions> | null | undefined): Promise<VFile>
import compileCompile MDX to JS.
- @param vfileCompatible MDX document to parse.
- @param compileOptions Compile configuration (optional).
- @return Promise to compiled file.
const code: stringvar String: StringConstructor
(value?: any) => stringAllows manipulation and formatting of text strings and determination and location of substrings within strings.
(alias) compile(vfileCompatible: Readonly<Compatible>, compileOptions?: Readonly<CompileOptions> | null | undefined): Promise<VFile>
import compileCompile MDX to JS.
- @param vfileCompatible MDX document to parse.
- @param compileOptions Compile configuration (optional).
- @return Promise to compiled file.
(property) outputFormat?: "function-body" | "program" | null | undefinedOutput format to generate (default: 'program'); in most cases 'program' should be used, it results in a whole program; internally evaluate uses 'function-body' to compile to code that can be passed to run; in some cases, you might want what evaluate does in separate steps, such as when compiling on the server and running on the client.
On the client:
import {run} from '@mdx-js/mdx'
import * as runtime from 'react/jsx-runtime'
const code = '' // To do: get `code` from server somehow.
const {default: Content} = await run(code, {...runtime, baseUrl: import.meta.url})
(alias) function run(code: {
toString(): string;
}, options: RunOptions): Promise<MDXModule>
import runRun code compiled with outputFormat: 'function-body'.
☢️ Danger: this
evals JavaScript.
- @param code JavaScript function body to run.
- @param options Configuration (required).
- @return Promise to a module; the result is an object with a
defaultfield set to the component; anything else that was exported is available too.
import runtimeconst code: ""(property) MDXModule.default: MDXContentA functional JSX component which renders the content of the MDX file.
const Content: MDXContentA functional JSX component which renders the content of the MDX file.
(alias) run(code: {
toString(): string;
}, options: RunOptions): Promise<MDXModule>
import runRun code compiled with outputFormat: 'function-body'.
☢️ Danger: this
evals JavaScript.
- @param code JavaScript function body to run.
- @param options Configuration (required).
- @return Promise to a module; the result is an object with a
defaultfield set to the component; anything else that was exported is available too.
const code: ""import runtime(property) baseUrl?: string | URL | null | undefinedUse this URL as import.meta.url and resolve import and export … from relative to it (optional, example: import.meta.url); this option can also be given at compile time in CompileOptions; you should pass this (likely at runtime), as you might get runtime errors when using import.meta.url / import / export … from otherwise.
The type of import.meta.
If you need to declare that a given property exists on import.meta, this type may be augmented via interface merging.
(property) ImportMeta.url: stringThe absolute file: URL of the module.
Content is now an MDXContent component that you can use like normal in your framework (see § Using MDX).
More information is available in the API docs of @mdx-js/mdx for compile and run. For other use cases, you can also use evaluate, which both compiles and runs in one.
Note: MDX is not a bundler (esbuild, webpack, and Rollup are bundlers): you can’t import other code from the server within the string of MDX and get a nicely minified bundle out or so.
Next.js example
Some frameworks let you write the server and client code in one file, such as Next.
/**
* @import {MDXModule} from 'mdx/types.js'
* @import {Dispatch, SetStateAction} from 'react'
*/
import {compile, run} from '@mdx-js/mdx'
import {Fragment, useEffect, useState} from 'react'
import * as runtime from 'react/jsx-runtime'
/**
* @param {{code: string}} props
* @returns {JSX.Element}
*/
export default function Page({code}) {
/** @type {[MDXModule | undefined, Dispatch<SetStateAction<MDXModule | undefined>>]} */
const [mdxModule, setMdxModule] = useState()
const Content = mdxModule ? mdxModule.default : Fragment
useEffect(
function () {
;(async function () {
setMdxModule(await run(code, {...runtime, baseUrl: import.meta.url}))
})()
},
[code]
)
return <Content />
}
export async function getStaticProps() {
const code = String(
await compile('# hi', {
outputFormat: 'function-body'
/* …otherOptions */
})
)
return {props: {code}}
}
(alias) function compile(vfileCompatible: Readonly<Compatible>, compileOptions?: Readonly<CompileOptions> | null | undefined): Promise<VFile>
import compileCompile MDX to JS.
- @param vfileCompatible MDX document to parse.
- @param compileOptions Compile configuration (optional).
- @return Promise to compiled file.
(alias) function run(code: {
toString(): string;
}, options: RunOptions): Promise<MDXModule>
import runRun code compiled with outputFormat: 'function-body'.
☢️ Danger: this
evals JavaScript.
- @param code JavaScript function body to run.
- @param options Configuration (required).
- @return Promise to a module; the result is an object with a
defaultfield set to the component; anything else that was exported is available too.
(alias) const Fragment: React.ExoticComponent<{
children?: React.ReactNode | undefined;
}>
import FragmentLets you group elements without a wrapper node.
- @see {@link https://react.dev/reference/react/Fragment React Docs}
- @example
import { Fragment } from 'react'; <Fragment> <td>Hello</td> <td>World</td> </Fragment> - @example
// Using the <></> shorthand syntax: <> <td>Hello</td> <td>World</td> </>
(alias) function useEffect(effect: React.EffectCallback, deps?: React.DependencyList): void
import useEffectAccepts a function that contains imperative, possibly effectful code.
- @param effect Imperative function that can return a cleanup function
- @param deps If present, effect will only activate if the values in the list change.
- @version 16.8.0
- @see {@link https://react.dev/reference/react/useEffect}
(alias) function useState<S>(initialState: S | (() => S)): [S, Dispatch<SetStateAction<S>>] (+1 overload)
import useStateReturns a stateful value, and a function to update it.
- @version 16.8.0
- @see {@link https://react.dev/reference/react/useState}
import runtimefunction Page({ code }: {
code: string;
}): JSX.Element- @param props
- @returns
(parameter) code: stringconst mdxModule: MDXModule | undefinedconst setMdxModule: Dispatch<SetStateAction<MDXModule | undefined>>(alias) useState<MDXModule>(): [MDXModule | undefined, Dispatch<SetStateAction<MDXModule | undefined>>] (+1 overload)
import useStateReturns a stateful value, and a function to update it.
- @version 16.8.0
- @see {@link https://react.dev/reference/react/useState}
const Content: React.ExoticComponent<{
children?: React.ReactNode | undefined;
}> | MDXContentconst mdxModule: MDXModule | undefinedconst mdxModule: MDXModule(property) MDXModule.default: MDXContentA functional JSX component which renders the content of the MDX file.
(alias) const Fragment: React.ExoticComponent<{
children?: React.ReactNode | undefined;
}>
import FragmentLets you group elements without a wrapper node.
- @see {@link https://react.dev/reference/react/Fragment React Docs}
- @example
import { Fragment } from 'react'; <Fragment> <td>Hello</td> <td>World</td> </Fragment> - @example
// Using the <></> shorthand syntax: <> <td>Hello</td> <td>World</td> </>
(alias) useEffect(effect: React.EffectCallback, deps?: React.DependencyList): void
import useEffectAccepts a function that contains imperative, possibly effectful code.
- @param effect Imperative function that can return a cleanup function
- @param deps If present, effect will only activate if the values in the list change.
- @version 16.8.0
- @see {@link https://react.dev/reference/react/useEffect}
const setMdxModule: (value: SetStateAction<MDXModule | undefined>) => void(alias) run(code: {
toString(): string;
}, options: RunOptions): Promise<MDXModule>
import runRun code compiled with outputFormat: 'function-body'.
☢️ Danger: this
evals JavaScript.
- @param code JavaScript function body to run.
- @param options Configuration (required).
- @return Promise to a module; the result is an object with a
defaultfield set to the component; anything else that was exported is available too.
(parameter) code: stringimport runtime(property) baseUrl?: string | URL | null | undefinedUse this URL as import.meta.url and resolve import and export … from relative to it (optional, example: import.meta.url); this option can also be given at compile time in CompileOptions; you should pass this (likely at runtime), as you might get runtime errors when using import.meta.url / import / export … from otherwise.
The type of import.meta.
If you need to declare that a given property exists on import.meta, this type may be augmented via interface merging.
(property) ImportMeta.url: stringThe absolute file: URL of the module.
(parameter) code: stringconst Content: React.ExoticComponent<{
children?: React.ReactNode | undefined;
}> | MDXContentfunction getStaticProps(): Promise<{
props: {
code: string;
};
}>const code: stringvar String: StringConstructor
(value?: any) => stringAllows manipulation and formatting of text strings and determination and location of substrings within strings.
(alias) compile(vfileCompatible: Readonly<Compatible>, compileOptions?: Readonly<CompileOptions> | null | undefined): Promise<VFile>
import compileCompile MDX to JS.
- @param vfileCompatible MDX document to parse.
- @param compileOptions Compile configuration (optional).
- @return Promise to compiled file.
(property) outputFormat?: "function-body" | "program" | null | undefinedOutput format to generate (default: 'program'); in most cases 'program' should be used, it results in a whole program; internally evaluate uses 'function-body' to compile to code that can be passed to run; in some cases, you might want what evaluate does in separate steps, such as when compiling on the server and running on the client.
(property) props: {
code: string;
}(property) code: string