<!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>