_baseMergeDeep.js 3.0 KB

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