- // ==UserScript==
-
- // #region Info
-
- // @namespace https://gf.qytechs.cn/en/users/1123632-93akkord
- // @exclude *
- // @author Michael Barros (https://gf.qytechs.cn/en/users/1123632-93akkord)
- // @icon 
-
- // #endregion Info
-
- // ==UserLibrary==
-
- // @name akkd-common
- // @description Common functions
- // @copyright 2022+, Michael Barros (https://gf.qytechs.cn/en/users/1123632-93akkord)
- // @license CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode
- // @license GPL-3.0-or-later; https://www.gnu.org/licenses/gpl-3.0.txt
- // @version 0.0.20
-
- // ==/UserScript==
-
- // ==/UserLibrary==
-
- // ==OpenUserJS==
-
- // @author 93Akkord
-
- // ==/OpenUserJS==
-
- /// <reference path='C:/Users/mbarros/Documents/DevProjects/Web/Tampermonkey/Palantir/@types/__fullReferencePaths__.js' />
-
- /*
-
- # akkd-common
-
- A collection of commonly used classes and functions.
-
- ## Required requires:
-
- - https://code.jquery.com/jquery-3.2.1.min.js
- - https://gf.qytechs.cn/scripts/474546-loglevel/code/loglevel.js
-
- */
-
- // #region Events
-
- // Setup location change events
- /**
- *
- * Example usage:
- * ```javascript
- * window.addEventListener('popstate', () => {
- * window.dispatchEvent(new Event('locationchange'));
- * });
- */
- (() => {
- class LocationChangeEvent extends Event {
- constructor(type, prevUrl, newUrl) {
- super(type);
-
- this.prevUrl = prevUrl;
- this.newUrl = newUrl;
- }
- }
-
- let prevUrl = document.location.href;
- let oldPushState = history.pushState;
-
- history.pushState = function pushState() {
- let ret = oldPushState.apply(this, arguments);
- let newUrl = document.location.href;
-
- window.dispatchEvent(new LocationChangeEvent('pushstate', prevUrl, newUrl));
- window.dispatchEvent(new LocationChangeEvent('locationchange', prevUrl, newUrl));
-
- prevUrl = newUrl;
-
- return ret;
- };
-
- let oldReplaceState = history.replaceState;
-
- history.replaceState = function replaceState() {
- let ret = oldReplaceState.apply(this, arguments);
- let newUrl = document.location.href;
-
- window.dispatchEvent(new LocationChangeEvent('replacestate', prevUrl, newUrl));
- window.dispatchEvent(new LocationChangeEvent('locationchange', prevUrl, newUrl));
-
- prevUrl = newUrl;
-
- return ret;
- };
-
- window.addEventListener('popstate', () => {
- let newUrl = document.location.href;
-
- window.dispatchEvent(new LocationChangeEvent('locationchange', prevUrl, newUrl));
-
- prevUrl = newUrl;
- });
- })();
-
- // #endregion Events
-
- // #region Helper Classes
-
- class Logger {
- /**
- * Creates an instance of Logger.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Window} _window
- * @param {string | null} devTag
- * @memberof Logger
- */
- constructor(_window = null, devTag = null) {
- /**
- * @type {Window}
- * @private
- */
- this.window = _window || getWindow();
-
- /** @type {string | null} */
- this.devTag = devTag;
-
- /** @type {string[]} */
- this._additionalTags = [];
- }
-
- /**
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @type {string[]}
- * @public
- * @memberof Logger
- */
- get additionalTags() {
- return this._additionalTags;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string[]} value
- * @memberof Logger
- */
- set additionalTags(value) {
- if (getType(value) != 'array') {
- value = [value];
- }
-
- this._additionalTags = value;
- }
-
- /** @type {string} */
- get label() {
- return [].concat([this.devTag], this.additionalTags).join(' ');
- }
-
- /** @type {(...data: any[]) => void} */
- get log() {
- if (this.devTag) {
- return console.log.bind(console, `${this.label}`);
- } else {
- return console.log.bind(console);
- }
- }
-
- /** @type {(...data: any[]) => void} */
- get info() {
- if (this.devTag) {
- return console.info.bind(console, `${this.label}`);
- } else {
- return console.info.bind(console);
- }
- }
-
- /** @type {(...data: any[]) => void} */
- get error() {
- if (this.devTag) {
- return console.error.bind(console, `${this.label}`);
- } else {
- return console.error.bind(console);
- }
- }
-
- /** @type {(...data: any[]) => void} */
- get debug() {
- if (this.devTag) {
- return console.debug.bind(console, `${this.label}`);
- } else {
- return console.debug.bind(console);
- }
- }
-
- /** @type {(...data: any[]) => void} */
- get warn() {
- if (this.devTag) {
- return console.warn.bind(console, `${this.label}`);
- } else {
- return console.warn.bind(console);
- }
- }
-
- /**
- * Maybe use later?
- *
- * @memberof Logger
- */
- _setupFunctions() {
- let self = this;
- let funcs = ['log', 'info', 'error', 'debug', 'warn'];
-
- for (let i = 0; i < funcs.length; i++) {
- let func = funcs[i];
-
- self[func] = function () {
- let args = [...arguments];
-
- if (self.devTag) args.unshift(self.label);
-
- self.window.console.debug.bind(self.window.console, ...args);
- };
- }
- }
- }
-
- class Base64 {
- static keyStr = 'ABCDEFGHIJKLMNOP' + 'QRSTUVWXYZabcdef' + 'ghijklmnopqrstuv' + 'wxyz0123456789+/' + '=';
-
- /**
- *
- *
- * @static
- * @param {string} input
- * @returns {string}
- * @memberof Base64
- */
- static encode(input) {
- input = escape(input);
-
- let output = '';
-
- let chr1;
- let chr2;
- let chr3 = '';
-
- let enc1;
- let enc2;
- let enc3;
- let enc4 = '';
-
- let i = 0;
-
- do {
- chr1 = input.charCodeAt(i++);
- chr2 = input.charCodeAt(i++);
- chr3 = input.charCodeAt(i++);
- enc1 = chr1 >> 2;
- enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
- enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
- enc4 = chr3 & 63;
-
- if (isNaN(chr2)) {
- enc3 = enc4 = 64;
- } else if (isNaN(chr3)) {
- enc4 = 64;
- }
-
- output = output + Base64.keyStr.charAt(enc1) + Base64.keyStr.charAt(enc2) + Base64.keyStr.charAt(enc3) + Base64.keyStr.charAt(enc4);
- chr1 = chr2 = chr3 = '';
- enc1 = enc2 = enc3 = enc4 = '';
- } while (i < input.length);
-
- return output;
- }
-
- /**
- *
- *
- * @static
- * @param {string} input
- * @returns {string}
- * @memberof Base64
- */
- static decode(input) {
- let output = '';
-
- let chr1;
- let chr2;
- let chr3 = '';
-
- let enc1;
- let enc2;
- let enc3;
- let enc4 = '';
-
- let i = 0;
-
- let base64test = /[^A-Za-z0-9\+\/\=]/g;
-
- if (base64test.exec(input)) {
- throw new Error(`There were invalid base64 characters in the input text. Valid base64 characters are: ['A-Z', 'a-z', '0-9,' '+', '/', '=']`);
- }
-
- input = input.replace(/[^A-Za-z0-9\+\/\=]/g, '');
-
- do {
- enc1 = Base64.keyStr.indexOf(input.charAt(i++));
- enc2 = Base64.keyStr.indexOf(input.charAt(i++));
- enc3 = Base64.keyStr.indexOf(input.charAt(i++));
- enc4 = Base64.keyStr.indexOf(input.charAt(i++));
- chr1 = (enc1 << 2) | (enc2 >> 4);
- chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
- chr3 = ((enc3 & 3) << 6) | enc4;
- output = output + String.fromCharCode(chr1);
-
- if (enc3 != 64) output = output + String.fromCharCode(chr2);
-
- if (enc4 != 64) output = output + String.fromCharCode(chr3);
-
- chr1 = chr2 = chr3 = '';
- enc1 = enc2 = enc3 = enc4 = '';
- } while (i < input.length);
-
- return unescape(output);
- }
- }
-
- class MultiRegExp {
- constructor(baseRegExp) {
- const { regexp, groupIndexMapper, previousGroupsForGroup } = this._fillGroups(baseRegExp);
-
- this.regexp = regexp;
- this.groupIndexMapper = groupIndexMapper;
- this.previousGroupsForGroup = previousGroupsForGroup;
- }
-
- execForAllGroups(str, includeFullMatch) {
- let matches = RegExp.prototype.exec.call(this.regexp, str);
-
- if (!matches) return matches;
-
- let firstIndex = matches.index;
- let indexMapper = includeFullMatch
- ? Object.assign(
- {
- 0: 0,
- },
- this.groupIndexMapper
- )
- : this.groupIndexMapper;
- let previousGroups = includeFullMatch
- ? Object.assign(
- {
- 0: [],
- },
- this.previousGroupsForGroup
- )
- : this.previousGroupsForGroup;
-
- let res = Object.keys(indexMapper).map((group) => {
- let mapped = indexMapper[group];
- let match = matches[mapped];
- let start = firstIndex + previousGroups[group].reduce((sum, i) => sum + (matches[i] ? matches[i].length : 0), 0);
- let end = start + (matches[mapped] ? matches[mapped].length : 0);
- let lineColumnStart = LineColumnFinder(str).fromIndex(start);
- let lineColumnEnd = LineColumnFinder(str).fromIndex(end - 1);
-
- return {
- match,
- start,
- end,
- startLineNumber: lineColumnStart.line,
- startColumnNumber: lineColumnStart.col,
- endLineNumber: lineColumnEnd.line,
- endColumnNumber: lineColumnEnd.col,
- };
- });
-
- return res;
- }
-
- execForGroup(string, group) {
- const matches = RegExp.prototype.exec.call(this.regexp, string);
-
- if (!matches) return matches;
-
- const firstIndex = matches.index;
-
- const mapped = group == 0 ? 0 : this.groupIndexMapper[group];
- const previousGroups = group == 0 ? [] : this.previousGroupsForGroup[group];
-
- let r = {
- match: matches[mapped],
- start: firstIndex + previousGroups.reduce((sum, i) => sum + (matches[i] ? matches[i].length : 0), 0),
- };
-
- r.end = r.start + (matches[mapped] ? matches[mapped].length : 0);
-
- return r;
- }
-
- /**
- * Adds brackets before and after a part of string
- * @param str string the hole regex string
- * @param start int marks the position where ( should be inserted
- * @param end int marks the position where ) should be inserted
- * @param groupsAdded int defines the offset to the original string because of inserted brackets
- * @return {string}
- */
- _addGroupToRegexString(str, start, end, groupsAdded) {
- start += groupsAdded * 2;
- end += groupsAdded * 2;
-
- return str.substring(0, start) + '(' + str.substring(start, end + 1) + ')' + str.substring(end + 1);
- }
-
- /**
- * converts the given regex to a regex where all not captured string are going to be captured
- * it along sides generates a mapper which maps the original group index to the shifted group offset and
- * generates a list of groups indexes (including new generated capturing groups)
- * which have been closed before a given group index (unshifted)
- *
- * Example:
- * regexp: /a(?: )bc(def(ghi)xyz)/g => /(a(?: )bc)((def)(ghi)(xyz))/g
- * groupIndexMapper: {'1': 2, '2', 4}
- * previousGroupsForGroup: {'1': [1], '2': [1, 3]}
- *
- * @param regex RegExp
- * @return {{regexp: RegExp, groupIndexMapper: {}, previousGroupsForGroup: {}}}
- */
- _fillGroups(regex) {
- let regexString;
- let modifier;
-
- if (regex.source && regex.flags) {
- regexString = regex.source;
- modifier = regex.flags;
- } else {
- regexString = regex.toString();
- modifier = regexString.substring(regexString.lastIndexOf(regexString[0]) + 1); // sometimes order matters ;)
- regexString = regexString.substr(1, regex.toString().lastIndexOf(regexString[0]) - 1);
- }
-
- // regexp is greedy so it should match (? before ( right?
- // brackets may be not quoted by \
- // closing bracket may look like: ), )+, )+?, ){1,}?, ){1,1111}?
- const tester = /(\\\()|(\\\))|(\(\?)|(\()|(\)(?:\{\d+,?\d*}|[*+?])?\??)/g;
-
- let modifiedRegex = regexString;
-
- let lastGroupStartPosition = -1;
- let lastGroupEndPosition = -1;
- let lastNonGroupStartPosition = -1;
- let lastNonGroupEndPosition = -1;
- let groupsAdded = 0;
- let groupCount = 0;
- let matchArr;
- const nonGroupPositions = [];
- const groupPositions = [];
- const groupNumber = [];
- let currentLengthIndexes = [];
- const groupIndexMapper = {};
- const previousGroupsForGroup = {};
-
- while ((matchArr = tester.exec(regexString)) !== null) {
- if (matchArr[1] || matchArr[2]) {
- // ignore escaped brackets \(, \)
- }
-
- if (matchArr[3]) {
- // non capturing group (?
- let index = matchArr.index + matchArr[0].length - 1;
-
- lastNonGroupStartPosition = index;
- nonGroupPositions.push(index);
- } else if (matchArr[4]) {
- // capturing group (
- let index = matchArr.index + matchArr[0].length - 1;
- let lastGroupPosition = Math.max(lastGroupStartPosition, lastGroupEndPosition);
-
- // if a (? is found add ) before it
- if (lastNonGroupStartPosition > lastGroupPosition) {
- // check if between ) of capturing group lies a non capturing group
- if (lastGroupPosition < lastNonGroupEndPosition) {
- // add groups for x1 and x2 on (?:()x1)x2(?:...
- if (lastNonGroupEndPosition - 1 - (lastGroupPosition + 1) > 0) {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastGroupPosition + 1, lastNonGroupEndPosition - 1, groupsAdded);
- groupsAdded++;
- lastGroupEndPosition = lastNonGroupEndPosition - 1; // imaginary position as it is not in regex but modifiedRegex
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
-
- if (lastNonGroupStartPosition - 1 - (lastNonGroupEndPosition + 1) > 0) {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastNonGroupEndPosition + 1, lastNonGroupStartPosition - 2, groupsAdded);
- groupsAdded++;
- lastGroupEndPosition = lastNonGroupStartPosition - 1; // imaginary position as it is not in regex but modifiedRegex
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
- } else {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastGroupPosition + 1, lastNonGroupStartPosition - 2, groupsAdded);
- groupsAdded++;
- lastGroupEndPosition = lastNonGroupStartPosition - 1; // imaginary position as it is not in regex but modifiedRegex
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
-
- // if necessary also add group between (? and opening bracket
- if (index > lastNonGroupStartPosition + 2) {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastNonGroupStartPosition + 2, index - 1, groupsAdded);
- groupsAdded++;
- lastGroupEndPosition = index - 1; // imaginary position as it is not in regex but modifiedRegex
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
- } else if (lastGroupPosition < index - 1) {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastGroupPosition + 1, index - 1, groupsAdded);
- groupsAdded++;
- lastGroupEndPosition = index - 1; // imaginary position as it is not in regex but modifiedRegex
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
-
- groupCount++;
- lastGroupStartPosition = index;
- groupPositions.push(index);
- groupNumber.push(groupCount + groupsAdded);
- groupIndexMapper[groupCount] = groupCount + groupsAdded;
- previousGroupsForGroup[groupCount] = currentLengthIndexes.slice();
- } else if (matchArr[5]) {
- // closing bracket ), )+, )+?, ){1,}?, ){1,1111}?
- let index = matchArr.index + matchArr[0].length - 1;
-
- if ((groupPositions.length && !nonGroupPositions.length) || groupPositions[groupPositions.length - 1] > nonGroupPositions[nonGroupPositions.length - 1]) {
- if (lastGroupStartPosition < lastGroupEndPosition && lastGroupEndPosition < index - 1) {
- modifiedRegex = this._addGroupToRegexString(modifiedRegex, lastGroupEndPosition + 1, index - 1, groupsAdded);
- groupsAdded++;
-
- //lastGroupEndPosition = index - 1; will be set anyway
- currentLengthIndexes.push(groupCount + groupsAdded);
- }
-
- groupPositions.pop();
- lastGroupEndPosition = index;
-
- let toPush = groupNumber.pop();
- currentLengthIndexes.push(toPush);
- currentLengthIndexes = currentLengthIndexes.filter((index) => index <= toPush);
- } else if (nonGroupPositions.length) {
- nonGroupPositions.pop();
- lastNonGroupEndPosition = index;
- }
- }
- }
-
- return {
- regexp: new RegExp(modifiedRegex, modifier),
- groupIndexMapper,
- previousGroupsForGroup,
- };
- }
- }
-
- class MoveableElement {
- /**
- * Creates an instance of MoveableElement.
- * @param {HTMLElement} element
- * @param {boolean} requireKeyDown
- * @memberof MoveableElement
- */
- constructor(element, requireKeyDown) {
- this.element = element;
- this.requireKeyDown = requireKeyDown || false;
- this.handleMouseDown = this.handleMouseDown.bind(this);
- this.handleMouseUp = this.handleMouseUp.bind(this);
- this.handleMouseMove = this.handleMouseMove.bind(this);
-
- this.moving = false;
- this.keyPressed = false;
- this.originalCursor = getStyle(this.element, 'cursor');
-
- this.setupEvents();
- }
-
- setupEvents() {
- if (!document.body) {
- setTimeout(() => {
- this.setupEvents();
- }, 250);
- } else {
- document.body.addEventListener('keydown', (ev) => {
- if (ev.which == '17') {
- this.keyPressed = true;
- }
- });
-
- document.body.addEventListener('keyup', (ev) => {
- this.keyPressed = false;
- });
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {MouseEvent} ev
- * @memberof MoveableElement
- */
- handleMouseDown(ev) {
- if (this.keyPressed || !this.requireKeyDown) {
- ev.preventDefault();
-
- this.element.style.cursor = 'move';
-
- this.changePointerEvents('none');
-
- document.body.removeEventListener('mouseup', this.handleMouseUp);
- document.body.addEventListener('mouseup', this.handleMouseUp);
-
- document.body.removeEventListener('mousemove', this.handleMouseMove);
- document.body.removeEventListener('mouseleave', this.handleMouseUp);
-
- document.body.addEventListener('mousemove', this.handleMouseMove);
- document.body.addEventListener('mouseleave', this.handleMouseUp);
-
- try {
- document.querySelectorAll('iframe')[0].style.pointerEvents = 'none';
- } catch (error) {}
- }
- }
-
- changePointerEvents(value) {
- for (let i = 0; i < this.element.children.length; i++) {
- const child = this.element.children[i];
-
- child.style.pointerEvents = value;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {MouseEvent} ev
- * @memberof MoveableElement
- */
- handleMouseUp(ev) {
- this.moving = false;
- this.element.style.cursor = this.originalCursor;
- this.changePointerEvents('auto');
-
- document.body.removeEventListener('mouseup', this.handleMouseUp);
- document.body.removeEventListener('mousemove', this.handleMouseMove);
- document.body.removeEventListener('mouseleave', this.handleMouseUp);
-
- try {
- document.querySelectorAll('iframe')[0].style.pointerEvents = '';
- } catch (error) {}
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {MouseEvent} ev
- * @memberof MoveableElement
- */
- handleMouseMove(ev) {
- this.moving = true;
-
- let top = ev.clientY - getStyle(this.element, 'height') / 2;
- let bottom = ev.clientX - getStyle(this.element, 'width') / 2;
-
- this.element.style.top = `${top}px`;
- this.element.style.left = `${bottom}px`;
- }
-
- padCoord(coord) {
- return coord.toString().padStart(5, ' ');
- }
-
- init() {
- this.element.addEventListener('mousedown', this.handleMouseDown);
- }
- }
-
- class CurrentLine {
- /**
- * @typedef ILineInfo
- * @prop {string} method
- * @prop {number} line
- * @prop {string} file
- * @prop {string} filename
- */
-
- /**
- * Returns a single item
- *
- * @param {number} [level] Useful to return levels up on the stack. If not informed, the first (0, zero index) element of the stack will be returned
- * @returns {ILineInfo}
- */
- get(level = 0) {
- const stack = getStack();
- const i = Math.min(level + 1, stack.length - 1);
- const item = stack[i];
- const result = CurrentLine.parse(item);
-
- return result;
- }
-
- /**
- * Returns all stack
- *
- * @returns {ILineInfo[]}
- */
- all() {
- const stack = getStack();
- const result = [];
-
- for (let i = 1; i < stack.length; i++) {
- const item = stack[i];
-
- result.push(CurrentLine.parse(item));
- }
-
- return result;
- }
-
- /**
- *
- *
- * @param {NodeJS.CallSite} item
- * @returns {ILineInfo}
- */
- static parse(item) {
- const result = {
- method: item.getMethodName() || item.getFunctionName(),
- line: item.getLineNumber(),
- file: item.getFileName() || item.getScriptNameOrSourceURL(),
- };
-
- result.filename = result.file ? result.file.replace(/^.*\/|\\/gm, '').replace(/\.\w+$/gm, '') : null;
-
- return result;
- }
- }
-
- /**
- *
- *
- * @returns {NodeJS.CallSite[]}
- */
- function getStack() {
- const orig = Error.prepareStackTrace;
-
- Error.prepareStackTrace = function (_, stack) {
- return stack;
- };
-
- const err = new Error();
-
- Error.captureStackTrace(err, arguments.callee);
-
- const stack = err.stack;
-
- Error.prepareStackTrace = orig;
-
- return stack;
- }
-
- class ProgressTimer {
- /**
- * Creates an instance of ProgressTimer.
- * @param {number} total
- * @memberof ProgressTimer
- */
- constructor(total) {
- this.startTime;
- this.total = total;
- this.loaded = 0;
- this.estimatedFinishDt = '';
- this.progressMessage = '';
- }
-
- /**
- *
- *
- * @memberof ProgressTimer
- */
- start() {
- this.startTime = new Date();
- }
-
- /**
- *
- *
- * @param {number} loaded
- * @param {string} msg
- * @memberof ProgressTimer
- */
- updateProgress(loaded, msg) {
- this.loaded = loaded;
-
- this.progress = `${((this.loaded * 100) / this.total).toFixed(2)}%`;
- this.timeRemaining = this._estimatedTimeRemaining(this.startTime, this.loaded, this.total);
- this.downloaded = `${this.loaded}/${this.total}`;
- this.completionTime = `${this._dateToISOLikeButLocal(this.estimatedFinishDt)}`;
- this.totalRuntime = `${this._ms2Timestamp(this.timeTaken)}`;
-
- this.updateProgressMessage(msg);
- this.printProgress();
- }
-
- /**
- *
- *
- * @param {string} msg
- * @memberof ProgressTimer
- */
- updateProgressMessage(msg) {
- let msgLines = [];
-
- msgLines.push(` completed: ${this.progress}`);
- msgLines.push(` downloaded: ${this.downloaded}`);
- msgLines.push(` total runtime: ${this.totalRuntime}`);
- msgLines.push(` time remaining: ${this.timeRemaining}`);
- msgLines.push(`completion time: ${this.completionTime}`);
-
- if (msg) {
- msgLines.push(msg);
- }
-
- this.progressMessage = msgLines.join('\n');
- }
-
- /**
- *
- *
- * @memberof ProgressTimer
- */
- printProgress() {
- console.clear();
- console.debug(this.progressMessage);
- }
-
- /**
- *
- *
- * @param {Date} startTime
- * @param {number} itemsProcessed
- * @param {number} totalItems
- * @returns {string}
- * @memberof ProgressTimer
- */
- _estimatedTimeRemaining(startTime, itemsProcessed, totalItems) {
- // if (itemsProcessed == 0) {
- // return '';
- // }
-
- let currentTime = new Date();
- this.timeTaken = currentTime - startTime;
- this.timeLeft = itemsProcessed == 0 ? this.timeTaken * (totalItems - itemsProcessed) : (this.timeTaken / itemsProcessed) * (totalItems - itemsProcessed);
- this.estimatedFinishDt = new Date(currentTime.getTime() + this.timeLeft);
-
- return this._ms2Timestamp(this.timeLeft);
- }
-
- /**
- *
- *
- * @param {number} ms
- * @returns {string}
- * @memberof ProgressTimer
- */
- _ms2Timestamp(ms) {
- // 1- Convert to seconds:
- let seconds = ms / 1000;
-
- // 2- Extract hours:
- let hours = parseInt(seconds / 3600); // 3,600 seconds in 1 hour
- seconds = seconds % 3600; // seconds remaining after extracting hours
-
- // 3- Extract minutes:
- let minutes = parseInt(seconds / 60); // 60 seconds in 1 minute
-
- // 4- Keep only seconds not extracted to minutes:
- seconds = seconds % 60;
-
- let parts = seconds.toString().split('.');
-
- seconds = parseInt(parts[0]);
- let milliseconds = parts.length > 1 ? parts[1].substring(0, 3).padEnd(3, 0) : '000';
-
- hours = hours.toString().padStart(2, '0');
- minutes = minutes.toString().padStart(2, '0');
- seconds = seconds.toString().padStart(2, '0');
-
- return `${hours}:${minutes}:${seconds}.${milliseconds}`; // hours + ':' + minutes + ':' + seconds;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Date} date
- * @returns {string}
- * @memberof ProgressTimer
- */
- _dateToISOLikeButLocal(date) {
- let offsetMs = date.getTimezoneOffset() * 60 * 1000;
- let msLocal = date.getTime() - offsetMs;
- let dateLocal = new Date(msLocal);
- let iso = dateLocal.toISOString();
- let isoLocal = iso.slice(0, 19);
-
- return isoLocal.replace(/T/g, ' ');
- }
- }
-
- class Benchmark {
- constructor({ logger, printResults } = {}) {
- this.namedPerformances = {};
- this.defaultName = 'default';
- this.logger = logger;
- this.printResults = printResults == undefined ? (this.logger ? true : false) : printResults;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string=} name
- * @memberof Benchmark
- */
- start(name) {
- name = name || this.defaultName;
-
- this.namedPerformances[name] = {
- startAt: this._hrtime(),
- };
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string=} name
- * @memberof Benchmark
- */
- stop(name) {
- name = name || this.defaultName;
-
- const startAt = this.namedPerformances[name] && this.namedPerformances[name].startAt;
-
- if (!startAt) throw new Error(`Namespace: ${name} doesnt exist`);
-
- const diff = this._hrtime(startAt);
- const time = diff[0] * 1e3 + diff[1] * 1e-6;
- const words = this.getWords(diff);
- const preciseWords = this.getPreciseWords(diff);
- const verboseWords = this.getVerboseWords(diff);
- const verboseAbbrWords = this.getVerboseAbbrWords(diff);
-
- if (this.printResults) {
- let output = name != 'default' ? `[${name}] execution time:` : `execution time:`;
-
- this.logger(output, time); // words
- }
-
- return {
- name,
- time,
- words,
- preciseWords,
- verboseWords,
- verboseAbbrWords,
- diff,
- };
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} func
- * @param {{name: string, measure: boolean}=} { name, measure }
- * @returns {T}
- * @memberof Benchmark
- * @template T
- */
- wrapFunc(func, { name, measure = true } = {}) {
- name = this._getFuncName(func, name);
-
- let self = this;
-
- wrappedFunc.measure = measure;
- wrappedFunc.benchmark = {
- name,
- results: {
- runs: [],
- avg: null,
- min: null,
- max: null,
- total: null,
- times: null,
- runCount: 0,
- },
- reset: function () {
- this.results.runs = [];
- this.results.avg = null;
- this.results.min = null;
- this.results.max = null;
- this.results.total = null;
- this.results.times = null;
- this.results.runCount = 0;
- },
- printResults: function (logger = console.debug) {
- let output = this.name != 'default' ? `[${this.name}] execution summary:` : `execution summary:`;
-
- let times = wrappedFunc.benchmark.results.runs.map((run) => {
- return run.time;
- });
-
- wrappedFunc.benchmark.results.times = times;
- wrappedFunc.benchmark.results.avg = self._getAvgTime(times);
- wrappedFunc.benchmark.results.total = self._getSumTime(times);
- wrappedFunc.benchmark.results.min = Math.min(...times);
- wrappedFunc.benchmark.results.max = Math.max(...times);
-
- logger(output, `times: ${this.results.runCount} total: ${self.getWords(this.results.total)} min: ${self.getWords(this.results.min)} max: ${self.getWords(this.results.max)} avg: ${self.getWords(this.results.avg)} total: ${self.getWords(this.results.total)}`);
- },
- };
-
- function wrappedFunc() {
- if (wrappedFunc.measure) {
- self.start(name);
- }
-
- let res = func(...arguments);
-
- if (wrappedFunc.measure) {
- wrappedFunc.benchmark.results.runCount++;
-
- wrappedFunc.benchmark.results.runs.push(self.stop(name));
- }
-
- return res;
- }
-
- return this._defineWrappedFuncProperties(wrappedFunc, name);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} func
- * @param {{name: string, measure: boolean}=} { name, measure }
- * @returns {T}
- * @memberof Benchmark
- * @template T
- */
- wrapAsyncFunc(func, { name, measure = true } = {}) {
- name = this._getFuncName(func, name);
-
- let self = this;
-
- wrappedFunc.measure = measure;
- wrappedFunc.benchmark = {
- name,
- results: {
- runs: [],
- avg: null,
- min: null,
- max: null,
- total: null,
- times: null,
- runCount: 0,
- },
- reset: function () {
- this.results.runs = [];
- this.results.avg = null;
- this.results.min = null;
- this.results.max = null;
- this.results.total = null;
- this.results.times = null;
- this.results.runCount = 0;
- },
- printResults: function (logger = console.debug) {
- let output = this.name != 'default' ? `[${this.name}] execution summary:` : `execution summary:`;
-
- let times = wrappedFunc.benchmark.results.runs.map((run) => {
- return run.time;
- });
-
- wrappedFunc.benchmark.results.times = times;
- wrappedFunc.benchmark.results.avg = self._getAvgTime(times);
- wrappedFunc.benchmark.results.total = self._getSumTime(times);
- wrappedFunc.benchmark.results.min = Math.min(...times);
- wrappedFunc.benchmark.results.max = Math.max(...times);
-
- logger(output, `times: ${this.results.runCount} total: ${self.getWords(this.results.total)} min: ${self.getWords(this.results.min)} max: ${self.getWords(this.results.max)} avg: ${self.getWords(this.results.avg)} total: ${self.getWords(this.results.total)}`);
- },
- };
-
- async function wrappedFunc() {
- if (wrappedFunc.measure) {
- self.start(name);
- }
-
- let res = await func(...arguments);
-
- if (wrappedFunc.measure) {
- wrappedFunc.benchmark.results.runCount++;
-
- wrappedFunc.benchmark.results.runs.push(self.stop(name));
- }
-
- return res;
- }
-
- return this._defineWrappedFuncProperties(wrappedFunc, name);
- }
-
- getWords(diff) {
- let ms = typeof diff === 'number' ? diff : this._hrtime2Ms(diff);
-
- return this._msToHms(ms);
-
- // return this._prettyHrtime(diff);
- }
-
- getPreciseWords(diff) {
- return this._prettyHrtime(diff, { precise: true });
- }
-
- getVerboseWords(diff) {
- return this._prettyHrtime(diff, { verbose: true });
- }
-
- getVerboseAbbrWords(diff) {
- return this._prettyHrtime(diff, { verbose: true, verboseAbbrv: true, precise: true });
- }
-
- _msToHms(ms) {
- // Extract days
- let days = Math.floor(ms / (86400 * 1000));
- ms %= (86400 * 1000);
-
- // Extract hours
- let hours = Math.floor(ms / (3600 * 1000));
- ms %= (3600 * 1000);
-
- // Extract minutes
- let minutes = Math.floor(ms / (60 * 1000));
- ms = ms % (60 * 1000);
-
- // Extract seconds
- let seconds = Math.floor(ms / 1000);
- ms = ms % 1000;
-
- let result = [];
-
- if (days > 0) {
- result.push(`${days} day${days === 1 ? '' : 's'}`);
- }
-
- if (hours > 0) {
- result.push(`${hours} hr${hours === 1 ? '' : 's'}`);
- }
-
- if (minutes > 0) {
- result.push(`${minutes} min${minutes === 1 ? '' : 's'}`);
- }
-
- if (seconds > 0) {
- result.push(`${seconds} sec${seconds === 1 ? '' : 's'}`);
- }
-
- if (ms > 0) {
- result.push(`${ms} ms`);
- }
-
- return result.join(', ');
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number[][]} times
- * @returns {number}
- */
- _getAvgTime(times) {
- return this._getSumTime(times) / times.length;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number[][]} times
- * @returns {number}
- */
- _getSumTime(times) {
- return times.reduce((a, b) => a + b);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} ms
- * @returns {number[][]}
- * @memberof Benchmark
- */
- _ms2Hrtime(ms) {
- let seconds = Math.round(ms / 1000);
- let nanoSeconds = Math.round(ms * 1000000 - seconds * 1000000 * 1000);
-
- return [seconds, nanoSeconds];
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {[number, number]} hrtime
- * @returns {number}
- * @memberof Benchmark
- */
- _hrtime2Ms(hrtime) {
- return hrtime[0] * 1000 + hrtime[1] / 1000000;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} func
- * @param {string=} name
- * @returns {T}
- * @memberof Benchmark
- * @template T
- */
- _getFuncName(func, name) {
- return name ? name : 'name' in func && func.name.trim() !== '' ? func.name : '[wrapped.func]';
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Function} wrappedFunc
- * @param {string} name
- * @returns {Function}
- * @memberof Benchmark
- */
- _defineWrappedFuncProperties(wrappedFunc, name) {
- Object.defineProperty(wrappedFunc, 'name', {
- value: name,
- writable: false,
- configurable: false,
- enumerable: false,
- });
-
- Object.defineProperty(wrappedFunc, 'toString', {
- value: () => func.toString(),
- writable: false,
- configurable: false,
- enumerable: false,
- });
-
- return wrappedFunc;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {[number, number]=} time
- * @returns {[number, number]}
- * @memberof Benchmark
- */
- _hrtime(time) {
- if (typeof process !== 'undefined') return process.hrtime(time);
-
- var performance = typeof performance !== 'undefined' ? performance : {};
- let performanceNow = performance.now || performance.mozNow || performance.msNow || performance.oNow || performance.webkitNow || (() => new Date().getTime());
-
- let clocktime = performanceNow.call(performance) * 1e-3;
- let seconds = Math.floor(clocktime);
- let nanoseconds = Math.floor((clocktime % 1) * 1e9);
-
- if (time) {
- seconds = seconds - time[0];
- nanoseconds = nanoseconds - time[1];
-
- if (nanoseconds < 0) {
- seconds--;
- nanoseconds += 1e9;
- }
- }
-
- return [seconds, nanoseconds];
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {[number, number]=} time
- * @param {{verbose: boolean; verboseAbbrv: boolean; precise: boolean}} { verbose = false, verboseAbbrv = false, precise = false }
- * @returns {string}
- * @memberof Benchmark
- */
- _prettyHrtime(time, { verbose = false, verboseAbbrv = false, precise = false } = {}) {
- let i, spot, sourceAtStep, valAtStep, decimals, strAtStep, results, totalSeconds;
-
- let minimalDesc = ['h', 'min', 's', 'ms', 'μs', 'ns'];
- let verboseDesc = !verboseAbbrv ? ['hour', 'minute', 'second', 'millisecond', 'microsecond', 'nanosecond'] : minimalDesc;
- let convert = [60 * 60, 60, 1, 1e6, 1e3, 1];
-
- if (typeof time === 'number') {
- time = this._ms2Hrtime(time);
- }
-
- if (!Array.isArray(time) || time.length !== 2) return '';
-
- if (typeof time[0] !== 'number' || typeof time[1] !== 'number') return '';
-
- // normalize source array due to changes in node v5.4+
- if (time[1] < 0) {
- totalSeconds = time[0] + time[1] / 1e9;
- time[0] = parseInt(totalSeconds);
- time[1] = parseFloat((totalSeconds % 1).toPrecision(9)) * 1e9;
- }
-
- results = '';
-
- for (i = 0; i < 6; i++) {
- // grabbing first or second spot in source array
- spot = i < 3 ? 0 : 1;
- sourceAtStep = time[spot];
-
- if (i !== 3 && i !== 0) {
- // trim off previous portions
- sourceAtStep = sourceAtStep % convert[i - 1];
- }
-
- if (i === 2) {
- // get partial seconds from other portion of the array
- sourceAtStep += time[1] / 1e9;
- }
-
- // val at this unit
- valAtStep = sourceAtStep / convert[i];
-
- if (valAtStep >= 1) {
- if (verbose) {
- // deal in whole units, subsequent laps will get the decimal portion
- valAtStep = Math.floor(valAtStep);
- }
-
- if (!precise) {
- // don't fling too many decimals
- decimals = valAtStep >= 10 ? 0 : 2;
- strAtStep = valAtStep.toFixed(decimals);
- } else {
- strAtStep = valAtStep.toString();
- }
-
- if (strAtStep.indexOf('.') > -1 && strAtStep[strAtStep.length - 1] === '0') {
- // remove trailing zeros
- strAtStep = strAtStep.replace(/\.?0+$/, '');
- }
-
- if (results) {
- // append space if we have a previous value
- results += ' ';
- }
-
- // append the value
- results += strAtStep;
-
- // append units
- if (verbose) {
- results += verboseAbbrv ? `${verboseDesc[i]}` : ` ${verboseDesc[i]}`;
-
- if (!verboseAbbrv && strAtStep !== '1') {
- results += 's';
- }
- } else {
- results += ` ${minimalDesc[i]}`;
- }
-
- if (!verbose) {
- // verbose gets as many groups as necessary, the rest get only one
- break;
- }
- }
- }
-
- return results;
- }
- }
-
- class ArrayStat {
- /**
- * Creates an instance of ArrayStat.
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number[]} array
- * @memberof ArrayStat
- */
- constructor(array) {
- this.array = array;
- }
-
- _getCloned() {
- return this.array.slice(0);
- }
-
- min() {
- return Math.min.apply(null, this.array);
- }
-
- max() {
- return Math.max.apply(null, this.array);
- }
-
- range() {
- return this.max(this.array) - this.min(this.array);
- }
-
- midrange() {
- return this.range(this.array) / 2;
- }
-
- sum(array) {
- array = array || this.array;
-
- let total = 0;
-
- for (let i = 0, l = array.length; i < l; i++) total += array[i];
-
- return total;
- }
-
- mean(array) {
- array = array || this.array;
-
- return this.sum(array) / array.length;
- }
-
- median() {
- let array = this._getCloned();
-
- array.sort(function (a, b) {
- return a - b;
- });
-
- let mid = array.length / 2;
-
- return mid % 1 ? array[mid - 0.5] : (array[mid - 1] + array[mid]) / 2;
- }
-
- modes() {
- if (!this.array.length) return [];
-
- let modeMap = {};
- let maxCount = 0;
- let modes = [];
-
- this.array.forEach(function (val) {
- if (!modeMap[val]) modeMap[val] = 1;
- else modeMap[val]++;
-
- if (modeMap[val] > maxCount) {
- modes = [val];
- maxCount = modeMap[val];
- } else if (modeMap[val] === maxCount) {
- modes.push(val);
-
- maxCount = modeMap[val];
- }
- });
-
- return modes;
- }
-
- letiance() {
- let mean = this.mean();
-
- return this.mean(
- this._getCloned().map(function (num) {
- return Math.pow(num - mean, 2);
- })
- );
- }
-
- standardDeviation() {
- return Math.sqrt(this.letiance());
- }
-
- meanAbsoluteDeviation() {
- let mean = this.mean();
-
- return this.mean(
- this._getCloned().map(function (num) {
- return Math.abs(num - mean);
- })
- );
- }
-
- zScores() {
- let mean = this.mean();
- let standardDeviation = this.standardDeviation();
-
- return this._getCloned().map(function (num) {
- return (num - mean) / standardDeviation;
- });
- }
-
- withinStd(val, stdev) {
- let low = this.mean() - stdev * this.standardDeviation(); // x.deviation;
- let hi = this.mean() + stdev * this.standardDeviation(); // x.deviation;
- let res = val > low && val < hi;
-
- console.log(`val: ${val.toString().padEnd(5, ' ')} mean: ${this.mean()} stdev: ${this.standardDeviation()} hi: ${hi} low: ${low} res: ${res}`);
-
- return res;
- }
- }
-
- memoizeClass(ArrayStat);
-
- let LineColumnFinder = (function LineColumnFinder() {
- let isArray = Array.isArray;
- let isObject = (val) => val != null && typeof val === 'object' && Array.isArray(val) === false;
- let slice = Array.prototype.slice;
-
- /**
- * Finder for index and line-column from given string.
- *
- * You can call this without `new` operator as it returns an instance anyway.
- *
- * @class
- * @param {string} str - A string to be parsed.
- * @param {Object|number} [options] - Options.
- * This can be an index in the string for shorthand of `lineColumn(str, index)`.
- * @param {number} [options.origin=1] - The origin value of line and column.
- */
- function LineColumnFinder(str, options) {
- if (!(this instanceof LineColumnFinder)) {
- if (typeof options === 'number') {
- return new LineColumnFinder(str).fromIndex(options);
- }
-
- return new LineColumnFinder(str, options);
- }
-
- this.str = str || '';
- this.lineToIndex = buildLineToIndex(this.str);
-
- options = options || {};
-
- this.origin = typeof options.origin === 'undefined' ? 1 : options.origin;
- }
-
- /**
- * Find line and column from index in the string.
- *
- * @param {number} index - Index in the string. (0-origin)
- * @return {Object|null}
- * Found line number and column number in object `{ line: X, col: Y }`.
- * If the given index is out of range, it returns `null`.
- */
- LineColumnFinder.prototype.fromIndex = function (index) {
- if (index < 0 || index >= this.str.length || isNaN(index)) {
- return null;
- }
-
- let line = findLowerIndexInRangeArray(index, this.lineToIndex);
-
- return {
- line: line + this.origin,
- col: index - this.lineToIndex[line] + this.origin,
- };
- };
-
- /**
- * Find index from line and column in the string.
- *
- * @param {number|Object|Array} line - Line number in the string.
- * This can be an Object of `{ line: X, col: Y }`, or
- * an Array of `[line, col]`.
- * @param {number} [column] - Column number in the string.
- * This must be omitted or undefined when Object or Array is given
- * to the first argument.
- * @return {number}
- * Found index in the string. (always 0-origin)
- * If the given line or column is out of range, it returns `-1`.
- */
- LineColumnFinder.prototype.toIndex = function (line, column) {
- if (typeof column === 'undefined') {
- if (isArray(line) && line.length >= 2) {
- return this.toIndex(line[0], line[1]);
- }
-
- if (isObject(line) && 'line' in line && ('col' in line || 'column' in line)) {
- return this.toIndex(line.line, 'col' in line ? line.col : line.column);
- }
-
- return -1;
- }
-
- if (isNaN(line) || isNaN(column)) {
- return -1;
- }
-
- line -= this.origin;
- column -= this.origin;
-
- if (line >= 0 && column >= 0 && line < this.lineToIndex.length) {
- let lineIndex = this.lineToIndex[line];
- let nextIndex = line === this.lineToIndex.length - 1 ? this.str.length : this.lineToIndex[line + 1];
-
- if (column < nextIndex - lineIndex) {
- return lineIndex + column;
- }
- }
-
- return -1;
- };
-
- /**
- * Build an array of indexes of each line from a string.
- *
- * @private
- * @param str {string} An input string.
- * @return {number[]} Built array of indexes. The key is line number.
- */
- function buildLineToIndex(str) {
- let lines = str.split('\n');
- let lineToIndex = new Array(lines.length);
- let index = 0;
-
- for (let i = 0, l = lines.length; i < l; i++) {
- lineToIndex[i] = index;
- index += lines[i].length + /* "\n".length */ 1;
- }
-
- return lineToIndex;
- }
-
- /**
- * Find a lower-bound index of a value in a sorted array of ranges.
- *
- * Assume `arr = [0, 5, 10, 15, 20]` and
- * this returns `1` for `value = 7` (5 <= value < 10),
- * and returns `3` for `value = 18` (15 <= value < 20).
- *
- * @private
- * @param arr {number[]} An array of values representing ranges.
- * @param value {number} A value to be searched.
- * @return {number} Found index. If not found `-1`.
- */
- function findLowerIndexInRangeArray(value, arr) {
- if (value >= arr[arr.length - 1]) {
- return arr.length - 1;
- }
-
- let min = 0,
- max = arr.length - 2,
- mid;
-
- while (min < max) {
- mid = min + ((max - min) >> 1);
-
- if (value < arr[mid]) {
- max = mid - 1;
- } else if (value >= arr[mid + 1]) {
- min = mid + 1;
- } else {
- // value >= arr[mid] && value < arr[mid + 1]
- min = mid;
- break;
- }
- }
-
- return min;
- }
-
- return LineColumnFinder;
- })();
-
- class CustomContextMenu {
- /**
- * Example menuItems
- *
- * ```javascript
- * let menuItems = [
- * {
- * type: 'item',
- * label: 'Test1',
- * onClick: () => {
- * alert('test1');
- * },
- * },
- * {
- * type: 'item',
- * label: 'Test2',
- * onClick: () => {
- * console.debug('test2');
- * },
- * },
- * {
- * type: 'break',
- * },
- * {
- * type: 'item',
- * label: 'Test3',
- * onClick: () => {
- * console.debug('test3');
- * },
- * },
- * ];
- * ```
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} elemToAttachTo
- * @param {*} menuItems
- * @memberof CustomContextMenu
- */
- constructor(elemToAttachTo, menuItems, onContextMenu) {
- this.elem = elemToAttachTo;
- this.menuItems = menuItems;
- this.menu = null;
- this.onContextMenu = onContextMenu;
-
- this._createMenu();
- this._setupEvents();
-
- this.hide = debounce(this.hide.bind(this), 500, true);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} top
- * @param {number} left
- * @memberof CustomContextMenu
- */
- show(top, left) {
- document.body.appendChild(this.menu);
-
- this.menu.style.display = 'block';
-
- this.menu.style.top = `${top}px`;
- this.menu.style.left = `${left}px`;
-
- this.menu.setAttribute('tabindex', '0');
- this.menu.focus();
- }
-
- hide() {
- this.menu.style.display = 'none';
-
- if (document.body.contains(this.menu)) {
- this.menu.remove();
- }
- }
-
- _setupEvents() {
- this.elem.addEventListener('contextmenu', (ev) => {
- ev.preventDefault();
-
- if (this.onContextMenu) {
- this.onContextMenu(ev);
- }
-
- this.show(ev.pageY, ev.pageX);
- });
-
- document.addEventListener('click', (ev) => {
- if (document.body.contains(this.menu) && !this._isHover(this.menu)) {
- this.hide();
- }
- });
-
- window.addEventListener('blur', (ev) => {
- this.hide();
- });
-
- this.menu.addEventListener('blur', (ev) => {
- this.hide();
- });
- }
-
- _createMenu() {
- this.menu = this._createMenuContainer();
-
- for (let i = 0; i < this.menuItems.length; i++) {
- let itemConfig = this.menuItems[i];
-
- switch (itemConfig.type) {
- case 'item':
- this.menu.appendChild(this._createItem(itemConfig));
-
- break;
-
- case 'break':
- this.menu.appendChild(this._createBreak());
-
- break;
-
- default:
- break;
- }
- }
-
- // document.body.appendChild(this.menu);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {HTMLElement}
- * @memberof CustomContextMenu
- */
- _createMenuContainer() {
- let html = `<div class="context" hidden></div>`;
-
- let elem = this._createElementsFromHTML(html);
-
- return elem;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} itemConfig
- * @returns {HTMLElement}
- * @memberof CustomContextMenu
- */
- _createItem(itemConfig) {
- let html = `<div class="context_item">
- <div class="inner_item">
- ${itemConfig.label}
- </div>
- </div>`;
-
- let elem = this._createElementsFromHTML(html);
-
- if (itemConfig.id) {
- elem.id = itemConfig.id;
- }
-
- if (itemConfig.onClick) {
- elem.addEventListener('click', (ev) => {
- itemConfig.onClick(ev);
-
- this.hide();
- });
- }
-
- return elem;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {HTMLElement}
- * @memberof CustomContextMenu
- */
- _createBreak() {
- let html = `<div class="context_hr"></div>`;
-
- let elem = this._createElementsFromHTML(html);
-
- return elem;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} htmlStr
- * @returns {HTMLElement}
- */
- _createElementsFromHTML(htmlStr) {
- let div = document.createElement('div');
-
- div.innerHTML = htmlStr.trim();
-
- return div.firstChild;
- }
-
- _isHover(elem) {
- return elem.parentElement.querySelector(':hover') === elem;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @class LocalStorageEx
- */
- class LocalStorageEx {
- /**
- * Creates an instance of LocalStorageEx.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @memberof LocalStorageEx
- */
- constructor() {
- this.__storage = localStorage;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @readonly
- * @memberof LocalStorageEx
- */
- get UNDEFINED_SAVED_VALUE() {
- return '__** undefined **__';
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @readonly
- * @memberof LocalStorageEx
- */
- get size() {
- let total = 0;
-
- for (let x in this.__storage) {
- // Value is multiplied by 2 due to data being stored in `utf-16` format, which requires twice the space.
- let amount = this.__storage[x].length * 2;
-
- if (!isNaN(amount) && this.__storage.hasOwnProperty(x)) {
- total += amount;
- }
- }
-
- return total;
- }
-
- /**
- * Determine if browser supports local storage.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {boolean}
- * @memberof LocalStorageEx
- */
- isSupported() {
- return typeof Storage !== 'undefined';
- }
-
- /**
- * Check if key exists in local storage.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} key
- * @returns {boolean}
- * @memberof LocalStorageEx
- */
- has(key) {
- if (typeof key === 'object') {
- key = JSON.stringify(key);
- }
-
- return this.__storage.hasOwnProperty(key);
- }
-
- /**
- * Retrieve an object from local storage.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} key
- * @param {*} [defaultValue=null]
- * @returns {*}
- * @memberof LocalStorageEx
- */
- get(key, defaultValue = null) {
- if (typeof key === 'object') {
- key = JSON.stringify(key);
- }
-
- if (!this.has(key)) {
- return defaultValue;
- }
-
- let item = this.__storage.getItem(key);
-
- try {
- if (item === '__** undefined **__') {
- return undefined;
- } else {
- return JSON.parse(item);
- }
- } catch (error) {
- return item;
- }
- }
-
- /**
- * Save some value to local storage.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} key
- * @param {*} value
- * @returns {void}
- * @memberof LocalStorageEx
- */
- set(key, value) {
- if (typeof key === 'object') {
- key = JSON.stringify(key);
- }
-
- if (value === undefined) {
- value = this.UNDEFINED_SAVED_VALUE;
- } else if (typeof value === 'object') {
- value = JSON.stringify(value);
- }
-
- this.__storage.setItem(key, value);
- }
-
- /**
- * Remove element from local storage.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} key
- * @returns {void}
- * @memberof LocalStorageEx
- */
- remove(key) {
- if (typeof key === 'object') {
- key = JSON.stringify(key);
- }
-
- this.__storage.removeItem(key);
- }
-
- toString() {
- return JSON.parse(JSON.stringify(this.__storage));
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @class SessionStorageEx
- * @extends {LocalStorageEx}
- */
- class SessionStorageEx extends LocalStorageEx {
- /**
- * Creates an instance of SessionStorageEx.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @memberof SessionStorageEx
- */
- constructor() {
- super();
-
- this.__storage = sessionStorage;
- }
- }
-
- class IgnoreCaseMap extends Map {
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} key
- * @param {*} value
- * @returns {this}
- * @memberof IgnoreCaseMap
- */
- set(key, value) {
- return super.set(key.toLocaleLowerCase(), value);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} key
- * @returns {*}
- * @memberof IgnoreCaseMap
- */
- get(key) {
- return super.get(key.toLocaleLowerCase());
- }
- }
-
- // #endregion Helper Classes
-
- // #region Helper Functions
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} name
- * @param {{logLevel: log.LogLevelDesc, tag: string}} logLevel
- * @return {log.Logger}
- */
- function getLogger(name, { logLevel, tag }) {
- prefix.reg(log);
-
- const colors = {
- TRACE: '220;86;220',
- DEBUG: '86;86;220',
- INFO: '134;134;221',
- WARN: '220;220;86',
- ERROR: '220;86;86',
- };
-
- /** @type {prefix.LoglevelPluginPrefixOptions} */
- let options = {
- // template: tag ? `[%t] %l [${tag}] %n:` : '[%t] %l %n:',
- levelFormatter: function (level) {
- return level.toUpperCase();
- },
- nameFormatter: function (name) {
- return name || 'root';
- },
- timestampFormatter: function (date) {
- return date.toTimeString().replace(/.*(\d{2}:\d{2}:\d{2}).*/, '$1');
- },
- format: function (level, name, timestamp) {
- let _timestamp = `\x1B[90m[${timestamp}]\x1B[m`;
- let _level = `\x1B[38;2;${colors[level.toUpperCase()]}m${level.toUpperCase()}\x1B[m`;
- let _name = `\x1B[38;2;38;177;38m${tag ? `[${tag}-` : '['}${name}]\x1B[m`;
-
- let _format = `${_timestamp} ${_level} ${_name}:`;
-
- return _format;
- },
- };
-
- const logger = log.getLogger(name);
-
- prefix.apply(logger, options);
-
- logger.setLevel(logLevel || 'WARN');
-
- return logger;
- }
-
- function pp(obj, fn) {
- fn = fn || console.log;
-
- fn(pformat(obj));
- }
-
- function pformat(obj, space = 4) {
- return JSON.stringify(obj, null, space);
- }
-
- function removeAllButLastStrPattern(string, token) {
- let parts = string.split(token);
-
- if (parts[1] === undefined) return string;
- else return parts.slice(0, -1).join('') + token + parts.slice(-1);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Array.<T> | Array} arr
- * @param {?function(T, T): boolean} callbackObjs
- * @return {T[]}
- * @template T
- */
- function dedupeArr(arr, callbackObjs) {
- if (callbackObjs) {
- let tempArr = /** @type {[]} */ (arr).filter((value, index) => {
- return (
- index ===
- arr.findIndex((other) => {
- return callbackObjs(value, other);
- })
- );
- });
-
- return tempArr;
- } else {
- return [...new Set(arr)];
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {any} obj
- * @returns {boolean}
- */
- function isClass(obj) {
- return typeof obj === 'function' && /^\s*class\s+/.test(obj.toString());
- }
-
- /**
- * Checks whether a variable is a class or an instance created with `new`.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} value The variable to check.
- * @returns {boolean} `true` if the variable is a class or an instance created with `new`, `false` otherwise.
- */
- function isClassOrInstance(value) {
- // prettier-ignore
- if (typeof value === 'function' &&
- value.prototype &&
- typeof value.prototype.constructor === 'function' &&
- value.prototype.constructor !== Array &&
- value.prototype.constructor !== Object) {
- return true; // It's a class
- } else if (typeof value === 'object' &&
- value.constructor &&
- typeof value.constructor === 'function' &&
- value.constructor.prototype &&
- typeof value.constructor.prototype.constructor === 'function' &&
- value.constructor !== Array &&
- value.constructor !== Object) {
- return true; // It's an instance created with new
- }
-
- return false;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {*} value The variable to check.
- * @returns {boolean}
- */
- function isFunction(value) {
- try {
- return typeof value == 'function';
- } catch (error) {
- return false;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {object} obj
- * @param {{ propsToExclude?: string[]; namesOnly: boolean; removeDuplicates: boolean; asObject: boolean }} [{ propsToExclude = [], namesOnly = false, removeDuplicates = true, asObject = false } = {}]
- * @returns
- */
- function getObjProps(obj, { propsToExclude = [], namesOnly = false, removeDuplicates = true, asObject = false } = {}) {
- // Default
- let _propsToExclude = [
- //
- '__defineGetter__',
- '__defineSetter__',
- '__lookupSetter__',
- '__lookupGetter__',
- '__proto__',
- '__original__',
-
- 'caller',
- 'callee',
- 'arguments',
-
- 'toString',
- 'valueOf',
- 'constructor',
- 'hasOwnProperty',
- 'isPrototypeOf',
- 'propertyIsEnumerable',
- 'toLocaleString',
- ];
-
- _propsToExclude = propsToExclude && Array.isArray(propsToExclude) ? _propsToExclude.concat(propsToExclude) : _propsToExclude;
-
- let objHierarchy = getObjHierarchy(obj);
- let propNames = getPropNames(objHierarchy);
- let plainObj = {};
- let objKeys = [];
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {any} obj
- * @returns {Array<any>}
- */
- function getObjHierarchy(obj) {
- let objs = [obj];
-
- obj = isClassOrInstance(obj) ? obj.prototype || obj.__proto__ : obj;
-
- do {
- objs.push(obj);
- } while ((obj = Object.getPrototypeOf(obj)));
-
- return objs;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Array<any>} objHierarchy
- * @returns {string[]}
- */
- function getPropNames(objHierarchy) {
- /** @type {string[]} */
- let propNames = [];
-
- for (let i = 0; i < objHierarchy.length; i++) {
- const _obj = objHierarchy[i];
-
- let getPropFuncs = [Object.getOwnPropertyNames, Object.getOwnPropertySymbols];
-
- getPropFuncs.forEach((func) => {
- let _propNames = func(_obj);
-
- _propNames.forEach((propName) => {
- if (!_propsToExclude.includes(propName) && !propNames.includes(propName)) {
- propNames.push(propName);
- }
- });
- });
- }
-
- return propNames;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{ name: string, value: any }[]} props
- * @return {{ name: string, value: any }[]}
- */
- function dedupeProps(props) {
- function findNonNullProp(props, name) {
- let res = props.find((prop) => prop.name == name && prop.value != null);
-
- if (!res) {
- res = props.find((prop) => prop.name == name);
- }
-
- return res;
- }
-
- function propsContains(props, name) {
- return props.some((prop) => prop.name == name);
- }
-
- let newProps = [];
-
- for (let i = 0; i < props.length; i++) {
- const prop = props[i];
-
- let tempProp = findNonNullProp(props, prop.name);
-
- if (!propsContains(newProps, tempProp.name)) {
- newProps.push(tempProp);
- }
- }
-
- return newProps;
- }
-
- function getProps(objHierarchy, doFuncs = false) {
- /** @type {{ name: string, value: any }} */
- let props = [];
-
- for (let o = 0; o < objHierarchy.length; o++) {
- const _obj = objHierarchy[o];
-
- for (let p = 0; p < propNames.length; p++) {
- const propName = propNames[p];
- let value;
-
- try {
- value = _obj[propName];
- } catch (error) {}
-
- if (!_propsToExclude.includes(propName)) {
- if (asObject) {
- if (!objKeys.includes(propName)) {
- objKeys.push(propName);
-
- plainObj[propName] = value;
- }
- } else {
- props.push({
- name: propName,
- value: value,
- });
- }
- }
- }
- }
-
- if (!asObject) {
- if (removeDuplicates) {
- props = dedupeProps(props);
- }
-
- props = props.filter(function (prop, i, props) {
- let exprs = [
- //
- !_propsToExclude.includes(prop.name),
- // props[i + 1] && prop.name != props[i + 1].name,
- ...(doFuncs ? [isFunction(prop.value)] : [!isFunction(prop.value)]),
- ];
-
- return exprs.every(Boolean);
- });
- }
-
- if (asObject) {
- return plainObj;
- } else {
- return props.sort(function (a, b) {
- let aName = typeof a.name == 'symbol' ? a.name.toString() : a.name;
- let bName = typeof b.name == 'symbol' ? b.name.toString() : b.name;
-
- if (aName < bName) return -1;
- if (aName > bName) return 1;
-
- return 0;
- });
- }
- }
-
- let res;
-
- if (asObject) {
- getProps(objHierarchy, true);
- getProps(objHierarchy);
-
- res = plainObj;
- } else {
- res = {
- funcs: getProps(objHierarchy, true),
- props: getProps(objHierarchy),
- };
-
- if (namesOnly) {
- res.funcs = res.funcs.filter((func) => func.name.toString() != 'Symbol(Symbol.hasInstance)').map((func) => func.name);
- res.props = res.props.filter((prop) => prop.name.toString() != 'Symbol(Symbol.hasInstance)').map((prop) => prop.name);
- }
- }
-
- objHierarchy = null;
-
- return res;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Window} [_window=window]
- * @param {{ namesOnly: boolean; asObject: boolean }} [{ namesOnly = false, asObject = false } = {}]
- * @returns
- */
- function getUserDefinedGlobalProps(_window = null, { namesOnly = false, asObject = false } = {}) {
- _window = _window || getWindow();
-
- let iframe = document.createElement('iframe');
-
- iframe.style.display = 'none';
-
- document.body.appendChild(iframe);
-
- let plainObj = {};
- let objKeys = [];
-
- function getProps(obj, doFuncs = false) {
- let props = [];
- let _obj = obj;
-
- let getPropFuncs = [Object.getOwnPropertyNames, Object.getOwnPropertySymbols];
-
- getPropFuncs.forEach((func) => {
- let propNames = func(_obj);
-
- for (let i = 0; i < propNames.length; i++) {
- const propName = propNames[i];
- let value;
-
- try {
- value = _obj[propName];
- } catch (error) {}
-
- if (isNumber(propName) && value?.constructor?.name == 'Window') continue;
-
- if (!iframe.contentWindow.hasOwnProperty(propName)) {
- if (asObject) {
- if (!objKeys.includes(propName)) {
- objKeys.push(propName);
-
- plainObj[propName] = value;
- }
- } else {
- props.push({
- name: propName,
- value: value,
- });
- }
- }
- }
- });
-
- if (!asObject) {
- props = props.filter(function (prop, i, props) {
- let propName1 = prop.name;
- let propName2 = props[i + 1] ? props[i + 1].name : undefined;
- let propValue1 = prop.value;
- let propValue2 = props[i + 1] ? props[i + 1].value : undefined;
-
- let exprs = [
- //
- // props[i + 1] && propName1 != propName2,
- (props[i + 1] && propName1.constructor.name == 'Symbol' && propName2.constructor.name == 'Symbol' && propValue1 != propValue2) || propName1 != propName2,
- ...(doFuncs ? [isFunction(obj[propName1])] : [!isFunction(obj[propName1])]),
- ];
-
- return exprs.every(Boolean);
- });
- }
-
- if (asObject) {
- return plainObj;
- } else {
- return props.sort(function (a, b) {
- let aName = typeof a.name == 'symbol' ? a.name.toString() : a.name;
- let bName = typeof b.name == 'symbol' ? b.name.toString() : b.name;
-
- if (aName < bName) return -1;
- if (aName > bName) return 1;
-
- return 0;
- });
- }
- }
-
- let res;
-
- if (asObject) {
- getProps(_window, true);
- getProps(_window);
-
- res = plainObj;
- } else {
- res = {
- funcs: getProps(_window, true),
- props: getProps(_window),
- };
-
- if (namesOnly) {
- res.funcs = res.funcs.filter((func) => func.name.toString() != 'Symbol(Symbol.hasInstance)').map((func) => func.name);
- res.props = res.props.filter((prop) => prop.name.toString() != 'Symbol(Symbol.hasInstance)').map((prop) => prop.name);
- }
- }
-
- document.body.removeChild(iframe);
-
- return res;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} obj
- * @param {T | boolean} thisArg
- * @returns {T}
- * @template T
- */
- function storeObjOriginalFuncs(obj, thisArg = true) {
- let props = getObjProps(obj);
-
- obj.__original__ = {};
-
- for (let i = 0; i < props.funcs.length; i++) {
- const func = props.funcs[i];
-
- if (thisArg == true) {
- obj.__original__[func.name] = func.value.bind(obj);
- } else {
- obj.__original__[func.name] = thisArg != false && thisArg != null && thisArg != undefined ? func.value.bind(thisArg) : func.value;
- }
- }
-
- return obj;
- }
-
- function printProps(obj, title) {
- let headerFooterBanner = '*********************************************************';
-
- console.log(headerFooterBanner);
- console.log(`* ${title || ''}`);
- console.log(headerFooterBanner);
-
- for (let key in obj) console.log(key + ': ', [obj[key]]);
-
- console.log(headerFooterBanner);
- }
-
- function sortObject(o, desc) {
- let sorted = {};
- let key;
- let a = [];
-
- for (key in o) {
- if (o.hasOwnProperty(key)) a.push(key);
- }
-
- if (desc) a.sort(sortDescending);
- else a.sort(sortAscending);
-
- for (key = 0; key < a.length; key++) sorted[a[key]] = o[a[key]];
-
- return sorted;
- }
-
- function sortAscending(a, b) {
- if (typeof a == 'string') {
- a = a.toLowerCase();
- b = b.toLowerCase();
- }
-
- if (a < b) return -1;
- else if (a > b) return 1;
- else return 0;
- }
-
- function sortDescending(a, b) {
- if (typeof a == 'string') {
- a = a.toLowerCase();
- b = b.toLowerCase();
- }
-
- if (a > b) return -1;
- else if (a < b) return 1;
- else return 0;
- }
-
- function getFileExtension(sFile) {
- return sFile.replace(/^(.*)(\.[^/.]+)$/, '$2');
- }
-
- /**
- * Async wait function.
- * Example:
- * (async () => {
- * await wait(4000).then(() => {
- * console.log(new Date().toLocaleTimeString());
- * }).then(() => {
- * console.log('here');
- * });
- * })();
- *
- * @param {number} ms - Milliseconds to wait.
- * @param {boolean} [synchronous=false] - Wait synchronously.
- */
- async function wait(ms, synchronous = false) {
- let _wait = (ms, synchronous) => {
- if (synchronous) {
- let start = Date.now();
- let now = start;
-
- while (now - start < ms) now = Date.now();
- } else {
- return new Promise((resolve) => setTimeout(resolve, ms));
- }
- };
-
- await _wait(ms, synchronous);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {() => bool} condition
- * @param {{ timeout?: number; callback: () => T; conditionIsAsync: boolean; }} [{ timeout, callback, conditionIsAsync = false } = {}]
- * @returns {T}
- * @template T
- */
- async function waitUntil(condition, { timeout, callback, conditionIsAsync = false } = {}) {
- timeout = timeout || -1;
- let maxTime = timeout == -1 ? 20000 : -1;
- let startTime = new Date();
-
- let timeRanOut = false;
-
- let done = (() => {
- let deferred = {};
-
- deferred.promise = new Promise((resolve, reject) => {
- deferred.resolve = resolve;
- deferred.reject = reject;
- });
-
- return deferred;
- })();
-
- /** @type {number} */
- let timeoutId;
-
- if (timeout && timeout > 0) {
- timeoutId = setTimeout(() => {
- timeRanOut = true;
-
- return done.reject();
- }, timeout);
- }
-
- let loop = async () => {
- let endTime = new Date();
- let elapsed = endTime - startTime;
-
- let conditionResult = conditionIsAsync ? await condition() : condition();
-
- if (conditionResult || timeRanOut || (maxTime != -1 && elapsed > maxTime)) {
- clearTimeout(timeoutId);
-
- return done.resolve(callback ? await callback() : undefined);
- }
-
- setTimeout(loop, 0);
- };
-
- setTimeout(loop, 0);
-
- return done.promise;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {any} obj
- * @param {boolean} [getInherited=false]
- * @returns {string}
- */
- function getType(obj, getInherited = false) {
- let _typeVar = (function (global) {
- let cache = {};
-
- return function (obj) {
- let key;
-
- // null
- if (obj == null) return 'null';
-
- // window/global
- if (obj == global) return 'global';
-
- // basic: string, boolean, number, undefined
- if (!['object', 'function'].includes((key = typeof obj))) return key;
-
- if (obj.constructor != undefined && obj.constructor.name != 'Object' && !getInherited) return obj.constructor.name;
-
- // cached. date, regexp, error, object, array, math
- // and get XXXX from [object XXXX], and cache it
- return cache[(key = {}.toString.call(obj))] || (cache[key] = key.slice(8, -1));
- };
- })(globalThis);
-
- return _typeVar(obj);
- }
-
- /**
- * Returns a function, that, as long as it continues to be invoked, will not
- * be triggered. The function will be called after it stops being called for
- * N milliseconds. If `immediate` is passed, trigger the function on the
- * leading edge, instead of the trailing.
- *
- * @param {function} func
- * @param {Number} wait
- * @param {Boolean} immediate
- * @returns
- */
- function debounce(func, wait, immediate) {
- let timeout;
-
- return function () {
- let context = this,
- args = arguments;
-
- let later = function () {
- timeout = null;
-
- if (!immediate) func.apply(context, args);
- };
-
- let callNow = immediate && !timeout;
-
- clearTimeout(timeout);
- timeout = setTimeout(later, wait);
-
- if (callNow) func.apply(context, args);
- };
- }
-
- function equals(x, y) {
- if (x === y) return true;
- // if both x and y are null or undefined and exactly the same
-
- if (!(x instanceof Object) || !(y instanceof Object)) return false;
- // if they are not strictly equal, they both need to be Objects
-
- if (x.constructor !== y.constructor) return false;
- // they must have the exact same prototype chain, the closest we can do is
- // test there constructor.
-
- for (let p in x) {
- if (!x.hasOwnProperty(p)) continue;
- // other properties were tested using x.constructor === y.constructor
-
- if (!y.hasOwnProperty(p)) return false;
- // allows to compare x[ p ] and y[ p ] when set to undefined
-
- if (x[p] === y[p]) continue;
- // if they have the same strict value or identity then they are equal
-
- if (typeof x[p] !== 'object') return false;
- // Numbers, Strings, Functions, Booleans must be strictly equal
-
- if (!equals(x[p], y[p])) return false;
- // Objects and Arrays must be tested recursively
- }
-
- for (p in y) {
- if (y.hasOwnProperty(p) && !x.hasOwnProperty(p)) return false;
- // allows x[ p ] to be set to undefined
- }
- return true;
- }
-
- function isEncoded(uri) {
- uri = uri || '';
-
- return uri !== decodeURIComponent(uri);
- }
-
- function fullyDecodeURI(uri) {
- while (isEncoded(uri)) uri = decodeURIComponent(uri);
-
- return uri;
- }
-
- /**
- * Get difference in days between two dates.
- *
- * @param {Date} a
- * @param {Date} b
- * @returns
- */
- function dateDiffInDays(a, b) {
- let _MS_PER_DAY = 1000 * 60 * 60 * 24;
-
- // Discard the time and time-zone information.
- let utc1 = Date.UTC(a.getFullYear(), a.getMonth(), a.getDate());
- let utc2 = Date.UTC(b.getFullYear(), b.getMonth(), b.getDate());
-
- return Math.floor((utc1 - utc2) / _MS_PER_DAY);
- }
-
- function randomInt(min, max) {
- return Math.floor(Math.random() * (max - min + 1) + min);
- }
-
- function randomFloat(min, max) {
- return Math.random() * (max - min + 1) + min;
- }
-
- function keySort(keys, desc) {
- return function (a, b) {
- let aVal = null;
- let bVal = null;
-
- for (let i = 0; i < keys.length; i++) {
- const key = keys[i];
-
- if (i == 0) {
- aVal = a[key];
- bVal = b[key];
- } else {
- aVal = aVal[key];
- bVal = bVal[key];
- }
- }
- return desc ? ~~(aVal < bVal) : ~~(aVal > bVal);
- };
- }
-
- function observe(obj, handler) {
- return new Proxy(obj, {
- get(target, key) {
- return target[key];
- },
- set(target, key, value) {
- target[key] = value;
-
- if (handler) handler();
- },
- });
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Console} console
- */
- function addSaveToConsole(console) {
- console.save = function (data, filename) {
- if (!data) {
- console.error('Console.save: No data');
-
- return;
- }
-
- if (!filename) filename = 'console.json';
-
- if (typeof data === 'object') data = JSON.stringify(data, undefined, 4);
-
- let blob = new Blob([data], {
- type: 'text/json',
- });
- let event = document.createEvent('MouseEvents');
- let tempElem = document.createElement('a');
-
- tempElem.download = filename;
- tempElem.href = window.URL.createObjectURL(blob);
- tempElem.dataset.downloadurl = ['text/json', tempElem.download, tempElem.href].join(':');
-
- event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
- tempElem.dispatchEvent(event);
- };
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Window} _window
- * @param {string} propName
- * @param {{} | [] | any} value
- */
- function setupWindowProps(_window, propName, value) {
- if (getType(value) == 'object') {
- if (typeof _window[propName] === 'undefined' || _window[propName] == null) {
- _window[propName] = {};
- }
-
- let keys = Object.keys(value);
-
- for (let i = 0; i < keys.length; i++) {
- const key = keys[i];
-
- if (!(/** @type {{}} */ (_window[propName].hasOwnProperty(key)))) {
- _window[propName][key] = null;
- }
-
- if (_window[propName][key] == null) {
- _window[propName][key] = value[key];
- }
- }
- } else {
- if (typeof _window[propName] === 'undefined' || _window[propName] == null) {
- _window[propName] = value;
- }
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{ name: string; value: any; }[]} variables
- */
- function exposeGlobalVariables(variables) {
- variables.forEach((variable, index, variables) => {
- try {
- setupWindowProps(getWindow(), variable.name, variable.value);
- } catch (error) {
- logger.error(`Unable to expose variable ${variable.name} into the global scope.`);
- }
- });
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} str
- * @returns {string}
- */
- function htmlEntitiesDecode(str) {
- return str
- .replace(/&/g, '&')
- .replace(/</g, '<')
- .replace(/>/g, '>')
- .replace(/"/g, '"');
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} metaName
- * @returns {string}
- */
- function getMeta(metaName) {
- const metas = document.getElementsByTagName('meta');
-
- for (let i = 0; i < metas.length; i++) {
- if (metas[i].getAttribute('name') === metaName) {
- return metas[i].getAttribute('content');
- }
- }
-
- return '';
- }
-
- /**
- *
- *
- * @returns {Window & typeof globalThis}
- */
- function getWindow() {
- return globalThis.GM_info && GM_info.script.grant.includes('unsafeWindow') ? unsafeWindow : globalThis;
- }
-
- /**
- *
- *
- * @param {Window} _window
- */
- function getTopWindow(_window = null) {
- _window = _window || getWindow();
-
- try {
- if (_window.self !== _window.top) {
- _window = getTopWindow(_window.parent);
- }
- } catch (e) {}
-
- return _window;
- }
-
- /**
- * Setup global error handler
- *
- * **Example:**
- * ```javascript
- * setupGlobalErrorHandler({
- * callback: (error) => console.error('Error:', error),
- * continuous: true,
- * prevent_default: true,
- * tag: '[test-global-error-handler]',
- * });
- * ```
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{ callback: (error: ErrorEx) => void; continuous?: boolean; prevent_default?: boolean; tag?: string; logFunc?: (...data: any[]) => void; _window: Window; }} [{ callback, continuous = true, prevent_default = false, tag = '[akkd]', logFunc = console.error, _window = window } = {}]
- */
- function setupGlobalErrorHandler({ callback, continuous = true, prevent_default = false, tag = null, logFunc = console.error, _window = window } = {}) {
- // respect existing onerror handlers
- let _onerror_original = _window.onerror;
-
- // install our new error handler
- _window.onerror = function (event, source, lineno, colno, error) {
- if (_onerror_original) {
- _onerror_original(event, source, lineno, colno, error);
- }
-
- // unset onerror to prevent loops and spamming
- let _onerror = _window.onerror;
-
- _window.onerror = null;
-
- // now deal with the error
- let errorObject = new ErrorEx(event, source, lineno, colno, error);
- let errorMessage = createErrorMessage(errorObject);
-
- if (tag) {
- let rgb = '38;177;38';
-
- tag = `\x1B[38;2;${rgb}m${tag}\x1B[m`;
-
- logFunc(tag, errorMessage);
- } else {
- logFunc(errorMessage);
- }
-
- // run callback if provided
- if (callback) {
- callback(errorObject);
- }
-
- // re-install this error handler again if continuous mode
- if (continuous) {
- _window.onerror = _onerror;
- }
-
- // true if normal error propagation should be suppressed
- // (i.e. normally console.error is logged by the browser)
- return prevent_default;
- };
-
- class ErrorEx {
- /**
- * Creates an instance of ErrorEx.
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string | Event} event
- * @param {string} source
- * @param {number} lineno
- * @param {number} colno
- * @param {Error} error
- * @memberof ErrorEx
- */
- constructor(event, source, lineno, colno, error) {
- this.name = error.name;
- this.message = error && error.message ? error.message : null;
- this.stack = error && error.stack ? error.stack : null;
- this.event = event;
- this.location = document.location.href;
- this.url = source;
- this.lineno = lineno;
- this.colno = colno;
- this.useragent = navigator.userAgent;
- this.fileName = error && error.fileName ? error.fileName : null;
- this.description = error && error.description ? error.description : null;
- this.name = error && error.name ? error.name : null;
- this.error = error;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {ErrorEx} error
- * @returns {string}
- */
- function createErrorMessage(error) {
- let name = error && error.name ? error.name : 'Error';
- let message = error && error.message ? error.message : 'Unknown error occured';
- let stack = error && error.stack ? error.stack.split('\n').splice(1).join('\n') : 'Error';
-
- let errorMessage = `Uncaught Global ${name}: ${message}\n${stack}`;
-
- return errorMessage;
- }
- }
-
- function applyCss(cssFiles) {
- /** @type {{ css: string, node?: HTMLElement }[]} */
- let cssArr = [];
-
- for (let i = 0; i < cssFiles.length; i++) {
- let cssStr = GM_getResourceText(cssFiles[i]);
-
- cssArr.push({
- css: cssStr,
- });
- }
-
- addStyles(cssArr);
- }
-
- function applyCss2(cssFiles) {
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} cssStyleStr
- * @returns {HTMLStyleElement}
- */
- function createStyleElementFromCss(cssStyleStr) {
- let style = document.createElement('style');
-
- style.innerHTML = cssStyleStr.trim();
-
- return style;
- }
-
- let ranOnce = false;
-
- /** @type {HTMLStyleElement[]} */
- getWindow().akkd.styleElements = [];
-
- function removeStyleElements() {
- for (let i = 0; i < getWindow().akkd.styleElements.length; i++) {
- let styleElement = getWindow().akkd.styleElements[i];
-
- styleElement.remove();
- }
-
- getWindow().akkd.styleElements = [];
- }
-
- function _editStyleSheets() {
- $(document).arrive('style, link', async function () {
- if (this.tagName == 'LINK' && this.href.includes('.css')) {
- removeStyleElements();
-
- for (let i = 0; i < cssFiles.length; i++) {
- let cssFile = cssFiles[i];
- let css = GM_getResourceText(cssFile);
-
- let styleElem = createStyleElementFromCss(css);
-
- styleElem.id = `akkd-transform-style-${(i + 1).toString().padStart(2, '0')}`;
-
- getWindow().akkd.styleElements.push(styleElem);
-
- document.body.appendChild(styleElem);
- }
- }
- });
-
- if (!ranOnce) {
- for (let i = 0; i < cssFiles.length; i++) {
- let cssFile = cssFiles[i];
- let css = GM_getResourceText(cssFile);
-
- let styleElem = createStyleElementFromCss(css);
-
- styleElem.id = `akkd-transform-style-${(i + 1).toString().padStart(2, '0')}`;
-
- getWindow().akkd.styleElements.push(styleElem);
-
- document.body.appendChild(styleElem);
- }
-
- ranOnce = true;
- }
- }
-
- _editStyleSheets();
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{ css: string, node?: HTMLElement }[]} cssArr
- */
- let addStyles = (function () {
- /** @type {string[]} */
- const addedStyleIds = [];
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{ css: string, node?: HTMLElement }[]} cssArr
- * @param {{ useGM: boolean }} cssArr { useGM = true } = {}
- */
- function _addStyles(cssArr, { useGM = true } = {}) {
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} css
- * @returns {HTMLStyleElement}
- */
- function createStyleElementFromCss(css) {
- let style = document.createElement('style');
-
- style.innerHTML = css.trim();
-
- return style;
- }
-
- function removeStyleElements() {
- for (let i = addedStyleIds.length - 1; i >= 0; i--) {
- /** @type {HTMLStyleElement} */
- let styleElem = document.getElementById(addedStyleIds[i]);
-
- if (styleElem) {
- styleElem.remove();
-
- addedStyleIds.splice(i, 1);
- }
- }
- }
-
- function addStyleElements() {
- for (let i = 0; i < cssArr.length; i++) {
- try {
- const css = cssArr[i].css;
- const node = cssArr[i].node || document.head;
-
- /** @type {HTMLStyleElement} */
- let elem = useGM ? GM_addStyle(css) : createStyleElementFromCss(css);
-
- elem.id = `akkd-custom-style-${(i + 1).toString().padStart(2, '0')}`;
-
- node.append(elem);
-
- addedStyleIds.push(elem.id);
- } catch (error) {
- console.error(error);
- }
- }
- }
-
- removeStyleElements();
- addStyleElements();
-
- return addedStyleIds;
- }
-
- return _addStyles;
- })();
-
- /**
- * Return uuid of form xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {string}
- */
- function uuid4() {
- let uuid = '';
- let ii;
-
- for (ii = 0; ii < 32; ii += 1) {
- switch (ii) {
- case 8:
- case 20:
- uuid += '-';
- uuid += ((Math.random() * 16) | 0).toString(16);
-
- break;
-
- case 12:
- uuid += '-';
- uuid += '4';
-
- break;
-
- case 16:
- uuid += '-';
- uuid += ((Math.random() * 4) | 8).toString(16);
-
- break;
-
- default:
- uuid += ((Math.random() * 16) | 0).toString(16);
- }
- }
-
- return uuid;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{} | []} obj
- * @param {string | {oldName: string, newName: string}[]} oldName
- * @param {string=} newName
- * @returns
- */
- function renameProperty(obj, oldName, newName) {
- function _renameProperty(obj, oldName, newName) {
- let keys = Object.keys(obj);
-
- for (let i = 0; i < keys.length; i++) {
- let key = keys[i];
- let value = obj[key];
-
- if (value && typeof value == 'object') {
- obj[key] = _renameProperty(value, oldName, newName);
- }
-
- if (obj.hasOwnProperty(oldName)) {
- obj[newName] = obj[oldName];
-
- delete obj[oldName];
- }
- }
-
- return obj;
- }
-
- let renames = Array.isArray(oldName) ? oldName : [{ oldName, newName }];
-
- for (let i = 0; i < renames.length; i++) {
- const rename = renames[i];
-
- obj = _renameProperty(obj, rename.oldName, rename.newName);
- }
-
- return obj;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Promise<T>} fn
- * @param {{ retries?: number; interval?: number; maxTime?: number; throwError?: boolean; }} { retries = 3, interval = 100, maxTime = null, throwError = false }
- * @returns {Promise<T>}
- * @template T
- */
- async function retry(fn, { retries = 3, interval = 100, maxTime = null, throwError = false }) {
- let start = new Date();
- let timeLapsed;
-
- async function _retry() {
- try {
- return await fn;
- } catch (error) {
- timeLapsed = new Date() - start;
-
- await wait(interval);
-
- if (maxTime) {
- if (timeLapsed >= maxTime) {
- if (throwError) {
- throw error;
- } else {
- return null;
- }
- }
- } else {
- --retries;
-
- if (retries === 0) {
- if (throwError) {
- throw error;
- } else {
- return null;
- }
- }
- }
-
- return await _retry();
- }
- }
-
- return await _retry();
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} htmlStr
- * @returns {NodeListOf<ChildNode>}
- */
- function createElementsFromHTML(htmlStr) {
- let div = document.createElement('div');
-
- div.innerHTML = htmlStr.trim();
-
- return div.childNodes;
- }
-
- /**
- * Checks if a variable is a number.
- *
- * @param {*} variable - The variable to check.
- * @returns {boolean} - `true` if the variable is a number or a number represented as a string, `false` otherwise.
- */
- function isNumber(variable) {
- return (typeof variable == 'string' || typeof variable == 'number') && !isNaN(variable - 0) && variable !== '';
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string | number} val
- * @returns
- */
- function parseNumberSafe(val) {
- if (isNumber(val)) {
- val = parseFloat(val);
- }
-
- return val;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} num
- * @returns {boolean}
- */
- function isInt(num) {
- return Number(num) === num && num % 1 === 0;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} num
- * @returns {boolean}
- */
- function isFloat(num) {
- return Number(num) === num && num % 1 !== 0;
- }
-
- /*
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} elem
- * @param {string} prop
- * @param {Window=} _window
- * @returns {string | number | null}
- */
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} elem
- * @param {string} prop
- * @param {Window} [_window=getWindow()]
- * @returns {string | number | null}
- */
- function getStyle(elem, prop, _window = null) {
- _window = _window || getWindow();
-
- let value = parseNumberSafe(
- window
- .getComputedStyle(elem, null)
- .getPropertyValue(prop)
- .replace(/^(\d+)px$/, '$1')
- );
-
- return value;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} beforeElem
- * @param {HTMLElement=} afterElem
- */
- function attachHorizontalResizer(beforeElem, afterElem) {
- let resizer = document.createElement('span');
-
- resizer.className = 'akkd-horz-resizer';
-
- beforeElem.after(resizer);
-
- afterElem = afterElem ? afterElem : resizer.nextElementSibling;
-
- // resizer.addEventListener('mousedown', init, false);
-
- // /**
- // *
- // *
- // * @author Michael Barros <michaelcbarros@gmail.com>
- // * @param {MouseEvent} ev
- // */
- // function init(ev) {
- // getWindow().addEventListener('mousemove', resize, false);
- // getWindow().addEventListener('mouseup', stopResize, false);
- // }
-
- // /**
- // *
- // *
- // * @author Michael Barros <michaelcbarros@gmail.com>
- // * @param {MouseEvent} ev
- // */
- // function resize(ev) {
- // beforeElem.style.height = `${ev.clientY - beforeElem.offsetTop}px`;
- // }
-
- // /**
- // *
- // *
- // * @author Michael Barros <michaelcbarros@gmail.com>
- // * @param {MouseEvent} ev
- // */
- // function stopResize(ev) {
- // getWindow().removeEventListener('mousemove', resize, false);
- // getWindow().removeEventListener('mouseup', stopResize, false);
- // }
-
- let prevX = -1;
- let prevY = -1;
- let dir = null;
-
- $(resizer).on('mousedown', function (e) {
- prevX = e.clientX;
- prevY = e.clientY;
- dir = 'n'; // $(this).attr('id');
-
- $(document).on('mousemove', resize);
- $(document).on('mouseup', stopResize);
- });
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {JQuery.MouseMoveEvent<Document, undefined, Document, Document>} ev
- */
- function resize(ev) {
- if (prevX == -1) return;
-
- let boxX = $(afterElem).position().left;
- let boxY = $(afterElem).position().top;
- let boxW = $(afterElem).width();
- let boxH = $(afterElem).height();
-
- let dx = ev.clientX - prevX;
- let dy = ev.clientY - prevY;
-
- switch (dir) {
- case 'n':
- // north
- boxY += dy;
- boxH -= dy;
-
- break;
-
- case 's':
- // south
- boxH += dy;
-
- break;
-
- case 'w':
- // west
- boxX += dx;
- boxW -= dx;
-
- break;
-
- case 'e':
- // east
- boxW += dx;
-
- break;
-
- default:
- break;
- }
-
- $(afterElem).css({
- // top: boxY + 'px',
- // left: boxX + 'px',
- // width: boxW + 'px',
- height: boxH + 'px',
- });
-
- let lines = [
- //
- // ['newHeight', newHeight],
- ['clientY', ev.clientY],
- ['beforeElem.top', roundNumber($(beforeElem).position().top)],
- ['beforeElem.height', $(beforeElem).height()],
- '',
- ['afterElem.top', roundNumber($(afterElem).position().top)],
- ['afterElem.height', $(afterElem).height()],
- ];
-
- // writeDebugMsg(lines);
- console.debug([`y: ${ev.clientY}`, `b.top: ${roundNumber($(beforeElem).position().top)}`, `b.height: ${$(beforeElem).height()}`, `a.top: ${roundNumber($(afterElem).position().top)}`, `a.height: ${$(afterElem).height()}`].join(' '));
-
- function writeDebugMsg(lines) {
- let outputLines = ['*'.repeat(60)];
-
- let tags = lines.map((line) => (Array.isArray(line) ? line[0] : line));
-
- lines.forEach((line) => {
- if (Array.isArray(line)) {
- // need to require lpad-align
- // outputLines.push(`${lpadAlign(line[0], tags)}: ${line[1]}`);
- } else {
- outputLines.push(line);
- }
- });
-
- outputLines.push('*'.repeat(60));
-
- console.debug(outputLines.join('\n'));
- }
-
- function roundNumber(num, places = 2) {
- return parseFloat(parseFloat(num.toString()).toFixed(places));
- // Math.round(num * 100) / 100
- }
-
- prevX = ev.clientX;
- prevY = ev.clientY;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {JQuery.MouseMoveEvent<Document, undefined, Document, Document>} ev
- */
- function stopResize(ev) {
- prevX = -1;
- prevY = -1;
-
- $(document).off('mousemove', resize);
- $(document).off('mouseup', stopResize);
- }
- }
-
- function traceMethodCalls(obj) {
- /** @type {ProxyHandler} */
- let handler = {
- get(target, propKey, receiver) {
- if (propKey == 'isProxy') return true;
-
- const prop = target[propKey];
-
- if (typeof prop == 'undefined') return;
-
- if (typeof prop === 'object' && target[propKey] !== null) {
- if (!prop.isProxy) {
- target[propKey] = new Proxy(prop, handler);
-
- return target[propKey];
- } else {
- return target[propKey];
- }
- }
-
- if (typeof target[propKey] == 'function') {
- const origMethod = target[propKey];
-
- return function (...args) {
- let result = origMethod.apply(this, args);
-
- console.log(propKey + JSON.stringify(args) + ' -> ' + JSON.stringify(result));
-
- return result;
- };
- } else {
- return target[propKey];
- }
- },
- };
-
- return new Proxy(obj, handler);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} elem
- * @param {number} [topOffset=0]
- * @returns {boolean}
- */
- function isVisible(elem, topOffset = 0) {
- /**
- * Checks if a DOM element is visible. Takes into
- * consideration its parents and overflow.
- *
- * @param {HTMLElement} el the DOM element to check if is visible
- *
- * These params are optional that are sent in recursively,
- * you typically won't use these:
- *
- * @param {number} top Top corner position number
- * @param {number} right Right corner position number
- * @param {number} bottom Bottom corner position number
- * @param {number} left Left corner position number
- * @param {number} width Element width number
- * @param {number} height Element height number
- * @returns {boolean}
- */
- function _isVisible(el, top, right, bottom, left, width, height) {
- let parent = el.parentNode;
- let VISIBLE_PADDING = 2;
-
- if (!_elementInDocument(el)) {
- return false;
- }
-
- // Return true for document node
- if (9 === parent.nodeType) {
- return true;
- }
-
- // Return false if our element is invisible
- if ('0' === _getStyle(el, 'opacity') || 'none' === _getStyle(el, 'display') || 'hidden' === _getStyle(el, 'visibility')) {
- return false;
- }
-
- if ('undefined' === typeof top || 'undefined' === typeof right || 'undefined' === typeof bottom || 'undefined' === typeof left || 'undefined' === typeof width || 'undefined' === typeof height) {
- top = el.offsetTop + topOffset;
- left = el.offsetLeft;
- bottom = top + el.offsetHeight;
- right = left + el.offsetWidth;
- width = el.offsetWidth;
- height = el.offsetHeight;
- }
-
- // If we have a parent, let's continue:
- if (parent) {
- // Check if the parent can hide its children.
- if ('hidden' === _getStyle(parent, 'overflow') || 'scroll' === _getStyle(parent, 'overflow')) {
- // Only check if the offset is different for the parent
- if (
- // If the target element is to the right of the parent elm
- left + VISIBLE_PADDING > parent.offsetWidth + parent.scrollLeft ||
- // If the target element is to the left of the parent elm
- left + width - VISIBLE_PADDING < parent.scrollLeft ||
- // If the target element is under the parent elm
- top + VISIBLE_PADDING > parent.offsetHeight + parent.scrollTop ||
- // If the target element is above the parent elm
- top + height - VISIBLE_PADDING < parent.scrollTop
- ) {
- // Our target element is out of bounds:
- return false;
- }
- }
- // Add the offset parent's left/top coords to our element's offset:
- if (el.offsetParent === parent) {
- left += parent.offsetLeft;
- top += parent.offsetTop;
- }
- // Let's recursively check upwards:
- return _isVisible(parent, top, right, bottom, left, width, height);
- }
-
- return true;
- }
-
- // Cross browser method to get style properties:
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} el
- * @param {string} property
- * @returns
- */
- function _getStyle(el, property) {
- let value;
-
- if (window.getComputedStyle) {
- value = document.defaultView.getComputedStyle(el, null)[property];
- }
-
- if (el.currentStyle) {
- value = el.currentStyle[property];
- }
-
- return value;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} element
- * @returns {boolean}
- */
- function _elementInDocument(element) {
- while ((element = element.parentNode)) {
- if (element == document) {
- return true;
- }
- }
-
- return false;
- }
-
- return _isVisible(elem);
- }
-
- /**
- * @summary
- * High-order function that memoizes a function, by creating a scope
- * to store the result of each function call, returning the cached
- * result when the same inputs is given.
- *
- * @description
- * Memoization is an optimization technique used primarily to speed up
- * functions by storing the results of expensive function calls, and returning
- * the cached result when the same inputs occur again.
- *
- * Each time a memoized function is called, its parameters are used as keys to index the cache.
- * If the index (key) is present, then it can be returned, without executing the entire function.
- * If the index is not cached, then all the body of the function is executed, and the result is
- * added to the cache.
- *
- * @see https://www.sitepoint.com/implementing-memoization-in-javascript/
- *
- * @export
- * @param {Function} func: function to memoize
- * @returns {Function}
- */
- function memoize(func) {
- const cache = {};
-
- function memoized(...args) {
- const key = JSON.stringify(args);
-
- if (key in cache) return cache[key];
-
- if (globalThis instanceof this.constructor) {
- return (cache[key] = func.apply(null, args));
- } else {
- return (cache[key] = func.apply(this, args));
- }
- }
-
- memoized.toString = () => func.toString();
-
- return memoized;
- }
-
- function memoizeClass(clazz, options = { toIgnore: [] }) {
- let funcs = getObjProps(clazz, { namesOnly: true }).funcs;
-
- for (let i = 0; i < funcs.length; i++) {
- let funcName = funcs[i];
-
- if (options.toIgnore.includes(funcName)) continue;
-
- let func = Object.getOwnPropertyDescriptor(clazz.prototype, funcName);
-
- let memFunc = memoize(func.value);
-
- Object.defineProperty(clazz.prototype, funcName, {
- get: function () {
- return memFunc;
- },
- });
- }
-
- let props = getObjProps(clazz, { namesOnly: true }).props;
-
- for (let i = 0; i < props.length; i++) {
- let propName = props[i];
-
- if (options.toIgnore.includes(propName)) continue;
-
- let prop = Object.getOwnPropertyDescriptor(clazz.prototype, propName);
- let cacheKey = `_${propName}-cache_`;
-
- Object.defineProperty(clazz.prototype, propName, {
- get: function () {
- return (this[cacheKey] = this[cacheKey] || prop.get.call(this));
- },
- });
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {any[]} objects
- * @returns {object}
- */
- function merge(...objects) {
- const isObject = (obj) => Object.prototype.toString.call(obj) == '[object Object]' && obj.constructor && obj.constructor.name == 'Object';
-
- let _merge = (_target, _source, _isMergingArrays) => {
- if (!isObject(_target) || !isObject(_source)) {
- return _source;
- }
-
- Object.keys(_source).forEach((key) => {
- const targetValue = _target[key];
- const sourceValue = _source[key];
-
- if (Array.isArray(targetValue) && Array.isArray(sourceValue)) {
- if (_isMergingArrays) {
- _target[key] = targetValue.map((x, i) => (sourceValue.length <= i ? x : _merge(x, sourceValue[i], _isMergingArrays)));
-
- if (sourceValue.length > targetValue.length) {
- _target[key] = _target[key].concat(sourceValue.slice(targetValue.length));
- }
- } else {
- _target[key] = targetValue.concat(sourceValue);
- }
- } else if (isObject(targetValue) && isObject(sourceValue)) {
- _target[key] = _merge(Object.assign({}, targetValue), sourceValue, _isMergingArrays);
- } else {
- _target[key] = sourceValue;
- }
- });
-
- return _target;
- };
-
- const isMergingArrays = typeof objects[objects.length - 1] == 'boolean' ? objects.pop() : false;
-
- if (objects.length < 2) throw new Error('mergeEx: this function expects at least 2 objects to be provided');
-
- if (objects.some((object) => !isObject(object))) throw new Error('mergeEx: all values should be of type "object"');
-
- const target = objects.shift();
- let source;
-
- while ((source = objects.shift())) {
- _merge(target, source, isMergingArrays);
- }
-
- return target;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {Window} [windowOrFrame=getTopWindow()]
- * @param {Window[]} [allFrameArray=[]]
- * @returns {Window[]}
- */
- function getAllFrames(windowOrFrame = getTopWindow(), allFrameArray = []) {
- allFrameArray.push(windowOrFrame.frames);
-
- for (var i = 0; i < windowOrFrame.frames.length; i++) {
- getAllFrames(windowOrFrame.frames[i], allFrameArray);
- }
-
- return allFrameArray;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {boolean}
- */
- function windowIsFocused() {
- let frames = getAllFrames();
-
- for (let i = 0; i < frames.length; i++) {
- const frame = frames[i];
-
- try {
- if (frame.document.hasFocus()) {
- return true;
- }
- } catch (error) {}
- }
-
- return false;
- }
-
- function setupWindowHasFocused() {
- let frames = getAllFrames();
-
- for (let i = 0; i < frames.length; i++) {
- const frame = frames[i];
-
- try {
- frame.hasFocus = windowIsFocused;
- } catch (error) {}
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {[]} array
- * @param {any} element
- * @param {number} index
- */
- function moveArrayElement(array, filter, index) {
- let item = array.filter((item) => item === filter)[0];
-
- if (item) {
- array = array.filter((item) => item !== filter);
-
- array.unshift(item);
- }
-
- return array;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} elem
- * @param {(elem: HTMLElement, level: number) => void} callback
- * @param {number} [level=0]
- */
- function walkDom(elem, callback, level = 0) {
- let children = elem.children;
-
- callback(elem, level);
-
- for (let i = 0; i < children.length; i++) {
- /** @type {HTMLElement} */
- let child = children[i];
-
- walkDom(child, callback, level + 1);
-
- if (child.shadowRoot) {
- walkDom(child.shadowRoot, callback, level + 2);
- }
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} obj
- * @param {(key: string | number, value: any, keyPath: string, callbackRes: { doBreak: boolean, returnValue: any | null }, obj: T) => boolean} callback
- * @template T
- * @returns {{ dottedObj: T, returnValue: any }}
- */
- function walkObj(obj, callback) {
- let callbackRes = {
- doBreak: false,
- returnValue: null,
- };
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{}} _obj
- * @param {string[]} keyPath
- * @param {{}} newObj
- */
- function _walk(_obj, keyPath, newObj) {
- keyPath = typeof keyPath === 'undefined' ? [] : keyPath;
- newObj = typeof newObj === 'undefined' ? {} : newObj;
-
- for (let key in _obj) {
- if (_obj.hasOwnProperty(key)) {
- let value = _obj[key];
-
- keyPath.push(key);
-
- callback.apply(this, [key, value, keyPath.join('.'), callbackRes, obj]);
-
- if (typeof value === 'object' && value !== null) {
- newObj = _walk(value, keyPath, newObj);
- } else {
- let newKey = keyPath.join('.');
-
- newObj[newKey] = value;
- }
-
- keyPath.pop();
-
- if (callbackRes.doBreak) {
- break;
- }
- }
- }
-
- return newObj;
- }
-
- let newObj = _walk(obj);
-
- return {
- dottedObj: newObj,
- returnValue: callbackRes.returnValue,
- };
- }
-
- /**
- * A function to take a string written in dot notation style, and use it to
- * find a nested object property inside of an object.
- *
- * Useful in a plugin or module that accepts a JSON array of objects, but
- * you want to let the user specify where to find various bits of data
- * inside of each custom object instead of forcing a standardized
- * property list.
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{}} obj
- * @param {string} dotPath
- * @returns {*}
- */
- function getNestedDot(obj, dotPath) {
- let parts = dotPath.split('.');
- let length = parts.length;
- let property = obj || this;
-
- for (let i = 0; i < length; i++) {
- property = property[parts[i]];
- }
-
- return property;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{}} obj
- * @param {number} maxLevel
- */
- function getDottedObj(obj, maxLevel = 50) {
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {{}} _obj
- * @param {string[]} keyPath
- * @param {{}} newObj
- * @param {number} level
- */
- function _worker(_obj, keyPath, newObj, level = 0) {
- keyPath = typeof keyPath === 'undefined' ? [] : keyPath;
- newObj = typeof newObj === 'undefined' ? {} : newObj;
-
- for (let key in _obj) {
- if (_obj.hasOwnProperty(key)) {
- let value = _obj[key];
-
- keyPath.push(key);
-
- if (typeof value === 'object' && value !== null) {
- newObj = _worker(value, keyPath, newObj, level++);
- } else {
- let newKey = keyPath.join('.');
-
- newObj[newKey] = value;
- }
-
- keyPath.pop();
-
- if (maxLevel > 0 && level >= maxLevel) {
- break;
- }
- }
- }
-
- return newObj;
- }
-
- let dottedObj = _worker(obj);
-
- return dottedObj;
- }
-
- function isNode() {
- return !(typeof window !== 'undefined' && typeof window.document !== 'undefined');
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {number}
- */
- function getCurrentTimeMs() {
- if (isNode()) {
- const NS_PER_MS = 1e6;
-
- let time = process.hrtime();
-
- return time[0] * 1000 + time[1] / NS_PER_MS;
- } else {
- return performance.now();
- }
- }
-
- const intervalIDsMap = new Map();
- const timeoutIDsMap = new Map();
-
- /**
- * Schedules the repeated execution of a function (callback) with a fixed time delay between each call.
- * @param {TimerHandler} handler - A function to be executed repeatedly.
- * @param {number} [timeout] - The time, in milliseconds, between each function call. Default is 0.
- * @param {...any} [args] - Additional arguments to be passed to the function.
- * @returns {number} - An identifier representing the interval. This value can be used with clearInterval to cancel the interval.
- */
- function setIntervalEx(handler, timeout, ...args) {
- if (isNode()) {
- return setInterval(handler, timeout, ...args);
- } else {
- let startTime = getCurrentTimeMs();
- let elapsedTime = 0;
- /** @type {number} */
- let intervalId;
- let intervalIdTemp;
-
- function loop(currentTime) {
- if (intervalIDsMap.get(intervalId)) {
- const deltaTime = currentTime - startTime;
-
- elapsedTime += deltaTime;
-
- if (elapsedTime >= timeout) {
- handler(...args);
-
- elapsedTime = 0;
- }
-
- startTime = currentTime;
-
- intervalIdTemp = window.requestAnimationFrame(loop);
- } else {
- window.cancelAnimationFrame(intervalIdTemp);
- intervalIDsMap.delete(intervalId);
- }
- }
-
- intervalId = window.requestAnimationFrame(loop);
-
- intervalIDsMap.set(intervalId, true);
-
- return intervalId;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} intervalId
- */
- function clearIntervalEx(intervalId) {
- if (isNode()) {
- clearInterval(intervalId);
- } else {
- intervalIDsMap.set(intervalId, false);
-
- window.cancelAnimationFrame(intervalId);
- }
- }
-
- /**
- * Schedules the execution of a function (callback) after a specified time delay.
- * @param {TimerHandler} handler - A function to be executed.
- * @param {number} [timeout] - The time, in milliseconds, to wait before executing the function. Default is 0.
- * @param {...any} [args] - Additional arguments to be passed to the function.
- * @returns {number} - An identifier representing the timeout. This value can be used with clearTimeout to cancel the timeout.
- */
- function setTimeoutEx(handler, timeout, ...args) {
- if (isNode()) {
- return setTimeout(handler, timeout, ...args);
- } else {
- let startTime = getCurrentTimeMs();
- /** @type {number} */
- let timeoutId;
- let timeoutIdTemp;
-
- function loop(currentTime) {
- if (timeoutIDsMap.get(timeoutId)) {
- const deltaTime = currentTime - startTime;
-
- if (deltaTime >= timeout) {
- handler(...args);
- } else {
- timeoutIdTemp = window.requestAnimationFrame(loop);
- }
- } else {
- window.cancelAnimationFrame(timeoutIdTemp);
- timeoutIDsMap.delete(timeoutId);
- }
- }
-
- timeoutId = window.requestAnimationFrame(loop);
-
- timeoutIDsMap.set(timeoutId, true);
-
- return timeoutId;
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} timeoutId
- */
- function clearTimeoutEx(timeoutId) {
- if (isNode()) {
- clearTimeout(timeoutId);
- } else {
- timeoutIDsMap.set(timeoutId, false);
-
- window.cancelAnimationFrame(timeoutId);
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} attribute
- * @param {string} value
- * @param {string} elementType
- * @returns {HTMLElement[]}
- */
- function findByAttributeValue(attribute, value, elementType) {
- elementType = elementType || '*';
-
- let all = document.getElementsByTagName(elementType);
- let foundElements = [];
-
- for (let i = 0; i < all.length; i++) {
- if (all[i].getAttribute(attribute).includes(value)) {
- foundElements.push(all[i]);
- }
- }
-
- return foundElements;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {number}
- */
- function getLocalStorageSize() {
- let total = 0;
-
- for (let x in localStorage) {
- // Value is multiplied by 2 due to data being stored in `utf-16` format, which requires twice the space.
- let amount = localStorage[x].length * 2;
-
- if (!isNaN(amount) && localStorage.hasOwnProperty(x)) {
- total += amount;
- }
- }
-
- return total;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} bytes
- * @param {boolean} [si=false]
- * @returns {string}
- */
- function bytes2HumanReadable(bytes, si = false) {
- let thresh = si ? 1000 : 1024;
-
- if (Math.abs(bytes) < thresh) {
- return bytes + ' B';
- }
-
- let units = si ? ['kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'] : ['KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB', 'ZiB', 'YiB'];
- let u = -1;
-
- while (true) {
- bytes = bytes / thresh;
- u++;
-
- if (!(Math.abs(bytes) >= thresh && u < units.length - 1)) {
- break;
- }
- }
-
- return bytes.toFixed(1) + ' ' + units[u];
- }
-
- async function GM_fetch(url, fetchInit = {}) {
- if (!window.GM_xmlhttpRequest) {
- console.warn('GM_xmlhttpRequest not required. Using native fetch.');
-
- return await fetch(url, fetchInit);
- }
-
- let parseHeaders = function (headersString) {
- const headers = new Headers();
-
- for (const line of headersString.trim().split('\n')) {
- const [key, ...valueParts] = line.split(':');
-
- if (key) {
- headers.set(key.trim().toLowerCase(), valueParts.join(':').trim());
- }
- }
-
- return headers;
- };
-
- const defaultFetchInit = { method: 'get' };
- const { headers, method } = { ...defaultFetchInit, ...fetchInit };
- const isStreamSupported = GM_xmlhttpRequest?.RESPONSE_TYPE_STREAM;
- const HEADERS_RECEIVED = 2;
-
- if (!isStreamSupported) {
- return new Promise((resolve, _reject) => {
- const blobPromise = new Promise((resolve, reject) => {
- GM_xmlhttpRequest({
- url,
- method,
- headers,
- responseType: 'blob',
- onload: async (response) => resolve(response.response),
- onerror: reject,
- onreadystatechange: onHeadersReceived,
- });
- });
-
- blobPromise.catch(_reject);
-
- function onHeadersReceived(gmResponse) {
- const { readyState, responseHeaders, status, statusText } = gmResponse;
-
- if (readyState === HEADERS_RECEIVED) {
- const headers = parseHeaders(responseHeaders);
-
- resolve({
- headers,
- status,
- statusText,
- arrayBuffer: () => blobPromise.then((blob) => blob.arrayBuffer()),
- blob: () => blobPromise,
- json: () => blobPromise.then((blob) => blob.text()).then((text) => JSON.parse(text)),
- text: () => blobPromise.then((blob) => blob.text()),
- });
- }
- }
- });
- } else {
- return new Promise((resolve, _reject) => {
- const responsePromise = new Promise((resolve, reject) => {
- void GM_xmlhttpRequest({
- url,
- method,
- headers,
- responseType: 'stream',
- onerror: reject,
- onreadystatechange: onHeadersReceived,
- // onloadstart: (gmResponse) => logDebug('[onloadstart]', gmResponse), // debug
- });
- });
-
- responsePromise.catch(_reject);
-
- function onHeadersReceived(gmResponse) {
- const { readyState, responseHeaders, status, statusText, response: readableStream } = gmResponse;
-
- if (readyState === HEADERS_RECEIVED) {
- const headers = parseHeaders(responseHeaders);
- let newResp;
-
- if (status === 0) {
- newResp = new Response(readableStream, { headers /*status, statusText*/ });
- } else {
- newResp = new Response(readableStream, { headers, status, statusText });
- }
-
- resolve(newResp);
- }
- }
- });
- }
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {T} items
- * @param {{name: string, desc: boolean, case_sensitive: boolean}[]} columns
- * @param {{cmpFunc: any}} [{ cmpFunc = cmp }={}]
- * @returns {T}
- * @template T
- */
- function multiKeySort(items, columns, { cmpFunc = null } = {}) {
- function cmp(a, b) {
- if (a < b) {
- return -1;
- } else {
- if (a > b) {
- return 1;
- } else {
- return 0;
- }
- }
- }
-
- cmpFunc = cmpFunc != null ? cmpFunc : cmp;
-
- let comparers = [];
-
- columns.forEach((col) => {
- let column = col.name;
- let desc = 'desc' in col ? col.desc : false;
- let case_sensitive = 'case_sensitive' in col ? col.case_sensitive : true;
-
- comparers.push([column, desc ? -1 : 1, case_sensitive]);
- });
-
- function comparer(left, right) {
- for (let i = 0; i < comparers.length; i++) {
- const column = comparers[i][0];
- const polarity = comparers[i][1];
- const case_sensitive = comparers[i][2];
-
- let result = 0;
-
- if (case_sensitive) {
- result = cmpFunc(left[column], right[column]);
- } else {
- result = cmpFunc(left[column].toLowerCase(), right[column].toLowerCase());
- }
-
- if (result) {
- return polarity * result;
- }
- }
-
- return 0;
- }
-
- return items.sort(comparer);
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} text
- * @param {string} [nodeType='div']
- * @returns {HTMLElement}
- */
- function getElementByTextContent(text, nodeType = 'div') {
- let xpath = `//${nodeType}[text()='${text}']`;
-
- /** @type {HTMLElement} */
- let elem = document.evaluate(xpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
-
- return elem;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} text
- * @param {string} [nodeType='div']
- * @returns {HTMLElement}
- */
- function getElementByTextContentContains(text, nodeType = 'div') {
- let xpath = `//${nodeType}[contains(text(),'${text}')]`;
-
- /** @type {HTMLElement} */
- let elem = document.evaluate(xpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
-
- return elem;
- }
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {HTMLElement} element
- * @param {(elem: HTMLElement) => void} callback
- */
- function onRemove(element, callback) {
- const parent = element.parentNode;
-
- if (!parent) {
- throw new Error('The node must already be attached');
- }
-
- const observer = new MutationObserver((mutations) => {
- let removed = false;
-
- for (const mutation of mutations) {
- for (const node of mutation.removedNodes) {
- if (node === element) {
- observer.disconnect();
-
- callback(element);
-
- removed = true;
-
- break;
- }
- }
-
- if (removed) {
- break;
- }
- }
- });
-
- observer.observe(parent, {
- childList: true,
- });
- }
-
- // #endregion Helper Functions
-
- // #region Prototype Functions
-
- // #region Array
-
- /**
- * Function to setup custom prototype functions for the Array class
- * (For no conflicts, Object.defineProperty must be used)
- *
- */
- function setupArrayPrototypes() {
- let funcs = [
- function pushUnique(item) {
- let index = -1;
-
- for (let i = 0; i < this.length; i++) {
- if (equals(this[i], item)) index = i;
- }
-
- if (index === -1) this.push(item);
- },
- ];
-
- for (let i = 0; i < funcs.length; i++) {
- let func = funcs[i];
-
- Object.defineProperty(Array.prototype, func.name, {
- enumerable: false,
- configurable: true,
- writable: true,
- value: func,
- });
- }
- }
-
- setupArrayPrototypes();
-
- // #endregion Array
-
- // #endregion Prototype Functions
-
- // #region jQuery
-
- function setupJqueryExtendedFuncs() {
- if ('jQuery' in getWindow() || 'jQuery' in window) {
- jQuery.fn.extend({
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {boolean}
- * @this {JQuery<HTMLElement>}
- */
- exists: function exists() {
- return this.length !== 0;
- },
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {() => void} callback
- * @returns {JQuery<HTMLElement>}
- * @this {JQuery<HTMLElement>}
- */
- ready: function ready(callback) {
- let cb = function cb() {
- return setTimeout(callback, 0, jQuery);
- };
-
- if (document.readyState !== 'loading') {
- cb();
- } else {
- document.addEventListener('DOMContentLoaded', cb);
- }
-
- return this;
- },
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {string} method
- * @param {{}} options
- * @returns {number | string | null}
- * @this {JQuery<HTMLElement>}
- */
- actual: function actual(method, options) {
- // check if the jQuery method exist
- if (!this[method]) {
- throw '$.actual => The jQuery method "' + method + '" you called does not exist';
- }
-
- let defaults = {
- absolute: false,
- clone: false,
- includeMargin: false,
- display: 'block',
- };
-
- let configs = jQuery.extend(defaults, options);
-
- let $target = this.eq(0);
- let fix;
- let restore;
-
- if (configs.clone === true) {
- fix = function () {
- let style = 'position: absolute !important; top: -1000 !important; ';
-
- // this is useful with css3pie
- $target = $target.clone().attr('style', style).appendTo('body');
- };
-
- restore = function () {
- // remove DOM element after getting the width
- $target.remove();
- };
- } else {
- let tmp = [];
- let style = '';
- let $hidden;
-
- fix = function () {
- // get all hidden parents
- $hidden = $target.parents().addBack().filter(':hidden');
- style += 'visibility: hidden !important; display: ' + configs.display + ' !important; ';
-
- if (configs.absolute === true) {
- style += 'position: absolute !important; ';
- }
-
- // save the origin style props
- // set the hidden el css to be got the actual value later
- $hidden.each(function () {
- // Save original style. If no style was set, attr() returns undefined
- let $this = jQuery(this);
- let thisStyle = $this.attr('style');
-
- tmp.push(thisStyle);
-
- // Retain as much of the original style as possible, if there is one
- $this.attr('style', thisStyle ? thisStyle + ';' + style : style);
- });
- };
-
- restore = function () {
- // restore origin style values
- $hidden.each(function (i) {
- let $this = jQuery(this);
- let _tmp = tmp[i];
-
- if (_tmp === undefined) {
- $this.removeAttr('style');
- } else {
- $this.attr('style', _tmp);
- }
- });
- };
- }
-
- fix();
- // get the actual value with user specific methed
- // it can be 'width', 'height', 'outerWidth', 'innerWidth'... etc
- // configs.includeMargin only works for 'outerWidth' and 'outerHeight'
- let actual = /(outer)/.test(method) ? $target[method](configs.includeMargin) : $target[method]();
-
- restore();
- // IMPORTANT, this plugin only return the value of the first element
- return actual;
- },
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @returns {DOMRect}
- * @this {JQuery<HTMLElement>}
- */
- rect: function rect() {
- return this[0].getBoundingClientRect();
- },
-
- /**
- *
- *
- * @author Michael Barros <michaelcbarros@gmail.com>
- * @param {number} levels
- * @returns {JQuery<HTMLElement>}
- * @this {JQuery<HTMLElement>}
- */
- parentEx: function parentEx(levels = 1) {
- let parent = this;
-
- for (let i = 0; i < levels; i++) {
- if (parent.parent().length == 0) {
- break;
- }
-
- parent = parent.parent();
- }
-
- return parent;
- },
- });
- }
- }
-
- setupJqueryExtendedFuncs();
-
- // #endregion jQuery