mirror of
				https://github.com/zadam/trilium.git
				synced 2025-10-31 10:26:08 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			237 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			237 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| const tags = {
 | |
|     1: "<b>",
 | |
|     2: "</b>",
 | |
|     3: "<i>",
 | |
|     4: "</i>",
 | |
|     5: "<u>",
 | |
|     6: "</u>",
 | |
|     9: "<s>",
 | |
|     10: "</s>"
 | |
| };
 | |
| 
 | |
| 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 = /<a[^>]+?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;
 | |
| } |