Last active
June 30, 2016 17:08
-
-
Save tomyam1/a4da104a2793840e6b65e3043684c88b to your computer and use it in GitHub Desktop.
lodash-issue
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
/** | |
* @license | |
* lodash (Custom Build) <https://lodash.com/> | |
* Build: `lodash core --development plus="add,assign,capitalize,cloneDeep,deburr,findIndex,forIn,intersection,isError,isPlainObject,merge,method,pickBy,property,range,remove,times,uniq,values,String,maxBy,isNil,chunk"` | |
* Copyright jQuery Foundation and other contributors <https://jquery.org/> | |
* Released under MIT license <https://lodash.com/license> | |
* Based on Underscore.js 1.8.3 <http://underscorejs.org/LICENSE> | |
* Copyright Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors | |
*/ | |
;(function() { | |
/** Used as a safe reference for `undefined` in pre-ES5 environments. */ | |
var undefined; | |
/** Used as the semantic version number. */ | |
var VERSION = '4.13.1'; | |
/** Used as the size to enable large array optimizations. */ | |
var LARGE_ARRAY_SIZE = 200; | |
/** Used as the `TypeError` message for "Functions" methods. */ | |
var FUNC_ERROR_TEXT = 'Expected a function'; | |
/** Used to stand-in for `undefined` hash values. */ | |
var HASH_UNDEFINED = '__lodash_hash_undefined__'; | |
/** Used to compose bitmasks for wrapper metadata. */ | |
var BIND_FLAG = 1, | |
PARTIAL_FLAG = 32; | |
/** Used to compose bitmasks for comparison styles. */ | |
var UNORDERED_COMPARE_FLAG = 1, | |
PARTIAL_COMPARE_FLAG = 2; | |
/** Used as default options for `_.truncate`. */ | |
var DEFAULT_TRUNC_LENGTH = 30, | |
DEFAULT_TRUNC_OMISSION = '...'; | |
/** Used as references for various `Number` constants. */ | |
var INFINITY = 1 / 0, | |
MAX_SAFE_INTEGER = 9007199254740991, | |
NAN = 0 / 0; | |
/** Used as references for the maximum length and index of an array. */ | |
var MAX_ARRAY_LENGTH = 4294967295; | |
/** `Object#toString` result references. */ | |
var argsTag = '[object Arguments]', | |
arrayTag = '[object Array]', | |
boolTag = '[object Boolean]', | |
dateTag = '[object Date]', | |
errorTag = '[object Error]', | |
funcTag = '[object Function]', | |
genTag = '[object GeneratorFunction]', | |
mapTag = '[object Map]', | |
numberTag = '[object Number]', | |
objectTag = '[object Object]', | |
promiseTag = '[object Promise]', | |
regexpTag = '[object RegExp]', | |
setTag = '[object Set]', | |
stringTag = '[object String]', | |
symbolTag = '[object Symbol]', | |
weakMapTag = '[object WeakMap]'; | |
var arrayBufferTag = '[object ArrayBuffer]', | |
dataViewTag = '[object DataView]', | |
float32Tag = '[object Float32Array]', | |
float64Tag = '[object Float64Array]', | |
int8Tag = '[object Int8Array]', | |
int16Tag = '[object Int16Array]', | |
int32Tag = '[object Int32Array]', | |
uint8Tag = '[object Uint8Array]', | |
uint8ClampedTag = '[object Uint8ClampedArray]', | |
uint16Tag = '[object Uint16Array]', | |
uint32Tag = '[object Uint32Array]'; | |
/** Used to match empty string literals in compiled template source. */ | |
var reEmptyStringLeading = /\b__p \+= '';/g, | |
reEmptyStringMiddle = /\b(__p \+=) '' \+/g, | |
reEmptyStringTrailing = /(__e\(.*?\)|\b__t\)) \+\n'';/g; | |
/** Used to match HTML entities and HTML characters. */ | |
var reEscapedHtml = /&(?:amp|lt|gt|quot|#39|#96);/g, | |
reUnescapedHtml = /[&<>"'`]/g, | |
reHasEscapedHtml = RegExp(reEscapedHtml.source), | |
reHasUnescapedHtml = RegExp(reUnescapedHtml.source); | |
/** Used to match template delimiters. */ | |
var reEscape = /<%-([\s\S]+?)%>/g, | |
reEvaluate = /<%([\s\S]+?)%>/g, | |
reInterpolate = /<%=([\s\S]+?)%>/g; | |
/** Used to match property names within property paths. */ | |
var reIsDeepProp = /\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/, | |
reIsPlainProp = /^\w*$/, | |
rePropName = /[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(\.|\[\])(?:\4|$))/g; | |
/** | |
* Used to match `RegExp` | |
* [syntax characters](http://ecma-international.org/ecma-262/6.0/#sec-patterns). | |
*/ | |
var reRegExpChar = /[\\^$.*+?()[\]{}|]/g, | |
reHasRegExpChar = RegExp(reRegExpChar.source); | |
/** Used to match leading and trailing whitespace. */ | |
var reTrim = /^\s+|\s+$/g, | |
reTrimStart = /^\s+/, | |
reTrimEnd = /\s+$/; | |
/** Used to match non-compound words composed of alphanumeric characters. */ | |
var reBasicWord = /[a-zA-Z0-9]+/g; | |
/** Used to match backslashes in property paths. */ | |
var reEscapeChar = /\\(\\)?/g; | |
/** | |
* Used to match | |
* [ES template delimiters](http://ecma-international.org/ecma-262/6.0/#sec-template-literal-lexical-components). | |
*/ | |
var reEsTemplate = /\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g; | |
/** Used to match `RegExp` flags from their coerced string values. */ | |
var reFlags = /\w*$/; | |
/** Used to detect hexadecimal string values. */ | |
var reHasHexPrefix = /^0x/i; | |
/** Used to detect host constructors (Safari). */ | |
var reIsHostCtor = /^\[object .+?Constructor\]$/; | |
/** Used to detect unsigned integer values. */ | |
var reIsUint = /^(?:0|[1-9]\d*)$/; | |
/** Used to match latin-1 supplementary letters (excluding mathematical operators). */ | |
var reLatin1 = /[\xc0-\xd6\xd8-\xde\xdf-\xf6\xf8-\xff]/g; | |
/** Used to ensure capturing order of template delimiters. */ | |
var reNoMatch = /($^)/; | |
/** Used to match unescaped characters in compiled string literals. */ | |
var reUnescapedString = /['\n\r\u2028\u2029\\]/g; | |
/** Used to compose unicode character classes. */ | |
var rsAstralRange = '\\ud800-\\udfff', | |
rsComboMarksRange = '\\u0300-\\u036f\\ufe20-\\ufe23', | |
rsComboSymbolsRange = '\\u20d0-\\u20f0', | |
rsDingbatRange = '\\u2700-\\u27bf', | |
rsLowerRange = 'a-z\\xdf-\\xf6\\xf8-\\xff', | |
rsMathOpRange = '\\xac\\xb1\\xd7\\xf7', | |
rsNonCharRange = '\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf', | |
rsPunctuationRange = '\\u2000-\\u206f', | |
rsSpaceRange = ' \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000', | |
rsUpperRange = 'A-Z\\xc0-\\xd6\\xd8-\\xde', | |
rsVarRange = '\\ufe0e\\ufe0f', | |
rsBreakRange = rsMathOpRange + rsNonCharRange + rsPunctuationRange + rsSpaceRange; | |
/** Used to compose unicode capture groups. */ | |
var rsApos = "['\u2019]", | |
rsAstral = '[' + rsAstralRange + ']', | |
rsBreak = '[' + rsBreakRange + ']', | |
rsCombo = '[' + rsComboMarksRange + rsComboSymbolsRange + ']', | |
rsDigits = '\\d+', | |
rsDingbat = '[' + rsDingbatRange + ']', | |
rsLower = '[' + rsLowerRange + ']', | |
rsMisc = '[^' + rsAstralRange + rsBreakRange + rsDigits + rsDingbatRange + rsLowerRange + rsUpperRange + ']', | |
rsFitz = '\\ud83c[\\udffb-\\udfff]', | |
rsModifier = '(?:' + rsCombo + '|' + rsFitz + ')', | |
rsNonAstral = '[^' + rsAstralRange + ']', | |
rsRegional = '(?:\\ud83c[\\udde6-\\uddff]){2}', | |
rsSurrPair = '[\\ud800-\\udbff][\\udc00-\\udfff]', | |
rsUpper = '[' + rsUpperRange + ']', | |
rsZWJ = '\\u200d'; | |
/** Used to compose unicode regexes. */ | |
var rsLowerMisc = '(?:' + rsLower + '|' + rsMisc + ')', | |
rsUpperMisc = '(?:' + rsUpper + '|' + rsMisc + ')', | |
rsOptLowerContr = '(?:' + rsApos + '(?:d|ll|m|re|s|t|ve))?', | |
rsOptUpperContr = '(?:' + rsApos + '(?:D|LL|M|RE|S|T|VE))?', | |
reOptMod = rsModifier + '?', | |
rsOptVar = '[' + rsVarRange + ']?', | |
rsOptJoin = '(?:' + rsZWJ + '(?:' + [rsNonAstral, rsRegional, rsSurrPair].join('|') + ')' + rsOptVar + reOptMod + ')*', | |
rsSeq = rsOptVar + reOptMod + rsOptJoin, | |
rsEmoji = '(?:' + [rsDingbat, rsRegional, rsSurrPair].join('|') + ')' + rsSeq, | |
rsSymbol = '(?:' + [rsNonAstral + rsCombo + '?', rsCombo, rsRegional, rsSurrPair, rsAstral].join('|') + ')'; | |
/** Used to match apostrophes. */ | |
var reApos = RegExp(rsApos, 'g'); | |
/** | |
* Used to match [combining diacritical marks](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks) and | |
* [combining diacritical marks for symbols](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks_for_Symbols). | |
*/ | |
var reComboMark = RegExp(rsCombo, 'g'); | |
/** Used to match [string symbols](https://mathiasbynens.be/notes/javascript-unicode). */ | |
var reComplexSymbol = RegExp(rsFitz + '(?=' + rsFitz + ')|' + rsSymbol + rsSeq, 'g'); | |
/** Used to match complex or compound words. */ | |
var reComplexWord = RegExp([ | |
rsUpper + '?' + rsLower + '+' + rsOptLowerContr + '(?=' + [rsBreak, rsUpper, '$'].join('|') + ')', | |
rsUpperMisc + '+' + rsOptUpperContr + '(?=' + [rsBreak, rsUpper + rsLowerMisc, '$'].join('|') + ')', | |
rsUpper + '?' + rsLowerMisc + '+' + rsOptLowerContr, | |
rsUpper + '+' + rsOptUpperContr, | |
rsDigits, | |
rsEmoji | |
].join('|'), 'g'); | |
/** Used to detect strings with [zero-width joiners or code points from the astral planes](http://eev.ee/blog/2015/09/12/dark-corners-of-unicode/). */ | |
var reHasComplexSymbol = RegExp('[' + rsZWJ + rsAstralRange + rsComboMarksRange + rsComboSymbolsRange + rsVarRange + ']'); | |
/** Used to detect strings that need a more robust regexp to match words. */ | |
var reHasComplexWord = /[a-z][A-Z]|[A-Z]{2,}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/; | |
/** Used to make template sourceURLs easier to identify. */ | |
var templateCounter = -1; | |
/** Used to identify `toStringTag` values of typed arrays. */ | |
var typedArrayTags = {}; | |
typedArrayTags[float32Tag] = typedArrayTags[float64Tag] = | |
typedArrayTags[int8Tag] = typedArrayTags[int16Tag] = | |
typedArrayTags[int32Tag] = typedArrayTags[uint8Tag] = | |
typedArrayTags[uint8ClampedTag] = typedArrayTags[uint16Tag] = | |
typedArrayTags[uint32Tag] = true; | |
typedArrayTags[argsTag] = typedArrayTags[arrayTag] = | |
typedArrayTags[arrayBufferTag] = typedArrayTags[boolTag] = | |
typedArrayTags[dataViewTag] = typedArrayTags[dateTag] = | |
typedArrayTags[errorTag] = typedArrayTags[funcTag] = | |
typedArrayTags[mapTag] = typedArrayTags[numberTag] = | |
typedArrayTags[objectTag] = typedArrayTags[regexpTag] = | |
typedArrayTags[setTag] = typedArrayTags[stringTag] = | |
typedArrayTags[weakMapTag] = false; | |
/** Used to identify `toStringTag` values supported by `_.clone`. */ | |
var cloneableTags = {}; | |
cloneableTags[argsTag] = cloneableTags[arrayTag] = | |
cloneableTags[arrayBufferTag] = cloneableTags[dataViewTag] = | |
cloneableTags[boolTag] = cloneableTags[dateTag] = | |
cloneableTags[float32Tag] = cloneableTags[float64Tag] = | |
cloneableTags[int8Tag] = cloneableTags[int16Tag] = | |
cloneableTags[int32Tag] = cloneableTags[mapTag] = | |
cloneableTags[numberTag] = cloneableTags[objectTag] = | |
cloneableTags[regexpTag] = cloneableTags[setTag] = | |
cloneableTags[stringTag] = cloneableTags[symbolTag] = | |
cloneableTags[uint8Tag] = cloneableTags[uint8ClampedTag] = | |
cloneableTags[uint16Tag] = cloneableTags[uint32Tag] = true; | |
cloneableTags[errorTag] = cloneableTags[funcTag] = | |
cloneableTags[weakMapTag] = false; | |
/** Used to map latin-1 supplementary letters to basic latin letters. */ | |
var deburredLetters = { | |
'\xc0': 'A', '\xc1': 'A', '\xc2': 'A', '\xc3': 'A', '\xc4': 'A', '\xc5': 'A', | |
'\xe0': 'a', '\xe1': 'a', '\xe2': 'a', '\xe3': 'a', '\xe4': 'a', '\xe5': 'a', | |
'\xc7': 'C', '\xe7': 'c', | |
'\xd0': 'D', '\xf0': 'd', | |
'\xc8': 'E', '\xc9': 'E', '\xca': 'E', '\xcb': 'E', | |
'\xe8': 'e', '\xe9': 'e', '\xea': 'e', '\xeb': 'e', | |
'\xcC': 'I', '\xcd': 'I', '\xce': 'I', '\xcf': 'I', | |
'\xeC': 'i', '\xed': 'i', '\xee': 'i', '\xef': 'i', | |
'\xd1': 'N', '\xf1': 'n', | |
'\xd2': 'O', '\xd3': 'O', '\xd4': 'O', '\xd5': 'O', '\xd6': 'O', '\xd8': 'O', | |
'\xf2': 'o', '\xf3': 'o', '\xf4': 'o', '\xf5': 'o', '\xf6': 'o', '\xf8': 'o', | |
'\xd9': 'U', '\xda': 'U', '\xdb': 'U', '\xdc': 'U', | |
'\xf9': 'u', '\xfa': 'u', '\xfb': 'u', '\xfc': 'u', | |
'\xdd': 'Y', '\xfd': 'y', '\xff': 'y', | |
'\xc6': 'Ae', '\xe6': 'ae', | |
'\xde': 'Th', '\xfe': 'th', | |
'\xdf': 'ss' | |
}; | |
/** Used to map characters to HTML entities. */ | |
var htmlEscapes = { | |
'&': '&', | |
'<': '<', | |
'>': '>', | |
'"': '"', | |
"'": ''', | |
'`': '`' | |
}; | |
/** Used to map HTML entities to characters. */ | |
var htmlUnescapes = { | |
'&': '&', | |
'<': '<', | |
'>': '>', | |
'"': '"', | |
''': "'", | |
'`': '`' | |
}; | |
/** Used to escape characters for inclusion in compiled string literals. */ | |
var stringEscapes = { | |
'\\': '\\', | |
"'": "'", | |
'\n': 'n', | |
'\r': 'r', | |
'\u2028': 'u2028', | |
'\u2029': 'u2029' | |
}; | |
/** Detect free variable `exports`. */ | |
var freeExports = typeof exports == 'object' && exports; | |
/** Detect free variable `module`. */ | |
var freeModule = freeExports && typeof module == 'object' && module; | |
/** Detect the popular CommonJS extension `module.exports`. */ | |
var moduleExports = freeModule && freeModule.exports === freeExports; | |
/** Detect free variable `global` from Node.js. */ | |
var freeGlobal = checkGlobal(typeof global == 'object' && global); | |
/** Detect free variable `self`. */ | |
var freeSelf = checkGlobal(typeof self == 'object' && self); | |
/** Detect `this` as the global object. */ | |
var thisGlobal = checkGlobal(typeof this == 'object' && this); | |
/** Used as a reference to the global object. */ | |
var root = freeGlobal || freeSelf || thisGlobal || Function('return this')(); | |
/*--------------------------------------------------------------------------*/ | |
/** | |
* 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 doesn't return the map instance 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) { | |
set.add(value); | |
return set; | |
} | |
/** | |
* A specialized version of `_.includes` for arrays without support for | |
* specifying an index to search from. | |
* | |
* @private | |
* @param {Array} [array] The array to search. | |
* @param {*} target The value to search for. | |
* @returns {boolean} Returns `true` if `target` is found, else `false`. | |
*/ | |
function arrayIncludes(array, value) { | |
var length = array ? array.length : 0; | |
return !!length && baseIndexOf(array, value, 0) > -1; | |
} | |
/** | |
* This function is like `arrayIncludes` except that it accepts a comparator. | |
* | |
* @private | |
* @param {Array} [array] The array to search. | |
* @param {*} target The value to search for. | |
* @param {Function} comparator The comparator invoked per element. | |
* @returns {boolean} Returns `true` if `target` is found, else `false`. | |
*/ | |
function arrayIncludesWith(array, value, comparator) { | |
var index = -1, | |
length = array ? array.length : 0; | |
while (++index < length) { | |
if (comparator(value, array[index])) { | |
return true; | |
} | |
} | |
return false; | |
} | |
/** | |
* Appends the elements of `values` to `array`. | |
* | |
* @private | |
* @param {Array} array The array to modify. | |
* @param {Array} values The values to append. | |
* @returns {Array} Returns `array`. | |
*/ | |
function arrayPush(array, values) { | |
array.push.apply(array, values); | |
return array; | |
} | |
/** | |
* The base implementation of `_.findIndex` and `_.findLastIndex` without | |
* support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array} array The array to search. | |
* @param {Function} predicate The function invoked per iteration. | |
* @param {number} fromIndex The index to search from. | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {number} Returns the index of the matched value, else `-1`. | |
*/ | |
function baseFindIndex(array, predicate, fromIndex, fromRight) { | |
var length = array.length, | |
index = fromIndex + (fromRight ? 1 : -1); | |
while ((fromRight ? index-- : ++index < length)) { | |
if (predicate(array[index], index, array)) { | |
return index; | |
} | |
} | |
return -1; | |
} | |
/** | |
* The base implementation of `_.indexOf` without `fromIndex` bounds checks. | |
* | |
* @private | |
* @param {Array} array The array to search. | |
* @param {*} value The value to search for. | |
* @param {number} fromIndex The index to search from. | |
* @returns {number} Returns the index of the matched value, else `-1`. | |
*/ | |
function baseIndexOf(array, value, fromIndex) { | |
if (value !== value) { | |
return indexOfNaN(array, fromIndex); | |
} | |
var index = fromIndex - 1, | |
length = array.length; | |
while (++index < length) { | |
if (array[index] === value) { | |
return index; | |
} | |
} | |
return -1; | |
} | |
/** | |
* The base implementation of `_.reduce` and `_.reduceRight`, without support | |
* for iteratee shorthands, which iterates over `collection` using `eachFunc`. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @param {*} accumulator The initial value. | |
* @param {boolean} initAccum Specify using the first or last element of | |
* `collection` as the initial value. | |
* @param {Function} eachFunc The function to iterate over `collection`. | |
* @returns {*} Returns the accumulated value. | |
*/ | |
function baseReduce(collection, iteratee, accumulator, initAccum, eachFunc) { | |
eachFunc(collection, function(value, index, collection) { | |
accumulator = initAccum | |
? (initAccum = false, value) | |
: iteratee(accumulator, value, index, collection); | |
}); | |
return accumulator; | |
} | |
/** | |
* The base implementation of `_.times` without support for iteratee shorthands | |
* or max array length checks. | |
* | |
* @private | |
* @param {number} n The number of times to invoke `iteratee`. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @returns {Array} Returns the array of results. | |
*/ | |
function baseTimes(n, iteratee) { | |
var index = -1, | |
result = Array(n); | |
while (++index < n) { | |
result[index] = iteratee(index); | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.unary` without support for storing wrapper metadata. | |
* | |
* @private | |
* @param {Function} func The function to cap arguments for. | |
* @returns {Function} Returns the new capped function. | |
*/ | |
function baseUnary(func) { | |
return function(value) { | |
return func(value); | |
}; | |
} | |
/** | |
* The base implementation of `_.values` and `_.valuesIn` which creates an | |
* array of `object` property values corresponding to the property names | |
* of `props`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {Array} props The property names to get values for. | |
* @returns {Object} Returns the array of property values. | |
*/ | |
function baseValues(object, props) { | |
return baseMap(props, function(key) { | |
return object[key]; | |
}); | |
} | |
/** | |
* Checks if a cache value for `key` exists. | |
* | |
* @private | |
* @param {Object} cache The cache to query. | |
* @param {string} key The key of the entry to check. | |
* @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. | |
*/ | |
function cacheHas(cache, key) { | |
return cache.has(key); | |
} | |
/** | |
* Used by `_.trim` and `_.trimStart` to get the index of the first string symbol | |
* that is not found in the character symbols. | |
* | |
* @private | |
* @param {Array} strSymbols The string symbols to inspect. | |
* @param {Array} chrSymbols The character symbols to find. | |
* @returns {number} Returns the index of the first unmatched string symbol. | |
*/ | |
function charsStartIndex(strSymbols, chrSymbols) { | |
var index = -1, | |
length = strSymbols.length; | |
while (++index < length && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {} | |
return index; | |
} | |
/** | |
* Used by `_.trim` and `_.trimEnd` to get the index of the last string symbol | |
* that is not found in the character symbols. | |
* | |
* @private | |
* @param {Array} strSymbols The string symbols to inspect. | |
* @param {Array} chrSymbols The character symbols to find. | |
* @returns {number} Returns the index of the last unmatched string symbol. | |
*/ | |
function charsEndIndex(strSymbols, chrSymbols) { | |
var index = strSymbols.length; | |
while (index-- && baseIndexOf(chrSymbols, strSymbols[index], 0) > -1) {} | |
return index; | |
} | |
/** | |
* Checks if `value` is a global object. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {null|Object} Returns `value` if it's a global object, else `null`. | |
*/ | |
function checkGlobal(value) { | |
return (value && value.Object === Object) ? value : null; | |
} | |
/** | |
* Used by `_.deburr` to convert latin-1 supplementary letters to basic latin letters. | |
* | |
* @private | |
* @param {string} letter The matched letter to deburr. | |
* @returns {string} Returns the deburred letter. | |
*/ | |
function deburrLetter(letter) { | |
return deburredLetters[letter]; | |
} | |
/** | |
* Used by `_.escape` to convert characters to HTML entities. | |
* | |
* @private | |
* @param {string} chr The matched character to escape. | |
* @returns {string} Returns the escaped character. | |
*/ | |
function escapeHtmlChar(chr) { | |
return htmlEscapes[chr]; | |
} | |
/** | |
* Used by `_.template` to escape characters for inclusion in compiled string literals. | |
* | |
* @private | |
* @param {string} chr The matched character to escape. | |
* @returns {string} Returns the escaped character. | |
*/ | |
function escapeStringChar(chr) { | |
return '\\' + stringEscapes[chr]; | |
} | |
/** | |
* Gets the value at `key` of `object`. | |
* | |
* @private | |
* @param {Object} [object] The object to query. | |
* @param {string} key The key of the property to get. | |
* @returns {*} Returns the property value. | |
*/ | |
function getValue(object, key) { | |
return object == null ? undefined : object[key]; | |
} | |
/** | |
* Gets the index at which the first occurrence of `NaN` is found in `array`. | |
* | |
* @private | |
* @param {Array} array The array to search. | |
* @param {number} fromIndex The index to search from. | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {number} Returns the index of the matched `NaN`, else `-1`. | |
*/ | |
function indexOfNaN(array, fromIndex, fromRight) { | |
var length = array.length, | |
index = fromIndex + (fromRight ? 1 : -1); | |
while ((fromRight ? index-- : ++index < length)) { | |
var other = array[index]; | |
if (other !== other) { | |
return index; | |
} | |
} | |
return -1; | |
} | |
/** | |
* Checks if `value` is a host object in IE < 9. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a host object, else `false`. | |
*/ | |
function isHostObject() { | |
return false; | |
} | |
/** | |
* Converts `map` to its key-value pairs. | |
* | |
* @private | |
* @param {Object} map The map to convert. | |
* @returns {Array} Returns the key-value pairs. | |
*/ | |
function mapToArray(map) { | |
var index = -1, | |
result = Array(map.size); | |
map.forEach(function(value, key) { | |
result[++index] = [key, value]; | |
}); | |
return result; | |
} | |
/** | |
* Converts `set` to an array of its values. | |
* | |
* @private | |
* @param {Object} set The set to convert. | |
* @returns {Array} Returns the values. | |
*/ | |
function setToArray(set) { | |
var index = -1, | |
result = Array(set.size); | |
set.forEach(function(value) { | |
result[++index] = value; | |
}); | |
return result; | |
} | |
/** | |
* Gets the number of symbols in `string`. | |
* | |
* @private | |
* @param {string} string The string to inspect. | |
* @returns {number} Returns the string size. | |
*/ | |
function stringSize(string) { | |
if (!(string && reHasComplexSymbol.test(string))) { | |
return string.length; | |
} | |
var result = reComplexSymbol.lastIndex = 0; | |
while (reComplexSymbol.test(string)) { | |
result++; | |
} | |
return result; | |
} | |
/** | |
* Converts `string` to an array. | |
* | |
* @private | |
* @param {string} string The string to convert. | |
* @returns {Array} Returns the converted array. | |
*/ | |
function stringToArray(string) { | |
return string.match(reComplexSymbol); | |
} | |
/** | |
* Used by `_.unescape` to convert HTML entities to characters. | |
* | |
* @private | |
* @param {string} chr The matched character to unescape. | |
* @returns {string} Returns the unescaped character. | |
*/ | |
function unescapeHtmlChar(chr) { | |
return htmlUnescapes[chr]; | |
} | |
/*--------------------------------------------------------------------------*/ | |
/** Used for built-in method references. */ | |
var arrayProto = Array.prototype, | |
objectProto = Object.prototype, | |
stringProto = String.prototype; | |
/** Used to detect overreaching core-js shims. */ | |
var coreJsData = root['__core-js_shared__']; | |
/** Used to detect methods masquerading as native. */ | |
var maskSrcKey = (function() { | |
var uid = /[^.]+$/.exec(coreJsData && coreJsData.keys && coreJsData.keys.IE_PROTO || ''); | |
return uid ? ('Symbol(src)_1.' + uid) : ''; | |
}()); | |
/** Used to resolve the decompiled source of functions. */ | |
var funcToString = Function.prototype.toString; | |
/** Used to check objects for own properties. */ | |
var hasOwnProperty = objectProto.hasOwnProperty; | |
/** Used to generate unique IDs. */ | |
var idCounter = 0; | |
/** Used to infer the `Object` constructor. */ | |
var objectCtorString = funcToString.call(Object); | |
/** | |
* Used to resolve the | |
* [`toStringTag`](http://ecma-international.org/ecma-262/6.0/#sec-object.prototype.tostring) | |
* of values. | |
*/ | |
var objectToString = objectProto.toString; | |
/** Used to restore the original `_` reference in `_.noConflict`. */ | |
var oldDash = root._; | |
/** Used to detect if a method is native. */ | |
var reIsNative = RegExp('^' + | |
funcToString.call(hasOwnProperty).replace(reRegExpChar, '\\$&') | |
.replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g, '$1.*?') + '$' | |
); | |
/** Built-in value references. */ | |
var Buffer = moduleExports ? root.Buffer : undefined, | |
getOwnPropertySymbols = Object.getOwnPropertySymbols, | |
objectCreate = Object.create, | |
propertyIsEnumerable = objectProto.propertyIsEnumerable, | |
splice = arrayProto.splice; | |
/* Built-in method references for those with the same name as other `lodash` methods. */ | |
var nativeCeil = Math.ceil, | |
nativeFloor = Math.floor, | |
nativeGetPrototype = Object.getPrototypeOf, | |
nativeIsFinite = root.isFinite, | |
nativeKeys = Object.keys, | |
nativeMax = Math.max, | |
nativeMin = Math.min, | |
nativeParseInt = root.parseInt, | |
nativeReplace = stringProto.replace, | |
nativeSplit = stringProto.split; | |
/* Built-in method references that are verified to be native. */ | |
var nativeCreate = getNative(Object, 'create'); | |
/** Used to detect maps, sets, and weakmaps. */ | |
var dataViewCtorString = toSource(DataView), | |
mapCtorString = toSource(Map), | |
promiseCtorString = toSource(Promise), | |
setCtorString = toSource(Set), | |
weakMapCtorString = toSource(WeakMap); | |
/** Used to convert symbols to primitives and strings. */ | |
var symbolProto = Symbol ? Symbol.prototype : undefined, | |
symbolValueOf = symbolProto ? symbolProto.valueOf : undefined, | |
symbolToString = symbolProto ? symbolProto.toString : undefined; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a `lodash` object which wraps `value` to enable implicit method | |
* chain sequences. Methods that operate on and return arrays, collections, | |
* and functions can be chained together. Methods that retrieve a single value | |
* or may return a primitive value will automatically end the chain sequence | |
* and return the unwrapped value. Otherwise, the value must be unwrapped | |
* with `_#value`. | |
* | |
* Explicit chain sequences, which must be unwrapped with `_#value`, may be | |
* enabled using `_.chain`. | |
* | |
* The execution of chained methods is lazy, that is, it's deferred until | |
* `_#value` is implicitly or explicitly called. | |
* | |
* Lazy evaluation allows several methods to support shortcut fusion. | |
* Shortcut fusion is an optimization to merge iteratee calls; this avoids | |
* the creation of intermediate arrays and can greatly reduce the number of | |
* iteratee executions. Sections of a chain sequence qualify for shortcut | |
* fusion if the section is applied to an array of at least `200` elements | |
* and any iteratees accept only one argument. The heuristic for whether a | |
* section qualifies for shortcut fusion is subject to change. | |
* | |
* Chaining is supported in custom builds as long as the `_#value` method is | |
* directly or indirectly included in the build. | |
* | |
* In addition to lodash methods, wrappers have `Array` and `String` methods. | |
* | |
* The wrapper `Array` methods are: | |
* `concat`, `join`, `pop`, `push`, `shift`, `sort`, `splice`, and `unshift` | |
* | |
* The wrapper `String` methods are: | |
* `replace` and `split` | |
* | |
* The wrapper methods that support shortcut fusion are: | |
* `at`, `compact`, `drop`, `dropRight`, `dropWhile`, `filter`, `find`, | |
* `findLast`, `head`, `initial`, `last`, `map`, `reject`, `reverse`, `slice`, | |
* `tail`, `take`, `takeRight`, `takeRightWhile`, `takeWhile`, and `toArray` | |
* | |
* The chainable wrapper methods are: | |
* `after`, `ary`, `assign`, `assignIn`, `assignInWith`, `assignWith`, `at`, | |
* `before`, `bind`, `bindAll`, `bindKey`, `castArray`, `chain`, `chunk`, | |
* `commit`, `compact`, `concat`, `conforms`, `constant`, `countBy`, `create`, | |
* `curry`, `debounce`, `defaults`, `defaultsDeep`, `defer`, `delay`, | |
* `difference`, `differenceBy`, `differenceWith`, `drop`, `dropRight`, | |
* `dropRightWhile`, `dropWhile`, `extend`, `extendWith`, `fill`, `filter`, | |
* `flatMap`, `flatMapDeep`, `flatMapDepth`, `flatten`, `flattenDeep`, | |
* `flattenDepth`, `flip`, `flow`, `flowRight`, `fromPairs`, `functions`, | |
* `functionsIn`, `groupBy`, `initial`, `intersection`, `intersectionBy`, | |
* `intersectionWith`, `invert`, `invertBy`, `invokeMap`, `iteratee`, `keyBy`, | |
* `keys`, `keysIn`, `map`, `mapKeys`, `mapValues`, `matches`, `matchesProperty`, | |
* `memoize`, `merge`, `mergeWith`, `method`, `methodOf`, `mixin`, `negate`, | |
* `nthArg`, `omit`, `omitBy`, `once`, `orderBy`, `over`, `overArgs`, | |
* `overEvery`, `overSome`, `partial`, `partialRight`, `partition`, `pick`, | |
* `pickBy`, `plant`, `property`, `propertyOf`, `pull`, `pullAll`, `pullAllBy`, | |
* `pullAllWith`, `pullAt`, `push`, `range`, `rangeRight`, `rearg`, `reject`, | |
* `remove`, `rest`, `reverse`, `sampleSize`, `set`, `setWith`, `shuffle`, | |
* `slice`, `sort`, `sortBy`, `splice`, `spread`, `tail`, `take`, `takeRight`, | |
* `takeRightWhile`, `takeWhile`, `tap`, `throttle`, `thru`, `toArray`, | |
* `toPairs`, `toPairsIn`, `toPath`, `toPlainObject`, `transform`, `unary`, | |
* `union`, `unionBy`, `unionWith`, `uniq`, `uniqBy`, `uniqWith`, `unset`, | |
* `unshift`, `unzip`, `unzipWith`, `update`, `updateWith`, `values`, | |
* `valuesIn`, `without`, `wrap`, `xor`, `xorBy`, `xorWith`, `zip`, | |
* `zipObject`, `zipObjectDeep`, and `zipWith` | |
* | |
* The wrapper methods that are **not** chainable by default are: | |
* `add`, `attempt`, `camelCase`, `capitalize`, `ceil`, `clamp`, `clone`, | |
* `cloneDeep`, `cloneDeepWith`, `cloneWith`, `deburr`, `divide`, `each`, | |
* `eachRight`, `endsWith`, `eq`, `escape`, `escapeRegExp`, `every`, `find`, | |
* `findIndex`, `findKey`, `findLast`, `findLastIndex`, `findLastKey`, `first`, | |
* `floor`, `forEach`, `forEachRight`, `forIn`, `forInRight`, `forOwn`, | |
* `forOwnRight`, `get`, `gt`, `gte`, `has`, `hasIn`, `head`, `identity`, | |
* `includes`, `indexOf`, `inRange`, `invoke`, `isArguments`, `isArray`, | |
* `isArrayBuffer`, `isArrayLike`, `isArrayLikeObject`, `isBoolean`, | |
* `isBuffer`, `isDate`, `isElement`, `isEmpty`, `isEqual`, `isEqualWith`, | |
* `isError`, `isFinite`, `isFunction`, `isInteger`, `isLength`, `isMap`, | |
* `isMatch`, `isMatchWith`, `isNaN`, `isNative`, `isNil`, `isNull`, | |
* `isNumber`, `isObject`, `isObjectLike`, `isPlainObject`, `isRegExp`, | |
* `isSafeInteger`, `isSet`, `isString`, `isUndefined`, `isTypedArray`, | |
* `isWeakMap`, `isWeakSet`, `join`, `kebabCase`, `last`, `lastIndexOf`, | |
* `lowerCase`, `lowerFirst`, `lt`, `lte`, `max`, `maxBy`, `mean`, `meanBy`, | |
* `min`, `minBy`, `multiply`, `noConflict`, `noop`, `now`, `nth`, `pad`, | |
* `padEnd`, `padStart`, `parseInt`, `pop`, `random`, `reduce`, `reduceRight`, | |
* `repeat`, `result`, `round`, `runInContext`, `sample`, `shift`, `size`, | |
* `snakeCase`, `some`, `sortedIndex`, `sortedIndexBy`, `sortedLastIndex`, | |
* `sortedLastIndexBy`, `startCase`, `startsWith`, `stubArray`, `stubFalse`, | |
* `stubObject`, `stubString`, `stubTrue`, `subtract`, `sum`, `sumBy`, | |
* `template`, `times`, `toFinite`, `toInteger`, `toJSON`, `toLength`, | |
* `toLower`, `toNumber`, `toSafeInteger`, `toString`, `toUpper`, `trim`, | |
* `trimEnd`, `trimStart`, `truncate`, `unescape`, `uniqueId`, `upperCase`, | |
* `upperFirst`, `value`, and `words` | |
* | |
* @name _ | |
* @constructor | |
* @category Seq | |
* @param {*} value The value to wrap in a `lodash` instance. | |
* @returns {Object} Returns the new `lodash` wrapper instance. | |
* @example | |
* | |
* function square(n) { | |
* return n * n; | |
* } | |
* | |
* var wrapped = _([1, 2, 3]); | |
* | |
* // Returns an unwrapped value. | |
* wrapped.reduce(_.add); | |
* // => 6 | |
* | |
* // Returns a wrapped value. | |
* var squares = wrapped.map(square); | |
* | |
* _.isArray(squares); | |
* // => false | |
* | |
* _.isArray(squares.value()); | |
* // => true | |
*/ | |
function lodash(value) { | |
return value instanceof LodashWrapper | |
? value | |
: new LodashWrapper(value); | |
} | |
/** | |
* The base constructor for creating `lodash` wrapper objects. | |
* | |
* @private | |
* @param {*} value The value to wrap. | |
* @param {boolean} [chainAll] Enable explicit method chain sequences. | |
*/ | |
function LodashWrapper(value, chainAll) { | |
this.__wrapped__ = value; | |
this.__actions__ = []; | |
this.__chain__ = !!chainAll; | |
} | |
/** | |
* By default, the template delimiters used by lodash are like those in | |
* embedded Ruby (ERB). Change the following template settings to use | |
* alternative delimiters. | |
* | |
* @static | |
* @memberOf _ | |
* @type {Object} | |
*/ | |
lodash.templateSettings = { | |
/** | |
* Used to detect `data` property values to be HTML-escaped. | |
* | |
* @memberOf _.templateSettings | |
* @type {RegExp} | |
*/ | |
'escape': reEscape, | |
/** | |
* Used to detect code to be evaluated. | |
* | |
* @memberOf _.templateSettings | |
* @type {RegExp} | |
*/ | |
'evaluate': reEvaluate, | |
/** | |
* Used to detect `data` property values to inject. | |
* | |
* @memberOf _.templateSettings | |
* @type {RegExp} | |
*/ | |
'interpolate': reInterpolate, | |
/** | |
* Used to reference the data object in the template text. | |
* | |
* @memberOf _.templateSettings | |
* @type {string} | |
*/ | |
'variable': '', | |
/** | |
* Used to import variables into the compiled template. | |
* | |
* @memberOf _.templateSettings | |
* @type {Object} | |
*/ | |
'imports': { | |
/** | |
* A reference to the `lodash` function. | |
* | |
* @memberOf _.templateSettings.imports | |
* @type {Function} | |
*/ | |
'_': lodash | |
} | |
}; | |
LodashWrapper.prototype = baseCreate(lodash.prototype); | |
LodashWrapper.prototype.constructor = LodashWrapper; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a hash object. | |
* | |
* @private | |
* @constructor | |
* @param {Array} [entries] The key-value pairs to cache. | |
*/ | |
function Hash(entries) { | |
var index = -1, | |
length = entries ? entries.length : 0; | |
this.clear(); | |
while (++index < length) { | |
var entry = entries[index]; | |
this.set(entry[0], entry[1]); | |
} | |
} | |
/** | |
* Removes all key-value entries from the hash. | |
* | |
* @private | |
* @name clear | |
* @memberOf Hash | |
*/ | |
function hashClear() { | |
this.__data__ = nativeCreate ? nativeCreate(null) : {}; | |
} | |
/** | |
* Removes `key` and its value from the hash. | |
* | |
* @private | |
* @name delete | |
* @memberOf Hash | |
* @param {Object} hash The hash to modify. | |
* @param {string} key The key of the value to remove. | |
* @returns {boolean} Returns `true` if the entry was removed, else `false`. | |
*/ | |
function hashDelete(key) { | |
return this.has(key) && delete this.__data__[key]; | |
} | |
/** | |
* Gets the hash value for `key`. | |
* | |
* @private | |
* @name get | |
* @memberOf Hash | |
* @param {string} key The key of the value to get. | |
* @returns {*} Returns the entry value. | |
*/ | |
function hashGet(key) { | |
var data = this.__data__; | |
if (nativeCreate) { | |
var result = data[key]; | |
return result === HASH_UNDEFINED ? undefined : result; | |
} | |
return hasOwnProperty.call(data, key) ? data[key] : undefined; | |
} | |
/** | |
* Checks if a hash value for `key` exists. | |
* | |
* @private | |
* @name has | |
* @memberOf Hash | |
* @param {string} key The key of the entry to check. | |
* @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. | |
*/ | |
function hashHas(key) { | |
var data = this.__data__; | |
return nativeCreate ? data[key] !== undefined : hasOwnProperty.call(data, key); | |
} | |
/** | |
* Sets the hash `key` to `value`. | |
* | |
* @private | |
* @name set | |
* @memberOf Hash | |
* @param {string} key The key of the value to set. | |
* @param {*} value The value to set. | |
* @returns {Object} Returns the hash instance. | |
*/ | |
function hashSet(key, value) { | |
var data = this.__data__; | |
data[key] = (nativeCreate && value === undefined) ? HASH_UNDEFINED : value; | |
return this; | |
} | |
// Add methods to `Hash`. | |
Hash.prototype.clear = hashClear; | |
Hash.prototype['delete'] = hashDelete; | |
Hash.prototype.get = hashGet; | |
Hash.prototype.has = hashHas; | |
Hash.prototype.set = hashSet; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates an list cache object. | |
* | |
* @private | |
* @constructor | |
* @param {Array} [entries] The key-value pairs to cache. | |
*/ | |
function ListCache(entries) { | |
var index = -1, | |
length = entries ? entries.length : 0; | |
this.clear(); | |
while (++index < length) { | |
var entry = entries[index]; | |
this.set(entry[0], entry[1]); | |
} | |
} | |
/** | |
* Removes all key-value entries from the list cache. | |
* | |
* @private | |
* @name clear | |
* @memberOf ListCache | |
*/ | |
function listCacheClear() { | |
this.__data__ = []; | |
} | |
/** | |
* Removes `key` and its value from the list cache. | |
* | |
* @private | |
* @name delete | |
* @memberOf ListCache | |
* @param {string} key The key of the value to remove. | |
* @returns {boolean} Returns `true` if the entry was removed, else `false`. | |
*/ | |
function listCacheDelete(key) { | |
var data = this.__data__, | |
index = assocIndexOf(data, key); | |
if (index < 0) { | |
return false; | |
} | |
var lastIndex = data.length - 1; | |
if (index == lastIndex) { | |
data.pop(); | |
} else { | |
splice.call(data, index, 1); | |
} | |
return true; | |
} | |
/** | |
* Gets the list cache value for `key`. | |
* | |
* @private | |
* @name get | |
* @memberOf ListCache | |
* @param {string} key The key of the value to get. | |
* @returns {*} Returns the entry value. | |
*/ | |
function listCacheGet(key) { | |
var data = this.__data__, | |
index = assocIndexOf(data, key); | |
return index < 0 ? undefined : data[index][1]; | |
} | |
/** | |
* Checks if a list cache value for `key` exists. | |
* | |
* @private | |
* @name has | |
* @memberOf ListCache | |
* @param {string} key The key of the entry to check. | |
* @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. | |
*/ | |
function listCacheHas(key) { | |
return assocIndexOf(this.__data__, key) > -1; | |
} | |
/** | |
* Sets the list cache `key` to `value`. | |
* | |
* @private | |
* @name set | |
* @memberOf ListCache | |
* @param {string} key The key of the value to set. | |
* @param {*} value The value to set. | |
* @returns {Object} Returns the list cache instance. | |
*/ | |
function listCacheSet(key, value) { | |
var data = this.__data__, | |
index = assocIndexOf(data, key); | |
if (index < 0) { | |
data.push([key, value]); | |
} else { | |
data[index][1] = value; | |
} | |
return this; | |
} | |
// Add methods to `ListCache`. | |
ListCache.prototype.clear = listCacheClear; | |
ListCache.prototype['delete'] = listCacheDelete; | |
ListCache.prototype.get = listCacheGet; | |
ListCache.prototype.has = listCacheHas; | |
ListCache.prototype.set = listCacheSet; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a map cache object to store key-value pairs. | |
* | |
* @private | |
* @constructor | |
* @param {Array} [entries] The key-value pairs to cache. | |
*/ | |
function MapCache(entries) { | |
var index = -1, | |
length = entries ? entries.length : 0; | |
this.clear(); | |
while (++index < length) { | |
var entry = entries[index]; | |
this.set(entry[0], entry[1]); | |
} | |
} | |
/** | |
* Removes all key-value entries from the map. | |
* | |
* @private | |
* @name clear | |
* @memberOf MapCache | |
*/ | |
function mapCacheClear() { | |
this.__data__ = { | |
'hash': new Hash, | |
'map': new (Map || ListCache), | |
'string': new Hash | |
}; | |
} | |
/** | |
* Removes `key` and its value from the map. | |
* | |
* @private | |
* @name delete | |
* @memberOf MapCache | |
* @param {string} key The key of the value to remove. | |
* @returns {boolean} Returns `true` if the entry was removed, else `false`. | |
*/ | |
function mapCacheDelete(key) { | |
return getMapData(this, key)['delete'](key); | |
} | |
/** | |
* Gets the map value for `key`. | |
* | |
* @private | |
* @name get | |
* @memberOf MapCache | |
* @param {string} key The key of the value to get. | |
* @returns {*} Returns the entry value. | |
*/ | |
function mapCacheGet(key) { | |
return getMapData(this, key).get(key); | |
} | |
/** | |
* Checks if a map value for `key` exists. | |
* | |
* @private | |
* @name has | |
* @memberOf MapCache | |
* @param {string} key The key of the entry to check. | |
* @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. | |
*/ | |
function mapCacheHas(key) { | |
return getMapData(this, key).has(key); | |
} | |
/** | |
* Sets the map `key` to `value`. | |
* | |
* @private | |
* @name set | |
* @memberOf MapCache | |
* @param {string} key The key of the value to set. | |
* @param {*} value The value to set. | |
* @returns {Object} Returns the map cache instance. | |
*/ | |
function mapCacheSet(key, value) { | |
getMapData(this, key).set(key, value); | |
return this; | |
} | |
// Add methods to `MapCache`. | |
MapCache.prototype.clear = mapCacheClear; | |
MapCache.prototype['delete'] = mapCacheDelete; | |
MapCache.prototype.get = mapCacheGet; | |
MapCache.prototype.has = mapCacheHas; | |
MapCache.prototype.set = mapCacheSet; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* | |
* Creates an array cache object to store unique values. | |
* | |
* @private | |
* @constructor | |
* @param {Array} [values] The values to cache. | |
*/ | |
function SetCache(values) { | |
var index = -1, | |
length = values ? values.length : 0; | |
this.__data__ = new MapCache; | |
while (++index < length) { | |
this.add(values[index]); | |
} | |
} | |
/** | |
* Adds `value` to the array cache. | |
* | |
* @private | |
* @name add | |
* @memberOf SetCache | |
* @alias push | |
* @param {*} value The value to cache. | |
* @returns {Object} Returns the cache instance. | |
*/ | |
function setCacheAdd(value) { | |
this.__data__.set(value, HASH_UNDEFINED); | |
return this; | |
} | |
/** | |
* Checks if `value` is in the array cache. | |
* | |
* @private | |
* @name has | |
* @memberOf SetCache | |
* @param {*} value The value to search for. | |
* @returns {number} Returns `true` if `value` is found, else `false`. | |
*/ | |
function setCacheHas(value) { | |
return this.__data__.has(value); | |
} | |
// Add methods to `SetCache`. | |
SetCache.prototype.add = SetCache.prototype.push = setCacheAdd; | |
SetCache.prototype.has = setCacheHas; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a stack cache object to store key-value pairs. | |
* | |
* @private | |
* @constructor | |
* @param {Array} [entries] The key-value pairs to cache. | |
*/ | |
function Stack(entries) { | |
this.__data__ = new ListCache(entries); | |
} | |
/** | |
* Removes all key-value entries from the stack. | |
* | |
* @private | |
* @name clear | |
* @memberOf Stack | |
*/ | |
function stackClear() { | |
this.__data__ = new ListCache; | |
} | |
/** | |
* Removes `key` and its value from the stack. | |
* | |
* @private | |
* @name delete | |
* @memberOf Stack | |
* @param {string} key The key of the value to remove. | |
* @returns {boolean} Returns `true` if the entry was removed, else `false`. | |
*/ | |
function stackDelete(key) { | |
return this.__data__['delete'](key); | |
} | |
/** | |
* Gets the stack value for `key`. | |
* | |
* @private | |
* @name get | |
* @memberOf Stack | |
* @param {string} key The key of the value to get. | |
* @returns {*} Returns the entry value. | |
*/ | |
function stackGet(key) { | |
return this.__data__.get(key); | |
} | |
/** | |
* Checks if a stack value for `key` exists. | |
* | |
* @private | |
* @name has | |
* @memberOf Stack | |
* @param {string} key The key of the entry to check. | |
* @returns {boolean} Returns `true` if an entry for `key` exists, else `false`. | |
*/ | |
function stackHas(key) { | |
return this.__data__.has(key); | |
} | |
/** | |
* Sets the stack `key` to `value`. | |
* | |
* @private | |
* @name set | |
* @memberOf Stack | |
* @param {string} key The key of the value to set. | |
* @param {*} value The value to set. | |
* @returns {Object} Returns the stack cache instance. | |
*/ | |
function stackSet(key, value) { | |
var cache = this.__data__; | |
if (cache instanceof ListCache && cache.__data__.length == LARGE_ARRAY_SIZE) { | |
cache = this.__data__ = new MapCache(cache.__data__); | |
} | |
cache.set(key, value); | |
return this; | |
} | |
// Add methods to `Stack`. | |
Stack.prototype.clear = stackClear; | |
Stack.prototype['delete'] = stackDelete; | |
Stack.prototype.get = stackGet; | |
Stack.prototype.has = stackHas; | |
Stack.prototype.set = stackSet; | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Used by `_.defaults` to customize its `_.assignIn` use. | |
* | |
* @private | |
* @param {*} objValue The destination value. | |
* @param {*} srcValue The source value. | |
* @param {string} key The key of the property to assign. | |
* @param {Object} object The parent object of `objValue`. | |
* @returns {*} Returns the value to assign. | |
*/ | |
function assignInDefaults(objValue, srcValue, key, object) { | |
if (objValue === undefined || | |
(eq(objValue, objectProto[key]) && !hasOwnProperty.call(object, key))) { | |
return srcValue; | |
} | |
return objValue; | |
} | |
/** | |
* This function is like `assignValue` except that it doesn't assign | |
* `undefined` values. | |
* | |
* @private | |
* @param {Object} object The object to modify. | |
* @param {string} key The key of the property to assign. | |
* @param {*} value The value to assign. | |
*/ | |
function assignMergeValue(object, key, value) { | |
if ((value !== undefined && !eq(object[key], value)) || | |
(typeof key == 'number' && value === undefined && !(key in object))) { | |
object[key] = value; | |
} | |
} | |
/** | |
* Assigns `value` to `key` of `object` if the existing value is not equivalent | |
* using [`SameValueZero`](http://ecma-international.org/ecma-262/6.0/#sec-samevaluezero) | |
* for equality comparisons. | |
* | |
* @private | |
* @param {Object} object The object to modify. | |
* @param {string} key The key of the property to assign. | |
* @param {*} value The value to assign. | |
*/ | |
function assignValue(object, key, value) { | |
var objValue = object[key]; | |
if (!(hasOwnProperty.call(object, key) && eq(objValue, value)) || | |
(value === undefined && !(key in object))) { | |
object[key] = value; | |
} | |
} | |
/** | |
* Gets the index at which the `key` is found in `array` of key-value pairs. | |
* | |
* @private | |
* @param {Array} array The array to search. | |
* @param {*} key The key to search for. | |
* @returns {number} Returns the index of the matched value, else `-1`. | |
*/ | |
function assocIndexOf(array, key) { | |
var length = array.length; | |
while (length--) { | |
if (eq(array[length][0], key)) { | |
return length; | |
} | |
} | |
return -1; | |
} | |
/** | |
* The base implementation of `_.assign` without support for multiple sources | |
* or `customizer` functions. | |
* | |
* @private | |
* @param {Object} object The destination object. | |
* @param {Object} source The source object. | |
* @returns {Object} Returns `object`. | |
*/ | |
function baseAssign(object, source) { | |
return object && copyObject(source, keys(source), object); | |
} | |
/** | |
* The base implementation of `_.clamp` which doesn't coerce arguments to numbers. | |
* | |
* @private | |
* @param {number} number The number to clamp. | |
* @param {number} [lower] The lower bound. | |
* @param {number} upper The upper bound. | |
* @returns {number} Returns the clamped number. | |
*/ | |
function baseClamp(number, lower, upper) { | |
if (number === number) { | |
if (upper !== undefined) { | |
number = number <= upper ? number : upper; | |
} | |
if (lower !== undefined) { | |
number = number >= lower ? number : lower; | |
} | |
} | |
return number; | |
} | |
/** | |
* The base implementation of `_.clone` and `_.cloneDeep` which tracks | |
* traversed objects. | |
* | |
* @private | |
* @param {*} value The value to clone. | |
* @param {boolean} [isDeep] Specify a deep clone. | |
* @param {boolean} [isFull] Specify a clone including symbols. | |
* @param {Function} [customizer] The function to customize cloning. | |
* @param {string} [key] The key of `value`. | |
* @param {Object} [object] The parent object of `value`. | |
* @param {Object} [stack] Tracks traversed objects and their clone counterparts. | |
* @returns {*} Returns the cloned value. | |
*/ | |
function baseClone(value, isDeep, isFull, customizer, key, object, stack) { | |
var result; | |
if (customizer) { | |
result = object ? customizer(value, key, object, stack) : customizer(value); | |
} | |
if (result !== undefined) { | |
return result; | |
} | |
if (!isObject(value)) { | |
return value; | |
} | |
var isArr = isArray(value); | |
if (isArr) { | |
result = initCloneArray(value); | |
if (!isDeep) { | |
return copyArray(value, result); | |
} | |
} else { | |
var tag = getTag(value), | |
isFunc = tag == funcTag || tag == genTag; | |
if (isBuffer(value)) { | |
return cloneBuffer(value, isDeep); | |
} | |
if (tag == objectTag || tag == argsTag || (isFunc && !object)) { | |
if (isHostObject(value)) { | |
return object ? value : {}; | |
} | |
result = initCloneObject(isFunc ? {} : value); | |
if (!isDeep) { | |
return copySymbols(value, baseAssign(result, value)); | |
} | |
} else { | |
if (!cloneableTags[tag]) { | |
return object ? value : {}; | |
} | |
result = initCloneByTag(value, tag, baseClone, isDeep); | |
} | |
} | |
// Check for circular references and return its corresponding clone. | |
stack || (stack = new Stack); | |
var stacked = stack.get(value); | |
if (stacked) { | |
return stacked; | |
} | |
stack.set(value, result); | |
if (!isArr) { | |
var props = isFull ? getAllKeys(value) : keys(value); | |
} | |
// Recursively populate clone (susceptible to call stack limits). | |
baseEach(props || value, function(subValue, key) { | |
if (props) { | |
key = subValue; | |
subValue = value[key]; | |
} | |
assignValue(result, key, baseClone(subValue, isDeep, isFull, customizer, key, value, stack)); | |
}); | |
return result; | |
} | |
/** | |
* The base implementation of `_.create` without support for assigning | |
* properties to the created object. | |
* | |
* @private | |
* @param {Object} prototype The object to inherit from. | |
* @returns {Object} Returns the new object. | |
*/ | |
function baseCreate(proto) { | |
return isObject(proto) ? objectCreate(proto) : {}; | |
} | |
/** | |
* The base implementation of `_.delay` and `_.defer` which accepts an array | |
* of `func` arguments. | |
* | |
* @private | |
* @param {Function} func The function to delay. | |
* @param {number} wait The number of milliseconds to delay invocation. | |
* @param {Object} args The arguments to provide to `func`. | |
* @returns {number} Returns the timer id. | |
*/ | |
function baseDelay(func, wait, args) { | |
if (typeof func != 'function') { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
return setTimeout(function() { func.apply(undefined, args); }, wait); | |
} | |
/** | |
* The base implementation of `_.forEach` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @returns {Array|Object} Returns `collection`. | |
*/ | |
var baseEach = createBaseEach(baseForOwn); | |
/** | |
* The base implementation of `_.every` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} predicate The function invoked per iteration. | |
* @returns {boolean} Returns `true` if all elements pass the predicate check, | |
* else `false` | |
*/ | |
function baseEvery(collection, predicate) { | |
var result = true; | |
baseEach(collection, function(value, index, collection) { | |
result = !!predicate(value, index, collection); | |
return result; | |
}); | |
return result; | |
} | |
/** | |
* The base implementation of methods like `_.max` and `_.min` which accepts a | |
* `comparator` to determine the extremum value. | |
* | |
* @private | |
* @param {Array} array The array to iterate over. | |
* @param {Function} iteratee The iteratee invoked per iteration. | |
* @param {Function} comparator The comparator used to compare values. | |
* @returns {*} Returns the extremum value. | |
*/ | |
function baseExtremum(array, iteratee, comparator) { | |
var index = -1, | |
length = array.length; | |
while (++index < length) { | |
var value = array[index], | |
current = iteratee(value); | |
if (current != null && (computed === undefined | |
? (current === current && !false) | |
: comparator(current, computed) | |
)) { | |
var computed = current, | |
result = value; | |
} | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.filter` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} predicate The function invoked per iteration. | |
* @returns {Array} Returns the new filtered array. | |
*/ | |
function baseFilter(collection, predicate) { | |
var result = []; | |
baseEach(collection, function(value, index, collection) { | |
if (predicate(value, index, collection)) { | |
result.push(value); | |
} | |
}); | |
return result; | |
} | |
/** | |
* The base implementation of `_.flatten` with support for restricting flattening. | |
* | |
* @private | |
* @param {Array} array The array to flatten. | |
* @param {number} depth The maximum recursion depth. | |
* @param {boolean} [predicate=isFlattenable] The function invoked per iteration. | |
* @param {boolean} [isStrict] Restrict to values that pass `predicate` checks. | |
* @param {Array} [result=[]] The initial result value. | |
* @returns {Array} Returns the new flattened array. | |
*/ | |
function baseFlatten(array, depth, predicate, isStrict, result) { | |
var index = -1, | |
length = array.length; | |
predicate || (predicate = isFlattenable); | |
result || (result = []); | |
while (++index < length) { | |
var value = array[index]; | |
if (depth > 0 && predicate(value)) { | |
if (depth > 1) { | |
// Recursively flatten arrays (susceptible to call stack limits). | |
baseFlatten(value, depth - 1, predicate, isStrict, result); | |
} else { | |
arrayPush(result, value); | |
} | |
} else if (!isStrict) { | |
result[result.length] = value; | |
} | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `baseForOwn` which iterates over `object` | |
* properties returned by `keysFunc` and invokes `iteratee` for each property. | |
* Iteratee functions may exit iteration early by explicitly returning `false`. | |
* | |
* @private | |
* @param {Object} object The object to iterate over. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @param {Function} keysFunc The function to get the keys of `object`. | |
* @returns {Object} Returns `object`. | |
*/ | |
var baseFor = createBaseFor(); | |
/** | |
* The base implementation of `_.forOwn` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Object} object The object to iterate over. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @returns {Object} Returns `object`. | |
*/ | |
function baseForOwn(object, iteratee) { | |
return object && baseFor(object, iteratee, keys); | |
} | |
/** | |
* The base implementation of `_.functions` which creates an array of | |
* `object` function property names filtered from `props`. | |
* | |
* @private | |
* @param {Object} object The object to inspect. | |
* @param {Array} props The property names to filter. | |
* @returns {Array} Returns the function names. | |
*/ | |
function baseFunctions(object, props) { | |
return baseFilter(props, function(key) { | |
return isFunction(object[key]); | |
}); | |
} | |
/** | |
* The base implementation of `_.get` without support for default values. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {Array|string} path The path of the property to get. | |
* @returns {*} Returns the resolved value. | |
*/ | |
function baseGet(object, path) { | |
path = isKey(path, object) ? [path] : castPath(path); | |
var index = 0, | |
length = path.length; | |
while (object != null && index < length) { | |
object = object[toKey(path[index++])]; | |
} | |
return (index && index == length) ? object : undefined; | |
} | |
/** | |
* The base implementation of `getAllKeys` and `getAllKeysIn` which uses | |
* `keysFunc` and `symbolsFunc` to get the enumerable property names and | |
* symbols of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {Function} keysFunc The function to get the keys of `object`. | |
* @param {Function} symbolsFunc The function to get the symbols of `object`. | |
* @returns {Array} Returns the array of property names and symbols. | |
*/ | |
function baseGetAllKeys(object, keysFunc, symbolsFunc) { | |
var result = keysFunc(object); | |
return isArray(object) ? result : arrayPush(result, symbolsFunc(object)); | |
} | |
/** | |
* The base implementation of `_.gt` which doesn't coerce arguments to numbers. | |
* | |
* @private | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @returns {boolean} Returns `true` if `value` is greater than `other`, | |
* else `false`. | |
*/ | |
function baseGt(value, other) { | |
return value > other; | |
} | |
/** | |
* The base implementation of methods like `_.intersection`, without support | |
* for iteratee shorthands, that accepts an array of arrays to inspect. | |
* | |
* @private | |
* @param {Array} arrays The arrays to inspect. | |
* @param {Function} [iteratee] The iteratee invoked per element. | |
* @param {Function} [comparator] The comparator invoked per element. | |
* @returns {Array} Returns the new array of shared values. | |
*/ | |
function baseIntersection(arrays, iteratee, comparator) { | |
var includes = comparator ? arrayIncludesWith : arrayIncludes, | |
length = arrays[0].length, | |
othLength = arrays.length, | |
othIndex = othLength, | |
caches = Array(othLength), | |
maxLength = Infinity, | |
result = []; | |
while (othIndex--) { | |
var array = arrays[othIndex]; | |
if (othIndex && iteratee) { | |
array = baseMap(array, baseUnary(iteratee)); | |
} | |
maxLength = nativeMin(array.length, maxLength); | |
caches[othIndex] = !comparator && (iteratee || (length >= 120 && array.length >= 120)) | |
? new SetCache(othIndex && array) | |
: undefined; | |
} | |
array = arrays[0]; | |
var index = -1, | |
seen = caches[0]; | |
outer: | |
while (++index < length && result.length < maxLength) { | |
var value = array[index], | |
computed = iteratee ? iteratee(value) : value; | |
value = (comparator || value !== 0) ? value : 0; | |
if (!(seen | |
? cacheHas(seen, computed) | |
: includes(result, computed, comparator) | |
)) { | |
othIndex = othLength; | |
while (--othIndex) { | |
var cache = caches[othIndex]; | |
if (!(cache | |
? cacheHas(cache, computed) | |
: includes(arrays[othIndex], computed, comparator)) | |
) { | |
continue outer; | |
} | |
} | |
if (seen) { | |
seen.push(computed); | |
} | |
result.push(value); | |
} | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.invoke` without support for individual | |
* method arguments. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {Array|string} path The path of the method to invoke. | |
* @param {Array} args The arguments to invoke the method with. | |
* @returns {*} Returns the result of the invoked method. | |
*/ | |
function baseInvoke(object, path, args) { | |
if (!isKey(path, object)) { | |
path = castPath(path); | |
object = parent(object, path); | |
path = last(path); | |
} | |
var func = object == null ? object : object[toKey(path)]; | |
return func == null ? undefined : func.apply(object, args); | |
} | |
/** | |
* The base implementation of `_.isEqual` which supports partial comparisons | |
* and tracks traversed objects. | |
* | |
* @private | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @param {Function} [customizer] The function to customize comparisons. | |
* @param {boolean} [bitmask] The bitmask of comparison flags. | |
* The bitmask may be composed of the following flags: | |
* 1 - Unordered comparison | |
* 2 - Partial comparison | |
* @param {Object} [stack] Tracks traversed `value` and `other` objects. | |
* @returns {boolean} Returns `true` if the values are equivalent, else `false`. | |
*/ | |
function baseIsEqual(value, other, customizer, bitmask, stack) { | |
if (value === other) { | |
return true; | |
} | |
if (value == null || other == null || (!isObject(value) && !isObjectLike(other))) { | |
return value !== value && other !== other; | |
} | |
return baseIsEqualDeep(value, other, baseIsEqual, customizer, bitmask, stack); | |
} | |
/** | |
* A specialized version of `baseIsEqual` for arrays and objects which performs | |
* deep comparisons and tracks traversed objects enabling objects with circular | |
* references to be compared. | |
* | |
* @private | |
* @param {Object} object The object to compare. | |
* @param {Object} other The other object to compare. | |
* @param {Function} equalFunc The function to determine equivalents of values. | |
* @param {Function} [customizer] The function to customize comparisons. | |
* @param {number} [bitmask] The bitmask of comparison flags. See `baseIsEqual` | |
* for more details. | |
* @param {Object} [stack] Tracks traversed `object` and `other` objects. | |
* @returns {boolean} Returns `true` if the objects are equivalent, else `false`. | |
*/ | |
function baseIsEqualDeep(object, other, equalFunc, customizer, bitmask, stack) { | |
var objIsArr = isArray(object), | |
othIsArr = isArray(other), | |
objTag = arrayTag, | |
othTag = arrayTag; | |
if (!objIsArr) { | |
objTag = objectToString.call(object); | |
objTag = objTag == argsTag ? objectTag : objTag; | |
} | |
if (!othIsArr) { | |
othTag = objectToString.call(other); | |
othTag = othTag == argsTag ? objectTag : othTag; | |
} | |
var objIsObj = objTag == objectTag && !isHostObject(object), | |
othIsObj = othTag == objectTag && !isHostObject(other), | |
isSameTag = objTag == othTag; | |
stack || (stack = []); | |
var stacked = find(stack, function(entry) { | |
return entry[0] === object; | |
}); | |
if (stacked && stacked[1]) { | |
return stacked[1] == other; | |
} | |
stack.push([object, other]); | |
if (isSameTag && !objIsObj) { | |
var result = (objIsArr) | |
? equalArrays(object, other, equalFunc, customizer, bitmask, stack) | |
: equalByTag(object, other, objTag, equalFunc, customizer, bitmask, stack); | |
stack.pop(); | |
return result; | |
} | |
if (!(bitmask & PARTIAL_COMPARE_FLAG)) { | |
var objIsWrapped = objIsObj && hasOwnProperty.call(object, '__wrapped__'), | |
othIsWrapped = othIsObj && hasOwnProperty.call(other, '__wrapped__'); | |
if (objIsWrapped || othIsWrapped) { | |
var objUnwrapped = objIsWrapped ? object.value() : object, | |
othUnwrapped = othIsWrapped ? other.value() : other; | |
var result = equalFunc(objUnwrapped, othUnwrapped, customizer, bitmask, stack); | |
stack.pop(); | |
return result; | |
} | |
} | |
if (!isSameTag) { | |
return false; | |
} | |
var result = equalObjects(object, other, equalFunc, customizer, bitmask, stack); | |
stack.pop(); | |
return result; | |
} | |
/** | |
* The base implementation of `_.isNative` without bad shim checks. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a native function, | |
* else `false`. | |
*/ | |
function baseIsNative(value) { | |
if (!isObject(value) || isMasked(value)) { | |
return false; | |
} | |
var pattern = (isFunction(value) || isHostObject(value)) ? reIsNative : reIsHostCtor; | |
return pattern.test(toSource(value)); | |
} | |
/** | |
* The base implementation of `_.iteratee`. | |
* | |
* @private | |
* @param {*} [value=_.identity] The value to convert to an iteratee. | |
* @returns {Function} Returns the iteratee. | |
*/ | |
function baseIteratee(func) { | |
if (typeof func == 'function') { | |
return func; | |
} | |
if (func == null) { | |
return identity; | |
} | |
return (typeof func == 'object' ? baseMatches : baseProperty)(func); | |
} | |
/** | |
* The base implementation of `_.keys` which doesn't skip the constructor | |
* property of prototypes or treat sparse arrays as dense. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names. | |
*/ | |
function baseKeys(object) { | |
return nativeKeys(Object(object)); | |
} | |
/** | |
* The base implementation of `_.keysIn` which doesn't skip the constructor | |
* property of prototypes or treat sparse arrays as dense. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names. | |
*/ | |
function baseKeysIn(object) { | |
object = object == null ? object : Object(object); | |
var result = []; | |
for (var key in object) { | |
result.push(key); | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.lt` which doesn't coerce arguments to numbers. | |
* | |
* @private | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @returns {boolean} Returns `true` if `value` is less than `other`, | |
* else `false`. | |
*/ | |
function baseLt(value, other) { | |
return value < other; | |
} | |
/** | |
* The base implementation of `_.map` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} iteratee The function invoked per iteration. | |
* @returns {Array} Returns the new mapped array. | |
*/ | |
function baseMap(collection, iteratee) { | |
var index = -1, | |
result = isArrayLike(collection) ? Array(collection.length) : []; | |
baseEach(collection, function(value, key, collection) { | |
result[++index] = iteratee(value, key, collection); | |
}); | |
return result; | |
} | |
/** | |
* The base implementation of `_.matches` which doesn't clone `source`. | |
* | |
* @private | |
* @param {Object} source The object of property values to match. | |
* @returns {Function} Returns the new spec function. | |
*/ | |
function baseMatches(source) { | |
var props = keys(source); | |
return function(object) { | |
var length = props.length; | |
if (object == null) { | |
return !length; | |
} | |
object = Object(object); | |
while (length--) { | |
var key = props[length]; | |
if (!(key in object && | |
baseIsEqual(source[key], object[key], undefined, UNORDERED_COMPARE_FLAG | PARTIAL_COMPARE_FLAG) | |
)) { | |
return false; | |
} | |
} | |
return true; | |
}; | |
} | |
/** | |
* The base implementation of `_.merge` without support for multiple sources. | |
* | |
* @private | |
* @param {Object} object The destination object. | |
* @param {Object} source The source object. | |
* @param {number} srcIndex The index of `source`. | |
* @param {Function} [customizer] The function to customize merged values. | |
* @param {Object} [stack] Tracks traversed source values and their merged | |
* counterparts. | |
*/ | |
function baseMerge(object, source, srcIndex, customizer, stack) { | |
if (object === source) { | |
return; | |
} | |
if (!(isArray(source) || isTypedArray(source))) { | |
var props = keysIn(source); | |
} | |
baseEach(props || source, function(srcValue, key) { | |
if (props) { | |
key = srcValue; | |
srcValue = source[key]; | |
} | |
if (isObject(srcValue)) { | |
stack || (stack = new Stack); | |
baseMergeDeep(object, source, key, srcIndex, baseMerge, customizer, stack); | |
} | |
else { | |
var newValue = customizer | |
? customizer(object[key], srcValue, (key + ''), object, source, stack) | |
: undefined; | |
if (newValue === undefined) { | |
newValue = srcValue; | |
} | |
assignMergeValue(object, key, newValue); | |
} | |
}); | |
} | |
/** | |
* 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 = object[key], | |
srcValue = 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) { | |
newValue = srcValue; | |
if (isArray(srcValue) || isTypedArray(srcValue)) { | |
if (isArray(objValue)) { | |
newValue = objValue; | |
} | |
else if (isArrayLikeObject(objValue)) { | |
newValue = copyArray(objValue); | |
} | |
else { | |
isCommon = false; | |
newValue = baseClone(srcValue, true); | |
} | |
} | |
else if (isPlainObject(srcValue) || isArguments(srcValue)) { | |
if (isArguments(objValue)) { | |
newValue = toPlainObject(objValue); | |
} | |
else if (!isObject(objValue) || (srcIndex && isFunction(objValue))) { | |
isCommon = false; | |
newValue = baseClone(srcValue, true); | |
} | |
else { | |
newValue = objValue; | |
} | |
} | |
else { | |
isCommon = false; | |
} | |
} | |
stack.set(srcValue, newValue); | |
if (isCommon) { | |
// Recursively merge objects and arrays (susceptible to call stack limits). | |
mergeFunc(newValue, srcValue, srcIndex, customizer, stack); | |
} | |
stack['delete'](srcValue); | |
assignMergeValue(object, key, newValue); | |
} | |
/** | |
* The base implementation of `_.pick` without support for individual | |
* property identifiers. | |
* | |
* @private | |
* @param {Object} object The source object. | |
* @param {string[]} props The property identifiers to pick. | |
* @returns {Object} Returns the new object. | |
*/ | |
function basePick(object, props) { | |
object = Object(object); | |
return reduce(props, function(result, key) { | |
if (key in object) { | |
result[key] = object[key]; | |
} | |
return result; | |
}, {}); | |
} | |
/** | |
* The base implementation of `_.pickBy` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Object} object The source object. | |
* @param {Function} predicate The function invoked per property. | |
* @returns {Object} Returns the new object. | |
*/ | |
function basePickBy(object, predicate) { | |
var index = -1, | |
props = getAllKeysIn(object), | |
length = props.length, | |
result = {}; | |
while (++index < length) { | |
var key = props[index], | |
value = object[key]; | |
if (predicate(value, key)) { | |
result[key] = value; | |
} | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.property` without support for deep paths. | |
* | |
* @private | |
* @param {string} key The key of the property to get. | |
* @returns {Function} Returns the new accessor function. | |
*/ | |
function baseProperty(key) { | |
return function(object) { | |
return object == null ? undefined : object[key]; | |
}; | |
} | |
/** | |
* The base implementation of `_.pullAt` without support for individual | |
* indexes or capturing the removed elements. | |
* | |
* @private | |
* @param {Array} array The array to modify. | |
* @param {number[]} indexes The indexes of elements to remove. | |
* @returns {Array} Returns `array`. | |
*/ | |
function basePullAt(array, indexes) { | |
var length = array ? indexes.length : 0, | |
lastIndex = length - 1; | |
while (length--) { | |
var index = indexes[length]; | |
if (length == lastIndex || index !== previous) { | |
var previous = index; | |
if (isIndex(index)) { | |
splice.call(array, index, 1); | |
} | |
else if (!isKey(index, array)) { | |
var path = castPath(index), | |
object = parent(array, path); | |
if (object != null) { | |
delete object[toKey(last(path))]; | |
} | |
} | |
else { | |
delete array[toKey(index)]; | |
} | |
} | |
} | |
return array; | |
} | |
/** | |
* The base implementation of `_.range` and `_.rangeRight` which doesn't | |
* coerce arguments to numbers. | |
* | |
* @private | |
* @param {number} start The start of the range. | |
* @param {number} end The end of the range. | |
* @param {number} step The value to increment or decrement by. | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {Array} Returns the range of numbers. | |
*/ | |
function baseRange(start, end, step, fromRight) { | |
var index = -1, | |
length = nativeMax(nativeCeil((end - start) / (step || 1)), 0), | |
result = Array(length); | |
while (length--) { | |
result[fromRight ? length : ++index] = start; | |
start += step; | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `_.repeat` which doesn't coerce arguments. | |
* | |
* @private | |
* @param {string} string The string to repeat. | |
* @param {number} n The number of times to repeat the string. | |
* @returns {string} Returns the repeated string. | |
*/ | |
function baseRepeat(string, n) { | |
var result = ''; | |
if (!string || n < 1 || n > MAX_SAFE_INTEGER) { | |
return result; | |
} | |
// Leverage the exponentiation by squaring algorithm for a faster repeat. | |
// See https://en.wikipedia.org/wiki/Exponentiation_by_squaring for more details. | |
do { | |
if (n % 2) { | |
result += string; | |
} | |
n = nativeFloor(n / 2); | |
if (n) { | |
string += string; | |
} | |
} while (n); | |
return result; | |
} | |
/** | |
* The base implementation of `_.slice` without an iteratee call guard. | |
* | |
* @private | |
* @param {Array} array The array to slice. | |
* @param {number} [start=0] The start position. | |
* @param {number} [end=array.length] The end position. | |
* @returns {Array} Returns the slice of `array`. | |
*/ | |
function baseSlice(array, start, end) { | |
var index = -1, | |
length = array.length; | |
if (start < 0) { | |
start = -start > length ? 0 : (length + start); | |
} | |
end = end > length ? length : end; | |
if (end < 0) { | |
end += length; | |
} | |
length = start > end ? 0 : ((end - start) >>> 0); | |
start >>>= 0; | |
var result = Array(length); | |
while (++index < length) { | |
result[index] = array[index + start]; | |
} | |
return result; | |
} | |
/** | |
* Copies the values of `source` to `array`. | |
* | |
* @private | |
* @param {Array} source The array to copy values from. | |
* @param {Array} [array=[]] The array to copy values to. | |
* @returns {Array} Returns `array`. | |
*/ | |
function copyArray(source) { | |
return baseSlice(source, 0, source.length); | |
} | |
/** | |
* The base implementation of `_.some` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} predicate The function invoked per iteration. | |
* @returns {boolean} Returns `true` if any element passes the predicate check, | |
* else `false`. | |
*/ | |
function baseSome(collection, predicate) { | |
var result; | |
baseEach(collection, function(value, index, collection) { | |
result = predicate(value, index, collection); | |
return !result; | |
}); | |
return !!result; | |
} | |
/** | |
* The base implementation of `_.toNumber` which doesn't ensure correct | |
* conversions of binary, hexadecimal, or octal string values. | |
* | |
* @private | |
* @param {*} value The value to process. | |
* @returns {number} Returns the number. | |
*/ | |
function baseToNumber(value) { | |
if (typeof value == 'number') { | |
return value; | |
} | |
if (false) { | |
return NAN; | |
} | |
return +value; | |
} | |
/** | |
* The base implementation of `_.toString` which doesn't convert nullish | |
* values to empty strings. | |
* | |
* @private | |
* @param {*} value The value to process. | |
* @returns {string} Returns the string. | |
*/ | |
function baseToString(value) { | |
// Exit early for strings to avoid a performance hit in some environments. | |
if (typeof value == 'string') { | |
return value; | |
} | |
if (false) { | |
return symbolToString ? symbolToString.call(value) : ''; | |
} | |
var result = (value + ''); | |
return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result; | |
} | |
/** | |
* The base implementation of `_.uniqBy` without support for iteratee shorthands. | |
* | |
* @private | |
* @param {Array} array The array to inspect. | |
* @param {Function} [iteratee] The iteratee invoked per element. | |
* @param {Function} [comparator] The comparator invoked per element. | |
* @returns {Array} Returns the new duplicate free array. | |
*/ | |
function baseUniq(array, iteratee, comparator) { | |
var index = -1, | |
includes = arrayIncludes, | |
length = array.length, | |
isCommon = true, | |
result = [], | |
seen = result; | |
if (comparator) { | |
isCommon = false; | |
includes = arrayIncludesWith; | |
} | |
else if (length >= LARGE_ARRAY_SIZE) { | |
var set = iteratee ? null : createSet(array); | |
if (set) { | |
return setToArray(set); | |
} | |
isCommon = false; | |
includes = cacheHas; | |
seen = new SetCache; | |
} | |
else { | |
seen = iteratee ? [] : result; | |
} | |
outer: | |
while (++index < length) { | |
var value = array[index], | |
computed = iteratee ? iteratee(value) : value; | |
value = (comparator || value !== 0) ? value : 0; | |
if (isCommon && computed === computed) { | |
var seenIndex = seen.length; | |
while (seenIndex--) { | |
if (seen[seenIndex] === computed) { | |
continue outer; | |
} | |
} | |
if (iteratee) { | |
seen.push(computed); | |
} | |
result.push(value); | |
} | |
else if (!includes(seen, computed, comparator)) { | |
if (seen !== result) { | |
seen.push(computed); | |
} | |
result.push(value); | |
} | |
} | |
return result; | |
} | |
/** | |
* The base implementation of `wrapperValue` which returns the result of | |
* performing a sequence of actions on the unwrapped `value`, where each | |
* successive action is supplied the return value of the previous. | |
* | |
* @private | |
* @param {*} value The unwrapped value. | |
* @param {Array} actions Actions to perform to resolve the unwrapped value. | |
* @returns {*} Returns the resolved value. | |
*/ | |
function baseWrapperValue(value, actions) { | |
var result = value; | |
return reduce(actions, function(result, action) { | |
return action.func.apply(action.thisArg, arrayPush([result], action.args)); | |
}, result); | |
} | |
/** | |
* Casts `value` to an empty array if it's not an array like object. | |
* | |
* @private | |
* @param {*} value The value to inspect. | |
* @returns {Array|Object} Returns the cast array-like object. | |
*/ | |
function castArrayLikeObject(value) { | |
return isArrayLikeObject(value) ? value : []; | |
} | |
/** | |
* Casts `value` to a path array if it's not one. | |
* | |
* @private | |
* @param {*} value The value to inspect. | |
* @returns {Array} Returns the cast property path array. | |
*/ | |
function castPath(value) { | |
return isArray(value) ? value : stringToPath(value); | |
} | |
/** | |
* Casts `array` to a slice if it's needed. | |
* | |
* @private | |
* @param {Array} array The array to inspect. | |
* @param {number} start The start position. | |
* @param {number} [end=array.length] The end position. | |
* @returns {Array} Returns the cast slice. | |
*/ | |
function castSlice(array, start, end) { | |
var length = array.length; | |
end = end === undefined ? length : end; | |
return (!start && end >= length) ? array : baseSlice(array, start, end); | |
} | |
/** | |
* Creates a clone of `buffer`. | |
* | |
* @private | |
* @param {Buffer} buffer The buffer to clone. | |
* @param {boolean} [isDeep] Specify a deep clone. | |
* @returns {Buffer} Returns the cloned buffer. | |
*/ | |
function cloneBuffer(buffer, isDeep) { | |
if (isDeep) { | |
return buffer.slice(); | |
} | |
var result = new buffer.constructor(buffer.length); | |
buffer.copy(result); | |
return result; | |
} | |
/** | |
* Creates a clone of `arrayBuffer`. | |
* | |
* @private | |
* @param {ArrayBuffer} arrayBuffer The array buffer to clone. | |
* @returns {ArrayBuffer} Returns the cloned array buffer. | |
*/ | |
function cloneArrayBuffer(arrayBuffer) { | |
var result = new arrayBuffer.constructor(arrayBuffer.byteLength); | |
new Uint8Array(result).set(new Uint8Array(arrayBuffer)); | |
return result; | |
} | |
/** | |
* Creates a clone of `dataView`. | |
* | |
* @private | |
* @param {Object} dataView The data view to clone. | |
* @param {boolean} [isDeep] Specify a deep clone. | |
* @returns {Object} Returns the cloned data view. | |
*/ | |
function cloneDataView(dataView, isDeep) { | |
var buffer = isDeep ? cloneArrayBuffer(dataView.buffer) : dataView.buffer; | |
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), true) : mapToArray(map); | |
return reduce(array, addMapEntry, new map.constructor); | |
} | |
/** | |
* Creates a clone of `regexp`. | |
* | |
* @private | |
* @param {Object} regexp The regexp to clone. | |
* @returns {Object} Returns the cloned regexp. | |
*/ | |
function cloneRegExp(regexp) { | |
var result = new regexp.constructor(regexp.source, reFlags.exec(regexp)); | |
result.lastIndex = regexp.lastIndex; | |
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), true) : setToArray(set); | |
return reduce(array, addSetEntry, new set.constructor); | |
} | |
/** | |
* Creates a clone of the `symbol` object. | |
* | |
* @private | |
* @param {Object} symbol The symbol object to clone. | |
* @returns {Object} Returns the cloned symbol object. | |
*/ | |
function cloneSymbol(symbol) { | |
return symbolValueOf ? Object(symbolValueOf.call(symbol)) : {}; | |
} | |
/** | |
* Creates a clone of `typedArray`. | |
* | |
* @private | |
* @param {Object} typedArray The typed array to clone. | |
* @param {boolean} [isDeep] Specify a deep clone. | |
* @returns {Object} Returns the cloned typed array. | |
*/ | |
function cloneTypedArray(typedArray, isDeep) { | |
var buffer = isDeep ? cloneArrayBuffer(typedArray.buffer) : typedArray.buffer; | |
return new typedArray.constructor(buffer, typedArray.byteOffset, typedArray.length); | |
} | |
/** | |
* Compares values to sort them in ascending order. | |
* | |
* @private | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @returns {number} Returns the sort order indicator for `value`. | |
*/ | |
function compareAscending(value, other) { | |
if (value !== other) { | |
var valIsDefined = value !== undefined, | |
valIsNull = value === null, | |
valIsReflexive = value === value, | |
valIsSymbol = false; | |
var othIsDefined = other !== undefined, | |
othIsNull = other === null, | |
othIsReflexive = other === other, | |
othIsSymbol = false; | |
if ((!othIsNull && !othIsSymbol && !valIsSymbol && value > other) || | |
(valIsSymbol && othIsDefined && othIsReflexive && !othIsNull && !othIsSymbol) || | |
(valIsNull && othIsDefined && othIsReflexive) || | |
(!valIsDefined && othIsReflexive) || | |
!valIsReflexive) { | |
return 1; | |
} | |
if ((!valIsNull && !valIsSymbol && !othIsSymbol && value < other) || | |
(othIsSymbol && valIsDefined && valIsReflexive && !valIsNull && !valIsSymbol) || | |
(othIsNull && valIsDefined && valIsReflexive) || | |
(!othIsDefined && valIsReflexive) || | |
!othIsReflexive) { | |
return -1; | |
} | |
} | |
return 0; | |
} | |
/** | |
* Copies properties of `source` to `object`. | |
* | |
* @private | |
* @param {Object} source The object to copy properties from. | |
* @param {Array} props The property identifiers to copy. | |
* @param {Object} [object={}] The object to copy properties to. | |
* @param {Function} [customizer] The function to customize copied values. | |
* @returns {Object} Returns `object`. | |
*/ | |
function copyObject(source, props, object, customizer) { | |
object || (object = {}); | |
var index = -1, | |
length = props.length; | |
while (++index < length) { | |
var key = props[index]; | |
var newValue = customizer | |
? customizer(object[key], source[key], key, object, source) | |
: source[key]; | |
assignValue(object, key, newValue); | |
} | |
return object; | |
} | |
/** | |
* Copies own symbol properties of `source` to `object`. | |
* | |
* @private | |
* @param {Object} source The object to copy symbols from. | |
* @param {Object} [object={}] The object to copy symbols to. | |
* @returns {Object} Returns `object`. | |
*/ | |
function copySymbols(source, object) { | |
return copyObject(source, getSymbols(source), object); | |
} | |
/** | |
* Creates a function like `_.assign`. | |
* | |
* @private | |
* @param {Function} assigner The function to assign values. | |
* @returns {Function} Returns the new assigner function. | |
*/ | |
function createAssigner(assigner) { | |
return rest(function(object, sources) { | |
var index = -1, | |
length = sources.length, | |
customizer = length > 1 ? sources[length - 1] : undefined; | |
customizer = (assigner.length > 3 && typeof customizer == 'function') | |
? (length--, customizer) | |
: undefined; | |
object = Object(object); | |
while (++index < length) { | |
var source = sources[index]; | |
if (source) { | |
assigner(object, source, index, customizer); | |
} | |
} | |
return object; | |
}); | |
} | |
/** | |
* Creates a `baseEach` or `baseEachRight` function. | |
* | |
* @private | |
* @param {Function} eachFunc The function to iterate over a collection. | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {Function} Returns the new base function. | |
*/ | |
function createBaseEach(eachFunc, fromRight) { | |
return function(collection, iteratee) { | |
if (collection == null) { | |
return collection; | |
} | |
if (!isArrayLike(collection)) { | |
return eachFunc(collection, iteratee); | |
} | |
var length = collection.length, | |
index = fromRight ? length : -1, | |
iterable = Object(collection); | |
while ((fromRight ? index-- : ++index < length)) { | |
if (iteratee(iterable[index], index, iterable) === false) { | |
break; | |
} | |
} | |
return collection; | |
}; | |
} | |
/** | |
* Creates a base function for methods like `_.forIn` and `_.forOwn`. | |
* | |
* @private | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {Function} Returns the new base function. | |
*/ | |
function createBaseFor(fromRight) { | |
return function(object, iteratee, keysFunc) { | |
var index = -1, | |
iterable = Object(object), | |
props = keysFunc(object), | |
length = props.length; | |
while (length--) { | |
var key = props[fromRight ? length : ++index]; | |
if (iteratee(iterable[key], key, iterable) === false) { | |
break; | |
} | |
} | |
return object; | |
}; | |
} | |
/** | |
* Creates a function like `_.lowerFirst`. | |
* | |
* @private | |
* @param {string} methodName The name of the `String` case method to use. | |
* @returns {Function} Returns the new case function. | |
*/ | |
function createCaseFirst(methodName) { | |
return function(string) { | |
string = toString(string); | |
var strSymbols = reHasComplexSymbol.test(string) | |
? stringToArray(string) | |
: undefined; | |
var chr = strSymbols | |
? strSymbols[0] | |
: string.charAt(0); | |
var trailing = strSymbols | |
? castSlice(strSymbols, 1).join('') | |
: string.slice(1); | |
return chr[methodName]() + trailing; | |
}; | |
} | |
/** | |
* Creates a function like `_.camelCase`. | |
* | |
* @private | |
* @param {Function} callback The function to combine each word. | |
* @returns {Function} Returns the new compounder function. | |
*/ | |
function createCompounder(callback) { | |
return function(string) { | |
return reduce(words(deburr(string).replace(reApos, '')), callback, ''); | |
}; | |
} | |
/** | |
* Creates a function that produces an instance of `Ctor` regardless of | |
* whether it was invoked as part of a `new` expression or by `call` or `apply`. | |
* | |
* @private | |
* @param {Function} Ctor The constructor to wrap. | |
* @returns {Function} Returns the new wrapped function. | |
*/ | |
function createCtorWrapper(Ctor) { | |
return function() { | |
// Use a `switch` statement to work with class constructors. See | |
// http://ecma-international.org/ecma-262/6.0/#sec-ecmascript-function-objects-call-thisargument-argumentslist | |
// for more details. | |
var args = arguments; | |
var thisBinding = baseCreate(Ctor.prototype), | |
result = Ctor.apply(thisBinding, args); | |
// Mimic the constructor's `return` behavior. | |
// See https://es5.github.io/#x13.2.2 for more details. | |
return isObject(result) ? result : thisBinding; | |
}; | |
} | |
/** | |
* Creates a `_.find` or `_.findLast` function. | |
* | |
* @private | |
* @param {Function} findIndexFunc The function to find the collection index. | |
* @returns {Function} Returns the new find function. | |
*/ | |
function createFind(findIndexFunc) { | |
return function(collection, predicate, fromIndex) { | |
var iterable = Object(collection); | |
predicate = baseIteratee(predicate, 3); | |
if (!isArrayLike(collection)) { | |
var props = keys(collection); | |
} | |
var index = findIndexFunc(props || collection, function(value, key) { | |
if (props) { | |
key = value; | |
value = iterable[key]; | |
} | |
return predicate(value, key, iterable); | |
}, fromIndex); | |
return index > -1 ? collection[props ? props[index] : index] : undefined; | |
}; | |
} | |
/** | |
* Creates a function that performs a mathematical operation on two values. | |
* | |
* @private | |
* @param {Function} operator The function to perform the operation. | |
* @returns {Function} Returns the new mathematical operation function. | |
*/ | |
function createMathOperation(operator) { | |
return function(value, other) { | |
var result; | |
if (value === undefined && other === undefined) { | |
return 0; | |
} | |
if (value !== undefined) { | |
result = value; | |
} | |
if (other !== undefined) { | |
if (result === undefined) { | |
return other; | |
} | |
if (typeof value == 'string' || typeof other == 'string') { | |
value = baseToString(value); | |
other = baseToString(other); | |
} else { | |
value = baseToNumber(value); | |
other = baseToNumber(other); | |
} | |
result = operator(value, other); | |
} | |
return result; | |
}; | |
} | |
/** | |
* Creates the padding for `string` based on `length`. The `chars` string | |
* is truncated if the number of characters exceeds `length`. | |
* | |
* @private | |
* @param {number} length The padding length. | |
* @param {string} [chars=' '] The string used as padding. | |
* @returns {string} Returns the padding for `string`. | |
*/ | |
function createPadding(length, chars) { | |
chars = chars === undefined ? ' ' : baseToString(chars); | |
var charsLength = chars.length; | |
if (charsLength < 2) { | |
return charsLength ? baseRepeat(chars, length) : chars; | |
} | |
var result = baseRepeat(chars, nativeCeil(length / stringSize(chars))); | |
return reHasComplexSymbol.test(chars) | |
? castSlice(stringToArray(result), 0, length).join('') | |
: result.slice(0, length); | |
} | |
/** | |
* Creates a function that wraps `func` to invoke it with the `this` binding | |
* of `thisArg` and `partials` prepended to the arguments it receives. | |
* | |
* @private | |
* @param {Function} func The function to wrap. | |
* @param {number} bitmask The bitmask of wrapper flags. See `createWrapper` | |
* for more details. | |
* @param {*} thisArg The `this` binding of `func`. | |
* @param {Array} partials The arguments to prepend to those provided to | |
* the new function. | |
* @returns {Function} Returns the new wrapped function. | |
*/ | |
function createPartialWrapper(func, bitmask, thisArg, partials) { | |
if (typeof func != 'function') { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
var isBind = bitmask & BIND_FLAG, | |
Ctor = createCtorWrapper(func); | |
function wrapper() { | |
var argsIndex = -1, | |
argsLength = arguments.length, | |
leftIndex = -1, | |
leftLength = partials.length, | |
args = Array(leftLength + argsLength), | |
fn = (this && this !== root && this instanceof wrapper) ? Ctor : func; | |
while (++leftIndex < leftLength) { | |
args[leftIndex] = partials[leftIndex]; | |
} | |
while (argsLength--) { | |
args[leftIndex++] = arguments[++argsIndex]; | |
} | |
return fn.apply(isBind ? thisArg : this, args); | |
} | |
return wrapper; | |
} | |
/** | |
* Creates a `_.range` or `_.rangeRight` function. | |
* | |
* @private | |
* @param {boolean} [fromRight] Specify iterating from right to left. | |
* @returns {Function} Returns the new range function. | |
*/ | |
function createRange(fromRight) { | |
return function(start, end, step) { | |
if (step && typeof step != 'number' && isIterateeCall(start, end, step)) { | |
end = step = undefined; | |
} | |
// Ensure the sign of `-0` is preserved. | |
start = toNumber(start); | |
start = start === start ? start : 0; | |
if (end === undefined) { | |
end = start; | |
start = 0; | |
} else { | |
end = toNumber(end) || 0; | |
} | |
step = step === undefined ? (start < end ? 1 : -1) : (toNumber(step) || 0); | |
return baseRange(start, end, step, fromRight); | |
}; | |
} | |
/** | |
* Creates a set of `values`. | |
* | |
* @private | |
* @param {Array} values The values to add to the set. | |
* @returns {Object} Returns the new set. | |
*/ | |
var createSet = !(Set && (1 / setToArray(new Set([,-0]))[1]) == INFINITY) ? noop : function(values) { | |
return new Set(values); | |
}; | |
/** | |
* A specialized version of `baseIsEqualDeep` for arrays with support for | |
* partial deep comparisons. | |
* | |
* @private | |
* @param {Array} array The array to compare. | |
* @param {Array} other The other array to compare. | |
* @param {Function} equalFunc The function to determine equivalents of values. | |
* @param {Function} customizer The function to customize comparisons. | |
* @param {number} bitmask The bitmask of comparison flags. See `baseIsEqual` | |
* for more details. | |
* @param {Object} stack Tracks traversed `array` and `other` objects. | |
* @returns {boolean} Returns `true` if the arrays are equivalent, else `false`. | |
*/ | |
function equalArrays(array, other, equalFunc, customizer, bitmask, stack) { | |
var isPartial = bitmask & PARTIAL_COMPARE_FLAG, | |
arrLength = array.length, | |
othLength = other.length; | |
if (arrLength != othLength && !(isPartial && othLength > arrLength)) { | |
return false; | |
} | |
var index = -1, | |
result = true, | |
seen = (bitmask & UNORDERED_COMPARE_FLAG) ? [] : undefined; | |
// Ignore non-index properties. | |
while (++index < arrLength) { | |
var arrValue = array[index], | |
othValue = other[index]; | |
var compared; | |
if (compared !== undefined) { | |
if (compared) { | |
continue; | |
} | |
result = false; | |
break; | |
} | |
// Recursively compare arrays (susceptible to call stack limits). | |
if (seen) { | |
if (!baseSome(other, function(othValue, othIndex) { | |
if (!indexOf(seen, othIndex) && | |
(arrValue === othValue || equalFunc(arrValue, othValue, customizer, bitmask, stack))) { | |
return seen.push(othIndex); | |
} | |
})) { | |
result = false; | |
break; | |
} | |
} else if (!( | |
arrValue === othValue || | |
equalFunc(arrValue, othValue, customizer, bitmask, stack) | |
)) { | |
result = false; | |
break; | |
} | |
} | |
return result; | |
} | |
/** | |
* A specialized version of `baseIsEqualDeep` for comparing objects of | |
* the same `toStringTag`. | |
* | |
* **Note:** This function only supports comparing values with tags of | |
* `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`. | |
* | |
* @private | |
* @param {Object} object The object to compare. | |
* @param {Object} other The other object to compare. | |
* @param {string} tag The `toStringTag` of the objects to compare. | |
* @param {Function} equalFunc The function to determine equivalents of values. | |
* @param {Function} customizer The function to customize comparisons. | |
* @param {number} bitmask The bitmask of comparison flags. See `baseIsEqual` | |
* for more details. | |
* @param {Object} stack Tracks traversed `object` and `other` objects. | |
* @returns {boolean} Returns `true` if the objects are equivalent, else `false`. | |
*/ | |
function equalByTag(object, other, tag, equalFunc, customizer, bitmask, stack) { | |
switch (tag) { | |
case boolTag: | |
case dateTag: | |
// Coerce dates and booleans to numbers, dates to milliseconds and | |
// booleans to `1` or `0` treating invalid dates coerced to `NaN` as | |
// not equal. | |
return +object == +other; | |
case errorTag: | |
return object.name == other.name && object.message == other.message; | |
case numberTag: | |
// Treat `NaN` vs. `NaN` as equal. | |
return (object != +object) ? other != +other : object == +other; | |
case regexpTag: | |
case stringTag: | |
// Coerce regexes to strings and treat strings, primitives and objects, | |
// as equal. See http://www.ecma-international.org/ecma-262/6.0/#sec-regexp.prototype.tostring | |
// for more details. | |
return object == (other + ''); | |
} | |
return false; | |
} | |
/** | |
* A specialized version of `baseIsEqualDeep` for objects with support for | |
* partial deep comparisons. | |
* | |
* @private | |
* @param {Object} object The object to compare. | |
* @param {Object} other The other object to compare. | |
* @param {Function} equalFunc The function to determine equivalents of values. | |
* @param {Function} customizer The function to customize comparisons. | |
* @param {number} bitmask The bitmask of comparison flags. See `baseIsEqual` | |
* for more details. | |
* @param {Object} stack Tracks traversed `object` and `other` objects. | |
* @returns {boolean} Returns `true` if the objects are equivalent, else `false`. | |
*/ | |
function equalObjects(object, other, equalFunc, customizer, bitmask, stack) { | |
var isPartial = bitmask & PARTIAL_COMPARE_FLAG, | |
objProps = keys(object), | |
objLength = objProps.length, | |
othProps = keys(other), | |
othLength = othProps.length; | |
if (objLength != othLength && !isPartial) { | |
return false; | |
} | |
var index = objLength; | |
while (index--) { | |
var key = objProps[index]; | |
if (!(isPartial ? key in other : hasOwnProperty.call(other, key))) { | |
return false; | |
} | |
} | |
var result = true; | |
var skipCtor = isPartial; | |
while (++index < objLength) { | |
key = objProps[index]; | |
var objValue = object[key], | |
othValue = other[key]; | |
var compared; | |
// Recursively compare objects (susceptible to call stack limits). | |
if (!(compared === undefined | |
? (objValue === othValue || equalFunc(objValue, othValue, customizer, bitmask, stack)) | |
: compared | |
)) { | |
result = false; | |
break; | |
} | |
skipCtor || (skipCtor = key == 'constructor'); | |
} | |
if (result && !skipCtor) { | |
var objCtor = object.constructor, | |
othCtor = other.constructor; | |
// Non `Object` object instances with different constructors are not equal. | |
if (objCtor != othCtor && | |
('constructor' in object && 'constructor' in other) && | |
!(typeof objCtor == 'function' && objCtor instanceof objCtor && | |
typeof othCtor == 'function' && othCtor instanceof othCtor)) { | |
result = false; | |
} | |
} | |
return result; | |
} | |
/** | |
* Creates an array of own enumerable property names and symbols of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names and symbols. | |
*/ | |
function getAllKeys(object) { | |
return baseGetAllKeys(object, keys, getSymbols); | |
} | |
/** | |
* Creates an array of own and inherited enumerable property names and | |
* symbols of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names and symbols. | |
*/ | |
function getAllKeysIn(object) { | |
return baseGetAllKeys(object, keysIn, getSymbolsIn); | |
} | |
/** | |
* Gets the "length" property value of `object`. | |
* | |
* **Note:** This function is used to avoid a | |
* [JIT bug](https://bugs.webkit.org/show_bug.cgi?id=142792) that affects | |
* Safari on at least iOS 8.1-8.3 ARM64. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {*} Returns the "length" value. | |
*/ | |
var getLength = baseProperty('length'); | |
/** | |
* Gets the data for `map`. | |
* | |
* @private | |
* @param {Object} map The map to query. | |
* @param {string} key The reference key. | |
* @returns {*} Returns the map data. | |
*/ | |
function getMapData(map, key) { | |
var data = map.__data__; | |
return isKeyable(key) | |
? data[typeof key == 'string' ? 'string' : 'hash'] | |
: data.map; | |
} | |
/** | |
* Gets the native function at `key` of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {string} key The key of the method to get. | |
* @returns {*} Returns the function if it's native, else `undefined`. | |
*/ | |
function getNative(object, key) { | |
var value = getValue(object, key); | |
return baseIsNative(value) ? value : undefined; | |
} | |
/** | |
* Gets the `[[Prototype]]` of `value`. | |
* | |
* @private | |
* @param {*} value The value to query. | |
* @returns {null|Object} Returns the `[[Prototype]]`. | |
*/ | |
function getPrototype(value) { | |
return nativeGetPrototype(Object(value)); | |
} | |
/** | |
* Creates an array of the own enumerable symbol properties of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of symbols. | |
*/ | |
function getSymbols(object) { | |
// Coerce `object` to an object to avoid non-object errors in V8. | |
// See https://bugs.chromium.org/p/v8/issues/detail?id=3443 for more details. | |
return getOwnPropertySymbols(Object(object)); | |
} | |
// Fallback for IE < 11. | |
if (!getOwnPropertySymbols) { | |
getSymbols = stubArray; | |
} | |
/** | |
* Creates an array of the own and inherited enumerable symbol properties | |
* of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of symbols. | |
*/ | |
var getSymbolsIn = !getOwnPropertySymbols ? getSymbols : function(object) { | |
var result = []; | |
while (object) { | |
arrayPush(result, getSymbols(object)); | |
object = getPrototype(object); | |
} | |
return result; | |
}; | |
/** | |
* Gets the `toStringTag` of `value`. | |
* | |
* @private | |
* @param {*} value The value to query. | |
* @returns {string} Returns the `toStringTag`. | |
*/ | |
function getTag(value) { | |
return objectToString.call(value); | |
} | |
// Fallback for data views, maps, sets, and weak maps in IE 11, | |
// for data views in Edge, and promises in Node.js. | |
if ((DataView && getTag(new DataView(new ArrayBuffer(1))) != dataViewTag) || | |
(Map && getTag(new Map) != mapTag) || | |
(Promise && getTag(Promise.resolve()) != promiseTag) || | |
(Set && getTag(new Set) != setTag) || | |
(WeakMap && getTag(new WeakMap) != weakMapTag)) { | |
getTag = function(value) { | |
var result = objectToString.call(value), | |
Ctor = result == objectTag ? value.constructor : undefined, | |
ctorString = Ctor ? toSource(Ctor) : undefined; | |
if (ctorString) { | |
switch (ctorString) { | |
case dataViewCtorString: return dataViewTag; | |
case mapCtorString: return mapTag; | |
case promiseCtorString: return promiseTag; | |
case setCtorString: return setTag; | |
case weakMapCtorString: return weakMapTag; | |
} | |
} | |
return result; | |
}; | |
} | |
/** | |
* Initializes an array clone. | |
* | |
* @private | |
* @param {Array} array The array to clone. | |
* @returns {Array} Returns the initialized clone. | |
*/ | |
function initCloneArray(array) { | |
var length = array.length, | |
result = array.constructor(length); | |
// Add properties assigned by `RegExp#exec`. | |
if (length && typeof array[0] == 'string' && hasOwnProperty.call(array, 'index')) { | |
result.index = array.index; | |
result.input = array.input; | |
} | |
return result; | |
} | |
/** | |
* Initializes an object clone. | |
* | |
* @private | |
* @param {Object} object The object to clone. | |
* @returns {Object} Returns the initialized clone. | |
*/ | |
function initCloneObject(object) { | |
return (typeof object.constructor == 'function' && !isPrototype(object)) | |
? baseCreate(getPrototype(object)) | |
: {}; | |
} | |
/** | |
* Initializes an object clone based on its `toStringTag`. | |
* | |
* **Note:** This function only supports cloning values with tags of | |
* `Boolean`, `Date`, `Error`, `Number`, `RegExp`, or `String`. | |
* | |
* @private | |
* @param {Object} object 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. | |
* @returns {Object} Returns the initialized clone. | |
*/ | |
function initCloneByTag(object, tag, cloneFunc, isDeep) { | |
var Ctor = object.constructor; | |
switch (tag) { | |
case arrayBufferTag: | |
return cloneArrayBuffer(object); | |
case boolTag: | |
case dateTag: | |
return new Ctor(+object); | |
case dataViewTag: | |
return cloneDataView(object, isDeep); | |
case float32Tag: case float64Tag: | |
case int8Tag: case int16Tag: case int32Tag: | |
case uint8Tag: case uint8ClampedTag: case uint16Tag: case uint32Tag: | |
return cloneTypedArray(object, isDeep); | |
case mapTag: | |
return cloneMap(object, isDeep, cloneFunc); | |
case numberTag: | |
case stringTag: | |
return new Ctor(object); | |
case regexpTag: | |
return cloneRegExp(object); | |
case setTag: | |
return cloneSet(object, isDeep, cloneFunc); | |
case symbolTag: | |
return cloneSymbol(object); | |
} | |
} | |
/** | |
* Checks if `value` is a flattenable `arguments` object or array. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is flattenable, else `false`. | |
*/ | |
function isFlattenable(value) { | |
return isArray(value) || isArguments(value); | |
} | |
/** | |
* Checks if `value` is a valid array-like index. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @param {number} [length=MAX_SAFE_INTEGER] The upper bounds of a valid index. | |
* @returns {boolean} Returns `true` if `value` is a valid index, else `false`. | |
*/ | |
function isIndex(value, length) { | |
length = length == null ? MAX_SAFE_INTEGER : length; | |
return !!length && | |
(typeof value == 'number' || reIsUint.test(value)) && | |
(value > -1 && value % 1 == 0 && value < length); | |
} | |
/** | |
* Checks if the given arguments are from an iteratee call. | |
* | |
* @private | |
* @param {*} value The potential iteratee value argument. | |
* @param {*} index The potential iteratee index or key argument. | |
* @param {*} object The potential iteratee object argument. | |
* @returns {boolean} Returns `true` if the arguments are from an iteratee call, | |
* else `false`. | |
*/ | |
function isIterateeCall(value, index, object) { | |
if (!isObject(object)) { | |
return false; | |
} | |
var type = typeof index; | |
if (type == 'number' | |
? (isArrayLike(object) && isIndex(index, object.length)) | |
: (type == 'string' && index in object) | |
) { | |
return eq(object[index], value); | |
} | |
return false; | |
} | |
/** | |
* Checks if `value` is a property name and not a property path. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @param {Object} [object] The object to query keys on. | |
* @returns {boolean} Returns `true` if `value` is a property name, else `false`. | |
*/ | |
function isKey(value, object) { | |
if (isArray(value)) { | |
return false; | |
} | |
var type = typeof value; | |
if (type == 'number' || type == 'symbol' || type == 'boolean' || | |
value == null || false) { | |
return true; | |
} | |
return reIsPlainProp.test(value) || !reIsDeepProp.test(value) || | |
(object != null && value in Object(object)); | |
} | |
/** | |
* Checks if `value` is suitable for use as unique object key. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is suitable, else `false`. | |
*/ | |
function isKeyable(value) { | |
var type = typeof value; | |
return (type == 'string' || type == 'number' || type == 'symbol' || type == 'boolean') | |
? (value !== '__proto__') | |
: (value === null); | |
} | |
/** | |
* Checks if `func` has its source masked. | |
* | |
* @private | |
* @param {Function} func The function to check. | |
* @returns {boolean} Returns `true` if `func` is masked, else `false`. | |
*/ | |
function isMasked(func) { | |
return !!maskSrcKey && (maskSrcKey in func); | |
} | |
/** | |
* Checks if `value` is likely a prototype object. | |
* | |
* @private | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a prototype, else `false`. | |
*/ | |
function isPrototype(value) { | |
var Ctor = value && value.constructor, | |
proto = (typeof Ctor == 'function' && Ctor.prototype) || objectProto; | |
return value === proto; | |
} | |
/** | |
* Gets the parent value at `path` of `object`. | |
* | |
* @private | |
* @param {Object} object The object to query. | |
* @param {Array} path The path to get the parent value of. | |
* @returns {*} Returns the parent value. | |
*/ | |
function parent(object, path) { | |
return path.length == 1 ? object : baseGet(object, baseSlice(path, 0, -1)); | |
} | |
/** | |
* Converts `string` to a property path array. | |
* | |
* @private | |
* @param {string} string The string to convert. | |
* @returns {Array} Returns the property path array. | |
*/ | |
var stringToPath = memoize(function(string) { | |
var result = []; | |
toString(string).replace(rePropName, function(match, number, quote, string) { | |
result.push(quote ? string.replace(reEscapeChar, '$1') : (number || match)); | |
}); | |
return result; | |
}); | |
/** | |
* Converts `value` to a string key if it's not a string or symbol. | |
* | |
* @private | |
* @param {*} value The value to inspect. | |
* @returns {string|symbol} Returns the key. | |
*/ | |
var toKey = String; | |
/** | |
* Converts `func` to its source code. | |
* | |
* @private | |
* @param {Function} func The function to process. | |
* @returns {string} Returns the source code. | |
*/ | |
function toSource(func) { | |
if (func != null) { | |
try { | |
return funcToString.call(func); | |
} catch (e) {} | |
try { | |
return (func + ''); | |
} catch (e) {} | |
} | |
return ''; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates an array of elements split into groups the length of `size`. | |
* If `array` can't be split evenly, the final chunk will be the remaining | |
* elements. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Array | |
* @param {Array} array The array to process. | |
* @param {number} [size=1] The length of each chunk | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {Array} Returns the new array of chunks. | |
* @example | |
* | |
* _.chunk(['a', 'b', 'c', 'd'], 2); | |
* // => [['a', 'b'], ['c', 'd']] | |
* | |
* _.chunk(['a', 'b', 'c', 'd'], 3); | |
* // => [['a', 'b', 'c'], ['d']] | |
*/ | |
function chunk(array, size, guard) { | |
if ((guard ? isIterateeCall(array, size, guard) : size === undefined)) { | |
size = 1; | |
} else { | |
size = nativeMax(toInteger(size), 0); | |
} | |
var length = array ? array.length : 0; | |
if (!length || size < 1) { | |
return []; | |
} | |
var index = 0, | |
resIndex = 0, | |
result = Array(nativeCeil(length / size)); | |
while (index < length) { | |
result[resIndex++] = baseSlice(array, index, (index += size)); | |
} | |
return result; | |
} | |
/** | |
* Creates an array with all falsey values removed. The values `false`, `null`, | |
* `0`, `""`, `undefined`, and `NaN` are falsey. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {Array} array The array to compact. | |
* @returns {Array} Returns the new array of filtered values. | |
* @example | |
* | |
* _.compact([0, 1, false, 2, '', 3]); | |
* // => [1, 2, 3] | |
*/ | |
function compact(array) { | |
return baseFilter(array, Boolean); | |
} | |
/** | |
* Creates a new array concatenating `array` with any additional arrays | |
* and/or values. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Array | |
* @param {Array} array The array to concatenate. | |
* @param {...*} [values] The values to concatenate. | |
* @returns {Array} Returns the new concatenated array. | |
* @example | |
* | |
* var array = [1]; | |
* var other = _.concat(array, 2, [3], [[4]]); | |
* | |
* console.log(other); | |
* // => [1, 2, 3, [4]] | |
* | |
* console.log(array); | |
* // => [1] | |
*/ | |
function concat() { | |
var length = arguments.length, | |
args = Array(length ? length - 1 : 0), | |
array = arguments[0], | |
index = length; | |
while (index--) { | |
args[index - 1] = arguments[index]; | |
} | |
return length | |
? arrayPush(isArray(array) ? copyArray(array) : [array], baseFlatten(args, 1)) | |
: []; | |
} | |
/** | |
* This method is like `_.find` except that it returns the index of the first | |
* element `predicate` returns truthy for instead of the element itself. | |
* | |
* @static | |
* @memberOf _ | |
* @since 1.1.0 | |
* @category Array | |
* @param {Array} array The array to search. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @param {number} [fromIndex=0] The index to search from. | |
* @returns {number} Returns the index of the found element, else `-1`. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'active': false }, | |
* { 'user': 'fred', 'active': false }, | |
* { 'user': 'pebbles', 'active': true } | |
* ]; | |
* | |
* _.findIndex(users, function(o) { return o.user == 'barney'; }); | |
* // => 0 | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.findIndex(users, { 'user': 'fred', 'active': false }); | |
* // => 1 | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.findIndex(users, ['active', false]); | |
* // => 0 | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.findIndex(users, 'active'); | |
* // => 2 | |
*/ | |
function findIndex(array, predicate, fromIndex) { | |
var length = array ? array.length : 0; | |
if (!length) { | |
return -1; | |
} | |
var index = fromIndex == null ? 0 : toInteger(fromIndex); | |
if (index < 0) { | |
index = nativeMax(length + index, 0); | |
} | |
return baseFindIndex(array, baseIteratee(predicate, 3), index); | |
} | |
/** | |
* Flattens `array` a single level deep. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {Array} array The array to flatten. | |
* @returns {Array} Returns the new flattened array. | |
* @example | |
* | |
* _.flatten([1, [2, [3, [4]], 5]]); | |
* // => [1, 2, [3, [4]], 5] | |
*/ | |
function flatten(array) { | |
var length = array ? array.length : 0; | |
return length ? baseFlatten(array, 1) : []; | |
} | |
/** | |
* Recursively flattens `array`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Array | |
* @param {Array} array The array to flatten. | |
* @returns {Array} Returns the new flattened array. | |
* @example | |
* | |
* _.flattenDeep([1, [2, [3, [4]], 5]]); | |
* // => [1, 2, 3, 4, 5] | |
*/ | |
function flattenDeep(array) { | |
var length = array ? array.length : 0; | |
return length ? baseFlatten(array, INFINITY) : []; | |
} | |
/** | |
* Gets the first element of `array`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @alias first | |
* @category Array | |
* @param {Array} array The array to query. | |
* @returns {*} Returns the first element of `array`. | |
* @example | |
* | |
* _.head([1, 2, 3]); | |
* // => 1 | |
* | |
* _.head([]); | |
* // => undefined | |
*/ | |
function head(array) { | |
return (array && array.length) ? array[0] : undefined; | |
} | |
/** | |
* Gets the index at which the first occurrence of `value` is found in `array` | |
* using [`SameValueZero`](http://ecma-international.org/ecma-262/6.0/#sec-samevaluezero) | |
* for equality comparisons. If `fromIndex` is negative, it's used as the | |
* offset from the end of `array`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {Array} array The array to search. | |
* @param {*} value The value to search for. | |
* @param {number} [fromIndex=0] The index to search from. | |
* @returns {number} Returns the index of the matched value, else `-1`. | |
* @example | |
* | |
* _.indexOf([1, 2, 1, 2], 2); | |
* // => 1 | |
* | |
* // Search from the `fromIndex`. | |
* _.indexOf([1, 2, 1, 2], 2, 2); | |
* // => 3 | |
*/ | |
function indexOf(array, value, fromIndex) { | |
var length = array ? array.length : 0; | |
if (typeof fromIndex == 'number') { | |
fromIndex = fromIndex < 0 ? nativeMax(length + fromIndex, 0) : fromIndex; | |
} else { | |
fromIndex = 0; | |
} | |
var index = (fromIndex || 0) - 1, | |
isReflexive = value === value; | |
while (++index < length) { | |
var other = array[index]; | |
if ((isReflexive ? other === value : other !== other)) { | |
return index; | |
} | |
} | |
return -1; | |
} | |
/** | |
* Creates an array of unique values that are included in all given arrays | |
* using [`SameValueZero`](http://ecma-international.org/ecma-262/6.0/#sec-samevaluezero) | |
* for equality comparisons. The order of result values is determined by the | |
* order they occur in the first array. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {...Array} [arrays] The arrays to inspect. | |
* @returns {Array} Returns the new array of intersecting values. | |
* @example | |
* | |
* _.intersection([2, 1], [2, 3]); | |
* // => [2] | |
*/ | |
var intersection = rest(function(arrays) { | |
var mapped = baseMap(arrays, castArrayLikeObject); | |
return (mapped.length && mapped[0] === arrays[0]) | |
? baseIntersection(mapped) | |
: []; | |
}); | |
/** | |
* Gets the last element of `array`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {Array} array The array to query. | |
* @returns {*} Returns the last element of `array`. | |
* @example | |
* | |
* _.last([1, 2, 3]); | |
* // => 3 | |
*/ | |
function last(array) { | |
var length = array ? array.length : 0; | |
return length ? array[length - 1] : undefined; | |
} | |
/** | |
* Removes all elements from `array` that `predicate` returns truthy for | |
* and returns an array of the removed elements. The predicate is invoked | |
* with three arguments: (value, index, array). | |
* | |
* **Note:** Unlike `_.filter`, this method mutates `array`. Use `_.pull` | |
* to pull elements from an array by value. | |
* | |
* @static | |
* @memberOf _ | |
* @since 2.0.0 | |
* @category Array | |
* @param {Array} array The array to modify. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @returns {Array} Returns the new array of removed elements. | |
* @example | |
* | |
* var array = [1, 2, 3, 4]; | |
* var evens = _.remove(array, function(n) { | |
* return n % 2 == 0; | |
* }); | |
* | |
* console.log(array); | |
* // => [1, 3] | |
* | |
* console.log(evens); | |
* // => [2, 4] | |
*/ | |
function remove(array, predicate) { | |
var result = []; | |
if (!(array && array.length)) { | |
return result; | |
} | |
var index = -1, | |
indexes = [], | |
length = array.length; | |
predicate = baseIteratee(predicate, 3); | |
while (++index < length) { | |
var value = array[index]; | |
if (predicate(value, index, array)) { | |
result.push(value); | |
indexes.push(index); | |
} | |
} | |
basePullAt(array, indexes); | |
return result; | |
} | |
/** | |
* Creates a slice of `array` from `start` up to, but not including, `end`. | |
* | |
* **Note:** This method is used instead of | |
* [`Array#slice`](https://mdn.io/Array/slice) to ensure dense arrays are | |
* returned. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Array | |
* @param {Array} array The array to slice. | |
* @param {number} [start=0] The start position. | |
* @param {number} [end=array.length] The end position. | |
* @returns {Array} Returns the slice of `array`. | |
*/ | |
function slice(array, start, end) { | |
var length = array ? array.length : 0; | |
start = start == null ? 0 : +start; | |
end = end === undefined ? length : +end; | |
return length ? baseSlice(array, start, end) : []; | |
} | |
/** | |
* Creates a duplicate-free version of an array, using | |
* [`SameValueZero`](http://ecma-international.org/ecma-262/6.0/#sec-samevaluezero) | |
* for equality comparisons, in which only the first occurrence of each | |
* element is kept. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Array | |
* @param {Array} array The array to inspect. | |
* @returns {Array} Returns the new duplicate free array. | |
* @example | |
* | |
* _.uniq([2, 1, 2]); | |
* // => [2, 1] | |
*/ | |
function uniq(array) { | |
return (array && array.length) | |
? baseUniq(array) | |
: []; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a `lodash` wrapper instance that wraps `value` with explicit method | |
* chain sequences enabled. The result of such sequences must be unwrapped | |
* with `_#value`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 1.3.0 | |
* @category Seq | |
* @param {*} value The value to wrap. | |
* @returns {Object} Returns the new `lodash` wrapper instance. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36 }, | |
* { 'user': 'fred', 'age': 40 }, | |
* { 'user': 'pebbles', 'age': 1 } | |
* ]; | |
* | |
* var youngest = _ | |
* .chain(users) | |
* .sortBy('age') | |
* .map(function(o) { | |
* return o.user + ' is ' + o.age; | |
* }) | |
* .head() | |
* .value(); | |
* // => 'pebbles is 1' | |
*/ | |
function chain(value) { | |
var result = lodash(value); | |
result.__chain__ = true; | |
return result; | |
} | |
/** | |
* This method invokes `interceptor` and returns `value`. The interceptor | |
* is invoked with one argument; (value). The purpose of this method is to | |
* "tap into" a method chain sequence in order to modify intermediate results. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Seq | |
* @param {*} value The value to provide to `interceptor`. | |
* @param {Function} interceptor The function to invoke. | |
* @returns {*} Returns `value`. | |
* @example | |
* | |
* _([1, 2, 3]) | |
* .tap(function(array) { | |
* // Mutate input array. | |
* array.pop(); | |
* }) | |
* .reverse() | |
* .value(); | |
* // => [2, 1] | |
*/ | |
function tap(value, interceptor) { | |
interceptor(value); | |
return value; | |
} | |
/** | |
* This method is like `_.tap` except that it returns the result of `interceptor`. | |
* The purpose of this method is to "pass thru" values replacing intermediate | |
* results in a method chain sequence. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Seq | |
* @param {*} value The value to provide to `interceptor`. | |
* @param {Function} interceptor The function to invoke. | |
* @returns {*} Returns the result of `interceptor`. | |
* @example | |
* | |
* _(' abc ') | |
* .chain() | |
* .trim() | |
* .thru(function(value) { | |
* return [value]; | |
* }) | |
* .value(); | |
* // => ['abc'] | |
*/ | |
function thru(value, interceptor) { | |
return interceptor(value); | |
} | |
/** | |
* Creates a `lodash` wrapper instance with explicit method chain sequences enabled. | |
* | |
* @name chain | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Seq | |
* @returns {Object} Returns the new `lodash` wrapper instance. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36 }, | |
* { 'user': 'fred', 'age': 40 } | |
* ]; | |
* | |
* // A sequence without explicit chaining. | |
* _(users).head(); | |
* // => { 'user': 'barney', 'age': 36 } | |
* | |
* // A sequence with explicit chaining. | |
* _(users) | |
* .chain() | |
* .head() | |
* .pick('user') | |
* .value(); | |
* // => { 'user': 'barney' } | |
*/ | |
function wrapperChain() { | |
return chain(this); | |
} | |
/** | |
* Executes the chain sequence to resolve the unwrapped value. | |
* | |
* @name value | |
* @memberOf _ | |
* @since 0.1.0 | |
* @alias toJSON, valueOf | |
* @category Seq | |
* @returns {*} Returns the resolved unwrapped value. | |
* @example | |
* | |
* _([1, 2, 3]).value(); | |
* // => [1, 2, 3] | |
*/ | |
function wrapperValue() { | |
return baseWrapperValue(this.__wrapped__, this.__actions__); | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Checks if `predicate` returns truthy for **all** elements of `collection`. | |
* Iteration is stopped once `predicate` returns falsey. The predicate is | |
* invoked with three arguments: (value, index|key, collection). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {boolean} Returns `true` if all elements pass the predicate check, | |
* else `false`. | |
* @example | |
* | |
* _.every([true, 1, null, 'yes'], Boolean); | |
* // => false | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36, 'active': false }, | |
* { 'user': 'fred', 'age': 40, 'active': false } | |
* ]; | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.every(users, { 'user': 'barney', 'active': false }); | |
* // => false | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.every(users, ['active', false]); | |
* // => true | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.every(users, 'active'); | |
* // => false | |
*/ | |
function every(collection, predicate, guard) { | |
predicate = guard ? undefined : predicate; | |
return baseEvery(collection, baseIteratee(predicate)); | |
} | |
/** | |
* Iterates over elements of `collection`, returning an array of all elements | |
* `predicate` returns truthy for. The predicate is invoked with three | |
* arguments: (value, index|key, collection). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @returns {Array} Returns the new filtered array. | |
* @see _.reject | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36, 'active': true }, | |
* { 'user': 'fred', 'age': 40, 'active': false } | |
* ]; | |
* | |
* _.filter(users, function(o) { return !o.active; }); | |
* // => objects for ['fred'] | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.filter(users, { 'age': 36, 'active': true }); | |
* // => objects for ['barney'] | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.filter(users, ['active', false]); | |
* // => objects for ['fred'] | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.filter(users, 'active'); | |
* // => objects for ['barney'] | |
*/ | |
function filter(collection, predicate) { | |
return baseFilter(collection, baseIteratee(predicate)); | |
} | |
/** | |
* Iterates over elements of `collection`, returning the first element | |
* `predicate` returns truthy for. The predicate is invoked with three | |
* arguments: (value, index|key, collection). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to search. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @param {number} [fromIndex=0] The index to search from. | |
* @returns {*} Returns the matched element, else `undefined`. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36, 'active': true }, | |
* { 'user': 'fred', 'age': 40, 'active': false }, | |
* { 'user': 'pebbles', 'age': 1, 'active': true } | |
* ]; | |
* | |
* _.find(users, function(o) { return o.age < 40; }); | |
* // => object for 'barney' | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.find(users, { 'age': 1, 'active': true }); | |
* // => object for 'pebbles' | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.find(users, ['active', false]); | |
* // => object for 'fred' | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.find(users, 'active'); | |
* // => object for 'barney' | |
*/ | |
var find = createFind(findIndex); | |
/** | |
* Iterates over elements of `collection` and invokes `iteratee` for each element. | |
* The iteratee is invoked with three arguments: (value, index|key, collection). | |
* Iteratee functions may exit iteration early by explicitly returning `false`. | |
* | |
* **Note:** As with other "Collections" methods, objects with a "length" | |
* property are iterated like arrays. To avoid this behavior use `_.forIn` | |
* or `_.forOwn` for object iteration. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @alias each | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} [iteratee=_.identity] The function invoked per iteration. | |
* @returns {Array|Object} Returns `collection`. | |
* @see _.forEachRight | |
* @example | |
* | |
* _([1, 2]).forEach(function(value) { | |
* console.log(value); | |
* }); | |
* // => Logs `1` then `2`. | |
* | |
* _.forEach({ 'a': 1, 'b': 2 }, function(value, key) { | |
* console.log(key); | |
* }); | |
* // => Logs 'a' then 'b' (iteration order is not guaranteed). | |
*/ | |
function forEach(collection, iteratee) { | |
return baseEach(collection, baseIteratee(iteratee)); | |
} | |
/** | |
* Creates an array of values by running each element in `collection` thru | |
* `iteratee`. The iteratee is invoked with three arguments: | |
* (value, index|key, collection). | |
* | |
* Many lodash methods are guarded to work as iteratees for methods like | |
* `_.every`, `_.filter`, `_.map`, `_.mapValues`, `_.reject`, and `_.some`. | |
* | |
* The guarded methods are: | |
* `ary`, `chunk`, `curry`, `curryRight`, `drop`, `dropRight`, `every`, | |
* `fill`, `invert`, `parseInt`, `random`, `range`, `rangeRight`, `repeat`, | |
* `sampleSize`, `slice`, `some`, `sortBy`, `split`, `take`, `takeRight`, | |
* `template`, `trim`, `trimEnd`, `trimStart`, and `words` | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Array|Function|Object|string} [iteratee=_.identity] | |
* The function invoked per iteration. | |
* @returns {Array} Returns the new mapped array. | |
* @example | |
* | |
* function square(n) { | |
* return n * n; | |
* } | |
* | |
* _.map([4, 8], square); | |
* // => [16, 64] | |
* | |
* _.map({ 'a': 4, 'b': 8 }, square); | |
* // => [16, 64] (iteration order is not guaranteed) | |
* | |
* var users = [ | |
* { 'user': 'barney' }, | |
* { 'user': 'fred' } | |
* ]; | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.map(users, 'user'); | |
* // => ['barney', 'fred'] | |
*/ | |
function map(collection, iteratee) { | |
return baseMap(collection, baseIteratee(iteratee)); | |
} | |
/** | |
* Reduces `collection` to a value which is the accumulated result of running | |
* each element in `collection` thru `iteratee`, where each successive | |
* invocation is supplied the return value of the previous. If `accumulator` | |
* is not given, the first element of `collection` is used as the initial | |
* value. The iteratee is invoked with four arguments: | |
* (accumulator, value, index|key, collection). | |
* | |
* Many lodash methods are guarded to work as iteratees for methods like | |
* `_.reduce`, `_.reduceRight`, and `_.transform`. | |
* | |
* The guarded methods are: | |
* `assign`, `defaults`, `defaultsDeep`, `includes`, `merge`, `orderBy`, | |
* and `sortBy` | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Function} [iteratee=_.identity] The function invoked per iteration. | |
* @param {*} [accumulator] The initial value. | |
* @returns {*} Returns the accumulated value. | |
* @see _.reduceRight | |
* @example | |
* | |
* _.reduce([1, 2], function(sum, n) { | |
* return sum + n; | |
* }, 0); | |
* // => 3 | |
* | |
* _.reduce({ 'a': 1, 'b': 2, 'c': 1 }, function(result, value, key) { | |
* (result[value] || (result[value] = [])).push(key); | |
* return result; | |
* }, {}); | |
* // => { '1': ['a', 'c'], '2': ['b'] } (iteration order is not guaranteed) | |
*/ | |
function reduce(collection, iteratee, accumulator) { | |
return baseReduce(collection, baseIteratee(iteratee), accumulator, arguments.length < 3, baseEach); | |
} | |
/** | |
* Gets the size of `collection` by returning its length for array-like | |
* values or the number of own enumerable string keyed properties for objects. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to inspect. | |
* @returns {number} Returns the collection size. | |
* @example | |
* | |
* _.size([1, 2, 3]); | |
* // => 3 | |
* | |
* _.size({ 'a': 1, 'b': 2 }); | |
* // => 2 | |
* | |
* _.size('pebbles'); | |
* // => 7 | |
*/ | |
function size(collection) { | |
if (collection == null) { | |
return 0; | |
} | |
collection = isArrayLike(collection) ? collection : keys(collection); | |
return collection.length; | |
} | |
/** | |
* Checks if `predicate` returns truthy for **any** element of `collection`. | |
* Iteration is stopped once `predicate` returns truthy. The predicate is | |
* invoked with three arguments: (value, index|key, collection). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per iteration. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {boolean} Returns `true` if any element passes the predicate check, | |
* else `false`. | |
* @example | |
* | |
* _.some([null, 0, 'yes', false], Boolean); | |
* // => true | |
* | |
* var users = [ | |
* { 'user': 'barney', 'active': true }, | |
* { 'user': 'fred', 'active': false } | |
* ]; | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.some(users, { 'user': 'barney', 'active': false }); | |
* // => false | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.some(users, ['active', false]); | |
* // => true | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.some(users, 'active'); | |
* // => true | |
*/ | |
function some(collection, predicate, guard) { | |
predicate = guard ? undefined : predicate; | |
return baseSome(collection, baseIteratee(predicate)); | |
} | |
/** | |
* Creates an array of elements, sorted in ascending order by the results of | |
* running each element in a collection thru each iteratee. This method | |
* performs a stable sort, that is, it preserves the original sort order of | |
* equal elements. The iteratees are invoked with one argument: (value). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Collection | |
* @param {Array|Object} collection The collection to iterate over. | |
* @param {...(Array|Array[]|Function|Function[]|Object|Object[]|string|string[])} | |
* [iteratees=[_.identity]] The iteratees to sort by. | |
* @returns {Array} Returns the new sorted array. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'fred', 'age': 48 }, | |
* { 'user': 'barney', 'age': 36 }, | |
* { 'user': 'fred', 'age': 40 }, | |
* { 'user': 'barney', 'age': 34 } | |
* ]; | |
* | |
* _.sortBy(users, function(o) { return o.user; }); | |
* // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 40]] | |
* | |
* _.sortBy(users, ['user', 'age']); | |
* // => objects for [['barney', 34], ['barney', 36], ['fred', 40], ['fred', 48]] | |
* | |
* _.sortBy(users, 'user', function(o) { | |
* return Math.floor(o.age / 10); | |
* }); | |
* // => objects for [['barney', 36], ['barney', 34], ['fred', 48], ['fred', 40]] | |
*/ | |
function sortBy(collection, iteratee) { | |
var index = 0; | |
iteratee = baseIteratee(iteratee); | |
return baseMap(baseMap(collection, function(value, key, collection) { | |
return { 'value': value, 'index': index++, 'criteria': iteratee(value, key, collection) }; | |
}).sort(function(object, other) { | |
return compareAscending(object.criteria, other.criteria) || (object.index - other.index); | |
}), baseProperty('value')); | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a function that invokes `func`, with the `this` binding and arguments | |
* of the created function, while it's called less than `n` times. Subsequent | |
* calls to the created function return the result of the last `func` invocation. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Function | |
* @param {number} n The number of calls at which `func` is no longer invoked. | |
* @param {Function} func The function to restrict. | |
* @returns {Function} Returns the new restricted function. | |
* @example | |
* | |
* jQuery(element).on('click', _.before(5, addContactToList)); | |
* // => allows adding up to 4 contacts to the list | |
*/ | |
function before(n, func) { | |
var result; | |
if (typeof func != 'function') { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
n = toInteger(n); | |
return function() { | |
if (--n > 0) { | |
result = func.apply(this, arguments); | |
} | |
if (n <= 1) { | |
func = undefined; | |
} | |
return result; | |
}; | |
} | |
/** | |
* Creates a function that invokes `func` with the `this` binding of `thisArg` | |
* and `partials` prepended to the arguments it receives. | |
* | |
* The `_.bind.placeholder` value, which defaults to `_` in monolithic builds, | |
* may be used as a placeholder for partially applied arguments. | |
* | |
* **Note:** Unlike native `Function#bind`, this method doesn't set the "length" | |
* property of bound functions. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Function | |
* @param {Function} func The function to bind. | |
* @param {*} thisArg The `this` binding of `func`. | |
* @param {...*} [partials] The arguments to be partially applied. | |
* @returns {Function} Returns the new bound function. | |
* @example | |
* | |
* var greet = function(greeting, punctuation) { | |
* return greeting + ' ' + this.user + punctuation; | |
* }; | |
* | |
* var object = { 'user': 'fred' }; | |
* | |
* var bound = _.bind(greet, object, 'hi'); | |
* bound('!'); | |
* // => 'hi fred!' | |
* | |
* // Bound with placeholders. | |
* var bound = _.bind(greet, object, _, '!'); | |
* bound('hi'); | |
* // => 'hi fred!' | |
*/ | |
var bind = rest(function(func, thisArg, partials) { | |
return createPartialWrapper(func, BIND_FLAG | PARTIAL_FLAG, thisArg, partials); | |
}); | |
/** | |
* Defers invoking the `func` until the current call stack has cleared. Any | |
* additional arguments are provided to `func` when it's invoked. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Function | |
* @param {Function} func The function to defer. | |
* @param {...*} [args] The arguments to invoke `func` with. | |
* @returns {number} Returns the timer id. | |
* @example | |
* | |
* _.defer(function(text) { | |
* console.log(text); | |
* }, 'deferred'); | |
* // => Logs 'deferred' after one or more milliseconds. | |
*/ | |
var defer = rest(function(func, args) { | |
return baseDelay(func, 1, args); | |
}); | |
/** | |
* Invokes `func` after `wait` milliseconds. Any additional arguments are | |
* provided to `func` when it's invoked. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Function | |
* @param {Function} func The function to delay. | |
* @param {number} wait The number of milliseconds to delay invocation. | |
* @param {...*} [args] The arguments to invoke `func` with. | |
* @returns {number} Returns the timer id. | |
* @example | |
* | |
* _.delay(function(text) { | |
* console.log(text); | |
* }, 1000, 'later'); | |
* // => Logs 'later' after one second. | |
*/ | |
var delay = rest(function(func, wait, args) { | |
return baseDelay(func, toNumber(wait) || 0, args); | |
}); | |
/** | |
* Creates a function that memoizes the result of `func`. If `resolver` is | |
* provided, it determines the cache key for storing the result based on the | |
* arguments provided to the memoized function. By default, the first argument | |
* provided to the memoized function is used as the map cache key. The `func` | |
* is invoked with the `this` binding of the memoized function. | |
* | |
* **Note:** The cache is exposed as the `cache` property on the memoized | |
* function. Its creation may be customized by replacing the `_.memoize.Cache` | |
* constructor with one whose instances implement the | |
* [`Map`](http://ecma-international.org/ecma-262/6.0/#sec-properties-of-the-map-prototype-object) | |
* method interface of `delete`, `get`, `has`, and `set`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Function | |
* @param {Function} func The function to have its output memoized. | |
* @param {Function} [resolver] The function to resolve the cache key. | |
* @returns {Function} Returns the new memoized function. | |
* @example | |
* | |
* var object = { 'a': 1, 'b': 2 }; | |
* var other = { 'c': 3, 'd': 4 }; | |
* | |
* var values = _.memoize(_.values); | |
* values(object); | |
* // => [1, 2] | |
* | |
* values(other); | |
* // => [3, 4] | |
* | |
* object.a = 2; | |
* values(object); | |
* // => [1, 2] | |
* | |
* // Modify the result cache. | |
* values.cache.set(object, ['a', 'b']); | |
* values(object); | |
* // => ['a', 'b'] | |
* | |
* // Replace `_.memoize.Cache`. | |
* _.memoize.Cache = WeakMap; | |
*/ | |
function memoize(func, resolver) { | |
if (typeof func != 'function' || (resolver && typeof resolver != 'function')) { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
var memoized = function() { | |
var args = arguments, | |
key = resolver ? resolver.apply(this, args) : args[0], | |
cache = memoized.cache; | |
if (cache.has(key)) { | |
return cache.get(key); | |
} | |
var result = func.apply(this, args); | |
memoized.cache = cache.set(key, result); | |
return result; | |
}; | |
memoized.cache = new (memoize.Cache || MapCache); | |
return memoized; | |
} | |
// Assign cache to `_.memoize`. | |
memoize.Cache = MapCache; | |
/** | |
* Creates a function that negates the result of the predicate `func`. The | |
* `func` predicate is invoked with the `this` binding and arguments of the | |
* created function. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Function | |
* @param {Function} predicate The predicate to negate. | |
* @returns {Function} Returns the new negated function. | |
* @example | |
* | |
* function isEven(n) { | |
* return n % 2 == 0; | |
* } | |
* | |
* _.filter([1, 2, 3, 4, 5, 6], _.negate(isEven)); | |
* // => [1, 3, 5] | |
*/ | |
function negate(predicate) { | |
if (typeof predicate != 'function') { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
return function() { | |
return !predicate.apply(this, arguments); | |
}; | |
} | |
/** | |
* Creates a function that is restricted to invoking `func` once. Repeat calls | |
* to the function return the value of the first invocation. The `func` is | |
* invoked with the `this` binding and arguments of the created function. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Function | |
* @param {Function} func The function to restrict. | |
* @returns {Function} Returns the new restricted function. | |
* @example | |
* | |
* var initialize = _.once(createApplication); | |
* initialize(); | |
* initialize(); | |
* // `initialize` invokes `createApplication` once | |
*/ | |
function once(func) { | |
return before(2, func); | |
} | |
/** | |
* Creates a function that invokes `func` with the `this` binding of the | |
* created function and arguments from `start` and beyond provided as | |
* an array. | |
* | |
* **Note:** This method is based on the | |
* [rest parameter](https://mdn.io/rest_parameters). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Function | |
* @param {Function} func The function to apply a rest parameter to. | |
* @param {number} [start=func.length-1] The start position of the rest parameter. | |
* @returns {Function} Returns the new function. | |
* @example | |
* | |
* var say = _.rest(function(what, names) { | |
* return what + ' ' + _.initial(names).join(', ') + | |
* (_.size(names) > 1 ? ', & ' : '') + _.last(names); | |
* }); | |
* | |
* say('hello', 'fred', 'barney', 'pebbles'); | |
* // => 'hello fred, barney, & pebbles' | |
*/ | |
function rest(func, start) { | |
if (typeof func != 'function') { | |
throw new TypeError(FUNC_ERROR_TEXT); | |
} | |
start = nativeMax(start === undefined ? (func.length - 1) : toInteger(start), 0); | |
return function() { | |
var args = arguments, | |
index = -1, | |
length = nativeMax(args.length - start, 0), | |
array = Array(length); | |
while (++index < length) { | |
array[index] = args[start + index]; | |
} | |
var otherArgs = Array(start + 1); | |
index = -1; | |
while (++index < start) { | |
otherArgs[index] = args[index]; | |
} | |
otherArgs[start] = array; | |
return func.apply(this, otherArgs); | |
}; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Creates a shallow clone of `value`. | |
* | |
* **Note:** This method is loosely based on the | |
* [structured clone algorithm](https://mdn.io/Structured_clone_algorithm) | |
* and supports cloning arrays, array buffers, booleans, date objects, maps, | |
* numbers, `Object` objects, regexes, sets, strings, symbols, and typed | |
* arrays. The own enumerable properties of `arguments` objects are cloned | |
* as plain objects. An empty object is returned for uncloneable values such | |
* as error objects, functions, DOM nodes, and WeakMaps. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to clone. | |
* @returns {*} Returns the cloned value. | |
* @see _.cloneDeep | |
* @example | |
* | |
* var objects = [{ 'a': 1 }, { 'b': 2 }]; | |
* | |
* var shallow = _.clone(objects); | |
* console.log(shallow[0] === objects[0]); | |
* // => true | |
*/ | |
function clone(value) { | |
if (!isObject(value)) { | |
return value; | |
} | |
return isArray(value) ? copyArray(value) : copyObject(value, keys(value)); | |
} | |
/** | |
* This method is like `_.clone` except that it recursively clones `value`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 1.0.0 | |
* @category Lang | |
* @param {*} value The value to recursively clone. | |
* @returns {*} Returns the deep cloned value. | |
* @see _.clone | |
* @example | |
* | |
* var objects = [{ 'a': 1 }, { 'b': 2 }]; | |
* | |
* var deep = _.cloneDeep(objects); | |
* console.log(deep[0] === objects[0]); | |
* // => false | |
*/ | |
function cloneDeep(value) { | |
return baseClone(value, true, true); | |
} | |
/** | |
* Performs a | |
* [`SameValueZero`](http://ecma-international.org/ecma-262/6.0/#sec-samevaluezero) | |
* comparison between two values to determine if they are equivalent. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @returns {boolean} Returns `true` if the values are equivalent, else `false`. | |
* @example | |
* | |
* var object = { 'user': 'fred' }; | |
* var other = { 'user': 'fred' }; | |
* | |
* _.eq(object, object); | |
* // => true | |
* | |
* _.eq(object, other); | |
* // => false | |
* | |
* _.eq('a', 'a'); | |
* // => true | |
* | |
* _.eq('a', Object('a')); | |
* // => false | |
* | |
* _.eq(NaN, NaN); | |
* // => true | |
*/ | |
function eq(value, other) { | |
return value === other || (value !== value && other !== other); | |
} | |
/** | |
* Checks if `value` is likely an `arguments` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isArguments(function() { return arguments; }()); | |
* // => true | |
* | |
* _.isArguments([1, 2, 3]); | |
* // => false | |
*/ | |
function isArguments(value) { | |
// Safari 8.1 incorrectly makes `arguments.callee` enumerable in strict mode. | |
return isArrayLikeObject(value) && hasOwnProperty.call(value, 'callee') && | |
(!propertyIsEnumerable.call(value, 'callee') || objectToString.call(value) == argsTag); | |
} | |
/** | |
* Checks if `value` is classified as an `Array` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @type {Function} | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isArray([1, 2, 3]); | |
* // => true | |
* | |
* _.isArray(document.body.children); | |
* // => false | |
* | |
* _.isArray('abc'); | |
* // => false | |
* | |
* _.isArray(_.noop); | |
* // => false | |
*/ | |
var isArray = Array.isArray; | |
/** | |
* Checks if `value` is array-like. A value is considered array-like if it's | |
* not a function and has a `value.length` that's an integer greater than or | |
* equal to `0` and less than or equal to `Number.MAX_SAFE_INTEGER`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is array-like, else `false`. | |
* @example | |
* | |
* _.isArrayLike([1, 2, 3]); | |
* // => true | |
* | |
* _.isArrayLike(document.body.children); | |
* // => true | |
* | |
* _.isArrayLike('abc'); | |
* // => true | |
* | |
* _.isArrayLike(_.noop); | |
* // => false | |
*/ | |
function isArrayLike(value) { | |
return value != null && isLength(getLength(value)) && !isFunction(value); | |
} | |
/** | |
* This method is like `_.isArrayLike` except that it also checks if `value` | |
* is an object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is an array-like object, | |
* else `false`. | |
* @example | |
* | |
* _.isArrayLikeObject([1, 2, 3]); | |
* // => true | |
* | |
* _.isArrayLikeObject(document.body.children); | |
* // => true | |
* | |
* _.isArrayLikeObject('abc'); | |
* // => false | |
* | |
* _.isArrayLikeObject(_.noop); | |
* // => false | |
*/ | |
function isArrayLikeObject(value) { | |
return isObjectLike(value) && isArrayLike(value); | |
} | |
/** | |
* Checks if `value` is classified as a boolean primitive or object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isBoolean(false); | |
* // => true | |
* | |
* _.isBoolean(null); | |
* // => false | |
*/ | |
function isBoolean(value) { | |
return value === true || value === false || | |
(isObjectLike(value) && objectToString.call(value) == boolTag); | |
} | |
/** | |
* Checks if `value` is a buffer. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.3.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a buffer, else `false`. | |
* @example | |
* | |
* _.isBuffer(new Buffer(2)); | |
* // => true | |
* | |
* _.isBuffer(new Uint8Array(2)); | |
* // => false | |
*/ | |
var isBuffer = !Buffer ? stubFalse : function(value) { | |
return value instanceof Buffer; | |
}; | |
/** | |
* Checks if `value` is classified as a `Date` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isDate(new Date); | |
* // => true | |
* | |
* _.isDate('Mon April 23 2012'); | |
* // => false | |
*/ | |
function isDate(value) { | |
return isObjectLike(value) && objectToString.call(value) == dateTag; | |
} | |
/** | |
* Checks if `value` is an empty object, collection, map, or set. | |
* | |
* Objects are considered empty if they have no own enumerable string keyed | |
* properties. | |
* | |
* Array-like values such as `arguments` objects, arrays, buffers, strings, or | |
* jQuery-like collections are considered empty if they have a `length` of `0`. | |
* Similarly, maps and sets are considered empty if they have a `size` of `0`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is empty, else `false`. | |
* @example | |
* | |
* _.isEmpty(null); | |
* // => true | |
* | |
* _.isEmpty(true); | |
* // => true | |
* | |
* _.isEmpty(1); | |
* // => true | |
* | |
* _.isEmpty([1, 2, 3]); | |
* // => false | |
* | |
* _.isEmpty({ 'a': 1 }); | |
* // => false | |
*/ | |
function isEmpty(value) { | |
if (isArrayLike(value) && | |
(isArray(value) || isString(value) || | |
isFunction(value.splice) || isArguments(value))) { | |
return !value.length; | |
} | |
return !keys(value).length; | |
} | |
/** | |
* Performs a deep comparison between two values to determine if they are | |
* equivalent. | |
* | |
* **Note:** This method supports comparing arrays, array buffers, booleans, | |
* date objects, error objects, maps, numbers, `Object` objects, regexes, | |
* sets, strings, symbols, and typed arrays. `Object` objects are compared | |
* by their own, not inherited, enumerable properties. Functions and DOM | |
* nodes are **not** supported. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to compare. | |
* @param {*} other The other value to compare. | |
* @returns {boolean} Returns `true` if the values are equivalent, | |
* else `false`. | |
* @example | |
* | |
* var object = { 'user': 'fred' }; | |
* var other = { 'user': 'fred' }; | |
* | |
* _.isEqual(object, other); | |
* // => true | |
* | |
* object === other; | |
* // => false | |
*/ | |
function isEqual(value, other) { | |
return baseIsEqual(value, other); | |
} | |
/** | |
* Checks if `value` is an `Error`, `EvalError`, `RangeError`, `ReferenceError`, | |
* `SyntaxError`, `TypeError`, or `URIError` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is an error object, | |
* else `false`. | |
* @example | |
* | |
* _.isError(new Error); | |
* // => true | |
* | |
* _.isError(Error); | |
* // => false | |
*/ | |
function isError(value) { | |
if (!isObjectLike(value)) { | |
return false; | |
} | |
return (objectToString.call(value) == errorTag) || | |
(typeof value.message == 'string' && typeof value.name == 'string'); | |
} | |
/** | |
* Checks if `value` is a finite primitive number. | |
* | |
* **Note:** This method is based on | |
* [`Number.isFinite`](https://mdn.io/Number/isFinite). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a finite number, | |
* else `false`. | |
* @example | |
* | |
* _.isFinite(3); | |
* // => true | |
* | |
* _.isFinite(Number.MIN_VALUE); | |
* // => true | |
* | |
* _.isFinite(Infinity); | |
* // => false | |
* | |
* _.isFinite('3'); | |
* // => false | |
*/ | |
function isFinite(value) { | |
return typeof value == 'number' && nativeIsFinite(value); | |
} | |
/** | |
* Checks if `value` is classified as a `Function` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isFunction(_); | |
* // => true | |
* | |
* _.isFunction(/abc/); | |
* // => false | |
*/ | |
function isFunction(value) { | |
// The use of `Object#toString` avoids issues with the `typeof` operator | |
// in Safari 8 which returns 'object' for typed array and weak map constructors, | |
// and PhantomJS 1.9 which returns 'function' for `NodeList` instances. | |
var tag = isObject(value) ? objectToString.call(value) : ''; | |
return tag == funcTag || tag == genTag; | |
} | |
/** | |
* Checks if `value` is a valid array-like length. | |
* | |
* **Note:** This function is loosely based on | |
* [`ToLength`](http://ecma-international.org/ecma-262/6.0/#sec-tolength). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a valid length, | |
* else `false`. | |
* @example | |
* | |
* _.isLength(3); | |
* // => true | |
* | |
* _.isLength(Number.MIN_VALUE); | |
* // => false | |
* | |
* _.isLength(Infinity); | |
* // => false | |
* | |
* _.isLength('3'); | |
* // => false | |
*/ | |
function isLength(value) { | |
return typeof value == 'number' && | |
value > -1 && value % 1 == 0 && value <= MAX_SAFE_INTEGER; | |
} | |
/** | |
* Checks if `value` is the | |
* [language type](http://www.ecma-international.org/ecma-262/6.0/#sec-ecmascript-language-types) | |
* of `Object`. (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`) | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is an object, else `false`. | |
* @example | |
* | |
* _.isObject({}); | |
* // => true | |
* | |
* _.isObject([1, 2, 3]); | |
* // => true | |
* | |
* _.isObject(_.noop); | |
* // => true | |
* | |
* _.isObject(null); | |
* // => false | |
*/ | |
function isObject(value) { | |
var type = typeof value; | |
return !!value && (type == 'object' || type == 'function'); | |
} | |
/** | |
* Checks if `value` is object-like. A value is object-like if it's not `null` | |
* and has a `typeof` result of "object". | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is object-like, else `false`. | |
* @example | |
* | |
* _.isObjectLike({}); | |
* // => true | |
* | |
* _.isObjectLike([1, 2, 3]); | |
* // => true | |
* | |
* _.isObjectLike(_.noop); | |
* // => false | |
* | |
* _.isObjectLike(null); | |
* // => false | |
*/ | |
function isObjectLike(value) { | |
return !!value && typeof value == 'object'; | |
} | |
/** | |
* Checks if `value` is `NaN`. | |
* | |
* **Note:** This method is based on | |
* [`Number.isNaN`](https://mdn.io/Number/isNaN) and is not the same as | |
* global [`isNaN`](https://mdn.io/isNaN) which returns `true` for | |
* `undefined` and other non-number values. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is `NaN`, else `false`. | |
* @example | |
* | |
* _.isNaN(NaN); | |
* // => true | |
* | |
* _.isNaN(new Number(NaN)); | |
* // => true | |
* | |
* isNaN(undefined); | |
* // => true | |
* | |
* _.isNaN(undefined); | |
* // => false | |
*/ | |
function isNaN(value) { | |
// An `NaN` primitive is the only value that is not equal to itself. | |
// Perform the `toStringTag` check first to avoid errors with some | |
// ActiveX objects in IE. | |
return isNumber(value) && value != +value; | |
} | |
/** | |
* Checks if `value` is `null`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is `null`, else `false`. | |
* @example | |
* | |
* _.isNull(null); | |
* // => true | |
* | |
* _.isNull(void 0); | |
* // => false | |
*/ | |
function isNull(value) { | |
return value === null; | |
} | |
/** | |
* Checks if `value` is `null` or `undefined`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is nullish, else `false`. | |
* @example | |
* | |
* _.isNil(null); | |
* // => true | |
* | |
* _.isNil(void 0); | |
* // => true | |
* | |
* _.isNil(NaN); | |
* // => false | |
*/ | |
function isNil(value) { | |
return value == null; | |
} | |
/** | |
* Checks if `value` is classified as a `Number` primitive or object. | |
* | |
* **Note:** To exclude `Infinity`, `-Infinity`, and `NaN`, which are | |
* classified as numbers, use the `_.isFinite` method. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isNumber(3); | |
* // => true | |
* | |
* _.isNumber(Number.MIN_VALUE); | |
* // => true | |
* | |
* _.isNumber(Infinity); | |
* // => true | |
* | |
* _.isNumber('3'); | |
* // => false | |
*/ | |
function isNumber(value) { | |
return typeof value == 'number' || | |
(isObjectLike(value) && objectToString.call(value) == numberTag); | |
} | |
/** | |
* Checks if `value` is a plain object, that is, an object created by the | |
* `Object` constructor or one with a `[[Prototype]]` of `null`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.8.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is a plain object, | |
* else `false`. | |
* @example | |
* | |
* function Foo() { | |
* this.a = 1; | |
* } | |
* | |
* _.isPlainObject(new Foo); | |
* // => false | |
* | |
* _.isPlainObject([1, 2, 3]); | |
* // => false | |
* | |
* _.isPlainObject({ 'x': 0, 'y': 0 }); | |
* // => true | |
* | |
* _.isPlainObject(Object.create(null)); | |
* // => true | |
*/ | |
function isPlainObject(value) { | |
if (!isObjectLike(value) || | |
objectToString.call(value) != objectTag || isHostObject(value)) { | |
return false; | |
} | |
var proto = getPrototype(value); | |
if (proto === null) { | |
return true; | |
} | |
var Ctor = hasOwnProperty.call(proto, 'constructor') && proto.constructor; | |
return (typeof Ctor == 'function' && | |
Ctor instanceof Ctor && funcToString.call(Ctor) == objectCtorString); | |
} | |
/** | |
* Checks if `value` is classified as a `RegExp` object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.1.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isRegExp(/abc/); | |
* // => true | |
* | |
* _.isRegExp('/abc/'); | |
* // => false | |
*/ | |
function isRegExp(value) { | |
return isObject(value) && objectToString.call(value) == regexpTag; | |
} | |
/** | |
* Checks if `value` is classified as a `String` primitive or object. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isString('abc'); | |
* // => true | |
* | |
* _.isString(1); | |
* // => false | |
*/ | |
function isString(value) { | |
return typeof value == 'string' || | |
(!isArray(value) && isObjectLike(value) && objectToString.call(value) == stringTag); | |
} | |
/** | |
* Checks if `value` is classified as a typed array. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is correctly classified, | |
* else `false`. | |
* @example | |
* | |
* _.isTypedArray(new Uint8Array); | |
* // => true | |
* | |
* _.isTypedArray([]); | |
* // => false | |
*/ | |
function isTypedArray(value) { | |
return isObjectLike(value) && | |
isLength(value.length) && !!typedArrayTags[objectToString.call(value)]; | |
} | |
/** | |
* Checks if `value` is `undefined`. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Lang | |
* @param {*} value The value to check. | |
* @returns {boolean} Returns `true` if `value` is `undefined`, else `false`. | |
* @example | |
* | |
* _.isUndefined(void 0); | |
* // => true | |
* | |
* _.isUndefined(null); | |
* // => false | |
*/ | |
function isUndefined(value) { | |
return value === undefined; | |
} | |
/** | |
* Converts `value` to an array. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Lang | |
* @param {*} value The value to convert. | |
* @returns {Array} Returns the converted array. | |
* @example | |
* | |
* _.toArray({ 'a': 1, 'b': 2 }); | |
* // => [1, 2] | |
* | |
* _.toArray('abc'); | |
* // => ['a', 'b', 'c'] | |
* | |
* _.toArray(1); | |
* // => [] | |
* | |
* _.toArray(null); | |
* // => [] | |
*/ | |
function toArray(value) { | |
if (!isArrayLike(value)) { | |
return values(value); | |
} | |
return value.length ? copyArray(value) : []; | |
} | |
/** | |
* Converts `value` to an integer. | |
* | |
* **Note:** This method is loosely based on | |
* [`ToInteger`](http://www.ecma-international.org/ecma-262/6.0/#sec-tointeger). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to convert. | |
* @returns {number} Returns the converted integer. | |
* @example | |
* | |
* _.toInteger(3.2); | |
* // => 3 | |
* | |
* _.toInteger(Number.MIN_VALUE); | |
* // => 0 | |
* | |
* _.toInteger(Infinity); | |
* // => 1.7976931348623157e+308 | |
* | |
* _.toInteger('3.2'); | |
* // => 3 | |
*/ | |
var toInteger = Number; | |
/** | |
* Converts `value` to a number. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to process. | |
* @returns {number} Returns the number. | |
* @example | |
* | |
* _.toNumber(3.2); | |
* // => 3.2 | |
* | |
* _.toNumber(Number.MIN_VALUE); | |
* // => 5e-324 | |
* | |
* _.toNumber(Infinity); | |
* // => Infinity | |
* | |
* _.toNumber('3.2'); | |
* // => 3.2 | |
*/ | |
var toNumber = Number; | |
/** | |
* Converts `value` to a plain object flattening inherited enumerable string | |
* keyed properties of `value` to own properties of the plain object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Lang | |
* @param {*} value The value to convert. | |
* @returns {Object} Returns the converted plain object. | |
* @example | |
* | |
* function Foo() { | |
* this.b = 2; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* | |
* _.assign({ 'a': 1 }, new Foo); | |
* // => { 'a': 1, 'b': 2 } | |
* | |
* _.assign({ 'a': 1 }, _.toPlainObject(new Foo)); | |
* // => { 'a': 1, 'b': 2, 'c': 3 } | |
*/ | |
function toPlainObject(value) { | |
return copyObject(value, keysIn(value)); | |
} | |
/** | |
* Converts `value` to a string. An empty string is returned for `null` | |
* and `undefined` values. The sign of `-0` is preserved. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Lang | |
* @param {*} value The value to process. | |
* @returns {string} Returns the string. | |
* @example | |
* | |
* _.toString(null); | |
* // => '' | |
* | |
* _.toString(-0); | |
* // => '-0' | |
* | |
* _.toString([1, 2, 3]); | |
* // => '1,2,3' | |
*/ | |
function toString(value) { | |
if (typeof value == 'string') { | |
return value; | |
} | |
return value == null ? '' : (value + ''); | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Assigns own enumerable string keyed properties of source objects to the | |
* destination object. Source objects are applied from left to right. | |
* Subsequent sources overwrite property assignments of previous sources. | |
* | |
* **Note:** This method mutates `object` and is loosely based on | |
* [`Object.assign`](https://mdn.io/Object/assign). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.10.0 | |
* @category Object | |
* @param {Object} object The destination object. | |
* @param {...Object} [sources] The source objects. | |
* @returns {Object} Returns `object`. | |
* @see _.assignIn | |
* @example | |
* | |
* function Foo() { | |
* this.c = 3; | |
* } | |
* | |
* function Bar() { | |
* this.e = 5; | |
* } | |
* | |
* Foo.prototype.d = 4; | |
* Bar.prototype.f = 6; | |
* | |
* _.assign({ 'a': 1 }, new Foo, new Bar); | |
* // => { 'a': 1, 'c': 3, 'e': 5 } | |
*/ | |
var assign = createAssigner(function(object, source) { | |
copyObject(source, keys(source), object); | |
}); | |
/** | |
* This method is like `_.assign` except that it iterates over own and | |
* inherited source properties. | |
* | |
* **Note:** This method mutates `object`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @alias extend | |
* @category Object | |
* @param {Object} object The destination object. | |
* @param {...Object} [sources] The source objects. | |
* @returns {Object} Returns `object`. | |
* @see _.assign | |
* @example | |
* | |
* function Foo() { | |
* this.b = 2; | |
* } | |
* | |
* function Bar() { | |
* this.d = 4; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* Bar.prototype.e = 5; | |
* | |
* _.assignIn({ 'a': 1 }, new Foo, new Bar); | |
* // => { 'a': 1, 'b': 2, 'c': 3, 'd': 4, 'e': 5 } | |
*/ | |
var assignIn = createAssigner(function(object, source) { | |
copyObject(source, keysIn(source), object); | |
}); | |
/** | |
* This method is like `_.assignIn` except that it accepts `customizer` | |
* which is invoked to produce the assigned values. If `customizer` returns | |
* `undefined`, assignment is handled by the method instead. The `customizer` | |
* is invoked with five arguments: (objValue, srcValue, key, object, source). | |
* | |
* **Note:** This method mutates `object`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @alias extendWith | |
* @category Object | |
* @param {Object} object The destination object. | |
* @param {...Object} sources The source objects. | |
* @param {Function} [customizer] The function to customize assigned values. | |
* @returns {Object} Returns `object`. | |
* @see _.assignWith | |
* @example | |
* | |
* function customizer(objValue, srcValue) { | |
* return _.isUndefined(objValue) ? srcValue : objValue; | |
* } | |
* | |
* var defaults = _.partialRight(_.assignInWith, customizer); | |
* | |
* defaults({ 'a': 1 }, { 'b': 2 }, { 'a': 3 }); | |
* // => { 'a': 1, 'b': 2 } | |
*/ | |
var assignInWith = createAssigner(function(object, source, srcIndex, customizer) { | |
copyObject(source, keysIn(source), object, customizer); | |
}); | |
/** | |
* Creates an object that inherits from the `prototype` object. If a | |
* `properties` object is given, its own enumerable string keyed properties | |
* are assigned to the created object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 2.3.0 | |
* @category Object | |
* @param {Object} prototype The object to inherit from. | |
* @param {Object} [properties] The properties to assign to the object. | |
* @returns {Object} Returns the new object. | |
* @example | |
* | |
* function Shape() { | |
* this.x = 0; | |
* this.y = 0; | |
* } | |
* | |
* function Circle() { | |
* Shape.call(this); | |
* } | |
* | |
* Circle.prototype = _.create(Shape.prototype, { | |
* 'constructor': Circle | |
* }); | |
* | |
* var circle = new Circle; | |
* circle instanceof Circle; | |
* // => true | |
* | |
* circle instanceof Shape; | |
* // => true | |
*/ | |
function create(prototype, properties) { | |
var result = baseCreate(prototype); | |
return properties ? assign(result, properties) : result; | |
} | |
/** | |
* Assigns own and inherited enumerable string keyed properties of source | |
* objects to the destination object for all destination properties that | |
* resolve to `undefined`. Source objects are applied from left to right. | |
* Once a property is set, additional values of the same property are ignored. | |
* | |
* **Note:** This method mutates `object`. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The destination object. | |
* @param {...Object} [sources] The source objects. | |
* @returns {Object} Returns `object`. | |
* @see _.defaultsDeep | |
* @example | |
* | |
* _.defaults({ 'user': 'barney' }, { 'age': 36 }, { 'user': 'fred' }); | |
* // => { 'user': 'barney', 'age': 36 } | |
*/ | |
var defaults = rest(function(args) { | |
args.push(undefined, assignInDefaults); | |
return assignInWith.apply(undefined, args); | |
}); | |
/** | |
* Iterates over own and inherited enumerable string keyed properties of an | |
* object and invokes `iteratee` for each property. The iteratee is invoked | |
* with three arguments: (value, key, object). Iteratee functions may exit | |
* iteration early by explicitly returning `false`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.3.0 | |
* @category Object | |
* @param {Object} object The object to iterate over. | |
* @param {Function} [iteratee=_.identity] The function invoked per iteration. | |
* @returns {Object} Returns `object`. | |
* @see _.forInRight | |
* @example | |
* | |
* function Foo() { | |
* this.a = 1; | |
* this.b = 2; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* | |
* _.forIn(new Foo, function(value, key) { | |
* console.log(key); | |
* }); | |
* // => Logs 'a', 'b', then 'c' (iteration order is not guaranteed). | |
*/ | |
function forIn(object, iteratee) { | |
return object == null | |
? object | |
: baseFor(object, baseIteratee(iteratee, 3), keysIn); | |
} | |
/** | |
* Checks if `path` is a direct property of `object`. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The object to query. | |
* @param {Array|string} path The path to check. | |
* @returns {boolean} Returns `true` if `path` exists, else `false`. | |
* @example | |
* | |
* var object = { 'a': { 'b': 2 } }; | |
* var other = _.create({ 'a': _.create({ 'b': 2 }) }); | |
* | |
* _.has(object, 'a'); | |
* // => true | |
* | |
* _.has(object, 'a.b'); | |
* // => true | |
* | |
* _.has(object, ['a', 'b']); | |
* // => true | |
* | |
* _.has(other, 'a'); | |
* // => false | |
*/ | |
function has(object, path) { | |
return object != null && hasOwnProperty.call(object, path); | |
} | |
/** | |
* Creates an array of the own enumerable property names of `object`. | |
* | |
* **Note:** Non-object values are coerced to objects. See the | |
* [ES spec](http://ecma-international.org/ecma-262/6.0/#sec-object.keys) | |
* for more details. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names. | |
* @example | |
* | |
* function Foo() { | |
* this.a = 1; | |
* this.b = 2; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* | |
* _.keys(new Foo); | |
* // => ['a', 'b'] (iteration order is not guaranteed) | |
* | |
* _.keys('hi'); | |
* // => ['0', '1'] | |
*/ | |
var keys = baseKeys; | |
/** | |
* Creates an array of the own and inherited enumerable property names of `object`. | |
* | |
* **Note:** Non-object values are coerced to objects. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Object | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property names. | |
* @example | |
* | |
* function Foo() { | |
* this.a = 1; | |
* this.b = 2; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* | |
* _.keysIn(new Foo); | |
* // => ['a', 'b', 'c'] (iteration order is not guaranteed) | |
*/ | |
var keysIn = baseKeysIn; | |
/** | |
* This method is like `_.assign` except that it recursively merges own and | |
* inherited enumerable string keyed properties of source objects into the | |
* destination object. Source properties that resolve to `undefined` are | |
* skipped if a destination value exists. Array and plain object properties | |
* are merged recursively. Other objects and value types are overridden by | |
* assignment. Source objects are applied from left to right. Subsequent | |
* sources overwrite property assignments of previous sources. | |
* | |
* **Note:** This method mutates `object`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.5.0 | |
* @category Object | |
* @param {Object} object The destination object. | |
* @param {...Object} [sources] The source objects. | |
* @returns {Object} Returns `object`. | |
* @example | |
* | |
* var users = { | |
* 'data': [{ 'user': 'barney' }, { 'user': 'fred' }] | |
* }; | |
* | |
* var ages = { | |
* 'data': [{ 'age': 36 }, { 'age': 40 }] | |
* }; | |
* | |
* _.merge(users, ages); | |
* // => { 'data': [{ 'user': 'barney', 'age': 36 }, { 'user': 'fred', 'age': 40 }] } | |
*/ | |
var merge = createAssigner(function(object, source, srcIndex) { | |
baseMerge(object, source, srcIndex); | |
}); | |
/** | |
* Creates an object composed of the picked `object` properties. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The source object. | |
* @param {...(string|string[])} [props] The property identifiers to pick. | |
* @returns {Object} Returns the new object. | |
* @example | |
* | |
* var object = { 'a': 1, 'b': '2', 'c': 3 }; | |
* | |
* _.pick(object, ['a', 'c']); | |
* // => { 'a': 1, 'c': 3 } | |
*/ | |
var pick = rest(function(object, props) { | |
return object == null ? {} : basePick(object, baseMap(baseFlatten(props, 1), toKey)); | |
}); | |
/** | |
* Creates an object composed of the `object` properties `predicate` returns | |
* truthy for. The predicate is invoked with two arguments: (value, key). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Object | |
* @param {Object} object The source object. | |
* @param {Array|Function|Object|string} [predicate=_.identity] | |
* The function invoked per property. | |
* @returns {Object} Returns the new object. | |
* @example | |
* | |
* var object = { 'a': 1, 'b': '2', 'c': 3 }; | |
* | |
* _.pickBy(object, _.isNumber); | |
* // => { 'a': 1, 'c': 3 } | |
*/ | |
function pickBy(object, predicate) { | |
return object == null ? {} : basePickBy(object, baseIteratee(predicate)); | |
} | |
/** | |
* This method is like `_.get` except that if the resolved value is a | |
* function it's invoked with the `this` binding of its parent object and | |
* its result is returned. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The object to query. | |
* @param {Array|string} path The path of the property to resolve. | |
* @param {*} [defaultValue] The value returned for `undefined` resolved values. | |
* @returns {*} Returns the resolved value. | |
* @example | |
* | |
* var object = { 'a': [{ 'b': { 'c1': 3, 'c2': _.constant(4) } }] }; | |
* | |
* _.result(object, 'a[0].b.c1'); | |
* // => 3 | |
* | |
* _.result(object, 'a[0].b.c2'); | |
* // => 4 | |
* | |
* _.result(object, 'a[0].b.c3', 'default'); | |
* // => 'default' | |
* | |
* _.result(object, 'a[0].b.c3', _.constant('default')); | |
* // => 'default' | |
*/ | |
function result(object, path, defaultValue) { | |
var value = object == null ? undefined : object[path]; | |
if (value === undefined) { | |
value = defaultValue; | |
} | |
return isFunction(value) ? value.call(object) : value; | |
} | |
/** | |
* Creates an array of the own enumerable string keyed property values of `object`. | |
* | |
* **Note:** Non-object values are coerced to objects. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Object | |
* @param {Object} object The object to query. | |
* @returns {Array} Returns the array of property values. | |
* @example | |
* | |
* function Foo() { | |
* this.a = 1; | |
* this.b = 2; | |
* } | |
* | |
* Foo.prototype.c = 3; | |
* | |
* _.values(new Foo); | |
* // => [1, 2] (iteration order is not guaranteed) | |
* | |
* _.values('hi'); | |
* // => ['h', 'i'] | |
*/ | |
function values(object) { | |
return object ? baseValues(object, keys(object)) : []; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Converts `string` to [camel case](https://en.wikipedia.org/wiki/CamelCase). | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the camel cased string. | |
* @example | |
* | |
* _.camelCase('Foo Bar'); | |
* // => 'fooBar' | |
* | |
* _.camelCase('--foo-bar--'); | |
* // => 'fooBar' | |
* | |
* _.camelCase('__FOO_BAR__'); | |
* // => 'fooBar' | |
*/ | |
var camelCase = createCompounder(function(result, word, index) { | |
word = word.toLowerCase(); | |
return result + (index ? capitalize(word) : word); | |
}); | |
/** | |
* Converts the first character of `string` to upper case and the remaining | |
* to lower case. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to capitalize. | |
* @returns {string} Returns the capitalized string. | |
* @example | |
* | |
* _.capitalize('FRED'); | |
* // => 'Fred' | |
*/ | |
function capitalize(string) { | |
return upperFirst(toString(string).toLowerCase()); | |
} | |
/** | |
* Deburrs `string` by converting | |
* [latin-1 supplementary letters](https://en.wikipedia.org/wiki/Latin-1_Supplement_(Unicode_block)#Character_table) | |
* to basic latin letters and removing | |
* [combining diacritical marks](https://en.wikipedia.org/wiki/Combining_Diacritical_Marks). | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to deburr. | |
* @returns {string} Returns the deburred string. | |
* @example | |
* | |
* _.deburr('déjà vu'); | |
* // => 'deja vu' | |
*/ | |
function deburr(string) { | |
string = toString(string); | |
return string && string.replace(reLatin1, deburrLetter).replace(reComboMark, ''); | |
} | |
/** | |
* Checks if `string` ends with the given target string. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to search. | |
* @param {string} [target] The string to search for. | |
* @param {number} [position=string.length] The position to search up to. | |
* @returns {boolean} Returns `true` if `string` ends with `target`, | |
* else `false`. | |
* @example | |
* | |
* _.endsWith('abc', 'c'); | |
* // => true | |
* | |
* _.endsWith('abc', 'b'); | |
* // => false | |
* | |
* _.endsWith('abc', 'b', 2); | |
* // => true | |
*/ | |
function endsWith(string, target, position) { | |
string = toString(string); | |
target = baseToString(target); | |
var length = string.length; | |
position = position === undefined | |
? length | |
: baseClamp(toInteger(position), 0, length); | |
position -= target.length; | |
return position >= 0 && string.indexOf(target, position) == position; | |
} | |
/** | |
* Converts the characters "&", "<", ">", '"', "'", and "\`" in `string` to | |
* their corresponding HTML entities. | |
* | |
* **Note:** No other characters are escaped. To escape additional | |
* characters use a third-party library like [_he_](https://mths.be/he). | |
* | |
* Though the ">" character is escaped for symmetry, characters like | |
* ">" and "/" don't need escaping in HTML and have no special meaning | |
* unless they're part of a tag or unquoted attribute value. See | |
* [Mathias Bynens's article](https://mathiasbynens.be/notes/ambiguous-ampersands) | |
* (under "semi-related fun fact") for more details. | |
* | |
* Backticks are escaped because in IE < 9, they can break out of | |
* attribute values or HTML comments. See [#59](https://html5sec.org/#59), | |
* [#102](https://html5sec.org/#102), [#108](https://html5sec.org/#108), and | |
* [#133](https://html5sec.org/#133) of the | |
* [HTML5 Security Cheatsheet](https://html5sec.org/) for more details. | |
* | |
* When working with HTML you should always | |
* [quote attribute values](http://wonko.com/post/html-escaping) to reduce | |
* XSS vectors. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category String | |
* @param {string} [string=''] The string to escape. | |
* @returns {string} Returns the escaped string. | |
* @example | |
* | |
* _.escape('fred, barney, & pebbles'); | |
* // => 'fred, barney, & pebbles' | |
*/ | |
function escape(string) { | |
string = toString(string); | |
return (string && reHasUnescapedHtml.test(string)) | |
? string.replace(reUnescapedHtml, escapeHtmlChar) | |
: string; | |
} | |
/** | |
* Escapes the `RegExp` special characters "^", "$", "\", ".", "*", "+", | |
* "?", "(", ")", "[", "]", "{", "}", and "|" in `string`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to escape. | |
* @returns {string} Returns the escaped string. | |
* @example | |
* | |
* _.escapeRegExp('[lodash](https://lodash.com/)'); | |
* // => '\[lodash\]\(https://lodash\.com/\)' | |
*/ | |
function escapeRegExp(string) { | |
string = toString(string); | |
return (string && reHasRegExpChar.test(string)) | |
? string.replace(reRegExpChar, '\\$&') | |
: string; | |
} | |
/** | |
* Converts `string` to | |
* [kebab case](https://en.wikipedia.org/wiki/Letter_case#Special_case_styles). | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the kebab cased string. | |
* @example | |
* | |
* _.kebabCase('Foo Bar'); | |
* // => 'foo-bar' | |
* | |
* _.kebabCase('fooBar'); | |
* // => 'foo-bar' | |
* | |
* _.kebabCase('__FOO_BAR__'); | |
* // => 'foo-bar' | |
*/ | |
var kebabCase = createCompounder(function(result, word, index) { | |
return result + (index ? '-' : '') + word.toLowerCase(); | |
}); | |
/** | |
* Converts `string`, as space separated words, to lower case. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the lower cased string. | |
* @example | |
* | |
* _.lowerCase('--Foo-Bar--'); | |
* // => 'foo bar' | |
* | |
* _.lowerCase('fooBar'); | |
* // => 'foo bar' | |
* | |
* _.lowerCase('__FOO_BAR__'); | |
* // => 'foo bar' | |
*/ | |
var lowerCase = createCompounder(function(result, word, index) { | |
return result + (index ? ' ' : '') + word.toLowerCase(); | |
}); | |
/** | |
* Converts the first character of `string` to lower case. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the converted string. | |
* @example | |
* | |
* _.lowerFirst('Fred'); | |
* // => 'fred' | |
* | |
* _.lowerFirst('FRED'); | |
* // => 'fRED' | |
*/ | |
var lowerFirst = createCaseFirst('toLowerCase'); | |
/** | |
* Pads `string` on the left and right sides if it's shorter than `length`. | |
* Padding characters are truncated if they can't be evenly divided by `length`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to pad. | |
* @param {number} [length=0] The padding length. | |
* @param {string} [chars=' '] The string used as padding. | |
* @returns {string} Returns the padded string. | |
* @example | |
* | |
* _.pad('abc', 8); | |
* // => ' abc ' | |
* | |
* _.pad('abc', 8, '_-'); | |
* // => '_-abc_-_' | |
* | |
* _.pad('abc', 3); | |
* // => 'abc' | |
*/ | |
function pad(string, length, chars) { | |
string = toString(string); | |
length = toInteger(length); | |
var strLength = length ? stringSize(string) : 0; | |
if (!length || strLength >= length) { | |
return string; | |
} | |
var mid = (length - strLength) / 2; | |
return ( | |
createPadding(nativeFloor(mid), chars) + | |
string + | |
createPadding(nativeCeil(mid), chars) | |
); | |
} | |
/** | |
* Pads `string` on the right side if it's shorter than `length`. Padding | |
* characters are truncated if they exceed `length`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to pad. | |
* @param {number} [length=0] The padding length. | |
* @param {string} [chars=' '] The string used as padding. | |
* @returns {string} Returns the padded string. | |
* @example | |
* | |
* _.padEnd('abc', 6); | |
* // => 'abc ' | |
* | |
* _.padEnd('abc', 6, '_-'); | |
* // => 'abc_-_' | |
* | |
* _.padEnd('abc', 3); | |
* // => 'abc' | |
*/ | |
function padEnd(string, length, chars) { | |
string = toString(string); | |
length = toInteger(length); | |
var strLength = length ? stringSize(string) : 0; | |
return (length && strLength < length) | |
? (string + createPadding(length - strLength, chars)) | |
: string; | |
} | |
/** | |
* Pads `string` on the left side if it's shorter than `length`. Padding | |
* characters are truncated if they exceed `length`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to pad. | |
* @param {number} [length=0] The padding length. | |
* @param {string} [chars=' '] The string used as padding. | |
* @returns {string} Returns the padded string. | |
* @example | |
* | |
* _.padStart('abc', 6); | |
* // => ' abc' | |
* | |
* _.padStart('abc', 6, '_-'); | |
* // => '_-_abc' | |
* | |
* _.padStart('abc', 3); | |
* // => 'abc' | |
*/ | |
function padStart(string, length, chars) { | |
string = toString(string); | |
length = toInteger(length); | |
var strLength = length ? stringSize(string) : 0; | |
return (length && strLength < length) | |
? (createPadding(length - strLength, chars) + string) | |
: string; | |
} | |
/** | |
* Converts `string` to an integer of the specified radix. If `radix` is | |
* `undefined` or `0`, a `radix` of `10` is used unless `value` is a | |
* hexadecimal, in which case a `radix` of `16` is used. | |
* | |
* **Note:** This method aligns with the | |
* [ES5 implementation](https://es5.github.io/#x15.1.2.2) of `parseInt`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 1.1.0 | |
* @category String | |
* @param {string} string The string to convert. | |
* @param {number} [radix=10] The radix to interpret `value` by. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {number} Returns the converted integer. | |
* @example | |
* | |
* _.parseInt('08'); | |
* // => 8 | |
* | |
* _.map(['6', '08', '10'], _.parseInt); | |
* // => [6, 8, 10] | |
*/ | |
function parseInt(string, radix, guard) { | |
// Chrome fails to trim leading <BOM> whitespace characters. | |
// See https://bugs.chromium.org/p/v8/issues/detail?id=3109 for more details. | |
if (guard || radix == null) { | |
radix = 0; | |
} else if (radix) { | |
radix = +radix; | |
} | |
string = toString(string).replace(reTrim, ''); | |
return nativeParseInt(string, radix || (reHasHexPrefix.test(string) ? 16 : 10)); | |
} | |
/** | |
* Repeats the given string `n` times. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to repeat. | |
* @param {number} [n=1] The number of times to repeat the string. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {string} Returns the repeated string. | |
* @example | |
* | |
* _.repeat('*', 3); | |
* // => '***' | |
* | |
* _.repeat('abc', 2); | |
* // => 'abcabc' | |
* | |
* _.repeat('abc', 0); | |
* // => '' | |
*/ | |
function repeat(string, n, guard) { | |
if ((guard ? isIterateeCall(string, n, guard) : n === undefined)) { | |
n = 1; | |
} else { | |
n = toInteger(n); | |
} | |
return baseRepeat(toString(string), n); | |
} | |
/** | |
* Replaces matches for `pattern` in `string` with `replacement`. | |
* | |
* **Note:** This method is based on | |
* [`String#replace`](https://mdn.io/String/replace). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to modify. | |
* @param {RegExp|string} pattern The pattern to replace. | |
* @param {Function|string} replacement The match replacement. | |
* @returns {string} Returns the modified string. | |
* @example | |
* | |
* _.replace('Hi Fred', 'Fred', 'Barney'); | |
* // => 'Hi Barney' | |
*/ | |
function replace() { | |
var args = arguments, | |
string = toString(args[0]); | |
return args.length < 3 ? string : nativeReplace.call(string, args[1], args[2]); | |
} | |
/** | |
* Converts `string` to | |
* [snake case](https://en.wikipedia.org/wiki/Snake_case). | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the snake cased string. | |
* @example | |
* | |
* _.snakeCase('Foo Bar'); | |
* // => 'foo_bar' | |
* | |
* _.snakeCase('fooBar'); | |
* // => 'foo_bar' | |
* | |
* _.snakeCase('--FOO-BAR--'); | |
* // => 'foo_bar' | |
*/ | |
var snakeCase = createCompounder(function(result, word, index) { | |
return result + (index ? '_' : '') + word.toLowerCase(); | |
}); | |
/** | |
* Splits `string` by `separator`. | |
* | |
* **Note:** This method is based on | |
* [`String#split`](https://mdn.io/String/split). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to split. | |
* @param {RegExp|string} separator The separator pattern to split by. | |
* @param {number} [limit] The length to truncate results to. | |
* @returns {Array} Returns the string segments. | |
* @example | |
* | |
* _.split('a-b-c', '-', 2); | |
* // => ['a', 'b'] | |
*/ | |
function split(string, separator, limit) { | |
if (limit && typeof limit != 'number' && isIterateeCall(string, separator, limit)) { | |
separator = limit = undefined; | |
} | |
limit = limit === undefined ? MAX_ARRAY_LENGTH : limit >>> 0; | |
if (!limit) { | |
return []; | |
} | |
string = toString(string); | |
if (string && ( | |
typeof separator == 'string' || | |
(separator != null && !isRegExp(separator)) | |
)) { | |
separator = baseToString(separator); | |
if (separator == '' && reHasComplexSymbol.test(string)) { | |
return castSlice(stringToArray(string), 0, limit); | |
} | |
} | |
return nativeSplit.call(string, separator, limit); | |
} | |
/** | |
* Converts `string` to | |
* [start case](https://en.wikipedia.org/wiki/Letter_case#Stylistic_or_specialised_usage). | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.1.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the start cased string. | |
* @example | |
* | |
* _.startCase('--foo-bar--'); | |
* // => 'Foo Bar' | |
* | |
* _.startCase('fooBar'); | |
* // => 'Foo Bar' | |
* | |
* _.startCase('__FOO_BAR__'); | |
* // => 'FOO BAR' | |
*/ | |
var startCase = createCompounder(function(result, word, index) { | |
return result + (index ? ' ' : '') + upperFirst(word); | |
}); | |
/** | |
* Checks if `string` starts with the given target string. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to search. | |
* @param {string} [target] The string to search for. | |
* @param {number} [position=0] The position to search from. | |
* @returns {boolean} Returns `true` if `string` starts with `target`, | |
* else `false`. | |
* @example | |
* | |
* _.startsWith('abc', 'a'); | |
* // => true | |
* | |
* _.startsWith('abc', 'b'); | |
* // => false | |
* | |
* _.startsWith('abc', 'b', 1); | |
* // => true | |
*/ | |
function startsWith(string, target, position) { | |
string = toString(string); | |
position = baseClamp(toInteger(position), 0, string.length); | |
return string.lastIndexOf(baseToString(target), position) == position; | |
} | |
/** | |
* Creates a compiled template function that can interpolate data properties | |
* in "interpolate" delimiters, HTML-escape interpolated data properties in | |
* "escape" delimiters, and execute JavaScript in "evaluate" delimiters. Data | |
* properties may be accessed as free variables in the template. If a setting | |
* object is given, it takes precedence over `_.templateSettings` values. | |
* | |
* **Note:** In the development build `_.template` utilizes | |
* [sourceURLs](http://www.html5rocks.com/en/tutorials/developertools/sourcemaps/#toc-sourceurl) | |
* for easier debugging. | |
* | |
* For more information on precompiling templates see | |
* [lodash's custom builds documentation](https://lodash.com/custom-builds). | |
* | |
* For more information on Chrome extension sandboxes see | |
* [Chrome's extensions documentation](https://developer.chrome.com/extensions/sandboxingEval). | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category String | |
* @param {string} [string=''] The template string. | |
* @param {Object} [options={}] The options object. | |
* @param {RegExp} [options.escape=_.templateSettings.escape] | |
* The HTML "escape" delimiter. | |
* @param {RegExp} [options.evaluate=_.templateSettings.evaluate] | |
* The "evaluate" delimiter. | |
* @param {Object} [options.imports=_.templateSettings.imports] | |
* An object to import into the template as free variables. | |
* @param {RegExp} [options.interpolate=_.templateSettings.interpolate] | |
* The "interpolate" delimiter. | |
* @param {string} [options.sourceURL='lodash.templateSources[n]'] | |
* The sourceURL of the compiled template. | |
* @param {string} [options.variable='obj'] | |
* The data object variable name. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {Function} Returns the compiled template function. | |
* @example | |
* | |
* // Use the "interpolate" delimiter to create a compiled template. | |
* var compiled = _.template('hello <%= user %>!'); | |
* compiled({ 'user': 'fred' }); | |
* // => 'hello fred!' | |
* | |
* // Use the HTML "escape" delimiter to escape data property values. | |
* var compiled = _.template('<b><%- value %></b>'); | |
* compiled({ 'value': '<script>' }); | |
* // => '<b><script></b>' | |
* | |
* // Use the "evaluate" delimiter to execute JavaScript and generate HTML. | |
* var compiled = _.template('<% _.forEach(users, function(user) { %><li><%- user %></li><% }); %>'); | |
* compiled({ 'users': ['fred', 'barney'] }); | |
* // => '<li>fred</li><li>barney</li>' | |
* | |
* // Use the internal `print` function in "evaluate" delimiters. | |
* var compiled = _.template('<% print("hello " + user); %>!'); | |
* compiled({ 'user': 'barney' }); | |
* // => 'hello barney!' | |
* | |
* // Use the ES delimiter as an alternative to the default "interpolate" delimiter. | |
* var compiled = _.template('hello ${ user }!'); | |
* compiled({ 'user': 'pebbles' }); | |
* // => 'hello pebbles!' | |
* | |
* // Use backslashes to treat delimiters as plain text. | |
* var compiled = _.template('<%= "\\<%- value %\\>" %>'); | |
* compiled({ 'value': 'ignored' }); | |
* // => '<%- value %>' | |
* | |
* // Use the `imports` option to import `jQuery` as `jq`. | |
* var text = '<% jq.each(users, function(user) { %><li><%- user %></li><% }); %>'; | |
* var compiled = _.template(text, { 'imports': { 'jq': jQuery } }); | |
* compiled({ 'users': ['fred', 'barney'] }); | |
* // => '<li>fred</li><li>barney</li>' | |
* | |
* // Use the `sourceURL` option to specify a custom sourceURL for the template. | |
* var compiled = _.template('hello <%= user %>!', { 'sourceURL': '/basic/greeting.jst' }); | |
* compiled(data); | |
* // => Find the source of "greeting.jst" under the Sources tab or Resources panel of the web inspector. | |
* | |
* // Use the `variable` option to ensure a with-statement isn't used in the compiled template. | |
* var compiled = _.template('hi <%= data.user %>!', { 'variable': 'data' }); | |
* compiled.source; | |
* // => function(data) { | |
* // var __t, __p = ''; | |
* // __p += 'hi ' + ((__t = ( data.user )) == null ? '' : __t) + '!'; | |
* // return __p; | |
* // } | |
* | |
* // Use custom template delimiters. | |
* _.templateSettings.interpolate = /{{([\s\S]+?)}}/g; | |
* var compiled = _.template('hello {{ user }}!'); | |
* compiled({ 'user': 'mustache' }); | |
* // => 'hello mustache!' | |
* | |
* // Use the `source` property to inline compiled templates for meaningful | |
* // line numbers in error messages and stack traces. | |
* fs.writeFileSync(path.join(process.cwd(), 'jst.js'), '\ | |
* var JST = {\ | |
* "main": ' + _.template(mainText).source + '\ | |
* };\ | |
* '); | |
*/ | |
function template(string, options, guard) { | |
// Based on John Resig's `tmpl` implementation | |
// (http://ejohn.org/blog/javascript-micro-templating/) | |
// and Laura Doktorova's doT.js (https://github.com/olado/doT). | |
var settings = lodash.templateSettings; | |
if (guard && isIterateeCall(string, options, guard)) { | |
options = undefined; | |
} | |
string = toString(string); | |
options = assignInWith({}, options, settings, assignInDefaults); | |
var imports = assignInWith({}, options.imports, settings.imports, assignInDefaults), | |
importsKeys = keys(imports), | |
importsValues = baseValues(imports, importsKeys); | |
var isEscaping, | |
isEvaluating, | |
index = 0, | |
interpolate = options.interpolate || reNoMatch, | |
source = "__p += '"; | |
// Compile the regexp to match each delimiter. | |
var reDelimiters = RegExp( | |
(options.escape || reNoMatch).source + '|' + | |
interpolate.source + '|' + | |
(interpolate === reInterpolate ? reEsTemplate : reNoMatch).source + '|' + | |
(options.evaluate || reNoMatch).source + '|$' | |
, 'g'); | |
// Use a sourceURL for easier debugging. | |
var sourceURL = '//# sourceURL=' + | |
('sourceURL' in options | |
? options.sourceURL | |
: ('lodash.templateSources[' + (++templateCounter) + ']') | |
) + '\n'; | |
string.replace(reDelimiters, function(match, escapeValue, interpolateValue, esTemplateValue, evaluateValue, offset) { | |
interpolateValue || (interpolateValue = esTemplateValue); | |
// Escape characters that can't be included in string literals. | |
source += string.slice(index, offset).replace(reUnescapedString, escapeStringChar); | |
// Replace delimiters with snippets. | |
if (escapeValue) { | |
isEscaping = true; | |
source += "' +\n__e(" + escapeValue + ") +\n'"; | |
} | |
if (evaluateValue) { | |
isEvaluating = true; | |
source += "';\n" + evaluateValue + ";\n__p += '"; | |
} | |
if (interpolateValue) { | |
source += "' +\n((__t = (" + interpolateValue + ")) == null ? '' : __t) +\n'"; | |
} | |
index = offset + match.length; | |
// The JS engine embedded in Adobe products needs `match` returned in | |
// order to produce the correct `offset` value. | |
return match; | |
}); | |
source += "';\n"; | |
// If `variable` is not specified wrap a with-statement around the generated | |
// code to add the data object to the top of the scope chain. | |
var variable = options.variable; | |
if (!variable) { | |
source = 'with (obj) {\n' + source + '\n}\n'; | |
} | |
// Cleanup code by stripping empty strings. | |
source = (isEvaluating ? source.replace(reEmptyStringLeading, '') : source) | |
.replace(reEmptyStringMiddle, '$1') | |
.replace(reEmptyStringTrailing, '$1;'); | |
// Frame code as the function body. | |
source = 'function(' + (variable || 'obj') + ') {\n' + | |
(variable | |
? '' | |
: 'obj || (obj = {});\n' | |
) + | |
"var __t, __p = ''" + | |
(isEscaping | |
? ', __e = _.escape' | |
: '' | |
) + | |
(isEvaluating | |
? ', __j = Array.prototype.join;\n' + | |
"function print() { __p += __j.call(arguments, '') }\n" | |
: ';\n' | |
) + | |
source + | |
'return __p\n}'; | |
var result = attempt(function() { | |
return Function(importsKeys, sourceURL + 'return ' + source) | |
.apply(undefined, importsValues); | |
}); | |
// Provide the compiled function's source by its `toString` method or | |
// the `source` property as a convenience for inlining compiled templates. | |
result.source = source; | |
if (isError(result)) { | |
throw result; | |
} | |
return result; | |
} | |
/** | |
* Converts `string`, as a whole, to lower case just like | |
* [String#toLowerCase](https://mdn.io/toLowerCase). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the lower cased string. | |
* @example | |
* | |
* _.toLower('--Foo-Bar--'); | |
* // => '--foo-bar--' | |
* | |
* _.toLower('fooBar'); | |
* // => 'foobar' | |
* | |
* _.toLower('__FOO_BAR__'); | |
* // => '__foo_bar__' | |
*/ | |
function toLower(value) { | |
return toString(value).toLowerCase(); | |
} | |
/** | |
* Converts `string`, as a whole, to upper case just like | |
* [String#toUpperCase](https://mdn.io/toUpperCase). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the upper cased string. | |
* @example | |
* | |
* _.toUpper('--foo-bar--'); | |
* // => '--FOO-BAR--' | |
* | |
* _.toUpper('fooBar'); | |
* // => 'FOOBAR' | |
* | |
* _.toUpper('__foo_bar__'); | |
* // => '__FOO_BAR__' | |
*/ | |
function toUpper(value) { | |
return toString(value).toUpperCase(); | |
} | |
/** | |
* Removes leading and trailing whitespace or specified characters from `string`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to trim. | |
* @param {string} [chars=whitespace] The characters to trim. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {string} Returns the trimmed string. | |
* @example | |
* | |
* _.trim(' abc '); | |
* // => 'abc' | |
* | |
* _.trim('-_-abc-_-', '_-'); | |
* // => 'abc' | |
* | |
* _.map([' foo ', ' bar '], _.trim); | |
* // => ['foo', 'bar'] | |
*/ | |
function trim(string, chars, guard) { | |
string = toString(string); | |
if (string && (guard || chars === undefined)) { | |
return string.replace(reTrim, ''); | |
} | |
if (!string || !(chars = baseToString(chars))) { | |
return string; | |
} | |
var strSymbols = stringToArray(string), | |
chrSymbols = stringToArray(chars), | |
start = charsStartIndex(strSymbols, chrSymbols), | |
end = charsEndIndex(strSymbols, chrSymbols) + 1; | |
return castSlice(strSymbols, start, end).join(''); | |
} | |
/** | |
* Removes trailing whitespace or specified characters from `string`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to trim. | |
* @param {string} [chars=whitespace] The characters to trim. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {string} Returns the trimmed string. | |
* @example | |
* | |
* _.trimEnd(' abc '); | |
* // => ' abc' | |
* | |
* _.trimEnd('-_-abc-_-', '_-'); | |
* // => '-_-abc' | |
*/ | |
function trimEnd(string, chars, guard) { | |
string = toString(string); | |
if (string && (guard || chars === undefined)) { | |
return string.replace(reTrimEnd, ''); | |
} | |
if (!string || !(chars = baseToString(chars))) { | |
return string; | |
} | |
var strSymbols = stringToArray(string), | |
end = charsEndIndex(strSymbols, stringToArray(chars)) + 1; | |
return castSlice(strSymbols, 0, end).join(''); | |
} | |
/** | |
* Removes leading whitespace or specified characters from `string`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to trim. | |
* @param {string} [chars=whitespace] The characters to trim. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {string} Returns the trimmed string. | |
* @example | |
* | |
* _.trimStart(' abc '); | |
* // => 'abc ' | |
* | |
* _.trimStart('-_-abc-_-', '_-'); | |
* // => 'abc-_-' | |
*/ | |
function trimStart(string, chars, guard) { | |
string = toString(string); | |
if (string && (guard || chars === undefined)) { | |
return string.replace(reTrimStart, ''); | |
} | |
if (!string || !(chars = baseToString(chars))) { | |
return string; | |
} | |
var strSymbols = stringToArray(string), | |
start = charsStartIndex(strSymbols, stringToArray(chars)); | |
return castSlice(strSymbols, start).join(''); | |
} | |
/** | |
* Truncates `string` if it's longer than the given maximum string length. | |
* The last characters of the truncated string are replaced with the omission | |
* string which defaults to "...". | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to truncate. | |
* @param {Object} [options={}] The options object. | |
* @param {number} [options.length=30] The maximum string length. | |
* @param {string} [options.omission='...'] The string to indicate text is omitted. | |
* @param {RegExp|string} [options.separator] The separator pattern to truncate to. | |
* @returns {string} Returns the truncated string. | |
* @example | |
* | |
* _.truncate('hi-diddly-ho there, neighborino'); | |
* // => 'hi-diddly-ho there, neighbo...' | |
* | |
* _.truncate('hi-diddly-ho there, neighborino', { | |
* 'length': 24, | |
* 'separator': ' ' | |
* }); | |
* // => 'hi-diddly-ho there,...' | |
* | |
* _.truncate('hi-diddly-ho there, neighborino', { | |
* 'length': 24, | |
* 'separator': /,? +/ | |
* }); | |
* // => 'hi-diddly-ho there...' | |
* | |
* _.truncate('hi-diddly-ho there, neighborino', { | |
* 'omission': ' [...]' | |
* }); | |
* // => 'hi-diddly-ho there, neig [...]' | |
*/ | |
function truncate(string, options) { | |
var length = DEFAULT_TRUNC_LENGTH, | |
omission = DEFAULT_TRUNC_OMISSION; | |
if (isObject(options)) { | |
var separator = 'separator' in options ? options.separator : separator; | |
length = 'length' in options ? toInteger(options.length) : length; | |
omission = 'omission' in options ? baseToString(options.omission) : omission; | |
} | |
string = toString(string); | |
var strLength = string.length; | |
if (reHasComplexSymbol.test(string)) { | |
var strSymbols = stringToArray(string); | |
strLength = strSymbols.length; | |
} | |
if (length >= strLength) { | |
return string; | |
} | |
var end = length - stringSize(omission); | |
if (end < 1) { | |
return omission; | |
} | |
var result = strSymbols | |
? castSlice(strSymbols, 0, end).join('') | |
: string.slice(0, end); | |
if (separator === undefined) { | |
return result + omission; | |
} | |
if (strSymbols) { | |
end += (result.length - end); | |
} | |
if (isRegExp(separator)) { | |
if (string.slice(end).search(separator)) { | |
var match, | |
substring = result; | |
if (!separator.global) { | |
separator = RegExp(separator.source, toString(reFlags.exec(separator)) + 'g'); | |
} | |
separator.lastIndex = 0; | |
while ((match = separator.exec(substring))) { | |
var newEnd = match.index; | |
} | |
result = result.slice(0, newEnd === undefined ? end : newEnd); | |
} | |
} else if (string.indexOf(baseToString(separator), end) != end) { | |
var index = result.lastIndexOf(separator); | |
if (index > -1) { | |
result = result.slice(0, index); | |
} | |
} | |
return result + omission; | |
} | |
/** | |
* The inverse of `_.escape`; this method converts the HTML entities | |
* `&`, `<`, `>`, `"`, `'`, and ``` in `string` to | |
* their corresponding characters. | |
* | |
* **Note:** No other HTML entities are unescaped. To unescape additional | |
* HTML entities use a third-party library like [_he_](https://mths.be/he). | |
* | |
* @static | |
* @memberOf _ | |
* @since 0.6.0 | |
* @category String | |
* @param {string} [string=''] The string to unescape. | |
* @returns {string} Returns the unescaped string. | |
* @example | |
* | |
* _.unescape('fred, barney, & pebbles'); | |
* // => 'fred, barney, & pebbles' | |
*/ | |
function unescape(string) { | |
string = toString(string); | |
return (string && reHasEscapedHtml.test(string)) | |
? string.replace(reEscapedHtml, unescapeHtmlChar) | |
: string; | |
} | |
/** | |
* Converts `string`, as space separated words, to upper case. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the upper cased string. | |
* @example | |
* | |
* _.upperCase('--foo-bar'); | |
* // => 'FOO BAR' | |
* | |
* _.upperCase('fooBar'); | |
* // => 'FOO BAR' | |
* | |
* _.upperCase('__foo_bar__'); | |
* // => 'FOO BAR' | |
*/ | |
var upperCase = createCompounder(function(result, word, index) { | |
return result + (index ? ' ' : '') + word.toUpperCase(); | |
}); | |
/** | |
* Converts the first character of `string` to upper case. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category String | |
* @param {string} [string=''] The string to convert. | |
* @returns {string} Returns the converted string. | |
* @example | |
* | |
* _.upperFirst('fred'); | |
* // => 'Fred' | |
* | |
* _.upperFirst('FRED'); | |
* // => 'FRED' | |
*/ | |
var upperFirst = createCaseFirst('toUpperCase'); | |
/** | |
* Splits `string` into an array of its words. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category String | |
* @param {string} [string=''] The string to inspect. | |
* @param {RegExp|string} [pattern] The pattern to match words. | |
* @param- {Object} [guard] Enables use as an iteratee for methods like `_.map`. | |
* @returns {Array} Returns the words of `string`. | |
* @example | |
* | |
* _.words('fred, barney, & pebbles'); | |
* // => ['fred', 'barney', 'pebbles'] | |
* | |
* _.words('fred, barney, & pebbles', /[^, ]+/g); | |
* // => ['fred', 'barney', '&', 'pebbles'] | |
*/ | |
function words(string, pattern, guard) { | |
string = toString(string); | |
pattern = guard ? undefined : pattern; | |
if (pattern === undefined) { | |
pattern = reHasComplexWord.test(string) ? reComplexWord : reBasicWord; | |
} | |
return string.match(pattern) || []; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Attempts to invoke `func`, returning either the result or the caught error | |
* object. Any additional arguments are provided to `func` when it's invoked. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Util | |
* @param {Function} func The function to attempt. | |
* @param {...*} [args] The arguments to invoke `func` with. | |
* @returns {*} Returns the `func` result or error object. | |
* @example | |
* | |
* // Avoid throwing errors for invalid selectors. | |
* var elements = _.attempt(function(selector) { | |
* return document.querySelectorAll(selector); | |
* }, '>_>'); | |
* | |
* if (_.isError(elements)) { | |
* elements = []; | |
* } | |
*/ | |
var attempt = rest(function(func, args) { | |
try { | |
return func.apply(undefined, args); | |
} catch (e) { | |
return isError(e) ? e : new Error(e); | |
} | |
}); | |
/** | |
* This method returns the first argument given to it. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @param {*} value Any value. | |
* @returns {*} Returns `value`. | |
* @example | |
* | |
* var object = { 'user': 'fred' }; | |
* | |
* console.log(_.identity(object) === object); | |
* // => true | |
*/ | |
function identity(value) { | |
return value; | |
} | |
/** | |
* Creates a function that invokes `func` with the arguments of the created | |
* function. If `func` is a property name, the created function returns the | |
* property value for a given element. If `func` is an array or object, the | |
* created function returns `true` for elements that contain the equivalent | |
* source properties, otherwise it returns `false`. | |
* | |
* @static | |
* @since 4.0.0 | |
* @memberOf _ | |
* @category Util | |
* @param {*} [func=_.identity] The value to convert to a callback. | |
* @returns {Function} Returns the callback. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36, 'active': true }, | |
* { 'user': 'fred', 'age': 40, 'active': false } | |
* ]; | |
* | |
* // The `_.matches` iteratee shorthand. | |
* _.filter(users, _.iteratee({ 'user': 'barney', 'active': true })); | |
* // => [{ 'user': 'barney', 'age': 36, 'active': true }] | |
* | |
* // The `_.matchesProperty` iteratee shorthand. | |
* _.filter(users, _.iteratee(['user', 'fred'])); | |
* // => [{ 'user': 'fred', 'age': 40 }] | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.map(users, _.iteratee('user')); | |
* // => ['barney', 'fred'] | |
* | |
* // Create custom iteratee shorthands. | |
* _.iteratee = _.wrap(_.iteratee, function(iteratee, func) { | |
* return !_.isRegExp(func) ? iteratee(func) : function(string) { | |
* return func.test(string); | |
* }; | |
* }); | |
* | |
* _.filter(['abc', 'def'], /ef/); | |
* // => ['def'] | |
*/ | |
var iteratee = baseIteratee; | |
/** | |
* Creates a function that performs a partial deep comparison between a given | |
* object and `source`, returning `true` if the given object has equivalent | |
* property values, else `false`. The created function is equivalent to | |
* `_.isMatch` with a `source` partially applied. | |
* | |
* **Note:** This method supports comparing the same values as `_.isEqual`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.0.0 | |
* @category Util | |
* @param {Object} source The object of property values to match. | |
* @returns {Function} Returns the new spec function. | |
* @example | |
* | |
* var users = [ | |
* { 'user': 'barney', 'age': 36, 'active': true }, | |
* { 'user': 'fred', 'age': 40, 'active': false } | |
* ]; | |
* | |
* _.filter(users, _.matches({ 'age': 40, 'active': false })); | |
* // => [{ 'user': 'fred', 'age': 40, 'active': false }] | |
*/ | |
function matches(source) { | |
return baseMatches(assign({}, source)); | |
} | |
/** | |
* Creates a function that invokes the method at `path` of a given object. | |
* Any additional arguments are provided to the invoked method. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.7.0 | |
* @category Util | |
* @param {Array|string} path The path of the method to invoke. | |
* @param {...*} [args] The arguments to invoke the method with. | |
* @returns {Function} Returns the new invoker function. | |
* @example | |
* | |
* var objects = [ | |
* { 'a': { 'b': _.constant(2) } }, | |
* { 'a': { 'b': _.constant(1) } } | |
* ]; | |
* | |
* _.map(objects, _.method('a.b')); | |
* // => [2, 1] | |
* | |
* _.map(objects, _.method(['a', 'b'])); | |
* // => [2, 1] | |
*/ | |
var method = rest(function(path, args) { | |
return function(object) { | |
return baseInvoke(object, path, args); | |
}; | |
}); | |
/** | |
* Adds all own enumerable string keyed function properties of a source | |
* object to the destination object. If `object` is a function, then methods | |
* are added to its prototype as well. | |
* | |
* **Note:** Use `_.runInContext` to create a pristine `lodash` function to | |
* avoid conflicts caused by modifying the original. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @param {Function|Object} [object=lodash] The destination object. | |
* @param {Object} source The object of functions to add. | |
* @param {Object} [options={}] The options object. | |
* @param {boolean} [options.chain=true] Specify whether mixins are chainable. | |
* @returns {Function|Object} Returns `object`. | |
* @example | |
* | |
* function vowels(string) { | |
* return _.filter(string, function(v) { | |
* return /[aeiou]/i.test(v); | |
* }); | |
* } | |
* | |
* _.mixin({ 'vowels': vowels }); | |
* _.vowels('fred'); | |
* // => ['e'] | |
* | |
* _('fred').vowels().value(); | |
* // => ['e'] | |
* | |
* _.mixin({ 'vowels': vowels }, { 'chain': false }); | |
* _('fred').vowels(); | |
* // => ['e'] | |
*/ | |
function mixin(object, source, options) { | |
var props = keys(source), | |
methodNames = baseFunctions(source, props); | |
if (options == null && | |
!(isObject(source) && (methodNames.length || !props.length))) { | |
options = source; | |
source = object; | |
object = this; | |
methodNames = baseFunctions(source, keys(source)); | |
} | |
var chain = !(isObject(options) && 'chain' in options) || !!options.chain, | |
isFunc = isFunction(object); | |
baseEach(methodNames, function(methodName) { | |
var func = source[methodName]; | |
object[methodName] = func; | |
if (isFunc) { | |
object.prototype[methodName] = function() { | |
var chainAll = this.__chain__; | |
if (chain || chainAll) { | |
var result = object(this.__wrapped__), | |
actions = result.__actions__ = copyArray(this.__actions__); | |
actions.push({ 'func': func, 'args': arguments, 'thisArg': object }); | |
result.__chain__ = chainAll; | |
return result; | |
} | |
return func.apply(object, arrayPush([this.value()], arguments)); | |
}; | |
} | |
}); | |
return object; | |
} | |
/** | |
* Reverts the `_` variable to its previous value and returns a reference to | |
* the `lodash` function. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @returns {Function} Returns the `lodash` function. | |
* @example | |
* | |
* var lodash = _.noConflict(); | |
*/ | |
function noConflict() { | |
if (root._ === this) { | |
root._ = oldDash; | |
} | |
return this; | |
} | |
/** | |
* A method that returns `undefined`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 2.3.0 | |
* @category Util | |
* @example | |
* | |
* _.times(2, _.noop); | |
* // => [undefined, undefined] | |
*/ | |
function noop() { | |
// No operation performed. | |
} | |
/** | |
* Creates a function that returns the value at `path` of a given object. | |
* | |
* @static | |
* @memberOf _ | |
* @since 2.4.0 | |
* @category Util | |
* @param {Array|string} path The path of the property to get. | |
* @returns {Function} Returns the new accessor function. | |
* @example | |
* | |
* var objects = [ | |
* { 'a': { 'b': 2 } }, | |
* { 'a': { 'b': 1 } } | |
* ]; | |
* | |
* _.map(objects, _.property('a.b')); | |
* // => [2, 1] | |
* | |
* _.map(_.sortBy(objects, _.property(['a', 'b'])), 'a.b'); | |
* // => [1, 2] | |
*/ | |
var property = baseProperty; | |
/** | |
* Creates an array of numbers (positive and/or negative) progressing from | |
* `start` up to, but not including, `end`. A step of `-1` is used if a negative | |
* `start` is specified without an `end` or `step`. If `end` is not specified, | |
* it's set to `start` with `start` then set to `0`. | |
* | |
* **Note:** JavaScript follows the IEEE-754 standard for resolving | |
* floating-point values which can produce unexpected results. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @param {number} [start=0] The start of the range. | |
* @param {number} end The end of the range. | |
* @param {number} [step=1] The value to increment or decrement by. | |
* @returns {Array} Returns the range of numbers. | |
* @see _.inRange, _.rangeRight | |
* @example | |
* | |
* _.range(4); | |
* // => [0, 1, 2, 3] | |
* | |
* _.range(-4); | |
* // => [0, -1, -2, -3] | |
* | |
* _.range(1, 5); | |
* // => [1, 2, 3, 4] | |
* | |
* _.range(0, 20, 5); | |
* // => [0, 5, 10, 15] | |
* | |
* _.range(0, -4, -1); | |
* // => [0, -1, -2, -3] | |
* | |
* _.range(1, 4, 0); | |
* // => [1, 1, 1] | |
* | |
* _.range(0); | |
* // => [] | |
*/ | |
var range = createRange(); | |
/** | |
* A method that returns a new empty array. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.13.0 | |
* @category Util | |
* @returns {Array} Returns the new empty array. | |
* @example | |
* | |
* var arrays = _.times(2, _.stubArray); | |
* | |
* console.log(arrays); | |
* // => [[], []] | |
* | |
* console.log(arrays[0] === arrays[1]); | |
* // => false | |
*/ | |
function stubArray() { | |
return []; | |
} | |
/** | |
* A method that returns `false`. | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.13.0 | |
* @category Util | |
* @returns {boolean} Returns `false`. | |
* @example | |
* | |
* _.times(2, _.stubFalse); | |
* // => [false, false] | |
*/ | |
function stubFalse() { | |
return false; | |
} | |
/** | |
* Invokes the iteratee `n` times, returning an array of the results of | |
* each invocation. The iteratee is invoked with one argument; (index). | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @param {number} n The number of times to invoke `iteratee`. | |
* @param {Function} [iteratee=_.identity] The function invoked per iteration. | |
* @returns {Array} Returns the array of results. | |
* @example | |
* | |
* _.times(3, String); | |
* // => ['0', '1', '2'] | |
* | |
* _.times(4, _.constant(0)); | |
* // => [0, 0, 0, 0] | |
*/ | |
function times(n, iteratee) { | |
n = toInteger(n); | |
if (n < 1 || n > MAX_SAFE_INTEGER) { | |
return []; | |
} | |
var index = MAX_ARRAY_LENGTH, | |
length = nativeMin(n, MAX_ARRAY_LENGTH); | |
iteratee = baseIteratee(iteratee); | |
n -= MAX_ARRAY_LENGTH; | |
var result = baseTimes(length, iteratee); | |
while (++index < n) { | |
iteratee(index); | |
} | |
return result; | |
} | |
/** | |
* Generates a unique ID. If `prefix` is given, the ID is appended to it. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Util | |
* @param {string} [prefix=''] The value to prefix the ID with. | |
* @returns {string} Returns the unique ID. | |
* @example | |
* | |
* _.uniqueId('contact_'); | |
* // => 'contact_104' | |
* | |
* _.uniqueId(); | |
* // => '105' | |
*/ | |
function uniqueId(prefix) { | |
var id = ++idCounter; | |
return toString(prefix) + id; | |
} | |
/*------------------------------------------------------------------------*/ | |
/** | |
* Adds two numbers. | |
* | |
* @static | |
* @memberOf _ | |
* @since 3.4.0 | |
* @category Math | |
* @param {number} augend The first number in an addition. | |
* @param {number} addend The second number in an addition. | |
* @returns {number} Returns the total. | |
* @example | |
* | |
* _.add(6, 4); | |
* // => 10 | |
*/ | |
var add = createMathOperation(function(augend, addend) { | |
return augend + addend; | |
}); | |
/** | |
* Computes the maximum value of `array`. If `array` is empty or falsey, | |
* `undefined` is returned. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Math | |
* @param {Array} array The array to iterate over. | |
* @returns {*} Returns the maximum value. | |
* @example | |
* | |
* _.max([4, 2, 8, 6]); | |
* // => 8 | |
* | |
* _.max([]); | |
* // => undefined | |
*/ | |
function max(array) { | |
return (array && array.length) | |
? baseExtremum(array, identity, baseGt) | |
: undefined; | |
} | |
/** | |
* This method is like `_.max` except that it accepts `iteratee` which is | |
* invoked for each element in `array` to generate the criterion by which | |
* the value is ranked. The iteratee is invoked with one argument: (value). | |
* | |
* @static | |
* @memberOf _ | |
* @since 4.0.0 | |
* @category Math | |
* @param {Array} array The array to iterate over. | |
* @param {Array|Function|Object|string} [iteratee=_.identity] | |
* The iteratee invoked per element. | |
* @returns {*} Returns the maximum value. | |
* @example | |
* | |
* var objects = [{ 'n': 1 }, { 'n': 2 }]; | |
* | |
* _.maxBy(objects, function(o) { return o.n; }); | |
* // => { 'n': 2 } | |
* | |
* // The `_.property` iteratee shorthand. | |
* _.maxBy(objects, 'n'); | |
* // => { 'n': 2 } | |
*/ | |
function maxBy(array, iteratee) { | |
return (array && array.length) | |
? baseExtremum(array, baseIteratee(iteratee), baseGt) | |
: undefined; | |
} | |
/** | |
* Computes the minimum value of `array`. If `array` is empty or falsey, | |
* `undefined` is returned. | |
* | |
* @static | |
* @since 0.1.0 | |
* @memberOf _ | |
* @category Math | |
* @param {Array} array The array to iterate over. | |
* @returns {*} Returns the minimum value. | |
* @example | |
* | |
* _.min([4, 2, 8, 6]); | |
* // => 2 | |
* | |
* _.min([]); | |
* // => undefined | |
*/ | |
function min(array) { | |
return (array && array.length) | |
? baseExtremum(array, identity, baseLt) | |
: undefined; | |
} | |
/*------------------------------------------------------------------------*/ | |
// Add methods that return wrapped values in chain sequences. | |
lodash.assign = assign; | |
lodash.assignIn = assignIn; | |
lodash.before = before; | |
lodash.bind = bind; | |
lodash.chain = chain; | |
lodash.chunk = chunk; | |
lodash.compact = compact; | |
lodash.concat = concat; | |
lodash.create = create; | |
lodash.defaults = defaults; | |
lodash.defer = defer; | |
lodash.delay = delay; | |
lodash.filter = filter; | |
lodash.flatten = flatten; | |
lodash.flattenDeep = flattenDeep; | |
lodash.intersection = intersection; | |
lodash.iteratee = iteratee; | |
lodash.keys = keys; | |
lodash.map = map; | |
lodash.matches = matches; | |
lodash.merge = merge; | |
lodash.method = method; | |
lodash.mixin = mixin; | |
lodash.negate = negate; | |
lodash.once = once; | |
lodash.pick = pick; | |
lodash.pickBy = pickBy; | |
lodash.property = property; | |
lodash.range = range; | |
lodash.remove = remove; | |
lodash.slice = slice; | |
lodash.sortBy = sortBy; | |
lodash.split = split; | |
lodash.tap = tap; | |
lodash.thru = thru; | |
lodash.toArray = toArray; | |
lodash.uniq = uniq; | |
lodash.values = values; | |
lodash.words = words; | |
// Add aliases. | |
lodash.extend = assignIn; | |
// Add methods to `lodash.prototype`. | |
mixin(lodash, lodash); | |
/*------------------------------------------------------------------------*/ | |
// Add methods that return unwrapped values in chain sequences. | |
lodash.add = add; | |
lodash.camelCase = camelCase; | |
lodash.capitalize = capitalize; | |
lodash.clone = clone; | |
lodash.cloneDeep = cloneDeep; | |
lodash.deburr = deburr; | |
lodash.endsWith = endsWith; | |
lodash.escape = escape; | |
lodash.escapeRegExp = escapeRegExp; | |
lodash.every = every; | |
lodash.find = find; | |
lodash.findIndex = findIndex; | |
lodash.forEach = forEach; | |
lodash.forIn = forIn; | |
lodash.has = has; | |
lodash.head = head; | |
lodash.identity = identity; | |
lodash.indexOf = indexOf; | |
lodash.isArguments = isArguments; | |
lodash.isArray = isArray; | |
lodash.isBoolean = isBoolean; | |
lodash.isDate = isDate; | |
lodash.isEmpty = isEmpty; | |
lodash.isEqual = isEqual; | |
lodash.isError = isError; | |
lodash.isFinite = isFinite; | |
lodash.isFunction = isFunction; | |
lodash.isNaN = isNaN; | |
lodash.isNil = isNil; | |
lodash.isNull = isNull; | |
lodash.isNumber = isNumber; | |
lodash.isObject = isObject; | |
lodash.isPlainObject = isPlainObject; | |
lodash.isRegExp = isRegExp; | |
lodash.isString = isString; | |
lodash.isUndefined = isUndefined; | |
lodash.kebabCase = kebabCase; | |
lodash.last = last; | |
lodash.lowerCase = lowerCase; | |
lodash.lowerFirst = lowerFirst; | |
lodash.max = max; | |
lodash.maxBy = maxBy; | |
lodash.min = min; | |
lodash.noConflict = noConflict; | |
lodash.noop = noop; | |
lodash.pad = pad; | |
lodash.padEnd = padEnd; | |
lodash.padStart = padStart; | |
lodash.parseInt = parseInt; | |
lodash.reduce = reduce; | |
lodash.repeat = repeat; | |
lodash.replace = replace; | |
lodash.result = result; | |
lodash.size = size; | |
lodash.snakeCase = snakeCase; | |
lodash.some = some; | |
lodash.startCase = startCase; | |
lodash.startsWith = startsWith; | |
lodash.template = template; | |
lodash.times = times; | |
lodash.toLower = toLower; | |
lodash.toUpper = toUpper; | |
lodash.trim = trim; | |
lodash.trimEnd = trimEnd; | |
lodash.trimStart = trimStart; | |
lodash.truncate = truncate; | |
lodash.unescape = unescape; | |
lodash.uniqueId = uniqueId; | |
lodash.upperCase = upperCase; | |
lodash.upperFirst = upperFirst; | |
// Add aliases. | |
lodash.each = forEach; | |
lodash.first = head; | |
mixin(lodash, (function() { | |
var source = {}; | |
baseForOwn(lodash, function(func, methodName) { | |
if (!hasOwnProperty.call(lodash.prototype, methodName)) { | |
source[methodName] = func; | |
} | |
}); | |
return source; | |
}()), { 'chain': false }); | |
/*------------------------------------------------------------------------*/ | |
/** | |
* The semantic version number. | |
* | |
* @static | |
* @memberOf _ | |
* @type {string} | |
*/ | |
lodash.VERSION = VERSION; | |
// Add `Array` methods to `lodash.prototype`. | |
baseEach(['pop', 'join', 'replace', 'reverse', 'split', 'push', 'shift', 'sort', 'splice', 'unshift'], function(methodName) { | |
var func = (/^(?:replace|split)$/.test(methodName) ? String.prototype : arrayProto)[methodName], | |
chainName = /^(?:push|sort|unshift)$/.test(methodName) ? 'tap' : 'thru', | |
retUnwrapped = /^(?:pop|join|replace|shift)$/.test(methodName); | |
lodash.prototype[methodName] = function() { | |
var args = arguments; | |
if (retUnwrapped && !this.__chain__) { | |
var value = this.value(); | |
return func.apply(isArray(value) ? value : [], args); | |
} | |
return this[chainName](function(value) { | |
return func.apply(isArray(value) ? value : [], args); | |
}); | |
}; | |
}); | |
// Add chain sequence methods to the `lodash` wrapper. | |
lodash.prototype.toJSON = lodash.prototype.valueOf = lodash.prototype.value = wrapperValue; | |
/*--------------------------------------------------------------------------*/ | |
// Expose Lodash on the free variable `window` or `self` when available so it's | |
// globally accessible, even when bundled with Browserify, Webpack, etc. This | |
// also prevents errors in cases where Lodash is loaded by a script tag in the | |
// presence of an AMD loader. See http://requirejs.org/docs/errors.html#mismatch | |
// for more details. Use `_.noConflict` to remove Lodash from the global object. | |
(freeSelf || {})._ = lodash; | |
// Some AMD build optimizers like r.js check for condition patterns like the following: | |
if (typeof define == 'function' && typeof define.amd == 'object' && define.amd) { | |
// Define as an anonymous module so, through path mapping, it can be | |
// referenced as the "underscore" module. | |
define(function() { | |
return lodash; | |
}); | |
} | |
// Check for `exports` after `define` in case a build optimizer adds an `exports` object. | |
else if (freeModule) { | |
// Export for Node.js. | |
(freeModule.exports = lodash)._ = lodash; | |
// Export for CommonJS support. | |
freeExports._ = lodash; | |
} | |
else { | |
// Export to the global object. | |
root._ = lodash; | |
} | |
}.call(this)); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment