[79a0317] | 1 | var assignMergeValue = require('./_assignMergeValue'),
|
---|
| 2 | cloneBuffer = require('./_cloneBuffer'),
|
---|
| 3 | cloneTypedArray = require('./_cloneTypedArray'),
|
---|
| 4 | copyArray = require('./_copyArray'),
|
---|
| 5 | initCloneObject = require('./_initCloneObject'),
|
---|
| 6 | isArguments = require('./isArguments'),
|
---|
| 7 | isArray = require('./isArray'),
|
---|
| 8 | isArrayLikeObject = require('./isArrayLikeObject'),
|
---|
| 9 | isBuffer = require('./isBuffer'),
|
---|
| 10 | isFunction = require('./isFunction'),
|
---|
| 11 | isObject = require('./isObject'),
|
---|
| 12 | isPlainObject = require('./isPlainObject'),
|
---|
| 13 | isTypedArray = require('./isTypedArray'),
|
---|
| 14 | safeGet = require('./_safeGet'),
|
---|
| 15 | toPlainObject = require('./toPlainObject');
|
---|
| 16 |
|
---|
| 17 | /**
|
---|
| 18 | * A specialized version of `baseMerge` for arrays and objects which performs
|
---|
| 19 | * deep merges and tracks traversed objects enabling objects with circular
|
---|
| 20 | * references to be merged.
|
---|
| 21 | *
|
---|
| 22 | * @private
|
---|
| 23 | * @param {Object} object The destination object.
|
---|
| 24 | * @param {Object} source The source object.
|
---|
| 25 | * @param {string} key The key of the value to merge.
|
---|
| 26 | * @param {number} srcIndex The index of `source`.
|
---|
| 27 | * @param {Function} mergeFunc The function to merge values.
|
---|
| 28 | * @param {Function} [customizer] The function to customize assigned values.
|
---|
| 29 | * @param {Object} [stack] Tracks traversed source values and their merged
|
---|
| 30 | * counterparts.
|
---|
| 31 | */
|
---|
| 32 | function baseMergeDeep(object, source, key, srcIndex, mergeFunc, customizer, stack) {
|
---|
| 33 | var objValue = safeGet(object, key),
|
---|
| 34 | srcValue = safeGet(source, key),
|
---|
| 35 | stacked = stack.get(srcValue);
|
---|
| 36 |
|
---|
| 37 | if (stacked) {
|
---|
| 38 | assignMergeValue(object, key, stacked);
|
---|
| 39 | return;
|
---|
| 40 | }
|
---|
| 41 | var newValue = customizer
|
---|
| 42 | ? customizer(objValue, srcValue, (key + ''), object, source, stack)
|
---|
| 43 | : undefined;
|
---|
| 44 |
|
---|
| 45 | var isCommon = newValue === undefined;
|
---|
| 46 |
|
---|
| 47 | if (isCommon) {
|
---|
| 48 | var isArr = isArray(srcValue),
|
---|
| 49 | isBuff = !isArr && isBuffer(srcValue),
|
---|
| 50 | isTyped = !isArr && !isBuff && isTypedArray(srcValue);
|
---|
| 51 |
|
---|
| 52 | newValue = srcValue;
|
---|
| 53 | if (isArr || isBuff || isTyped) {
|
---|
| 54 | if (isArray(objValue)) {
|
---|
| 55 | newValue = objValue;
|
---|
| 56 | }
|
---|
| 57 | else if (isArrayLikeObject(objValue)) {
|
---|
| 58 | newValue = copyArray(objValue);
|
---|
| 59 | }
|
---|
| 60 | else if (isBuff) {
|
---|
| 61 | isCommon = false;
|
---|
| 62 | newValue = cloneBuffer(srcValue, true);
|
---|
| 63 | }
|
---|
| 64 | else if (isTyped) {
|
---|
| 65 | isCommon = false;
|
---|
| 66 | newValue = cloneTypedArray(srcValue, true);
|
---|
| 67 | }
|
---|
| 68 | else {
|
---|
| 69 | newValue = [];
|
---|
| 70 | }
|
---|
| 71 | }
|
---|
| 72 | else if (isPlainObject(srcValue) || isArguments(srcValue)) {
|
---|
| 73 | newValue = objValue;
|
---|
| 74 | if (isArguments(objValue)) {
|
---|
| 75 | newValue = toPlainObject(objValue);
|
---|
| 76 | }
|
---|
| 77 | else if (!isObject(objValue) || isFunction(objValue)) {
|
---|
| 78 | newValue = initCloneObject(srcValue);
|
---|
| 79 | }
|
---|
| 80 | }
|
---|
| 81 | else {
|
---|
| 82 | isCommon = false;
|
---|
| 83 | }
|
---|
| 84 | }
|
---|
| 85 | if (isCommon) {
|
---|
| 86 | // Recursively merge objects and arrays (susceptible to call stack limits).
|
---|
| 87 | stack.set(srcValue, newValue);
|
---|
| 88 | mergeFunc(newValue, srcValue, srcIndex, customizer, stack);
|
---|
| 89 | stack['delete'](srcValue);
|
---|
| 90 | }
|
---|
| 91 | assignMergeValue(object, key, newValue);
|
---|
| 92 | }
|
---|
| 93 |
|
---|
| 94 | module.exports = baseMergeDeep;
|
---|