mirror of
https://github.com/pesde-pkg/pesde.git
synced 2025-04-04 10:50:55 +01:00
215 lines
5 KiB
TypeScript
215 lines
5 KiB
TypeScript
import rehypeShikiFromHighlighter from "@shikijs/rehype/core"
|
|
import type { Nodes } from "hast"
|
|
import { heading } from "hast-util-heading"
|
|
import { headingRank } from "hast-util-heading-rank"
|
|
import { toText } from "hast-util-to-text"
|
|
import { h, type Child } from "hastscript"
|
|
import type { ContainerDirective } from "mdast-util-directive"
|
|
import type { Handler } from "mdast-util-to-hast"
|
|
import rehypeInferDescriptionMeta from "rehype-infer-description-meta"
|
|
import rehypeRaw from "rehype-raw"
|
|
import rehypeSanitize, { defaultSchema } from "rehype-sanitize"
|
|
import rehypeSlug from "rehype-slug"
|
|
import rehypeStringify from "rehype-stringify"
|
|
import remarkDirective from "remark-directive"
|
|
import remarkFrontmatter from "remark-frontmatter"
|
|
import remarkGemoji from "remark-gemoji"
|
|
import remarkGfm from "remark-gfm"
|
|
import remarkGithubAdmonitionsToDirectives from "remark-github-admonitions-to-directives"
|
|
import remarkParse from "remark-parse"
|
|
import remarkRehype from "remark-rehype"
|
|
import { createCssVariablesTheme, createHighlighter, loadWasm } from "shiki"
|
|
import { unified } from "unified"
|
|
import type { Node } from "unist"
|
|
import { map } from "unist-util-map"
|
|
|
|
// @ts-expect-error - typescript doesn't like the wasm import
|
|
import onigWasm from "shiki/onig.wasm"
|
|
|
|
const loadOnigWasm = (async () => {
|
|
await loadWasm(onigWasm())
|
|
})()
|
|
|
|
const highlighter = (async () => {
|
|
await loadOnigWasm
|
|
|
|
return await createHighlighter({
|
|
themes: [],
|
|
langs: [],
|
|
})
|
|
})()
|
|
|
|
const ADMONITION_TYPES = {
|
|
note: {
|
|
label: "Note",
|
|
},
|
|
tip: {
|
|
label: "Tip",
|
|
},
|
|
info: {
|
|
label: "Info",
|
|
},
|
|
warning: {
|
|
label: "Warning",
|
|
},
|
|
danger: {
|
|
label: "Danger",
|
|
},
|
|
}
|
|
|
|
const containerDirectiveHandler: Handler = (state, node: ContainerDirective) => {
|
|
const type = node.name as keyof typeof ADMONITION_TYPES
|
|
if (!type || !(type in ADMONITION_TYPES)) {
|
|
return
|
|
}
|
|
|
|
const typeInfo = ADMONITION_TYPES[type]
|
|
|
|
let label: Child = typeInfo.label
|
|
|
|
const firstChild = node.children[0]
|
|
if (firstChild?.type === "paragraph" && firstChild.data?.directiveLabel) {
|
|
node.children.shift()
|
|
label = state.all(firstChild)
|
|
}
|
|
|
|
return h(
|
|
"div",
|
|
{
|
|
class: `admonition admonition-${type}`,
|
|
},
|
|
[
|
|
h(
|
|
"p",
|
|
{
|
|
class: "admonition-title",
|
|
},
|
|
[
|
|
h("span", {
|
|
class: "admonition-icon",
|
|
}),
|
|
h(
|
|
"span",
|
|
{
|
|
class: "admonition-label",
|
|
},
|
|
label,
|
|
),
|
|
],
|
|
),
|
|
state.all(node),
|
|
],
|
|
)
|
|
}
|
|
|
|
const sanitizeSchema: typeof defaultSchema = {
|
|
...defaultSchema,
|
|
attributes: {
|
|
...defaultSchema.attributes,
|
|
"*": [...(defaultSchema.attributes?.["*"] ?? []), ["className", "admonition", /^admonition-/]],
|
|
},
|
|
}
|
|
|
|
const remarkRehypeHandlers = {
|
|
containerDirective: containerDirectiveHandler,
|
|
}
|
|
|
|
export const markdown = (async () => {
|
|
return unified()
|
|
.use(remarkParse)
|
|
.use(remarkFrontmatter)
|
|
.use(remarkGfm)
|
|
.use(remarkGemoji)
|
|
.use(remarkGithubAdmonitionsToDirectives)
|
|
.use(remarkDirective)
|
|
.use(remarkRehype, { allowDangerousHtml: true, handlers: remarkRehypeHandlers })
|
|
.use(rehypeRaw)
|
|
.use(rehypeSanitize, sanitizeSchema)
|
|
.use(rehypeShikiFromHighlighter, await highlighter, {
|
|
lazy: true,
|
|
theme: createCssVariablesTheme({
|
|
name: "css-variables",
|
|
variablePrefix: "--shiki-",
|
|
variableDefaults: {},
|
|
fontStyle: true,
|
|
}),
|
|
fallbackLanguage: "text",
|
|
})
|
|
.use(rehypeStringify)
|
|
.freeze()
|
|
})()
|
|
|
|
export type TocItem = {
|
|
id: string
|
|
title: string
|
|
level: number
|
|
}
|
|
|
|
export const docsMarkdown = (async () => {
|
|
return unified()
|
|
.use(remarkParse)
|
|
.use(remarkFrontmatter)
|
|
.use(remarkGfm)
|
|
.use(remarkGemoji)
|
|
.use(remarkGithubAdmonitionsToDirectives)
|
|
.use(remarkDirective)
|
|
.use(remarkRehype, {
|
|
allowDangerousHtml: true,
|
|
clobberPrefix: "",
|
|
handlers: remarkRehypeHandlers,
|
|
})
|
|
.use(rehypeSlug)
|
|
.use(() => (node, file) => {
|
|
const toc: TocItem[] = []
|
|
file.data.toc = toc
|
|
|
|
return map(node as Nodes, (node) => {
|
|
if (node.type === "element" && node.tagName === "a") {
|
|
const fullUrl = new URL(node.properties.href as string, `file://${file.path}`)
|
|
|
|
let href = node.properties.href as string
|
|
if (fullUrl.protocol === "file:") {
|
|
href = file.data.basePath + fullUrl.pathname.replace(/\.mdx?$/, "") + fullUrl.hash
|
|
}
|
|
|
|
return {
|
|
...node,
|
|
properties: {
|
|
...node.properties,
|
|
href,
|
|
},
|
|
}
|
|
}
|
|
|
|
if (heading(node)) {
|
|
const rank = headingRank(node)
|
|
if (rank && typeof node.properties.id === "string" && rank >= 2 && rank <= 3) {
|
|
toc.push({
|
|
id: node.properties.id,
|
|
title: toText(node),
|
|
level: rank,
|
|
})
|
|
}
|
|
}
|
|
|
|
return node
|
|
}) as Node
|
|
})
|
|
.use(rehypeRaw)
|
|
.use(rehypeSanitize, sanitizeSchema)
|
|
.use(rehypeShikiFromHighlighter, await highlighter, {
|
|
lazy: true,
|
|
theme: createCssVariablesTheme({
|
|
name: "css-variables",
|
|
variablePrefix: "--shiki-",
|
|
variableDefaults: {},
|
|
fontStyle: true,
|
|
}),
|
|
fallbackLanguage: "text",
|
|
})
|
|
.use(rehypeInferDescriptionMeta, {
|
|
selector: "p",
|
|
})
|
|
.use(rehypeStringify)
|
|
.freeze()
|
|
})()
|