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;