mirror of
				https://github.com/TriliumNext/Notes.git
				synced 2025-10-23 07:41:37 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			334 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
			
		
		
	
	
			334 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			HTML
		
	
	
	
	
	
| <!DOCTYPE html>
 | |
| <html lang="en">
 | |
| <head>
 | |
|     <meta charset="utf-8">
 | |
|     <title>JSDoc: Source: becca/entities/bbranch.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/bbranch.js</h1>
 | |
| 
 | |
|     
 | |
| 
 | |
| 
 | |
| 
 | |
|     
 | |
|     <section>
 | |
|         <article>
 | |
|             <pre class="prettyprint source linenums"><code>"use strict";
 | |
| 
 | |
| const BNote = require('./bnote');
 | |
| const AbstractBeccaEntity = require("./abstract_becca_entity");
 | |
| const dateUtils = require("../../services/date_utils");
 | |
| const utils = require("../../services/utils");
 | |
| const TaskContext = require("../../services/task_context");
 | |
| const cls = require("../../services/cls");
 | |
| const log = require("../../services/log");
 | |
| 
 | |
| /**
 | |
|  * Branch represents a relationship between a child note and its parent note. Trilium allows a note to have multiple
 | |
|  * parents.
 | |
|  *
 | |
|  * Note that you should not rely on the branch's identity, since it can change easily with a note's move.
 | |
|  * Always check noteId instead.
 | |
|  *
 | |
|  * @extends AbstractBeccaEntity
 | |
|  */
 | |
| class BBranch extends AbstractBeccaEntity {
 | |
|     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]) {
 | |
|         /** @type {string} */
 | |
|         this.branchId = branchId;
 | |
|         /** @type {string} */
 | |
|         this.noteId = noteId;
 | |
|         /** @type {string} */
 | |
|         this.parentNoteId = parentNoteId;
 | |
|         /** @type {string|null} */
 | |
|         this.prefix = prefix;
 | |
|         /** @type {int} */
 | |
|         this.notePosition = notePosition;
 | |
|         /** @type {boolean} */
 | |
|         this.isExpanded = !!isExpanded;
 | |
|         /** @type {string} */
 | |
|         this.utcDateModified = utcDateModified;
 | |
| 
 | |
|         return this;
 | |
|     }
 | |
| 
 | |
