2018-01-24 11:52:05 +00:00
|
|
|
const BaseDocument = require('./document');
|
2018-01-16 06:09:17 +00:00
|
|
|
const frappe = require('frappejs');
|
2019-09-17 08:14:09 +00:00
|
|
|
const model = require('./index');
|
2018-07-09 14:55:24 +00:00
|
|
|
const indicatorColor = require('frappejs/ui/constants/indicators');
|
2018-01-12 12:25:07 +00:00
|
|
|
|
2018-01-24 11:52:05 +00:00
|
|
|
module.exports = class BaseMeta extends BaseDocument {
|
2019-09-17 08:14:09 +00:00
|
|
|
constructor(data) {
|
2019-10-05 21:47:09 +00:00
|
|
|
if (data.basedOn) {
|
|
|
|
let config = frappe.models[data.basedOn];
|
|
|
|
Object.assign(data, config, {
|
|
|
|
name: data.name,
|
|
|
|
label: data.label,
|
|
|
|
filters: data.filters
|
2019-10-08 11:16:54 +00:00
|
|
|
});
|
2019-10-05 21:47:09 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
super(data);
|
|
|
|
this.setDefaultIndicators();
|
|
|
|
if (this.setupMeta) {
|
|
|
|
this.setupMeta();
|
|
|
|
}
|
|
|
|
if (!this.titleField) {
|
|
|
|
this.titleField = 'name';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-08 11:16:54 +00:00
|
|
|
setValues(data) {
|
|
|
|
Object.assign(this, data);
|
2019-12-02 12:18:11 +00:00
|
|
|
this.processFields();
|
|
|
|
}
|
|
|
|
|
|
|
|
processFields() {
|
|
|
|
// add name field
|
2019-10-26 14:46:19 +00:00
|
|
|
if (!this.fields.find(df => df.fieldname === 'name') && !this.isSingle) {
|
2019-10-19 14:37:24 +00:00
|
|
|
this.fields = [
|
|
|
|
{
|
2019-11-21 19:21:43 +00:00
|
|
|
label: frappe._('ID'),
|
2019-10-19 14:37:24 +00:00
|
|
|
fieldname: 'name',
|
|
|
|
fieldtype: 'Data',
|
2019-10-29 19:39:07 +00:00
|
|
|
required: 1,
|
|
|
|
readOnly: 1
|
2019-10-19 14:37:24 +00:00
|
|
|
}
|
|
|
|
].concat(this.fields);
|
|
|
|
}
|
2019-12-02 12:18:11 +00:00
|
|
|
|
|
|
|
// attach default precision to Float and Currency
|
|
|
|
this.fields = this.fields.map(df => {
|
|
|
|
if (['Float', 'Currency'].includes(df.fieldtype)) {
|
|
|
|
df.precision = df.precision || frappe.SystemSettings.floatPrecision;
|
|
|
|
}
|
|
|
|
return df;
|
|
|
|
});
|
2019-10-08 11:16:54 +00:00
|
|
|
}
|
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
hasField(fieldname) {
|
|
|
|
return this.getField(fieldname) ? true : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
getField(fieldname) {
|
|
|
|
if (!this._field_map) {
|
|
|
|
this._field_map = {};
|
|
|
|
for (let field of this.fields) {
|
|
|
|
this._field_map[field.fieldname] = field;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this._field_map[fieldname];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get fields filtered by filters
|
|
|
|
* @param {Object} filters
|
|
|
|
*
|
|
|
|
* Usage:
|
|
|
|
* meta = frappe.getMeta('ToDo')
|
|
|
|
* dataFields = meta.getFieldsWith({ fieldtype: 'Data' })
|
|
|
|
*/
|
|
|
|
getFieldsWith(filters) {
|
|
|
|
return this.fields.filter(df => {
|
|
|
|
let match = true;
|
|
|
|
for (const key in filters) {
|
|
|
|
const value = filters[key];
|
|
|
|
match = df[key] === value;
|
|
|
|
}
|
|
|
|
return match;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
getLabel(fieldname) {
|
|
|
|
let df = this.getField(fieldname);
|
|
|
|
return df.getLabel || df.label;
|
|
|
|
}
|
|
|
|
|
|
|
|
getTableFields() {
|
|
|
|
if (this._tableFields === undefined) {
|
|
|
|
this._tableFields = this.fields.filter(
|
|
|
|
field => field.fieldtype === 'Table'
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return this._tableFields;
|
|
|
|
}
|
|
|
|
|
|
|
|
getFormulaFields() {
|
|
|
|
if (this._formulaFields === undefined) {
|
|
|
|
this._formulaFields = this.fields.filter(field => field.formula);
|
|
|
|
}
|
|
|
|
return this._formulaFields;
|
|
|
|
}
|
|
|
|
|
|
|
|
hasFormula() {
|
|
|
|
if (this._hasFormula === undefined) {
|
|
|
|
this._hasFormula = false;
|
|
|
|
if (this.getFormulaFields().length) {
|
|
|
|
this._hasFormula = true;
|
|
|
|
} else {
|
|
|
|
for (let tablefield of this.getTableFields()) {
|
|
|
|
if (frappe.getMeta(tablefield.childtype).getFormulaFields().length) {
|
|
|
|
this._hasFormula = true;
|
|
|
|
break;
|
|
|
|
}
|
2018-01-12 12:25:07 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return this._hasFormula;
|
|
|
|
}
|
|
|
|
|
2019-10-05 21:47:09 +00:00
|
|
|
getBaseDocType() {
|
|
|
|
return this.basedOn || this.name;
|
|
|
|
}
|
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
async set(fieldname, value) {
|
|
|
|
this[fieldname] = value;
|
|
|
|
await this.trigger(fieldname);
|
|
|
|
}
|
|
|
|
|
|
|
|
get(fieldname) {
|
|
|
|
return this[fieldname];
|
|
|
|
}
|
|
|
|
|
|
|
|
getValidFields({ withChildren = true } = {}) {
|
|
|
|
if (!this._validFields) {
|
|
|
|
this._validFields = [];
|
|
|
|
this._validFieldsWithChildren = [];
|
|
|
|
|
|
|
|
const _add = field => {
|
|
|
|
this._validFields.push(field);
|
|
|
|
this._validFieldsWithChildren.push(field);
|
|
|
|
};
|
|
|
|
|
|
|
|
const doctype_fields = this.fields.map(field => field.fieldname);
|
|
|
|
|
|
|
|
// standard fields
|
|
|
|
for (let field of model.commonFields) {
|
|
|
|
if (
|
|
|
|
frappe.db.typeMap[field.fieldtype] &&
|
|
|
|
!doctype_fields.includes(field.fieldname)
|
|
|
|
) {
|
|
|
|
_add(field);
|
2018-02-22 11:21:42 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
2018-01-12 12:25:07 +00:00
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
if (this.isSubmittable) {
|
|
|
|
_add({
|
|
|
|
fieldtype: 'Check',
|
|
|
|
fieldname: 'submitted',
|
|
|
|
label: frappe._('Submitted')
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.isChild) {
|
|
|
|
// child fields
|
|
|
|
for (let field of model.childFields) {
|
|
|
|
if (
|
|
|
|
frappe.db.typeMap[field.fieldtype] &&
|
|
|
|
!doctype_fields.includes(field.fieldname)
|
|
|
|
) {
|
|
|
|
_add(field);
|
|
|
|
}
|
2018-01-12 12:25:07 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
} else {
|
|
|
|
// parent fields
|
|
|
|
for (let field of model.parentFields) {
|
|
|
|
if (
|
|
|
|
frappe.db.typeMap[field.fieldtype] &&
|
|
|
|
!doctype_fields.includes(field.fieldname)
|
|
|
|
) {
|
|
|
|
_add(field);
|
|
|
|
}
|
2018-02-01 11:07:36 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.isTree) {
|
|
|
|
// tree fields
|
|
|
|
for (let field of model.treeFields) {
|
|
|
|
if (
|
|
|
|
frappe.db.typeMap[field.fieldtype] &&
|
|
|
|
!doctype_fields.includes(field.fieldname)
|
|
|
|
) {
|
|
|
|
_add(field);
|
|
|
|
}
|
2018-02-12 12:01:31 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
2018-01-12 12:25:07 +00:00
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
// doctype fields
|
|
|
|
for (let field of this.fields) {
|
|
|
|
let include = frappe.db.typeMap[field.fieldtype];
|
2018-03-08 13:31:22 +00:00
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
if (include) {
|
|
|
|
_add(field);
|
2018-01-12 12:25:07 +00:00
|
|
|
}
|
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
// include tables if (withChildren = True)
|
|
|
|
if (!include && field.fieldtype === 'Table') {
|
|
|
|
this._validFieldsWithChildren.push(field);
|
2018-02-22 11:21:42 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (withChildren) {
|
|
|
|
return this._validFieldsWithChildren;
|
|
|
|
} else {
|
|
|
|
return this._validFields;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getKeywordFields() {
|
|
|
|
if (!this._keywordFields) {
|
|
|
|
this._keywordFields = this.keywordFields;
|
|
|
|
if (!(this._keywordFields && this._keywordFields.length && this.fields)) {
|
|
|
|
this._keywordFields = this.fields
|
|
|
|
.filter(field => field.fieldtype !== 'Table' && field.required)
|
|
|
|
.map(field => field.fieldname);
|
|
|
|
}
|
|
|
|
if (!(this._keywordFields && this._keywordFields.length)) {
|
|
|
|
this._keywordFields = ['name'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this._keywordFields;
|
|
|
|
}
|
|
|
|
|
2019-10-05 21:47:09 +00:00
|
|
|
getQuickEditFields() {
|
2019-10-19 14:37:24 +00:00
|
|
|
if (this.quickEditFields) {
|
|
|
|
return this.quickEditFields.map(fieldname => this.getField(fieldname));
|
|
|
|
}
|
|
|
|
return this.getFieldsWith({ required: 1 });
|
2019-10-05 21:47:09 +00:00
|
|
|
}
|
|
|
|
|
2019-09-17 08:14:09 +00:00
|
|
|
validateSelect(field, value) {
|
|
|
|
let options = field.options;
|
|
|
|
if (!options) return;
|
|
|
|
|
|
|
|
if (typeof options === 'string') {
|
|
|
|
// values given as string
|
|
|
|
options = field.options.split('\n');
|
|
|
|
}
|
|
|
|
if (!options.includes(value)) {
|
|
|
|
throw new frappe.errors.ValueError(
|
|
|
|
`${value} must be one of ${options.join(', ')}`
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
|
|
|
|
async trigger(event, params = {}) {
|
|
|
|
Object.assign(params, {
|
|
|
|
doc: this,
|
|
|
|
name: event
|
|
|
|
});
|
|
|
|
|
|
|
|
await super.trigger(event, params);
|
|
|
|
}
|
|
|
|
|
|
|
|
setDefaultIndicators() {
|
|
|
|
if (!this.indicators) {
|
|
|
|
if (this.isSubmittable) {
|
|
|
|
this.indicators = {
|
|
|
|
key: 'submitted',
|
|
|
|
colors: {
|
|
|
|
0: indicatorColor.GRAY,
|
|
|
|
1: indicatorColor.BLUE
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
getIndicatorColor(doc) {
|
|
|
|
if (frappe.isDirty(this.name, doc.name)) {
|
|
|
|
return indicatorColor.ORANGE;
|
|
|
|
} else {
|
|
|
|
if (this.indicators) {
|
|
|
|
let value = doc[this.indicators.key];
|
|
|
|
if (value) {
|
|
|
|
return this.indicators.colors[value] || indicatorColor.GRAY;
|
2018-02-22 11:21:42 +00:00
|
|
|
} else {
|
2019-09-17 08:14:09 +00:00
|
|
|
return indicatorColor.GRAY;
|
2018-02-22 11:21:42 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
} else {
|
|
|
|
return indicatorColor.GRAY;
|
|
|
|
}
|
2018-02-22 11:21:42 +00:00
|
|
|
}
|
2019-09-17 08:14:09 +00:00
|
|
|
}
|
|
|
|
};
|