mirror of
				https://github.com/zadam/trilium.git
				synced 2025-11-03 11:56:01 +01: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>
 |