From b0e572b16cd2dee7039a4b894d5259c3c4737ec2 Mon Sep 17 00:00:00 2001 From: thefalconx33 Date: Tue, 17 Sep 2019 13:44:09 +0530 Subject: [PATCH] Dynamic Labels --- model/document.js | 2 +- model/meta.js | 454 ++++++++++++++++-------------- ui/components/Form/FormLayout.vue | 19 +- ui/components/controls/Base.vue | 17 +- utils/format.js | 10 +- utils/numberFormat.js | 29 +- 6 files changed, 259 insertions(+), 272 deletions(-) diff --git a/model/document.js b/model/document.js index 2b82c1d6..42eed550 100644 --- a/model/document.js +++ b/model/document.js @@ -358,7 +358,7 @@ module.exports = class BaseDocument extends Observable { // helper functions getSum(tablefield, childfield) { return this[tablefield] - .map(d => frappe.parseNumber(d[childfield]) || 0) + .map(d => d[childfield] || 0) .reduce((a, b) => a + b, 0); } diff --git a/model/meta.js b/model/meta.js index 2d3a9c09..20aaba78 100644 --- a/model/meta.js +++ b/model/meta.js @@ -1,232 +1,254 @@ const BaseDocument = require('./document'); const frappe = require('frappejs'); -const model = require('./index') +const model = require('./index'); const indicatorColor = require('frappejs/ui/constants/indicators'); module.exports = class BaseMeta extends BaseDocument { - constructor(data) { - super(data); - this.setDefaultIndicators(); - if (this.setupMeta) { - this.setupMeta(); + constructor(data) { + super(data); + this.setDefaultIndicators(); + if (this.setupMeta) { + this.setupMeta(); + } + if (!this.titleField) { + this.titleField = 'name'; + } + } + + 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; + } } - if (!this.titleField) { - this.titleField = 'name'; + } + } + return this._hasFormula; + } + + 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); } - } + } - 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) { - return this.getField(fieldname).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; - } - } - } - } - return this._hasFormula; - } - - 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); - } - } - - 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); - } - } - } else { - // parent fields - for (let field of model.parentFields) { - if (frappe.db.typeMap[field.fieldtype] && !doctype_fields.includes(field.fieldname)) { - _add(field); - } - } - } - - if (this.isTree) { - // tree fields - for (let field of model.treeFields) { - if (frappe.db.typeMap[field.fieldtype] && !doctype_fields.includes(field.fieldname)) { - _add(field); - } - } - } - - // doctype fields - for (let field of this.fields) { - let include = frappe.db.typeMap[field.fieldtype]; - - if (include) { - _add(field); - } - - // include tables if (withChildren = True) - if (!include && field.fieldtype === 'Table') { - this._validFieldsWithChildren.push(field); - } - } - } - - 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; - } - - 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 + if (this.isSubmittable) { + _add({ + fieldtype: 'Check', + fieldname: 'submitted', + label: frappe._('Submitted') }); + } - await super.trigger(event, params); - } - - setDefaultIndicators() { - if (!this.indicators) { - if (this.isSubmittable) { - this.indicators = { - key: 'submitted', - colors: { - 0: indicatorColor.GRAY, - 1: indicatorColor.BLUE - } - } - } + if (this.isChild) { + // child fields + for (let field of model.childFields) { + if ( + frappe.db.typeMap[field.fieldtype] && + !doctype_fields.includes(field.fieldname) + ) { + _add(field); + } } + } else { + // parent fields + for (let field of model.parentFields) { + if ( + frappe.db.typeMap[field.fieldtype] && + !doctype_fields.includes(field.fieldname) + ) { + _add(field); + } + } + } + + if (this.isTree) { + // tree fields + for (let field of model.treeFields) { + if ( + frappe.db.typeMap[field.fieldtype] && + !doctype_fields.includes(field.fieldname) + ) { + _add(field); + } + } + } + + // doctype fields + for (let field of this.fields) { + let include = frappe.db.typeMap[field.fieldtype]; + + if (include) { + _add(field); + } + + // include tables if (withChildren = True) + if (!include && field.fieldtype === 'Table') { + this._validFieldsWithChildren.push(field); + } + } } - getIndicatorColor(doc) { - if (frappe.isDirty(this.name, doc.name)) { - return indicatorColor.ORANGE; + 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; + } + + 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; } else { - if (this.indicators) { - let value = doc[this.indicators.key]; - if (value) { - return this.indicators.colors[value] || indicatorColor.GRAY; - } else { - return indicatorColor.GRAY; - } - } else { - return indicatorColor.GRAY; - } + return indicatorColor.GRAY; } + } else { + return indicatorColor.GRAY; + } } -} \ No newline at end of file + } +}; diff --git a/ui/components/Form/FormLayout.vue b/ui/components/Form/FormLayout.vue index 96123da2..4befcbe6 100644 --- a/ui/components/Form/FormLayout.vue +++ b/ui/components/Form/FormLayout.vue @@ -8,9 +8,10 @@ >
{ + control.docfield.label = control.docfield.getLabel + ? control.docfield.getLabel(this.doc) + : control.docfield.label; + }); + }, getDocField(fieldname) { return this.fields.find(df => df.fieldname === fieldname); }, @@ -73,13 +81,6 @@ export default { return true; }, - setLabelOptions() { - this.fields.forEach(field => { - if (field.labelOption) { - field.labelOption = field.labelOption(this.doc); - } - }); - }, updateDoc(fieldname, value) { this.doc.set(fieldname, value); this.$emit('updateDoc', { diff --git a/ui/components/controls/Base.vue b/ui/components/controls/Base.vue index 9bcc6478..2e814e0e 100644 --- a/ui/components/controls/Base.vue +++ b/ui/components/controls/Base.vue @@ -6,6 +6,11 @@ export default { } return this.getWrapperElement(h); }, + data() { + return { + label: this.docfield.label + }; + }, props: { docfield: Object, value: [String, Number, Array, FileList], @@ -68,23 +73,13 @@ export default { ]; }, getLabelElement(h) { - const hasLabelOptions = Boolean(this.docfield.labelOption); - let label = this.docfield.label; - - if (hasLabelOptions) { - const replaceableKeys = Object.keys(this.docfield.labelOption); - for (let key of replaceableKeys) { - label = label.replace(key.toString(), this.docfield.labelOption[key]); - } - } - return h('label', { class: [this.labelClass, 'text-muted'], attrs: { for: this.id }, domProps: { - textContent: label + textContent: this.label } }); }, diff --git a/utils/format.js b/utils/format.js index 4b714fa8..10b0b208 100644 --- a/utils/format.js +++ b/utils/format.js @@ -9,15 +9,7 @@ module.exports = { field = { fieldtype: field }; } if (field.fieldtype === 'Currency') { - if (field.currencyInfo) { - value = numberFormat.formatNumber( - value, - field.currencyInfo.numberFormat, - field.currencyInfo.symbol - ); - } else { - value = numberFormat.formatNumber(value); - } + value = numberFormat.formatNumber(value); } else if (field.fieldtype === 'Text') { // value = markdown.makeHtml(value || ''); } else if (field.fieldtype === 'Date') { diff --git a/utils/numberFormat.js b/utils/numberFormat.js index bd561ac3..a7a54ee6 100644 --- a/utils/numberFormat.js +++ b/utils/numberFormat.js @@ -14,39 +14,21 @@ const numberFormats = { module.exports = { // parse a formatted number string // from "4,555,000.34" -> 4555000.34 - parseNumber(number, format, symbol) { + parseNumber(number, format = '#,###.##') { if (!number) { return 0; } - if (!format) { - format = frappe.AccountingSettings.numberFormat || '#,###.##'; - } - if (!symbol) { - symbol = frappe.AccountingSettings.symbol || ''; - } if (typeof number === 'number') { return number; } - - if (isNaN(parseFloat(number))) { - // remove symbol - number = number.substr(2); - } const info = this.getFormatInfo(format); - return parseFloat(this.removeSeparator(number, info.groupSep)); }, - formatNumber(number, format, symbol, precision = null) { + formatNumber(number, format = '#,###.##', precision = null) { if (!number) { number = 0; } - if (!format) { - format = frappe.AccountingSettings.numberFormat || '#,###.##'; - } - if (!symbol) { - symbol = frappe.AccountingSettings.symbol || ''; - } let info = this.getFormatInfo(format); if (precision) { info.precision = precision; @@ -96,12 +78,7 @@ module.exports = { parts[1] = parts[1] && info.fractionSep ? info.fractionSep + parts[1] : ''; // join - return ( - (symbol.length ? `${symbol} ` : '') + - (is_negative ? '-' : '') + - parts[0] + - parts[1] - ); + return (is_negative ? '-' : '') + parts[0] + parts[1]; }, getFormatInfo(format) {