silverbullet/plugs/template/template.ts

381 lines
11 KiB
TypeScript
Raw Normal View History

2023-11-06 16:14:16 +08:00
import { editor, handlebars, markdown, space, YAML } from "$sb/syscalls.ts";
2023-11-13 22:49:21 +08:00
import {
extractFrontmatter,
prepareFrontmatterDispatch,
} from "$sb/lib/frontmatter.ts";
2022-10-14 21:11:33 +08:00
import { renderToText } from "$sb/lib/tree.ts";
import { niceDate, niceTime } from "$sb/lib/dates.ts";
2022-10-14 21:11:33 +08:00
import { readSettings } from "$sb/lib/settings_page.ts";
import { cleanPageRef } from "$sb/lib/resolve.ts";
import { PageMeta } from "$sb/types.ts";
2023-11-06 16:14:16 +08:00
import { CompleteEvent, SlashCompletion } from "$sb/app_event.ts";
import { getObjectByRef, queryObjects } from "../index/plug_api.ts";
import { TemplateObject } from "./types.ts";
import { renderTemplate } from "./api.ts";
2023-11-06 16:14:16 +08:00
export async function templateSlashComplete(
completeEvent: CompleteEvent,
): Promise<SlashCompletion[]> {
const allTemplates = await queryObjects<TemplateObject>("template", {
// Only return templates that have a trigger
filter: ["!=", ["attr", "trigger"], ["null"]],
});
2023-11-06 16:14:16 +08:00
return allTemplates.map((template) => ({
label: template.trigger!,
detail: "template",
templatePage: template.ref,
pageName: completeEvent.pageName,
invoke: "template.insertSlashTemplate",
}));
}
export async function insertSlashTemplate(slashCompletion: SlashCompletion) {
const pageObject = await loadPageObject(slashCompletion.pageName);
2023-11-13 22:49:21 +08:00
const templateText = await space.readPage(slashCompletion.templatePage);
let { frontmatter, text } = await renderTemplate(templateText, pageObject);
2023-11-06 16:14:16 +08:00
const cursorPos = await editor.getCursor();
2023-11-13 22:49:21 +08:00
if (frontmatter) {
frontmatter = frontmatter.trim();
const pageText = await editor.getText();
const tree = await markdown.parseMarkdown(pageText);
const dispatch = await prepareFrontmatterDispatch(tree, frontmatter);
if (cursorPos === 0) {
dispatch.selection = { anchor: frontmatter.length + 9 };
}
await editor.dispatch(dispatch);
}
const carretPos = text.indexOf("|^|");
text = text.replace("|^|", "");
await editor.insertAtCursor(text);
2023-11-06 16:14:16 +08:00
if (carretPos !== -1) {
await editor.moveCursor(cursorPos + carretPos);
}
}
export async function instantiateTemplateCommand() {
2022-10-14 21:11:33 +08:00
const allPages = await space.listPages();
const { pageTemplatePrefix } = await readSettings({
pageTemplatePrefix: "template/page/",
});
2022-10-14 21:11:33 +08:00
const selectedTemplate = await editor.filterBox(
"Template",
2022-08-08 19:56:04 +08:00
allPages
.filter((pageMeta) => pageMeta.name.startsWith(pageTemplatePrefix))
.map((pageMeta) => ({
...pageMeta,
name: pageMeta.name.slice(pageTemplatePrefix.length),
})),
`Select the template to create a new page from (listing any page starting with <tt>${pageTemplatePrefix}</tt>)`,
);
if (!selectedTemplate) {
return;
}
console.log("Selected template", selectedTemplate);
2022-10-14 21:11:33 +08:00
const text = await space.readPage(
`${pageTemplatePrefix}${selectedTemplate.name}`,
2022-08-08 19:56:04 +08:00
);
2022-10-14 21:11:33 +08:00
const parseTree = await markdown.parseMarkdown(text);
const additionalPageMeta = await extractFrontmatter(parseTree, {
removeKeys: [
"$name",
"$disableDirectives",
],
});
const tempPageMeta: PageMeta = {
2023-11-06 16:14:16 +08:00
tags: ["page"],
ref: "",
name: "",
2023-11-06 16:14:16 +08:00
created: "",
lastModified: "",
perm: "rw",
};
2023-02-27 22:20:16 +08:00
if (additionalPageMeta.$name) {
additionalPageMeta.$name = await replaceTemplateVars(
2023-02-27 22:20:16 +08:00
additionalPageMeta.$name,
tempPageMeta,
2023-02-27 22:20:16 +08:00
);
}
2022-10-14 21:11:33 +08:00
const pageName = await editor.prompt(
"Name of new page",
additionalPageMeta.$name,
);
if (!pageName) {
return;
}
tempPageMeta.name = pageName;
2023-02-27 22:51:54 +08:00
try {
// Fails if doesn't exist
2023-03-06 21:17:03 +08:00
await space.getPageMeta(pageName);
2023-02-27 22:51:54 +08:00
// So, page exists, let's warn
if (
!await editor.confirm(
`Page ${pageName} already exists, are you sure you want to override it?`,
)
) {
return;
}
} catch {
// The preferred scenario, let's keep going
}
const pageText = await replaceTemplateVars(
renderToText(parseTree),
tempPageMeta,
);
2022-10-14 21:11:33 +08:00
await space.writePage(pageName, pageText);
await editor.navigate(pageName);
}
export async function insertSnippet() {
2022-10-14 21:11:33 +08:00
const allPages = await space.listPages();
const { snippetPrefix } = await readSettings({
snippetPrefix: "snippet/",
});
2022-10-14 21:11:33 +08:00
const cursorPos = await editor.getCursor();
const page = await editor.getCurrentPage();
const pageMeta = await space.getPageMeta(page);
2022-10-14 21:11:33 +08:00
const allSnippets = allPages
.filter((pageMeta) => pageMeta.name.startsWith(snippetPrefix))
.map((pageMeta) => ({
...pageMeta,
name: pageMeta.name.slice(snippetPrefix.length),
}));
2022-10-14 21:11:33 +08:00
const selectedSnippet = await editor.filterBox(
"Snippet",
allSnippets,
`Select the snippet to insert (listing any page starting with <tt>${snippetPrefix}</tt>)`,
);
if (!selectedSnippet) {
return;
}
2022-10-14 21:11:33 +08:00
const text = await space.readPage(`${snippetPrefix}${selectedSnippet.name}`);
let templateText = await replaceTemplateVars(text, pageMeta);
2022-10-14 21:11:33 +08:00
const carretPos = templateText.indexOf("|^|");
templateText = templateText.replace("|^|", "");
templateText = await replaceTemplateVars(templateText, pageMeta);
2022-10-14 21:11:33 +08:00
await editor.insertAtCursor(templateText);
if (carretPos !== -1) {
2022-10-14 21:11:33 +08:00
await editor.moveCursor(cursorPos + carretPos);
}
}
export async function applyPageTemplateCommand() {
const allPages = await space.listPages();
const { pageTemplatePrefix } = await readSettings({
pageTemplatePrefix: "template/page/",
});
const cursorPos = await editor.getCursor();
const page = await editor.getCurrentPage();
const pageMeta = await space.getPageMeta(page);
const allSnippets = allPages
.filter((pageMeta) => pageMeta.name.startsWith(pageTemplatePrefix))
.map((pageMeta) => ({
...pageMeta,
name: pageMeta.name.slice(pageTemplatePrefix.length),
}));
const selectedPage = await editor.filterBox(
"Page template",
allSnippets,
`Select the page template to apply (listing any page starting with <tt>${pageTemplatePrefix}</tt>)`,
);
if (!selectedPage) {
return;
}
const text = await space.readPage(
`${pageTemplatePrefix}${selectedPage.name}`,
);
let templateText = await replaceTemplateVars(text, pageMeta);
const carretPos = templateText.indexOf("|^|");
templateText = templateText.replace("|^|", "");
templateText = await replaceTemplateVars(templateText, pageMeta);
await editor.insertAtCursor(templateText);
if (carretPos !== -1) {
await editor.moveCursor(cursorPos + carretPos);
}
}
2023-11-06 16:14:16 +08:00
export async function loadPageObject(pageName: string): Promise<PageMeta> {
return (await getObjectByRef<PageMeta>(
pageName,
"page",
pageName,
)) || {
ref: pageName,
name: pageName,
tags: ["page"],
lastModified: "",
created: "",
} as PageMeta;
}
export function replaceTemplateVars(
s: string,
pageMeta: PageMeta,
): Promise<string> {
return handlebars.renderTemplate(s, {}, { page: pageMeta });
}
2022-05-07 00:55:04 +08:00
export async function quickNoteCommand() {
2022-10-14 21:11:33 +08:00
const { quickNotePrefix } = await readSettings({
2022-08-08 19:56:04 +08:00
quickNotePrefix: "📥 ",
});
const date = niceDate(new Date());
const time = niceTime(new Date());
2022-10-14 21:11:33 +08:00
const pageName = `${quickNotePrefix}${date} ${time}`;
await editor.navigate(pageName);
2022-05-07 00:55:04 +08:00
}
2022-07-06 18:18:47 +08:00
export async function dailyNoteCommand() {
2022-10-14 21:11:33 +08:00
const { dailyNoteTemplate, dailyNotePrefix } = await readSettings({
dailyNoteTemplate: "[[template/page/Daily Note]]",
2022-08-08 19:56:04 +08:00
dailyNotePrefix: "📅 ",
});
2022-10-14 21:11:33 +08:00
const date = niceDate(new Date());
const pageName = `${dailyNotePrefix}${date}`;
let carretPos = 0;
try {
await space.getPageMeta(pageName);
} catch {
// Doesn't exist, let's create
let dailyNoteTemplateText = "";
2022-08-08 19:56:04 +08:00
try {
dailyNoteTemplateText = await space.readPage(
cleanPageRef(dailyNoteTemplate),
);
carretPos = dailyNoteTemplateText.indexOf("|^|");
if (carretPos === -1) {
carretPos = 0;
}
dailyNoteTemplateText = dailyNoteTemplateText.replace("|^|", "");
2022-08-08 19:56:04 +08:00
} catch {
console.warn(`No daily note template found at ${dailyNoteTemplate}`);
2022-08-08 19:56:04 +08:00
}
await space.writePage(
pageName,
await replaceTemplateVars(dailyNoteTemplateText, {
2023-11-06 16:14:16 +08:00
tags: ["page"],
ref: pageName,
name: pageName,
2023-11-06 16:14:16 +08:00
created: "",
lastModified: "",
perm: "rw",
}),
);
2022-08-08 19:56:04 +08:00
}
await editor.navigate(pageName, carretPos);
2022-07-06 18:18:47 +08:00
}
2022-11-02 16:06:30 +08:00
function getWeekStartDate(monday = false) {
const d = new Date();
const day = d.getDay();
let diff = d.getDate() - day;
if (monday) {
diff += day == 0 ? -6 : 1;
}
return new Date(d.setDate(diff));
}
export async function weeklyNoteCommand() {
const { weeklyNoteTemplate, weeklyNotePrefix, weeklyNoteMonday } =
await readSettings({
weeklyNoteTemplate: "[[template/page/Weekly Note]]",
2022-11-02 16:06:30 +08:00
weeklyNotePrefix: "🗓️ ",
weeklyNoteMonday: false,
});
let weeklyNoteTemplateText = "";
try {
weeklyNoteTemplateText = await space.readPage(
cleanPageRef(weeklyNoteTemplate),
);
2022-11-02 16:06:30 +08:00
} catch {
console.warn(`No weekly note template found at ${weeklyNoteTemplate}`);
}
const date = niceDate(getWeekStartDate(weeklyNoteMonday));
const pageName = `${weeklyNotePrefix}${date}`;
if (weeklyNoteTemplateText) {
try {
await space.getPageMeta(pageName);
} catch {
// Doesn't exist, let's create
await space.writePage(
pageName,
await replaceTemplateVars(weeklyNoteTemplateText, {
name: pageName,
2023-11-06 16:14:16 +08:00
ref: pageName,
tags: ["page"],
created: "",
lastModified: "",
perm: "rw",
}),
2022-11-02 16:06:30 +08:00
);
}
await editor.navigate(pageName);
} else {
await editor.navigate(pageName);
}
}
export async function insertTemplateText(cmdDef: any) {
2022-10-14 21:11:33 +08:00
const cursorPos = await editor.getCursor();
const page = await editor.getCurrentPage();
2023-11-06 16:14:16 +08:00
const pageMeta = await loadPageObject(page);
let templateText: string = cmdDef.value;
2022-10-14 21:11:33 +08:00
const carretPos = templateText.indexOf("|^|");
templateText = templateText.replace("|^|", "");
2023-11-06 16:14:16 +08:00
templateText = await replaceTemplateVars(templateText, pageMeta);
2022-10-14 21:11:33 +08:00
await editor.insertAtCursor(templateText);
if (carretPos !== -1) {
2022-10-14 21:11:33 +08:00
await editor.moveCursor(cursorPos + carretPos);
}
2022-05-07 00:55:04 +08:00
}
export async function applyLineReplace(cmdDef: any) {
const cursorPos = await editor.getCursor();
const text = await editor.getText();
const matchRegex = new RegExp(cmdDef.match);
let startOfLine = cursorPos;
while (startOfLine > 0 && text[startOfLine - 1] !== "\n") {
startOfLine--;
}
let currentLine = text.slice(startOfLine, cursorPos);
const emptyLine = !currentLine;
currentLine = currentLine.replace(matchRegex, cmdDef.replace);
await editor.dispatch({
changes: {
from: startOfLine,
to: cursorPos,
insert: currentLine,
},
selection: emptyLine
? {
anchor: startOfLine + currentLine.length,
}
: undefined,
});
}