import { editor, markdown, YAML } from "@silverbulletmd/silverbullet/syscalls"; import type { CodeWidgetContent } from "@silverbulletmd/silverbullet/types"; import { stripMarkdown } from "@silverbulletmd/silverbullet/lib/markdown"; import { traverseTree } from "@silverbulletmd/silverbullet/lib/tree"; type Header = { name: string; pos: number; level: number; }; type TocConfig = { // Only show the TOC if there are at least this many headers minHeaders?: number; // Don't show the TOC if there are more than this many headers maxHeaders?: number; header?: boolean; headerText?: string; maxLevel?: number; minLevel?: number; }; export async function widget( bodyText: string, ): Promise { let config: TocConfig = {}; if (bodyText.trim() !== "") { config = await YAML.parse(bodyText); } const page = await editor.getCurrentPage(); const text = await editor.getText(); const tree = await markdown.parseMarkdown(text); const headers: Header[] = []; traverseTree(tree, (n) => { if (n.type?.startsWith("ATXHeading")) { headers.push({ name: n.children! .slice(1) .map(stripMarkdown) .join("") .trim(), pos: n.from!, level: +n.type[n.type.length - 1], }); return true; } return false; }); if (headers.length === 0) { return null; } if (config.minHeaders && headers.length < config.minHeaders) { // Not enough headers, not showing TOC return null; } if (config.maxHeaders && headers.length > config.maxHeaders) { // Too many headers, not showing TOC return null; } let headerText = (config.headerText ?? "# Table of Contents") + "\n"; if (config.header === false) { headerText = ""; } // Adjust level down if only sub-headers are used let minLevel = headers.reduce( (min, header) => Math.min(min, header.level), 6, ); if (config.minLevel && config.minLevel > minLevel) minLevel = config.minLevel; let renderedMd = headerText; for (const header of headers) { if ( config.maxLevel && header.level > config.maxLevel || (config.minLevel && header.level < config.minLevel) ) { continue; } renderedMd = renderedMd + " ".repeat((header.level - minLevel) * 2) + "* [[" + page + "@" + header.pos + "|" + header.name + "]]\n"; } // console.log("Markdown", renderedMd); return { markdown: renderedMd, buttons: [ { description: "Bake result", svg: ``, invokeFunction: ["query.bakeButton", bodyText], }, { description: "Edit", svg: ``, invokeFunction: ["query.editButton", bodyText], }, { description: "Reload", svg: ``, invokeFunction: ["index.refreshWidgets"], }, ], }; }