Important
If you are using react18
, use ver.1 of next-mdx-remote-client
, currently v1.0.6
If you are using react19
, use ver.2 of next-mdx-remote-client
, currently v2.0.0
The both serve the same features and APIs. I am going to maintain both.
The next-mdx-remote-client
is a wrapper of @mdx-js/mdx
for nextjs
applications in order to load MDX content. It is a fork of next-mdx-remote
.
See some blog applications in which next-mdx-remote-client
is used:
- for a demo application which uses
app
router, visit source code or living web site, - for a demo application which uses
pages
router, visit source code or living web site, - for a testing application which uses both
app
andpages
router, visit source code or living web site.
I started to create the next-mdx-remote-client
in line with the mindset of the @mdx-js/mdx
in early 2024 considering next-mdx-remote had not been updated for a long time, and finally, a brand new package emerged.
The next-mdx-remote-client
serves as a viable alternative to next-mdx-remote
having more features.
I would like to highlight some main features:
- It supports MDX version 3.
- It provides well designed components and functions for both "pages" router and "app" router, which completely isolated from eachother.
- It provides internal error handling mechanism.
- It supports
import statements
andexport statements
in MDX source, which can be disabled as well. - Creating table of contents (TOC) is so easy since it supports passing
vfile.data
into thescope
. - You can get frontmatter without compiling the source while listing the articles/posts via
getFrontmatter
. - It exports some components and types from
@mdx-js/mdx
so as you don't need to install.
Let's compare the features of next-mdx-remote
and next-mdx-remote-client
.
Features | next-mdx-remote |
next-mdx-remote-client |
---|---|---|
support MDX version 3 | ✅ (as of v5) |
✅ |
ensure internal error handling mechanism in app router |
❌ | ✅ |
ensure internal error handling mechanism in pages router |
❌ | ✅ |
support export-from-MDX in app router |
❌ | ✅ |
support export-from-MDX in pages router |
❌ | ✅ |
support import-into-MDX in app router |
❌ | ✅ |
support import-into-MDX in pages router |
❌ | ❌ |
get frontmatter and mutated scope in app router |
❌ | ✅ |
get frontmatter and mutated scope in pages router |
✅ | ✅ |
support options for disabling imports and exports in MDX | ✅ | ✅ |
support passing vfile.data into the scope |
❌ | ✅ |
provide utility for getting frontmatter without compiling | ❌ | ✅ |
expose MDXProvider from @mdx-js/mdx |
❌ | ✅ |
provide option for disabling parent MDXProvider contexts |
❌ | ✅ |
expose the necessary types from mdx/types |
❌ | ✅ |
Important
You will see a lot the abbreviatons csr
and rsc
. Pay attention to the both are spelled backwards.
csr
stands for "client side rendering" which is related with pages
router
rsc
stands for "react server component" which is related with app
router
- It is ESM only package
- Needs
react
version 19.0+, works with latest next@15 (tested) - Needs
node
version 18.18+ in line with nextjs does - Vitest is used instead of jest for testing
- Rollup is removed for bundling
- Test coverage is 100%
- Type coverage is 100%
- The parts client side (csr) and server side (rsc) are completely isolated from each other
- Exported a small utility to get frontmatter without compiling the source
- All functions take named parameters
- Supports
import statements
andexport statements
in MDX - Export statements in MDX work for both
app
andpages
router - Import statements in MDX work for only
app
router
Important
Imported modules in MDX with relative path should be transpiled into javascript before or during build process, otherwise will not work. I believe the community can find a solution to import reqular .jsx
or .tsx
modules into MDX. With the support of the next/mdx
, it is viable to import .mdx
into MDX, but not tested yet.
This package is ESM only, requires Node.js (version 18.18+).
# in general
npm install next-mdx-remote-client
# specifically for react18 users
npm install next-mdx-remote-client@1
# specifically for react19 users
npm install next-mdx-remote-client@2
or
yarn add next-mdx-remote-client
Warning
The next-mdx-remote
users may follow the migration guide.
The main entry point /
also refers to /csr
subpath.
// main entry point, which is related "pages" router
import /* */ from "next-mdx-remote-client";
// isolated subpath for the "serialize" function
import /* */ from "next-mdx-remote-client/serialize";
// sub entry point related with "pages" router
import /* */ from "next-mdx-remote-client/csr";
// sub entry point related with "app" router
import /* */ from "next-mdx-remote-client/rsc";
// isolated subpath for the utils
import /* */ from "next-mdx-remote-client/utils";
Go to the part associated with Next.js pages router
The next-mdx-remote-client
exposes evaluate
function and MDXRemote
component for "app" router.
import { evaluate, MDXRemote } from "next-mdx-remote-client/rsc";
Tip
If you need to get the exports from MDX --> use evaluate
If you don't need --> use MDXRemote
If you need to get the frontmatter and the mutated scope --> use evaluate
If you don't need --> use MDXRemote
Let's give some examples how to use next-mdx-remote-client
in "app" router first, then explain the exposed function and component.
See a demo application with app
router, visit source code or living web site.
import { Suspense } from "react";
import { MDXRemote } from "next-mdx-remote-client/rsc";
import { ErrorComponent, LoadingComponent } from "../components";
import { Test } from '../mdxComponents';
const components = {
Test,
wrapper: ({ children }) => <div className="mdx-wrapper">{children}</div>,
}
export default async function Page() {
const source = "Some **bold text** in MDX, with a component <Test />";
return (
<Suspense fallback={<LoadingComponent />}>
<MDXRemote
source={source}
components={components}
onError={ErrorComponent}
/>
</Suspense>
);
};
import { Suspense } from "react";
import { MDXRemote } from "next-mdx-remote-client/rsc";
import type { MDXRemoteOptions, MDXComponents } from "next-mdx-remote-client/rsc";
import { calculateSomeHow, getSourceSomeHow } from "../utils";
import { ErrorComponent, LoadingComponent } from "../components";
import { Test } from '../mdxComponents';
const components: MDXComponents = {
Test,
wrapper: function ({ children }: React.ComponentPropsWithoutRef<"div">) {
return <div className="mdx-wrapper">{children}</div>;
},
}
export default async function Page() {
const source = await getSourceSomeHow();
if (!source) {
return <ErrorComponent error="The source could not found !" />;
}
const options: MDXRemoteOptions = {
mdxOptions: {
// ...
},
parseFrontmatter: true,
scope: {
readingTime: calculateSomeHow(source),
},
};
return (
<Suspense fallback={<LoadingComponent />}>
<MDXRemote
source={source}
options={options}
components={components}
onError={ErrorComponent}
/>
</Suspense>
);
}
I assume you have a MDX file having <TableOfContentComponent />
inside; and you provide some MDX components which have an entry for TableOfContentComponent: (props) => { ... }
.
---
title: My Article
---
# {frontmatter.title}
<TableOfContentComponent toc={toc} />
rest of the article...
You can have a look at an example TOC component in the demo application.
In order to create a table of contents (TOC) I use remark-flexible-toc
in the remark plugin list and pass the table of contents objects vFile.data.toc
into the scope
via the option vfileDataIntoScope
.
That's it ! So easy !
import { Suspense } from "react";
import { MDXRemote, type MDXRemoteOptions } from "next-mdx-remote-client/rsc";
import remarkFlexibleToc from "remark-flexible-toc"; // <---------
import { calculateSomeHow, getSourceSomeHow } from "../utils";
import { ErrorComponent, LoadingComponent } from "../components";
import { components } from '../mdxComponents';
export default async function Page() {
const source = await getSourceSomeHow();
if (!source) {
return <ErrorComponent error="The source could not found !" />;
}
const options: MDXRemoteOptions = {
mdxOptions: {
remarkPlugins: [
// ...
remarkFlexibleToc, // <---------
],
},
parseFrontmatter: true,
scope: {
readingTime: calculateSomeHow(source),
},
vfileDataIntoScope: "toc", // <---------
};
return (
<Suspense fallback={<LoadingComponent />}>
<MDXRemote
source={source}
options={options}
components={components}
onError={ErrorComponent}
/>
</Suspense>
);
}
import { Suspense } from "react";
import { evaluate, type EvaluateOptions } from "next-mdx-remote-client/rsc";
import remarkFlexibleToc, { type TocItem } from "remark-flexible-toc";
import { calculateSomeHow, getSourceSomeHow } from "../utils";
import { ErrorComponent, LoadingComponent, TableOfContentComponent } from "../components";
import { components } from "../mdxComponents";
type Scope = {
readingTime: string;
toc?: TocItem[];
};
type Frontmatter = {
title: string;
author: string;
};
export default async function Page() {
const source = await getSourceSomeHow();
if (!source) {
return <ErrorComponent error="The source could not found !" />;
}
const options: EvaluateOptions<Scope> = {
mdxOptions: {
remarkPlugins: [
// ...
remarkFlexibleToc,
],
},
parseFrontmatter: true,
scope: {
readingTime: calculateSomeHow(source),
},
vfileDataIntoScope: "toc",
};
const { content, frontmatter, scope, error } = await evaluate<Frontmatter, Scope>({
source,
options,
components,
});
if (error) {
return <ErrorComponent error={error} />;
}
return (
<>
<h1>{frontmatter.title}</h1>
<p>Written by {frontmatter.author}; read in {scope.readingTime}</p>
<TableOfContentComponent toc={scope.toc} />
<Suspense fallback={<LoadingComponent />}>
{content}
</Suspense>
</>
);
}
Actually, you may not need to access the "frontmatter" and "scope" in JSX, you can use them within MDX directly, and return just content
only.
// ...
export default async function Page({ source }: Props) {
// ...
return (
<Suspense fallback={<LoadingComponent />}>
{content}
</Suspense>
);
}
article.mdx
# {frontmatter.title}
Written by {frontmatter.author}; read in {readingTime}
<TableOfContentComponent toc={toc} />
rest of the article...
After the examples given, let's dive into the exposed function and component by next-mdx-remote-client
for "app" router.
Go to the MDXRemote component
The evaluate
function is used for compiling the MDX source, constructing the compiled source, getting exported information from MDX and returning MDX content to be rendered on the server side, as a react server component.
async function evaluate(props: EvaluateProps): Promise<EvaluateResult> {}
The evaluate
function takes EvaluateProps
and returns EvaluateResult
as a promise.
Props of the evaluate
function
type EvaluateProps<TScope> = {
source: Compatible;
options?: EvaluateOptions<TScope>;
components?: MDXComponents;
};
Result of the evaluate
function
type EvaluateResult<TFrontmatter, TScope> = {
content: JSX.Element;
mod: Record<string, unknown>;
frontmatter: TFrontmatter;
scope: TScope;
error?: Error;
};
The evaluate
has internal error handling mechanism as much as it can, in order to do so, it returns an error
object if it is catched.
Caution
The eval of the compiled source returns a module MDXModule
, and does not throw errors except syntax errors. Some errors throw during the render process which needs you to use an ErrorBoundary.
import { Suspense } from "react";
import { evaluate, type EvaluateOptions } from "next-mdx-remote-client/rsc";
import { ErrorComponent, LoadingComponent, TableOfContentComponent } from "../components";
import { components } from "../mdxComponents";
import type { Frontmatter, Scope } from "../types"
export default async function MDXComponent({ source }: {source?: string}) {
if (!source) {
return <ErrorComponent error="The source could not found !" />;
}
const options: EvaluateOptions = {
/* */
};
const { content, mod, frontmatter, scope, error } = await evaluate<Frontmatter, Scope>({
source,
options,
components,
});
if (error) {
return <ErrorComponent error={error} />;
}
/**
* Use "mod", "frontmatter" and "scope" as you wish
*
* "mod" object is for exported information from MDX
* "frontmatter" is available even if a MDX syntax error occurs
* "scope" is for mutated scope if the `vfileDataIntoScope` option is used
*/
return (
<>
<h1>{frontmatter.title}</h1>
<div><em>{mod.something}</em></div>
<TableOfContentComponent toc={scope.toc} />
<Suspense fallback={<LoadingComponent />}>
{content}
</Suspense>
</>
);
};
If you provide the generic type parameters like await evaluate<Frontmatter, Scope>(){}
, the frontmatter
and the scope
get the types, otherwise Record<string, unknown>
by default for both.
Warning
Pay attention to the order of the generic type parameters.
The type parameters Frontmatter
and Scope
should extend Record<string, unknown>
. You should use type
instead of interface
for type parameters otherwise, you will receive an error saying Type 'Xxxx' does not satisfy the constraint 'Record<string, unknown>'.
See this issue for more explanation.
In the above example, I assume you use remark-flexible-toc
remark plugin in order to collect the headings from the MDX content, and you pass that information into the scope
via vfileDataIntoScope
option.
All options are optional.
type EvaluateOptions<TScope> = {
mdxOptions?: EvaluateMdxOptions;
disableExports?: boolean;
disableImports?: boolean;
parseFrontmatter?: boolean;
scope?: TScope;
vfileDataIntoScope?: VfileDataIntoScope;
};
It is an EvaluateMdxOptions
option to be passed to the @mdx-js/mdx
compiler.
import { type EvaluateOptions as OriginalEvaluateOptions } from "@mdx-js/mdx";
type EvaluateMdxOptions = Omit<
OriginalEvaluateOptions,
| "Fragment"
| "jsx"
| "jsxs"
| "jsxDEV"
| "useMDXComponents"
| "providerImportSource"
| "outputFormat"
>;
As you see, some of the options are omitted and opinionated within the package. For example the outputFormat
is always function-body
by default. Visit https://mdxjs.com/packages/mdx/#evaluateoptions for available mdxOptions.
const options: EvaluateOptions = {
// ...
mdxOptions: {
format: "mdx",
baseUrl: import.meta.url,
development: true,
remarkPlugins: [/* */],
rehypePlugins: [/* */],
recmaPlugins: [/* */],
remarkRehypeOptions: {handlers: {/* */}},
// ...
};
};
For more information see the MDX documentation.
It is a boolean option whether or not stripping the export statements
out from the MDX source.
By default it is false, meaningly the export statements
work as expected.
const options: EvaluateOptions = {
disableExports: true;
};
Now, the export statements
will be stripped out from the MDX.
It is a boolean option whether or not stripping the import statements
out from the MDX source.
By default it is false, meaningly the import statements
work as expected.
const options: EvaluateOptions = {
disableImports: true;
};
Now, the import statements
will be stripped out from the MDX.
It is a boolean option whether or not the frontmatter should be parsed out of the MDX.
By default it is false, meaningly the frontmatter
will not be parsed and extracted.
const options: EvaluateOptions = {
parseFrontmatter: true;
};
Now, the frontmatter
part of the MDX file is parsed and extracted from the MDX source; and will be supplied into the MDX file so as you to use it within the javascript statements.
Note
Frontmatter is a way to identify metadata in Markdown files. Metadata can literally be anything you want it to be, but often it's used for data elements your page needs and you don't want to show directly.
---
title: "My Article"
author: "ipikuka"
---
# {frontmatter.title}
It is written by {frontmatter.author}
The package uses the vfile-matter
internally to parse the frontmatter.
It is an Record<string, unknown>
option which is an arbitrary object of data which will be supplied to the MDX. For example, in cases where you want to provide template variables to the MDX, like my name is {name}
, you could provide scope as { name: "ipikuka" }
.
Here is another example:
const options: EvaluateOptions = {
scope: {
readingTime: calculateSomeHow(source)
};
};
Now, the scope
will be supplied into the MDX file so as you to use it within the statements.
# My article
read in {readingTime} min.
The variables within the expression in the MDX content should be valid javascript variable names. Therefore, each key of the scope must be a valid variable name.
My name is {name} valid expression.
My name is {my-name} is not valid expression, which will throw error
So, we can say for the scope
, here:
const options: EvaluateOptions = {
scope: {
name: "ipikuka", // valid usage
"my-name": "ipikuka", // is not valid and error prone for the MDX content !!!
};
};
Tip
The scope variables can be consumed not only as a property of a component, but also within the texts.
my name is {name}
<BarComponent name={name} />
It is an union type option. It is for passing some fields of vfile.data
into the scope
by mutating the scope
.
Important
It provides referencial copy for objects and arrays. If the scope
has the same key already, vfile.data
overrides it.
The reason behind of this option is that vfile.data
may hold some extra information added by some remark plugins. Some fields of the vfile.data
may be needed to pass into the scope
so as you to use in the MDX.
type VfileDataIntoScope =
| true // all fields from vfile.data
| string // one specific field
| { name: string; as: string } // one specific field but change the key as
| Array<string | { name: string; as: string }>; // more than one field
const options: EvaluateOptions = {
// Let's assume you use "remark-flexible-toc" plugin which composes
// the table of content (TOC) within the 'vfile.data.toc'
vfileDataIntoScope: "toc"; // or fileDataIntoScope: ["toc"];
};
Now, vfile.data.toc
is copied into the scope as scope["toc"]
, and will be supplied to the MDX via scope
.
# My article
<TableOfContentComponent toc={toc} />
If you need to change the name of the field, specify it for example { name: "toc", as: "headings" }
.
const options: EvaluateOptions = {
vfileDataIntoScope: { name: "toc", as: "headings" };
};
# My article
<TableOfContentComponent headings={headings} />
If you need to pass all the fields from vfile.data
, specify it as true
const options: EvaluateOptions = {
vfileDataIntoScope: true;
};
Go to the evaluate function
The MDXRemote
component is used for rendering the MDX content on the server side. It is a react server component.
async function MDXRemote(props: MDXRemoteProps): Promise<JSX.Element> {}
The MDXRemote
component takes MDXRemoteProps
and returns JSX.Element
as a promise.
Props of the MDXRemote
component
type MDXRemoteProps<TScope> = {
source: Compatible;
options?: MDXRemoteOptions<TScope>;
components?: MDXComponents;
onError?: React.ComponentType<{ error: Error }>
};
The MDXRemote
has internal error handling mechanism as much as it can, in order to do so, it takes onError
prop in addition to evaluate
function.
Caution
The eval of the compiled source returns a module MDXModule
, and does not throw errors except syntax errors. Some errors throw during the render process which needs you to use an ErrorBoundary.
import { Suspense } from "react";
import { MDXRemote, type MDXRemoteOptions } from "next-mdx-remote-client/rsc";
import { ErrorComponent, LoadingComponent } from "../components";
import { components } from "../mdxComponents";
export default async function MDXComponent({ source }: {source?: string}) {
if (!source) {
return <ErrorComponent error="The source could not found !" />;
}
const options: MDXRemoteOptions = {
/* */
};
return (
<Suspense fallback={<LoadingComponent />}>
<MDXRemote
source={source}
options={options}
components={components}
onError={ErrorComponent}
/>
</Suspense>
);
};
All options are optional.
type MDXRemoteOptions<TScope> = {
mdxOptions?: EvaluateMdxOptions;
disableExports?: boolean;
disableImports?: boolean;
parseFrontmatter?: boolean;
scope?: TScope;
vfileDataIntoScope?: VfileDataIntoScope;
};
The details are the same with the EvaluateOptions.
Go to the part associated with Next.js app router
The next-mdx-remote-client
exposes serialize
, hydrate
functions and MDXClient
component for "pages" router.
The serialize
function is used on the server side in "pages" router, while as the hydrate
and the MDXClient
are used on the client side in "pages" router. That is why the "serialize" function is purposefully isolated considering it is intended to run on the server side.
Let's give some examples how to use next-mdx-remote-client
in "pages" router first, then explain the exposed functions and component.
See a demo application with pages
router, visit source code or living web site.
import { serialize } from 'next-mdx-remote-client/serialize';
import { MDXClient } from 'next-mdx-remote-client';
import ErrorComponent from '../components/ErrorComponent';
import Test from '../mdxComponents/Test';
const components = {
Test,
wrapper: ({children}) => <div className="mdx-wrapper">{children}</div>,
}
export default function Page({ mdxSource }) {
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
return <MDXClient {...mdxSource} components={components} />;
}
export async function getStaticProps() {
const source = "Some **bold text** in MDX, with a component <Test />";
const mdxSource = await serialize({source});
return { props: { mdxSource } };
}
import { MDXClient, type MDXComponents } from 'next-mdx-remote-client';
import { serialize } from "next-mdx-remote-client/serialize";
import type { SerializeOptions, SerializeResult } from "next-mdx-remote-client/serialize";
import { calculateSomeHow, getSourceSomeHow } from "../utils";
import ErrorComponent from '../components/ErrorComponent';
import Test from '../mdxComponents/Test';
type Scope = {
readingTime: string;
};
type Frontmatter = {
title: string;
author: string;
};
const components: MDXComponents = {
Test,
wrapper: function ({ children }: React.ComponentPropsWithoutRef<"div">) {
return <div className="mdx-wrapper">{children}</div>;
},
}
type Props = {
mdxSource?: SerializeResult<Frontmatter, Scope>;
}
export default function Page({ mdxSource }: Props) {
if (!mdxSource) {
return <ErrorComponent error="The source could not found !" />;
}
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
return (
<>
<h1>{mdxSource.frontmatter.title}</h1>
<p>Written by {mdxSource.frontmatter.author}; read in {mdxSource.scope.readingTime}</p>
<MDXClient {...mdxSource} components={components} />
</>
);
}
export async function getStaticProps() {
const source = await getSourceSomeHow();
if (!source) return { props: {} };
const options: SerializeOptions<Scope> = {
disableImports: true,
mdxOptions: {
// ...
},
parseFrontmatter: true,
scope: {
readingTime: calculateSomeHow(source),
},
};
const mdxSource = await serialize<Frontmatter, Scope>({source, options});
return { props: { mdxSource } };
}
I assume you have a MDX file having <TableOfContentComponent />
inside; and you provide some MDX components which have an entry for TableOfContentComponent: (props) => { ... }
.
---
title: My Article
---
# {frontmatter.title}
<TableOfContentComponent toc={toc} />
rest of the article...
You can have a look at an example TOC component in the demo application.
In order to create a table of contents (TOC) I use remark-flexible-toc
in the remark plugin list and pass the table of contents objects vFile.data.toc
into the scope
via the option vfileDataIntoScope
.
That's it! So easy!
import { MDXClient, type MDXComponents } from 'next-mdx-remote-client';
import { serialize } from "next-mdx-remote-client/serialize";
import type { SerializeOptions, SerializeResult } from "next-mdx-remote-client/serialize";
import remarkFlexibleToc, {type TocItem} from "remark-flexible-toc"; // <---------
import { calculateSomeHow, getSourceSomeHow } from "../utils";
import { ErrorComponent, TableOfContentComponent } from '../components';
import { Test } from '../mdxComponents';
type Scope = {
readingTime: string;
};
type Frontmatter = {
title: string;
author: string;
};
const components: MDXComponents = {
Test,
wrapper: function ({ children }: React.ComponentPropsWithoutRef<"div">) {
return <div className="mdx-wrapper">{children}</div>;
},
}
type Props = {
mdxSource?: SerializeResult<Frontmatter, Scope & {toc: TocItem[]}>;
}
export default function Page({ mdxSource }: Props) {
if (!mdxSource) {
return <ErrorComponent error="The source could not found !" />;
}
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
return (
<>
<h1>{mdxSource.frontmatter.title}</h1>
<p>Written by {mdxSource.frontmatter.author}; read in {mdxSource.scope.readingTime}</p>
<TableOfContentComponent toc={mdxSource.scope.toc /* <----- here added TOC */} />
<MDXClient {...mdxSource} components={components} />
</>
);
}
export async function getStaticProps() {
const source = await getSourceSomeHow();
if (!source) return { props: {} };
const options: SerializeOptions<Scope> = {
disableImports: true,
mdxOptions: {
remarkPlugins: [
// ...
remarkFlexibleToc, // <---------
],
},
parseFrontmatter: true,
scope: {
readingTime: calculateSomeHow(source),
},
vfileDataIntoScope: "toc", // <---------
};
const mdxSource = await serialize<Frontmatter, Scope>({source, options});
return { props: { mdxSource } };
}
Actually, you may not need to access the "frontmatter" and "scope" in JSX, you can use them within MDX directly, and return just <MDXClient />
only.
// ...
const components: MDXComponents = {
TableOfContentComponent, // <---------
wrapper: function ({ children }: React.ComponentPropsWithoutRef<"div">) {
return <div className="mdx-wrapper">{children}</div>;
},
}
// ...
export default function Page({ mdxSource }: Props) {
// ...
return (
<MDXClient {...mdxSource} components={components} />
);
}
article.mdx
# {frontmatter.title}
Written by {frontmatter.author}; read in {readingTime}
<TableOfContentComponent toc={toc} />
rest of the article...
After the examples given, let's dive into the exposed functions and component by next-mdx-remote-client
for "pages" router.
Go to the hydrate function or the MDXClient component
import { serialize } from "next-mdx-remote-client/serialize";
The serialize
function is used for compiling the MDX source, in other words, producing the compiled source from MDX source, intended to run on server side at build time.
Warning
The serialize
function is asyncronous and to be used within the getStaticProps
or the getServerSideProps
on the server side. (Off the record, it can be used within an useEffect
as well, but this is not recommended because it is a heavy function as having more dependencies).
async function serialize(props: SerializeProps): Promise<SerializeResult> {}
The serialize
function takes SerializeProps
and returns SerializeResult
as a promise.
Props of the serialize
function
type SerializeProps<TScope> = {
source: Compatible;
options?: SerializeOptions<TScope>;
};
Result of the serialize
function
Either the compiledSource
or the error
exists, in addition to frontmatter
and scope
.
type SerializeResult<TFrontmatter, TScope> =
({ compiledSource: string } | { error: Error })
& {
frontmatter: TFrontmatter;
scope: TScope;
};
The serialize
function has internal error handling mechanism for the MDX syntax errors. The catched error is serialized via serialize-error
package and attached into the serialize results, further you can deserialize the error on the client, if necessary. You don't need to implement error handling by yourself.
import { serialize, type SerializeOptions } from "next-mdx-remote-client/serialize";
import type { Frontmatter, Scope } from "./types"
export async function getStaticProps() {
const source = await getSourceSomeHow();
if (!source) {
return { props: {} };
}
const options: SerializeOptions = {
/* */
};
const mdxSource = await serialize<Frontmatter, Scope>({
source,
options,
});
return {
props: {
mdxSource,
},
};
}
If you provide the generic type parameters like await serialize<Frontmatter, Scope>(){}
, the frontmatter
and the scope
get the types, otherwise Record<string, unknown>
by default for both.
Warning
Pay attention to the order of the generic type parameters.
The type parameters Frontmatter
and Scope
should extend Record<string, unknown>
. You should use type
instead of interface
for type parameters otherwise, you will receive an error saying Type 'Xxxx' does not satisfy the constraint 'Record<string, unknown>'.
See this issue for more explanation.
The nextjs
will send the mdxSource
((compiledSource
or error
) + frontmatter
+ scope
) to client side.
On client side, you need first to narrow the mdxSource
by checking if ("error" in mdxSource) {}
.
type Props = {
mdxSource?: SerializeResult<Frontmatter, Scope>;
}
export default function Page({ mdxSource }: Props) {
// ...
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
// ...
};
All options are optional.
type SerializeOptions<TScope> = {
mdxOptions?: SerializeMdxOptions;
disableExports?: boolean;
disableImports?: boolean;
parseFrontmatter?: boolean;
scope?: TScope;
vfileDataIntoScope?: VfileDataIntoScope;
};
Except the mdxOptions
, the details are the same with the EvaluateOptions.
It is a SerializeMdxOptions
option to be passed to the @mdx-js/mdx
compiler.
import { type CompileOptions as OriginalCompileOptions } from "@mdx-js/mdx";
type SerializeMdxOptions = Omit<
OriginalCompileOptions,
"outputFormat" | "providerImportSource"
>;
As you see, some of the options are omitted and opinionated within the package. For example the outputFormat
is always function-body
by default. Visit https://mdxjs.com/packages/mdx/#compileoptions for available mdxOptions.
const options: SerializeOptions = {
// ...
mdxOptions: {
format: "mdx",
baseUrl: import.meta.url,
development: true,
remarkPlugins: [/* */],
rehypePlugins: [/* */],
recmaPlugins: [/* */],
remarkRehypeOptions: {handlers: {/* */}},
// ...
};
};
Warning
Here I need to mention about the scope
option again for the serialize
.
scope
Actually, the serialize
doesn't do so much with the scope
except you provide the option vfileDataIntoScope
for passing data from vfile.data
into the scope
. Since the scope
is passed from the server to the client by nextjs
, the scope
must be serializable. The scope
can not hold function, component , Date, undefined, Error object etc.
If the scope has to have unserializable information or if you don't need or don't want to pass the scope
into the serialize
, you can pass it into hydrate
or MDXClient
directly on the client side.
Go to the serialize function or the MDXClient component
import { hydrate } from "next-mdx-remote-client/csr";
The hydrate
function is used for constructing the compiled source, getting exported information from MDX and returning MDX content to be rendered on the client side.
function hydrate(props: HydrateProps): HydrateResult {}
The hydrate
function takes HydrateProps
and returns HydrateResult
. The hydrate
has no "options" parameter.
Props of the hydrate
function
type HydrateProps = {
compiledSource: string;
frontmatter?: Record<string, unknown>;
scope?: Record<string, unknown>;
components?: MDXComponents;
disableParentContext?: boolean;
};
The option disableParentContext
is a feature of @mdx-js/mdx
. If it is false
, the mdx components provided by parent MDXProvider
s are going to be disregarded.
Result of the hydrate
function
type HydrateResult = {
content: JSX.Element;
mod: Record<string, unknown>;
error?: Error;
};
The mod
object is for exported information from MDX source.
Tip
If you need to get the exports from MDX --> use hydrate
If you don't need --> use MDXClient
The hydrate
has internal error handling mechanism as much as it can, in order to do so, it returns an error
object if it is catched.
Caution
The eval of the compiled source returns a module MDXModule
, and does not throw errors except syntax errors. Some errors throw during the render process which needs you to use an ErrorBoundary.
import { hydrate, type SerializeResult } from "next-mdx-remote-client/csr";
import { ErrorComponent, TableOfContentComponent } from "../components";
import { components } from "../mdxComponents";
import type { Frontmatter, Scope } from "../types"
type Props = {
mdxSource?: SerializeResult<Frontmatter, Scope>;
}
export default function Page({ mdxSource }: Props) {
if (!mdxSource) {
return <ErrorComponent error="The source could not found !" />;
}
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
// Now, mdxSource has {compiledSource, frontmatter, scope}
const { content, mod, error } = hydrate({ ...mdxSource, components });
if (error) {
return <ErrorComponent error={error} />;
}
// You can use the "mod" object for exported information from the MDX as you wish
return (
<>
<h1>{mdxSource.frontmatter.title}</h1>
<div><em>{mod.something}</em></div>
<TableOfContentComponent toc={mdxSource.scope.toc} />
{content}
</>
);
};
In the above example, I assume you use remark-flexible-toc
remark plugin in order to collect the headings from the MDX content, and you pass that information into the scope
via vfileDataIntoScope
option within the serialize on the server side.
Go to the serialize function or the hydrate function
import { MDXClient } from "next-mdx-remote-client/csr";
The MDXClient
component is used for rendering the MDX content on the client side.
function MDXClient(props: MDXClientProps): JSX.Element {}
The MDXClient
component takes MDXClientProps
and returns JSX.Element
. The MDXClient
has no "options" parameter like hydrate
.
Props of the MDXClient
component
type MDXClientProps = {
compiledSource: string;
frontmatter?: Record<string, unknown>;
scope?: Record<string, unknown>;
components?: MDXComponents;
disableParentContext?: boolean;
onError?: React.ComponentType<{ error: Error }>
};
The option disableParentContext
is a feature of @mdx-js/mdx
. If it is false
, the mdx components provided by parent MDXProvider
s are going to be disregarded.
Tip
If you need to get the exports from MDX --> use hydrate
If you don't need --> use MDXClient
The MDXClient
has internal error handling mechanism as much as it can, in order to do so, it takes onError
prop in addition to hydrate
function.
Caution
The eval of the compiled source returns a module MDXModule
, and does not throw errors except syntax errors. Some errors throw during the render process which needs you to use an ErrorBoundary.
import { MDXClient, type SerializeResult } from "next-mdx-remote-client/csr";
import { ErrorComponent, TableOfContentComponent } from "../components";
import { components } from "../mdxComponents";
import type { Frontmatter, Scope } from "../types"
type Props = {
mdxSource?: SerializeResult<Frontmatter, Scope>;
}
export default function Page({ mdxSource }: Props) {
if (!mdxSource) {
return <ErrorComponent error="The source could not found !" />;
}
if ("error" in mdxSource) {
return <ErrorComponent error={mdxSource.error} />;
}
// Now, mdxSource has {compiledSource, frontmatter, scope}
return (
<>
<h1>{mdxSource.frontmatter.title}</h1>
<TableOfContentComponent toc={mdxSource.scope.toc} />
<MDXClient
{...mdxSource}
components={components}
onError={ErrorComponent}
/>
</>
);
};
In the above example, I assume you use remark-flexible-toc
remark plugin in order to collect the headings from the MDX content, and you pass that information into the scope
via vfileDataIntoScope
option within the serialize on the server side.
The next-mdx-remote-client
exports additional versions, say, the hydrateLazy
and the MDXClientLazy
, which both have the same functionality, props, results with the hydrate
and the MDXClient
, correspondently.
The only difference is the hydration process takes place lazily on the browser within a window.requestIdleCallback
in a useEffect. You can use hydrateLazy
or MDXClientLazy
in order to defer hydration of the content and immediately serve the static markup.
import { hydrateLazy, MDXClientLazy } from "next-mdx-remote-client/csr";
When you use hydrateLazy
, and want to get the exports from MDX via mod
object, please be aware that the mod
object is always empty {}
at first render, then it will get actual exports at second render.
Note
Lazy hydration defers hydration of the components on the client. This is an optimization technique to improve the initial load of the application, but may introduce unexpected delays in interactivity for any dynamic content within the MDX content.
This will add an additional wrapping div around the rendered MDX, which is necessary to avoid hydration mismatches during render.
For further explanation about the lazy hydration see next-mdx-remote
notes.
The next-mdx-remote-client
exports additional versions, say, the hydrateAsync
and the MDXClientAsync
.
These have additional props and options, but here, I don't want to give the details since I created them for experimental to show the import statements
on the client side don't work. You can have a look at the github repository for the code and the tests.
The main difference is that the eval of the compiled source takes place in a useEffect on the browser, since the compile source has await
keyword for import statements
.
import { hydrateAsync, MDXClientAsync } from "next-mdx-remote-client/csr";
Note
I believe, it is viable somehow using dynamic
API if the vercel
supports for a solution for the pages
router via import.meta
APIs. During the compilation of the MDX in the serialize
, a remark/recma plugin can register the imported modules into the import.meta.url
via a nextjs API (needs support of vercel) for them will be available to download/import on the client side via dynamic
api. This is my imagination.
The package exports the MDXProvider
from @mdx-js/react
, in order the developers don't need to install the @mdx-js/react
.
import { MDXProvider } from "next-mdx-remote-client/csr";
The <MDXProvider />
makes the mdx components available to any <MDXClient />
or hydrate's { content }
being rendered in the application, as a child status of that provider.
For example, you can wrap the whole application so as you do not need to supply the mdx components into any <MDXClient />
or hydrate's { content }
.
import { MDXProvider } from 'next-mdx-remote-client';
import { components } from "../mdxComponents";
export default function App({ Component, pageProps }) {
return (
<MDXProvider components={components}>
<Component {...pageProps} />
</MDXProvider>
)
}
Note
How this happens, because the next-mdx-remote-client
injects the useMdxComponents
context hook from @mdx-js/react
during the function construction of the compiled source, internally. Pay attention that it is valid for only MDXClient
and hydrate
functions.
Caution
Since MDXRemote
as a react server component can not read the context, MDXProvider
is effectless when used within the nextjs app
router for MDXRemote
, which is also for evaluate
.
You can provide a map of custom MDX components, which is a feature of @mdx-js/mdx
, in order to replace HTML tags (see the list of markdown syntax and equivalent HTML tags) with the custom components.
Typescript users can use MDXComponents
from mdx/types
, which is exported by this package as well.
../mdxComponents/index.ts
import { type MDXComponents } from "next-mdx-remote-client";
import dynamic from "next/dynamic";
import Image from "next/image";
import Link from "next/link";
import { Typography } from "@material-ui/core";
import { motion } from 'framer-motion'
import Hello from "./Hello";
import CountButton from "./CountButton";
import BlockQuote, { default as blockquote } from "./BlockQuote";
import pre from "./pre";
export const mdxComponents: MDXComponents = {
Hello,
CountButton,
Dynamic: dynamic(() => import("./dynamic")),
Image,
Link,
motion: { div: () => <div>Hello world</div> },
h2: (props: React.ComponentPropsWithoutRef<"h2">) => (
<Typography variant="h2" {...props} />
),
strong: (props: React.ComponentPropsWithoutRef<"strong">) => (
<strong className="custom-strong" {...props} />
),
em: (props: React.ComponentPropsWithoutRef<"em">) => (
<em className="custom-em" {...props} />
),
pre,
blockquote,
BlockQuote,
wrapper: (props: { children: any }) => {
return <div id="mdx-layout">{props.children}</div>;
}
};
Note
The wrapper
is a special key, if you want to wrap the MDX content with a HTML container element.
./data/my-article.mdx
---
title: "My Article"
author: "ipikuka"
---
_Read in {readingTime}, written by <Link href="#">**{frontmatter.author}**</Link>_
# {frontmatter.title}
## Sub heading for custom components
<Hello name={foo} />
<CountButton />
<Dynamic />
<Image src="/images/cover.png" alt="cover" width={180} height={40} />
<BlockQuote>
I am blackquote content
</BlockQuote>
<motion.div animate={{ x: 100 }} />
## Sub heading for some markdown elements
![cover](/images/cover.png)
Here is _italic text_ and **strong text**
> I am blackquote content
The package exports one utility getFrontmatter
which is for getting frontmatter without compiling the source. You can get the fronmatter and the stripped source by using the getFrontmatter
which employs the same frontmatter extractor vfile-matter
used within the package.
import { getFrontmatter } from "next-mdx-remote-client/utils";
const { frontmatter, strippedSource } = getFrontmatter<TFrontmatter>(source);
If you provide the generic type parameter, it ensures the frontmatter
gets the type, otherwise Record<string, unknown>
by default.
If there is no frontmatter in the source, the frontmatter
will be empty object {}
.
Important
If you use next-mdx-remote
and want to get frontmatter
without compiling the source !
The subpath next-mdx-remote-client/utils
is isolated from other features of the package and it does cost minimum. So, anyone can use next-mdx-remote-client/utils
while using next-mdx-remote
.
The next-mdx-remote-client
is fully typed with TypeScript.
The package exports the types for server side (rsc):
EvaluateProps
EvaluateOptions
EvaluateResult
MDXRemoteProps
MDXRemoteOptions
The package exports the types for client side (csr):
HydrateProps
HydrateResult
MDXClientProps
SerializeResult
The package exports the types for the serialize function:
SerializeProps
SerializeOptions
SerializeResult
In addition, the package exports the types from mdx/types
so that developers do not need to import mdx/types
:
MDXComponents
MDXContent
MDXProps
MDXModule
Element
The next-mdx-remote-client
works with unified version 6+ ecosystem since it is compatible with MDX version 3.
Allowance of the export declarations
and the import declarations
in MDX source, if you don't have exact control on the content, may cause vulnerabilities and harmful activities. The next-mdx-remote-client gives options for disabling them.
But, you need to use a custom recma plugin for disabiling the import expressions
like await import("xyz")
since the next-mdx-remote-client doesn't touch the import expressions.
Eval
a string of JavaScript can be a dangerous and may cause enabling XSS attacks, which is how the next-mdx-remote-client APIs do. Please, take your own measures while passing the user input.
If there is a Content Security Policy (CSP) on the website that disallows code evaluation via eval
or new Function()
, it is needed to loosen that restriction in order to utilize next-mdx-remote-client
, which can be done using unsafe-eval.
I like to contribute the Unified / Remark / MDX ecosystem, so I recommend you to have a look my plugins.
remark-flexible-code-titles
– Remark plugin to add titles or/and containers for the code blocks with customizable propertiesremark-flexible-containers
– Remark plugin to add custom containers with customizable properties in markdownremark-ins
– Remark plugin to addins
element in markdownremark-flexible-paragraphs
– Remark plugin to add custom paragraphs with customizable properties in markdownremark-flexible-markers
– Remark plugin to add custommark
element with customizable properties in markdownremark-flexible-toc
– Remark plugin to expose the table of contents via Vfile.data or via an option referenceremark-mdx-remove-esm
– Remark plugin to remove import and/or export statements (mdxjsEsm)
rehype-pre-language
– Rehype plugin to add language information as a property topre
elementrehype-highlight-code-lines
– Rehype plugin to add line numbers to code blocks and allow highlighting of desired code lines
recma-mdx-escape-missing-components
– Recma plugin to set the default value() => null
for the Components in MDX in case of missing or not provided so as not to throw an errorrecma-mdx-change-props
– Recma plugin to change theprops
parameter into the_props
in thefunction _createMdxContent(props) {/* */}
in the compiled source in order to be able to use{props.foo}
like expressions. It is useful for thenext-mdx-remote
ornext-mdx-remote-client
users innextjs
applications.
MPL 2.0 License © ipikuka
🟩 @mdx-js 🟩 next/mdx 🟩 next-mdx-remote 🟩 next-mdx-remote-client