2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								<!DOCTYPE html>  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< html  lang = "en" >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< head >  
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < meta  charset = "utf-8" > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < title > JSDoc: Source: becca/entities/note.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/note.js< / h1 > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    < section > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        < article > 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            < pre  class = "prettyprint source linenums" > < code > "use strict";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const protectedSessionService = require('../../services/protected_session');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const log = require('../../services/log');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const sql = require('../../services/sql');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const utils = require('../../services/utils');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const dateUtils = require('../../services/date_utils');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const entityChangesService = require('../../services/entity_changes');
							 
						 
					
						
							
								
									
										
										
										
											2022-01-10 19:54:38 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const AbstractEntity = require("./abstract_entity");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const NoteRevision = require("./note_revision");
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const TaskContext = require("../../services/task_context");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const dayjs = require("dayjs");
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								const utc = require('dayjs/plugin/utc');
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								dayjs.extend(utc)
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const LABEL = 'label';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								const RELATION = 'relation';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								/**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 * Trilium's main entity which can represent text note, image, code note, file attachment etc.
							 
						 
					
						
							
								
									
										
										
										
											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 Note extends AbstractEntity {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    static get entityName() { return "notes"; }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    static get primaryKeyName() { return "noteId"; }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    static get hashedProperties() { return ["noteId", "title", "isProtected", "type", "mime"]; }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    constructor(row) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        super();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!row) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.updateFromRow(row);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.init();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    updateFromRow(row) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.update([
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.title,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.mime,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.isProtected,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.dateCreated,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.dateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.utcDateCreated,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            row.utcDateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        ]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    update([noteId, title, type, mime, isProtected, dateCreated, dateModified, utcDateCreated, utcDateModified]) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // ------ Database persisted attributes ------
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											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.title = title;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {boolean} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.isProtected = !!isProtected;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.type = type;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.mime = mime;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.dateCreated = dateCreated || dateUtils.localNowDateTime();
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.dateModified = dateModified;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.utcDateCreated = utcDateCreated || dateUtils.utcNowDateTime();
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.utcDateModified = utcDateModified;
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {boolean} - set during the deletion operation, before it is completed (removed from becca completely) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.isBeingDeleted = false;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // ------ Derived attributes ------
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {boolean} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.isDecrypted = !this.noteId || !this.isProtected;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.decrypt();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {string|null} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.flatTextCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    init() {
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Branch[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.parentBranches = [];
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Note[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.parents = [];
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Note[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private*/
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.children = [];
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Attribute[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.ownedAttributes = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Attribute[]|null}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.__attributeCache = null;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Attribute[]|null}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private*/
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.inheritableAttributeCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Attribute[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private*/
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.targetRelations = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.becca.addNote(this.noteId, this);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /** @type {Note[]|null}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.ancestorCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        // following attributes are filled during searching from database
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * size of the content in bytes
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @type {int|null}
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         * @private
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.contentSize = null;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * size of the content and note revision contents in bytes
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @type {int|null}
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         * @private
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.noteSize = null;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * number of note revisions for this note
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								         * @type {int|null}
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         * @private
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								         */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.revisionCount = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isContentAvailable() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return !this.noteId // new note which was not encrypted yet
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            || !this.isProtected
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            || protectedSessionService.isProtectedSessionAvailable()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-10 19:54:38 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    getTitleOrProtected() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.isContentAvailable() ? this.title : '[protected]';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Branch[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getParentBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.parentBranches;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Returns < i>strong< /i> (as opposed to < i>weak< /i>) parent branches. See isWeak for details.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Branch[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getStrongParentBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getParentBranches().filter(branch => !branch.isWeak);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-05 20:31:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Branch[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @deprecated use getParentBranches() instead
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.parentBranches;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Note[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getParentNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.parents;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Note[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getChildNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.children;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {boolean} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    hasChildren() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.children & &  this.children.length > 0;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Branch[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getChildBranches() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.children.map(childNote => this.becca.getBranchFromChildAndParent(childNote.noteId, this.noteId));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /*
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Note content has quite special handling - it's not a separate entity, but a lazily loaded
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * part of Note entity with it's own sync. Reasons behind this hybrid design has been:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - content can be quite large and it's not necessary to load it / fill memory for any note access even if we don't need a content, especially for bulk operations like search
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - changes in the note metadata or title should not trigger note content sync (so we keep separate utcDateModified and entity changes records)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - but to the user note content and title changes are one and the same - single dateModified (so all changes must go through Note and content is not a separate entity)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {*} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getContent(silentNotFoundError = false) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const row = sql.getRow(`SELECT content FROM note_contents WHERE noteId = ?`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!row) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (silentNotFoundError) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return undefined;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                throw new Error(`Cannot find note content for noteId=${this.noteId}`);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        let content = row.content;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isProtected) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (protectedSessionService.isProtectedSessionAvailable()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                content = content === null ? null : protectedSessionService.decrypt(content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                content = "";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isStringNote()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return content === null
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                ? ""
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                : content.toString("UTF-8");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return content;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {{contentLength, dateModified, utcDateModified}} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getContentMetadata() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return sql.getRow(`
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            SELECT 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                LENGTH(content) AS contentLength, 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                dateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                utcDateModified 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            FROM note_contents 
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            WHERE noteId = ?`, [this.noteId]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    get dateCreatedObj() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.dateCreated === null ? null : dayjs(this.dateCreated);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get utcDateCreatedObj() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.utcDateCreated === null ? null : dayjs.utc(this.utcDateCreated);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get dateModifiedObj() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.dateModified === null ? null : dayjs(this.dateModified);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get utcDateModifiedObj() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.utcDateModified === null ? null : dayjs.utc(this.utcDateModified);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    /** @returns {*} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getJsonContent() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const content = this.getContent();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!content || !content.trim()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return JSON.parse(content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    setContent(content, ignoreMissingProtectedSession = false) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (content === null || content === undefined) {
							 
						 
					
						
							
								
									
										
										
										
											2022-05-03 23:25:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            throw new Error(`Cannot set null content to note '${this.noteId}'`);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isStringNote()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            content = content.toString();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            content = Buffer.isBuffer(content) ? content : Buffer.from(content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const pojo = {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            content: content,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            dateModified: dateUtils.localNowDateTime(),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateModified: dateUtils.utcNowDateTime()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        };
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isProtected) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (protectedSessionService.isProtectedSessionAvailable()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                pojo.content = protectedSessionService.encrypt(pojo.content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else if (!ignoreMissingProtectedSession) {
							 
						 
					
						
							
								
									
										
										
										
											2022-05-03 23:25:28 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                throw new Error(`Cannot update content of noteId '${this.noteId}' since we're out of protected session.`);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        sql.upsert("note_contents", "noteId", pojo);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        const hash = utils.hash(`${this.noteId}|${pojo.content.toString()}`);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        entityChangesService.addEntityChange({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            entityName: 'note_contents',
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            entityId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            hash: hash,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            isErased: false,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateChanged: pojo.utcDateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            isSynced: true
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        });
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    setJsonContent(content) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.setContent(JSON.stringify(content, null, '\t'));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is the root of the note tree. Root note has "root" noteId */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isRoot() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.noteId === 'root';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is of application/json content type */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isJson() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.mime === "application/json";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is JavaScript (code or attachment) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isJavaScript() {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-17 21:46:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return (this.type === "code" || this.type === "file" || this.type === 'launcher')
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            & &  (this.mime.startsWith("application/javascript")
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                || this.mime === "application/x-javascript"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                || this.mime === "text/javascript");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if this note is HTML */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isHtml() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return ["code", "file", "render"].includes(this.type)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            & &  this.mime === "text/html";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {boolean} true if the note has string content (not binary) */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isStringNote() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return utils.isStringNote(this.type, this.mime);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /** @returns {string|null} JS script environment - either "frontend" or "backend" */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getScriptEnv() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isHtml() || (this.isJavaScript() & &  this.mime.endsWith('env=frontend'))) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "frontend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.type === 'render') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "frontend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isJavaScript() & &  this.mime.endsWith('env=backend')) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return "backend";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [type] - (optional) attribute type to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - (optional) attribute name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} all note's attributes, including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getAttributes(type, name) {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-17 21:46:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.__validateTypeName(type, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.__getAttributes([]);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (type & &  name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache.filter(attr => attr.type === type & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else if (type) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache.filter(attr => attr.type === type);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else if (name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache.filter(attr => attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            // a bit unsafe to return the original array, but defensive copy would be costly
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.__attributeCache;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @private */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    __getAttributes(path) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (path.includes(this.noteId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.__attributeCache) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const parentAttributes = this.ownedAttributes.slice();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const newPath = [...path, this.noteId];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (this.noteId !== 'root') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                for (const parentNote of this.parents) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    parentAttributes.push(...parentNote.__getInheritableAttributes(newPath));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const templateAttributes = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const ownedAttr of parentAttributes) { // parentAttributes so we process also inherited templates
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (ownedAttr.type === 'relation' & &  ownedAttr.name === 'template') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    const templateNote = this.becca.notes[ownedAttr.value];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    if (templateNote) {
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                        templateAttributes.push(
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                            ...templateNote.__getAttributes(newPath)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                                // template attr is used as a marker for templates, but it's not meant to be inherited
							 
						 
					
						
							
								
									
										
										
										
											2022-10-26 14:23:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                                .filter(attr => !(attr.type === 'label' & &  (attr.name === 'template' || attr.name === 'workspacetemplate')))
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                        );
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.__attributeCache = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const addedAttributeIds = new Set();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const attr of parentAttributes.concat(templateAttributes)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (!addedAttributeIds.has(attr.attributeId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    addedAttributeIds.add(attr.attributeId);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    this.__attributeCache.push(attr);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.inheritableAttributeCache = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const attr of this.__attributeCache) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (attr.isInheritable) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    this.inheritableAttributeCache.push(attr);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.__attributeCache;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @private
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    __getInheritableAttributes(path) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (path.includes(this.noteId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.inheritableAttributeCache) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.__getAttributes(path); // will refresh also this.inheritableAttributeCache
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.inheritableAttributeCache;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-17 21:46:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    __validateTypeName(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (type & &  type !== 'label' & &  type !== 'relation') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            throw new Error(`Unrecognized attribute type '${type}'. Only 'label' and 'relation' are possible values.`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const firstLetter = name.charAt(0);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (firstLetter === '#' || firstLetter === '~') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                throw new Error(`Detect '#' or '~' in the attribute's name. In the API, attribute names should be set without these characters.`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-07-11 23:15:16 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param type
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param [value]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasAttribute(type, name, value = null) {
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return !!this.getAttributes().find(attr =>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            attr.type === type
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            & &  attr.name === name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            & &  (value === undefined || value === null || attr.value === value)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        );
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getAttributeCaseInsensitive(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        name = name.toLowerCase();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        value = value ? value.toLowerCase() : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().find(
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            attr => attr.type === type
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            & &  attr.name.toLowerCase() === name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            & &  (!value || attr.value.toLowerCase() === value));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getRelationTarget(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const relation = this.getAttributes().find(attr => attr.type === 'relation' & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return relation ? relation.targetNote : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} true if label exists (including inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasLabel(name, value) { return this.hasAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} true if label exists (excluding inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasOwnedLabel(name, value) { return this.hasOwnedAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} true if relation exists (including inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasRelation(name, value) { return this.hasAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} true if relation exists (excluding inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasOwnedRelation(name, value) { return this.hasOwnedAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute|null} label if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getLabel(name) { return this.getAttribute(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute|null} label if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedLabel(name) { return this.getOwnedAttribute(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute|null} relation if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getRelation(name) { return this.getAttribute(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute|null} relation if it exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedRelation(name) { return this.getOwnedAttribute(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} label value if label exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getLabelValue(name) { return this.getAttributeValue(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} label value if label exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedLabelValue(name) { return this.getOwnedAttributeValue(LABEL, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} relation value if relation exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getRelationValue(name) { return this.getAttributeValue(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} relation value if relation exists, null otherwise
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedRelationValue(name) { return this.getOwnedAttributeValue(RELATION, name); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {boolean} true if note has an attribute with given type and name (excluding inherited)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    hasOwnedAttribute(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return !!this.getOwnedAttribute(type, name, value);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute} attribute of given type and name. If there's more such attributes, first is  returned. Returns null if there's no such attribute belonging to this note.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getAttribute(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = this.getAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attributes.find(attr => attr.type === type & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} attribute value of given type and name or null if no such attribute exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getAttributeValue(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attr = this.getAttribute(type, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attr ? attr.value : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string|null} attribute value of given type and name or null if no such attribute exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedAttributeValue(type, name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attr = this.getOwnedAttribute(type, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attr ? attr.value : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} all note's labels (attributes with type label), including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getLabels(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes(LABEL, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string[]} all note's label values, including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getLabelValues(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getLabels(name).map(l => l.value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} all note's labels (attributes with type label), excluding inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedLabels(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getOwnedAttributes(LABEL, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - label name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {string[]} all note's label values, excluding inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedLabelValues(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getOwnedAttributes(LABEL, name).map(l => l.value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - relation name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} all note's relations (attributes with type relation), including inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getRelations(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes(RELATION, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [name] - relation name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} all note's relations (attributes with type relation), excluding inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getOwnedRelations(name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getOwnedAttributes(RELATION, name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * @param {string|null} [type] - (optional) attribute type to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string|null} [name] - (optional) attribute name to filter
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string|null} [value] - (optional) attribute value to filter
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute[]} note's "owned" attributes - excluding inherited ones
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    getOwnedAttributes(type = null, name = null, value = null) {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-17 21:46:51 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.__validateTypeName(type, name);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if (type & &  name & &  value !== undefined & &  value !== null) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.ownedAttributes.filter(attr => attr.type === type & &  attr.name === name & &  attr.value === value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else if (type & &  name) {
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            return this.ownedAttributes.filter(attr => attr.type === type & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else if (type) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.ownedAttributes.filter(attr => attr.type === type);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else if (name) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.ownedAttributes.filter(attr => attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return this.ownedAttributes.slice();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {Attribute} attribute belonging to this specific note (excludes inherited attributes)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * This method can be significantly faster than the getAttribute()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    getOwnedAttribute(type, name, value = null) {
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        const attrs = this.getOwnedAttributes(type, name, value);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return attrs.length > 0 ? attrs[0] : null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get isArchived() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.hasAttribute('label', 'archived');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    hasInheritableOwnedArchivedLabel() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return !!this.ownedAttributes.find(attr => attr.type === 'label' & &  attr.name === 'archived' & &  attr.isInheritable);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // will sort the parents so that non-search &  non-archived are first and archived at the end
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // this is done so that non-search &  non-archived paths are always explored as first when looking for note path
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    sortParents() {
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        this.parentBranches.sort((a, b) =>
							 
						 
					
						
							
								
									
										
										
										
											2022-10-26 14:23:56 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            a.branchId.startsWith('virt-') // FIXME: search virtual notes appear only in froca so this is probably not necessary
							 
						 
					
						
							
								
									
										
										
										
											2022-08-27 13:45:22 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            || a.parentNote?.hasInheritableOwnedArchivedLabel() ? 1 : -1);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-08-27 13:45:22 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.parents = this.parentBranches
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .map(branch => branch.parentNote)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .filter(note => !!note);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * This is used for:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - fast searching
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * - note similarity evaluation
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {string} - returns flattened textual representation of note, prefixes and attributes
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getFlatText() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.flatTextCache) {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            this.flatTextCache = `${this.noteId} ${this.type} ${this.mime} `;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const branch of this.parentBranches) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (branch.prefix) {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                    this.flatTextCache += `${branch.prefix} `;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            this.flatTextCache += `${this.title} `;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const attr of this.getAttributes()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // it's best to use space as separator since spaces are filtered from the search string by the tokenization into words
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                this.flatTextCache += `${attr.type === 'label' ? '#' : '~'}${attr.name}`;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (attr.value) {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                    this.flatTextCache += `=${attr.value}`;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                this.flatTextCache += ' ';
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.flatTextCache = utils.normalize(this.flatTextCache);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.flatTextCache;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    invalidateThisCache() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.flatTextCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.__attributeCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.inheritableAttributeCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.ancestorCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    invalidateSubTree(path = []) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (path.includes(this.noteId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.invalidateThisCache();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.children.length || this.targetRelations.length) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            path = [...path, this.noteId];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const childNote of this.children) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            childNote.invalidateSubTree(path);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const targetRelation of this.targetRelations) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (targetRelation.name === 'template') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const note = targetRelation.note;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (note) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    note.invalidateSubTree(path);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    invalidateSubtreeFlatText() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.flatTextCache = null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const childNote of this.children) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            childNote.invalidateSubtreeFlatText();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const targetRelation of this.targetRelations) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (targetRelation.name === 'template') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const note = targetRelation.note;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (note) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    note.invalidateSubtreeFlatText();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getRelationDefinitions() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getLabels()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .filter(l => l.name.startsWith("relation:"));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getLabelDefinitions() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getLabels()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .filter(l => l.name.startsWith("relation:"));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isTemplate() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return !!this.targetRelations.find(rel => rel.name === 'template');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Note[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getSubtreeNotesIncludingTemplated() {
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        const set = new Set();
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        function inner(note) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (set.has(note)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            set.add(note);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            for (const childNote of note.children) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                inner(childNote);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const targetRelation of note.targetRelations) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (targetRelation.name === 'template') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    const targetNote = targetRelation.note;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    if (targetNote) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                        inner(targetNote);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        inner(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return Array.from(set);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-11-12 20:40:40 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @return {Note[]} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getSearchResultNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.type !== 'search') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        try {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const searchService = require("../../services/search/services/search");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const {searchResultNoteIds} = searchService.searchFromNote(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const becca = this.becca;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return searchResultNoteIds
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .map(resultNoteId => becca.notes[resultNoteId])
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .filter(note => !!note);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        catch (e) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            log.error(`Could not resolve search note ${this.noteId}: ${e.message}`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {{notes: Note[], relationships: Array.< {parentNoteId: string, childNoteId: string}>}}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getSubtree({includeArchived = true, resolveSearch = false} = {}) {
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        const noteSet = new Set();
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        const relationships = []; // list of tuples parentNoteId -> childNoteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        function resolveSearchNote(searchNote) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            try {
							 
						 
					
						
							
								
									
										
										
										
											2022-11-12 20:40:40 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                for (const resultNote of searchNote.getSearchResultNotes()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    addSubtreeNotesInner(resultNote, searchNote);
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            catch (e) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                log.error(`Could not resolve search note ${searchNote?.noteId}: ${e.message}`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        function addSubtreeNotesInner(note, parentNote = null) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // share can be removed after 0.57 since it will be put under hidden
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if (note.noteId === '_hidden' || note.noteId === '_share') {
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (parentNote) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // this needs to happen first before noteSet check to include all clone relationships
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                relationships.push({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    parentNoteId: parentNote.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    childNoteId: note.noteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                });
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (noteSet.has(note)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (!includeArchived & &  note.isArchived) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteSet.add(note);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            if (note.type === 'search') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (resolveSearch) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    resolveSearchNote(note);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                for (const childNote of note.children) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    addSubtreeNotesInner(childNote, note);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        addSubtreeNotesInner(this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            notes: Array.from(noteSet),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            relationships
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        };
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {String[]} */
							 
						 
					
						
							
								
									
										
										
										
											2022-11-07 21:26:13 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    getSubtreeNoteIds({includeArchived = true, resolveSearch = false} = {}) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getSubtree({includeArchived, resolveSearch})
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .notes
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .map(note => note.noteId);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getDescendantNoteIds() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getSubtreeNoteIds();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get parentCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.parents.length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get childrenCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.children.length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get labelCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().filter(attr => attr.type === 'label').length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get ownedLabelCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.ownedAttributes.filter(attr => attr.type === 'label').length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get relationCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().filter(attr => attr.type === 'relation' & &  !attr.isAutoLink()).length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get relationCountIncludingLinks() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().filter(attr => attr.type === 'relation').length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get ownedRelationCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.ownedAttributes.filter(attr => attr.type === 'relation' & &  !attr.isAutoLink()).length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get ownedRelationCountIncludingLinks() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.ownedAttributes.filter(attr => attr.type === 'relation').length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get targetRelationCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.targetRelations.filter(attr => !attr.isAutoLink()).length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get targetRelationCountIncludingLinks() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.targetRelations.length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get attributeCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    get ownedAttributeCount() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.getAttributes().length;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Note[]} */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getAncestors() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!this.ancestorCache) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            const noteIds = new Set();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.ancestorCache = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const parent of this.parents) {
							 
						 
					
						
							
								
									
										
										
										
											2022-01-14 19:53:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                if (noteIds.has(parent.noteId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    continue;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-14 19:53:59 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                this.ancestorCache.push(parent);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                noteIds.add(parent.noteId);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								                for (const ancestorNote of parent.getAncestors()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    if (!noteIds.has(ancestorNote.noteId)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                        this.ancestorCache.push(ancestorNote);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                        noteIds.add(ancestorNote.noteId);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.ancestorCache;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {boolean} */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    hasAncestor(ancestorNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const ancestorNote of this.getAncestors()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (ancestorNote.noteId === ancestorNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                return true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return false;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    getTargetRelations() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.targetRelations;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /** @returns {Note[]} - returns only notes which are templated, does not include their subtrees
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     *                     in effect returns notes which are influenced by note's non-inheritable attributes */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getTemplatedNotes() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const arr = [this];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const targetRelation of this.targetRelations) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (targetRelation.name === 'template') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                const note = targetRelation.note;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                if (note) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    arr.push(note);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return arr;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getDistanceToAncestor(ancestorNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.noteId === ancestorNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return 0;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        let minDistance = 999999;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const parent of this.parents) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            minDistance = Math.min(minDistance, parent.getDistanceToAncestor(ancestorNoteId) + 1);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return minDistance;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getNoteRevisions() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return sql.getRows("SELECT * FROM note_revisions WHERE noteId = ?", [this.noteId])
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .map(row => new NoteRevision(row));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {string[][]} - array of notePaths (each represented by array of noteIds constituting the particular note path)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getAllNotePaths() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.noteId === 'root') {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return [['root']];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const notePaths = [];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const parentNote of this.getParentNotes()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            for (const parentPath of parentNote.getAllNotePaths()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                parentPath.push(this.noteId);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                notePaths.push(parentPath);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return notePaths;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param ancestorNoteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {boolean} - true if ancestorNoteId occurs in at least one of the note's paths
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isDescendantOfNote(ancestorNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const notePaths = this.getAllNotePaths();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return notePaths.some(path => path.includes(ancestorNoteId));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Update's given attribute's value or creates it if it doesn't exist
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    setAttribute(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = this.getOwnedAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attr = attributes.find(attr => attr.type === type & &  attr.name === name);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        value = value?.toString() || "";
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (attr) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (attr.value !== value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                attr.value = value;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                attr.save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
									
										
										
										
											2022-01-10 19:54:38 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            const Attribute = require("./attribute");
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            new Attribute({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                type: type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                name: name,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                value: value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }).save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Removes given attribute name-value pair if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label, relation, etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    removeAttribute(type, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const attributes = this.getOwnedAttributes();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        for (const attribute of attributes) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (attribute.type === type & &  attribute.name === name & &  (value === undefined || value === attribute.value)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                attribute.markAsDeleted();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
									
										
										
										
											2022-08-27 13:42:44 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								     * Adds a new attribute to this note. The attribute is saved and returned.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * See addLabel, addRelation for more specific methods.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type (label / relation)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - name of the attribute, not including the leading ~/#
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - value of the attribute - text for labels, target note ID for relations; optional.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								     * @return {Attribute}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    addAttribute(type, name, value = "", isInheritable = false, position = 1000) {
							 
						 
					
						
							
								
									
										
										
										
											2022-01-10 19:54:38 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        const Attribute = require("./attribute");
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return new Attribute({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            type: type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            name: name,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            value: value,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            isInheritable: isInheritable,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            position: position
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }).save();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-08-27 13:42:44 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Adds a new label to this note. The label attribute is saved and returned.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - name of the label, not including the leading #
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - text value of the label; optional
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {Attribute}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    addLabel(name, value = "", isInheritable = false) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.addAttribute(LABEL, name, value, isInheritable);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-08-27 13:42:44 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Adds a new relation to this note. The relation attribute is saved and
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * returned.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - name of the relation, not including the leading ~
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} value - ID of the target note of the relation
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {Attribute}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    addRelation(name, targetNoteId, isInheritable = false) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.addAttribute(RELATION, name, targetNoteId, isInheritable);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, attribute is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} type - attribute type ('relation', 'label' etc.)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - attribute name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - attribute value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    toggleAttribute(type, enabled, name, value) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (enabled) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.setAttribute(type, name, value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            this.removeAttribute(type, name, value);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, label is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value (optional)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    toggleLabel(enabled, name, value) { return this.toggleAttribute(LABEL, enabled, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Based on enabled, relation is either set or removed.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {boolean} enabled - toggle On or Off
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    toggleRelation(enabled, name, value) { return this.toggleAttribute(RELATION, enabled, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Update's given label's value or creates it if it doesn't exist
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    setLabel(name, value) { return this.setAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Update's given relation's value or creates it if it doesn't exist
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} value - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    setRelation(name, value) { return this.setAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Remove label name-value pair, if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - label name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - label value
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    removeLabel(name, value) { return this.removeAttribute(LABEL, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * Remove relation name-value pair, if it exists.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} name - relation name
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [value] - relation value (noteId)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    removeRelation(name, value) { return this.removeAttribute(RELATION, name, value); }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    searchNotesInSubtree(searchString) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const searchService = require("../../services/search/services/search");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return searchService.searchNotes(searchString);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    searchNoteInSubtree(searchString) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.searchNotesInSubtree(searchString)[0];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-07-19 23:32:03 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param parentNoteId
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @returns {{success: boolean, message: string}}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    cloneTo(parentNoteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const cloningService = require("../../services/cloning");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const branch = this.becca.getNote(parentNoteId).getParentBranches()[0];
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-01-05 20:31:21 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return cloningService.cloneNoteToBranch(this.noteId, branch.branchId);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-19 23:06:46 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * (Soft) delete a note and all its descendants.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     *
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {string} [deleteId] - optional delete identified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @param {TaskContext} [taskContext]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    deleteNote(deleteId, taskContext) {
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if (this.isDeleted) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-19 23:06:46 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        if (!deleteId) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            deleteId = utils.randomString(10);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!taskContext) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            taskContext = new TaskContext('no-progress-reporting');
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        // needs to be run before branches and attributes are deleted and thus attached relations disappear
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const handlers = require("../../services/handlers");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        handlers.runAttachedRelations(this, 'runOnNoteDeletion', this);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        taskContext.noteDeletionHandlerTriggered = true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-19 23:06:46 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        for (const branch of this.getParentBranches()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            branch.deleteBranch(deleteId, taskContext);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    decrypt() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (this.isProtected & &  !this.isDecrypted & &  protectedSessionService.isProtectedSessionAvailable()) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            try {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                this.title = protectedSessionService.decryptString(this.title);
							 
						 
					
						
							
								
									
										
										
										
											2022-04-16 00:17:32 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                this.flatTextCache = null;
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                this.isDecrypted = true;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            catch (e) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                log.error(`Could not decrypt note ${this.noteId}: ${e.message} ${e.stack}`);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    isLaunchBarConfig() {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-22 14:57:00 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return this.type === 'launcher' || ['_lbRoot', '_lbAvailableLaunchers', '_lbVisibleLaunchers'].includes(this.noteId);
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    isOptions() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return this.noteId.startsWith("options");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    get isDeleted() {
							 
						 
					
						
							
								
									
										
										
										
											2022-12-14 23:51:56 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return !(this.noteId in this.becca.notes) || this.isBeingDeleted;
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-06-02 17:25:58 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    /**
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     * @return {NoteRevision|null}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								     */
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    saveNoteRevision() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const content = this.getContent();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (!content || (Buffer.isBuffer(content) & &  content.byteLength === 0)) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            return null;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const contentMetadata = this.getContentMetadata();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const noteRevision = new NoteRevision({
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // title and text should be decrypted now
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            title: this.title,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            type: this.type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            mime: this.mime,
							 
						 
					
						
							
								
									
										
										
										
											2022-06-19 14:06:00 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            isProtected: this.isProtected,
							 
						 
					
						
							
								
									
										
										
										
											2022-06-02 17:25:58 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            utcDateLastEdited: this.utcDateModified > contentMetadata.utcDateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                ? this.utcDateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                : contentMetadata.utcDateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateCreated: dateUtils.utcNowDateTime(),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateModified: dateUtils.utcNowDateTime(),
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            dateLastEdited: this.dateModified > contentMetadata.dateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                ? this.dateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                : contentMetadata.dateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            dateCreated: dateUtils.localNowDateTime()
							 
						 
					
						
							
								
									
										
										
										
											2022-07-11 23:15:16 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        }, true).save();
							 
						 
					
						
							
								
									
										
										
										
											2022-06-02 17:25:58 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        noteRevision.setContent(content);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return noteRevision;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    beforeSaving() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        super.beforeSaving();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 21:30:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        this.becca.addNote(this.noteId, this);
							 
						 
					
						
							
								
									
										
										
										
											2021-09-30 12:26:13 +02:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.dateModified = dateUtils.localNowDateTime();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        this.utcDateModified = dateUtils.utcNowDateTime();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getPojo() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            noteId: this.noteId,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            title: this.title,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            isProtected: this.isProtected,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            type: this.type,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            mime: this.mime,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            isDeleted: false,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            dateCreated: this.dateCreated,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            dateModified: this.dateModified,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateCreated: this.utcDateCreated,
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            utcDateModified: this.utcDateModified
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        };
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    getPojoToSave() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        const pojo = this.getPojo();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        if (pojo.isProtected) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            if (this.isDecrypted) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                pojo.title = protectedSessionService.encrypt(pojo.title);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            else {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // updating protected note outside of protected session means we will keep original ciphertexts
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                delete pojo.title;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return pojo;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								module.exports = Note;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								< / 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 >