mirror of
https://github.com/zadam/trilium.git
synced 2025-03-01 14:22:32 +01:00
168 lines
9.5 KiB
JavaScript
168 lines
9.5 KiB
JavaScript
import treeService from './tree.js';
|
|
import froca from "./froca.js";
|
|
import clipboard from './clipboard.js';
|
|
import noteCreateService from "./note_create.js";
|
|
import contextMenu from "./context_menu.js";
|
|
import appContext from "./app_context.js";
|
|
import server from "./server.js";
|
|
|
|
class TreeContextMenu {
|
|
/**
|
|
* @param {NoteTreeWidget} treeWidget
|
|
* @param {FancytreeNode} node
|
|
*/
|
|
constructor(treeWidget, node) {
|
|
this.treeWidget = treeWidget;
|
|
this.node = node;
|
|
}
|
|
|
|
async show(e) {
|
|
contextMenu.show({
|
|
x: e.pageX,
|
|
y: e.pageY,
|
|
items: await this.getMenuItems(),
|
|
selectMenuItemHandler: (item, e) => this.selectMenuItemHandler(item, e)
|
|
})
|
|
}
|
|
|
|
async getNoteTypeItems(command) {
|
|
const items = [
|
|
{ title: "Text", command: command, type: "text", uiIcon: "bx bx-note" },
|
|
{ title: "Code", command: command, type: "code", uiIcon: "bx bx-code" },
|
|
{ title: "Saved Search", command: command, type: "search", uiIcon: "bx bx-file-find" },
|
|
{ title: "Relation Map", command: command, type: "relation-map", uiIcon: "bx bx-map-alt" },
|
|
{ title: "Note Map", command: command, type: "note-map", uiIcon: "bx bx-map-alt" },
|
|
{ title: "Render Note", command: command, type: "render", uiIcon: "bx bx-extension" },
|
|
{ title: "Book", command: command, type: "book", uiIcon: "bx bx-book" },
|
|
{ title: "Mermaid Diagram", command: command, type: "mermaid", uiIcon: "bx bx-selection" },
|
|
{ title: "Canvas", command: command, type: "canvas", uiIcon: "bx bx-pen" },
|
|
{ title: "Web View", command: command, type: "iframe", uiIcon: "bx bx-globe-alt" },
|
|
];
|
|
|
|
const templateNoteIds = await server.get("search-templates");
|
|
const templateNotes = await froca.getNotes(templateNoteIds);
|
|
|
|
if (items.length > 0) {
|
|
items.push({ title: "----" });
|
|
|
|
for (const templateNote of templateNotes) {
|
|
items.push({
|
|
title: templateNote.title,
|
|
uiIcon: templateNote.getIcon()
|
|
});
|
|
}
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
async getMenuItems() {
|
|
const note = await froca.getNote(this.node.data.noteId);
|
|
const branch = froca.getBranch(this.node.data.branchId);
|
|
const isNotRoot = note.noteId !== 'root';
|
|
const isHoisted = note.noteId === appContext.tabManager.getActiveContext().hoistedNoteId;
|
|
const parentNote = isNotRoot ? await froca.getNote(branch.parentNoteId) : null;
|
|
|
|
// some actions don't support multi-note so they are disabled when notes are selected
|
|
// the only exception is when the only selected note is the one that was right-clicked, then
|
|
// it's clear what the user meant to do.
|
|
const selNodes = this.treeWidget.getSelectedNodes();
|
|
const noSelectedNotes = selNodes.length === 0
|
|
|| (selNodes.length === 1 && selNodes[0] === this.node);
|
|
|
|
const notSearch = note.type !== 'search';
|
|
const parentNotSearch = !parentNote || parentNote.type !== 'search';
|
|
const insertNoteAfterEnabled = isNotRoot && !isHoisted && parentNotSearch;
|
|
|
|
return [
|
|
{ title: 'Open in a new tab <kbd>Ctrl+Click</kbd>', command: "openInTab", uiIcon: "bx bx-empty", enabled: noSelectedNotes },
|
|
{ title: 'Open in a new split', command: "openNoteInSplit", uiIcon: "bx bx-dock-right", enabled: noSelectedNotes },
|
|
{ title: 'Insert note after <kbd data-command="createNoteAfter"></kbd>', command: "insertNoteAfter", uiIcon: "bx bx-plus",
|
|
items: insertNoteAfterEnabled ? await this.getNoteTypeItems("insertNoteAfter") : null,
|
|
enabled: insertNoteAfterEnabled && noSelectedNotes },
|
|
{ title: 'Insert child note <kbd data-command="createNoteInto"></kbd>', command: "insertChildNote", uiIcon: "bx bx-plus",
|
|
items: notSearch ? await this.getNoteTypeItems("insertChildNote") : null,
|
|
enabled: notSearch && noSelectedNotes },
|
|
{ title: 'Delete <kbd data-command="deleteNotes"></kbd>', command: "deleteNotes", uiIcon: "bx bx-trash",
|
|
enabled: isNotRoot && !isHoisted && parentNotSearch },
|
|
{ title: "----" },
|
|
{ title: 'Search in subtree <kbd data-command="searchInSubtree"></kbd>', command: "searchInSubtree", uiIcon: "bx bx-search",
|
|
enabled: notSearch && noSelectedNotes },
|
|
isHoisted ? null : { title: 'Hoist note <kbd data-command="toggleNoteHoisting"></kbd>', command: "toggleNoteHoisting", uiIcon: "bx bx-empty", enabled: noSelectedNotes && notSearch },
|
|
!isHoisted || !isNotRoot ? null : { title: 'Unhoist note <kbd data-command="toggleNoteHoisting"></kbd>', command: "toggleNoteHoisting", uiIcon: "bx bx-door-open" },
|
|
{ title: 'Edit branch prefix <kbd data-command="editBranchPrefix"></kbd>', command: "editBranchPrefix", uiIcon: "bx bx-empty",
|
|
enabled: isNotRoot && parentNotSearch && noSelectedNotes},
|
|
{ title: "Advanced", uiIcon: "bx bx-empty", enabled: true, items: [
|
|
{ title: 'Expand subtree <kbd data-command="expandSubtree"></kbd>', command: "expandSubtree", uiIcon: "bx bx-expand", enabled: noSelectedNotes },
|
|
{ title: 'Collapse subtree <kbd data-command="collapseSubtree"></kbd>', command: "collapseSubtree", uiIcon: "bx bx-collapse", enabled: noSelectedNotes },
|
|
{ title: "Force note sync", command: "forceNoteSync", uiIcon: "bx bx-refresh", enabled: noSelectedNotes },
|
|
{ title: 'Sort by ... <kbd data-command="sortChildNotes"></kbd>', command: "sortChildNotes", uiIcon: "bx bx-empty", enabled: noSelectedNotes && notSearch },
|
|
{ title: 'Recent changes in subtree', command: "recentChangesInSubtree", uiIcon: "bx bx-history", enabled: noSelectedNotes }
|
|
] },
|
|
{ title: "----" },
|
|
{ title: "Protect subtree", command: "protectSubtree", uiIcon: "bx bx-check-shield", enabled: noSelectedNotes },
|
|
{ title: "Unprotect subtree", command: "unprotectSubtree", uiIcon: "bx bx-shield", enabled: noSelectedNotes },
|
|
{ title: "----" },
|
|
{ title: 'Copy / clone <kbd data-command="copyNotesToClipboard"></kbd>', command: "copyNotesToClipboard", uiIcon: "bx bx-copy",
|
|
enabled: isNotRoot && !isHoisted },
|
|
{ title: 'Clone to ... <kbd data-command="cloneNotesTo"></kbd>', command: "cloneNotesTo", uiIcon: "bx bx-empty",
|
|
enabled: isNotRoot && !isHoisted },
|
|
{ title: 'Cut <kbd data-command="cutNotesToClipboard"></kbd>', command: "cutNotesToClipboard", uiIcon: "bx bx-cut",
|
|
enabled: isNotRoot && !isHoisted && parentNotSearch },
|
|
{ title: 'Move to ... <kbd data-command="moveNotesTo"></kbd>', command: "moveNotesTo", uiIcon: "bx bx-empty",
|
|
enabled: isNotRoot && !isHoisted && parentNotSearch },
|
|
{ title: 'Paste into <kbd data-command="pasteNotesFromClipboard"></kbd>', command: "pasteNotesFromClipboard", uiIcon: "bx bx-paste",
|
|
enabled: !clipboard.isClipboardEmpty() && notSearch && noSelectedNotes },
|
|
{ title: 'Paste after', command: "pasteNotesAfterFromClipboard", uiIcon: "bx bx-paste",
|
|
enabled: !clipboard.isClipboardEmpty() && isNotRoot && !isHoisted && parentNotSearch && noSelectedNotes },
|
|
{ title: `Duplicate subtree <kbd data-command="duplicateSubtree">`, command: "duplicateSubtree", uiIcon: "bx bx-empty",
|
|
enabled: parentNotSearch && isNotRoot && !isHoisted },
|
|
{ title: "----" },
|
|
{ title: "Export", command: "exportNote", uiIcon: "bx bx-empty",
|
|
enabled: notSearch && noSelectedNotes },
|
|
{ title: "Import into note", command: "importIntoNote", uiIcon: "bx bx-empty",
|
|
enabled: notSearch && noSelectedNotes },
|
|
{ title: "Bulk assign attributes", command: "bulkAssignAttributes", uiIcon: "bx bx-empty",
|
|
enabled: true }
|
|
].filter(row => row !== null);
|
|
}
|
|
|
|
async selectMenuItemHandler({command, type}) {
|
|
const notePath = treeService.getNotePath(this.node);
|
|
|
|
if (command === 'openInTab') {
|
|
appContext.tabManager.openTabWithNoteWithHoisting(notePath);
|
|
}
|
|
else if (command === "insertNoteAfter") {
|
|
const parentNotePath = treeService.getNotePath(this.node.getParent());
|
|
const isProtected = await treeService.getParentProtectedStatus(this.node);
|
|
|
|
noteCreateService.createNote(parentNotePath, {
|
|
target: 'after',
|
|
targetBranchId: this.node.data.branchId,
|
|
type: type,
|
|
isProtected: isProtected
|
|
});
|
|
}
|
|
else if (command === "insertChildNote") {
|
|
const parentNotePath = treeService.getNotePath(this.node);
|
|
|
|
noteCreateService.createNote(parentNotePath, {
|
|
type: type,
|
|
isProtected: this.node.data.isProtected
|
|
});
|
|
}
|
|
else if (command === 'openNoteInSplit') {
|
|
const subContexts = appContext.tabManager.getActiveContext().getSubContexts();
|
|
const {ntxId} = subContexts[subContexts.length - 1];
|
|
|
|
this.treeWidget.triggerCommand("openNewNoteSplit", {ntxId, notePath});
|
|
}
|
|
else {
|
|
this.treeWidget.triggerCommand(command, {node: this.node, notePath: notePath});
|
|
}
|
|
}
|
|
}
|
|
|
|
export default TreeContextMenu;
|