@ipikuka/plugins
v2.0.3
Published
A collection of useful remark, rehype, recma plugins for markdown / MDX
Maintainers
Readme
@ipikuka/plugins
A robust Next.js newsletter Next.js Weekly is sponsoring me 💖

A warm thanks 🙌 to @ErfanEbrahimnia, @recepkyk, and @LSeaburg for the support 💖
@ipikuka/plugins is a collection of unified (remark, rehype and recma) plugins for markdown / MDX that I used in my many projects.
unified is a project that transforms content with abstract syntax trees (ASTs) using the new parser micromark. remark adds support for markdown to unified. mdast is the Markdown Abstract Syntax Tree (AST) which is a specification for representing markdown in a syntax tree. rehype is a tool that transforms HTML with plugins. hast stands for HTML Abstract Syntax Tree (HAST) that rehype uses. recma adds support for producing a javascript code by transforming esast which stands for Ecma Script Abstract Syntax Tree (AST) that is used in production of compiled source for MDX.
@ipikuka/plugins provides remarkPlugins, rehypePlugins, recmaPlugins for @mdx-js/mdx and related projects like next-mdx-remote and next-mdx-remote-client.
When should I use this?
If you don't want to install and configure any specific remark, rehype and recma plugin; @ipikuka/plugins provides you a plugin list that is opinionated and well tested. It also helps creating table of contents (TOC) for markdown/mdx content out of the box via remark-flexible-toc.
The remark plugins that exposed by @ipikuka/plugins:
(exactly in specific order below)
- @richardtowers/remark-abbr
- remark-mdx-remove-expressions (onlyDangerousExpressions)
- remark-fix-guillemets
- remark-smartypants
- remark-flexible-markers
- remark-ins
- remark-gfm
- remark-textr (with custom textr-plugins)
- remark-definition-list
- remark-flexible-paragraphs
- remark-supersub
- remark-gemoji
- remark-emoji
- remark-flexible-containers
- remark-flexible-code-titles
- remark-custom-header-id
- remark-footnotes-extra
- remark-flexible-toc
The rehype plugins that exposed by @ipikuka/plugins:
(exactly in specific order below)
- rehype-code-meta
- rehype-raw
- rehype-slug
- rehype-autolink-headings
- rehype-prism-plus
- rehype-pre-language
- rehype-image-toolkit
The recma plugins (only for MDX content) that exposed by @ipikuka/plugins:
(exactly in specific order below)
- recma-mdx-escape-missing-components
- recma-mdx-change-props
- recma-mdx-interpolate
Installation
This package is suitable for ESM only. In Node.js (version 16+), install with npm:
npm install @ipikula/pluginsor
yarn add @ipikula/pluginsUsage
Let's create a wrapper for serialize function of next-mdx-remote-client and use @ipikuka/plugins inside.
// serialize.ts
import { serialize as serialize_ } from "next-mdx-remote-client/serialize";
import type { SerializeResult, SerializeProps } from "next-mdx-remote-client/serialize";
import { plugins, prepare, type TocItem } from "@ipikuka/plugins";
export async function serialize<
TFrontmatter extends Record<string, unknown> = Record<string, unknown>,
TScope extends Record<string, unknown> = Record<string, unknown>,
>({
source,
options,
}: SerializeProps<TScope>): Promise<SerializeResult<TFrontmatter, TScope & { toc?: TocItem[] }>> {
const { mdxOptions, ...rest } = options || {};
const format_ = mdxOptions?.format;
const format = format_ === "md" || format_ === "mdx" ? format_ : "mdx";
const processedSource = format === "mdx" ? prepare(source) : source;
return await serialize_<TFrontmatter, TScope>({
source: processedSource,
options: {
mdxOptions: {
...mdxOptions,
...plugins({ format }),
},
vfileDataIntoScope: "toc",
...rest,
},
});
};Let's create another wrapper for serialize function of next-mdx-remote and use @ipikuka/plugins inside.
// serialize.ts
import { serialize as serialize_, type SerializeOptions } from "next-mdx-remote/serialize";
import { plugins, prepare, type TocItem } from "@ipikuka/plugins";
import { type Compatible } from "vfile";
export async function serialize<
TFrontmatter extends Record<string, unknown> = Record<string, unknown>,
TScope extends Record<string, unknown> = Record<string, unknown>,
>(
source: Compatible,
{ mdxOptions, parseFrontmatter, scope }: SerializeOptions = {},
): Promise<MDXRemoteSerializeResult<TScope & { toc?: TocItem[] }, TFrontmatter>> {
const toc: TocItem[] = [];
const { format: format_, ...rest } = mdxOptions || {};
const format = format_ === "md" || format_ === "mdx" ? format_ : "mdx";
const processedSource = format === "mdx" ? prepare(source) : source;
return await serialize_<TScope & { toc?: TocItem[] }, TFrontmatter>(processedSource, {
parseFrontmatter,
scope: { ...scope, toc },
mdxOptions: {
format,
...rest,
...plugins({ format, toc }),
},
});
};You can use the serialize wrappers in pages router of nextjs applications.
[!NOTE] I will try to provide a complete example
nextjsapplications later.
Thanks to @ipikuka/plugins, the markdown/mdx content will support table of contents, containers, markers, aligned paragraphs, gfm syntax (tables, strikethrough, task lists, autolinks etc.), inserted texts, highlighted code fences, code titles, code line numbering and highlighting, autolink for headers, definition lists etc. in addition to standard markdown syntax like bold texts, italic texts, lists, blockquotes, headings etc.
Without @ipikuka/plugins the result would be a standart markdown result with no containers, no markers, no gfm syntax, no inserted texts, no highlighted code fences etc.
Options
type Options = {
format?: "md" | "mdx" | null | undefined;
toc?: TocItem[];
};format
It is "md" | "mdx" | "detect" | null | undefined option to adjust remark plugins and whether or not to employ recma plugins.
It is optional, and default is mdx.
toc
It is TocItem[] option to compose a table of content by remark-flexible-toc.
It is optional and have no default value.
If you want to have a table of content and supplied into the scope, I advise you use the option toc if you use next-mdx-remote, but you don't need it for next-mdx-remote-client thanks to the option vfileDataIntoScope: "toc".
Examples:
Example with @mdx-js/mdx
import { compile } from "@mdx-js/mdx";
import { plugins, type TocItem } from "@ipikuka/plugins";
// ...
// if you don't need a table of content you can omit it
const toc: TocItem[] = [];
const compiledSource = await compile(source, {
...plugins({ format: "md", toc }),
})
console.log(toc); // now it has table of contents
// ...Example with next-mdx-remote-client
import { serialize } from "next-mdx-remote-client/serialize";
import { plugins } from "@ipikuka/plugins";
// ...
const mdxSource = await serialize<TFrontmatter, TScope>({
source,
options: {
mdxOptions: {
...plugins({ format: "md" }),
},
parseFrontmatter: true,
scope: {},
vfileDataIntoScope: "toc", // it will ensure the scope has `toc`
},
});
console.log(mdxSource.scope.toc); // now it has table of contents
// ...Example with next-mdx-remote
import { serialize } from "next-mdx-remote/serialize";
import { plugins, type TocItem } from "@ipikuka/plugins";
// ...
// if you don't need a table of content you can omit it
const toc: TocItem[] = [];
const mdxSource = await serialize<TScope, TFrontmatter>(
source,
{
mdxOptions: {
...plugins({ format: "md", toc }),
},
parseFrontmatter: true,
scope: { toc },
},
);
console.log(mdxSource.scope.toc); // now it has table of contents
// ...Utils
The package exposes one utility function which is called prepare.
prepare(source: Compatible)
It is for MDX source (not markdown) to correct breaklines to <br/>, horizontal lines to <hr/>, guillements to « » and or equals signs to ≤ and ≥. The prepare function accepts Compatible (see vfile) but check if it is string, otherwise do nothing.
The reason for having prepare function is that remark parser and mdx parser are different.
Syntax tree
The plugins modifies the mdast (Markdown abstract syntax tree), the hast (HTML abstract syntax tree) and the esast (EcmaScript abstract syntax tree).
Types
This package is fully typed with TypeScript.
The package exports the type Options, TocItem.
Compatibility
The plugins that are provided by this package work with unified version 6+, MDX version 3+, next-mdx-remote, next-mdx-remote-client and @next/mdx.
Security
Use of some rehype plugins involves hast, but doesn't lead to cross-site scripting (XSS) attacks.
My Plugins
I like to contribute the Unified / Remark / MDX ecosystem, so I recommend you to have a look my plugins.
Support My Work (become a sponsor 🚀)
If you find @ipikuka/plugins or any of my projects is useful and helpful, please consider supporting my work. Your sponsorship means a lot to me and keeps these projects alive and updated! 💖
My sponsors are going to be featured at the very top of the page and proudly displayed on my Sponsor Wall.
Thank you for supporting open source! 🙌
My Remark 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 addinselement in markdownremark-flexible-paragraphs– Remark plugin to add custom paragraphs with customizable properties in markdownremark-flexible-markers– Remark plugin to add custommarkelement with customizable properties in markdownremark-flexible-toc– Remark plugin to expose the table of contents viavfile.dataor via an option referenceremark-mdx-remove-esm– Remark plugin to remove import and/or export statements (mdxjsEsm)remark-mdx-remove-expressions– Remark plugin to remove MDX expressions within curlybraces {} in MDX content
My Rehype Plugins
rehype-pre-language– Rehype plugin to add language information as a property topreelementrehype-highlight-code-lines– Rehype plugin to add line numbers to code blocks and allow highlighting of desired code linesrehype-code-meta– Rehype plugin to copycode.data.metatocode.properties.metastringrehype-image-toolkit– Rehype plugin to enhance Markdown image syntax![]()and Markdown/MDX media elements (<img>,<audio>,<video>) by auto-linking bracketed or parenthesized image URLs, wrapping them in<figure>with optional captions, unwrapping images/videos/audio from paragraph, parsing directives in title for styling and adding attributes, and dynamically converting images into<video>or<audio>elements based on file extension.
My Recma Plugins
recma-mdx-escape-missing-components– Recma plugin to set the default value() => nullfor 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 thepropsparameter into the_propsin thefunction _createMdxContent(props) {/* */}in the compiled source in order to be able to use{props.foo}like expressions. It is useful for thenext-mdx-remoteornext-mdx-remote-clientusers innextjsapplications.recma-mdx-change-imports– Recma plugin to convert import declarations for assets and media with relative links into variable declarations with string URLs, enabling direct asset URL resolution in compiled MDX.recma-mdx-import-media– Recma plugin to turn media relative paths into import declarations for both markdown and html syntax in MDX.recma-mdx-import-react– Recma plugin to ensure gettingReactinstance from the arguments and to make the runtime props{React, jsx, jsxs, jsxDev, Fragment}is available in the dynamically imported components in the compiled source of MDX.recma-mdx-html-override– Recma plugin to allow selected raw HTML elements to be overridden via MDX components.recma-mdx-interpolate– Recma plugin to enable interpolation of identifiers wrapped in curly braces within thealt,src,href, andtitleattributes of markdown link and image syntax in MDX.
My Unist Utils and Unified Plugins
I also build low-level utilities and plugins for the Unified ecosystem that can be used across Remark, Rehype, Recma, and other unist-based abstract syntax trees (ASTs).
unist-util-find-between– Unist utility to find the nodes between two nodes.unified-log-tree– Unified plugin to log abstract syntax trees (ASTs) for debugging without mutating.
License
MIT License © ipikuka
