Published on

Mermaid Plugin Design

Authors

In the previous post Mermaid MDX, I added a Mermaid component to the MDX components and use it to render the mermaid diagrams..

But it is more as ad-hoc, after I published the post, I was thinking about how to make it more general and reusable.

So I started to think about how to design a more general plugin for this.

There are two ways to do this,

  • one is to use the rehype plugin to transform the mermaid code blocks to svg at compile time,
  • the other is to use the Mermaid component to render the mermaid diagrams at runtime.

Better way to do may be when compiling the mdx file, if find the mermaid code blocks, add global script to the html head.

<script type="module">
  import mermaid from 'https://cdn.jsdelivr.net/npm/mermaid@11/dist/mermaid.esm.min.mjs'
  mermaid.initialize({ startOnLoad: true })
</script>

ContentLayer -> mdx-bundler -> @mdx-js/esbuild -> @

Libraries such as mermaid.js traditionally run inside browsers because they depend on browser-specific objects like document and window. However, their primary role is to perform data transformations rather than rendering final HTML output. This opens up the possibility of running them in environments like Node.js or edge runtimes by supplying mocked or virtual implementations of the required document and window APIs.

The core difficulty is rendering process requires layout, which need to measure, and it needs a powerful layout system to calculate the size like fonts, and lightweight jsdom is not enough. See https://github.com/mermaid-js/mermaid/issues/559#issuecomment-372685431

Yes, you can provide mock document and window APIs in a Node.js environment using libraries that simulate a browser-like environment. Here are some common solutions:

JSDOM is a popular library that emulates the DOM in Node.js. It provides window and document objects similar to those in a real browser.

import { JSDOM } from 'jsdom'

const dom = new JSDOM(`<!DOCTYPE html><html><body></body></html>`)
global.window = dom.window
global.document = dom.window.document
global.navigator = dom.window.navigator

This allows libraries like Mermaid.js to function as if they were in a browser.


2. HappyDOM (Faster Alternative to JSDOM)

HappyDOM is another DOM simulation library optimized for speed.

import { Window } from 'happy-dom'

const window = new Window()
global.window = window
global.document = window.document
global.navigator = window.navigator

HappyDOM is generally faster than JSDOM and may be more efficient for certain workloads.


3. Puppeteer (Headless Browser)

If you need full browser rendering capabilities, you can use Puppeteer, which runs a headless version of Chrome.

import puppeteer from 'puppeteer'
;(async () => {
  const browser = await puppeteer.launch()
  const page = await browser.newPage()

  await page.evaluate(() => {
    window.someCustomProperty = 'test' // Modify the window object
  })

  await browser.close()
})()

While Puppeteer provides a real browser environment, it's heavier and slower compared to JSDOM or HappyDOM.


Which One to Choose?

  • Use JSDOM if you need basic DOM emulation (good balance between compatibility and performance).
  • Use HappyDOM if performance is a priority and JSDOM is too slow.
  • Use Puppeteer if you require a full browser environment with real rendering.

For running Mermaid.js in Node.js, JSDOM is usually sufficient since Mermaid mainly manipulates the DOM but doesn't require full browser rendering.