2023-08-28 23:12:15 +08:00
|
|
|
import { events } from "$sb/syscalls.ts";
|
2023-05-24 02:53:53 +08:00
|
|
|
import { CompleteEvent } from "$sb/app_event.ts";
|
2023-07-24 17:09:17 +08:00
|
|
|
import { buildHandebarOptions } from "./util.ts";
|
|
|
|
import type { PageMeta } from "../../web/types.ts";
|
2023-08-28 23:12:15 +08:00
|
|
|
import type {
|
2023-08-08 22:35:46 +08:00
|
|
|
AttributeCompleteEvent,
|
|
|
|
AttributeCompletion,
|
2023-08-28 23:12:15 +08:00
|
|
|
} from "../index/attributes.ts";
|
2022-07-04 21:07:27 +08:00
|
|
|
|
2022-12-21 21:55:24 +08:00
|
|
|
export async function queryComplete(completeEvent: CompleteEvent) {
|
2023-08-02 03:35:19 +08:00
|
|
|
const querySourceMatch = /#query\s+([\w\-_]*)$/.exec(
|
|
|
|
completeEvent.linePrefix,
|
|
|
|
);
|
|
|
|
if (querySourceMatch) {
|
|
|
|
const allEvents = await events.listEvents();
|
2022-07-04 21:07:27 +08:00
|
|
|
|
2023-08-02 03:35:19 +08:00
|
|
|
return {
|
|
|
|
from: completeEvent.pos - querySourceMatch[1].length,
|
|
|
|
options: allEvents
|
|
|
|
.filter((eventName) => eventName.startsWith("query:"))
|
|
|
|
.map((source) => ({
|
|
|
|
label: source.substring("query:".length),
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
}
|
2022-07-04 21:07:27 +08:00
|
|
|
|
2023-08-02 03:35:19 +08:00
|
|
|
if (completeEvent.parentNodes.includes("DirectiveStart")) {
|
|
|
|
const querySourceMatch = /#query\s+([\w\-_]+)/.exec(
|
|
|
|
completeEvent.linePrefix,
|
|
|
|
);
|
|
|
|
const whereMatch =
|
|
|
|
/(where|order\s+by|and|select(\s+[\w\s,]+)?)\s+([\w\-_]*)$/.exec(
|
|
|
|
completeEvent.linePrefix,
|
|
|
|
);
|
|
|
|
if (querySourceMatch && whereMatch) {
|
|
|
|
const type = querySourceMatch[1];
|
|
|
|
const attributePrefix = whereMatch[3];
|
2023-08-08 22:35:46 +08:00
|
|
|
const completions = (await events.dispatchEvent(
|
|
|
|
`attribute:complete:${type}`,
|
|
|
|
{
|
|
|
|
source: type,
|
|
|
|
prefix: attributePrefix,
|
|
|
|
} as AttributeCompleteEvent,
|
|
|
|
)).flat() as AttributeCompletion[];
|
2023-08-02 03:35:19 +08:00
|
|
|
return {
|
|
|
|
from: completeEvent.pos - attributePrefix.length,
|
2023-08-08 22:35:46 +08:00
|
|
|
options: attributeCompletionsToCMCompletion(completions),
|
2023-08-02 03:35:19 +08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
2022-07-04 21:07:27 +08:00
|
|
|
}
|
2023-07-02 17:25:32 +08:00
|
|
|
|
2023-08-02 14:35:24 +08:00
|
|
|
export async function templateVariableComplete(completeEvent: CompleteEvent) {
|
2023-07-02 17:25:32 +08:00
|
|
|
const match = /\{\{([\w@]*)$/.exec(completeEvent.linePrefix);
|
|
|
|
if (!match) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const handlebarOptions = buildHandebarOptions({ name: "" } as PageMeta);
|
2023-08-02 14:35:24 +08:00
|
|
|
let allCompletions: any[] = Object.keys(handlebarOptions.helpers).map(
|
|
|
|
(name) => ({ label: name, detail: "helper" }),
|
|
|
|
);
|
|
|
|
allCompletions = allCompletions.concat(
|
|
|
|
Object.keys(handlebarOptions.data).map((key) => ({
|
|
|
|
label: `@${key}`,
|
|
|
|
detail: "global variable",
|
|
|
|
})),
|
|
|
|
);
|
|
|
|
|
2023-08-08 22:35:46 +08:00
|
|
|
const completions = (await events.dispatchEvent(
|
|
|
|
`attribute:complete:*`,
|
|
|
|
{
|
|
|
|
source: "*",
|
|
|
|
prefix: match[1],
|
|
|
|
} as AttributeCompleteEvent,
|
|
|
|
)).flat() as AttributeCompletion[];
|
|
|
|
|
2023-08-02 14:35:24 +08:00
|
|
|
allCompletions = allCompletions.concat(
|
2023-08-08 22:35:46 +08:00
|
|
|
attributeCompletionsToCMCompletion(completions),
|
2023-07-02 17:25:32 +08:00
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
|
|
|
from: completeEvent.pos - match[1].length,
|
2023-08-02 14:35:24 +08:00
|
|
|
options: allCompletions,
|
2023-07-02 17:25:32 +08:00
|
|
|
};
|
|
|
|
}
|
2023-08-08 22:35:46 +08:00
|
|
|
|
|
|
|
export function attributeCompletionsToCMCompletion(
|
|
|
|
completions: AttributeCompletion[],
|
|
|
|
) {
|
|
|
|
return completions.map(
|
|
|
|
(completion) => ({
|
|
|
|
label: completion.name,
|
|
|
|
detail: `${completion.type} (${completion.source})`,
|
|
|
|
type: "attribute",
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
}
|