2023-11-09 16:26:44 +08:00
|
|
|
import { handlebars, markdown, YAML } from "$sb/syscalls.ts";
|
|
|
|
import type { PageMeta } from "$sb/types.ts";
|
|
|
|
import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
|
|
|
|
import { TemplateObject } from "./types.ts";
|
|
|
|
import { renderToText } from "$sb/lib/tree.ts";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Strips the template from its frontmatter and renders it.
|
|
|
|
* The assumption is that the frontmatter has already been parsed and should not appear in thhe rendered output.
|
|
|
|
* @param templateText the template text
|
|
|
|
* @param data data to be rendered by the template
|
|
|
|
* @param globals a set of global variables
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
export async function renderTemplate(
|
|
|
|
templateText: string,
|
|
|
|
pageMeta: PageMeta,
|
|
|
|
data: any = {},
|
2023-11-13 22:49:21 +08:00
|
|
|
): Promise<{ frontmatter?: string; text: string }> {
|
2023-11-09 16:26:44 +08:00
|
|
|
const tree = await markdown.parseMarkdown(templateText);
|
|
|
|
const frontmatter: Partial<TemplateObject> = await extractFrontmatter(tree, {
|
|
|
|
removeFrontmatterSection: true,
|
|
|
|
removeTags: ["template"],
|
|
|
|
});
|
|
|
|
templateText = renderToText(tree).trimStart();
|
|
|
|
// If a 'frontmatter' key was specified in the frontmatter, use that as the frontmatter
|
2023-11-13 22:49:21 +08:00
|
|
|
let frontmatterText: string | undefined;
|
2023-11-09 16:26:44 +08:00
|
|
|
if (frontmatter.frontmatter) {
|
|
|
|
if (typeof frontmatter.frontmatter === "string") {
|
2023-11-13 22:49:21 +08:00
|
|
|
frontmatterText = frontmatter.frontmatter;
|
2023-11-09 16:26:44 +08:00
|
|
|
} else {
|
2023-11-13 22:49:21 +08:00
|
|
|
frontmatterText = await YAML.stringify(frontmatter.frontmatter);
|
2023-11-09 16:26:44 +08:00
|
|
|
}
|
|
|
|
}
|
2023-11-13 22:49:21 +08:00
|
|
|
return {
|
|
|
|
frontmatter: frontmatterText,
|
|
|
|
text: await handlebars.renderTemplate(templateText, data, {
|
|
|
|
page: pageMeta,
|
|
|
|
}),
|
|
|
|
};
|
2023-11-09 16:26:44 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Strips a template text from its frontmatter and #template tag
|
|
|
|
*/
|
|
|
|
export async function cleanTemplate(
|
|
|
|
templateText: string,
|
|
|
|
): Promise<string> {
|
|
|
|
const tree = await markdown.parseMarkdown(templateText);
|
|
|
|
await extractFrontmatter(tree, {
|
|
|
|
removeFrontmatterSection: true,
|
|
|
|
removeTags: ["template"],
|
|
|
|
});
|
|
|
|
return renderToText(tree).trimStart();
|
|
|
|
}
|