mirror of
https://github.com/versia-pub/docs.git
synced 2026-03-13 02:49:16 +01:00
feat: ✨ Initialize rewrite
This commit is contained in:
parent
47ce9bd9f8
commit
f39d34b769
143 changed files with 7257 additions and 4032 deletions
3
mdx/recma.mjs
Normal file
3
mdx/recma.mjs
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
import { mdxAnnotations } from "mdx-annotations";
|
||||
|
||||
export const recmaPlugins = [mdxAnnotations.recma];
|
||||
129
mdx/rehype.mjs
Normal file
129
mdx/rehype.mjs
Normal file
|
|
@ -0,0 +1,129 @@
|
|||
import { slugifyWithCounter } from "@sindresorhus/slugify";
|
||||
import * as acorn from "acorn";
|
||||
import { toString as mdastToString } from "mdast-util-to-string";
|
||||
import { mdxAnnotations } from "mdx-annotations";
|
||||
import shiki from "shiki";
|
||||
import { visit } from "unist-util-visit";
|
||||
|
||||
function rehypeParseCodeBlocks() {
|
||||
return (tree) => {
|
||||
// biome-ignore lint/style/useNamingConvention: <explanation>
|
||||
visit(tree, "element", (node, _, parentNode) => {
|
||||
if (node.tagName === "code" && node.properties.className) {
|
||||
parentNode.properties.language =
|
||||
node.properties.className[0]?.replace(/^language-/, "");
|
||||
}
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
let highlighter;
|
||||
|
||||
function rehypeShiki() {
|
||||
return async (tree) => {
|
||||
highlighter =
|
||||
highlighter ??
|
||||
(await shiki.getHighlighter({ theme: "css-variables" }));
|
||||
|
||||
visit(tree, "element", (node) => {
|
||||
if (
|
||||
node.tagName === "pre" &&
|
||||
node.children[0]?.tagName === "code"
|
||||
) {
|
||||
const codeNode = node.children[0];
|
||||
const textNode = codeNode.children[0];
|
||||
|
||||
node.properties.code = textNode.value;
|
||||
|
||||
if (node.properties.language) {
|
||||
const tokens = highlighter.codeToThemedTokens(
|
||||
textNode.value,
|
||||
node.properties.language,
|
||||
);
|
||||
|
||||
textNode.value = shiki.renderToHtml(tokens, {
|
||||
elements: {
|
||||
pre: ({ children }) => children,
|
||||
code: ({ children }) => children,
|
||||
line: ({ children }) => `<span>${children}</span>`,
|
||||
},
|
||||
});
|
||||
}
|
||||
}
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
function rehypeSlugify() {
|
||||
return (tree) => {
|
||||
const slugify = slugifyWithCounter();
|
||||
visit(tree, "element", (node) => {
|
||||
if (node.tagName === "h2" && !node.properties.id) {
|
||||
node.properties.id = slugify(mdastToString(node));
|
||||
}
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
function rehypeAddMDXExports(getExports) {
|
||||
return (tree) => {
|
||||
const exports = Object.entries(getExports(tree));
|
||||
|
||||
for (const [name, value] of exports) {
|
||||
for (const node of tree.children) {
|
||||
if (
|
||||
node.type === "mdxjsEsm" &&
|
||||
new RegExp(`export\\s+const\\s+${name}\\s*=`).test(
|
||||
node.value,
|
||||
)
|
||||
) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
const exportStr = `export const ${name} = ${value}`;
|
||||
|
||||
tree.children.push({
|
||||
type: "mdxjsEsm",
|
||||
value: exportStr,
|
||||
data: {
|
||||
estree: acorn.parse(exportStr, {
|
||||
sourceType: "module",
|
||||
ecmaVersion: "latest",
|
||||
}),
|
||||
},
|
||||
});
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
function getSections(node) {
|
||||
const sections = [];
|
||||
|
||||
for (const child of node.children ?? []) {
|
||||
if (child.type === "element" && child.tagName === "h2") {
|
||||
sections.push(`{
|
||||
title: ${JSON.stringify(mdastToString(child))},
|
||||
id: ${JSON.stringify(child.properties.id)},
|
||||
...${child.properties.annotation}
|
||||
}`);
|
||||
} else if (child.children) {
|
||||
sections.push(...getSections(child));
|
||||
}
|
||||
}
|
||||
|
||||
return sections;
|
||||
}
|
||||
|
||||
export const rehypePlugins = [
|
||||
mdxAnnotations.rehype,
|
||||
rehypeParseCodeBlocks,
|
||||
rehypeShiki,
|
||||
rehypeSlugify,
|
||||
[
|
||||
rehypeAddMDXExports,
|
||||
(tree) => ({
|
||||
sections: `[${getSections(tree).join()}]`,
|
||||
}),
|
||||
],
|
||||
];
|
||||
4
mdx/remark.mjs
Normal file
4
mdx/remark.mjs
Normal file
|
|
@ -0,0 +1,4 @@
|
|||
import { mdxAnnotations } from "mdx-annotations";
|
||||
import remarkGfm from "remark-gfm";
|
||||
|
||||
export const remarkPlugins = [mdxAnnotations.remark, remarkGfm];
|
||||
141
mdx/search.mjs
Normal file
141
mdx/search.mjs
Normal file
|
|
@ -0,0 +1,141 @@
|
|||
import * as fs from "node:fs";
|
||||
import * as path from "node:path";
|
||||
import * as url from "node:url";
|
||||
import { slugifyWithCounter } from "@sindresorhus/slugify";
|
||||
import glob from "fast-glob";
|
||||
import { toString as mdastToString } from "mdast-util-to-string";
|
||||
import { remark } from "remark";
|
||||
import remarkMdx from "remark-mdx";
|
||||
import { createLoader } from "simple-functional-loader";
|
||||
import { filter } from "unist-util-filter";
|
||||
import { SKIP, visit } from "unist-util-visit";
|
||||
|
||||
const __filename = url.fileURLToPath(import.meta.url);
|
||||
const processor = remark().use(remarkMdx).use(extractSections);
|
||||
const slugify = slugifyWithCounter();
|
||||
|
||||
function isObjectExpression(node) {
|
||||
return (
|
||||
node.type === "mdxTextExpression" &&
|
||||
node.data?.estree?.body?.[0]?.expression?.type === "ObjectExpression"
|
||||
);
|
||||
}
|
||||
|
||||
function excludeObjectExpressions(tree) {
|
||||
return filter(tree, (node) => !isObjectExpression(node));
|
||||
}
|
||||
|
||||
function extractSections() {
|
||||
return (tree, { sections }) => {
|
||||
slugify.reset();
|
||||
|
||||
visit(tree, (node) => {
|
||||
if (node.type === "heading" || node.type === "paragraph") {
|
||||
const content = mdastToString(excludeObjectExpressions(node));
|
||||
if (node.type === "heading" && node.depth <= 2) {
|
||||
const hash = node.depth === 1 ? null : slugify(content);
|
||||
sections.push([content, hash, []]);
|
||||
} else {
|
||||
sections.at(-1)?.[2].push(content);
|
||||
}
|
||||
return SKIP;
|
||||
}
|
||||
});
|
||||
};
|
||||
}
|
||||
|
||||
export default function Search(nextConfig = {}) {
|
||||
const cache = new Map();
|
||||
|
||||
return Object.assign({}, nextConfig, {
|
||||
webpack(config, options) {
|
||||
config.module.rules.push({
|
||||
test: __filename,
|
||||
use: [
|
||||
createLoader(function () {
|
||||
const appDir = path.resolve("./app");
|
||||
this.addContextDependency(appDir);
|
||||
|
||||
const files = glob.sync("**/*.mdx", { cwd: appDir });
|
||||
const data = files.map((file) => {
|
||||
const url = `/${file.replace(/(^|\/)page\.mdx$/, "")}`;
|
||||
const mdx = fs.readFileSync(
|
||||
path.join(appDir, file),
|
||||
"utf8",
|
||||
);
|
||||
|
||||
let sections = [];
|
||||
|
||||
if (cache.get(file)?.[0] === mdx) {
|
||||
sections = cache.get(file)[1];
|
||||
} else {
|
||||
const vfile = { value: mdx, sections };
|
||||
processor.runSync(
|
||||
processor.parse(vfile),
|
||||
vfile,
|
||||
);
|
||||
cache.set(file, [mdx, sections]);
|
||||
}
|
||||
|
||||
return { url, sections };
|
||||
});
|
||||
|
||||
// When this file is imported within the application
|
||||
// the following module is loaded:
|
||||
return `
|
||||
import FlexSearch from 'flexsearch'
|
||||
|
||||
let sectionIndex = new FlexSearch.Document({
|
||||
tokenize: 'full',
|
||||
document: {
|
||||
id: 'url',
|
||||
index: 'content',
|
||||
store: ['title', 'pageTitle'],
|
||||
},
|
||||
context: {
|
||||
resolution: 9,
|
||||
depth: 2,
|
||||
bidirectional: true
|
||||
}
|
||||
})
|
||||
|
||||
let data = ${JSON.stringify(data)}
|
||||
|
||||
for (let { url, sections } of data) {
|
||||
for (let [title, hash, content] of sections) {
|
||||
sectionIndex.add({
|
||||
url: url + (hash ? ('#' + hash) : ''),
|
||||
title,
|
||||
content: [title, ...content].join('\\n'),
|
||||
pageTitle: hash ? sections[0][0] : undefined,
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
export function search(query, options = {}) {
|
||||
let result = sectionIndex.search(query, {
|
||||
...options,
|
||||
enrich: true,
|
||||
})
|
||||
if (result.length === 0) {
|
||||
return []
|
||||
}
|
||||
return result[0].result.map((item) => ({
|
||||
url: item.id,
|
||||
title: item.doc.title,
|
||||
pageTitle: item.doc.pageTitle,
|
||||
}))
|
||||
}
|
||||
`;
|
||||
}),
|
||||
],
|
||||
});
|
||||
|
||||
if (typeof nextConfig.webpack === "function") {
|
||||
return nextConfig.webpack(config, options);
|
||||
}
|
||||
|
||||
return config;
|
||||
},
|
||||
});
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue