2024-02-09 04:00:45 +08:00
|
|
|
import { Hook, Manifest } from "../../lib/plugos/types.ts";
|
|
|
|
import { System } from "../../lib/plugos/system.ts";
|
2022-10-12 17:47:13 +08:00
|
|
|
import { Completion, CompletionContext, CompletionResult } from "../deps.ts";
|
2023-07-14 22:56:20 +08:00
|
|
|
import { Client } from "../client.ts";
|
2022-10-10 20:50:21 +08:00
|
|
|
import { syntaxTree } from "../deps.ts";
|
2024-02-24 16:26:00 +08:00
|
|
|
import { SlashCompletionOption, SlashCompletions } from "$type/types.ts";
|
|
|
|
import { safeRun } from "$lib/async.ts";
|
2022-03-29 17:21:32 +08:00
|
|
|
|
|
|
|
export type SlashCommandDef = {
|
|
|
|
name: string;
|
2022-07-04 21:07:27 +08:00
|
|
|
description?: string;
|
2022-11-18 23:04:37 +08:00
|
|
|
boost?: number;
|
2022-03-29 17:21:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
export type AppSlashCommand = {
|
|
|
|
slashCommand: SlashCommandDef;
|
|
|
|
run: () => Promise<void>;
|
|
|
|
};
|
|
|
|
|
|
|
|
export type SlashCommandHookT = {
|
|
|
|
slashCommand?: SlashCommandDef;
|
|
|
|
};
|
|
|
|
|
2024-02-03 02:19:07 +08:00
|
|
|
const slashCommandRegexp = /([^\w:]|^)\/[\w#\-]*/;
|
2022-08-02 19:22:10 +08:00
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
export class SlashCommandHook implements Hook<SlashCommandHookT> {
|
|
|
|
slashCommands = new Map<string, AppSlashCommand>();
|
2023-07-14 22:56:20 +08:00
|
|
|
private editor: Client;
|
2022-03-29 17:21:32 +08:00
|
|
|
|
2023-07-14 22:56:20 +08:00
|
|
|
constructor(editor: Client) {
|
2022-03-29 17:21:32 +08:00
|
|
|
this.editor = editor;
|
|
|
|
}
|
|
|
|
|
|
|
|
buildAllCommands(system: System<SlashCommandHookT>) {
|
|
|
|
this.slashCommands.clear();
|
2022-10-16 01:02:56 +08:00
|
|
|
for (const plug of system.loadedPlugs.values()) {
|
2022-10-10 20:50:21 +08:00
|
|
|
for (
|
|
|
|
const [name, functionDef] of Object.entries(
|
|
|
|
plug.manifest!.functions,
|
|
|
|
)
|
|
|
|
) {
|
2022-03-29 17:21:32 +08:00
|
|
|
if (!functionDef.slashCommand) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const cmd = functionDef.slashCommand;
|
|
|
|
this.slashCommands.set(cmd.name, {
|
|
|
|
slashCommand: cmd,
|
|
|
|
run: () => {
|
2022-07-04 21:07:27 +08:00
|
|
|
return plug.invoke(name, [cmd]);
|
2022-03-29 17:21:32 +08:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Completer for CodeMirror
|
2023-11-06 16:14:16 +08:00
|
|
|
public async slashCommandCompleter(
|
2022-10-10 20:50:21 +08:00
|
|
|
ctx: CompletionContext,
|
2023-11-06 16:14:16 +08:00
|
|
|
): Promise<CompletionResult | null> {
|
2022-10-16 01:02:56 +08:00
|
|
|
const prefix = ctx.matchBefore(slashCommandRegexp);
|
2022-03-29 17:21:32 +08:00
|
|
|
if (!prefix) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-08-03 18:24:35 +08:00
|
|
|
const prefixText = prefix.text;
|
2022-10-16 01:02:56 +08:00
|
|
|
const options: Completion[] = [];
|
2022-07-04 21:07:27 +08:00
|
|
|
|
|
|
|
// No slash commands in comment blocks (queries and such)
|
2022-10-16 01:02:56 +08:00
|
|
|
const currentNode = syntaxTree(ctx.state).resolveInner(ctx.pos);
|
2022-07-04 21:07:27 +08:00
|
|
|
if (currentNode.type.name === "CommentBlock") {
|
|
|
|
return null;
|
|
|
|
}
|
2023-11-06 16:14:16 +08:00
|
|
|
|
2022-10-16 01:02:56 +08:00
|
|
|
for (const def of this.slashCommands.values()) {
|
2022-03-29 17:21:32 +08:00
|
|
|
options.push({
|
|
|
|
label: def.slashCommand.name,
|
2022-07-04 21:07:27 +08:00
|
|
|
detail: def.slashCommand.description,
|
2022-11-18 23:04:37 +08:00
|
|
|
boost: def.slashCommand.boost,
|
2022-03-29 17:21:32 +08:00
|
|
|
apply: () => {
|
|
|
|
// Delete slash command part
|
2023-07-27 17:41:44 +08:00
|
|
|
this.editor.editorView.dispatch({
|
2022-03-29 17:21:32 +08:00
|
|
|
changes: {
|
2022-08-03 18:24:35 +08:00
|
|
|
from: prefix!.from + prefixText.indexOf("/"),
|
2022-03-29 17:21:32 +08:00
|
|
|
to: ctx.pos,
|
|
|
|
insert: "",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
// Replace with whatever the completion is
|
|
|
|
safeRun(async () => {
|
|
|
|
await def.run();
|
2022-07-04 21:07:27 +08:00
|
|
|
this.editor.focus();
|
2022-03-29 17:21:32 +08:00
|
|
|
});
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
2023-11-06 16:14:16 +08:00
|
|
|
|
2024-02-24 16:26:00 +08:00
|
|
|
const slashCompletions: CompletionResult | SlashCompletions | null =
|
|
|
|
await this.editor
|
|
|
|
.completeWithEvent(
|
|
|
|
ctx,
|
|
|
|
"slash:complete",
|
|
|
|
);
|
2023-11-06 16:14:16 +08:00
|
|
|
|
|
|
|
if (slashCompletions) {
|
2024-02-24 16:26:00 +08:00
|
|
|
for (
|
|
|
|
const slashCompletion of slashCompletions
|
|
|
|
.options as SlashCompletionOption[]
|
|
|
|
) {
|
2023-11-06 16:14:16 +08:00
|
|
|
options.push({
|
|
|
|
label: slashCompletion.label,
|
|
|
|
detail: slashCompletion.detail,
|
2024-02-03 02:19:07 +08:00
|
|
|
boost: slashCompletion.order && -slashCompletion.order,
|
2023-11-06 16:14:16 +08:00
|
|
|
apply: () => {
|
|
|
|
// Delete slash command part
|
|
|
|
this.editor.editorView.dispatch({
|
|
|
|
changes: {
|
|
|
|
from: prefix!.from + prefixText.indexOf("/"),
|
|
|
|
to: ctx.pos,
|
|
|
|
insert: "",
|
|
|
|
},
|
|
|
|
});
|
|
|
|
// Replace with whatever the completion is
|
|
|
|
safeRun(async () => {
|
2024-02-07 21:50:01 +08:00
|
|
|
await this.editor.clientSystem.system.invokeFunction(
|
2024-01-21 02:16:07 +08:00
|
|
|
slashCompletion.invoke,
|
|
|
|
[slashCompletion],
|
2023-11-06 16:14:16 +08:00
|
|
|
);
|
|
|
|
this.editor.focus();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-29 17:21:32 +08:00
|
|
|
return {
|
|
|
|
// + 1 because of the '/'
|
2022-08-03 18:24:35 +08:00
|
|
|
from: prefix.from + prefixText.indexOf("/") + 1,
|
2022-03-29 17:21:32 +08:00
|
|
|
options: options,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
apply(system: System<SlashCommandHookT>): void {
|
|
|
|
this.buildAllCommands(system);
|
|
|
|
system.on({
|
|
|
|
plugLoaded: () => {
|
|
|
|
this.buildAllCommands(system);
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
validateManifest(manifest: Manifest<SlashCommandHookT>): string[] {
|
2022-10-16 01:02:56 +08:00
|
|
|
const errors = [];
|
2022-03-29 17:21:32 +08:00
|
|
|
for (const [name, functionDef] of Object.entries(manifest.functions)) {
|
|
|
|
if (!functionDef.slashCommand) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const cmd = functionDef.slashCommand;
|
|
|
|
if (!cmd.name) {
|
|
|
|
errors.push(`Function ${name} has a command but no name`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|