Markdown plug APIS

pull/555/head
Zef Hemel 2023-11-02 12:35:30 +01:00
parent b04bd5041c
commit 2844c8bc25
3 changed files with 71 additions and 2 deletions

66
plugs/markdown/api.ts Normal file
View File

@ -0,0 +1,66 @@
import {
findNodeOfType,
ParseTree,
renderToText,
replaceNodesMatchingAsync,
} from "$sb/lib/tree.ts";
import { codeWidget } from "$sb/syscalls.ts";
import { parseMarkdown } from "$sb/silverbullet-syscall/markdown.ts";
import {
MarkdownRenderOptions,
renderMarkdownToHtml,
} from "./markdown_render.ts";
/**
* Finds code widgets, runs their plug code to render and inlines their content in the parse tree
* @param mdTree parsed markdown tree
* @param pageName name of the current page
* @returns modified mdTree
*/
export async function expandCodeWidgets(
mdTree: ParseTree,
pageName: string,
): Promise<ParseTree> {
await replaceNodesMatchingAsync(mdTree, async (n) => {
if (n.type === "FencedCode") {
const codeInfo = findNodeOfType(n, "CodeInfo");
if (!codeInfo) {
return;
}
const codeType = codeInfo.children![0].text!;
const codeTextNode = findNodeOfType(n, "CodeText");
try {
// This will error out if this is not a code wiget, which is fine
const result = await codeWidget.render(
codeType,
renderToText(codeTextNode!),
pageName,
);
// Only do this for "markdown" widgets, that is: that can render to markdown
if (result.markdown) {
const parsedBody = await parseMarkdown(result.markdown);
// Recursively process
return expandCodeWidgets(
parsedBody,
pageName,
);
}
} catch (e: any) {
// 'not found' is to be expected (no code widget configured for this language)
// Every other error should probably be reported
if (!e.message.includes("not found")) {
console.error("Error rendering code", e.message);
}
}
}
});
return mdTree;
}
export async function markdownToHtml(
markdown: string,
options: MarkdownRenderOptions = {},
) {
const mdTree = await parseMarkdown(markdown);
return renderMarkdownToHtml(mdTree, options);
}

View File

@ -2,10 +2,13 @@ name: markdown
assets:
- "assets/*"
functions:
# API
markdownContentWidget:
path: markdown_content_widget.ts:markdownContentWidget
expandCodeWidgets:
path: api.ts:expandCodeWidgets
markdownToHtml:
path: api.ts:markdownToHtml
# User facing

View File

@ -8,7 +8,7 @@ import {
} from "$sb/lib/tree.ts";
import { Fragment, renderHtml, Tag } from "./html_render.ts";
type MarkdownRenderOptions = {
export type MarkdownRenderOptions = {
failOnUnknown?: true;
smartHardBreak?: true;
annotationPositions?: true;