2024-02-29 22:23:05 +08:00
|
|
|
import type { ClickEvent } from "../../plug-api/types.ts";
|
2024-10-20 18:39:58 +08:00
|
|
|
import { extractHashtag } from "../../plug-api/lib/tags.ts";
|
2024-08-07 02:11:38 +08:00
|
|
|
import {
|
|
|
|
editor,
|
|
|
|
markdown,
|
|
|
|
system,
|
|
|
|
} from "@silverbulletmd/silverbullet/syscalls";
|
2022-11-18 23:04:37 +08:00
|
|
|
import {
|
|
|
|
addParentPointers,
|
2022-12-29 19:53:42 +08:00
|
|
|
findNodeOfType,
|
2022-11-18 23:04:37 +08:00
|
|
|
findParentMatching,
|
|
|
|
nodeAtPos,
|
2024-07-30 23:33:33 +08:00
|
|
|
type ParseTree,
|
2024-08-07 02:11:38 +08:00
|
|
|
} from "@silverbulletmd/silverbullet/lib/tree";
|
|
|
|
import {
|
|
|
|
isLocalPath,
|
|
|
|
resolvePath,
|
|
|
|
} from "@silverbulletmd/silverbullet/lib/resolve";
|
|
|
|
import {
|
|
|
|
looksLikePathWithExtension,
|
|
|
|
parsePageRef,
|
|
|
|
} from "@silverbulletmd/silverbullet/lib/page_ref";
|
2024-01-25 17:17:42 +08:00
|
|
|
import { tagPrefix } from "../index/constants.ts";
|
2022-03-29 23:02:28 +08:00
|
|
|
|
2022-10-29 15:23:12 +08:00
|
|
|
async function actionClickOrActionEnter(
|
|
|
|
mdTree: ParseTree | null,
|
|
|
|
inNewWindow = false,
|
|
|
|
) {
|
2022-04-04 00:42:12 +08:00
|
|
|
if (!mdTree) {
|
2022-02-28 21:58:18 +08:00
|
|
|
return;
|
|
|
|
}
|
2022-11-18 23:04:37 +08:00
|
|
|
const navigationNodeFinder = (t: ParseTree) =>
|
2022-12-19 20:42:20 +08:00
|
|
|
[
|
|
|
|
"WikiLink",
|
|
|
|
"Link",
|
|
|
|
"Image",
|
|
|
|
"URL",
|
|
|
|
"NakedURL",
|
|
|
|
"Link",
|
|
|
|
"CommandLink",
|
|
|
|
"PageRef",
|
2024-01-25 17:17:42 +08:00
|
|
|
"Hashtag",
|
2022-12-19 20:42:20 +08:00
|
|
|
]
|
2022-12-19 20:16:22 +08:00
|
|
|
.includes(
|
|
|
|
t.type!,
|
|
|
|
);
|
2022-11-18 23:04:37 +08:00
|
|
|
if (!navigationNodeFinder(mdTree)) {
|
|
|
|
mdTree = findParentMatching(mdTree, navigationNodeFinder);
|
|
|
|
if (!mdTree) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2023-07-02 17:25:32 +08:00
|
|
|
const currentPage = await editor.getCurrentPage();
|
2022-04-04 00:42:12 +08:00
|
|
|
switch (mdTree.type) {
|
2022-11-18 23:04:37 +08:00
|
|
|
case "WikiLink": {
|
2024-05-28 02:33:41 +08:00
|
|
|
const link = mdTree.children![1]!.children![0].text!;
|
|
|
|
// Assume is attachment if it has extension
|
2024-07-13 19:51:49 +08:00
|
|
|
if (looksLikePathWithExtension(link)) {
|
2024-05-28 02:33:41 +08:00
|
|
|
const attachmentPath = resolvePath(
|
|
|
|
currentPage,
|
|
|
|
"/" + decodeURI(link),
|
|
|
|
);
|
|
|
|
return editor.openUrl(attachmentPath);
|
|
|
|
} else {
|
|
|
|
const pageRef = parsePageRef(link);
|
|
|
|
pageRef.page = resolvePath(currentPage, "/" + pageRef.page);
|
|
|
|
if (!pageRef.page) {
|
|
|
|
pageRef.page = currentPage;
|
|
|
|
}
|
|
|
|
// This is an explicit navigate, move to the top
|
|
|
|
if (pageRef.pos === undefined) {
|
|
|
|
pageRef.pos = 0;
|
|
|
|
}
|
2024-07-03 02:13:41 +08:00
|
|
|
return editor.navigate(pageRef, false, inNewWindow);
|
2022-08-30 16:44:20 +08:00
|
|
|
}
|
2022-10-10 20:50:21 +08:00
|
|
|
}
|
2022-12-19 20:42:20 +08:00
|
|
|
case "PageRef": {
|
2024-01-24 18:58:33 +08:00
|
|
|
const pageName = parsePageRef(mdTree.children![0].text!).page;
|
2024-05-28 02:33:41 +08:00
|
|
|
return editor.navigate({ page: pageName, pos: 0 }, false, inNewWindow);
|
2022-12-19 20:42:20 +08:00
|
|
|
}
|
2022-04-12 19:33:07 +08:00
|
|
|
case "NakedURL":
|
2024-06-07 14:31:29 +08:00
|
|
|
case "URL":
|
2024-05-28 02:33:41 +08:00
|
|
|
return editor.openUrl(mdTree.children![0].text!);
|
2022-12-19 20:16:22 +08:00
|
|
|
case "Image":
|
2022-10-10 20:50:21 +08:00
|
|
|
case "Link": {
|
2022-12-29 19:53:42 +08:00
|
|
|
const urlNode = findNodeOfType(mdTree, "URL");
|
|
|
|
if (!urlNode) {
|
|
|
|
return;
|
|
|
|
}
|
2023-07-02 17:25:32 +08:00
|
|
|
const url = urlNode.children![0].text!;
|
2022-08-29 21:40:38 +08:00
|
|
|
if (url.length <= 1) {
|
2022-10-14 21:11:33 +08:00
|
|
|
return editor.flashNotification("Empty link, ignoring", "error");
|
2022-08-29 21:40:38 +08:00
|
|
|
}
|
2024-05-28 02:33:41 +08:00
|
|
|
if (isLocalPath(url)) {
|
|
|
|
if (/\.[a-zA-Z0-9>]+$/.test(url)) {
|
|
|
|
return editor.openUrl(
|
|
|
|
resolvePath(currentPage, decodeURI(url)),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return editor.navigate(
|
|
|
|
parsePageRef(resolvePath(currentPage, decodeURI(url))),
|
2024-07-03 02:13:41 +08:00
|
|
|
false,
|
|
|
|
inNewWindow,
|
2024-05-28 02:33:41 +08:00
|
|
|
);
|
|
|
|
}
|
2023-01-08 22:29:34 +08:00
|
|
|
} else {
|
2024-05-28 02:33:41 +08:00
|
|
|
return editor.openUrl(url);
|
2023-01-08 22:29:34 +08:00
|
|
|
}
|
2022-10-10 20:50:21 +08:00
|
|
|
}
|
|
|
|
case "CommandLink": {
|
2022-11-18 23:04:37 +08:00
|
|
|
const commandName = mdTree.children![1]!.children![0].text!;
|
2023-11-26 01:57:00 +08:00
|
|
|
const argsNode = findNodeOfType(mdTree, "CommandLinkArgs");
|
2024-07-30 21:17:34 +08:00
|
|
|
const argsText = argsNode?.children?.[0]?.text;
|
2023-11-26 01:57:00 +08:00
|
|
|
// Assume the arguments are can be parsed as the innards of a valid JSON list
|
|
|
|
try {
|
|
|
|
const args = argsText ? JSON.parse(`[${argsText}]`) : [];
|
|
|
|
await system.invokeCommand(commandName, args);
|
2023-12-20 00:55:11 +08:00
|
|
|
} catch (e: any) {
|
|
|
|
await editor.flashNotification(
|
|
|
|
`Error parsing command link arguments: ${e.message}`,
|
|
|
|
"error",
|
|
|
|
);
|
2023-11-26 01:57:00 +08:00
|
|
|
}
|
2022-07-18 22:44:43 +08:00
|
|
|
break;
|
2022-10-10 20:50:21 +08:00
|
|
|
}
|
2024-01-25 17:17:42 +08:00
|
|
|
case "Hashtag": {
|
2024-10-20 18:39:58 +08:00
|
|
|
const hashtag = extractHashtag(mdTree.children![0].text!);
|
2024-01-25 17:17:42 +08:00
|
|
|
await editor.navigate(
|
|
|
|
{ page: `${tagPrefix}${hashtag}`, pos: 0 },
|
|
|
|
false,
|
|
|
|
inNewWindow,
|
|
|
|
);
|
|
|
|
break;
|
|
|
|
}
|
2022-02-27 00:50:50 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-28 21:58:18 +08:00
|
|
|
export async function linkNavigate() {
|
2022-10-14 21:11:33 +08:00
|
|
|
const mdTree = await markdown.parseMarkdown(await editor.getText());
|
|
|
|
const newNode = nodeAtPos(mdTree, await editor.getCursor());
|
2022-11-18 23:04:37 +08:00
|
|
|
addParentPointers(mdTree);
|
2022-04-04 00:42:12 +08:00
|
|
|
await actionClickOrActionEnter(newNode);
|
2022-02-28 21:58:18 +08:00
|
|
|
}
|
2022-02-27 00:50:50 +08:00
|
|
|
|
2022-02-28 21:58:18 +08:00
|
|
|
export async function clickNavigate(event: ClickEvent) {
|
2022-10-29 15:23:12 +08:00
|
|
|
// Navigate by default, don't navigate when Alt is held
|
|
|
|
if (event.altKey) {
|
2022-04-12 02:34:09 +08:00
|
|
|
return;
|
2022-02-27 00:50:50 +08:00
|
|
|
}
|
2022-10-14 21:11:33 +08:00
|
|
|
const mdTree = await markdown.parseMarkdown(await editor.getText());
|
2022-11-18 23:04:37 +08:00
|
|
|
addParentPointers(mdTree);
|
2022-10-10 20:50:21 +08:00
|
|
|
const newNode = nodeAtPos(mdTree, event.pos);
|
2022-10-29 15:23:12 +08:00
|
|
|
await actionClickOrActionEnter(newNode, event.ctrlKey || event.metaKey);
|
2022-02-27 00:50:50 +08:00
|
|
|
}
|
2022-07-04 17:52:09 +08:00
|
|
|
|
|
|
|
export async function navigateCommand(cmdDef: any) {
|
2024-01-24 18:58:33 +08:00
|
|
|
await editor.navigate({ page: cmdDef.page, pos: 0 });
|
2022-07-04 17:52:09 +08:00
|
|
|
}
|
2024-01-25 18:42:36 +08:00
|
|
|
|
|
|
|
export async function navigateToPage(_cmdDef: any, pageName: string) {
|
|
|
|
await editor.navigate({ page: pageName, pos: 0 });
|
|
|
|
}
|
2024-02-03 02:19:07 +08:00
|
|
|
|
|
|
|
export async function navigateToURL(_cmdDef: any, url: string) {
|
|
|
|
await editor.openUrl(url, false);
|
|
|
|
}
|
2024-02-07 16:33:47 +08:00
|
|
|
|
|
|
|
export async function navigateBack() {
|
|
|
|
await editor.goHistory(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function navigateForward() {
|
|
|
|
await editor.goHistory(1);
|
|
|
|
}
|