2
0
mirror of https://github.com/frappe/books.git synced 2025-01-25 16:18:33 +00:00
books/model/document.js

333 lines
9.1 KiB
JavaScript
Raw Normal View History

2018-01-16 11:39:17 +05:30
const frappe = require('frappejs');
2018-02-13 17:24:57 +05:30
const Observable = require('frappejs/utils/observable');
2018-03-05 22:15:21 +05:30
const naming = require('./naming');
2018-01-12 17:55:07 +05:30
2018-02-13 17:24:57 +05:30
module.exports = class BaseDocument extends Observable {
2018-01-12 17:55:07 +05:30
constructor(data) {
2018-02-13 17:24:57 +05:30
super();
2018-03-27 19:25:26 +05:30
this.fetchValuesCache = {};
2018-03-05 22:15:21 +05:30
this.flags = {};
2018-01-12 17:55:07 +05:30
this.setup();
Object.assign(this, data);
2018-03-27 19:25:26 +05:30
// clear fetch-values cache
frappe.db.on('change', (params) => this.fetchValuesCache[`${params.doctype}:${params.name}`] = {});
2018-01-12 17:55:07 +05:30
}
setup() {
2018-02-13 17:24:57 +05:30
// add listeners
2018-01-12 17:55:07 +05:30
}
2018-02-12 17:31:31 +05:30
get meta() {
if (!this._meta) {
this._meta = frappe.getMeta(this.doctype);
}
return this._meta;
}
async getSettings() {
if (!this._settings) {
this._settings = await frappe.getSingle(this.meta.settings);
}
return this._settings;
}
// set value and trigger change
async set(fieldname, value) {
if (typeof fieldname === 'object') {
const valueDict = fieldname;
for (let fieldname in valueDict) {
await this.set(fieldname, valueDict[fieldname]);
}
return;
}
if (this[fieldname] !== value) {
this._dirty = true;
this[fieldname] = await this.validateField(fieldname, value);
await this.applyChange(fieldname);
}
}
async applyChange(fieldname) {
if (await this.applyFormula()) {
2018-02-08 17:15:32 +05:30
// multiple changes
await this.trigger('change', { doc: this });
} else {
// no other change, trigger control refresh
await this.trigger('change', { doc: this, fieldname: fieldname });
2018-02-08 17:15:32 +05:30
}
2018-01-12 17:55:07 +05:30
}
2018-02-12 17:31:31 +05:30
setDefaults() {
for (let field of this.meta.fields) {
if (this[field.fieldname]===null || this[field.fieldname]===undefined) {
let defaultValue = null;
if (field.fieldtype === 'Date') {
defaultValue = (new Date()).toISOString().substr(0, 10);
2018-02-14 18:20:56 +05:30
}
if (field.fieldtype === 'Table') {
defaultValue = [];
}
if (field.default) {
defaultValue = field.default;
}
this[field.fieldname] = defaultValue;
2018-02-12 17:31:31 +05:30
}
}
}
2018-02-08 17:15:32 +05:30
setKeywords() {
2018-01-12 17:55:07 +05:30
let keywords = [];
for (let fieldname of this.meta.getKeywordFields()) {
2018-01-12 17:55:07 +05:30
keywords.push(this[fieldname]);
}
this.keywords = keywords.join(', ');
}
append(key, document) {
if (!this[key]) {
this[key] = [];
}
2018-02-08 17:15:32 +05:30
this[key].push(this.initDoc(document));
2018-01-12 17:55:07 +05:30
}
2018-02-08 17:15:32 +05:30
initDoc(data) {
2018-01-12 17:55:07 +05:30
if (data.prototype instanceof Document) {
return data;
} else {
2018-02-01 14:59:25 +05:30
return new Document(data);
2018-01-12 17:55:07 +05:30
}
}
2018-02-08 17:15:32 +05:30
async validateField(key, value) {
let field = this.meta.getField(key);
2018-01-31 18:34:46 +05:30
if (field && field.fieldtype == 'Select') {
2018-02-13 17:24:57 +05:30
return this.meta.validateSelect(field, value);
2018-01-12 17:55:07 +05:30
}
return value;
2018-01-12 17:55:07 +05:30
}
2018-02-08 17:15:32 +05:30
getValidDict() {
2018-02-01 16:37:36 +05:30
let data = {};
for (let field of this.meta.getValidFields()) {
2018-02-01 16:37:36 +05:30
data[field.fieldname] = this[field.fieldname];
2018-01-12 17:55:07 +05:30
}
2018-02-01 16:37:36 +05:30
return data;
2018-01-12 17:55:07 +05:30
}
2018-03-05 22:15:21 +05:30
getFullDict() {
let data = this.getValidDict();
return data;
}
2018-02-08 17:15:32 +05:30
setStandardValues() {
// set standard values on server-side only
if (frappe.isServer) {
2018-03-05 22:15:21 +05:30
let now = (new Date()).toISOString();
2018-05-07 09:53:20 +05:30
if (!this.submitted) {
this.submitted = 0;
}
if (!this.owner) {
this.owner = frappe.session.user;
2018-05-07 09:53:20 +05:30
}
if (!this.creation) {
this.creation = now;
}
2018-04-30 19:57:41 +05:30
if (!this.modifiedBy) {
this.modifiedBy = frappe.session.user;
}
this.modified = now;
2018-01-12 17:55:07 +05:30
}
}
async load() {
let data = await frappe.db.get(this.doctype, this.name);
if (data.name) {
2018-02-08 17:15:32 +05:30
this.syncValues(data);
if (this.meta.isSingle) {
this.setDefaults();
}
} else {
throw new frappe.errors.NotFound(`Not Found: ${this.doctype} ${this.name}`);
}
2018-01-12 17:55:07 +05:30
}
2018-02-08 17:15:32 +05:30
syncValues(data) {
this.clearValues();
2018-02-01 16:37:36 +05:30
Object.assign(this, data);
this._dirty = false;
this.trigger('change', {doc: this});
2018-02-01 16:37:36 +05:30
}
2018-02-08 17:15:32 +05:30
clearValues() {
for (let field of this.meta.getValidFields()) {
2018-02-01 16:37:36 +05:30
if(this[field.fieldname]) {
delete this[field.fieldname];
}
}
}
2018-02-08 17:15:32 +05:30
setChildIdx() {
2018-02-07 18:53:52 +05:30
// renumber children
for (let field of this.meta.getValidFields()) {
2018-02-07 18:53:52 +05:30
if (field.fieldtype==='Table') {
for(let i=0; i < (this[field.fieldname] || []).length; i++) {
this[field.fieldname][i].idx = i;
}
}
}
}
async compareWithCurrentDoc() {
if (frappe.isServer && !this._notInserted) {
let currentDoc = await frappe.db.get(this.doctype, this.name);
// check for conflict
if (currentDoc && this.modified != currentDoc.modified) {
throw new frappe.errors.Conflict(frappe._('Document {0} {1} has been modified after loading', [this.doctype, this.name]));
}
if (this.submitted && !this.meta.isSubmittable) {
throw new frappe.errors.ValidationError(frappe._('Document type {1} is not submittable', [this.doctype]));
}
// set submit action flag
if (this.submitted && !currentDoc.submitted) {
2018-03-05 22:15:21 +05:30
this.flags.submitAction = true;
}
if (currentDoc.submitted && !this.submitted) {
2018-03-05 22:15:21 +05:30
this.flags.revertAction = true;
}
2018-03-05 22:15:21 +05:30
}
}
async applyFormula() {
if (!this.meta.hasFormula()) {
2018-02-08 17:15:32 +05:30
return false;
}
2018-02-09 18:25:55 +05:30
let doc = this;
2018-02-08 17:15:32 +05:30
// children
for (let tablefield of this.meta.getTableFields()) {
let formulaFields = frappe.getMeta(tablefield.childtype).getFormulaFields();
if (formulaFields.length) {
// for each row
2018-02-09 18:25:55 +05:30
for (let row of this[tablefield.fieldname]) {
2018-02-08 17:15:32 +05:30
for (let field of formulaFields) {
2018-03-08 19:01:22 +05:30
const val = await field.formula(row, doc);
if (val !== false) {
row[field.fieldname] = val;
}
2018-02-08 17:15:32 +05:30
}
}
}
}
// parent
for (let field of this.meta.getFormulaFields()) {
2018-03-08 19:01:22 +05:30
const val = await field.formula(doc);
if (val !== false) {
doc[field.fieldname] = val;
}
2018-02-08 17:15:32 +05:30
}
return true;
}
2018-02-07 18:53:52 +05:30
async commit() {
// re-run triggers
2018-02-08 17:15:32 +05:30
this.setStandardValues();
this.setKeywords();
this.setChildIdx();
await this.applyFormula();
await this.trigger('validate');
2018-02-07 18:53:52 +05:30
}
async insert() {
2018-03-07 16:07:58 +05:30
await naming.setName(this);
2018-02-07 18:53:52 +05:30
await this.commit();
await this.trigger('beforeInsert');
2018-03-05 22:15:21 +05:30
const data = await frappe.db.insert(this.doctype, this.getValidDict());
this.syncValues(data);
await this.trigger('afterInsert');
await this.trigger('afterSave');
2018-02-01 16:37:36 +05:30
return this;
}
2018-02-07 18:53:52 +05:30
async update() {
await this.compareWithCurrentDoc();
2018-02-07 18:53:52 +05:30
await this.commit();
await this.trigger('beforeUpdate');
2018-03-05 22:15:21 +05:30
// before submit
if (this.flags.submitAction) await this.trigger('beforeSubmit');
if (this.flags.revertAction) await this.trigger('beforeRevert');
const data = await frappe.db.update(this.doctype, this.getValidDict());
this.syncValues(data);
await this.trigger('afterUpdate');
await this.trigger('afterSave');
2018-03-05 22:15:21 +05:30
// after submit
if (this.flags.submitAction) await this.trigger('afterSubmit');
if (this.flags.revertAction) await this.trigger('afterRevert');
2018-02-01 16:37:36 +05:30
return this;
2018-01-12 17:55:07 +05:30
}
async delete() {
2018-06-25 13:19:11 +05:30
await this.trigger('beforeDelete');
2018-01-12 17:55:07 +05:30
await frappe.db.delete(this.doctype, this.name);
2018-06-25 13:19:11 +05:30
await this.trigger('afterDelete');
2018-01-12 17:55:07 +05:30
}
2018-03-05 22:15:21 +05:30
async submit() {
this.submitted = 1;
this.update();
}
async revert() {
this.submitted = 0;
this.update();
}
// trigger methods on the class if they match
// with the trigger name
2018-02-13 17:24:57 +05:30
async trigger(event, params) {
if (this[event]) {
await this[event](params);
2018-01-12 17:55:07 +05:30
}
2018-02-13 17:24:57 +05:30
await super.trigger(event, params);
2018-01-12 17:55:07 +05:30
}
2018-02-09 18:25:55 +05:30
// helper functions
getSum(tablefield, childfield) {
2018-02-21 15:13:21 +05:30
return this[tablefield].map(d => (d[childfield] || 0)).reduce((a, b) => a + b, 0);
2018-02-09 18:25:55 +05:30
}
async getFrom(doctype, name, fieldname) {
if (!name) return '';
2018-03-27 19:25:26 +05:30
let _values = this.fetchValuesCache[`${doctype}:${name}`] || (this.fetchValuesCache[`${doctype}:${name}`] = {});
2018-03-07 16:07:58 +05:30
if (!_values[fieldname]) {
_values[fieldname] = await frappe.db.getValue(doctype, name, fieldname);
2018-02-09 18:25:55 +05:30
}
2018-03-07 16:07:58 +05:30
return _values[fieldname];
2018-02-09 18:25:55 +05:30
}
2018-01-24 17:22:05 +05:30
};