Spaces:
Runtime error
Runtime error
var assignMergeValue = require('./_assignMergeValue'), | |
cloneBuffer = require('./_cloneBuffer'), | |
cloneTypedArray = require('./_cloneTypedArray'), | |
copyArray = require('./_copyArray'), | |
initCloneObject = require('./_initCloneObject'), | |
isArguments = require('./isArguments'), | |
isArray = require('./isArray'), | |
isArrayLikeObject = require('./isArrayLikeObject'), | |
isBuffer = require('./isBuffer'), | |
isFunction = require('./isFunction'), | |
isObject = require('./isObject'), | |
isPlainObject = require('./isPlainObject'), | |
isTypedArray = require('./isTypedArray'), | |
safeGet = require('./_safeGet'), | |
toPlainObject = require('./toPlainObject'); | |
/** | |
* A specialized version of `baseMerge` for arrays and objects which performs | |
* deep merges and tracks traversed objects enabling objects with circular | |
* references to be merged. | |
* | |
* @private | |
* @param {Object} object The destination object. | |
* @param {Object} source The source object. | |
* @param {string} key The key of the value to merge. | |
* @param {number} srcIndex The index of `source`. | |
* @param {Function} mergeFunc The function to merge values. | |
* @param {Function} [customizer] The function to customize assigned values. | |
* @param {Object} [stack] Tracks traversed source values and their merged | |
* counterparts. | |
*/ | |
function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) { | |
var objValue = safeGet(object, key), | |
srcValue = safeGet(source, key), | |
stacked = stack.get(srcValue); | |
if (stacked) { | |
assignMergeValue(object, key, stacked); | |
return; | |
} | |
var newValue = customizer | |
? customizer(objValue, srcValue, (key + ''), object, source, stack) | |
: undefined; | |
var isCommon = newValue === undefined; | |
if (isCommon) { | |
var isArr = isArray(srcValue), | |
isBuff = !isArr && isBuffer(srcValue), | |
isTyped = !isArr && !isBuff && isTypedArray(srcValue); | |
newValue = srcValue; | |
if (isArr || isBuff || isTyped) { | |
if (isArray(objValue)) { | |
newValue = objValue; | |
} | |
else if (isArrayLikeObject(objValue)) { | |
newValue = copyArray(objValue); | |
} | |
else if (isBuff) { | |
isCommon = false; | |
newValue = cloneBuffer(srcValue, true); | |
} | |
else if (isTyped) { | |
isCommon = false; | |
newValue = cloneTypedArray(srcValue, true); | |
} | |
else { | |
newValue = []; | |
} | |
} | |
else if (isPlainObject(srcValue) || isArguments(srcValue)) { | |
newValue = objValue; | |
if (isArguments(objValue)) { | |
newValue = toPlainObject(objValue); | |
} | |
else if (!isObject(objValue) || isFunction(objValue)) { | |
newValue = initCloneObject(srcValue); | |
} | |
} | |
else { | |
isCommon = false; | |
} | |
} | |
if (isCommon) { | |
// Recursively merge objects and arrays (susceptible to call stack limits). | |
stack.set(srcValue, newValue); | |
mergeFunc(newValue, srcValue, srcIndex, customizer, stack); | |
stack['delete'](srcValue); | |
} | |
assignMergeValue(object, key, newValue); | |
} | |
module.exports = baseMergeDeep; | |