|     init() {
 | |
|         if (this.branchId) {
 | |
|             this.becca.branches[this.branchId] = this;
 | |
|         }
 | |
| 
 | |
|         this.becca.childParentToBranch[`${this.noteId}-${this.parentNoteId}`] = this;
 | |
| 
 | |
|         const childNote = this.childNote;
 | |
| 
 | |
|         if (!childNote.parentBranches.includes(this)) {
 | |
|             childNote.parentBranches.push(this);
 | |
|         }
 | |
| 
 | |
|         if (this.noteId === 'root') {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         const parentNote = this.parentNote;
 | |
| 
 | |
|         if (!childNote.parents.includes(parentNote)) {
 | |
|             childNote.parents.push(parentNote);
 | |
|         }
 | |
| 
 | |
|         if (!parentNote.children.includes(childNote)) {
 | |
|             parentNote.children.push(childNote);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /** @returns {BNote} */
 | |
|     get childNote() {
 | |
|         if (!(this.noteId in this.becca.notes)) {
 | |
|             // entities can come out of order in sync/import, create skeleton which will be filled later
 | |
|             this.becca.addNote(this.noteId, new BNote({noteId: this.noteId}));
 | |
|         }
 | |
| 
 | |
|         return this.becca.notes[this.noteId];
 | |
|     }
 | |
| 
 | |
|     /** @returns {BNote} */
 | |
|     getNote() {
 | |
|         return this.childNote;
 | |
|     }
 | |
| 
 | |
|     /** @returns {BNote|undefined} - root branch will have undefined parent, all other branches have to have a parent note */
 | |
|     get parentNote() {
 | |
|         if (!(this.parentNoteId in this.becca.notes) && this.parentNoteId !== 'none') {
 | |
|             // entities can come out of order in sync/import, create skeleton which will be filled later
 | |
|             this.becca.addNote(this.parentNoteId, new BNote({noteId: this.parentNoteId}));
 | |
|         }
 | |
| 
 | |
|         return this.becca.notes[this.parentNoteId];
 | |
|     }
 | |
| 
 | |
|     get isDeleted() {
 | |
|         return !(this.branchId in this.becca.branches);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 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() {
 | |
|         return ['_share', '_lbBookmarks'].includes(this.parentNoteId);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * 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]
 | |
|      *
 | |
|      * @returns {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();
 | |
| 
 | |
|         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);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (this.noteId === 'root'
 | |
|             || this.noteId === cls.getHoistedNoteId()) {
 | |
| 
 | |
|             throw new Error("Can't delete root or hoisted branch/note");
 | |
|         }
 | |
| 
 | |
|         this.markAsDeleted(deleteId);
 | |
| 
 | |
|         const notDeletedBranches = note.getStrongParentBranches();
 | |
| 
 | |
|         if (notDeletedBranches.length === 0) {
 | |
|             for (const weakBranch of note.getParentBranches()) {
 | |
|                 weakBranch.markAsDeleted(deleteId);
 | |
|             }
 | |
| 
 | |
|             for (const childBranch of note.getChildBranches()) {
 | |
|                 childBranch.deleteBranch(deleteId, taskContext);
 | |
|             }
 | |
| 
 | |
|             // first delete children and then parent - this will show up better in recent changes
 | |
| 
 | |
|             log.info(`Deleting note '${note.noteId}'`);
 | |
| 
 | |
|             this.becca.notes[note.noteId].isBeingDeleted = true;
 | |
| 
 | |
|             for (const attribute of note.getOwnedAttributes().slice()) {
 | |
|                 attribute.markAsDeleted(deleteId);
 | |
|             }
 | |
| 
 | |
|             for (const relation of note.getTargetRelations()) {
 | |
|                 relation.markAsDeleted(deleteId);
 | |
|             }
 | |
| 
 | |
|             for (const attachment of note.getAttachments()) {
 | |
|                 attachment.markAsDeleted(deleteId);
 | |
|             }
 | |
| 
 | |
|             note.markAsDeleted(deleteId);
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
|         else {
 | |
|             return false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     beforeSaving() {
 | |
|         if (!this.noteId || !this.parentNoteId) {
 | |
|             throw new Error(`noteId and parentNoteId are mandatory properties for Branch`);
 | |
|         }
 | |
| 
 | |
|         this.branchId = `${this.parentNoteId}_${this.noteId}`;
 | |
| 
 | |
|         if (this.notePosition === undefined || this.notePosition === null) {
 | |
|             let maxNotePos = 0;
 | |
| 
 | |
|             for (const childBranch of this.parentNote.getChildBranches()) {
 | |
|                 if (maxNotePos < childBranch.notePosition
 | |
|                     && childBranch.noteId !== '_hidden' // hidden has a very large notePosition to always stay last
 | |
|                 ) {
 | |
|                     maxNotePos = childBranch.notePosition;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             this.notePosition = maxNotePos + 10;
 | |
|         }
 | |
| 
 | |
|         if (!this.isExpanded) {
 | |
|             this.isExpanded = false;
 | |
|         }
 | |
| 
 | |
|         if (!this.prefix?.trim()) {
 | |
|             this.prefix = null;
 | |
|         }
 | |
| 
 | |
|         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,
 | |
|             utcDateModified: this.utcDateModified
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     createClone(parentNoteId, notePosition) {
 | |
|         const existingBranch = this.becca.getBranchFromChildAndParent(this.noteId, parentNoteId);
 | |
| 
 | |
|         if (existingBranch) {
 | |
|             existingBranch.notePosition = notePosition;
 | |
|             return existingBranch;
 | |
|         } else {
 | |
|             return new BBranch({
 | |
|                 noteId: this.noteId,
 | |
|                 parentNoteId: parentNoteId,
 | |
|                 notePosition: notePosition,
 | |
|                 prefix: this.prefix,
 | |
|                 isExpanded: this.isExpanded
 | |
|             });
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| module.exports = BBranch;
 | |
| </code></pre>
 | |
|         </article>
 | |
|     </section>
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| </div>
 | |
| 
 | |
| <nav>
 | |
|     <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="AbstractBeccaEntity.html">AbstractBeccaEntity</a></li><li><a href="BAttachment.html">BAttachment</a></li><li><a href="BAttribute.html">BAttribute</a></li><li><a href="BBranch.html">BBranch</a></li><li><a href="BEtapiToken.html">BEtapiToken</a></li><li><a href="BNote.html">BNote</a></li><li><a href="BOption.html">BOption</a></li><li><a href="BRecentNote.html">BRecentNote</a></li><li><a href="BRevision.html">BRevision</a></li><li><a href="BackendScriptApi.html">BackendScriptApi</a></li></ul><h3>Global</h3><ul><li><a href="global.html#api">api</a></li></ul>
 | |
| </nav>
 | |
| 
 | |
| <br class="clear">
 | |
| 
 | |
| <footer>
 | |
|     Documentation generated by <a href="https://github.com/jsdoc/jsdoc">JSDoc 4.0.2</a>
 | |
| </footer>
 | |
| 
 | |
| <script> prettyPrint(); </script>
 | |
| <script src="scripts/linenumber.js"> </script>
 | |
| </body>
 | |
| </html>
 | 
