UNPKG

react-pur

Version:

React UI component library (using styled-components)

153 lines (117 loc) 3.76 kB
'use strict'; /* * OBJECT ASSIGN DEEP * Allows deep cloning of plain objects that contain primitives, nested plain objects, or nested plain arrays. */ /* * A unified way of returning a string that describes the type of the given variable. */ function getTypeOf (input) { if (input === null) { return 'null'; } else if (typeof input === 'undefined') { return 'undefined'; } else if (typeof input === 'object') { return (Array.isArray(input) ? 'array' : 'object'); } return typeof input; } /* * Branching logic which calls the correct function to clone the given value base on its type. */ function cloneValue (value) { // The value is an object so lets clone it. if (getTypeOf(value) === 'object') { return quickCloneObject(value); } // The value is an array so lets clone it. else if (getTypeOf(value) === 'array') { return quickCloneArray(value); } // Any other value can just be copied. return value; } /* * Enumerates the given array and returns a new array, with each of its values cloned (i.e. references broken). */ function quickCloneArray (input) { return input.map(cloneValue); } /* * Enumerates the properties of the given object (ignoring the prototype chain) and returns a new object, with each of * its values cloned (i.e. references broken). */ function quickCloneObject (input) { var output = {}; for (var key in input) { if (!input.hasOwnProperty(key)) { continue; } output[key] = cloneValue(input[key]); } return output; } /* * Does the actual deep merging. */ function executeDeepMerge (target, _objects = [], _options = {}) { var options = { arrayBehaviour: _options.arrayBehaviour || 'replace', // Can be "merge" or "replace". }; // Ensure we have actual objects for each. var objects = _objects.map(object => object || {}); var output = target || {}; // Enumerate the objects and their keys. for (let oindex = 0; oindex < objects.length; oindex++) { var object = objects[oindex]; var keys = Object.keys(object); for (let kindex = 0; kindex < keys.length; kindex++) { var key = keys[kindex]; var value = object[key]; var type = getTypeOf(value); var existingValueType = getTypeOf(output[key]); if (type === 'object') { if (existingValueType !== 'undefined') { var existingValue = (existingValueType === 'object' ? output[key] : {}); output[key] = executeDeepMerge({}, [existingValue, quickCloneObject(value)], options); } else { output[key] = quickCloneObject(value); } } else if (type === 'array') { if (existingValueType === 'array') { var newValue = quickCloneArray(value); output[key] = (options.arrayBehaviour === 'merge' ? output[key].concat(newValue) : newValue); } else { output[key] = quickCloneArray(value); } } else { output[key] = value; } } } return output; } /* * Merge all the supplied objects into the target object, breaking all references, including those of nested objects * and arrays, and even objects nested inside arrays. The first parameter is not mutated unlike Object.assign(). * Properties in later objects will always overwrite. */ module.exports = function objectAssignDeep (target, ...objects) { return executeDeepMerge(target, objects); }; /* * Same as objectAssignDeep() except it doesn't mutate the target object and returns an entirely new object. */ module.exports.noMutate = function objectAssignDeepInto (...objects) { return executeDeepMerge({}, objects); }; /* * Allows an options object to be passed in to customise the behaviour of the function. */ module.exports.withOptions = function objectAssignDeepInto (target, objects, options) { return executeDeepMerge(target, objects, options); };