@orbit/utils
Version:
Core utilities for Orbit.
224 lines • 20.1 kB
JavaScript
/* eslint-disable @typescript-eslint/explicit-module-boundary-types, valid-jsdoc */
/**
* Clones a value. If the value is an object, a deeply nested clone will be
* created.
*
* Traverses all object properties (but not prototype properties).
*/
export function clone(obj) {
if (obj === undefined || obj === null || typeof obj !== 'object') {
return obj;
}
let dup;
let type = Object.prototype.toString.call(obj);
if (type === '[object Date]') {
dup = new Date();
dup.setTime(obj.getTime());
}
else if (type === '[object RegExp]') {
dup = obj.constructor(obj);
}
else if (type === '[object Array]') {
dup = [];
for (let i = 0, len = obj.length; i < len; i++) {
if (obj.hasOwnProperty(i)) {
dup.push(clone(obj[i]));
}
}
}
else {
let val;
dup = {};
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
val = obj[key];
if (typeof val === 'object') {
val = clone(val);
}
dup[key] = val;
}
}
}
return dup;
}
/**
* Expose properties and methods from one object on another.
*
* Methods will be called on `source` and will maintain `source` as the context.
*
* @deprecated since v0.17
*/
export function expose(destination, source) {
let properties;
if (arguments.length > 2) {
properties = Array.prototype.slice.call(arguments, 2);
}
else {
properties = Object.keys(source);
}
properties.forEach((p) => {
if (typeof source[p] === 'function') {
destination[p] = function () {
return source[p].apply(source, arguments);
};
}
else {
destination[p] = source[p];
}
});
}
/**
* Extend an object with the properties of one or more other objects.
*
* @deprecated since v0.17
*/
export function extend(destination, ...sources) {
sources.forEach((source) => {
for (let p in source) {
if (source.hasOwnProperty(p)) {
destination[p] = source[p];
}
}
});
return destination;
}
/**
* Converts an object to an `Array` if it's not already.
*
* @export
* @param {*} obj
* @returns {any[]}
*/
export function toArray(obj) {
if (isNone(obj)) {
return [];
}
else {
return Array.isArray(obj) ? obj : [obj];
}
}
/**
* Checks whether a value is a non-null object
*
* @export
* @param {*} obj
* @returns {boolean}
*/
export function isObject(obj) {
return obj !== null && typeof obj === 'object';
}
/**
* Checks whether an object is null or undefined
*
* @export
* @param {*} obj
* @returns {boolean}
*/
export function isNone(obj) {
return obj === undefined || obj === null;
}
/**
* Merges properties from other objects into a base object. Properties that
* resolve to `undefined` will not overwrite properties on the base object
* that already exist.
*
* @deprecated since v0.17
*/
export function merge(object, ...sources) {
sources.forEach((source) => {
Object.keys(source).forEach((field) => {
if (source.hasOwnProperty(field)) {
let value = source[field];
if (value !== undefined) {
object[field] = clone(value);
}
}
});
});
return object;
}
/**
* Merges properties from other objects into a base object, traversing and
* merging any objects that are encountered.
*
* Properties that resolve to `undefined` will not overwrite properties on the
* base object that already exist.
*/
export function deepMerge(object, ...sources) {
sources.forEach((source) => {
Object.keys(source).forEach((field) => {
if (source.hasOwnProperty(field)) {
let a = object[field];
let b = source[field];
if (isObject(a) &&
isObject(b) &&
!Array.isArray(a) &&
!Array.isArray(b)) {
deepMerge(a, b);
}
else if (b !== undefined) {
object[field] = clone(b);
}
}
});
});
return object;
}
/**
* Retrieves a value from a nested path on an object.
*
* Returns any falsy value encountered while traversing the path.
*/
export function deepGet(obj, path) {
let index = -1;
let result = obj;
while (++index < path.length) {
result = result[path[index]];
if (!result) {
return result;
}
}
return result;
}
/**
* Sets a value on an object at a nested path.
*
* This function will create objects along the path if necessary to allow
* setting a deeply nested value.
*
* Returns `false` only if the current value is already strictly equal to the
* requested `value` argument. Otherwise returns `true`.
*/
export function deepSet(obj, path, value) {
let ptr = obj;
let prop = path.pop();
let segment;
for (let i = 0, l = path.length; i < l; i++) {
segment = path[i];
if (ptr[segment] === undefined) {
ptr[segment] = typeof segment === 'number' ? [] : {};
}
ptr = ptr[segment];
}
if (ptr[prop] === value) {
return false;
}
else {
ptr[prop] = value;
return true;
}
}
/**
* Find an array of values that correspond to the keys of an object.
*
* This is a ponyfill for `Object.values`, which is still experimental.
*/
export function objectValues(obj) {
if (Object.values) {
return Object.values(obj);
}
else {
return Object.keys(obj).map((k) => obj[k]);
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"objects.js","sourceRoot":"","sources":["../../src/objects.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/explicit-module-boundary-types, valid-jsdoc */\n\n/**\n * Clones a value. If the value is an object, a deeply nested clone will be\n * created.\n *\n * Traverses all object properties (but not prototype properties).\n */\nexport function clone(obj: any): any {\n  if (obj === undefined || obj === null || typeof obj !== 'object') {\n    return obj;\n  }\n\n  let dup: any;\n  let type = Object.prototype.toString.call(obj);\n\n  if (type === '[object Date]') {\n    dup = new Date();\n    dup.setTime(obj.getTime());\n  } else if (type === '[object RegExp]') {\n    dup = obj.constructor(obj);\n  } else if (type === '[object Array]') {\n    dup = [];\n    for (let i = 0, len = obj.length; i < len; i++) {\n      if (obj.hasOwnProperty(i)) {\n        dup.push(clone(obj[i]));\n      }\n    }\n  } else {\n    let val;\n\n    dup = {};\n    for (let key in obj) {\n      if (obj.hasOwnProperty(key)) {\n        val = obj[key];\n        if (typeof val === 'object') {\n          val = clone(val);\n        }\n        dup[key] = val;\n      }\n    }\n  }\n  return dup;\n}\n\n/**\n * Expose properties and methods from one object on another.\n *\n * Methods will be called on `source` and will maintain `source` as the context.\n *\n * @deprecated since v0.17\n */\nexport function expose(destination: any, source: any): void {\n  let properties: string[];\n  if (arguments.length > 2) {\n    properties = Array.prototype.slice.call(arguments, 2);\n  } else {\n    properties = Object.keys(source);\n  }\n\n  properties.forEach((p) => {\n    if (typeof source[p] === 'function') {\n      destination[p] = function () {\n        return source[p].apply(source, arguments);\n      };\n    } else {\n      destination[p] = source[p];\n    }\n  });\n}\n\n/**\n * Extend an object with the properties of one or more other objects.\n *\n * @deprecated since v0.17\n */\nexport function extend(destination: any, ...sources: any[]): any {\n  sources.forEach((source) => {\n    for (let p in source) {\n      if (source.hasOwnProperty(p)) {\n        destination[p] = source[p];\n      }\n    }\n  });\n  return destination;\n}\n\n/**\n * Converts an object to an `Array` if it's not already.\n *\n * @export\n * @param {*} obj\n * @returns {any[]}\n */\nexport function toArray(obj: unknown): any[] {\n  if (isNone(obj)) {\n    return [];\n  } else {\n    return Array.isArray(obj) ? obj : [obj];\n  }\n}\n\n/**\n * Checks whether a value is a non-null object\n *\n * @export\n * @param {*} obj\n * @returns {boolean}\n */\nexport function isObject(obj: unknown): boolean {\n  return obj !== null && typeof obj === 'object';\n}\n\n/**\n * Checks whether an object is null or undefined\n *\n * @export\n * @param {*} obj\n * @returns {boolean}\n */\nexport function isNone(obj: unknown): boolean {\n  return obj === undefined || obj === null;\n}\n\n/**\n * Merges properties from other objects into a base object. Properties that\n * resolve to `undefined` will not overwrite properties on the base object\n * that already exist.\n *\n * @deprecated since v0.17\n */\nexport function merge(object: any, ...sources: any[]): any {\n  sources.forEach((source) => {\n    Object.keys(source).forEach((field) => {\n      if (source.hasOwnProperty(field)) {\n        let value = source[field];\n        if (value !== undefined) {\n          object[field] = clone(value);\n        }\n      }\n    });\n  });\n  return object;\n}\n\n/**\n * Merges properties from other objects into a base object, traversing and\n * merging any objects that are encountered.\n *\n * Properties that resolve to `undefined` will not overwrite properties on the\n * base object that already exist.\n */\nexport function deepMerge(object: any, ...sources: any[]): any {\n  sources.forEach((source) => {\n    Object.keys(source).forEach((field) => {\n      if (source.hasOwnProperty(field)) {\n        let a = object[field];\n        let b = source[field];\n        if (\n          isObject(a) &&\n          isObject(b) &&\n          !Array.isArray(a) &&\n          !Array.isArray(b)\n        ) {\n          deepMerge(a, b);\n        } else if (b !== undefined) {\n          object[field] = clone(b);\n        }\n      }\n    });\n  });\n  return object;\n}\n\n/**\n * Retrieves a value from a nested path on an object.\n *\n * Returns any falsy value encountered while traversing the path.\n */\nexport function deepGet(obj: any, path: string[]): any {\n  let index = -1;\n  let result = obj;\n\n  while (++index < path.length) {\n    result = result[path[index]];\n    if (!result) {\n      return result;\n    }\n  }\n\n  return result;\n}\n\n/**\n * Sets a value on an object at a nested path.\n *\n * This function will create objects along the path if necessary to allow\n * setting a deeply nested value.\n *\n * Returns `false` only if the current value is already strictly equal to the\n * requested `value` argument. Otherwise returns `true`.\n */\nexport function deepSet(obj: any, path: string[], value: any): boolean {\n  let ptr = obj;\n  let prop = path.pop() as string;\n  let segment;\n  for (let i = 0, l = path.length; i < l; i++) {\n    segment = path[i];\n    if (ptr[segment] === undefined) {\n      ptr[segment] = typeof segment === 'number' ? [] : {};\n    }\n    ptr = ptr[segment];\n  }\n  if (ptr[prop] === value) {\n    return false;\n  } else {\n    ptr[prop] = value;\n    return true;\n  }\n}\n\n/**\n * Find an array of values that correspond to the keys of an object.\n *\n * This is a ponyfill for `Object.values`, which is still experimental.\n */\nexport function objectValues(obj: any): any[] {\n  if (Object.values) {\n    return Object.values(obj);\n  } else {\n    return Object.keys(obj).map((k) => obj[k]);\n  }\n}\n"]}