mirror of
				https://github.com/zadam/trilium.git
				synced 2025-11-04 13:39:01 +01:00 
			
		
		
		
	context menu with delete
This commit is contained in:
		
							parent
							
								
									28185af158
								
							
						
					
					
						commit
						d2f72529b3
					
				@ -32,3 +32,5 @@ class Attribute {
 | 
			
		||||
        return `Attribute(attributeId=${this.attributeId}, type=${this.type}, name=${this.name})`;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default Attribute;
 | 
			
		||||
@ -1,12 +1,12 @@
 | 
			
		||||
const $contextMenuContainer = $("#context-menu-container");
 | 
			
		||||
 | 
			
		||||
function initContextMenu(event, contextMenuItems, selectContextMenuItem) {
 | 
			
		||||
function initContextMenu(event, itemContainer, selectContextMenuItem) {
 | 
			
		||||
    event.stopPropagation();
 | 
			
		||||
 | 
			
		||||
    $contextMenuContainer.empty();
 | 
			
		||||
 | 
			
		||||
    for (const item of contextMenuItems) {
 | 
			
		||||
        if (item.hidden) {
 | 
			
		||||
            continue;
 | 
			
		||||
        }
 | 
			
		||||
    for (const item of itemContainer.getItems()) {
 | 
			
		||||
        console.log("Adding " + item.title);
 | 
			
		||||
 | 
			
		||||
        if (item.title === '----') {
 | 
			
		||||
            $contextMenuContainer.append($("<div>").addClass("dropdown-divider"));
 | 
			
		||||
 | 
			
		||||
@ -0,0 +1,28 @@
 | 
			
		||||
class ContextMenuItemsContainer {
 | 
			
		||||
    constructor(items) {
 | 
			
		||||
        // clone the item array and the items themselves
 | 
			
		||||
        this.items = items.map(item => Object.assign({}, item));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    hideItem(cmd, hidden = true) {
 | 
			
		||||
        if (hidden) {
 | 
			
		||||
            this.items = this.items.filter(item => item.cmd !== cmd);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    enableItem(cmd, enabled) {
 | 
			
		||||
        const item = this.items.find(item => item.cmd === cmd);
 | 
			
		||||
 | 
			
		||||
        if (!item) {
 | 
			
		||||
            throw new Error(`Command ${cmd} has not been found!`);
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        item.enabled = enabled;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    getItems() {
 | 
			
		||||
        return this.items;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
export default ContextMenuItemsContainer;
 | 
			
		||||
@ -3,6 +3,12 @@ import noteDetailService from "./note_detail.js";
 | 
			
		||||
import dragAndDropSetup from "./drag_and_drop.js";
 | 
			
		||||
import treeCache from "./tree_cache.js";
 | 
			
		||||
import treeBuilder from "./tree_builder.js";
 | 
			
		||||
import contextMenuWidget from "./context_menu.js";
 | 
			
		||||
import confirmDialog from "../dialogs/confirm.js";
 | 
			
		||||
import server from "./server.js";
 | 
			
		||||
import promptDialog from "../dialogs/prompt.js";
 | 
			
		||||
import ContextMenuItemsContainer from "./context_menu_items_container.js";
 | 
			
		||||
import treeChangesService from "./branches.js";
 | 
			
		||||
 | 
			
		||||
const $leftPane = $("#left-pane");
 | 
			
		||||
const $tree = $("#tree");
 | 
			
		||||
@ -69,7 +75,36 @@ async function showTree() {
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    });
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
$("#note-menu-button").click(async e => {
 | 
			
		||||
    const node = treeService.getCurrentNode();
 | 
			
		||||
    const branch = await treeCache.getBranch(node.data.branchId);
 | 
			
		||||
    const note = await treeCache.getNote(node.data.noteId);
 | 
			
		||||
    const parentNote = await treeCache.getNote(branch.parentNoteId);
 | 
			
		||||
    const isNotRoot = note.noteId !== 'root';
 | 
			
		||||
 | 
			
		||||
    const itemsContainer = new ContextMenuItemsContainer([
 | 
			
		||||
        {title: "Delete note", cmd: "delete", uiIcon: "trash"}
 | 
			
		||||
    ]);
 | 
			
		||||
 | 
			
		||||
    itemsContainer.enableItem("delete", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
 | 
			
		||||
    contextMenuWidget.initContextMenu(e, itemsContainer, noteContextMenuHandler);
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
async function noteContextMenuHandler(event, cmd) {
 | 
			
		||||
    const node = treeService.getCurrentNode();
 | 
			
		||||
 | 
			
		||||
    if (cmd === "delete") {
 | 
			
		||||
        treeChangesService.deleteNodes([node]);
 | 
			
		||||
 | 
			
		||||
        // move to the tree
 | 
			
		||||
        togglePanes();
 | 
			
		||||
    }
 | 
			
		||||
    else {
 | 
			
		||||
        throw new Error("Unrecognized command " + cmd);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
showTree();
 | 
			
		||||
@ -9,6 +9,7 @@ import attributeAutocompleteService from "./attribute_autocomplete.js";
 | 
			
		||||
import promptDialog from "../dialogs/prompt.js";
 | 
			
		||||
import infoDialog from "../dialogs/info.js";
 | 
			
		||||
import confirmDialog from "../dialogs/confirm.js";
 | 
			
		||||
import ContextMenuItemsContainer from "./context_menu_items_container.js";
 | 
			
		||||
 | 
			
		||||
const $component = $("#note-detail-relation-map");
 | 
			
		||||
const $relationMapContainer = $("#relation-map-container");
 | 
			
		||||
@ -304,9 +305,9 @@ function connectionContextMenuHandler(connection, event) {
 | 
			
		||||
    event.preventDefault();
 | 
			
		||||
    event.stopPropagation();
 | 
			
		||||
 | 
			
		||||
    const contextMenuItems = [ {title: "Remove relation", cmd: "remove", uiIcon: "trash"} ];
 | 
			
		||||
    const contextMenuItemsContainer = new ContextMenuItemsContainer([ {title: "Remove relation", cmd: "remove", uiIcon: "trash"} ]);
 | 
			
		||||
 | 
			
		||||
    contextMenuWidget.initContextMenu(event, contextMenuItems, async (event, cmd) => {
 | 
			
		||||
    contextMenuWidget.initContextMenu(event, contextMenuItemsContainer, async (event, cmd) => {
 | 
			
		||||
        if (cmd === 'remove') {
 | 
			
		||||
            if (!await confirmDialog.confirm("Are you sure you want to remove the relation?")) {
 | 
			
		||||
                return;
 | 
			
		||||
@ -380,12 +381,12 @@ async function connectionCreatedHandler(info, originalEvent) {
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
$relationMapContainer.on("contextmenu", ".note-box", e => {
 | 
			
		||||
    const contextMenuItems = [
 | 
			
		||||
    const contextMenuItemsContainer = new ContextMenuItemsContainer([
 | 
			
		||||
        {title: "Remove note", cmd: "remove", uiIcon: "trash"},
 | 
			
		||||
        {title: "Edit title", cmd: "edit-title", uiIcon: "pencil"},
 | 
			
		||||
    ];
 | 
			
		||||
    ]);
 | 
			
		||||
 | 
			
		||||
    contextMenuWidget.initContextMenu(e, contextMenuItems, noteContextMenuHandler);
 | 
			
		||||
    contextMenuWidget.initContextMenu(e, contextMenuItemsContainer, noteContextMenuHandler);
 | 
			
		||||
 | 
			
		||||
    return false;
 | 
			
		||||
});
 | 
			
		||||
 | 
			
		||||
@ -418,8 +418,8 @@ function initFancyTree(tree) {
 | 
			
		||||
    });
 | 
			
		||||
 | 
			
		||||
    $tree.on('contextmenu', '.fancytree-node', function(e) {
 | 
			
		||||
        treeContextMenuService.getContextMenuItems(e).then(contextMenuItems => {
 | 
			
		||||
            contextMenuWidget.initContextMenu(e, contextMenuItems, treeContextMenuService.selectContextMenuItem);
 | 
			
		||||
        treeContextMenuService.getContextMenuItems(e).then(contextMenuItemContainer => {
 | 
			
		||||
            contextMenuWidget.initContextMenu(e, contextMenuItemContainer, treeContextMenuService.selectContextMenuItem);
 | 
			
		||||
        });
 | 
			
		||||
 | 
			
		||||
        return false; // blocks default browser right click menu
 | 
			
		||||
 | 
			
		||||
@ -11,6 +11,7 @@ import infoService from "./info.js";
 | 
			
		||||
import treeCache from "./tree_cache.js";
 | 
			
		||||
import syncService from "./sync.js";
 | 
			
		||||
import hoistedNoteService from './hoisted_note.js';
 | 
			
		||||
import ContextMenuItemsContainer from './context_menu_items_container.js';
 | 
			
		||||
 | 
			
		||||
const $tree = $("#tree");
 | 
			
		||||
 | 
			
		||||
@ -81,10 +82,10 @@ function cut(nodes) {
 | 
			
		||||
const contextMenuItems = [
 | 
			
		||||
    {title: "Insert note here <kbd>Ctrl+O</kbd>", cmd: "insertNoteHere", uiIcon: "plus"},
 | 
			
		||||
    {title: "Insert child note <kbd>Ctrl+P</kbd>", cmd: "insertChildNote", uiIcon: "plus"},
 | 
			
		||||
    {title: "Delete", cmd: "delete", uiIcon: "trash"},
 | 
			
		||||
    {title: "Delete <kbd>Delete</kbd>", cmd: "delete", uiIcon: "trash"},
 | 
			
		||||
    {title: "----"},
 | 
			
		||||
    {title: "Hoist note <kbd>CTRL-H</kbd>", cmd: "hoist", uiIcon: "arrow-up"},
 | 
			
		||||
    {title: "Unhoist note <kbd>CTRL-H</kbd>", cmd: "unhoist", uiIcon: "arrow-up"},
 | 
			
		||||
    {title: "Hoist note <kbd>Ctrl-H</kbd>", cmd: "hoist", uiIcon: "arrow-up"},
 | 
			
		||||
    {title: "Unhoist note <kbd>Ctrl-H</kbd>", cmd: "unhoist", uiIcon: "arrow-up"},
 | 
			
		||||
    {title: "Edit branch prefix <kbd>F2</kbd>", cmd: "editBranchPrefix", uiIcon: "pencil"},
 | 
			
		||||
    {title: "----"},
 | 
			
		||||
    {title: "Protect subtree", cmd: "protectSubtree", uiIcon: "shield-check"},
 | 
			
		||||
@ -103,26 +104,6 @@ const contextMenuItems = [
 | 
			
		||||
    {title: "Sort alphabetically <kbd>Alt+S</kbd>", cmd: "sortAlphabetically", uiIcon: "arrows-v"}
 | 
			
		||||
];
 | 
			
		||||
 | 
			
		||||
function hideItem(cmd, hidden) {
 | 
			
		||||
    const item = contextMenuItems.find(item => item.cmd === cmd);
 | 
			
		||||
 | 
			
		||||
    if (!item) {
 | 
			
		||||
        throw new Error(`Command ${cmd} has not been found!`);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    item.hidden = hidden;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function enableItem(cmd, enabled) {
 | 
			
		||||
    const item = contextMenuItems.find(item => item.cmd === cmd);
 | 
			
		||||
    
 | 
			
		||||
    if (!item) {
 | 
			
		||||
        throw new Error(`Command ${cmd} has not been found!`);
 | 
			
		||||
    }
 | 
			
		||||
    
 | 
			
		||||
    item.enabled = enabled;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
async function getContextMenuItems(event) {
 | 
			
		||||
    const node = $.ui.fancytree.getNode(event);
 | 
			
		||||
    const branch = await treeCache.getBranch(node.data.branchId);
 | 
			
		||||
@ -130,22 +111,24 @@ async function getContextMenuItems(event) {
 | 
			
		||||
    const parentNote = await treeCache.getNote(branch.parentNoteId);
 | 
			
		||||
    const isNotRoot = note.noteId !== 'root';
 | 
			
		||||
 | 
			
		||||
    const itemsContainer = new ContextMenuItemsContainer(contextMenuItems);
 | 
			
		||||
 | 
			
		||||
    // Modify menu entries depending on node status
 | 
			
		||||
    enableItem("insertNoteHere", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    enableItem("insertChildNote", note.type !== 'search');
 | 
			
		||||
    enableItem("delete", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    enableItem("copy", isNotRoot);
 | 
			
		||||
    enableItem("cut", isNotRoot);
 | 
			
		||||
    enableItem("pasteAfter", clipboardIds.length > 0 && isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    enableItem("pasteInto", clipboardIds.length > 0 && note.type !== 'search');
 | 
			
		||||
    enableItem("importIntoNote", note.type !== 'search');
 | 
			
		||||
    enableItem("export", note.type !== 'search');
 | 
			
		||||
    enableItem("editBranchPrefix", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("insertNoteHere", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("insertChildNote", note.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("delete", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("copy", isNotRoot);
 | 
			
		||||
    itemsContainer.enableItem("cut", isNotRoot);
 | 
			
		||||
    itemsContainer.enableItem("pasteAfter", clipboardIds.length > 0 && isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("pasteInto", clipboardIds.length > 0 && note.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("importIntoNote", note.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("export", note.type !== 'search');
 | 
			
		||||
    itemsContainer.enableItem("editBranchPrefix", isNotRoot && parentNote.type !== 'search');
 | 
			
		||||
 | 
			
		||||
    const hoistedNoteId = await hoistedNoteService.getHoistedNoteId();
 | 
			
		||||
 | 
			
		||||
    hideItem("hoist", note.noteId === hoistedNoteId);
 | 
			
		||||
    hideItem("unhoist", note.noteId !== hoistedNoteId || !isNotRoot);
 | 
			
		||||
    itemsContainer.hideItem("hoist", note.noteId === hoistedNoteId);
 | 
			
		||||
    itemsContainer.hideItem("unhoist", note.noteId !== hoistedNoteId || !isNotRoot);
 | 
			
		||||
 | 
			
		||||
    // Activate node on right-click
 | 
			
		||||
    node.setActive();
 | 
			
		||||
@ -156,7 +139,7 @@ async function getContextMenuItems(event) {
 | 
			
		||||
    node.setSelected(true);
 | 
			
		||||
    treeService.clearSelectedNodes();
 | 
			
		||||
 | 
			
		||||
    return contextMenuItems;
 | 
			
		||||
    return itemsContainer;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
function selectContextMenuItem(event, cmd) {
 | 
			
		||||
 | 
			
		||||
@ -81,3 +81,7 @@ html, body {
 | 
			
		||||
kbd {
 | 
			
		||||
    display: none;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
.dropdown-menu {
 | 
			
		||||
    font-size: larger;
 | 
			
		||||
}
 | 
			
		||||
@ -65,7 +65,7 @@
 | 
			
		||||
        <div id="global-buttons">
 | 
			
		||||
            <a id="create-top-level-note-button" title="Create new top level note" class="icon-action jam jam-plus-circle"></a>
 | 
			
		||||
 | 
			
		||||
            <a id="collapse-tree-button" title="Collapse note tree. Shortcut ALT+C" class="icon-action jam jam-align-justify"></a>
 | 
			
		||||
            <a id="collapse-tree-button" title="Collapse note tree. Shortcut ALT+C" class="icon-action jam jam-layers"></a>
 | 
			
		||||
 | 
			
		||||
            <a id="scroll-to-current-note-button" title="Scroll to current note. Shortcut CTRL+." class="icon-action jam jam-download"></a>
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
@ -11,7 +11,7 @@
 | 
			
		||||
        <div id="global-buttons">
 | 
			
		||||
            <a id="create-top-level-note-button" title="Create new top level note" class="icon-action jam jam-plus-circle"></a>
 | 
			
		||||
 | 
			
		||||
            <a id="collapse-tree-button" title="Collapse note tree. Shortcut ALT+C" class="icon-action jam jam-align-justify"></a>
 | 
			
		||||
            <a id="collapse-tree-button" title="Collapse note tree. Shortcut ALT+C" class="icon-action jam jam-layers"></a>
 | 
			
		||||
 | 
			
		||||
            <a id="scroll-to-current-note-button" title="Scroll to current note. Shortcut CTRL+." class="icon-action jam jam-download"></a>
 | 
			
		||||
        </div>
 | 
			
		||||
@ -52,6 +52,8 @@
 | 
			
		||||
    </div>
 | 
			
		||||
 | 
			
		||||
    <% include dialogs/protected_session_password.ejs %>
 | 
			
		||||
 | 
			
		||||
    <div class="dropdown-menu dropdown-menu-sm" id="context-menu-container"></div>
 | 
			
		||||
</div>
 | 
			
		||||
 | 
			
		||||
<script type="text/javascript">
 | 
			
		||||
 | 
			
		||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user