mirror of
https://github.com/whoisclebs/lodash.git
synced 2026-02-12 20:07:49 +00:00
Fix cloneDeep with circularly dependent Sets/Maps. [closes #3122]
This commit is contained in:
83
lodash.js
83
lodash.js
@@ -451,34 +451,6 @@
|
|||||||
|
|
||||||
/*--------------------------------------------------------------------------*/
|
/*--------------------------------------------------------------------------*/
|
||||||
|
|
||||||
/**
|
|
||||||
* Adds the key-value `pair` to `map`.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
* @param {Object} map The map to modify.
|
|
||||||
* @param {Array} pair The key-value pair to add.
|
|
||||||
* @returns {Object} Returns `map`.
|
|
||||||
*/
|
|
||||||
function addMapEntry(map, pair) {
|
|
||||||
// Don't return `map.set` because it's not chainable in IE 11.
|
|
||||||
map.set(pair[0], pair[1]);
|
|
||||||
return map;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Adds `value` to `set`.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
* @param {Object} set The set to modify.
|
|
||||||
* @param {*} value The value to add.
|
|
||||||
* @returns {Object} Returns `set`.
|
|
||||||
*/
|
|
||||||
function addSetEntry(set, value) {
|
|
||||||
// Don't return `set.add` because it's not chainable in IE 11.
|
|
||||||
set.add(value);
|
|
||||||
return set;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A faster alternative to `Function#apply`, this function invokes `func`
|
* A faster alternative to `Function#apply`, this function invokes `func`
|
||||||
* with the `this` binding of `thisArg` and the arguments of `args`.
|
* with the `this` binding of `thisArg` and the arguments of `args`.
|
||||||
@@ -2691,7 +2663,7 @@
|
|||||||
if (!cloneableTags[tag]) {
|
if (!cloneableTags[tag]) {
|
||||||
return object ? value : {};
|
return object ? value : {};
|
||||||
}
|
}
|
||||||
result = initCloneByTag(value, tag, baseClone, isDeep);
|
result = initCloneByTag(value, tag, isDeep);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Check for circular references and return its corresponding clone.
|
// Check for circular references and return its corresponding clone.
|
||||||
@@ -2702,6 +2674,22 @@
|
|||||||
}
|
}
|
||||||
stack.set(value, result);
|
stack.set(value, result);
|
||||||
|
|
||||||
|
if (isSet(value)) {
|
||||||
|
value.forEach(function(subValue) {
|
||||||
|
result.add(baseClone(subValue, bitmask, customizer, subValue, value, stack));
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isMap(value)) {
|
||||||
|
value.forEach(function(subValue, key) {
|
||||||
|
result.set(key, baseClone(subValue, bitmask, customizer, key, value, stack));
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
var keysFunc = isFull
|
var keysFunc = isFull
|
||||||
? (isFlat ? getAllKeysIn : getAllKeys)
|
? (isFlat ? getAllKeysIn : getAllKeys)
|
||||||
: (isFlat ? keysIn : keys);
|
: (isFlat ? keysIn : keys);
|
||||||
@@ -4565,20 +4553,6 @@
|
|||||||
return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);
|
return new dataView.constructor(buffer, dataView.byteOffset, dataView.byteLength);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a clone of `map`.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
* @param {Object} map The map to clone.
|
|
||||||
* @param {Function} cloneFunc The function to clone values.
|
|
||||||
* @param {boolean} [isDeep] Specify a deep clone.
|
|
||||||
* @returns {Object} Returns the cloned map.
|
|
||||||
*/
|
|
||||||
function cloneMap(map, isDeep, cloneFunc) {
|
|
||||||
var array = isDeep ? cloneFunc(mapToArray(map), CLONE_DEEP_FLAG) : mapToArray(map);
|
|
||||||
return arrayReduce(array, addMapEntry, new map.constructor);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a clone of `regexp`.
|
* Creates a clone of `regexp`.
|
||||||
*
|
*
|
||||||
@@ -4592,20 +4566,6 @@
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Creates a clone of `set`.
|
|
||||||
*
|
|
||||||
* @private
|
|
||||||
* @param {Object} set The set to clone.
|
|
||||||
* @param {Function} cloneFunc The function to clone values.
|
|
||||||
* @param {boolean} [isDeep] Specify a deep clone.
|
|
||||||
* @returns {Object} Returns the cloned set.
|
|
||||||
*/
|
|
||||||
function cloneSet(set, isDeep, cloneFunc) {
|
|
||||||
var array = isDeep ? cloneFunc(setToArray(set), CLONE_DEEP_FLAG) : setToArray(set);
|
|
||||||
return arrayReduce(array, addSetEntry, new set.constructor);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a clone of the `symbol` object.
|
* Creates a clone of the `symbol` object.
|
||||||
*
|
*
|
||||||
@@ -6227,16 +6187,15 @@
|
|||||||
* Initializes an object clone based on its `toStringTag`.
|
* Initializes an object clone based on its `toStringTag`.
|
||||||
*
|
*
|
||||||
* **Note:** This function only supports cloning values with tags of
|
* **Note:** This function only supports cloning values with tags of
|
||||||
* `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`.
|
* `Boolean`, `Date`, `Error`, `Map`, `Number`, `RegExp`, `Set`, or `String`.
|
||||||
*
|
*
|
||||||
* @private
|
* @private
|
||||||
* @param {Object} object The object to clone.
|
* @param {Object} object The object to clone.
|
||||||
* @param {string} tag The `toStringTag` of the object to clone.
|
* @param {string} tag The `toStringTag` of the object to clone.
|
||||||
* @param {Function} cloneFunc The function to clone values.
|
|
||||||
* @param {boolean} [isDeep] Specify a deep clone.
|
* @param {boolean} [isDeep] Specify a deep clone.
|
||||||
* @returns {Object} Returns the initialized clone.
|
* @returns {Object} Returns the initialized clone.
|
||||||
*/
|
*/
|
||||||
function initCloneByTag(object, tag, cloneFunc, isDeep) {
|
function initCloneByTag(object, tag, isDeep) {
|
||||||
var Ctor = object.constructor;
|
var Ctor = object.constructor;
|
||||||
switch (tag) {
|
switch (tag) {
|
||||||
case arrayBufferTag:
|
case arrayBufferTag:
|
||||||
@@ -6255,7 +6214,7 @@
|
|||||||
return cloneTypedArray(object, isDeep);
|
return cloneTypedArray(object, isDeep);
|
||||||
|
|
||||||
case mapTag:
|
case mapTag:
|
||||||
return cloneMap(object, isDeep, cloneFunc);
|
return new Ctor;
|
||||||
|
|
||||||
case numberTag:
|
case numberTag:
|
||||||
case stringTag:
|
case stringTag:
|
||||||
@@ -6265,7 +6224,7 @@
|
|||||||
return cloneRegExp(object);
|
return cloneRegExp(object);
|
||||||
|
|
||||||
case setTag:
|
case setTag:
|
||||||
return cloneSet(object, isDeep, cloneFunc);
|
return new Ctor;
|
||||||
|
|
||||||
case symbolTag:
|
case symbolTag:
|
||||||
return cloneSymbol(object);
|
return cloneSymbol(object);
|
||||||
|
|||||||
Reference in New Issue
Block a user