BlogPodcastAbout💌 Tiny Improvements

Next.js with MDX tips: Provide shortcuts to article headings

This tutorial will teach you how to automatically add links to heading tags in your mdx posts on your Next.js site with a plugin called rehype-slug. This should work for most nextJS sites that use MDX for content, as well as many other JavaScript-based sites which use MDX.

You may have come across this pattern in articles and posts on sites you frequent - article headings (think <h1>, <h2>, <h3>, <h4>, <h5>, and <h6> in html) will be be wrapped in links that point to themselves. This allows readers to link to specific headings in your articles, jumping to relevant bits of content without forcing someone to read through an entire article. Generally speaking, it will look something like this:

1
<a href="#some-unique-id">
2
<h1 id="some-unique-id">My first blog post</h1>
3
</a>

The <a> tag here has an href value of #some-unique-id - this is the id of the heading tag. This is based on an HTML standard defined by the W3C. In short, you can link to any element on an HTML page which has a unique id attribute defined, by appending #[id] to the end of the URL, like www.example.com#id-of-the-element.

This is tricky with Markdown and MDX

In most static site generators and JAMStack frameworks which allow you to use Markdown and MDX to generate content, the goal is simple: give authors a very simple way to author content using Markdown syntax. The unfortunate side effect in this case is that there's not a way to specify IDs for the headings in Markdown posts (at least, not one that I'm aware of).

A sample markdown post might look like this:

1
---
2
title: Hello, world
3
---
4
5
# A fish called wanda
6
7
In this essay, I will explain the difference between...

This results in the following output:

1
<h1>A fish called wanda</h1>
2
<p>In this essay, I will explain the difference between...</p>

Fantastic! That's a nice, easy way to write, but there's not a way to add an id to the heading tag. At least, not out of the box. This is where MDX's plugins come in handy.

Automatically linking to headings in your mdx posts with rehype plugins

Note: This tutorial assumes you're using MDX with NextJS, althought it may be applicable to other systems. Feel free to send me any hurdles you encounter with other frameworks, and I'll try to document them here.

Step 1: Generate IDs for all headings automatically with rehype-slug

rehype-slug is a plugin that works with MDX, and will automatically generate IDs for your headings by generating a slug based on the text they contain.

  1. Install rehype-slug in your project by running npm install --save rehype-slug or yarn add rehype-slug

  2. Add rehype-slug to the list of rehype plugins MDX uses. In the case of next.js sites, it is likely wherever you call serialize() from next-mdx-remote.

1
import rehypeSlug from 'rehype-slug';
2
3
// ...
4
5
const options = {
6
mdxOptions: {
7
rehypePlugins: [
8
rehypeSlug, // add IDs to any h1-h6 tag that doesn't have one, using a slug made from its text
9
],
10
},
11
};
12
13
const mdxSource = await serialize(post.content, options);
14
15
// ...

Note: My site uses serialize() in several places, so I extracted options to its own file. This avoids repeated code, and allows me to manage my plugins for MDX from one place.

At this point, if you fire up your dev environment, and use your browser devtools to inspect any of the headings generated from markdown for your site, they should all have an id property added. For the example above, you'd see:

1
<h1 id="a-fish-called-wanda">A fish called wanda</h1>

We're halfway there - you can now link to www.example.com#a-fish-called-wanda, and the browser will automatically scroll to the heading.

Step 2: use MDXProvider to customize the way heading tags render

MDXProvider is a wrapper component which allows you to customize the way your MDX renders by providing a list of components.

This step will depend heavily on the UI frameworks you've chosen for your site - I use Chakra UI for my nextjs site, but you can use whatever you like - tailwindcss, Material UI, etc will all have similar parallels.

Here's a simplified version of the code, which I'll show just for <h1> - you'd want to extend this for all title tags, i.e. <h1> through <h6>:

