449 lines
14 KiB
JavaScript
Executable File
449 lines
14 KiB
JavaScript
Executable File
const ObjectUtils = {
|
|
equals(obj1, obj2, field) {
|
|
if (field) return this.resolveFieldData(obj1, field) === this.resolveFieldData(obj2, field);
|
|
else return this.deepEquals(obj1, obj2);
|
|
},
|
|
|
|
deepEquals(a, b) {
|
|
if (a === b) return true;
|
|
|
|
if (a && b && typeof a == 'object' && typeof b == 'object') {
|
|
var arrA = Array.isArray(a),
|
|
arrB = Array.isArray(b),
|
|
i,
|
|
length,
|
|
key;
|
|
|
|
if (arrA && arrB) {
|
|
length = a.length;
|
|
if (length != b.length) return false;
|
|
for (i = length; i-- !== 0; ) if (!this.deepEquals(a[i], b[i])) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
if (arrA != arrB) return false;
|
|
|
|
var dateA = a instanceof Date,
|
|
dateB = b instanceof Date;
|
|
|
|
if (dateA != dateB) return false;
|
|
if (dateA && dateB) return a.getTime() == b.getTime();
|
|
|
|
var regexpA = a instanceof RegExp,
|
|
regexpB = b instanceof RegExp;
|
|
|
|
if (regexpA != regexpB) return false;
|
|
if (regexpA && regexpB) return a.toString() == b.toString();
|
|
|
|
var keys = Object.keys(a);
|
|
|
|
length = keys.length;
|
|
|
|
if (length !== Object.keys(b).length) return false;
|
|
|
|
for (i = length; i-- !== 0; ) if (!Object.prototype.hasOwnProperty.call(b, keys[i])) return false;
|
|
|
|
for (i = length; i-- !== 0; ) {
|
|
key = keys[i];
|
|
if (!this.deepEquals(a[key], b[key])) return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
return a !== a && b !== b;
|
|
},
|
|
|
|
resolveFieldData(data, field) {
|
|
if (!data || !field) {
|
|
// short circuit if there is nothing to resolve
|
|
return null;
|
|
}
|
|
|
|
try {
|
|
const value = data[field];
|
|
|
|
if (this.isNotEmpty(value)) return value;
|
|
} catch {
|
|
// Performance optimization: https://github.com/primefaces/primereact/issues/4797
|
|
// do nothing and continue to other methods to resolve field data
|
|
}
|
|
|
|
if (Object.keys(data).length) {
|
|
if (this.isFunction(field)) {
|
|
return field(data);
|
|
} else if (field.indexOf('.') === -1) {
|
|
return data[field];
|
|
} else {
|
|
let fields = field.split('.');
|
|
let value = data;
|
|
|
|
for (var i = 0, len = fields.length; i < len; ++i) {
|
|
if (value == null) {
|
|
return null;
|
|
}
|
|
|
|
value = value[fields[i]];
|
|
}
|
|
|
|
return value;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
getItemValue(obj, ...params) {
|
|
return this.isFunction(obj) ? obj(...params) : obj;
|
|
},
|
|
|
|
filter(value, fields, filterValue) {
|
|
var filteredItems = [];
|
|
|
|
if (value) {
|
|
for (let item of value) {
|
|
for (let field of fields) {
|
|
if (String(this.resolveFieldData(item, field)).toLowerCase().indexOf(filterValue.toLowerCase()) > -1) {
|
|
filteredItems.push(item);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return filteredItems;
|
|
},
|
|
|
|
reorderArray(value, from, to) {
|
|
if (value && from !== to) {
|
|
if (to >= value.length) {
|
|
to %= value.length;
|
|
from %= value.length;
|
|
}
|
|
|
|
value.splice(to, 0, value.splice(from, 1)[0]);
|
|
}
|
|
},
|
|
|
|
findIndexInList(value, list) {
|
|
let index = -1;
|
|
|
|
if (list) {
|
|
for (let i = 0; i < list.length; i++) {
|
|
if (list[i] === value) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return index;
|
|
},
|
|
|
|
test(regex, str) {
|
|
if (regex) {
|
|
const match = regex.test(str);
|
|
|
|
regex.lastIndex = 0;
|
|
|
|
return match;
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
contains(value, list) {
|
|
if (value != null && list && list.length) {
|
|
for (let val of list) {
|
|
if (this.equals(value, val)) return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
},
|
|
|
|
insertIntoOrderedArray(item, index, arr, sourceArr) {
|
|
if (arr.length > 0) {
|
|
let injected = false;
|
|
|
|
for (let i = 0; i < arr.length; i++) {
|
|
let currentItemIndex = this.findIndexInList(arr[i], sourceArr);
|
|
|
|
if (currentItemIndex > index) {
|
|
arr.splice(i, 0, item);
|
|
injected = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!injected) {
|
|
arr.push(item);
|
|
}
|
|
} else {
|
|
arr.push(item);
|
|
}
|
|
},
|
|
|
|
removeAccents(str) {
|
|
if (str && str.search(/[\xC0-\xFF]/g) > -1) {
|
|
str = str
|
|
.replace(/[\xC0-\xC5]/g, 'A')
|
|
.replace(/[\xC6]/g, 'AE')
|
|
.replace(/[\xC7]/g, 'C')
|
|
.replace(/[\xC8-\xCB]/g, 'E')
|
|
.replace(/[\xCC-\xCF]/g, 'I')
|
|
.replace(/[\xD0]/g, 'D')
|
|
.replace(/[\xD1]/g, 'N')
|
|
.replace(/[\xD2-\xD6\xD8]/g, 'O')
|
|
.replace(/[\xD9-\xDC]/g, 'U')
|
|
.replace(/[\xDD]/g, 'Y')
|
|
.replace(/[\xDE]/g, 'P')
|
|
.replace(/[\xE0-\xE5]/g, 'a')
|
|
.replace(/[\xE6]/g, 'ae')
|
|
.replace(/[\xE7]/g, 'c')
|
|
.replace(/[\xE8-\xEB]/g, 'e')
|
|
.replace(/[\xEC-\xEF]/g, 'i')
|
|
.replace(/[\xF1]/g, 'n')
|
|
.replace(/[\xF2-\xF6\xF8]/g, 'o')
|
|
.replace(/[\xF9-\xFC]/g, 'u')
|
|
.replace(/[\xFE]/g, 'p')
|
|
.replace(/[\xFD\xFF]/g, 'y');
|
|
}
|
|
|
|
return str;
|
|
},
|
|
|
|
getVNodeProp(vnode, prop) {
|
|
if (vnode) {
|
|
let props = vnode.props;
|
|
|
|
if (props) {
|
|
let kebabProp = prop.replace(/([a-z])([A-Z])/g, '$1-$2').toLowerCase();
|
|
let propName = Object.prototype.hasOwnProperty.call(props, kebabProp) ? kebabProp : prop;
|
|
|
|
return vnode.type.extends.props[prop].type === Boolean && props[propName] === '' ? true : props[propName];
|
|
}
|
|
}
|
|
|
|
return null;
|
|
},
|
|
|
|
toFlatCase(str) {
|
|
// convert snake, kebab, camel and pascal cases to flat case
|
|
return this.isString(str) ? str.replace(/(-|_)/g, '').toLowerCase() : str;
|
|
},
|
|
|
|
toKebabCase(str) {
|
|
// convert snake, camel and pascal cases to kebab case
|
|
return this.isString(str)
|
|
? str
|
|
.replace(/(_)/g, '-')
|
|
.replace(/[A-Z]/g, (c, i) => (i === 0 ? c : '-' + c.toLowerCase()))
|
|
.toLowerCase()
|
|
: str;
|
|
},
|
|
|
|
toCapitalCase(str) {
|
|
return this.isString(str, { empty: false }) ? str[0].toUpperCase() + str.slice(1) : str;
|
|
},
|
|
|
|
isEmpty(value) {
|
|
return value === null || value === undefined || value === '' || (Array.isArray(value) && value.length === 0) || (!(value instanceof Date) && typeof value === 'object' && Object.keys(value).length === 0);
|
|
},
|
|
|
|
isNotEmpty(value) {
|
|
return !this.isEmpty(value);
|
|
},
|
|
|
|
isFunction(value) {
|
|
return !!(value && value.constructor && value.call && value.apply);
|
|
},
|
|
|
|
isObject(value, empty = true) {
|
|
return value instanceof Object && value.constructor === Object && (empty || Object.keys(value).length !== 0);
|
|
},
|
|
|
|
isDate(value) {
|
|
return value instanceof Date && value.constructor === Date;
|
|
},
|
|
|
|
isArray(value, empty = true) {
|
|
return Array.isArray(value) && (empty || value.length !== 0);
|
|
},
|
|
|
|
isString(value, empty = true) {
|
|
return typeof value === 'string' && (empty || value !== '');
|
|
},
|
|
|
|
isNumber(value) {
|
|
return !isNaN(value);
|
|
},
|
|
|
|
isPrintableCharacter(char = '') {
|
|
return this.isNotEmpty(char) && char.length === 1 && char.match(/\S| /);
|
|
},
|
|
|
|
/**
|
|
* Firefox-v103 does not currently support the "findLast" method. It is stated that this method will be supported with Firefox-v104.
|
|
* https://caniuse.com/mdn-javascript_builtins_array_findlast
|
|
*/
|
|
findLast(arr, callback) {
|
|
let item;
|
|
|
|
if (this.isNotEmpty(arr)) {
|
|
try {
|
|
item = arr.findLast(callback);
|
|
} catch {
|
|
item = [...arr].reverse().find(callback);
|
|
}
|
|
}
|
|
|
|
return item;
|
|
},
|
|
|
|
/**
|
|
* Firefox-v103 does not currently support the "findLastIndex" method. It is stated that this method will be supported with Firefox-v104.
|
|
* https://caniuse.com/mdn-javascript_builtins_array_findlastindex
|
|
*/
|
|
findLastIndex(arr, callback) {
|
|
let index = -1;
|
|
|
|
if (this.isNotEmpty(arr)) {
|
|
try {
|
|
index = arr.findLastIndex(callback);
|
|
} catch {
|
|
index = arr.lastIndexOf([...arr].reverse().find(callback));
|
|
}
|
|
}
|
|
|
|
return index;
|
|
},
|
|
|
|
sort(value1, value2, order = 1, comparator, nullSortOrder = 1) {
|
|
const result = this.compare(value1, value2, comparator, order);
|
|
let finalSortOrder = order;
|
|
|
|
// nullSortOrder == 1 means Excel like sort nulls at bottom
|
|
if (this.isEmpty(value1) || this.isEmpty(value2)) {
|
|
finalSortOrder = nullSortOrder === 1 ? order : nullSortOrder;
|
|
}
|
|
|
|
return finalSortOrder * result;
|
|
},
|
|
|
|
compare(value1, value2, comparator, order = 1) {
|
|
let result = -1;
|
|
const emptyValue1 = this.isEmpty(value1);
|
|
const emptyValue2 = this.isEmpty(value2);
|
|
|
|
if (emptyValue1 && emptyValue2) result = 0;
|
|
else if (emptyValue1) result = order;
|
|
else if (emptyValue2) result = -order;
|
|
else if (typeof value1 === 'string' && typeof value2 === 'string') result = comparator(value1, value2);
|
|
else result = value1 < value2 ? -1 : value1 > value2 ? 1 : 0;
|
|
|
|
return result;
|
|
},
|
|
|
|
localeComparator() {
|
|
//performance gain using Int.Collator. It is not recommended to use localeCompare against large arrays.
|
|
return new Intl.Collator(undefined, { numeric: true }).compare;
|
|
},
|
|
|
|
nestedKeys(obj = {}, parentKey = '') {
|
|
return Object.entries(obj).reduce((o, [key, value]) => {
|
|
const currentKey = parentKey ? `${parentKey}.${key}` : key;
|
|
|
|
this.isObject(value) ? (o = o.concat(this.nestedKeys(value, currentKey))) : o.push(currentKey);
|
|
|
|
return o;
|
|
}, []);
|
|
},
|
|
|
|
stringify(value, indent = 2, currentIndent = 0) {
|
|
const currentIndentStr = ' '.repeat(currentIndent);
|
|
const nextIndentStr = ' '.repeat(currentIndent + indent);
|
|
|
|
if (this.isArray(value)) {
|
|
return '[' + value.map((v) => this.stringify(v, indent, currentIndent + indent)).join(', ') + ']';
|
|
} else if (this.isDate(value)) {
|
|
return value.toISOString();
|
|
} else if (this.isFunction(value)) {
|
|
return value.toString();
|
|
} else if (this.isObject(value)) {
|
|
return (
|
|
'{\n' +
|
|
Object.entries(value)
|
|
.map(([k, v]) => `${nextIndentStr}${k}: ${this.stringify(v, indent, currentIndent + indent)}`)
|
|
.join(',\n') +
|
|
`\n${currentIndentStr}` +
|
|
'}'
|
|
);
|
|
} else {
|
|
return JSON.stringify(value);
|
|
}
|
|
},
|
|
|
|
css: {
|
|
setProperty(properties, key, value) {
|
|
properties.push(this.getDeclaration(key, value));
|
|
},
|
|
getDeclaration(key, value) {
|
|
return `${key}:${value};`;
|
|
},
|
|
getRule(selector, properties) {
|
|
return selector ? `${selector}{${properties}}` : '';
|
|
},
|
|
toUnit(value, variable = '') {
|
|
const excludedProperties = ['opacity', 'z-index', 'line-height', 'font-weight', 'flex', 'flex-grow', 'flex-shrink', 'order'];
|
|
|
|
if (!excludedProperties.some((property) => variable.endsWith(property))) {
|
|
const val = `${value}`.trim();
|
|
const valArr = val.split(' ');
|
|
|
|
return valArr.map((v) => (ObjectUtils.isNumber(v) ? `${v}px` : v)).join(' ');
|
|
}
|
|
|
|
return value;
|
|
},
|
|
toNormalizePrefix(prefix) {
|
|
return prefix.replaceAll(/ /g, '').replace(/[^\w]/g, '-');
|
|
},
|
|
toNormalizeVariable(prefix = '', variable = '') {
|
|
return this.toNormalizePrefix(`${ObjectUtils.isNotEmpty(prefix) && ObjectUtils.isNotEmpty(variable) ? `${prefix}-` : prefix}${variable}`);
|
|
},
|
|
getVariableName(prefix = '', variable = '') {
|
|
return `--${this.toNormalizeVariable(prefix, variable)}`;
|
|
},
|
|
getVariableValue(value, variable = '', prefix = '', excludedKeyRegexes = []) {
|
|
if (ObjectUtils.isString(value)) {
|
|
const regex = /{([^}]*)}/g;
|
|
const val = value.trim();
|
|
|
|
if (ObjectUtils.test(regex, val)) {
|
|
const _val = val.replaceAll(regex, (v) => {
|
|
const path = v.replace(/{|}/g, '');
|
|
const keys = path.split('.').filter((_v) => !excludedKeyRegexes.some((_r) => this.test(_r, _v)));
|
|
|
|
return `var(${this.getVariableName(prefix, ObjectUtils.toKebabCase(keys.join('-')))})`;
|
|
});
|
|
|
|
const calculationRegex = /(\d+\s+[\+\-\*\/]\s+\d+)/g;
|
|
const cleanedVarRegex = /var\([^)]+\)/g;
|
|
|
|
return ObjectUtils.test(calculationRegex, _val.replace(cleanedVarRegex, '0')) ? `calc(${_val})` : _val;
|
|
}
|
|
|
|
return this.toUnit(val, variable);
|
|
} else if (ObjectUtils.isNumber(value)) {
|
|
return this.toUnit(value, variable);
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
}
|
|
};
|
|
|
|
export default ObjectUtils;
|