import treeService from './tree.js'; import linkContextMenuService from "../menus/link_context_menu.js"; import appContext from "../components/app_context.js"; import froca from "./froca.js"; import utils from "./utils.js"; function getNotePathFromUrl(url) { const notePathMatch = /#(root[A-Za-z0-9_/]*)$/.exec(url); return notePathMatch === null ? null : notePathMatch[1]; } async function createLink(notePath, options = {}) { if (!notePath || !notePath.trim()) { logError("Missing note path"); return $("").text("[missing note]"); } if (!notePath.startsWith("root")) { // all note paths should start with "root/" (except for "root" itself) // used, e.g., to find internal links notePath = `root/${notePath}`; } const showTooltip = options.showTooltip === undefined ? true : options.showTooltip; const showNotePath = options.showNotePath === undefined ? false : options.showNotePath; const showNoteIcon = options.showNoteIcon === undefined ? false : options.showNoteIcon; const referenceLink = options.referenceLink === undefined ? false : options.referenceLink; const { noteId, parentNoteId } = treeService.getNoteIdAndParentIdFromNotePath(notePath); const viewScope = options.viewScope || {}; const viewMode = viewScope.viewMode || 'default'; let linkTitle = options.title; if (!linkTitle) { if (viewMode === 'attachments' && viewScope.attachmentId) { const attachment = await froca.getAttachment(viewScope.attachmentId); linkTitle = attachment ? attachment.title : '[missing attachment]'; } else { linkTitle = await treeService.getNoteTitle(noteId, parentNoteId); } } const $container = $(""); if (showNoteIcon) { let icon; if (viewMode === 'default') { const note = await froca.getNote(noteId); icon = note.getIcon(); } else if (viewMode === 'source') { icon = 'bx-code-curly'; } else if (viewMode === 'attachments') { icon = 'bx-file'; } if (icon) { $container .append($("").addClass(`bx ${icon}`)) .append(" "); } } const hash = calculateHash({ notePath, viewScope: viewScope }); const $noteLink = $("", { href: hash, text: linkTitle }); if (!showTooltip) { $noteLink.addClass("no-tooltip-preview"); } if (referenceLink) { $noteLink.addClass("reference-link"); } $container.append($noteLink); if (showNotePath) { const resolvedNotePathSegments = await treeService.resolveNotePathToSegments(notePath); if (notePath) { resolvedNotePathSegments.pop(); // remove last element const parentNotePath = resolvedNotePathSegments.join("/").trim(); if (parentNotePath) { $container.append($("").text(` (${await treeService.getNotePathTitle(parentNotePath)})`)); } } } return $container; } function calculateHash({notePath, ntxId, hoistedNoteId, viewScope = {}}) { notePath = notePath || ""; const params = [ ntxId ? { ntxId: ntxId } : null, (hoistedNoteId && hoistedNoteId !== 'root') ? { hoistedNoteId: hoistedNoteId } : null, viewScope.viewMode && viewScope.viewMode !== 'default' ? { viewMode: viewScope.viewMode } : null, viewScope.attachmentId ? { attachmentId: viewScope.attachmentId } : null ].filter(p => !!p); const paramStr = params.map(pair => { const name = Object.keys(pair)[0]; const value = pair[name]; return `${encodeURIComponent(name)}=${encodeURIComponent(value)}`; }).join("&"); if (!notePath && !paramStr) { return ""; } let hash = `#${notePath}`; if (paramStr) { hash += `?${paramStr}`; } return hash; } function parseNavigationStateFromUrl(url) { if (!url) { return {}; } const hashIdx = url.indexOf('#'); if (hashIdx === -1) { return {}; } const hash = url.substr(hashIdx + 1); // strip also the initial '#' const [notePath, paramString] = hash.split("?"); const viewScope = { viewMode: 'default' }; let ntxId = null; let hoistedNoteId = null; if (paramString) { for (const pair of paramString.split("&")) { let [name, value] = pair.split("="); name = decodeURIComponent(name); value = decodeURIComponent(value); if (name === 'ntxId') { ntxId = value; } else if (name === 'hoistedNoteId') { hoistedNoteId = value; } else if (['viewMode', 'attachmentId'].includes(name)) { viewScope[name] = value; } else { console.warn(`Unrecognized hash parameter '${name}'.`); } } } return { notePath, noteId: treeService.getNoteIdFromNotePath(notePath), ntxId, hoistedNoteId, viewScope }; } function goToLink(evt) { const $link = $(evt.target).closest("a,.block-link"); const hrefLink = $link.attr('href') || $link.attr('data-href'); if (hrefLink?.startsWith("data:")) { return true; } evt.preventDefault(); evt.stopPropagation(); const { notePath, viewScope } = parseNavigationStateFromUrl(hrefLink); const ctrlKey = utils.isCtrlKey(evt); const isLeftClick = evt.which === 1; const isMiddleClick = evt.which === 2; const openInNewTab = (isLeftClick && ctrlKey) || isMiddleClick; const leftClick = evt.which === 1; const middleClick = evt.which === 2; if (notePath) { if (openInNewTab) { appContext.tabManager.openTabWithNoteWithHoisting(notePath, { viewScope }); } else if (isLeftClick) { const ntxId = $(evt.target).closest("[data-ntx-id]").attr("data-ntx-id"); const noteContext = ntxId ? appContext.tabManager.getNoteContextById(ntxId) : appContext.tabManager.getActiveContext(); noteContext.setNote(notePath, { viewScope }).then(() => { if (noteContext !== appContext.tabManager.getActiveContext()) { appContext.tabManager.activateNoteContext(noteContext.ntxId); } }); } } else if (hrefLink) { const withinEditLink = $link.hasClass("ck-link-actions__preview"); const outsideOfCKEditor = $link.closest("[contenteditable]").length === 0; if (openInNewTab || (withinEditLink && (leftClick || middleClick)) || (outsideOfCKEditor && (leftClick || middleClick)) ) { if (hrefLink.toLowerCase().startsWith('http')) { window.open(hrefLink, '_blank'); } else if (hrefLink.toLowerCase().startsWith('file:') && utils.isElectron()) { const electron = utils.dynamicRequire('electron'); electron.shell.openPath(hrefLink); } } } return true; } function linkContextMenu(e) { const $link = $(e.target).closest("a"); const url = $link.attr("href") || $link.attr("data-href"); const { notePath, viewScope } = parseNavigationStateFromUrl(url); if (!notePath) { return; } e.preventDefault(); linkContextMenuService.openContextMenu(notePath, e, viewScope, null); } async function loadReferenceLinkTitle($el) { const url = $el.attr("href"); if (!url) { console.warn("Empty URL for parsing"); return; } const {noteId} = parseNavigationStateFromUrl(url); const note = await froca.getNote(noteId, true); let title; if (!note) { title = '[missing]'; } else { title = note.isDeleted ? `${note.title} (deleted)` : note.title; } if (note) { $el.addClass(note.getColorClass()); } $el.text(title); if (note) { $el.prepend($("").addClass(note.getIcon())); } } $(document).on('click', "a", goToLink); $(document).on('auxclick', "a", goToLink); // to handle the middle button $(document).on('contextmenu', 'a', linkContextMenu); $(document).on('dblclick', "a", e => { e.preventDefault(); e.stopPropagation(); const $link = $(e.target).closest("a"); const address = $link.attr('href'); if (address && address.startsWith('http')) { window.open(address, '_blank'); } }); $(document).on('mousedown', 'a', e => { if (e.which === 2) { // prevent paste on middle click // https://github.com/zadam/trilium/issues/2995 // https://developer.mozilla.org/en-US/docs/Web/API/Element/auxclick_event#preventing_default_actions e.preventDefault(); return false; } }); export default { getNotePathFromUrl, createLink, goToLink, loadReferenceLinkTitle, calculateHash, parseNavigationStateFromUrl };