2024-02-09 04:00:45 +08:00
|
|
|
import { codeWidget, editor, language, markdown, space } from "$sb/syscalls.ts";
|
2024-02-29 22:23:05 +08:00
|
|
|
import { parseTreeToAST, renderToText } from "$sb/lib/tree.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import type { CodeWidgetContent } from "../../plug-api/types.ts";
|
2024-01-21 02:16:07 +08:00
|
|
|
import { loadPageObject } from "../template/page.ts";
|
2024-01-05 03:08:12 +08:00
|
|
|
import { queryObjects } from "./api.ts";
|
2024-07-30 23:33:33 +08:00
|
|
|
import { type TemplateObject, WidgetConfig } from "../template/types.ts";
|
2024-01-05 03:08:12 +08:00
|
|
|
import { expressionToKvQueryExpression } from "$sb/lib/parse-query.ts";
|
2024-02-03 02:19:07 +08:00
|
|
|
import { evalQueryExpression } from "$sb/lib/query_expression.ts";
|
2024-01-09 00:08:26 +08:00
|
|
|
import { renderTemplate } from "../template/plug_api.ts";
|
|
|
|
import { extractFrontmatter } from "$sb/lib/frontmatter.ts";
|
|
|
|
import { rewritePageRefs } from "$sb/lib/resolve.ts";
|
2024-01-05 03:08:12 +08:00
|
|
|
|
2024-01-09 00:08:26 +08:00
|
|
|
export async function refreshWidgets() {
|
|
|
|
await codeWidget.refreshAll();
|
|
|
|
}
|
2024-01-05 03:08:12 +08:00
|
|
|
|
2024-01-09 00:08:26 +08:00
|
|
|
export async function renderTemplateWidgets(side: "top" | "bottom"): Promise<
|
2024-01-05 03:08:12 +08:00
|
|
|
CodeWidgetContent | null
|
|
|
|
> {
|
|
|
|
const text = await editor.getText();
|
2024-01-21 02:16:07 +08:00
|
|
|
let pageMeta = await loadPageObject(await editor.getCurrentPage());
|
2024-01-05 03:08:12 +08:00
|
|
|
const parsedMd = await markdown.parseMarkdown(text);
|
|
|
|
const frontmatter = await extractFrontmatter(parsedMd);
|
|
|
|
|
2024-01-21 02:16:07 +08:00
|
|
|
pageMeta = { ...pageMeta, ...frontmatter };
|
|
|
|
|
|
|
|
const blockTemplates = await queryObjects<TemplateObject>(
|
2024-01-05 03:08:12 +08:00
|
|
|
"template",
|
|
|
|
{
|
2024-01-21 02:16:07 +08:00
|
|
|
// where hooks.top/bottom exists
|
|
|
|
filter: ["attr", ["attr", "hooks"], side],
|
|
|
|
orderBy: [{
|
|
|
|
// order by hooks.top/bottom.order asc
|
|
|
|
expr: ["attr", ["attr", ["attr", "hooks"], side], "order"],
|
|
|
|
desc: false,
|
|
|
|
}],
|
2024-01-05 03:08:12 +08:00
|
|
|
},
|
|
|
|
);
|
2024-01-21 02:16:07 +08:00
|
|
|
// console.log(`Found the following ${side} templates`, blockTemplates);
|
2024-01-09 00:08:26 +08:00
|
|
|
const templateBits: string[] = [];
|
2024-01-05 03:08:12 +08:00
|
|
|
// Strategy: walk through all matching templates, evaluate the 'where' expression, and pick the first one that matches
|
2024-01-21 02:16:07 +08:00
|
|
|
for (const template of blockTemplates) {
|
|
|
|
if (!template.hooks) {
|
2024-01-09 16:37:06 +08:00
|
|
|
console.warn(
|
2024-01-21 02:16:07 +08:00
|
|
|
"No hooks specified for template",
|
2024-01-09 16:37:06 +08:00
|
|
|
template.ref,
|
2024-01-21 02:16:07 +08:00
|
|
|
"this should never happen",
|
2024-01-09 16:37:06 +08:00
|
|
|
);
|
|
|
|
continue;
|
|
|
|
}
|
2024-01-21 02:16:07 +08:00
|
|
|
const blockDef = WidgetConfig.parse(template.hooks[side]!);
|
2024-01-05 03:08:12 +08:00
|
|
|
const exprAST = parseTreeToAST(
|
2024-01-21 02:16:07 +08:00
|
|
|
await language.parseLanguage("expression", blockDef.where!),
|
2024-01-05 03:08:12 +08:00
|
|
|
);
|
|
|
|
const parsedExpression = expressionToKvQueryExpression(exprAST[1]);
|
2024-07-30 03:21:16 +08:00
|
|
|
|
2024-02-03 02:19:07 +08:00
|
|
|
if (await evalQueryExpression(parsedExpression, pageMeta, {}, {})) {
|
2024-01-05 03:08:12 +08:00
|
|
|
// Match! We're happy
|
2024-01-09 00:08:26 +08:00
|
|
|
const templateText = await space.readPage(template.ref);
|
|
|
|
let renderedTemplate = (await renderTemplate(
|
|
|
|
templateText,
|
|
|
|
frontmatter,
|
2024-02-03 02:19:07 +08:00
|
|
|
{ page: pageMeta },
|
2024-01-09 00:08:26 +08:00
|
|
|
)).text;
|
|
|
|
|
|
|
|
const parsedMarkdown = await markdown.parseMarkdown(renderedTemplate);
|
|
|
|
rewritePageRefs(parsedMarkdown, template.ref);
|
|
|
|
renderedTemplate = renderToText(parsedMarkdown);
|
|
|
|
|
2024-01-09 16:37:06 +08:00
|
|
|
templateBits.push(renderedTemplate.trim());
|
2024-01-05 03:08:12 +08:00
|
|
|
}
|
|
|
|
}
|
2024-01-09 16:37:06 +08:00
|
|
|
const summaryText = templateBits.join("\n");
|
2024-03-02 21:48:02 +08:00
|
|
|
// console.log("Summary:", summaryText);
|
2024-01-05 03:08:12 +08:00
|
|
|
return {
|
2024-01-09 00:08:26 +08:00
|
|
|
markdown: summaryText,
|
2024-01-05 03:08:12 +08:00
|
|
|
buttons: [
|
|
|
|
{
|
|
|
|
description: "Reload",
|
|
|
|
svg:
|
|
|
|
`<svg xmlns="http://www.w3.org/2000/svg" width="15" height="15" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="feather feather-refresh-cw"><polyline points="23 4 23 10 17 10"></polyline><polyline points="1 20 1 14 7 14"></polyline><path d="M3.51 9a9 9 0 0 1 14.85-3.36L23 10M1 14l4.64 4.36A9 9 0 0 0 20.49 15"></path></svg>`,
|
|
|
|
invokeFunction: "index.refreshWidgets",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
}
|