
Security News
CVE Volume Surges Past 48,000 in 2025 as WordPress Plugin Ecosystem Drives Growth
CVE disclosures hit a record 48,185 in 2025, driven largely by vulnerabilities in third-party WordPress plugins.
panda-docs
Advanced tools
What's black and white and read all over?

This is a documentation build system that takes Markdown files as sources, and produces HTML files. It runs on Node.js, and uses Jade as its templating engine.
h1, h2, e.t.c.), as well as automatic table of headers generationMake sure you have a recent build of Node.js (this was tested on v0.6.0). Install it using npm:
npm install panda-docs -g
Want to try a demonstration? Then clone this repository, and run
node build.js
That'll turn this README into a better looking HTML file in the /out directory.
Use panda-docs in a script! Simply define a file similar to this one:
var options = {
title: "Panda (from script)"
}
panda.make(["./src/"], options, function(err, cbReturn) {
if (err) {
console.error(err);
}
});
You can find out more information on options you can use below:
There are a number of arguments you can pass to Panda that affect the entire build. They are:
-h, --help` Show this help message and exit.-v, --version Show program's version number and exit.-o PATH, --output PATH Resulting file(s) location [out]-oa PATH, --outputAssets PATH Resulting file(s) location for assets [out/assets]-t STRING, --title STRING` Title of the index page [Panda: Default Title Here]--skin PATH The location of your Jade templates [./templates/default]--assets PATH The location of your asset files (CSS, Javascript, e.t.c.) [./templates/default/assets]-d, --disableTests Disables the test suite that runs at the end of an HTML build. This is NOT recommended.--keepFirstHeader If set, keeps the first header (<h1>) detected--baseUrl STRING Base url of all links [./]--keepOutDir Does not wipe output directory before building (defaults to false)--safeWords An array of words not to complain about when performing a spellcheck testYou need at least one Jade template layout.jade. Optional you can
have a template per file, for example if you have the file index.md
you can add index.jade to your template folder and it will be used
only for this file.
Within your Jade template, you have access to the following variables:
content is the transformed HTML content of your Markdown filemetadata is an object containing your document-based metadata valuesmanifest is an object containing the Manifest.json propertiestoh is an object containing the headings for each file (h1, h2, e.t.c.). See below for more information on this object.headingTable is a function you can use to generate a list of your page's table of contents. See below for more information on using thisfileName is the name of the resulting file (without the extension)title is the title of the documentationpageTitle is the title of the current pagemtime indicates the last modified time of your source filemarkdown references the Markdown converter; since this is based on namp, you'll want to add .html at the end to get the actual HTMLAll your passed in options are also available.
The toh object has the following structure:
[{
rank: 1, // the hierarchy in the toc (based on h1, h2, ..., h6)
name: "My first header", // the content of the header
link: "#my-first-header", // a direct internal url to be used
line: 0 // the line number in the original markdown file
text: "# My first header" // the actual Markdown header text
}, {
rank: 2,
name: "Subtitle",
link: "#subtitle",
line: 4
}, {
rank: 4,
name: "Minor Header",
link: "#minor-header",
line: 25
},{
rank: 2,
name: "Another Subtitle!",
link: "#another-subtitle",
line: 58
}]
Each non-h1 header is also automatically an anchor. The resulting HTML for an H2 called "Testing Your Highlighter" looks like this:
<h2 id="testing-your-highlighter">
<a class="heading_anchor" href="#testing-your-highlighter"></a>
<i class="headerLinkIcon"></i>
Testing Your Highlighter
</h2>
You can add an icon for headerLinkIcon to make it more discoverable, i.e. by using something from Font Awesome.
Finally, you also have access to a function, called headingTable, that automatically generates a table of contents for each page's heading. The resulting HTML produced by this function might look like this:
<ol class="tocContainer level_1">
<li class="tocItem level_2">
<a href="#defining-a-mode">Defining a Mode</a>
</li>
<li class="tocItem level_2">
<a href="#defining-syntax-highlighting-rules">Defining Syntax Highlighting Rules</a>
<ol class="tocContainer level_2">
<li class="tocItem level_3">
<a href="#defining-tokens">Defining Tokens</a>
</li>
<li class="tocItem level_3">
<a href="#defining-regular-expressions">Defining Regular Expressions</a>
<ol class="tocContainer level_3">
<li class="tocItem level_4">
<a href="#groupings">Groupings</a>
</li>
</ol>
</li>
</ol>
</li>
<li class="tocItem level_2">
<a href="#defining-states">Defining States</a>
</li>
<li class="tocItem level_2">
<a href="#code-folding">Code Folding</a>
</li>
</ol>
Basically, every sub-heading is nested underneath a parent heading of larger size. In the example above, we have a page with an <h2> tag called "Defining a Mode", followed by another <h2>, "Defining Syntax Highlighting Rules", which itself is followed by two <h3> tags, "Defining Tokens" and "Defining Regular Expressions." The last <h3> has an <h4> called "Groupings." We then go back to some regular old <h2> tags.
This generated table always ignores the <h1> tag. You can customize it by by embedding the following signature into your Jade template:
headingTable(toh, maxLevel, classes)
where
toh is your page's toh objectmaxLevel is optional, and refers to the maximum heading number you want to display; this defaults to 4. Basically, any h tag greater than this number is ignoredclasses is optional, and it's an array of two strings. The first is a class that applies to all the <ol> tags; the second applies to the <li> tags. Every <ol> and <li> automatically gets a level_<POSITION> class that is set to the current item's positionThus, to generate the above, your Jade template might go:
!= headingTable(toh, 5, ['tocContainer', 'tocItem'])
The callback for panda returns a JSON with one key: files, which is a listing of all the files generated. files is an array of objects, containing the following keys:
filename: the filename (minus the suffix)mtime: the last modified time of your source filepageTitle: the title of the page (text only, meaning minus any # or <h1> indicators)You could use this information to provide a list of Recently Updated content--which is exactly what the Cloud9 IDE User Documentation does.
FAQs
A complete documentation generation tool for Markdown files
We found that panda-docs demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 1 open source maintainer collaborating on the project.
Did you know?

Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.

Security News
CVE disclosures hit a record 48,185 in 2025, driven largely by vulnerabilities in third-party WordPress plugins.

Security News
Socket CEO Feross Aboukhadijeh joins Insecure Agents to discuss CVE remediation and why supply chain attacks require a different security approach.

Security News
Tailwind Labs laid off 75% of its engineering team after revenue dropped 80%, as LLMs redirect traffic away from documentation where developers discover paid products.