UNPKG

5.2 kBJavaScriptView Raw
1var Stack = require('./_Stack'),
2 arrayEach = require('./_arrayEach'),
3 assignValue = require('./_assignValue'),
4 baseAssign = require('./_baseAssign'),
5 baseAssignIn = require('./_baseAssignIn'),
6 cloneBuffer = require('./_cloneBuffer'),
7 copyArray = require('./_copyArray'),
8 copySymbols = require('./_copySymbols'),
9 copySymbolsIn = require('./_copySymbolsIn'),
10 getAllKeys = require('./_getAllKeys'),
11 getAllKeysIn = require('./_getAllKeysIn'),
12 getTag = require('./_getTag'),
13 initCloneArray = require('./_initCloneArray'),
14 initCloneByTag = require('./_initCloneByTag'),
15 initCloneObject = require('./_initCloneObject'),
16 isArray = require('./isArray'),
17 isBuffer = require('./isBuffer'),
18 isObject = require('./isObject'),
19 keys = require('./keys');
20
21/** Used to compose bitmasks for cloning. */
22var CLONE_DEEP_FLAG = 1,
23 CLONE_FLAT_FLAG = 2,
24 CLONE_SYMBOLS_FLAG = 4;
25
26/** `Object#toString` result references. */
27var argsTag = '[object Arguments]',
28 arrayTag = '[object Array]',
29 boolTag = '[object Boolean]',
30 dateTag = '[object Date]',
31 errorTag = '[object Error]',
32 funcTag = '[object Function]',
33 genTag = '[object GeneratorFunction]',
34 mapTag = '[object Map]',
35 numberTag = '[object Number]',
36 objectTag = '[object Object]',
37 regexpTag = '[object RegExp]',
38 setTag = '[object Set]',
39 stringTag = '[object String]',
40 symbolTag = '[object Symbol]',
41 weakMapTag = '[object WeakMap]';
42
43var arrayBufferTag = '[object ArrayBuffer]',
44 dataViewTag = '[object DataView]',
45 float32Tag = '[object Float32Array]',
46 float64Tag = '[object Float64Array]',
47 int8Tag = '[object Int8Array]',
48 int16Tag = '[object Int16Array]',
49 int32Tag = '[object Int32Array]',
50 uint8Tag = '[object Uint8Array]',
51 uint8ClampedTag = '[object Uint8ClampedArray]',
52 uint16Tag = '[object Uint16Array]',
53 uint32Tag = '[object Uint32Array]';
54
55/** Used to identify `toStringTag` values supported by `_.clone`. */
56var cloneableTags = {};
57cloneableTags[argsTag] = cloneableTags[arrayTag] =
58cloneableTags[arrayBufferTag] = cloneableTags[dataViewTag] =
59cloneableTags[boolTag] = cloneableTags[dateTag] =
60cloneableTags[float32Tag] = cloneableTags[float64Tag] =
61cloneableTags[int8Tag] = cloneableTags[int16Tag] =
62cloneableTags[int32Tag] = cloneableTags[mapTag] =
63cloneableTags[numberTag] = cloneableTags[objectTag] =
64cloneableTags[regexpTag] = cloneableTags[setTag] =
65cloneableTags[stringTag] = cloneableTags[symbolTag] =
66cloneableTags[uint8Tag] = cloneableTags[uint8ClampedTag] =
67cloneableTags[uint16Tag] = cloneableTags[uint32Tag] = true;
68cloneableTags[errorTag] = cloneableTags[funcTag] =
69cloneableTags[weakMapTag] = false;
70
71/**
72 * The base implementation of `_.clone` and `_.cloneDeep` which tracks
73 * traversed objects.
74 *
75 * @private
76 * @param {*} value The value to clone.
77 * @param {boolean} bitmask The bitmask flags.
78 * 1 - Deep clone
79 * 2 - Flatten inherited properties
80 * 4 - Clone symbols
81 * @param {Function} [customizer] The function to customize cloning.
82 * @param {string} [key] The key of `value`.
83 * @param {Object} [object] The parent object of `value`.
84 * @param {Object} [stack] Tracks traversed objects and their clone counterparts.
85 * @returns {*} Returns the cloned value.
86 */
87function baseClone(value, bitmask, customizer, key, object, stack) {
88 var result,
89 isDeep = bitmask & CLONE_DEEP_FLAG,
90 isFlat = bitmask & CLONE_FLAT_FLAG,
91 isFull = bitmask & CLONE_SYMBOLS_FLAG;
92
93 if (customizer) {
94 result = object ? customizer(value, key, object, stack) : customizer(value);
95 }
96 if (result !== undefined) {
97 return result;
98 }
99 if (!isObject(value)) {
100 return value;
101 }
102 var isArr = isArray(value);
103 if (isArr) {
104 result = initCloneArray(value);
105 if (!isDeep) {
106 return copyArray(value, result);
107 }
108 } else {
109 var tag = getTag(value),
110 isFunc = tag == funcTag || tag == genTag;
111
112 if (isBuffer(value)) {
113 return cloneBuffer(value, isDeep);
114 }
115 if (tag == objectTag || tag == argsTag || (isFunc && !object)) {
116 result = (isFlat || isFunc) ? {} : initCloneObject(value);
117 if (!isDeep) {
118 return isFlat
119 ? copySymbolsIn(value, baseAssignIn(result, value))
120 : copySymbols(value, baseAssign(result, value));
121 }
122 } else {
123 if (!cloneableTags[tag]) {
124 return object ? value : {};
125 }
126 result = initCloneByTag(value, tag, baseClone, isDeep);
127 }
128 }
129 // Check for circular references and return its corresponding clone.
130 stack || (stack = new Stack);
131 var stacked = stack.get(value);
132 if (stacked) {
133 return stacked;
134 }
135 stack.set(value, result);
136
137 var keysFunc = isFull
138 ? (isFlat ? getAllKeysIn : getAllKeys)
139 : (isFlat ? keysIn : keys);
140
141 var props = isArr ? undefined : keysFunc(value);
142 arrayEach(props || value, function(subValue, key) {
143 if (props) {
144 key = subValue;
145 subValue = value[key];
146 }
147 // Recursively populate clone (susceptible to call stack limits).
148 assignValue(result, key, baseClone(subValue, bitmask, customizer, key, value, stack));
149 });
150 return result;
151}
152
153module.exports = baseClone;