mirror of
https://github.com/espocrm/espocrm.git
synced 2026-06-28 06:56:05 +00:00
510 lines
16 KiB
JavaScript
510 lines
16 KiB
JavaScript
/************************************************************************
|
|
* This file is part of EspoCRM.
|
|
*
|
|
* EspoCRM - Open Source CRM application.
|
|
* Copyright (C) 2014-2023 Yurii Kuznietsov, Taras Machyshyn, Oleksii Avramenko
|
|
* Website: https://www.espocrm.com
|
|
*
|
|
* EspoCRM is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* EspoCRM is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with EspoCRM. If not, see http://www.gnu.org/licenses/.
|
|
*
|
|
* The interactive user interfaces in modified source and object code versions
|
|
* of this program must display Appropriate Legal Notices, as required under
|
|
* Section 5 of the GNU General Public License version 3.
|
|
*
|
|
* In accordance with Section 7(b) of the GNU General Public License version 3,
|
|
* these Appropriate Legal Notices must retain the display of the "EspoCRM" word.
|
|
************************************************************************/
|
|
|
|
define('views/fields/enum', ['views/fields/base', 'ui/multi-select', 'ui/select'],
|
|
function (Dep, /** module:ui/multi-select*/MultiSelect, /** module:ui/select*/Select) {
|
|
|
|
/**
|
|
* An enum field (select-box).
|
|
*
|
|
* @class
|
|
* @name Class
|
|
* @extends module:views/fields/base.Class
|
|
* @memberOf module:views/fields/enum
|
|
*/
|
|
return Dep.extend(/** @lends module:views/fields/enum.Class# */{
|
|
|
|
type: 'enum',
|
|
|
|
listTemplate: 'fields/enum/list',
|
|
listLinkTemplate: 'fields/enum/list-link',
|
|
detailTemplate: 'fields/enum/detail',
|
|
editTemplate: 'fields/enum/edit',
|
|
searchTemplate: 'fields/enum/search',
|
|
|
|
translatedOptions: null,
|
|
|
|
/**
|
|
* @todo Remove? Always treat as true.
|
|
*/
|
|
fetchEmptyValueAsNull: true,
|
|
|
|
searchTypeList: [
|
|
'anyOf',
|
|
'noneOf',
|
|
'isEmpty',
|
|
'isNotEmpty',
|
|
],
|
|
|
|
validationElementSelector: '.selectize-control',
|
|
|
|
data: function () {
|
|
let data = Dep.prototype.data.call(this);
|
|
|
|
data.translatedOptions = this.translatedOptions;
|
|
|
|
let value = this.model.get(this.name);
|
|
|
|
if (this.isReadMode() && this.styleMap) {
|
|
data.style = this.styleMap[value || ''] || 'default';
|
|
}
|
|
|
|
if (this.isReadMode()) {
|
|
if (this.params.displayAsLabel && data.style && data.style !== 'default') {
|
|
data.class = 'label label-md label';
|
|
} else {
|
|
data.class = 'text';
|
|
}
|
|
}
|
|
|
|
let translationKey = value || '';
|
|
|
|
if (
|
|
typeof value !== 'undefined' && value !== null && value !== ''
|
|
||
|
|
translationKey === '' && (
|
|
translationKey in (this.translatedOptions || {}) &&
|
|
(this.translatedOptions || {})[translationKey] !== ''
|
|
)
|
|
) {
|
|
data.isNotEmpty = true;
|
|
}
|
|
|
|
data.valueIsSet = this.model.has(this.name);
|
|
|
|
return data;
|
|
},
|
|
|
|
setup: function () {
|
|
if (!this.params.options) {
|
|
let methodName = 'get' + Espo.Utils.upperCaseFirst(this.name) + 'Options';
|
|
|
|
if (typeof this.model[methodName] === 'function') {
|
|
this.params.options = this.model[methodName].call(this.model);
|
|
}
|
|
}
|
|
|
|
let optionsPath = this.params.optionsPath;
|
|
/** @type {?string} */
|
|
let optionsReference = this.params.optionsReference;
|
|
|
|
if (!optionsPath && optionsReference) {
|
|
let [refEntityType, refField] = optionsReference.split('.');
|
|
|
|
optionsPath = `entityDefs.${refEntityType}.fields.${refField}.options`;
|
|
}
|
|
|
|
if (optionsPath) {
|
|
this.params.options = Espo.Utils.clone(this.getMetadata().get(optionsPath)) || [];
|
|
}
|
|
|
|
this.styleMap = this.params.style || this.model.getFieldParam(this.name, 'style') || {};
|
|
|
|
this.setupOptions();
|
|
|
|
if ('translatedOptions' in this.options) {
|
|
this.translatedOptions = this.options.translatedOptions;
|
|
}
|
|
|
|
if ('translatedOptions' in this.params) {
|
|
this.translatedOptions = this.params.translatedOptions;
|
|
}
|
|
|
|
this.setupTranslation();
|
|
|
|
if (this.translatedOptions === null) {
|
|
this.translatedOptions = this.getLanguage()
|
|
.translate(this.name, 'options', this.model.name) || {};
|
|
|
|
if (this.translatedOptions === this.name) {
|
|
this.translatedOptions = null;
|
|
}
|
|
}
|
|
|
|
if (this.params.isSorted && this.translatedOptions) {
|
|
this.params.options = Espo.Utils.clone(this.params.options) || [];
|
|
|
|
this.params.options = this.params.options.sort((v1, v2) => {
|
|
return (this.translatedOptions[v1] || v1)
|
|
.localeCompare(this.translatedOptions[v2] || v2);
|
|
});
|
|
}
|
|
|
|
if (this.options.customOptionList) {
|
|
this.setOptionList(this.options.customOptionList);
|
|
}
|
|
},
|
|
|
|
setupTranslation: function () {
|
|
let translation = this.params.translation;
|
|
/** @type {?string} */
|
|
let optionsReference = this.params.optionsReference;
|
|
|
|
if (!translation && optionsReference) {
|
|
let [refEntityType, refField] = optionsReference.split('.');
|
|
|
|
translation = `${refEntityType}.options.${refField}`;
|
|
}
|
|
|
|
if (!translation) {
|
|
return;
|
|
}
|
|
|
|
let translationObj;
|
|
|
|
let arr = translation.split('.');
|
|
let pointer = this.getLanguage().data;
|
|
|
|
arr.forEach(key => {
|
|
if (key in pointer) {
|
|
pointer = pointer[key];
|
|
translationObj = pointer;
|
|
}
|
|
});
|
|
|
|
this.translatedOptions = null;
|
|
|
|
let translatedOptions = {};
|
|
|
|
if (!this.params.options) {
|
|
return;
|
|
}
|
|
|
|
this.params.options.forEach(item => {
|
|
if (typeof translationObj === 'object' && item in translationObj) {
|
|
translatedOptions[item] = translationObj[item];
|
|
}
|
|
else if (
|
|
Array.isArray(translationObj) &&
|
|
typeof item === 'number' &&
|
|
typeof translationObj[item] !== 'undefined'
|
|
) {
|
|
translatedOptions[item.toString()] = translationObj[item];
|
|
}
|
|
else {
|
|
translatedOptions[item] = item;
|
|
}
|
|
});
|
|
|
|
let value = this.model.get(this.name);
|
|
|
|
if ((value || value === '') && !(value in translatedOptions)) {
|
|
if (typeof translationObj === 'object' && value in translationObj) {
|
|
translatedOptions[value] = translationObj[value];
|
|
}
|
|
}
|
|
|
|
this.translatedOptions = translatedOptions;
|
|
},
|
|
|
|
/**
|
|
* Set up options.
|
|
*/
|
|
setupOptions: function () {},
|
|
|
|
/**
|
|
* Set an option list.
|
|
*
|
|
* @param {string[]} optionList An option list.
|
|
*/
|
|
setOptionList: function (optionList) {
|
|
let previousOptions = this.params.options;
|
|
|
|
if (!this.originalOptionList) {
|
|
this.originalOptionList = this.params.options;
|
|
}
|
|
|
|
let newOptions = Espo.Utils.clone(optionList) || [];
|
|
|
|
this.params.options = newOptions;
|
|
|
|
let isChanged = !_(previousOptions).isEqual(optionList);
|
|
|
|
if (!this.isEditMode() || !isChanged) {
|
|
return;
|
|
}
|
|
|
|
let triggerChange = false;
|
|
let currentValue = this.model.get(this.name);
|
|
|
|
if (!newOptions.includes(currentValue) && this.isReady) {
|
|
this.model.set(this.name, newOptions[0] ?? null, {silent: true});
|
|
|
|
triggerChange = true;
|
|
}
|
|
|
|
this.reRender()
|
|
.then(() => {
|
|
if (triggerChange) {
|
|
this.trigger('change');
|
|
}
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Reset a previously set option list.
|
|
*/
|
|
resetOptionList: function () {
|
|
if (!this.originalOptionList) {
|
|
return;
|
|
}
|
|
|
|
let previousOptions = this.params.options;
|
|
|
|
this.params.options = Espo.Utils.clone(this.originalOptionList);
|
|
|
|
let isChanged = !_(previousOptions).isEqual(this.originalOptionList);
|
|
|
|
if (!this.isEditMode() || !isChanged) {
|
|
return;
|
|
}
|
|
|
|
if (this.isRendered()) {
|
|
this.reRender();
|
|
}
|
|
},
|
|
|
|
setupSearch: function () {
|
|
this.events = _.extend({
|
|
'change select.search-type': (e) => {
|
|
this.handleSearchType($(e.currentTarget).val());
|
|
},
|
|
}, this.events || {});
|
|
},
|
|
|
|
handleSearchType: function (type) {
|
|
var $inputContainer = this.$el.find('div.input-container');
|
|
|
|
if (~['anyOf', 'noneOf'].indexOf(type)) {
|
|
$inputContainer.removeClass('hidden');
|
|
} else {
|
|
$inputContainer.addClass('hidden');
|
|
}
|
|
},
|
|
|
|
afterRender: function () {
|
|
Dep.prototype.afterRender.call(this);
|
|
|
|
if (this.isSearchMode()) {
|
|
this.$element = this.$el.find('.main-element');
|
|
|
|
let type = this.$el.find('select.search-type').val();
|
|
|
|
this.handleSearchType(type);
|
|
|
|
let valueList = this.getSearchParamsData().valueList || this.searchParams.value || [];
|
|
|
|
this.$element.val(valueList.join(':,:'));
|
|
|
|
let items = [];
|
|
|
|
(this.params.options || []).forEach(value => {
|
|
let label = this.getLanguage().translateOption(value, this.name, this.scope);
|
|
|
|
if (this.translatedOptions) {
|
|
if (value in this.translatedOptions) {
|
|
label = this.translatedOptions[value];
|
|
}
|
|
}
|
|
|
|
if (label === '') {
|
|
return;
|
|
}
|
|
|
|
items.push({
|
|
value: value,
|
|
text: label,
|
|
});
|
|
});
|
|
|
|
/** @type {module:ui/multi-select~Options} */
|
|
let multiSelectOptions = {
|
|
items: items,
|
|
delimiter: ':,:',
|
|
matchAnyWord: true,
|
|
};
|
|
|
|
MultiSelect.init(this.$element, multiSelectOptions);
|
|
|
|
this.$el.find('.selectize-dropdown-content').addClass('small');
|
|
this.$el.find('select.search-type').on('change', () => this.trigger('change'));
|
|
this.$element.on('change', () => this.trigger('change'));
|
|
}
|
|
|
|
if (this.isEditMode() || this.isSearchMode()) {
|
|
Select.init(this.$element, {matchAnyWord: true});
|
|
}
|
|
},
|
|
|
|
focusOnInlineEdit: function () {
|
|
Select.focus(this.$element);
|
|
},
|
|
|
|
validateRequired: function () {
|
|
if (this.isRequired()) {
|
|
if (!this.model.get(this.name)) {
|
|
let msg = this.translate('fieldIsRequired', 'messages')
|
|
.replace('{field}', this.getLabelText());
|
|
|
|
this.showValidationMessage(msg);
|
|
|
|
return true;
|
|
}
|
|
}
|
|
},
|
|
|
|
fetch: function () {
|
|
let value = this.$element.val();
|
|
|
|
if (this.fetchEmptyValueAsNull && !value) {
|
|
value = null;
|
|
}
|
|
|
|
let data = {};
|
|
|
|
data[this.name] = value;
|
|
|
|
return data;
|
|
},
|
|
|
|
parseItemForSearch: function (item) {
|
|
return item;
|
|
},
|
|
|
|
fetchSearch: function () {
|
|
let type = this.fetchSearchType();
|
|
|
|
let list = this.$element.val().split(':,:');
|
|
|
|
if (list.length === 1 && list[0] === '') {
|
|
list = [];
|
|
}
|
|
|
|
list.forEach((item, i) => {
|
|
list[i] = this.parseItemForSearch(item);
|
|
});
|
|
|
|
if (type === 'anyOf') {
|
|
if (list.length === 0) {
|
|
return {
|
|
type: 'any',
|
|
data: {
|
|
type: 'anyOf',
|
|
valueList: list,
|
|
},
|
|
};
|
|
}
|
|
|
|
return {
|
|
type: 'in',
|
|
value: list,
|
|
data: {
|
|
type: 'anyOf',
|
|
valueList: list,
|
|
},
|
|
};
|
|
}
|
|
|
|
if (type === 'noneOf') {
|
|
if (list.length === 0) {
|
|
return {
|
|
type: 'any',
|
|
data: {
|
|
type: 'noneOf',
|
|
valueList: list,
|
|
},
|
|
};
|
|
}
|
|
|
|
return {
|
|
type: 'or',
|
|
value: [
|
|
{
|
|
type: 'isNull',
|
|
attribute: this.name,
|
|
},
|
|
{
|
|
type: 'notIn',
|
|
value: list,
|
|
attribute: this.name,
|
|
},
|
|
],
|
|
data: {
|
|
type: 'noneOf',
|
|
valueList: list,
|
|
},
|
|
};
|
|
}
|
|
|
|
if (type === 'isEmpty') {
|
|
return {
|
|
type: 'or',
|
|
value: [
|
|
{
|
|
type: 'isNull',
|
|
attribute: this.name,
|
|
},
|
|
{
|
|
type: 'equals',
|
|
value: '',
|
|
attribute: this.name,
|
|
}
|
|
],
|
|
data: {
|
|
type: 'isEmpty',
|
|
},
|
|
};
|
|
}
|
|
|
|
if (type === 'isNotEmpty') {
|
|
return {
|
|
type: 'and',
|
|
value: [
|
|
{
|
|
type: 'isNotNull',
|
|
attribute: this.name,
|
|
},
|
|
{
|
|
type: 'notEquals',
|
|
value: '',
|
|
attribute: this.name,
|
|
},
|
|
],
|
|
data: {
|
|
type: 'isNotEmpty',
|
|
},
|
|
};
|
|
}
|
|
},
|
|
|
|
getSearchType: function () {
|
|
return this.getSearchParamsData().type || 'anyOf';
|
|
},
|
|
});
|
|
});
|