const tags = { 1: "", 2: "", 3: "", 4: "", 5: "", 6: "", 9: "", 10: "" }; let noteChangeDisabled = false; let isNoteChanged = false; function noteChanged() { if (noteChangeDisabled) { return; } isNoteChanged = true; } async function saveNoteIfChanged() { if (!isNoteChanged) { return; } const note = glob.currentNote; updateNoteFromInputs(note); encryptNoteIfNecessary(note); await saveNoteToServer(note); } setInterval(saveNoteIfChanged, 5000); $(document).ready(() => { $("#note-title").on('input', () => { noteChanged(); }); $('#note-detail').summernote({ airMode: true, height: 300, callbacks: { onChange: noteChanged } }); // so that tab jumps from note title (which has tabindex 1) $(".note-editable").attr("tabindex", 2); }); function parseHtml(contents, note) { note.links = []; note.images = []; note.detail.note_text = contents; if (!note.detail.encryption) { const linkRegexp = /]+?href="[^"]*kapp#([A-Za-z0-9]{22})"[^>]*?>[^<]+?<\/a>/g; let match; while (match = linkRegexp.exec(contents)) { console.log("adding link for " + match[1]); note.links.push({ note_id: note.detail.note_id, target_note_id: match[1] }); } } } function updateNoteFromInputs(note) { let contents = $('#note-detail').summernote('code'); parseHtml(contents, note); let title = $('#note-title').val(); getNodeByKey(note.detail.note_id).setTitle(title); note.detail.note_title = title; } async function saveNoteToServer(note) { await $.ajax({ url: baseApiUrl + 'notes/' + note.detail.note_id, type: 'PUT', data: JSON.stringify(note), contentType: "application/json", error: () => { error("Error saving the note!"); } }); isNoteChanged = false; message("Saved!"); } glob.currentNote = null; glob.currentNoteLoadTime = null; function createNewTopLevelNote() { let rootNode = glob.tree.fancytree("getRootNode"); createNote(rootNode, "root", "into"); } let newNoteCreated = false; async function createNote(node, parentKey, target, encryption) { // if encryption isn't available (user didn't enter password yet), then note is created as unencrypted // but this is quite weird since user doesn't see where the note is being created so it shouldn't occur often if (!encryption || !isEncryptionAvailable()) { encryption = 0; } const newNoteName = "new note"; const newNoteNameEncryptedIfNecessary = encryption > 0 ? encryptString(newNoteName) : newNoteName; const result = await $.ajax({ url: baseApiUrl + 'notes/' + parentKey + '/children' , type: 'POST', data: JSON.stringify({ note_title: newNoteNameEncryptedIfNecessary, target: target, target_note_id: node.key, encryption: encryption }), contentType: "application/json" }); const newNode = { title: newNoteName, key: result.note_id, note_id: result.note_id, encryption: encryption, extraClasses: encryption ? "encrypted" : "" }; glob.allNoteIds.push(result.note_id); newNoteCreated = true; if (target === 'after') { node.appendSibling(newNode).setActive(true); } else { node.addChildren(newNode).setActive(true); node.folder = true; node.renderTitle(); } message("Created!"); } function setTreeBasedOnEncryption(note) { const node = getNodeByKey(note.detail.note_id); node.toggleClass("encrypted", note.detail.encryption > 0); } function setNoteBackgroundIfEncrypted(note) { if (note.detail.encryption > 0) { $(".note-editable").addClass("encrypted"); $("#encrypt-button").hide(); $("#decrypt-button").show(); } else { $(".note-editable").removeClass("encrypted"); $("#encrypt-button").show(); $("#decrypt-button").hide(); } setTreeBasedOnEncryption(note); } async function loadNoteToEditor(noteId) { const note = await $.get(baseApiUrl + 'notes/' + noteId); glob.currentNote = note; glob.currentNoteLoadTime = Math.floor(new Date().getTime() / 1000); if (newNoteCreated) { newNoteCreated = false; $("#note-title").focus().select(); } await handleEncryption(note.detail.encryption > 0, false); $("#note-detail-wrapper").show(); // this may fal if the dialog has not been previously opened try { $("#encryption-password-dialog").dialog('close'); } catch(e) {} $("#encryption-password").val(''); decryptNoteIfNecessary(note); $("#note-title").val(note.detail.note_title); noteChangeDisabled = true; // Clear contents and remove all stored history. This is to prevent undo from going across notes $('#note-detail').summernote('reset'); $('#note-detail').summernote('code', note.detail.note_text); document.location.hash = noteId; recentNotes.addRecentNote(noteId, note.detail.note_id); noteChangeDisabled = false; setNoteBackgroundIfEncrypted(note); } async function loadNote(noteId) { const note = await $.get(baseApiUrl + 'notes/' + noteId); if (note.detail.encryption > 0 && !isEncryptionAvailable()) { return; } decryptNoteIfNecessary(note); return note; }