2024-08-07 02:11:38 +08:00
|
|
|
import { collectNodesOfType } from "@silverbulletmd/silverbullet/lib/tree";
|
2024-05-28 02:33:41 +08:00
|
|
|
import type {
|
|
|
|
CompleteEvent,
|
|
|
|
IndexTreeEvent,
|
|
|
|
ObjectValue,
|
|
|
|
QueryExpression,
|
2024-08-07 02:11:38 +08:00
|
|
|
} from "@silverbulletmd/silverbullet/types";
|
2023-10-03 20:16:33 +08:00
|
|
|
import { indexObjects, queryObjects } from "./api.ts";
|
2024-08-07 02:11:38 +08:00
|
|
|
import { parsePageRef } from "@silverbulletmd/silverbullet/lib/page_ref";
|
2022-08-30 16:44:20 +08:00
|
|
|
|
2023-10-03 20:16:33 +08:00
|
|
|
type AnchorObject = ObjectValue<{
|
|
|
|
name: string;
|
|
|
|
page: string;
|
|
|
|
pos: number;
|
|
|
|
}>;
|
2022-08-30 16:44:20 +08:00
|
|
|
|
|
|
|
export async function indexAnchors({ name: pageName, tree }: IndexTreeEvent) {
|
2023-10-03 20:16:33 +08:00
|
|
|
const anchors: ObjectValue<AnchorObject>[] = [];
|
2024-03-15 03:39:38 +08:00
|
|
|
const anchorNames = new Set<string>();
|
2022-08-30 16:44:20 +08:00
|
|
|
|
|
|
|
collectNodesOfType(tree, "NamedAnchor").forEach((n) => {
|
2022-10-25 01:40:52 +08:00
|
|
|
const aName = n.children![0].text!.substring(1);
|
2024-03-15 03:39:38 +08:00
|
|
|
if (anchorNames.has(aName)) {
|
|
|
|
console.warn("Duplicate anchor", aName, "in", pageName);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
anchorNames.add(aName);
|
|
|
|
}
|
2022-08-30 16:44:20 +08:00
|
|
|
anchors.push({
|
2023-11-03 19:01:33 +08:00
|
|
|
ref: `${pageName}$${aName}`,
|
2024-01-11 20:20:50 +08:00
|
|
|
tag: "anchor",
|
2023-10-03 20:16:33 +08:00
|
|
|
name: aName,
|
|
|
|
page: pageName,
|
|
|
|
pos: n.from!,
|
2022-08-30 16:44:20 +08:00
|
|
|
});
|
|
|
|
});
|
2022-10-21 16:00:43 +08:00
|
|
|
// console.log("Found", anchors.length, "anchors(s)");
|
2023-10-03 20:16:33 +08:00
|
|
|
await indexObjects(pageName, anchors);
|
2022-08-30 16:44:20 +08:00
|
|
|
}
|
|
|
|
|
2022-12-21 21:55:24 +08:00
|
|
|
export async function anchorComplete(completeEvent: CompleteEvent) {
|
2024-01-24 18:58:33 +08:00
|
|
|
const match = /\[\[([^\]$:]*\$[\w\.\-\/]*)$/.exec(
|
2023-11-03 19:01:33 +08:00
|
|
|
completeEvent.linePrefix,
|
|
|
|
);
|
2022-12-21 21:55:24 +08:00
|
|
|
if (!match) {
|
2022-08-30 16:44:20 +08:00
|
|
|
return null;
|
|
|
|
}
|
2022-12-21 21:55:24 +08:00
|
|
|
|
2024-01-24 18:58:33 +08:00
|
|
|
const pageRef = parsePageRef(match[1]).page;
|
2023-11-02 23:19:33 +08:00
|
|
|
let filter: QueryExpression | undefined = ["=", ["attr", "page"], [
|
|
|
|
"string",
|
|
|
|
pageRef,
|
|
|
|
]];
|
2022-08-30 16:44:20 +08:00
|
|
|
if (!pageRef) {
|
2023-11-02 23:19:33 +08:00
|
|
|
// "bare" anchor, match any page for completion purposes
|
|
|
|
filter = undefined;
|
2022-08-30 16:44:20 +08:00
|
|
|
}
|
2024-01-15 23:43:12 +08:00
|
|
|
const allAnchors = await queryObjects<AnchorObject>("anchor", { filter }, 5);
|
2022-08-30 16:44:20 +08:00
|
|
|
return {
|
2023-11-02 23:19:33 +08:00
|
|
|
from: completeEvent.pos - match[1].length,
|
2022-08-30 16:44:20 +08:00
|
|
|
options: allAnchors.map((a) => ({
|
2023-11-03 19:01:33 +08:00
|
|
|
label: a.page === completeEvent.pageName ? `\$${a.name}` : a.ref,
|
2022-08-30 16:44:20 +08:00
|
|
|
type: "anchor",
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
}
|