1
import Link from 'next/link';
2
3
const CustomH1 = ({ id, ...rest }) => {
4
if (id) {
5
return (
6
<Link href={`#${id}`}>
7
<h1 {...rest} />
8
</Link>
9
);
10
}
11
return <h1 {...rest} />;
12
};
13
14
const components = {
15
h1: CustomH1,
16
};
17
18
// this would also work in pages/_app.js
19
const Layout = ({ children }) => {
20
return <MDXProvider components={components}>{children}</MDXProvider>;
21
};

Doing it with Chakra UI

Like I mentioned above, my site uses Chakra UI to compose page layouts. I've added a bit of customization to links on my site - including a hover behavior which adds a nice # character before headings when they're hovered over. If you're curious about my implementation with Chakra UI, it looks a bit like this:

1
import NextLink from 'next/link';
2
import { Link, Heading } from '@chakra-ui/react';
3
4
const CustomHeading = ({ as, id, ...props }) => {
5
if (id) {
6
return (
7
<Link href={`#${id}`}>
8
<NextLink href={`#${id}`}>
9
<Heading
10
as={as}
11
display="inline"
12
id={id}
13
lineHeight={'1em'}
14
{...props}
15
_hover={{
16
_before: {
17
content: '"#"',
18
position: 'relative',
19
marginLeft: '-1.2ch',
20
paddingRight: '0.2ch',
21
},
22
}}
23
/>
24
</NextLink>
25
</Link>
26
);
27
}
28
return <Heading as={as} {...props} />;
29
};
30
31
const H1 = (props) => <CustomHeading as="h1" {...props} />;
32
const H2 = (props) => <CustomHeading as="h2" {...props} />;
33
const H3 = (props) => <CustomHeading as="h3" {...props} />;
34
const H4 = (props) => <CustomHeading as="h4" {...props} />;
35
const H5 = (props) => <CustomHeading as="h5" {...props} />;
36
const H6 = (props) => <CustomHeading as="h6" {...props} />;
37
38
const components = {
39
h1: H1,
40
h2: H2,
41
h3: H3,
42
h4: H4,
43
h5: H5,
44
h6: H6,
45
};
46
47
// ...etc - components is passed to MDXProvider in my Layout component

The Result

The result is what you see on this page, and any of the other posts on my site! Every heading on my markdown pages contains an ID, and is wrapped in a link to itself. This makes it easy for readers to tap on the link to send it to their URL bar, or to right-click/long-press and copy a link to the part of the article they want to link to.

The final markup looks a bit like this:

1
<a href="#a-fish-called-wanda">
2
<h1 id="a-fish-called-wanda">A fish called wanda</h1>
3
</a>

I hope you found this helpful! If you run into any trouble, feel free to drop me a line on twitter. Beyond that, I'd love it if you shared this post with someone who you think could benefit from it.

Auto-linking headings with other frameworks

More reading

If you found this helpful, you may also be interested in:

Mike Bifulco headshot

Subscribe to Tiny Improvements

Learn about designing & building great products for the web, and my philosophy for living a life you love in an ever-changing world.

    Typically once a week, straight from me to you. 😘 Unsubscribe anytime.


    Get in touch to → Sponsor Tiny Improvements

    ***
    Hero
    Next.js with MDX tips: Provide shortcuts to article headings

    This tutorial will teach you how to automatically add links to heading tags in your mdx posts on your Next.js site with a plugin called rehype-slug. This should work for most nextJS sites that use MDX for content, as well as many other JavaScript-based sites which use MDX.

    reactmdxnextjs
    © 2019-2023 Mike Bifulco

    Get in touch to → Sponsor Tiny Improvements

    Disclaimer: 👋🏽 Hi there. I work as a co-founder & CTO at Craftwork. These are my opinions, and not necessarily the views of my employer.
    Built with Next. Source code on GitHub.

    More great resources

    Articles about React.jsArticles about Remix.runArticles about Next.jsArticles for developersArticles for JavaScript developersArticles about CSSArticles about User Experience (UX)Articles about tools I useArticles about productivityBrowse all topics →