react-pur
Version:
React UI component library (using styled-components)
153 lines (117 loc) • 3.76 kB
JavaScript
;
/*
* 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);
};