| /** |
| * lodash (Custom Build) <https://lodash.com/> |
| * Build: `lodash modularize exports="npm" -o ./` |
| * 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 |
| */ |
| |
| /** 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_INTEGER = 1.7976931348623157e+308, |
| NAN = 0 / 0; |
| |
| /** `Object#toString` result references. */ |
| var regexpTag = '[object RegExp]', |
| symbolTag = '[object Symbol]'; |
| |
| /** Used to match leading and trailing whitespace. */ |
| var reTrim = /^\s+|\s+$/g; |
| |
| /** Used to match `RegExp` flags from their coerced string values. */ |
| var reFlags = /\w*$/; |
| |
| /** Used to detect bad signed hexadecimal string values. */ |
| var reIsBadHex = /^[-+]0x[0-9a-f]+$/i; |
| |
| /** Used to detect binary string values. */ |
| var reIsBinary = /^0b[01]+$/i; |
| |
| /** Used to detect octal string values. */ |
| var reIsOctal = /^0o[0-7]+$/i; |
| |
| /** Used to compose unicode character classes. */ |
| var rsAstralRange = '\\ud800-\\udfff', |
| rsComboMarksRange = '\\u0300-\\u036f\\ufe20-\\ufe23', |
| rsComboSymbolsRange = '\\u20d0-\\u20f0', |
| rsVarRange = '\\ufe0e\\ufe0f'; |
| |
| /** Used to compose unicode capture groups. */ |
| var rsAstral = '[' + rsAstralRange + ']', |
| rsCombo = '[' + rsComboMarksRange + rsComboSymbolsRange + ']', |
| rsFitz = '\\ud83c[\\udffb-\\udfff]', |
| rsModifier = '(?:' + rsCombo + '|' + rsFitz + ')', |
| rsNonAstral = '[^' + rsAstralRange + ']', |
| rsRegional = '(?:\\ud83c[\\udde6-\\uddff]){2}', |
| rsSurrPair = '[\\ud800-\\udbff][\\udc00-\\udfff]', |
| rsZWJ = '\\u200d'; |
| |
| /** Used to compose unicode regexes. */ |
| var reOptMod = rsModifier + '?', |
| rsOptVar = '[' + rsVarRange + ']?', |
| rsOptJoin = '(?:' + rsZWJ + '(?:' + [rsNonAstral, rsRegional, rsSurrPair].join('|') + ')' + rsOptVar + reOptMod + ')*', |
| rsSeq = rsOptVar + reOptMod + rsOptJoin, |
| rsSymbol = '(?:' + [rsNonAstral + rsCombo + '?', rsCombo, rsRegional, rsSurrPair, rsAstral].join('|') + ')'; |
| |
| /** Used to match [string symbols](https://mathiasbynens.be/notes/javascript-unicode). */ |
| var reUnicode = RegExp(rsFitz + '(?=' + rsFitz + ')|' + rsSymbol + rsSeq, '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 reHasUnicode = RegExp('[' + rsZWJ + rsAstralRange + rsComboMarksRange + rsComboSymbolsRange + rsVarRange + ']'); |
| |
| /** Built-in method references without a dependency on `root`. */ |
| var freeParseInt = parseInt; |
| |
| /** Detect free variable `global` from Node.js. */ |
| var freeGlobal = typeof global == 'object' && global && global.Object === Object && global; |
| |
| /** Detect free variable `self`. */ |
| var freeSelf = typeof self == 'object' && self && self.Object === Object && self; |
| |
| /** Used as a reference to the global object. */ |
| var root = freeGlobal || freeSelf || Function('return this')(); |
| |
| /** Detect free variable `exports`. */ |
| var freeExports = typeof exports == 'object' && exports && !exports.nodeType && exports; |
| |
| /** Detect free variable `module`. */ |
| var freeModule = freeExports && typeof module == 'object' && module && !module.nodeType && module; |
| |
| /** Detect the popular CommonJS extension `module.exports`. */ |
| var moduleExports = freeModule && freeModule.exports === freeExports; |
| |
| /** Detect free variable `process` from Node.js. */ |
| var freeProcess = moduleExports && freeGlobal.process; |
| |
| /** Used to access faster Node.js helpers. */ |
| var nodeUtil = (function() { |
| try { |
| return freeProcess && freeProcess.binding('util'); |
| } catch (e) {} |
| }()); |
| |
| /* Node.js helper references. */ |
| var nodeIsRegExp = nodeUtil && nodeUtil.isRegExp; |
| |
| /** |
| * Gets the size of an ASCII `string`. |
| * |
| * @private |
| * @param {string} string The string inspect. |
| * @returns {number} Returns the string size. |
| */ |
| var asciiSize = baseProperty('length'); |
| |
| /** |
| * Converts an ASCII `string` to an array. |
| * |
| * @private |
| * @param {string} string The string to convert. |
| * @returns {Array} Returns the converted array. |
| */ |
| function asciiToArray(string) { |
| return string.split(''); |
| } |
| |
| /** |
| * 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 `_.unary` without support for storing 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); |
| }; |
| } |
| |
| /** |
| * Checks if `string` contains Unicode symbols. |
| * |
| * @private |
| * @param {string} string The string to inspect. |
| * @returns {boolean} Returns `true` if a symbol is found, else `false`. |
| */ |
| function hasUnicode(string) { |
| return reHasUnicode.test(string); |
| } |
| |
| /** |
| * Gets the number of symbols in `string`. |
| * |
| * @private |
| * @param {string} string The string to inspect. |
| * @returns {number} Returns the string size. |
| */ |
| function stringSize(string) { |
| return hasUnicode(string) |
| ? unicodeSize(string) |
| : asciiSize(string); |
| } |
| |
| /** |
| * Converts `string` to an array. |
| * |
| * @private |
| * @param {string} string The string to convert. |
| * @returns {Array} Returns the converted array. |
| */ |
| function stringToArray(string) { |
| return hasUnicode(string) |
| ? unicodeToArray(string) |
| : asciiToArray(string); |
| } |
| |
| /** |
| * Gets the size of a Unicode `string`. |
| * |
| * @private |
| * @param {string} string The string inspect. |
| * @returns {number} Returns the string size. |
| */ |
| function unicodeSize(string) { |
| var result = reUnicode.lastIndex = 0; |
| while (reUnicode.test(string)) { |
| result++; |
| } |
| return result; |
| } |
| |
| /** |
| * Converts a Unicode `string` to an array. |
| * |
| * @private |
| * @param {string} string The string to convert. |
| * @returns {Array} Returns the converted array. |
| */ |
| function unicodeToArray(string) { |
| return string.match(reUnicode) || []; |
| } |
| |
| /** Used for built-in method references. */ |
| var objectProto = Object.prototype; |
| |
| /** |
| * Used to resolve the |
| * [`toStringTag`](http://ecma-international.org/ecma-262/7.0/#sec-object.prototype.tostring) |
| * of values. |
| */ |
| var objectToString = objectProto.toString; |
| |
| /** Built-in value references. */ |
| var Symbol = root.Symbol; |
| |
| /** Used to convert symbols to primitives and strings. */ |
| var symbolProto = Symbol ? Symbol.prototype : undefined, |
| symbolToString = symbolProto ? symbolProto.toString : undefined; |
| |
| /** |
| * The base implementation of `_.isRegExp` without Node.js optimizations. |
| * |
| * @private |
| * @param {*} value The value to check. |
| * @returns {boolean} Returns `true` if `value` is a regexp, else `false`. |
| */ |
| function baseIsRegExp(value) { |
| return isObject(value) && objectToString.call(value) == regexpTag; |
| } |
| |
| /** |
| * 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; |
| } |
| |
| /** |
| * 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 (isSymbol(value)) { |
| return symbolToString ? symbolToString.call(value) : ''; |
| } |
| var result = (value + ''); |
| return (result == '0' && (1 / value) == -INFINITY) ? '-0' : result; |
| } |
| |
| /** |
| * 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); |
| } |
| |
| /** |
| * Checks if `value` is the |
| * [language type](http://www.ecma-international.org/ecma-262/7.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 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 a regexp, else `false`. |
| * @example |
| * |
| * _.isRegExp(/abc/); |
| * // => true |
| * |
| * _.isRegExp('/abc/'); |
| * // => false |
| */ |
| var isRegExp = nodeIsRegExp ? baseUnary(nodeIsRegExp) : baseIsRegExp; |
| |
| /** |
| * Checks if `value` is classified as a `Symbol` primitive or object. |
| * |
| * @static |
| * @memberOf _ |
| * @since 4.0.0 |
| * @category Lang |
| * @param {*} value The value to check. |
| * @returns {boolean} Returns `true` if `value` is a symbol, else `false`. |
| * @example |
| * |
| * _.isSymbol(Symbol.iterator); |
| * // => true |
| * |
| * _.isSymbol('abc'); |
| * // => false |
| */ |
| function isSymbol(value) { |
| return typeof value == 'symbol' || |
| (isObjectLike(value) && objectToString.call(value) == symbolTag); |
| } |
| |
| /** |
| * Converts `value` to a finite number. |
| * |
| * @static |
| * @memberOf _ |
| * @since 4.12.0 |
| * @category Lang |
| * @param {*} value The value to convert. |
| * @returns {number} Returns the converted number. |
| * @example |
| * |
| * _.toFinite(3.2); |
| * // => 3.2 |
| * |
| * _.toFinite(Number.MIN_VALUE); |
| * // => 5e-324 |
| * |
| * _.toFinite(Infinity); |
| * // => 1.7976931348623157e+308 |
| * |
| * _.toFinite('3.2'); |
| * // => 3.2 |
| */ |
| function toFinite(value) { |
| if (!value) { |
| return value === 0 ? value : 0; |
| } |
| value = toNumber(value); |
| if (value === INFINITY || value === -INFINITY) { |
| var sign = (value < 0 ? -1 : 1); |
| return sign * MAX_INTEGER; |
| } |
| return value === value ? value : 0; |
| } |
| |
| /** |
| * Converts `value` to an integer. |
| * |
| * **Note:** This method is loosely based on |
| * [`ToInteger`](http://www.ecma-international.org/ecma-262/7.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 |
| */ |
| function toInteger(value) { |
| var result = toFinite(value), |
| remainder = result % 1; |
| |
| return result === result ? (remainder ? result - remainder : result) : 0; |
| } |
| |
| /** |
| * 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 |
| */ |
| function toNumber(value) { |
| if (typeof value == 'number') { |
| return value; |
| } |
| if (isSymbol(value)) { |
| return NAN; |
| } |
| if (isObject(value)) { |
| var other = typeof value.valueOf == 'function' ? value.valueOf() : value; |
| value = isObject(other) ? (other + '') : other; |
| } |
| if (typeof value != 'string') { |
| return value === 0 ? value : +value; |
| } |
| value = value.replace(reTrim, ''); |
| var isBinary = reIsBinary.test(value); |
| return (isBinary || reIsOctal.test(value)) |
| ? freeParseInt(value.slice(2), isBinary ? 2 : 8) |
| : (reIsBadHex.test(value) ? NAN : +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) { |
| return value == null ? '' : baseToString(value); |
| } |
| |
| /** |
| * 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 (hasUnicode(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; |
| } |
| |
| module.exports = truncate; |