mirror of
https://github.com/TriliumNext/Notes.git
synced 2025-10-24 16:21:32 +08:00
286 lines
8.2 KiB
HTML
286 lines
8.2 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<title>JSDoc: Source: becca/entities/battribute.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/battribute.js</h1>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
<section>
|
|
<article>
|
|
<pre class="prettyprint source linenums"><code>"use strict";
|
|
|
|
const BNote = require('./bnote');
|
|
const AbstractBeccaEntity = require("./abstract_becca_entity");
|
|
const sql = require("../../services/sql");
|
|
const dateUtils = require("../../services/date_utils");
|
|
const promotedAttributeDefinitionParser = require("../../services/promoted_attribute_definition_parser");
|
|
const {sanitizeAttributeName} = require("../../services/sanitize_attribute_name");
|
|
|
|
/**
|
|
* Attribute is an abstract concept which has two real uses - label (key - value pair)
|
|
* and relation (representing named relationship between source and target note)
|
|
*
|
|
* @extends AbstractBeccaEntity
|
|
*/
|
|
class BAttribute extends AbstractBeccaEntity {
|
|
static get entityName() { return "attributes"; }
|
|
static get primaryKeyName() { return "attributeId"; }
|
|
static get hashedProperties() { return ["attributeId", "noteId", "type", "name", "value", "isInheritable"]; }
|
|
|
|
constructor(row) {
|
|
super();
|
|
|
|
if (!row) {
|
|
return;
|
|
}
|
|
|
|
this.updateFromRow(row);
|
|
this.init();
|
|
}
|
|
|
|
updateFromRow(row) {
|
|
this.update([
|
|
row.attributeId,
|
|
row.noteId,
|
|
row.type,
|
|
row.name,
|
|
row.value,
|
|
row.isInheritable,
|
|
row.position,
|
|
row.utcDateModified
|
|
]);
|
|
}
|
|
|
|
update([attributeId, noteId, type, name, value, isInheritable, position, utcDateModified]) {
|
|
/** @type {string} */
|
|
this.attributeId = attributeId;
|
|
/** @type {string} */
|
|
this.noteId = noteId;
|
|
/** @type {string} */
|
|
this.type = type;
|
|
/** @type {string} */
|
|
this.name = name;
|
|
/** @type {int} */
|
|
this.position = position;
|
|
/** @type {string} */
|
|
this.value = value || "";
|
|
/** @type {boolean} */
|
|
this.isInheritable = !!isInheritable;
|
|
/** @type {string} */
|
|
this.utcDateModified = utcDateModified;
|
|
|
|
return this;
|
|
}
|
|
|
|
init() {
|
|
if (this.attributeId) {
|
|
this.becca.attributes[this.attributeId] = this;
|
|
}
|
|
|
|
if (!(this.noteId in this.becca.notes)) {
|
|
// entities can come out of order in sync, create skeleton which will be filled later
|
|
this.becca.addNote(this.noteId, new BNote({noteId: this.noteId}));
|
|
}
|
|
|
|
this.becca.notes[this.noteId].ownedAttributes.push(this);
|
|
|
|
const key = `${this.type}-${this.name.toLowerCase()}`;
|
|
this.becca.attributeIndex[key] = this.becca.attributeIndex[key] || [];
|
|
this.becca.attributeIndex[key].push(this);
|
|
|
|
const targetNote = this.targetNote;
|
|
|
|
if (targetNote) {
|
|
targetNote.targetRelations.push(this);
|
|
}
|
|
}
|
|
|
|
validate() {
|
|
if (!["label", "relation"].includes(this.type)) {
|
|
throw new Error(`Invalid attribute type '${this.type}' in attribute '${this.attributeId}' of note '${this.noteId}'`);
|
|
}
|
|
|
|
if (!this.name?.trim()) {
|
|
throw new Error(`Invalid empty name in attribute '${this.attributeId}' of note '${this.noteId}'`);
|
|
}
|
|
|
|
if (this.type === 'relation' && !(this.value in this.becca.notes)) {
|
|
throw new Error(`Cannot save relation '${this.name}' of note '${this.noteId}' since it target not existing note '${this.value}'.`);
|
|
}
|
|
}
|
|
|
|
get isAffectingSubtree() {
|
|
return this.isInheritable
|
|
|| (this.type === 'relation' && ['template', 'inherit'].includes(this.name));
|
|
}
|
|
|
|
get targetNoteId() { // alias
|
|
return this.type === 'relation' ? this.value : undefined;
|
|
}
|
|
|
|
isAutoLink() {
|
|
return this.type === 'relation' && ['internalLink', 'imageLink', 'relationMapLink', 'includeNoteLink'].includes(this.name);
|
|
}
|
|
|
|
get note() {
|
|
return this.becca.notes[this.noteId];
|
|
}
|
|
|
|
get targetNote() {
|
|
if (this.type === 'relation') {
|
|
return this.becca.notes[this.value];
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @returns {BNote|null}
|
|
*/
|
|
getNote() {
|
|
const note = this.becca.getNote(this.noteId);
|
|
|
|
if (!note) {
|
|
throw new Error(`Note '${this.noteId}' of attribute '${this.attributeId}', type '${this.type}', name '${this.name}' does not exist.`);
|
|
}
|
|
|
|
return note;
|
|
}
|
|
|
|
/**
|
|
* @returns {BNote|null}
|
|
*/
|
|
getTargetNote() {
|
|
if (this.type !== 'relation') {
|
|
throw new Error(`Attribute ${this.attributeId} is not relation`);
|
|
}
|
|
|
|
if (!this.value) {
|
|
return null;
|
|
}
|
|
|
|
return this.becca.getNote(this.value);
|
|
}
|
|
|
|
/**
|
|
* @returns {boolean}
|
|
*/
|
|
isDefinition() {
|
|
return this.type === 'label' && (this.name.startsWith('label:') || this.name.startsWith('relation:'));
|
|
}
|
|
|
|
getDefinition() {
|
|
return promotedAttributeDefinitionParser.parse(this.value);
|
|
}
|
|
|
|
getDefinedName() {
|
|
if (this.type === 'label' && this.name.startsWith('label:')) {
|
|
return this.name.substr(6);
|
|
} else if (this.type === 'label' && this.name.startsWith('relation:')) {
|
|
return this.name.substr(9);
|
|
} else {
|
|
return this.name;
|
|
}
|
|
}
|
|
|
|
get isDeleted() {
|
|
return !(this.attributeId in this.becca.attributes);
|
|
}
|
|
|
|
beforeSaving(opts = {}) {
|
|
if (!opts.skipValidation) {
|
|
this.validate();
|
|
}
|
|
|
|
this.name = sanitizeAttributeName(this.name);
|
|
|
|
if (!this.value) {
|
|
// null value isn't allowed
|
|
this.value = "";
|
|
}
|
|
|
|
if (this.position === undefined) {
|
|
// TODO: can be calculated from becca
|
|
this.position = 1 + sql.getValue(`SELECT COALESCE(MAX(position), 0) FROM attributes WHERE noteId = ?`, [this.noteId]);
|
|
}
|
|
|
|
if (!this.isInheritable) {
|
|
this.isInheritable = false;
|
|
}
|
|
|
|
this.utcDateModified = dateUtils.utcNowDateTime();
|
|
|
|
super.beforeSaving();
|
|
|
|
this.becca.attributes[this.attributeId] = this;
|
|
}
|
|
|
|
getPojo() {
|
|
return {
|
|
attributeId: this.attributeId,
|
|
noteId: this.noteId,
|
|
type: this.type,
|
|
name: this.name,
|
|
position: this.position,
|
|
value: this.value,
|
|
isInheritable: this.isInheritable,
|
|
utcDateModified: this.utcDateModified,
|
|
isDeleted: false
|
|
};
|
|
}
|
|
|
|
createClone(type, name, value, isInheritable) {
|
|
return new BAttribute({
|
|
noteId: this.noteId,
|
|
type: type,
|
|
name: name,
|
|
value: value,
|
|
position: this.position,
|
|
isInheritable: isInheritable,
|
|
utcDateModified: this.utcDateModified
|
|
});
|
|
}
|
|
}
|
|
|
|
module.exports = BAttribute;
|
|
</code></pre>
|
|
</article>
|
|
</section>
|
|
|
|
|
|
|
|
|
|
</div>
|
|
|
|
<nav>
|
|
<h2><a href="index.html">Home</a></h2><h3>Modules</h3><ul><li><a href="module-sql.html">sql</a></li></ul><h3>Classes</h3><ul><li><a href="AbstractBeccaEntity.html">AbstractBeccaEntity</a></li><li><a href="BAttribute.html">BAttribute</a></li><li><a href="BBranch.html">BBranch</a></li><li><a href="BEtapiToken.html">BEtapiToken</a></li><li><a href="BNote.html">BNote</a></li><li><a href="BNoteRevision.html">BNoteRevision</a></li><li><a href="BOption.html">BOption</a></li><li><a href="BRecentNote.html">BRecentNote</a></li><li><a href="BackendScriptApi.html">BackendScriptApi</a></li></ul>
|
|
</nav>
|
|
|
|
<br class="clear">
|
|
|
|
<footer>
|
|
Documentation generated by <a href="https://github.com/jsdoc/jsdoc">JSDoc 4.0.1</a>
|
|
</footer>
|
|
|
|
<script> prettyPrint(); </script>
|
|
<script src="scripts/linenumber.js"> </script>
|
|
</body>
|
|
</html>
|