import type { ClickEvent } from "../../plug-api/types.ts"; import { editor, markdown, system } from "$sb/syscalls.ts"; import { addParentPointers, findNodeOfType, findParentMatching, nodeAtPos, type ParseTree, } from "$sb/lib/tree.ts"; import { isLocalPath, resolvePath } from "$sb/lib/resolve.ts"; import { looksLikePathWithExtension, parsePageRef } from "$sb/lib/page_ref.ts"; import { tagPrefix } from "../index/constants.ts"; async function actionClickOrActionEnter( mdTree: ParseTree | null, inNewWindow = false, ) { if (!mdTree) { return; } const navigationNodeFinder = (t: ParseTree) => [ "WikiLink", "Link", "Image", "URL", "NakedURL", "Link", "CommandLink", "PageRef", "Hashtag", ] .includes( t.type!, ); if (!navigationNodeFinder(mdTree)) { mdTree = findParentMatching(mdTree, navigationNodeFinder); if (!mdTree) { return; } } const currentPage = await editor.getCurrentPage(); switch (mdTree.type) { case "WikiLink": { const link = mdTree.children![1]!.children![0].text!; // Assume is attachment if it has extension if (looksLikePathWithExtension(link)) { 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; } return editor.navigate(pageRef, false, inNewWindow); } } case "PageRef": { const pageName = parsePageRef(mdTree.children![0].text!).page; return editor.navigate({ page: pageName, pos: 0 }, false, inNewWindow); } case "NakedURL": case "URL": return editor.openUrl(mdTree.children![0].text!); case "Image": case "Link": { const urlNode = findNodeOfType(mdTree, "URL"); if (!urlNode) { return; } const url = urlNode.children![0].text!; if (url.length <= 1) { return editor.flashNotification("Empty link, ignoring", "error"); } 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))), false, inNewWindow, ); } } else { return editor.openUrl(url); } } case "CommandLink": { const commandName = mdTree.children![1]!.children![0].text!; const argsNode = findNodeOfType(mdTree, "CommandLinkArgs"); const argsText = argsNode?.children?.[0]?.text; // 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); } catch (e: any) { await editor.flashNotification( `Error parsing command link arguments: ${e.message}`, "error", ); } break; } case "Hashtag": { const hashtag = mdTree.children![0].text!.slice(1); await editor.navigate( { page: `${tagPrefix}${hashtag}`, pos: 0 }, false, inNewWindow, ); break; } } } export async function linkNavigate() { const mdTree = await markdown.parseMarkdown(await editor.getText()); const newNode = nodeAtPos(mdTree, await editor.getCursor()); addParentPointers(mdTree); await actionClickOrActionEnter(newNode); } export async function clickNavigate(event: ClickEvent) { // Navigate by default, don't navigate when Alt is held if (event.altKey) { return; } const mdTree = await markdown.parseMarkdown(await editor.getText()); addParentPointers(mdTree); const newNode = nodeAtPos(mdTree, event.pos); await actionClickOrActionEnter(newNode, event.ctrlKey || event.metaKey); } export async function navigateCommand(cmdDef: any) { await editor.navigate({ page: cmdDef.page, pos: 0 }); } export async function navigateToPage(_cmdDef: any, pageName: string) { await editor.navigate({ page: pageName, pos: 0 }); } export async function navigateToURL(_cmdDef: any, url: string) { await editor.openUrl(url, false); } export async function navigateBack() { await editor.goHistory(-1); } export async function navigateForward() { await editor.goHistory(1); }