import type { Hook, Manifest } from "../../lib/plugos/types.ts"; import type { System } from "../../lib/plugos/system.ts"; import { EventEmitter } from "../../lib/plugos/event.ts"; import type { ObjectValue } from "../../plug-api/types.ts"; import type { FrontmatterConfig } from "../../plugs/template/types.ts"; import { throttle } from "../../lib/async.ts"; import type { AppCommand, CommandHookEvents } from "../../lib/command.ts"; import type { CommandHookT } from "$lib/manifest.ts"; export class CommandHook extends EventEmitter implements Hook { editorCommands = new Map(); system!: System; constructor( private readOnly: boolean, private additionalCommandsMap: Map, ) { super(); } throttledBuildAllCommands = throttle(() => { this.buildAllCommands().catch(console.error); }, 200); async buildAllCommands() { this.editorCommands.clear(); for (const plug of this.system.loadedPlugs.values()) { for ( const [name, functionDef] of Object.entries( plug.manifest!.functions, ) ) { if (!functionDef.command) { continue; } const cmd = functionDef.command; if (cmd.requireMode === "rw" && this.readOnly) { // Bit hacky, but don't expose commands that require write mode in read-only mode continue; } this.editorCommands.set(cmd.name, { command: cmd, run: (args?: string[]) => { return plug.invoke(name, [cmd, ...args ?? []]); }, }); } } await this.loadPageTemplateCommands(); for (const [name, cmd] of this.additionalCommandsMap) { this.editorCommands.set(name, cmd); } this.emit("commandsUpdated", this.editorCommands); } async loadPageTemplateCommands() { // This relies on two plugs being loaded: index and template const indexPlug = this.system.loadedPlugs.get("index"); const templatePlug = this.system.loadedPlugs.get("template"); if (!indexPlug || !templatePlug) { // Index and template plugs not yet loaded, let's wait return; } // Query all page templates that have a command configured const templateCommands: ObjectValue[] = await indexPlug .invoke( "queryObjects", ["template", { // where hooks.newPage.command or hooks.snippet.command filter: ["or", [ "attr", ["attr", ["attr", "hooks"], "newPage"], "command", ], [ "attr", ["attr", ["attr", "hooks"], "snippet"], "command", ]], }], ); // console.log("Template commands", templateCommands); for (const page of templateCommands) { try { if (page.hooks!.newPage) { const newPageConfig = page.hooks!.newPage; const cmdDef = { name: newPageConfig.command!, key: newPageConfig.key, mac: newPageConfig.mac, }; this.editorCommands.set(newPageConfig.command!, { command: cmdDef, run: () => { return templatePlug.invoke("newPageCommand", [cmdDef, page.ref]); }, }); } if (page.hooks!.snippet) { const snippetConfig = page.hooks!.snippet; const cmdDef = { name: snippetConfig.command!, key: snippetConfig.key, mac: snippetConfig.mac, }; this.editorCommands.set(snippetConfig.command!, { command: cmdDef, run: () => { return templatePlug.invoke("insertSnippetTemplate", [ { templatePage: page.ref }, ]); }, }); } } catch (e: any) { console.error("Error loading command from", page.ref, e); } } // console.log("Page template commands", pageTemplateCommands); } apply(system: System): void { this.system = system; system.on({ plugLoaded: () => { this.throttledBuildAllCommands(); }, }); // On next tick setTimeout(() => { this.throttledBuildAllCommands(); }); } validateManifest(manifest: Manifest): string[] { const errors = []; for (const [name, functionDef] of Object.entries(manifest.functions)) { if (!functionDef.command) { continue; } const cmd = functionDef.command; if (!cmd.name) { errors.push(`Function ${name} has a command but no name`); } } return []; } }