import { editor, markdown, space, sync } from "$sb/silverbullet-syscall/mod.ts"; import { removeParentPointers, renderToText, traverseTree, } from "$sb/lib/tree.ts"; import { renderDirectives } from "./directives.ts"; import { extractFrontmatter } from "$sb/lib/frontmatter.ts"; import { PageMeta } from "../../web/types.ts"; export async function updateDirectivesOnPageCommand(arg: any) { // If `arg` is a string, it's triggered automatically via an event, not explicitly via a command const explicitCall = typeof arg !== "string"; const pageMeta = await space.getPageMeta(await editor.getCurrentPage()); const text = await editor.getText(); const tree = await markdown.parseMarkdown(text); const metaData = await extractFrontmatter(tree, ["$disableDirectives"]); if (metaData.$disableDirectives) { // Not updating, directives disabled return; } // if (!(await sync.hasInitialSyncCompleted())) { // console.info("Initial sync hasn't completed yet, not updating directives."); // return; // } // If this page is shared ($share) via collab: disable directives as well // due to security concerns if (metaData.$share) { for (const uri of metaData.$share) { if (uri.startsWith("collab:")) { if (explicitCall) { await editor.flashNotification( "Directives are disabled for 'collab' pages (safety reasons).", "error", ); } return; } } } // Collect all directives and their body replacements const replacements: { fullMatch: string; textPromise: Promise }[] = []; // Convenience array to wait for all promises to resolve const allPromises: Promise[] = []; removeParentPointers(tree); traverseTree(tree, (tree) => { if (tree.type !== "Directive") { return false; } const fullMatch = text.substring(tree.from!, tree.to!); try { const promise = renderDirectives(pageMeta, tree); replacements.push({ textPromise: promise, fullMatch, }); allPromises.push(promise); } catch (e: any) { replacements.push({ fullMatch, textPromise: Promise.resolve( `${renderToText(tree.children![0])}\n**ERROR:** ${e.message}\n${ renderToText(tree.children![tree.children!.length - 1]) }`, ), }); } return true; }); // Wait for all to have processed await Promise.all(allPromises); // Iterate again and replace the bodies. Iterating again (not using previous positions) // because text may have changed in the mean time (directive processing may take some time) // Hypothetically in the mean time directives in text may have been changed/swapped, in which // case this will break. This would be a rare edge case, however. for (const replacement of replacements) { // Fetch the text every time, because dispatch() will have been made changes const text = await editor.getText(); // Determine the current position const index = text.indexOf(replacement.fullMatch); // This may happen if the query itself, or the user is editing inside the directive block (WHY!?) if (index === -1) { console.warn( "Text I got", text, ); console.warn( "Could not find directive in text, skipping", replacement.fullMatch, ); continue; } const from = index, to = index + replacement.fullMatch.length; const newText = await replacement.textPromise; if (text.substring(from, to) === newText) { // No change, skip continue; } await editor.dispatch({ changes: { from, to, insert: newText, }, }); } } // Pure server driven implementation of directive updating export async function updateDirectives( pageMeta: PageMeta, text: string, ) { const tree = await markdown.parseMarkdown(text); // Collect all directives and their body replacements const replacements: { fullMatch: string; textPromise: Promise }[] = []; const allPromises: Promise[] = []; traverseTree(tree, (tree) => { if (tree.type !== "Directive") { return false; } const fullMatch = text.substring(tree.from!, tree.to!); try { const promise = renderDirectives( pageMeta, tree, ); replacements.push({ textPromise: promise, fullMatch, }); allPromises.push(promise); } catch (e: any) { replacements.push({ fullMatch, textPromise: Promise.resolve( `${renderToText(tree.children![0])}\n**ERROR:** ${e.message}\n${ renderToText(tree.children![tree.children!.length - 1]) }`, ), }); } return true; }); // Wait for all to have processed await Promise.all(allPromises); // Iterate again and replace the bodies. for (const replacement of replacements) { text = text.replace(replacement.fullMatch, await replacement.textPromise); } return text; }