UNPKG

devextreme

Version:

HTML5 JavaScript Component Suite for Responsive Web Development

282 lines (265 loc) • 8.93 kB
/** * DevExtreme (esm/__internal/data/m_array_utils.js) * Version: 24.2.6 * Build date: Mon Mar 17 2025 * * Copyright (c) 2012 - 2025 Developer Express Inc. ALL RIGHTS RESERVED * Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/ */ import _extends from "@babel/runtime/helpers/esm/extends"; import { errors } from "../../common/data/errors"; import { keysEqual, rejectedPromise, trivialPromise } from "../../common/data/utils"; import config from "../../core/config"; import Guid from "../../core/guid"; import { compileGetter } from "../../core/utils/data"; import { extend } from "../../core/utils/extend"; import { deepExtendArraySafe } from "../../core/utils/object"; import { isDefined, isEmptyObject, isObject, isPlainObject } from "../../core/utils/type"; function hasKey(target, keyOrKeys) { let key; const keys = "string" === typeof keyOrKeys ? keyOrKeys.split() : keyOrKeys.slice(); while (keys.length) { key = keys.shift(); if (key in target) { return true } } return false } function findItems(keyInfo, items, key, groupCount) { let childItems; let result; if (groupCount) { for (let i = 0; i < items.length; i++) { childItems = items[i].items || items[i].collapsedItems || []; result = findItems(keyInfo, childItems || [], key, groupCount - 1); if (result) { return result } } } else if (indexByKey(keyInfo, items, key) >= 0) { return items } } function getItems(keyInfo, items, key, groupCount) { if (groupCount) { return findItems(keyInfo, items, key, groupCount) || [] } return items } function generateDataByKeyMap(keyInfo, array) { if (keyInfo.key() && (!array._dataByKeyMap || array._dataByKeyMapLength !== array.length)) { const dataByKeyMap = {}; const arrayLength = array.length; for (let i = 0; i < arrayLength; i++) { dataByKeyMap[JSON.stringify(keyInfo.keyOf(array[i]))] = array[i] } array._dataByKeyMap = dataByKeyMap; array._dataByKeyMapLength = arrayLength } } function getCacheValue(array, key) { if (array._dataByKeyMap) { return array._dataByKeyMap[JSON.stringify(key)] } } function getHasKeyCacheValue(array, key) { if (array._dataByKeyMap) { return array._dataByKeyMap[JSON.stringify(key)] } return true } function setDataByKeyMapValue(array, key, data) { if (array._dataByKeyMap) { array._dataByKeyMap[JSON.stringify(key)] = data; array._dataByKeyMapLength += data ? 1 : -1 } } function cloneInstanceWithChangedPaths(instance, changes, clonedInstances) { clonedInstances = clonedInstances || new WeakMap; const result = instance ? Object.create(Object.getPrototypeOf(instance)) : {}; if (instance) { clonedInstances.set(instance, result) } const instanceWithoutPrototype = _extends({}, instance); deepExtendArraySafe(result, instanceWithoutPrototype, true, true, true); for (const name in instanceWithoutPrototype) { const value = instanceWithoutPrototype[name]; const change = null === changes || void 0 === changes ? void 0 : changes[name]; if (isObject(value) && !isPlainObject(value) && isObject(change) && !clonedInstances.has(value)) { result[name] = cloneInstanceWithChangedPaths(value, change, clonedInstances) } } for (const name in result) { const prop = result[name]; if (isObject(prop) && clonedInstances.has(prop)) { result[name] = clonedInstances.get(prop) } } return result } function createObjectWithChanges(target, changes) { const result = cloneInstanceWithChangedPaths(target, changes); return deepExtendArraySafe(result, changes, true, true, true) } function applyBatch(_ref) { let { keyInfo: keyInfo, data: data, changes: changes, groupCount: groupCount, useInsertIndex: useInsertIndex, immutable: immutable, disableCache: disableCache, logError: logError, skipCopying: skipCopying } = _ref; const resultItems = true === immutable ? [...data] : data; changes.forEach((item => { const items = "insert" === item.type ? resultItems : getItems(keyInfo, resultItems, item.key, groupCount); !disableCache && generateDataByKeyMap(keyInfo, items); switch (item.type) { case "update": update(keyInfo, items, item.key, item.data, true, immutable, logError); break; case "insert": insert(keyInfo, items, item.data, useInsertIndex && isDefined(item.index) ? item.index : -1, true, logError, skipCopying); break; case "remove": remove(keyInfo, items, item.key, true, logError) } })); return resultItems } function getErrorResult(isBatch, logError, errorCode) { return !isBatch ? rejectedPromise(errors.Error(errorCode)) : logError && errors.log(errorCode) } function applyChanges(data, changes) { let options = arguments.length > 2 && void 0 !== arguments[2] ? arguments[2] : {}; const { keyExpr: keyExpr = "id", immutable: immutable = true } = options; const keyGetter = compileGetter(keyExpr); const keyInfo = { key: () => keyExpr, keyOf: obj => keyGetter(obj) }; return applyBatch({ keyInfo: keyInfo, data: data, changes: changes, immutable: immutable, disableCache: true, logError: true }) } function update(keyInfo, array, key, data, isBatch, immutable, logError) { let target; const keyExpr = keyInfo.key(); if (keyExpr) { if (hasKey(data, keyExpr) && !keysEqual(keyExpr, key, keyInfo.keyOf(data))) { return getErrorResult(isBatch, logError, "E4017") } target = getCacheValue(array, key); if (!target) { const index = indexByKey(keyInfo, array, key); if (index < 0) { return getErrorResult(isBatch, logError, "E4009") } target = array[index]; if (true === immutable && isDefined(target)) { const newTarget = createObjectWithChanges(target, data); array[index] = newTarget; return !isBatch && trivialPromise(newTarget, key) } } } else { target = key } deepExtendArraySafe(target, data, true, false, true, true); if (!isBatch) { if (config().useLegacyStoreResult) { return trivialPromise(key, data) } return trivialPromise(target, key) } } function insert(keyInfo, array, data, index, isBatch, logError, skipCopying) { let keyValue; const keyExpr = keyInfo.key(); const obj = isPlainObject(data) && !skipCopying ? extend({}, data) : data; if (keyExpr) { keyValue = keyInfo.keyOf(obj); if (void 0 === keyValue || "object" === typeof keyValue && isEmptyObject(keyValue)) { if (Array.isArray(keyExpr)) { throw errors.Error("E4007") } keyValue = obj[keyExpr] = String(new Guid) } else if (void 0 !== array[indexByKey(keyInfo, array, keyValue)]) { return getErrorResult(isBatch, logError, "E4008") } } else { keyValue = obj } if (index >= 0) { array.splice(index, 0, obj) } else { array.push(obj) } setDataByKeyMapValue(array, keyValue, obj); if (!isBatch) { return trivialPromise(config().useLegacyStoreResult ? data : obj, keyValue) } } function remove(keyInfo, array, key, isBatch, logError) { const index = indexByKey(keyInfo, array, key); if (index > -1) { array.splice(index, 1); setDataByKeyMapValue(array, key, null) } if (!isBatch) { return trivialPromise(key) } if (index < 0) { return getErrorResult(isBatch, logError, "E4009") } } function indexByKey(keyInfo, array, key) { const keyExpr = keyInfo.key(); if (!getHasKeyCacheValue(array, key)) { return -1 } for (let i = 0, arrayLength = array.length; i < arrayLength; i++) { if (keysEqual(keyExpr, keyInfo.keyOf(array[i]), key)) { return i } } return -1 } export { applyBatch, applyChanges, createObjectWithChanges, indexByKey, insert, remove, update };