| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  | <!DOCTYPE html> | 
					
						
							|  |  |  | <html lang="en"> | 
					
						
							|  |  |  | <head> | 
					
						
							|  |  |  |     <meta charset="utf-8"> | 
					
						
							|  |  |  |     <title>JSDoc: Source: becca/entities/branch.js</title> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     <script src="scripts/prettify/prettify.js"> </script> | 
					
						
							|  |  |  |     <script src="scripts/prettify/lang-css.js"> </script> | 
					
						
							|  |  |  |     <!--[if lt IE 9]>
 | 
					
						
							|  |  |  |       <script src="//html5shiv.googlecode.com/svn/trunk/html5.js"></script> | 
					
						
							|  |  |  |     <![endif]--> | 
					
						
							|  |  |  |     <link type="text/css" rel="stylesheet" href="styles/prettify-tomorrow.css"> | 
					
						
							|  |  |  |     <link type="text/css" rel="stylesheet" href="styles/jsdoc-default.css"> | 
					
						
							|  |  |  | </head> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <body> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <div id="main"> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     <h1 class="page-title">Source: becca/entities/branch.js</h1> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     <section> | 
					
						
							|  |  |  |         <article> | 
					
						
							|  |  |  |             <pre class="prettyprint source linenums"><code>"use strict"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-01-10 19:54:38 +01:00
										 |  |  | const Note = require('./note'); | 
					
						
							|  |  |  | const AbstractEntity = require("./abstract_entity"); | 
					
						
							|  |  |  | const sql = require("../../services/sql"); | 
					
						
							|  |  |  | const dateUtils = require("../../services/date_utils"); | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  | const utils = require("../../services/utils"); | 
					
						
							| 
									
										
										
										
											2022-06-19 14:06:00 +02:00
										 |  |  | const TaskContext = require("../../services/task_context"); | 
					
						
							|  |  |  | const cls = require("../../services/cls"); | 
					
						
							|  |  |  | const log = require("../../services/log"); | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Branch represents a relationship between a child note and its parent note. Trilium allows a note to have multiple | 
					
						
							|  |  |  |  * parents. | 
					
						
							| 
									
										
										
										
											2022-04-16 00:17:32 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * @extends AbstractEntity | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  | class Branch extends AbstractEntity { | 
					
						
							|  |  |  |     static get entityName() { return "branches"; } | 
					
						
							|  |  |  |     static get primaryKeyName() { return "branchId"; } | 
					
						
							|  |  |  |     // notePosition is not part of hash because it would produce a lot of updates in case of reordering | 
					
						
							|  |  |  |     static get hashedProperties() { return ["branchId", "noteId", "parentNoteId", "prefix"]; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     constructor(row) { | 
					
						
							|  |  |  |         super(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!row) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.updateFromRow(row); | 
					
						
							|  |  |  |         this.init(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     updateFromRow(row) { | 
					
						
							|  |  |  |         this.update([ | 
					
						
							|  |  |  |             row.branchId, | 
					
						
							|  |  |  |             row.noteId, | 
					
						
							|  |  |  |             row.parentNoteId, | 
					
						
							|  |  |  |             row.prefix, | 
					
						
							|  |  |  |             row.notePosition, | 
					
						
							|  |  |  |             row.isExpanded, | 
					
						
							|  |  |  |             row.utcDateModified | 
					
						
							|  |  |  |         ]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     update([branchId, noteId, parentNoteId, prefix, notePosition, isExpanded, utcDateModified]) { | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {string} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.branchId = branchId; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {string} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.noteId = noteId; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {string} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.parentNoteId = parentNoteId; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {string} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.prefix = prefix; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {int} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.notePosition = notePosition; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {boolean} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.isExpanded = !!isExpanded; | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         /** @type {string} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         this.utcDateModified = utcDateModified; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     init() { | 
					
						
							| 
									
										
										
										
											2022-01-05 20:31:21 +01:00
										 |  |  |         if (this.branchId) { | 
					
						
							|  |  |  |             this.becca.branches[this.branchId] = this; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.becca.childParentToBranch[`${this.noteId}-${this.parentNoteId}`] = this; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-27 13:45:22 -07:00
										 |  |  |         const childNote = this.childNote; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!childNote.parentBranches.includes(this)) { | 
					
						
							|  |  |  |             childNote.parentBranches.push(this); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         if (this.branchId === 'root') { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const parentNote = this.parentNote; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |         if (!childNote.parents.includes(parentNote)) { | 
					
						
							|  |  |  |             childNote.parents.push(parentNote); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!parentNote.children.includes(childNote)) { | 
					
						
							|  |  |  |             parentNote.children.push(childNote); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |     /** @returns {Note} */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |     get childNote() { | 
					
						
							|  |  |  |         if (!(this.noteId in this.becca.notes)) { | 
					
						
							| 
									
										
										
										
											2022-01-05 20:31:21 +01:00
										 |  |  |             // entities can come out of order in sync/import, create skeleton which will be filled later | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |             this.becca.addNote(this.noteId, new Note({noteId: this.noteId})); | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return this.becca.notes[this.noteId]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     getNote() { | 
					
						
							|  |  |  |         return this.childNote; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-27 13:45:22 -07:00
										 |  |  |     /** @returns {Note|undefined} - root branch will have undefined parent, all other branches have to have a parent note */ | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |     get parentNote() { | 
					
						
							| 
									
										
										
										
											2022-08-27 13:45:22 -07:00
										 |  |  |         if (!(this.parentNoteId in this.becca.notes) && this.parentNoteId !== 'none') { | 
					
						
							| 
									
										
										
										
											2022-01-05 20:31:21 +01:00
										 |  |  |             // entities can come out of order in sync/import, create skeleton which will be filled later | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |             this.becca.addNote(this.parentNoteId, new Note({noteId: this.parentNoteId})); | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return this.becca.notes[this.parentNoteId]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-10 21:30:54 +01:00
										 |  |  |     get isDeleted() { | 
					
						
							|  |  |  |         return !(this.branchId in this.becca.branches); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Branch is weak when its existence should not hinder deletion of its note. | 
					
						
							|  |  |  |      * As a result, note with only weak branches should be immediately deleted. | 
					
						
							|  |  |  |      * An example is shared or bookmarked clones - they are created automatically and exist for technical reasons, | 
					
						
							|  |  |  |      * not as user-intended actions. From user perspective, they don't count as real clones and for the purpose | 
					
						
							|  |  |  |      * of deletion should not act as a clone. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @returns {boolean} | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     get isWeak() { | 
					
						
							| 
									
										
										
										
											2022-12-22 14:57:00 +01:00
										 |  |  |         return ['_share', 'lbBookmarks'].includes(this.parentNoteId); | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Delete a branch. If this is a last note's branch, delete the note as well. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @param {string} [deleteId] - optional delete identified | 
					
						
							|  |  |  |      * @param {TaskContext} [taskContext] | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @return {boolean} - true if note has been deleted, false otherwise | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     deleteBranch(deleteId, taskContext) { | 
					
						
							|  |  |  |         if (!deleteId) { | 
					
						
							|  |  |  |             deleteId = utils.randomString(10); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!taskContext) { | 
					
						
							|  |  |  |             taskContext = new TaskContext('no-progress-reporting'); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         taskContext.increaseProgressCount(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-19 14:06:00 +02:00
										 |  |  |         const note = this.getNote(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!taskContext.noteDeletionHandlerTriggered) { | 
					
						
							|  |  |  |             const parentBranches = note.getParentBranches(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (parentBranches.length === 1 && parentBranches[0] === this) { | 
					
						
							|  |  |  |                 // needs to be run before branches and attributes are deleted and thus attached relations disappear | 
					
						
							|  |  |  |                 const handlers = require("../../services/handlers"); | 
					
						
							|  |  |  |                 handlers.runAttachedRelations(note, 'runOnNoteDeletion', note); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  |         if (this.branchId === 'root' | 
					
						
							|  |  |  |             || this.noteId === 'root' | 
					
						
							|  |  |  |             || this.noteId === cls.getHoistedNoteId()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             throw new Error("Can't delete root or hoisted branch/note"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.markAsDeleted(deleteId); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  |         const notDeletedBranches = note.getStrongParentBranches(); | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (notDeletedBranches.length === 0) { | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  |             for (const weakBranch of note.getParentBranches()) { | 
					
						
							|  |  |  |                 weakBranch.markAsDeleted(deleteId); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  |             for (const childBranch of note.getChildBranches()) { | 
					
						
							|  |  |  |                 childBranch.deleteBranch(deleteId, taskContext); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // first delete children and then parent - this will show up better in recent changes | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-22 14:57:00 +01:00
										 |  |  |             log.info(`Deleting note ${note.noteId}`); | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-12-14 23:51:28 +01:00
										 |  |  |             this.becca.notes[note.noteId].isBeingDeleted = true; | 
					
						
							| 
									
										
										
										
											2022-11-07 21:26:13 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-19 23:06:46 +02:00
										 |  |  |             for (const attribute of note.getOwnedAttributes()) { | 
					
						
							|  |  |  |                 attribute.markAsDeleted(deleteId); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (const relation of note.getTargetRelations()) { | 
					
						
							|  |  |  |                 relation.markAsDeleted(deleteId); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             note.markAsDeleted(deleteId); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         else { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |     beforeSaving() { | 
					
						
							|  |  |  |         if (this.notePosition === undefined || this.notePosition === null) { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:57:51 +01:00
										 |  |  |             let maxNotePos = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             for (const childBranch of this.parentNote.getChildBranches()) { | 
					
						
							| 
									
										
										
										
											2022-12-22 14:57:00 +01:00
										 |  |  |                 if (maxNotePos < childBranch.notePosition && childBranch.branchId !== '_hidden') { | 
					
						
							| 
									
										
										
										
											2022-11-30 16:57:51 +01:00
										 |  |  |                     maxNotePos = childBranch.notePosition; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             this.notePosition = maxNotePos + 10; | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (!this.isExpanded) { | 
					
						
							|  |  |  |             this.isExpanded = false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.utcDateModified = dateUtils.utcNowDateTime(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         super.beforeSaving(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         this.becca.branches[this.branchId] = this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     getPojo() { | 
					
						
							|  |  |  |         return { | 
					
						
							|  |  |  |             branchId: this.branchId, | 
					
						
							|  |  |  |             noteId: this.noteId, | 
					
						
							|  |  |  |             parentNoteId: this.parentNoteId, | 
					
						
							|  |  |  |             prefix: this.prefix, | 
					
						
							|  |  |  |             notePosition: this.notePosition, | 
					
						
							|  |  |  |             isExpanded: this.isExpanded, | 
					
						
							|  |  |  |             isDeleted: false, | 
					
						
							| 
									
										
										
										
											2022-01-05 20:31:21 +01:00
										 |  |  |             utcDateModified: this.utcDateModified | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     createClone(parentNoteId, notePosition) { | 
					
						
							|  |  |  |         return new Branch({ | 
					
						
							|  |  |  |             noteId: this.noteId, | 
					
						
							|  |  |  |             parentNoteId: parentNoteId, | 
					
						
							|  |  |  |             notePosition: notePosition, | 
					
						
							|  |  |  |             prefix: this.prefix, | 
					
						
							|  |  |  |             isExpanded: this.isExpanded | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = Branch; | 
					
						
							|  |  |  | </code></pre> | 
					
						
							|  |  |  |         </article> | 
					
						
							|  |  |  |     </section> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | </div> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <nav> | 
					
						
							| 
									
										
										
										
											2022-04-16 00:17:32 +02:00
										 |  |  |     <h2><a href="index.html">Home</a></h2><h3>Modules</h3><ul><li><a href="module-sql.html">sql</a></li></ul><h3>Classes</h3><ul><li><a href="AbstractEntity.html">AbstractEntity</a></li><li><a href="Attribute.html">Attribute</a></li><li><a href="BackendScriptApi.html">BackendScriptApi</a></li><li><a href="Branch.html">Branch</a></li><li><a href="EtapiToken.html">EtapiToken</a></li><li><a href="Note.html">Note</a></li><li><a href="NoteRevision.html">NoteRevision</a></li><li><a href="Option.html">Option</a></li><li><a href="RecentNote.html">RecentNote</a></li></ul><h3><a href="global.html">Global</a></h3> | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  | </nav> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <br class="clear"> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <footer> | 
					
						
							| 
									
										
										
										
											2022-11-07 21:26:13 +01:00
										 |  |  |     Documentation generated by <a href="https://github.com/jsdoc/jsdoc">JSDoc 4.0.0</a> | 
					
						
							| 
									
										
										
										
											2021-09-30 12:26:13 +02:00
										 |  |  | </footer> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <script> prettyPrint(); </script> | 
					
						
							|  |  |  | <script src="scripts/linenumber.js"> </script> | 
					
						
							|  |  |  | </body> | 
					
						
							|  |  |  | </html> |