keroosha.cybergulag.today/quartz/plugins/transformers/toc.ts

86 lines
2.4 KiB
TypeScript
Raw Normal View History

2023-06-10 09:06:02 +03:00
import { QuartzTransformerPlugin } from "../types"
import { Root } from "mdast"
import { visit } from "unist-util-visit"
import { toString } from "mdast-util-to-string"
import Slugger from "github-slugger"
import { wikilinkRegex } from "./ofm"
2023-06-10 09:06:02 +03:00
export interface Options {
2023-07-23 03:27:41 +03:00
maxDepth: 1 | 2 | 3 | 4 | 5 | 6
minEntries: 1
2023-06-10 09:06:02 +03:00
showByDefault: boolean
collapseByDefault: boolean
2023-06-10 09:06:02 +03:00
}
const defaultOptions: Options = {
maxDepth: 3,
minEntries: 1,
showByDefault: true,
collapseByDefault: false,
2023-06-10 09:06:02 +03:00
}
interface TocEntry {
2023-07-23 03:27:41 +03:00
depth: number
text: string
slug: string // this is just the anchor (#some-slug), not the canonical slug
2023-06-10 09:06:02 +03:00
}
const regexMdLinks = new RegExp(/\[([^\[]+)\](\(.*\))/, "g")
2023-07-23 03:27:41 +03:00
export const TableOfContents: QuartzTransformerPlugin<Partial<Options> | undefined> = (
userOpts,
) => {
const opts = { ...defaultOptions, ...userOpts }
return {
name: "TableOfContents",
markdownPlugins() {
2023-07-23 03:27:41 +03:00
return [
() => {
return async (tree: Root, file) => {
const display = file.data.frontmatter?.enableToc ?? opts.showByDefault
if (display) {
const slugAnchor = new Slugger()
2023-07-23 03:27:41 +03:00
const toc: TocEntry[] = []
let highestDepth: number = opts.maxDepth
visit(tree, "heading", (node) => {
if (node.depth <= opts.maxDepth) {
let text = toString(node)
// strip link formatting from toc entries
text = text.replace(wikilinkRegex, (_, rawFp, __, rawAlias) => {
const fp = rawFp?.trim() ?? ""
const alias = rawAlias?.slice(1).trim()
return alias ?? fp
})
text = text.replace(regexMdLinks, "$1")
2023-07-23 03:27:41 +03:00
highestDepth = Math.min(highestDepth, node.depth)
toc.push({
depth: node.depth,
text,
slug: slugAnchor.slug(text),
2023-07-23 03:27:41 +03:00
})
}
})
2023-06-10 09:06:02 +03:00
2023-07-23 03:27:41 +03:00
if (toc.length > opts.minEntries) {
file.data.toc = toc.map((entry) => ({
...entry,
depth: entry.depth - highestDepth,
}))
file.data.collapseToc = opts.collapseByDefault
2023-07-23 03:27:41 +03:00
}
2023-06-10 09:06:02 +03:00
}
}
2023-07-23 03:27:41 +03:00
},
]
},
2023-06-10 09:06:02 +03:00
}
}
2023-07-23 03:27:41 +03:00
declare module "vfile" {
2023-06-10 09:06:02 +03:00
interface DataMap {
toc: TocEntry[]
collapseToc: boolean
2023-06-10 09:06:02 +03:00
}
}