| 
									
										
										
										
											2017-11-04 19:38:50 -04:00
										 |  |  | "use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | const noteTree = (function() { | 
					
						
							|  |  |  |     const noteDetailEl = $('#note-detail'); | 
					
						
							|  |  |  |     const treeEl = $("#tree"); | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |     let startNoteId = null; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |     let treeLoadTime = null; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:49:26 -04:00
										 |  |  |     let clipboardNoteId = null; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function getTreeLoadTime() { | 
					
						
							|  |  |  |         return treeLoadTime; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:49:26 -04:00
										 |  |  |     function getClipboardNoteId() { | 
					
						
							|  |  |  |         return clipboardNoteId; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function setClipboardNoteId(cbNoteId) { | 
					
						
							|  |  |  |         clipboardNoteId = cbNoteId; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |     function prepareNoteTree(notes) { | 
					
						
							|  |  |  |         for (const note of notes) { | 
					
						
							|  |  |  |             glob.allNoteIds.push(note.note_id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-10 22:55:19 -05:00
										 |  |  |             note.title = note.note_title; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-14 21:54:12 -05:00
										 |  |  |             if (note.is_protected) { | 
					
						
							| 
									
										
										
										
											2017-11-14 22:21:56 -05:00
										 |  |  |                 note.extraClasses = "protected"; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 if (note.is_clone) { | 
					
						
							|  |  |  |                     note.title += " (clone)"; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             note.key = note.note_id; | 
					
						
							|  |  |  |             note.expanded = note.is_expanded; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (note.children && note.children.length > 0) { | 
					
						
							|  |  |  |                 prepareNoteTree(note.children); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function setExpandedToServer(note_id, is_expanded) { | 
					
						
							| 
									
										
										
										
											2017-11-04 19:38:50 -04:00
										 |  |  |         const expandedNum = is_expanded ? 1 : 0; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $.ajax({ | 
					
						
							| 
									
										
										
										
											2017-11-04 19:38:50 -04:00
										 |  |  |             url: baseApiUrl + 'notes/' + note_id + '/expanded/' + expandedNum, | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             type: 'PUT', | 
					
						
							|  |  |  |             contentType: "application/json", | 
					
						
							|  |  |  |             success: result => {} | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |     function initFancyTree(notes) { | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |         const keybindings = { | 
					
						
							|  |  |  |             "insert": node => { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:18:36 -04:00
										 |  |  |                 const parentKey = treeUtils.getParentKey(node); | 
					
						
							| 
									
										
										
										
											2017-11-14 23:01:23 -05:00
										 |  |  |                 const isProtected = treeUtils.getParentProtectedStatus(node); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-14 23:01:23 -05:00
										 |  |  |                 noteEditor.createNote(node, parentKey, 'after', isProtected); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             }, | 
					
						
							|  |  |  |             "ctrl+insert": node => { | 
					
						
							| 
									
										
										
										
											2017-11-14 21:54:12 -05:00
										 |  |  |                 noteEditor.createNote(node, node.key, 'into', node.data.is_protected); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             }, | 
					
						
							|  |  |  |             "del": node => { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:10:41 -04:00
										 |  |  |                 treeChanges.deleteNode(node); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             }, | 
					
						
							|  |  |  |             "shift+up": node => { | 
					
						
							|  |  |  |                 const beforeNode = node.getPrevSibling(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (beforeNode !== null) { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:10:41 -04:00
										 |  |  |                     treeChanges.moveBeforeNode(node, beforeNode); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |                 } | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             "shift+down": node => { | 
					
						
							|  |  |  |                 let afterNode = node.getNextSibling(); | 
					
						
							|  |  |  |                 if (afterNode !== null) { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:10:41 -04:00
										 |  |  |                     treeChanges.moveAfterNode(node, afterNode); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |                 } | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             "shift+left": node => { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:10:41 -04:00
										 |  |  |                 treeChanges.moveNodeUp(node); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             }, | 
					
						
							|  |  |  |             "shift+right": node => { | 
					
						
							|  |  |  |                 let toNode = node.getPrevSibling(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (toNode !== null) { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:10:41 -04:00
										 |  |  |                     treeChanges.moveToNode(node, toNode); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |                 } | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             "return": node => { | 
					
						
							|  |  |  |                 // doesn't work :-/
 | 
					
						
							|  |  |  |                 noteDetailEl.summernote('focus'); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         treeEl.fancytree({ | 
					
						
							|  |  |  |             autoScroll: true, | 
					
						
							|  |  |  |             extensions: ["hotkeys", "filter", "dnd"], | 
					
						
							|  |  |  |             source: notes, | 
					
						
							|  |  |  |             scrollParent: $("#tree"), | 
					
						
							|  |  |  |             activate: (event, data) => { | 
					
						
							|  |  |  |                 const node = data.node.data; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 10:06:49 -05:00
										 |  |  |                 noteEditor.switchToNote(node.note_id); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             }, | 
					
						
							|  |  |  |             expand: (event, data) => { | 
					
						
							|  |  |  |                 setExpandedToServer(data.node.key, true); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             collapse: (event, data) => { | 
					
						
							|  |  |  |                 setExpandedToServer(data.node.key, false); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             init: (event, data) => { | 
					
						
							|  |  |  |                 if (startNoteId) { | 
					
						
							|  |  |  |                     data.tree.activateKey(startNoteId); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             hotkeys: { | 
					
						
							|  |  |  |                 keydown: keybindings | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             filter: { | 
					
						
							|  |  |  |                 autoApply: true,   // Re-apply last filter if lazy data is loaded
 | 
					
						
							|  |  |  |                 autoExpand: true, // Expand all branches that contain matches while filtered
 | 
					
						
							|  |  |  |                 counter: false,     // Show a badge with number of matching child nodes near parent icons
 | 
					
						
							|  |  |  |                 fuzzy: false,      // Match single characters in order, e.g. 'fb' will match 'FooBar'
 | 
					
						
							|  |  |  |                 hideExpandedCounter: true,  // Hide counter badge if parent is expanded
 | 
					
						
							|  |  |  |                 hideExpanders: false,       // Hide expanders if all child nodes are hidden by filter
 | 
					
						
							|  |  |  |                 highlight: true,   // Highlight matches by wrapping inside <mark> tags
 | 
					
						
							|  |  |  |                 leavesOnly: false, // Match end nodes only
 | 
					
						
							|  |  |  |                 nodata: true,      // Display a 'no data' status node if result is empty
 | 
					
						
							|  |  |  |                 mode: "hide"       // Grayout unmatched nodes (pass "hide" to remove unmatched node instead)
 | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             dnd: dragAndDropSetup, | 
					
						
							|  |  |  |             keydown: (event, data) => { | 
					
						
							|  |  |  |                 const node = data.node; | 
					
						
							|  |  |  |                 // Eat keyboard events, when a menu is open
 | 
					
						
							|  |  |  |                 if ($(".contextMenu:visible").length > 0) | 
					
						
							|  |  |  |                     return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 switch (event.which) { | 
					
						
							|  |  |  |                     // Open context menu on [Space] key (simulate right click)
 | 
					
						
							|  |  |  |                     case 32: // [Space]
 | 
					
						
							|  |  |  |                         $(node.span).trigger("mousedown", { | 
					
						
							|  |  |  |                             preventDefault: true, | 
					
						
							|  |  |  |                             button: 2 | 
					
						
							|  |  |  |                         }) | 
					
						
							|  |  |  |                             .trigger("mouseup", { | 
					
						
							|  |  |  |                                 preventDefault: true, | 
					
						
							|  |  |  |                                 pageX: node.span.offsetLeft, | 
					
						
							|  |  |  |                                 pageY: node.span.offsetTop, | 
					
						
							|  |  |  |                                 button: 2 | 
					
						
							|  |  |  |                             }); | 
					
						
							|  |  |  |                         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     // Handle Ctrl-C, -X and -V
 | 
					
						
							|  |  |  |                     // case 67:
 | 
					
						
							|  |  |  |                     //     if (event.ctrlKey) { // Ctrl-C
 | 
					
						
							|  |  |  |                     //         copyPaste("copy", node);
 | 
					
						
							|  |  |  |                     //         return false;
 | 
					
						
							|  |  |  |                     //     }
 | 
					
						
							|  |  |  |                     //     break;
 | 
					
						
							|  |  |  |                     case 86: | 
					
						
							|  |  |  |                         console.log("CTRL-V"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (event.ctrlKey) { // Ctrl-V
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:33:39 -04:00
										 |  |  |                             contextMenu.pasteAfter(node); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |                             return false; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                     case 88: | 
					
						
							|  |  |  |                         console.log("CTRL-X"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         if (event.ctrlKey) { // Ctrl-X
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:33:39 -04:00
										 |  |  |                             contextMenu.cut(node); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |                             return false; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:33:39 -04:00
										 |  |  |         treeEl.contextmenu(contextMenu.contextMenuSettings); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |     async function reload() { | 
					
						
							|  |  |  |         const treeResp = await loadTree(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // this will also reload the note content
 | 
					
						
							|  |  |  |         await treeEl.fancytree('getTree').reload(treeResp.notes); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |     function loadTree() { | 
					
						
							|  |  |  |         return $.get(baseApiUrl + 'tree').then(resp => { | 
					
						
							|  |  |  |             const notes = resp.notes; | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |             startNoteId = resp.start_note_id; | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |             treeLoadTime = resp.tree_load_time; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (document.location.hash) { | 
					
						
							|  |  |  |                 startNoteId = document.location.hash.substr(1); // strip initial #
 | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             prepareNoteTree(notes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return { | 
					
						
							|  |  |  |                 notes: notes, | 
					
						
							|  |  |  |                 startNoteId: startNoteId | 
					
						
							|  |  |  |             }; | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |     $(() => loadTree().then(resp => initFancyTree(resp.notes))); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     function collapseTree() { | 
					
						
							|  |  |  |         treeEl.fancytree("getRootNode").visit(node => { | 
					
						
							|  |  |  |             node.setExpanded(false); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     $(document).bind('keydown', 'alt+c', collapseTree); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function scrollToCurrentNote() { | 
					
						
							| 
									
										
										
										
											2017-11-04 22:18:36 -04:00
										 |  |  |         const node = treeUtils.getNodeByKey(noteEditor.getCurrentNoteId()); | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (node) { | 
					
						
							|  |  |  |             node.makeVisible({scrollIntoView: true}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             node.setFocus(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function showSearch() { | 
					
						
							|  |  |  |         $("#search-box").show(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $("input[name=search]").focus(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function toggleSearch() { | 
					
						
							|  |  |  |         if ($("#search-box:hidden").length) { | 
					
						
							|  |  |  |             showSearch(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             resetSearch(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             $("#search-box").hide(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     function resetSearch() { | 
					
						
							|  |  |  |         $("input[name=search]").val(""); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const tree = treeEl.fancytree("getTree"); | 
					
						
							|  |  |  |         tree.clearFilter(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     $("button#reset-search-button").click(resetSearch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     $("input[name=search]").keyup(e => { | 
					
						
							|  |  |  |         const searchString = $("input[name=search]").val(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (e && e.which === $.ui.keyCode.ESCAPE || $.trim(searchString) === "") { | 
					
						
							|  |  |  |             $("button#reset-search-button").click(); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (e && e.which === $.ui.keyCode.ENTER) { | 
					
						
							|  |  |  |             $.get(baseApiUrl + 'notes?search=' + searchString).then(resp => { | 
					
						
							|  |  |  |                 console.log("search: ", resp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 // Pass a string to perform case insensitive matching
 | 
					
						
							|  |  |  |                 const tree = treeEl.fancytree("getTree"); | 
					
						
							|  |  |  |                 tree.filterBranches(node => { | 
					
						
							|  |  |  |                     return resp.includes(node.data.note_id); | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     }).focus(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     $(document).bind('keydown', 'alt+s', showSearch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |         getTreeLoadTime, | 
					
						
							| 
									
										
										
										
											2017-11-04 19:49:26 -04:00
										 |  |  |         getClipboardNoteId, | 
					
						
							|  |  |  |         setClipboardNoteId, | 
					
						
							| 
									
										
										
										
											2017-11-05 09:52:28 -05:00
										 |  |  |         reload, | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |         collapseTree, | 
					
						
							|  |  |  |         scrollToCurrentNote, | 
					
						
							| 
									
										
										
										
											2017-11-04 19:49:26 -04:00
										 |  |  |         toggleSearch, | 
					
						
							| 
									
										
										
										
											2017-11-04 19:28:49 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | })(); |