| /* |
| * Copyright (C) 2007 Apple Inc. All rights reserved. |
| * Copyright (C) 2012 Google Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| /** |
| * @param {number} m |
| * @param {number} n |
| * @return {number} |
| */ |
| self.mod = function(m, n) { |
| return ((m % n) + n) % n; |
| }; |
| |
| /** |
| * @param {string} string |
| * @return {!Array.<number>} |
| */ |
| String.prototype.findAll = function(string) { |
| const matches = []; |
| let i = this.indexOf(string); |
| while (i !== -1) { |
| matches.push(i); |
| i = this.indexOf(string, i + string.length); |
| } |
| return matches; |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.reverse = function() { |
| return this.split('').reverse().join(''); |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.replaceControlCharacters = function() { |
| // Replace C0 and C1 control character sets with printable character. |
| // Do not replace '\t', \n' and '\r'. |
| return this.replace(/[\u0000-\u0008\u000b\u000c\u000e-\u001f\u0080-\u009f]/g, '�'); |
| }; |
| |
| /** |
| * @return {boolean} |
| */ |
| String.prototype.isWhitespace = function() { |
| return /^\s*$/.test(this); |
| }; |
| |
| /** |
| * @return {!Array.<number>} |
| */ |
| String.prototype.computeLineEndings = function() { |
| const endings = this.findAll('\n'); |
| endings.push(this.length); |
| return endings; |
| }; |
| |
| /** |
| * @param {string} chars |
| * @return {string} |
| */ |
| String.prototype.escapeCharacters = function(chars) { |
| let foundChar = false; |
| for (let i = 0; i < chars.length; ++i) { |
| if (this.indexOf(chars.charAt(i)) !== -1) { |
| foundChar = true; |
| break; |
| } |
| } |
| |
| if (!foundChar) { |
| return String(this); |
| } |
| |
| let result = ''; |
| for (let i = 0; i < this.length; ++i) { |
| if (chars.indexOf(this.charAt(i)) !== -1) { |
| result += '\\'; |
| } |
| result += this.charAt(i); |
| } |
| |
| return result; |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.regexSpecialCharacters = function() { |
| return '^[]{}()\\.^$*+?|-,'; |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.escapeForRegExp = function() { |
| return this.escapeCharacters(String.regexSpecialCharacters()); |
| }; |
| |
| /** |
| * @param {string} query |
| * @return {!RegExp} |
| */ |
| String.filterRegex = function(query) { |
| const toEscape = String.regexSpecialCharacters(); |
| let regexString = ''; |
| for (let i = 0; i < query.length; ++i) { |
| let c = query.charAt(i); |
| if (toEscape.indexOf(c) !== -1) { |
| c = '\\' + c; |
| } |
| if (i) { |
| regexString += '[^\\0' + c + ']*'; |
| } |
| regexString += c; |
| } |
| return new RegExp(regexString, 'i'); |
| }; |
| |
| /** |
| * @param {string} text |
| * @return {string} |
| */ |
| String.escapeInvalidUnicodeCharacters = function(text) { |
| if (!String._invalidCharactersRegExp) { |
| // Escape orphan surrogates and invalid characters. |
| let invalidCharacters = ''; |
| for (let i = 0xfffe; i <= 0x10ffff; i += 0x10000) { |
| invalidCharacters += String.fromCodePoint(i, i + 1); |
| } |
| String._invalidCharactersRegExp = new RegExp(`[${invalidCharacters}\uD800-\uDFFF\uFDD0-\uFDEF]`, 'gu'); |
| } |
| let result = ''; |
| let lastPos = 0; |
| while (true) { |
| const match = String._invalidCharactersRegExp.exec(text); |
| if (!match) { |
| break; |
| } |
| result += text.substring(lastPos, match.index) + '\\u' + text.charCodeAt(match.index).toString(16); |
| if (match.index + 1 < String._invalidCharactersRegExp.lastIndex) { |
| result += '\\u' + text.charCodeAt(match.index + 1).toString(16); |
| } |
| lastPos = String._invalidCharactersRegExp.lastIndex; |
| } |
| return result + text.substring(lastPos); |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.escapeHTML = function() { |
| return this.replace(/&/g, '&') |
| .replace(/</g, '<') |
| .replace(/>/g, '>') |
| .replace(/"/g, '"'); // " doublequotes just for editor |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.unescapeHTML = function() { |
| return this.replace(/</g, '<') |
| .replace(/>/g, '>') |
| .replace(/:/g, ':') |
| .replace(/"/g, '"') |
| .replace(/</g, '<') |
| .replace(/>/g, '>') |
| .replace(/&/g, '&'); |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.collapseWhitespace = function() { |
| return this.replace(/[\s\xA0]+/g, ' '); |
| }; |
| |
| /** |
| * @param {number} maxLength |
| * @return {string} |
| */ |
| String.prototype.trimMiddle = function(maxLength) { |
| if (this.length <= maxLength) { |
| return String(this); |
| } |
| let leftHalf = maxLength >> 1; |
| let rightHalf = maxLength - leftHalf - 1; |
| if (this.codePointAt(this.length - rightHalf - 1) >= 0x10000) { |
| --rightHalf; |
| ++leftHalf; |
| } |
| if (leftHalf > 0 && this.codePointAt(leftHalf - 1) >= 0x10000) { |
| --leftHalf; |
| } |
| return this.substr(0, leftHalf) + '\u2026' + this.substr(this.length - rightHalf, rightHalf); |
| }; |
| |
| /** |
| * @param {number} maxLength |
| * @return {string} |
| */ |
| String.prototype.trimEndWithMaxLength = function(maxLength) { |
| if (this.length <= maxLength) { |
| return String(this); |
| } |
| return this.substr(0, maxLength - 1) + '\u2026'; |
| }; |
| |
| /** |
| * @param {?string=} baseURLDomain |
| * @return {string} |
| */ |
| String.prototype.trimURL = function(baseURLDomain) { |
| let result = this.replace(/^(https|http|file):\/\//i, ''); |
| if (baseURLDomain) { |
| if (result.toLowerCase().startsWith(baseURLDomain.toLowerCase())) { |
| result = result.substr(baseURLDomain.length); |
| } |
| } |
| return result; |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.toTitleCase = function() { |
| return this.substring(0, 1).toUpperCase() + this.substring(1); |
| }; |
| |
| /** |
| * @param {string} other |
| * @return {number} |
| */ |
| String.prototype.compareTo = function(other) { |
| if (this > other) { |
| return 1; |
| } |
| if (this < other) { |
| return -1; |
| } |
| return 0; |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.removeURLFragment = function() { |
| let fragmentIndex = this.indexOf('#'); |
| if (fragmentIndex === -1) { |
| fragmentIndex = this.length; |
| } |
| return this.substring(0, fragmentIndex); |
| }; |
| |
| /** |
| * @param {string|undefined} string |
| * @return {number} |
| */ |
| String.hashCode = function(string) { |
| if (!string) { |
| return 0; |
| } |
| // Hash algorithm for substrings is described in "Über die Komplexität der Multiplikation in |
| // eingeschränkten Branchingprogrammmodellen" by Woelfe. |
| // http://opendatastructures.org/versions/edition-0.1d/ods-java/node33.html#SECTION00832000000000000000 |
| const p = ((1 << 30) * 4 - 5); // prime: 2^32 - 5 |
| const z = 0x5033d967; // 32 bits from random.org |
| const z2 = 0x59d2f15d; // random odd 32 bit number |
| let s = 0; |
| let zi = 1; |
| for (let i = 0; i < string.length; i++) { |
| const xi = string.charCodeAt(i) * z2; |
| s = (s + zi * xi) % p; |
| zi = (zi * z) % p; |
| } |
| s = (s + zi * (p - 1)) % p; |
| return Math.abs(s | 0); |
| }; |
| |
| /** |
| * @param {string} string |
| * @param {number} index |
| * @return {boolean} |
| */ |
| String.isDigitAt = function(string, index) { |
| const c = string.charCodeAt(index); |
| return (48 <= c && c <= 57); |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| String.prototype.toBase64 = function() { |
| /** |
| * @param {number} b |
| * @return {number} |
| */ |
| function encodeBits(b) { |
| return b < 26 ? b + 65 : b < 52 ? b + 71 : b < 62 ? b - 4 : b === 62 ? 43 : b === 63 ? 47 : 65; |
| } |
| const encoder = new TextEncoder(); |
| const data = encoder.encode(this.toString()); |
| const n = data.length; |
| let encoded = ''; |
| if (n === 0) { |
| return encoded; |
| } |
| let shift; |
| let v = 0; |
| for (let i = 0; i < n; i++) { |
| shift = i % 3; |
| v |= data[i] << (16 >>> shift & 24); |
| if (shift === 2) { |
| encoded += String.fromCharCode( |
| encodeBits(v >>> 18 & 63), encodeBits(v >>> 12 & 63), encodeBits(v >>> 6 & 63), encodeBits(v & 63)); |
| v = 0; |
| } |
| } |
| if (shift === 0) { |
| encoded += String.fromCharCode(encodeBits(v >>> 18 & 63), encodeBits(v >>> 12 & 63), 61, 61); |
| } else if (shift === 1) { |
| encoded += String.fromCharCode(encodeBits(v >>> 18 & 63), encodeBits(v >>> 12 & 63), encodeBits(v >>> 6 & 63), 61); |
| } |
| return encoded; |
| }; |
| |
| /** |
| * @param {string} a |
| * @param {string} b |
| * @return {number} |
| */ |
| String.naturalOrderComparator = function(a, b) { |
| const chunk = /^\d+|^\D+/; |
| let chunka, chunkb, anum, bnum; |
| while (1) { |
| if (a) { |
| if (!b) { |
| return 1; |
| } |
| } else { |
| if (b) { |
| return -1; |
| } else { |
| return 0; |
| } |
| } |
| chunka = a.match(chunk)[0]; |
| chunkb = b.match(chunk)[0]; |
| anum = !isNaN(chunka); |
| bnum = !isNaN(chunkb); |
| if (anum && !bnum) { |
| return -1; |
| } |
| if (bnum && !anum) { |
| return 1; |
| } |
| if (anum && bnum) { |
| const diff = chunka - chunkb; |
| if (diff) { |
| return diff; |
| } |
| if (chunka.length !== chunkb.length) { |
| if (!+chunka && !+chunkb) // chunks are strings of all 0s (special case) |
| { |
| return chunka.length - chunkb.length; |
| } else { |
| return chunkb.length - chunka.length; |
| } |
| } |
| } else if (chunka !== chunkb) { |
| return (chunka < chunkb) ? -1 : 1; |
| } |
| a = a.substring(chunka.length); |
| b = b.substring(chunkb.length); |
| } |
| }; |
| |
| /** |
| * @param {string} a |
| * @param {string} b |
| * @return {number} |
| */ |
| String.caseInsensetiveComparator = function(a, b) { |
| a = a.toUpperCase(); |
| b = b.toUpperCase(); |
| if (a === b) { |
| return 0; |
| } |
| return a > b ? 1 : -1; |
| }; |
| |
| /** |
| * @param {number} num |
| * @param {number} min |
| * @param {number} max |
| * @return {number} |
| */ |
| Number.constrain = function(num, min, max) { |
| if (num < min) { |
| num = min; |
| } else if (num > max) { |
| num = max; |
| } |
| return num; |
| }; |
| |
| /** |
| * @param {number} a |
| * @param {number} b |
| * @return {number} |
| */ |
| Number.gcd = function(a, b) { |
| if (b === 0) { |
| return a; |
| } else { |
| return Number.gcd(b, a % b); |
| } |
| }; |
| |
| /** |
| * @param {string} value |
| * @return {string} |
| */ |
| Number.toFixedIfFloating = function(value) { |
| if (!value || isNaN(value)) { |
| return value; |
| } |
| const number = Number(value); |
| return number % 1 ? number.toFixed(3) : String(number); |
| }; |
| |
| /** |
| * @return {boolean} |
| */ |
| Date.prototype.isValid = function() { |
| return !isNaN(this.getTime()); |
| }; |
| |
| /** |
| * @return {string} |
| */ |
| Date.prototype.toISO8601Compact = function() { |
| /** |
| * @param {number} x |
| * @return {string} |
| */ |
| function leadZero(x) { |
| return (x > 9 ? '' : '0') + x; |
| } |
| return this.getFullYear() + leadZero(this.getMonth() + 1) + leadZero(this.getDate()) + 'T' + |
| leadZero(this.getHours()) + leadZero(this.getMinutes()) + leadZero(this.getSeconds()); |
| }; |
| |
| Object.defineProperty(Array.prototype, 'remove', { |
| /** |
| * @param {!T} value |
| * @param {boolean=} firstOnly |
| * @return {boolean} |
| * @this {Array.<!T>} |
| * @template T |
| */ |
| value: function(value, firstOnly) { |
| let index = this.indexOf(value); |
| if (index === -1) { |
| return false; |
| } |
| if (firstOnly) { |
| this.splice(index, 1); |
| return true; |
| } |
| for (let i = index + 1, n = this.length; i < n; ++i) { |
| if (this[i] !== value) { |
| this[index++] = this[i]; |
| } |
| } |
| this.length = index; |
| return true; |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'pushAll', { |
| /** |
| * @param {!Array<!T>} array |
| * @this {Array<!T>} |
| * @template T |
| */ |
| value: function(array) { |
| for (let i = 0; i < array.length; ++i) { |
| this.push(array[i]); |
| } |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'rotate', { |
| /** |
| * @param {number} index |
| * @return {!Array.<!T>} |
| * @this {Array.<!T>} |
| * @template T |
| */ |
| value: function(index) { |
| const result = []; |
| for (let i = index; i < index + this.length; ++i) { |
| result.push(this[i % this.length]); |
| } |
| return result; |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'sortNumbers', { |
| /** |
| * @this {Array.<number>} |
| */ |
| value: function() { |
| /** |
| * @param {number} a |
| * @param {number} b |
| * @return {number} |
| */ |
| function numericComparator(a, b) { |
| return a - b; |
| } |
| |
| this.sort(numericComparator); |
| } |
| }); |
| |
| (function() { |
| const partition = { |
| /** |
| * @this {Array.<number>} |
| * @param {function(number, number): number} comparator |
| * @param {number} left |
| * @param {number} right |
| * @param {number} pivotIndex |
| */ |
| value: function(comparator, left, right, pivotIndex) { |
| function swap(array, i1, i2) { |
| const temp = array[i1]; |
| array[i1] = array[i2]; |
| array[i2] = temp; |
| } |
| |
| const pivotValue = this[pivotIndex]; |
| swap(this, right, pivotIndex); |
| let storeIndex = left; |
| for (let i = left; i < right; ++i) { |
| if (comparator(this[i], pivotValue) < 0) { |
| swap(this, storeIndex, i); |
| ++storeIndex; |
| } |
| } |
| swap(this, right, storeIndex); |
| return storeIndex; |
| } |
| }; |
| Object.defineProperty(Array.prototype, 'partition', partition); |
| Object.defineProperty(Uint32Array.prototype, 'partition', partition); |
| |
| const sortRange = { |
| /** |
| * @param {function(number, number): number} comparator |
| * @param {number} leftBound |
| * @param {number} rightBound |
| * @param {number} sortWindowLeft |
| * @param {number} sortWindowRight |
| * @return {!Array.<number>} |
| * @this {Array.<number>} |
| */ |
| value: function(comparator, leftBound, rightBound, sortWindowLeft, sortWindowRight) { |
| function quickSortRange(array, comparator, left, right, sortWindowLeft, sortWindowRight) { |
| if (right <= left) { |
| return; |
| } |
| const pivotIndex = Math.floor(Math.random() * (right - left)) + left; |
| const pivotNewIndex = array.partition(comparator, left, right, pivotIndex); |
| if (sortWindowLeft < pivotNewIndex) { |
| quickSortRange(array, comparator, left, pivotNewIndex - 1, sortWindowLeft, sortWindowRight); |
| } |
| if (pivotNewIndex < sortWindowRight) { |
| quickSortRange(array, comparator, pivotNewIndex + 1, right, sortWindowLeft, sortWindowRight); |
| } |
| } |
| if (leftBound === 0 && rightBound === (this.length - 1) && sortWindowLeft === 0 && sortWindowRight >= rightBound) { |
| this.sort(comparator); |
| } else { |
| quickSortRange(this, comparator, leftBound, rightBound, sortWindowLeft, sortWindowRight); |
| } |
| return this; |
| } |
| }; |
| Object.defineProperty(Array.prototype, 'sortRange', sortRange); |
| Object.defineProperty(Uint32Array.prototype, 'sortRange', sortRange); |
| })(); |
| |
| Object.defineProperty(Array.prototype, 'lowerBound', { |
| /** |
| * Return index of the leftmost element that is equal or greater |
| * than the specimen object. If there's no such element (i.e. all |
| * elements are smaller than the specimen) returns right bound. |
| * The function works for sorted array. |
| * When specified, |left| (inclusive) and |right| (exclusive) indices |
| * define the search window. |
| * |
| * @param {!T} object |
| * @param {function(!T,!S):number=} comparator |
| * @param {number=} left |
| * @param {number=} right |
| * @return {number} |
| * @this {Array.<!S>} |
| * @template T,S |
| */ |
| value: function(object, comparator, left, right) { |
| function defaultComparator(a, b) { |
| return a < b ? -1 : (a > b ? 1 : 0); |
| } |
| comparator = comparator || defaultComparator; |
| let l = left || 0; |
| let r = right !== undefined ? right : this.length; |
| while (l < r) { |
| const m = (l + r) >> 1; |
| if (comparator(object, this[m]) > 0) { |
| l = m + 1; |
| } else { |
| r = m; |
| } |
| } |
| return r; |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'upperBound', { |
| /** |
| * Return index of the leftmost element that is greater |
| * than the specimen object. If there's no such element (i.e. all |
| * elements are smaller or equal to the specimen) returns right bound. |
| * The function works for sorted array. |
| * When specified, |left| (inclusive) and |right| (exclusive) indices |
| * define the search window. |
| * |
| * @param {!T} object |
| * @param {function(!T,!S):number=} comparator |
| * @param {number=} left |
| * @param {number=} right |
| * @return {number} |
| * @this {Array.<!S>} |
| * @template T,S |
| */ |
| value: function(object, comparator, left, right) { |
| function defaultComparator(a, b) { |
| return a < b ? -1 : (a > b ? 1 : 0); |
| } |
| comparator = comparator || defaultComparator; |
| let l = left || 0; |
| let r = right !== undefined ? right : this.length; |
| while (l < r) { |
| const m = (l + r) >> 1; |
| if (comparator(object, this[m]) >= 0) { |
| l = m + 1; |
| } else { |
| r = m; |
| } |
| } |
| return r; |
| } |
| }); |
| |
| Object.defineProperty(Uint32Array.prototype, 'lowerBound', {value: Array.prototype.lowerBound}); |
| |
| Object.defineProperty(Uint32Array.prototype, 'upperBound', {value: Array.prototype.upperBound}); |
| |
| Object.defineProperty(Int32Array.prototype, 'lowerBound', {value: Array.prototype.lowerBound}); |
| |
| Object.defineProperty(Int32Array.prototype, 'upperBound', {value: Array.prototype.upperBound}); |
| |
| Object.defineProperty(Float64Array.prototype, 'lowerBound', {value: Array.prototype.lowerBound}); |
| |
| Object.defineProperty(Array.prototype, 'binaryIndexOf', { |
| /** |
| * @param {!T} value |
| * @param {function(!T,!S):number} comparator |
| * @return {number} |
| * @this {Array.<!S>} |
| * @template T,S |
| */ |
| value: function(value, comparator) { |
| const index = this.lowerBound(value, comparator); |
| return index < this.length && comparator(value, this[index]) === 0 ? index : -1; |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'select', { |
| /** |
| * @param {string} field |
| * @return {!Array.<!T>} |
| * @this {Array.<!Object.<string,!T>>} |
| * @template T |
| */ |
| value: function(field) { |
| const result = new Array(this.length); |
| for (let i = 0; i < this.length; ++i) { |
| result[i] = this[i][field]; |
| } |
| return result; |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'peekLast', { |
| /** |
| * @return {!T|undefined} |
| * @this {Array.<!T>} |
| * @template T |
| */ |
| value: function() { |
| return this[this.length - 1]; |
| } |
| }); |
| |
| (function() { |
| /** |
| * @param {!Array.<T>} array1 |
| * @param {!Array.<T>} array2 |
| * @param {function(T,T):number} comparator |
| * @param {boolean} mergeNotIntersect |
| * @return {!Array.<T>} |
| * @template T |
| */ |
| function mergeOrIntersect(array1, array2, comparator, mergeNotIntersect) { |
| const result = []; |
| let i = 0; |
| let j = 0; |
| while (i < array1.length && j < array2.length) { |
| const compareValue = comparator(array1[i], array2[j]); |
| if (mergeNotIntersect || !compareValue) { |
| result.push(compareValue <= 0 ? array1[i] : array2[j]); |
| } |
| if (compareValue <= 0) { |
| i++; |
| } |
| if (compareValue >= 0) { |
| j++; |
| } |
| } |
| if (mergeNotIntersect) { |
| while (i < array1.length) { |
| result.push(array1[i++]); |
| } |
| while (j < array2.length) { |
| result.push(array2[j++]); |
| } |
| } |
| return result; |
| } |
| |
| Object.defineProperty(Array.prototype, 'intersectOrdered', { |
| /** |
| * @param {!Array.<T>} array |
| * @param {function(T,T):number} comparator |
| * @return {!Array.<T>} |
| * @this {!Array.<T>} |
| * @template T |
| */ |
| value: function(array, comparator) { |
| return mergeOrIntersect(this, array, comparator, false); |
| } |
| }); |
| |
| Object.defineProperty(Array.prototype, 'mergeOrdered', { |
| /** |
| * @param {!Array.<T>} array |
| * @param {function(T,T):number} comparator |
| * @return {!Array.<T>} |
| * @this {!Array.<T>} |
| * @template T |
| */ |
| value: function(array, comparator) { |
| return mergeOrIntersect(this, array, comparator, true); |
| } |
| }); |
| })(); |
| |
| /** |
| * @param {string} format |
| * @param {...*} var_arg |
| * @return {string} |
| */ |
| String.sprintf = function(format, var_arg) { |
| return String.vsprintf(format, Array.prototype.slice.call(arguments, 1)); |
| }; |
| |
| /** |
| * @param {string} format |
| * @param {!Object.<string, function(string, ...):*>} formatters |
| * @return {!Array.<!Object>} |
| */ |
| String.tokenizeFormatString = function(format, formatters) { |
| const tokens = []; |
| |
| function addStringToken(str) { |
| if (!str) { |
| return; |
| } |
| if (tokens.length && tokens[tokens.length - 1].type === 'string') { |
| tokens[tokens.length - 1].value += str; |
| } else { |
| tokens.push({type: 'string', value: str}); |
| } |
| } |
| |
| function addSpecifierToken(specifier, precision, substitutionIndex) { |
| tokens.push({type: 'specifier', specifier: specifier, precision: precision, substitutionIndex: substitutionIndex}); |
| } |
| |
| function addAnsiColor(code) { |
| const types = {3: 'color', 9: 'colorLight', 4: 'bgColor', 10: 'bgColorLight'}; |
| const colorCodes = ['black', 'red', 'green', 'yellow', 'blue', 'magenta', 'cyan', 'lightGray', '', 'default']; |
| const colorCodesLight = |
| ['darkGray', 'lightRed', 'lightGreen', 'lightYellow', 'lightBlue', 'lightMagenta', 'lightCyan', 'white', '']; |
| const colors = {color: colorCodes, colorLight: colorCodesLight, bgColor: colorCodes, bgColorLight: colorCodesLight}; |
| const type = types[Math.floor(code / 10)]; |
| if (!type) { |
| return; |
| } |
| const color = colors[type][code % 10]; |
| if (!color) { |
| return; |
| } |
| tokens.push({ |
| type: 'specifier', |
| specifier: 'c', |
| value: {description: (type.startsWith('bg') ? 'background : ' : 'color: ') + color} |
| }); |
| } |
| |
| let textStart = 0; |
| let substitutionIndex = 0; |
| const re = |
| new RegExp(`%%|%(?:(\\d+)\\$)?(?:\\.(\\d*))?([${Object.keys(formatters).join('')}])|\\u001b\\[(\\d+)m`, 'g'); |
| for (let match = re.exec(format); !!match; match = re.exec(format)) { |
| const matchStart = match.index; |
| if (matchStart > textStart) { |
| addStringToken(format.substring(textStart, matchStart)); |
| } |
| |
| if (match[0] === '%%') { |
| addStringToken('%'); |
| } else if (match[0].startsWith('%')) { |
| // eslint-disable-next-line no-unused-vars |
| const [_, substitionString, precisionString, specifierString] = match; |
| if (substitionString && Number(substitionString) > 0) { |
| substitutionIndex = Number(substitionString) - 1; |
| } |
| const precision = precisionString ? Number(precisionString) : -1; |
| addSpecifierToken(specifierString, precision, substitutionIndex); |
| ++substitutionIndex; |
| } else { |
| const code = Number(match[4]); |
| addAnsiColor(code); |
| } |
| textStart = matchStart + match[0].length; |
| } |
| addStringToken(format.substring(textStart)); |
| return tokens; |
| }; |
| |
| String.standardFormatters = { |
| /** |
| * @return {number} |
| */ |
| d: function(substitution) { |
| return !isNaN(substitution) ? substitution : 0; |
| }, |
| |
| /** |
| * @return {number} |
| */ |
| f: function(substitution, token) { |
| if (substitution && token.precision > -1) { |
| substitution = substitution.toFixed(token.precision); |
| } |
| return !isNaN(substitution) ? substitution : (token.precision > -1 ? Number(0).toFixed(token.precision) : 0); |
| }, |
| |
| /** |
| * @return {string} |
| */ |
| s: function(substitution) { |
| return substitution; |
| } |
| }; |
| |
| /** |
| * @param {string} format |
| * @param {!Array.<*>} substitutions |
| * @return {string} |
| */ |
| String.vsprintf = function(format, substitutions) { |
| return String |
| .format( |
| format, substitutions, String.standardFormatters, '', |
| function(a, b) { |
| return a + b; |
| }) |
| .formattedResult; |
| }; |
| |
| /** |
| * @param {string} format |
| * @param {?ArrayLike} substitutions |
| * @param {!Object.<string, function(string, ...):Q>} formatters |
| * @param {!T} initialValue |
| * @param {function(T, Q): T|undefined} append |
| * @param {!Array.<!Object>=} tokenizedFormat |
| * @return {!{formattedResult: T, unusedSubstitutions: ?ArrayLike}}; |
| * @template T, Q |
| */ |
| String.format = function(format, substitutions, formatters, initialValue, append, tokenizedFormat) { |
| if (!format || ((!substitutions || !substitutions.length) && format.search(/\u001b\[(\d+)m/) === -1)) { |
| return {formattedResult: append(initialValue, format), unusedSubstitutions: substitutions}; |
| } |
| |
| function prettyFunctionName() { |
| return 'String.format("' + format + '", "' + Array.prototype.join.call(substitutions, '", "') + '")'; |
| } |
| |
| function warn(msg) { |
| console.warn(prettyFunctionName() + ': ' + msg); |
| } |
| |
| function error(msg) { |
| console.error(prettyFunctionName() + ': ' + msg); |
| } |
| |
| let result = initialValue; |
| const tokens = tokenizedFormat || String.tokenizeFormatString(format, formatters); |
| const usedSubstitutionIndexes = {}; |
| |
| for (let i = 0; i < tokens.length; ++i) { |
| const token = tokens[i]; |
| |
| if (token.type === 'string') { |
| result = append(result, token.value); |
| continue; |
| } |
| |
| if (token.type !== 'specifier') { |
| error('Unknown token type "' + token.type + '" found.'); |
| continue; |
| } |
| |
| if (!token.value && token.substitutionIndex >= substitutions.length) { |
| // If there are not enough substitutions for the current substitutionIndex |
| // just output the format specifier literally and move on. |
| error( |
| 'not enough substitution arguments. Had ' + substitutions.length + ' but needed ' + |
| (token.substitutionIndex + 1) + ', so substitution was skipped.'); |
| result = append(result, '%' + (token.precision > -1 ? token.precision : '') + token.specifier); |
| continue; |
| } |
| |
| if (!token.value) { |
| usedSubstitutionIndexes[token.substitutionIndex] = true; |
| } |
| |
| if (!(token.specifier in formatters)) { |
| // Encountered an unsupported format character, treat as a string. |
| warn('unsupported format character \u201C' + token.specifier + '\u201D. Treating as a string.'); |
| result = append(result, token.value ? '' : substitutions[token.substitutionIndex]); |
| continue; |
| } |
| |
| result = append(result, formatters[token.specifier](token.value || substitutions[token.substitutionIndex], token)); |
| } |
| |
| const unusedSubstitutions = []; |
| for (let i = 0; i < substitutions.length; ++i) { |
| if (i in usedSubstitutionIndexes) { |
| continue; |
| } |
| unusedSubstitutions.push(substitutions[i]); |
| } |
| |
| return {formattedResult: result, unusedSubstitutions: unusedSubstitutions}; |
| }; |
| |
| /** |
| * @param {string} query |
| * @param {boolean} caseSensitive |
| * @param {boolean} isRegex |
| * @return {!RegExp} |
| */ |
| self.createSearchRegex = function(query, caseSensitive, isRegex) { |
| const regexFlags = caseSensitive ? 'g' : 'gi'; |
| let regexObject; |
| |
| if (isRegex) { |
| try { |
| regexObject = new RegExp(query, regexFlags); |
| } catch (e) { |
| // Silent catch. |
| } |
| } |
| |
| if (!regexObject) { |
| regexObject = self.createPlainTextSearchRegex(query, regexFlags); |
| } |
| |
| return regexObject; |
| }; |
| |
| /** |
| * @param {string} query |
| * @param {string=} flags |
| * @return {!RegExp} |
| */ |
| self.createPlainTextSearchRegex = function(query, flags) { |
| // This should be kept the same as the one in StringUtil.cpp. |
| const regexSpecialCharacters = String.regexSpecialCharacters(); |
| let regex = ''; |
| for (let i = 0; i < query.length; ++i) { |
| const c = query.charAt(i); |
| if (regexSpecialCharacters.indexOf(c) !== -1) { |
| regex += '\\'; |
| } |
| regex += c; |
| } |
| return new RegExp(regex, flags || ''); |
| }; |
| |
| /** |
| * @param {!RegExp} regex |
| * @param {string} content |
| * @return {number} |
| */ |
| self.countRegexMatches = function(regex, content) { |
| let text = content; |
| let result = 0; |
| let match; |
| while (text && (match = regex.exec(text))) { |
| if (match[0].length > 0) { |
| ++result; |
| } |
| text = text.substring(match.index + 1); |
| } |
| return result; |
| }; |
| |
| /** |
| * @param {number} spacesCount |
| * @return {string} |
| */ |
| self.spacesPadding = function(spacesCount) { |
| return '\xA0'.repeat(spacesCount); |
| }; |
| |
| /** |
| * @param {number} value |
| * @param {number} symbolsCount |
| * @return {string} |
| */ |
| self.numberToStringWithSpacesPadding = function(value, symbolsCount) { |
| const numberString = value.toString(); |
| const paddingLength = Math.max(0, symbolsCount - numberString.length); |
| return self.spacesPadding(paddingLength) + numberString; |
| }; |
| |
| /** |
| * @return {!Array.<T>} |
| * @template T |
| */ |
| Set.prototype.valuesArray = function() { |
| return Array.from(this.values()); |
| }; |
| |
| /** |
| * @return {?T} |
| * @template T |
| */ |
| Set.prototype.firstValue = function() { |
| if (!this.size) { |
| return null; |
| } |
| return this.values().next().value; |
| }; |
| |
| /** |
| * @param {!Iterable<T>|!Array<!T>} iterable |
| * @template T |
| */ |
| Set.prototype.addAll = function(iterable) { |
| for (const e of iterable) { |
| this.add(e); |
| } |
| }; |
| |
| /** |
| * @param {!Iterable<T>|!Array<!T>} iterable |
| * @return {boolean} |
| * @template T |
| */ |
| Set.prototype.containsAll = function(iterable) { |
| for (const e of iterable) { |
| if (!this.has(e)) { |
| return false; |
| } |
| } |
| return true; |
| }; |
| |
| /** |
| * @return {T} |
| * @template T |
| */ |
| Map.prototype.remove = function(key) { |
| const value = this.get(key); |
| this.delete(key); |
| return value; |
| }; |
| |
| /** |
| * @return {!Array<!VALUE>} |
| */ |
| Map.prototype.valuesArray = function() { |
| return Array.from(this.values()); |
| }; |
| |
| /** |
| * @return {!Array<!KEY>} |
| */ |
| Map.prototype.keysArray = function() { |
| return Array.from(this.keys()); |
| }; |
| |
| /** |
| * @return {!Platform.Multimap<!KEY, !VALUE>} |
| */ |
| Map.prototype.inverse = function() { |
| const result = new Platform.Multimap(); |
| for (const key of this.keys()) { |
| const value = this.get(key); |
| result.set(value, key); |
| } |
| return result; |
| }; |
| |
| /** |
| * @template K, V |
| */ |
| const Multimap = class { |
| constructor() { |
| /** @type {!Map.<K, !Set.<!V>>} */ |
| this._map = new Map(); |
| } |
| |
| /** |
| * @param {K} key |
| * @param {V} value |
| */ |
| set(key, value) { |
| let set = this._map.get(key); |
| if (!set) { |
| set = new Set(); |
| this._map.set(key, set); |
| } |
| set.add(value); |
| } |
| |
| /** |
| * @param {K} key |
| * @return {!Set<!V>} |
| */ |
| get(key) { |
| return this._map.get(key) || new Set(); |
| } |
| |
| /** |
| * @param {K} key |
| * @return {boolean} |
| */ |
| has(key) { |
| return this._map.has(key); |
| } |
| |
| /** |
| * @param {K} key |
| * @param {V} value |
| * @return {boolean} |
| */ |
| hasValue(key, value) { |
| const set = this._map.get(key); |
| if (!set) { |
| return false; |
| } |
| return set.has(value); |
| } |
| |
| /** |
| * @return {number} |
| */ |
| get size() { |
| return this._map.size; |
| } |
| |
| /** |
| * @param {K} key |
| * @param {V} value |
| * @return {boolean} |
| */ |
| delete(key, value) { |
| const values = this.get(key); |
| if (!values) { |
| return false; |
| } |
| const result = values.delete(value); |
| if (!values.size) { |
| this._map.delete(key); |
| } |
| return result; |
| } |
| |
| /** |
| * @param {K} key |
| */ |
| deleteAll(key) { |
| this._map.delete(key); |
| } |
| |
| /** |
| * @return {!Array.<K>} |
| */ |
| keysArray() { |
| return this._map.keysArray(); |
| } |
| |
| /** |
| * @return {!Array.<!V>} |
| */ |
| valuesArray() { |
| const result = []; |
| const keys = this.keysArray(); |
| for (let i = 0; i < keys.length; ++i) { |
| result.pushAll(this.get(keys[i]).valuesArray()); |
| } |
| return result; |
| } |
| |
| clear() { |
| this._map.clear(); |
| } |
| }; |
| |
| /** |
| * @param {string} url |
| * @return {!Promise.<string>} |
| */ |
| self.loadXHR = function(url) { |
| return new Promise(load); |
| |
| function load(successCallback, failureCallback) { |
| function onReadyStateChanged() { |
| if (xhr.readyState !== XMLHttpRequest.DONE) { |
| return; |
| } |
| if (xhr.status !== 200) { |
| xhr.onreadystatechange = null; |
| failureCallback(new Error(xhr.status)); |
| return; |
| } |
| xhr.onreadystatechange = null; |
| successCallback(xhr.responseText); |
| } |
| |
| const xhr = new XMLHttpRequest(); |
| xhr.withCredentials = false; |
| xhr.open('GET', url, true); |
| xhr.onreadystatechange = onReadyStateChanged; |
| xhr.send(null); |
| } |
| }; |
| |
| /** |
| * @param {*} value |
| */ |
| self.suppressUnused = function(value) {}; |
| |
| /** |
| * @param {function()} callback |
| * @return {number} |
| */ |
| self.setImmediate = function(callback) { |
| const args = [...arguments].slice(1); |
| Promise.resolve().then(() => callback(...args)); |
| return 0; |
| }; |
| |
| /** |
| * @param {function(...?)} callback |
| * @return {!Promise.<T>} |
| * @template T |
| */ |
| Promise.prototype.spread = function(callback) { |
| return this.then(spreadPromise); |
| |
| function spreadPromise(arg) { |
| return callback.apply(null, arg); |
| } |
| }; |
| |
| /** |
| * @param {T} defaultValue |
| * @return {!Promise.<T>} |
| * @template T |
| */ |
| Promise.prototype.catchException = function(defaultValue) { |
| return this.catch(function(error) { |
| console.error(error); |
| return defaultValue; |
| }); |
| }; |
| |
| /** |
| * @param {!Map<number, ?>} other |
| * @param {function(!VALUE,?):boolean} isEqual |
| * @return {!{removed: !Array<!VALUE>, added: !Array<?>, equal: !Array<!VALUE>}} |
| * @this {Map<number, VALUE>} |
| */ |
| Map.prototype.diff = function(other, isEqual) { |
| const leftKeys = this.keysArray(); |
| const rightKeys = other.keysArray(); |
| leftKeys.sort((a, b) => a - b); |
| rightKeys.sort((a, b) => a - b); |
| |
| const removed = []; |
| const added = []; |
| const equal = []; |
| let leftIndex = 0; |
| let rightIndex = 0; |
| while (leftIndex < leftKeys.length && rightIndex < rightKeys.length) { |
| const leftKey = leftKeys[leftIndex]; |
| const rightKey = rightKeys[rightIndex]; |
| if (leftKey === rightKey && isEqual(this.get(leftKey), other.get(rightKey))) { |
| equal.push(this.get(leftKey)); |
| ++leftIndex; |
| ++rightIndex; |
| continue; |
| } |
| if (leftKey <= rightKey) { |
| removed.push(this.get(leftKey)); |
| ++leftIndex; |
| continue; |
| } |
| added.push(other.get(rightKey)); |
| ++rightIndex; |
| } |
| while (leftIndex < leftKeys.length) { |
| const leftKey = leftKeys[leftIndex++]; |
| removed.push(this.get(leftKey)); |
| } |
| while (rightIndex < rightKeys.length) { |
| const rightKey = rightKeys[rightIndex++]; |
| added.push(other.get(rightKey)); |
| } |
| return {added: added, removed: removed, equal: equal}; |
| }; |
| |
| /** |
| * TODO: move into its own module |
| * @param {function()} callback |
| */ |
| self.runOnWindowLoad = function(callback) { |
| /** |
| * @suppressGlobalPropertiesCheck |
| */ |
| function windowLoaded() { |
| self.removeEventListener('DOMContentLoaded', windowLoaded, false); |
| callback(); |
| } |
| |
| if (document.readyState === 'complete' || document.readyState === 'interactive') { |
| callback(); |
| } else { |
| self.addEventListener('DOMContentLoaded', windowLoaded, false); |
| } |
| }; |
| |
| const _singletonSymbol = Symbol('singleton'); |
| |
| /** |
| * @template T |
| * @param {function(new:T, ...)} constructorFunction |
| * @return {!T} |
| */ |
| self.singleton = function(constructorFunction) { |
| if (_singletonSymbol in constructorFunction) { |
| return constructorFunction[_singletonSymbol]; |
| } |
| const instance = new constructorFunction(); |
| constructorFunction[_singletonSymbol] = instance; |
| return instance; |
| }; |
| |
| /** |
| * @param {?string} content |
| * @return {number} |
| */ |
| self.base64ToSize = function(content) { |
| if (!content) { |
| return 0; |
| } |
| let size = content.length * 3 / 4; |
| if (content[content.length - 1] === '=') { |
| size--; |
| } |
| if (content.length > 1 && content[content.length - 2] === '=') { |
| size--; |
| } |
| return size; |
| }; |
| |
| self.Platform = self.Platform || {}; |
| Platform = Platform || {}; |
| |
| /** @constructor */ |
| Platform.Multimap = Multimap; |