您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
none
// ==UserScript== // @name lib:allfuncs // @version 27 // @description none // @run-at document-start // @author rssaromeo // @license GPLv3 // @match *://*/* // @include * // @icon data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEgAAABICAMAAABiM0N1AAAAAXNSR0IB2cksfwAAAAlwSFlzAAAOxAAADsQBlSsOGwAAAHJQTFRFAAAAEIijAo2yAI60BYyuF4WaFIifAY6zBI2wB4usGIaZEYigIoiZCIyrE4igG4iYD4mjEomhFoedCoqpDIqnDomlBYyvE4efEYmiDYqlA42xBoytD4mkCYqqGYSUFYidC4qoC4upAo6yCoupDYqmCYur4zowOQAAACZ0Uk5TAO////9vr////1+/D/+/L+/Pf/////+f3///////H4////////+5G91rAAACgUlEQVR4nM2Y22KjIBCGidg1264liZqDadK03X3/V2wNKHMC7MpF/xthHD5mgERAqZhWhfYqH6K+Qf2qNNf625hCoFj9/gblMUi5q5jLkXLCKudgyiRm0FMK82cWJp1fLbV5VmvJbCIc0GCYaFqqlDJgADdBjncqAXYobm1xh72aFMflbysteFfdy2Yi1XGOm5HGBzQ1dq7TzEoxjeNTjQZb7VA3e1c7+ImgasAgQ9+xusNVNZIo5xmOMgihIS2PbCQIiHEUdTvhxCcS/kPomfFI2zHy2PkWmA6aNatIJpKFJyekyy02xh5Y3DI9T4aOT6VhIUrsNTFp1pf79Z4SIIVDegl6IJO6cHiL/GimIZDhgTu/BlYWCQzHMl0zBWT/T3KAhtxOuUB9FtBrpsz0RV4xsjHmW+UCaffcSy/5viMGer0/6HdFNMZBq/vjJL38H9Dqx4Fuy0Em12DbZy+9pGtiDijbglwAehyj11n0tRD3WUBm+lwulE/8h4BuA+iWAQQnteg2Xm63WQLTpnMnpjdge0Mgu/GRPsV4xdjQ94Lfi624fabhDkfUqIKNrM64Q837v8yL0prasepCgrtvw1sJpoqanGEX7b5mQboNW8eawXaWXTMfMGxub472hzWzHSn6Sg2G9+6TAyRruE71s+zAzjWaknoyJCQzwxrghH2k5FDT4eqWunuNxyN9QCGcxVod5oADbYnIUkDTGZEf1xDJnSFteQ3KdsT8zYDMQXcHxsevcLH1TrsABzkNPyA/L7b0jg704viMMlpQI96WsHknCt/3YH0kOEo9zcGkwrFK39ck72rmoehmKqo2RKlilzSy/nJKEV45CT38myJp456fezktHjN5aeMAAAAASUVORK5CYII= // @grant none // @namespace https://greasyfork.org/users/1184528 // ==/UserScript== // fix gettype in test func /* // @noregex // @name remove ai comments // @regex (//) \.\.\..* // @replace // @endregex // @regex maketype\((\w+), // @replace $1 = maketype($1, // @endregex */ /** * @typedef {Object} TestUtils * @property {function(...*): void} ifunset - Sets default values for undefined inputs. * @property {function(*): string} gettype - Checks the type of a value. * @property {function(number): void} pass - Validates that the current parameter is correct. * @property {function(): boolean} end - Validates that all parameters have been checked. * @property {function(*, Array<*>): boolean} isany - Checks if a value is in a list of options. * @property {function(string, RegExp): boolean} matchesregex - Checks if a value matches a regex pattern. * @property {function(*, *): boolean} issame - Checks if two values are strictly equal. * @property {function(number, Array<string>): boolean} trymaketype - Attempts to convert a value to one of the specified types. */ /** * @callback TestFunction * @param {TestUtils} testUtils - An object containing utility functions for testing. * @returns {boolean} Returns true if all validations pass. */ /** * @param {Function} func - The function to wrap in a test. * @param {TestFunction} testfunc - The function to test func against. * @returns {Function} - The wrapped function. */ ;(() => { function newfunc(func, testfunc) { var funcname = func.name || func.prototype.name var retfunc = function () { var i = 0 var inputargs = [...arguments] return testfunc({ funcname, args: inputargs, /** * * @param {*} thing - anything * @param {Array} _enum - list of posable things thing can be * @returns {Error|*} */ makeenum: function makeenum(thing, _enum) { for (var thing2 of _enum) { if (thing == thing2) return thing2 } throw new Error( `makeenum: in function ${this.funcname} input ${i} is invalid, should match enum [${_enum}] but is instead ${thing}` ) }.bind(retfunc), trymakeenum: function trymakeenum(thing, _enum) { try { this.makeenum(thing, _enum) return true } catch (e) { return false } }.bind(retfunc), /** * Sets default values for input if they are undefined. * @param {...*} input - The default values to set. */ ifunset: function ifunset(newinputs) { for (var i in newinputs) { if (inputargs[i] === undefined) inputargs[i] = newinputs[i] } }.bind(retfunc), /** * * @param {Number} i - tries to make thing become a type in types * @param {Array|string} types - array of types to try to be * @returns {boolean} */ trymaketype: function trymaketype(thing, types) { try { this.maketype(thing, types) return true } catch (e) { return false } }.bind(retfunc), /** * * @param {Number} i - tries to make thing become a type in types * @param {Array|string} types - array of types to try to be * @returns {Error|true} */ maketype: function maketype(thing, types) { // if (types.includes("any")) return true if (gt(thing, types)) return thing for (var type of types) { if (gt(thing, "string") && type == "number") { thing = Number(thing) } if (gt(thing, "number") && type == "string") { thing = String(thing) } if (gt(thing, "string") && type == "number") { thing = Number(thing) } if (gt(thing, "number") && type == "string") { thing = String(thing) } if (gt(thing, "boolean") && type == 1) { thing = true } if (type == "boolean" && (thing == 0 || thing == "")) { thing = false } if (gt(thing, type)) return thing } throw new Error( `trymaketype: in function ${ this.funcname } input ${i} is invalid, should be type [${types}] but is instead type ${gt( thing )}` ) }.bind(retfunc), /** * * @param {*} thing - the thing to get the type of * @param {string|Array|undefined} match - if set the type to check aganst or array of ytpes to check aganst * @returns {boolean|string} */ trygettype: gt, gettype: function gettype(a, s) { if (gt(a, s)) return true throw new Error( `gettype: in function ${ this.funcname } input ${i} is invalid, should be type [${s}] but is instead type ${gettype( a )}` ) }.bind(retfunc), /** * tells that the test function has ended and to try and call the main function * @returns {boolean|undefined} */ end: function end() { return func.call(func, ...inputargs) }.bind(retfunc), }) } retfunc.name = retfunc.prototype.name = "strict " + (funcname ?? "function") retfunc.funcname = funcname return retfunc.bind(retfunc) /** * * @param {*} thing - the thing to get the type of * @param {string|Array|undefined} match - if set the type to check aganst or array of ytpes to check aganst * @returns {boolean|string} */ function gt(thing, match) { if ( !match || (Object.prototype.toString .call(match) .toLowerCase() .match(/^\[[a-z]+ (.+)\]$/)[1] == "string" && !match.includes("|")) ) { var type = Object.prototype.toString .call(thing) .toLowerCase() .match(/^\[[a-z]+ (.+)\]$/)[1] if (type !== "function") if (type == match) return true if (match == "normalfunction") return type == "function" if (type == "htmldocument" && match == "document") return true if (match == "body" && type == "htmlbodyelement") return true if (match && new RegExp(`^html${match}element$`).test(type)) return true if (/^html\w+element$/.test(type)) type = "element" if (type == "htmldocument") type = "element" if (type == "asyncfunction") type = "function" if (type == "generatorfunction") type = "function" if (type == "regexp") type = "regex" if (match == "regexp") match = "regex" if (match == "element" && type == "window") return true if (match == "element" && type == "shadowroot") return true if (match == "event" && /\w+event$/.test(type)) return true if (/^(html|svg).*element$/.test(type)) type = "element" if (type == "function") { type = /^\s*class\s/.test( Function.prototype.toString.call(thing) ) ? "class" : "function" } if (match == "none") return ( type == "nan" || type == "undefined" || type == "null" ) try { if (type === "number" && isNaN(thing) && match == "nan") return true } catch (e) { error(thing) } return match ? match === type : type } else { if (match.includes("|")) match = match.split("|") match = [...new Set(match)] return !!match.find((e) => gt(thing, e)) } } } const a = {} var x = newfunc( function foreach(arr, func) { var type = a.gettype(arr) if (type == "array") arr.forEach(func) else if (type == "object") { Reflect.ownKeys(arr).forEach((e, i) => { func(e, arr[e], i) }) } else { ;[arr].forEach(func) } }, ({ args: [arr, func], end, maketype }) => { arr = maketype(arr, ["array", "object", "any"]) func = maketype(func, ["function"]) return end() } ) a.wait = newfunc( function wait(ms) { return new Promise(function (done) { var last = Date.now() setTimeout(() => done(Date.now() - last - ms), ms) }) }, function ({ ifunset, end, args: [ms], maketype }) { ifunset([0]) ms = maketype(ms, ["number"]) return end() } ) a.waituntil = newfunc( function waituntil(q, cb) { return new Promise((resolve) => { var last = Date.now() var int = setInterval( function (q, cb) { if (!!q()) { clearInterval(int) try { cb(Date.now() - last) } catch (e) {} resolve(Date.now() - last) } }, 0, q, cb ) }) }, function ({ end, args: [q, cb], maketype }) { q = maketype(q, ["function"]) cb = maketype(cb, ["function", "undefined"]) return end() } ) a.keeponlyone = newfunc( function keeponlyone(arr) { return [...new Set(arr)] }, function ({ end, args: [arr], maketype }) { arr = maketype(arr, ["array"]) return end() } ) a.matchall = newfunc( function matchall(x, y) { return [...x.matchAll(y)].map((e) => e[1] !== undefined ? [...e] : e[0] ) }, function ({ args: [x, y], end, maketype }) { x = maketype(x, ["string"]) y = maketype(y, ["regex"]) return end() } ) a.randfrom = newfunc( function randfrom(min, max) { if (max === undefined) return min.length ? min[this(0, min.length - 1)] : this(0, min) if (min == max) return min if (max) return Math.round(Math.random() * (max - min)) + min return min[Math.round(Math.random() * (min.length - 1))] }, function ({ args: [min, max], end, maketype }) { min = maketype(min, ["number", "array"]) max = maketype(max, ["number", "undefined"]) return end() } ) a.foreach = newfunc( function foreach( //none arr, //array|object|any func //function ) { var type = a.gettype(arr) if (type == "array") arr.forEach(func) else if (type == "object") { Reflect.ownKeys(arr).forEach((e, i) => { func(e, arr[e], i) }) } else { ;[arr].forEach(func) } }, function ({ args: [arr, func], end, maketype }) { func = maketype(func, ["function"]) return end() } ) a.listen = newfunc( function listen(elem, type, cb, istrue = false) { var all = [] if (a.gettype(elem, "array")) { return [...elem].map(listen).flat() } else { return listen(elem) } function listen(elem) { if (a.gettype(type, "array")) { var temp = {} a.foreach(type, (e) => (temp[e] = cb)) type = temp } if (a.gettype(type, "object")) { istrue = cb a.foreach(type, function (type, cb) { if (a.gettype(type, "string")) type = a.matchall(type, /[a-z]+/g) type.forEach((type) => { const newcb = function (...e) { cb.call(elem, ...e) } elem.addEventListener(type, newcb, istrue) all.push([elem, type, newcb, istrue]) }) }) } else if (a.gettype(type, "string")) { type = a.matchall(type, /[a-z]+/g) type.forEach((type) => { const newcb = function (e) { cb.call(elem, e, type) } elem.addEventListener(type, newcb, istrue) all.push([elem, type, newcb, istrue]) }) } return all } }, function ({ gettype, trygettype, args, //[elem, type, cb, istrue], end, maketype, }) { var [elem, type, cb, istrue] = args elem = maketype(elem, ["element", "window", "string", "array"]) if (trygettype(elem, "string")) { elem = a.qs(elem) elem = maketype(elem, ["element"]) args[0] = elem } type = maketype(type, ["array", "object", "string"]) if (trygettype(type, ["array", "any"])) { for (var temp in type) gettype(temp, "string") } else if (trygettype(type, "object")) { for (var temp of Object.values(type)) gettype(temp, "function") } if (trygettype(type, "object")) cb = maketype(cb, ["undefined"]) else cb = maketype(cb, ["function"]) istrue = maketype(istrue, ["boolean", "undefined"]) return end() } ) a.unlisten = newfunc( function listen(all) { for (var l of all) { l[0].removeEventListener(l[1], l[2], l[3]) } }, function ({ gettype, trygettype, args: [all], end, maketype }) { all = maketype(all, ["array"]) return end() } ) a.toelem = newfunc( function toelem(elem, single) { if (a.gettype(elem, "element")) return elem switch (a.gettype(elem)) { case "string": return single ? a.qs(elem) : a.qsa(elem) case "array": return elem.map((elem) => { return a.toelem(elem, single) }) case "object": var newobj = { ...elem, } if (single) return { [Object.keys(newobj)[0]]: a.toelem( newobj[Object.keys(newobj)[0]], single ), } a.foreach(newobj, function (a, s) { newobj[a] = a.toelem(s) }) return newobj default: error(elem, "inside [toelem] - not an element?") return undefined } }, function ({ ifunset, end, args: [elem, single], maketype }) { ifunset([undefined, false]) elem = maketype(elem, ["element", "string", "array", "object"]) single = maketype(single, ["boolean"]) return end() } ) a.geturlperams = newfunc( function geturlperams(e = location.href) { var arr = {} ;[ ...e.matchAll(/[?&]([^&\s]+?)(?:=([^&\s]*?))?(?=&|$|\s)/g), ].forEach((e) => { if (e[1].includes("#")) arr["#"] = e[1].match(/#(.*$)/)[1] if (e[2].includes("#")) arr["#"] = e[2].match(/#(.*$)/)[1] e[1] = e[1].replace(/#.*$/, "") e[2] = e[2].replace(/#.*$/, "") arr[decodeURIComponent(e[1]).replaceAll("+", " ")] = e[2] === undefined ? undefined : decodeURIComponent(e[2]).replaceAll("+", " ") }) return arr }, function ({ end, maketype, args: [e], ifunset }) { ifunset([location.href]) e = maketype(e, ["string", "undefined"]) return end() } ) a.updateurlperam = newfunc( function updateurlperam(key, value, cangoback) { var g = a.geturlperams() if (g[key] == value && !cangoback) return g[key] = value var k = "" var hash = "" a.foreach(g, function (key, value) { if (key == "#") return (hash = key + value) key = encodeURIComponent(key) value = encodeURIComponent(value) k += "&" + (value === undefined ? key : key + "=" + value) }) k = k.replace("&", "?") k += hash cangoback ? history.pushState(null, null, k) : history.replaceState(null, null, k) return key }, function ({ ifunset, end, maketype, args: [key, value, cangoback], }) { ifunset([undefined, undefined, true]) key = maketype(key, ["string"]) value = maketype(value, ["string"]) cangoback = maketype(cangoback, ["boolean"]) return end() } ) a.rerange = newfunc( function rerange(val, low1, high1, low2, high2) { return ((val - low1) / (high1 - low1)) * (high2 - low2) + low2 }, function ({ end, maketype, args: [val, low1, high1, low2, high2], }) { val = maketype(val, ["number"]) low1 = maketype(low1, ["number"]) high1 = maketype(high1, ["number"]) low2 = maketype(low2, ["number"]) high2 = maketype(high2, ["number"]) return end() } ) a.destring = newfunc( function destring(inp) { var out = inp if (/^[\-0-9]+$/.test(inp)) return Number(inp) if (a.gettype((out = JSON.parse(inp)), "array")) return out if ( a.gettype( (out = JSON.parse( inp.replaceAll("'", '"').replaceAll("`", '"') )), "object" ) ) return out if (inp == "true") return true if (inp == "false") return false if (inp == "undefined") return undefined if (inp == "NaN") return NaN return inp }, function ({ end, maketype, args: [inp] }) { inp = maketype(inp, ["string"]) return end() } ) a.eachelem = newfunc( function eachelem(arr1, cb) { var arr = [] var elem = [] if (a.gettype(arr1, "array")) { arr1.foreach((e) => { elem = [ ...elem, ...(a.gettype(e, "string") ? a.qsa(e) : [e]), ] }) } else { elem = a.gettype(arr1, "string") ? a.qsa(ar1) : [arr1] } elem = elem.filter((e) => { return e instanceof Element }) elem.foreach(function (...a) { arr.push(cb(...a)) }) if (arr.length == 1) arr = arr[0] return arr }, function ({ end, maketype, args: [arr1, cb] }) { arr1 = maketype(arr1, ["string", "array", "element"]) cb = maketype(cb, ["function"]) return end() } ) a.remove = newfunc( function remove(arr, idx, isidx) { arr = [...arr] idx = isidx ? idx : arr.indexOf(idx) if (idx < 0 || typeof idx !== "number") return arr arr.splice(idx, 1) return arr }, function ({ ifunset, gettype, end, maketype, trymaketype, trygettype, args: [arr, idx, isidx], }) { ifunset([undefined, undefined, true]) isidx = maketype(isidx, ["boolean"]) if (isidx) idx = maketype(idx, ["number"]) arr = maketype(arr, ["array"]) return end() } ) a.createelem = newfunc( function createelem(parent, elem, data = {}) { var type = elem var issvg = elem == "svg" || parent?.tagName?.toLowerCase?.() == "svg" elem = issvg ? document.createElementNS("http://www.w3.org/2000/svg", elem) : document.createElement(elem) if (data.class) data.class.split(" ").forEach((e) => { elem.classList.add(e) }) if (data.options && type == "select") data.options = data.options.map((e) => a.gettype(e, "array") ? a.createelem(elem, "option", { innerHTML: e[0], value: e[1], }) : a.createelem(elem, "option", { innerHTML: e, value: e, }) ) if (type == "label" && "for" in data) { data.htmlFor = data.for } Object.assign(elem.style, data) if (type == "select") { a.foreach(data, function (a, s) { elem[a] = s }) } else if (issvg) { Object.keys(data).forEach((e) => (elem[e] = data[e])) } else { Object.assign(elem, data) } if (parent !== null) { if (typeof parent == "string") parent = a.qs(parent) parent.appendChild(elem) } return elem }, function ({ ifunset, end, maketype, args: [parent, elem, data], }) { ifunset([undefined, undefined, {}]) parent = maketype(parent, ["element", "none"]) elem = maketype(elem, ["string"]) data = maketype(data, ["object"]) return end() } ) a.newelem = newfunc( function newelem(type, data = {}, inside = []) { var parent = a.createelem(null, type, data) inside.forEach((elem) => { parent.appendChild(elem) }) return parent }, function ({ ifunset, gettype, end, maketype, trymaketype, trygettype, args: [type, data, inside], }) { ifunset([null, {}, []]) type = maketype(type, ["string"]) data = maketype(data, ["object", "undefined", "null"]) maketype(inside, ["array", "undefined", "null"]) return end() } ) a.gettype = newfunc( function gettype(thing, match) { if ( !match || (Object.prototype.toString .call(match) .toLowerCase() .match(/^\[[a-z]+ (.+)\]$/)[1] == "string" && !match.includes("|")) ) { var type = Object.prototype.toString .call(thing) .toLowerCase() .match(/^\[[a-z]+ (.+)\]$/)[1] if (type !== "function") if (type == match) return true if (match == "normalfunction") return type == "function" if (type == "htmldocument" && match == "document") return true if (match == "body" && type == "htmlbodyelement") return true if (match && new RegExp(`^html${match}element$`).test(type)) return true if (/^html\w+element$/.test(type)) type = "element" if (type == "htmldocument") type = "element" if (type == "asyncfunction") type = "function" if (type == "generatorfunction") type = "function" if (type == "regexp") type = "regex" if (match == "regexp") match = "regex" if (match == "element" && type == "window") return true if (match == "element" && type == "shadowroot") return true if (match == "event" && /\w+event$/.test(type)) return true if (/^(html|svg).*element$/.test(type)) type = "element" if (type == "function") { type = /^\s*class\s/.test( Function.prototype.toString.call(thing) ) ? "class" : "function" } if (match == "none") return ( type == "nan" || type == "undefined" || type == "null" ) try { if (type === "number" && isNaN(thing) && match == "nan") return true } catch (e) { error(thing) } return match ? match === type : type } else { if (match.includes("|")) match = match.split("|") match = [...new Set(match)] return match.filter((e) => a.gettype(thing, e)).length > 0 } }, function ({ ifunset, gettype, end, maketype, trymaketype, trygettype, args: [_thing, match], }) { // _thing = maketype(_thing, ["any"]) match = maketype(match, ["array", "string", "none"]) return end() } ) a.waitforelem = newfunc( async function waitforelem(selector) { if (a.gettype(selector, "string")) { selector = [selector] } await a.bodyload() var g = false return new Promise((resolve) => { var observer = new MutationObserver(check) observer.observe(document.body, { childList: true, subtree: true, attributes: true, characterData: false, }) check() function check() { if (g) return if (selector.find((selector) => !a.qs(selector))) return observer.disconnect() resolve( selector.length == 1 ? a.qs(selector[0]) : selector.map((e) => a.qs(e)) ) } }) }, function ({ ifunset, end, args: [selector], maketype }) { ifunset([undefined]) selector = maketype(selector, ["string", "array"]) return end() } ) a.getallvars = newfunc( function getallvars() { var obj = {} var variables = [] for (var name in this) if ( !`window self document name location customElements history locationbar menubar personalbar scrollbars statusbar toolbar status closed frames length top opener parent frameElement navigator origin external screen innerWidth innerHeight scrollX pageXOffset scrollY pageYOffset visualViewport screenX screenY outerWidth outerHeight devicePixelRatio clientInformation screenLeft screenTop styleMedia onsearch isSecureContext trustedTypes performance onappinstalled onbeforeinstallprompt crypto indexedDB sessionStorage localStorage onbeforexrselect onabort onbeforeinput onblur oncancel oncanplay oncanplaythrough onchange onclick onclose oncontextlost oncontextmenu oncontextrestored oncuechange ondblclick ondrag ondragend ondragenter ondragleave ondragover ondragstart ondrop ondurationchange onemptied onended onerror onfocus onformdata oninput oninvalid onkeydown onkeypress onkeyup onload onloadeddata onloadedmetadata onloadstart onmousedown onmouseenter onmouseleave onmousemove onmouseout onmouseover onmouseup onmousewheel onpause onplay onplaying onprogress onratechange onreset onresize onscroll onsecuritypolicyviolation onseeked onseeking onselect onslotchange onstalled onsubmit onsuspend ontimeupdate ontoggle onvolumechange onwaiting onwebkitanimationend onwebkitanimationiteration onwebkitanimationstart onwebkittransitionend onwheel onauxclick ongotpointercapture onlostpointercapture onpointerdown onpointermove onpointerrawupdate onpointerup onpointercancel onpointerover onpointerout onpointerenter onpointerleave onselectstart onselectionchange onanimationend onanimationiteration onanimationstart ontransitionrun ontransitionstart ontransitionend ontransitioncancel onafterprint onbeforeprint onbeforeunload onhashchange onlanguagechange onmessage onmessageerror onoffline ononline onpagehide onpageshow onpopstate onrejectionhandled onstorage onunhandledrejection onunload crossOriginIsolated scheduler alert atob blur btoa cancelAnimationFrame cancelIdleCallback captureEvents clearInterval clearTimeout close confirm createImageBitmap fetch find focus getComputedStyle getSelection matchMedia moveBy moveTo open postMessage print prompt queueMicrotask releaseEvents reportError requestAnimationFrame requestIdleCallback resizeBy resizeTo scroll scrollBy scrollTo setInterval setTimeout stop structuredClone webkitCancelAnimationFrame webkitRequestAnimationFrame originAgentCluster navigation webkitStorageInfo speechSynthesis oncontentvisibilityautostatechange openDatabase webkitRequestFileSystem webkitResolveLocalFileSystemURL chrome caches cookieStore ondevicemotion ondeviceorientation ondeviceorientationabsolute launchQueue onbeforematch getDigitalGoodsService getScreenDetails queryLocalFonts showDirectoryPicker showOpenFilePicker showSaveFilePicker TEMPORARY PERSISTENT addEventListener dispatchEvent removeEventListener` .split(" ") .includes(name) ) variables.push(name) variables.forEach((e) => { var c = String(a.gettype(this[e])) if (c === "object") c = "variable" if (!obj[c]) obj[c] = [] obj[c].push(e) }) return obj }, function ({ end }) { return end() } ) a.sha = newfunc( function sha(s = "", includesymbols) { var tab if (typeof includesymbols == "string") { tab = includesymbols } else if (includesymbols) { tab = "`~\\|[];',./{}:<>?\"!@#$%^&*ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789" } else { tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789" } return binb2b64(core_sha1(str2binb(s), s.length * 8)) function core_sha1(x, len) { x[len >> 5] |= 0x80 << (24 - len) x[(((len + 64) >> 9) << 4) + 15] = len var w = Array(80) var a = 1732584193 var b = -271733879 var c = -1732584194 var d = 271733878 var e = -1009589776 for (var i = 0; i < x.length; i += 16) { var olda = a var oldb = b var oldc = c var oldd = d var olde = e for (var j = 0; j < 80; j++) { if (j < 16) w[j] = x[i + j] else w[j] = rol( w[j - 3] ^ w[j - 8] ^ w[j - 14] ^ w[j - 16], 1 ) var t = safe_add( safe_add(rol(a, 5), sha1_ft(j, b, c, d)), safe_add(safe_add(e, w[j]), sha1_kt(j)) ) e = d d = c c = rol(b, 30) b = a a = t } a = safe_add(a, olda) b = safe_add(b, oldb) c = safe_add(c, oldc) d = safe_add(d, oldd) e = safe_add(e, olde) } return Array(a, b, c, d, e) } function sha1_ft(t, b, c, d) { if (t < 20) return (b & c) | (~b & d) if (t < 40) return b ^ c ^ d if (t < 60) return (b & c) | (b & d) | (c & d) return b ^ c ^ d } function sha1_kt(t) { return t < 20 ? 1518500249 : t < 40 ? 1859775393 : t < 60 ? -1894007588 : -899497514 } function safe_add(x, y) { var lsw = (x & 0xffff) + (y & 0xffff) var msw = (x >> 16) + (y >> 16) + (lsw >> 16) return (msw << 16) | (lsw & 0xffff) } function rol(num, cnt) { return (num << cnt) | (num >>> (32 - cnt)) } function str2binb(str) { var bin = Array() var mask = (1 << 8) - 1 for (var i = 0; i < str.length * 8; i += 8) bin[i >> 5] |= (str.charCodeAt(i / 8) & mask) << (24 - i) return bin } function binb2b64(binarray) { var str = "" for (var i = 0; i < binarray.length * 4; i += 3) { var triplet = (((binarray[i >> 2] >> (8 * (3 - (i % 4)))) & 0xff) << 16) | (((binarray[(i + 1) >> 2] >> (8 * (3 - ((i + 1) % 4)))) & 0xff) << 8) | ((binarray[(i + 2) >> 2] >> (8 * (3 - ((i + 2) % 4)))) & 0xff) for (var j = 0; j < 4; j++) { if (i * 8 + j * 6 > binarray.length * 32) str += "" else str += tab.charAt((triplet >> (6 * (3 - j))) & 0x3f) } } return str } }, function ({ ifunset, end, args: [s, includesymbols], maketype }) { ifunset([undefined, false]) s = maketype(s, ["string"]) includesymbols = maketype(includesymbols, ["boolean", "string"]) return end() } ) a.qs = newfunc( function qs(text, parent = document) { return parent.querySelector(text) }, function ({ end, args: [text, parent], maketype }) { parent = maketype(parent, ["element", "undefined"]) text = maketype(text, ["string"]) return end() } ) a.qsa = newfunc( function qsa(text, parent = document) { return Array.from(parent.querySelectorAll(text)) }, function ({ end, args: [text, parent], maketype }) { parent = maketype(parent, ["element", "undefined"]) text = maketype(text, ["string"]) return end() } ) a.csspath = newfunc( function csspath(el) { if (a.gettype(el, "array")) return a.map(el, (e) => a.csspath(e)) if (!(el instanceof Element)) return var path = [] while (el.nodeType === Node.ELEMENT_NODE) { var selector = el.nodeName.toLowerCase() if (el.id) { selector += "#" + el.id path.unshift(selector) break } else { var sib = el, nth = 1 while ((sib = sib.previousElementSibling)) { if (sib.nodeName.toLowerCase() == selector) nth++ } if (nth != 1) selector += ":nth-of-type(" + nth + ")" } path.unshift(selector) el = el.parentNode } return path.join(" > ") }, function ({ end, args: [el], maketype }) { el = maketype(el, ["element", "array"]) return end() } ) a.fromms = newfunc( function fromms(ms) { ms = Number(ms) return { years: Math.floor(ms / 1000 / 60 / 60 / 24 / 365), days: Math.floor(ms / 1000 / 60 / 60 / 24) % 365, hours: Math.floor(ms / 1000 / 60 / 60) % 24, mins: Math.floor(ms / 1000 / 60) % 60, secs: Math.floor(ms / 1000) % 60, ms: Math.floor(ms) % 1000, } }, function ({ ifunset, end, args: [ms], maketype }) { ifunset([0]) // Default value for ms ms = maketype(ms, ["number"]) // Ensure ms is a number return end() } ) a.fromms = newfunc( function fromms(ms) { ms = Number(ms) return { years: Math.floor(ms / 1000 / 60 / 60 / 24 / 365), days: Math.floor(ms / 1000 / 60 / 60 / 24) % 365, hours: Math.floor(ms / 1000 / 60 / 60) % 24, mins: Math.floor(ms / 1000 / 60) % 60, secs: Math.floor(ms / 1000) % 60, ms: Math.floor(ms) % 1000, } }, function ({ ifunset, end, args: [ms], maketype }) { ifunset([0]) // Default value for ms ms = maketype(ms, ["number"]) // Ensure ms is a number return end() } ) a.rect = newfunc( function rect(e) { if (a.gettype(e, "string")) e = a.qs(e) var { x, y, width, height } = e.getBoundingClientRect().toJSON() return { x, y, w: width, h: height, } }, function ({ end, args: [e], maketype }) { e = maketype(e, ["element", "string"]) return end() } ) a.setelem = newfunc( function setelem(elem, data) { var issvg = elem == "svg" || parent?.tagName?.toLowerCase?.() == "svg" if (data.class) data.class.split(" ").forEach((e) => { elem.classList.add(e) }) if (data.options && elem.tagName.toLowerCase() == "select") data.options = data.options.map((e) => a.gettype(e, "array") ? a.createelem(elem, "option", { innerHTML: e[0], value: e[1], }) : a.createelem(elem, "option", { innerHTML: e, value: e, }) ) if (elem.tagName.toLowerCase() == "label" && "for" in data) { data.htmlFor = data.for } Object.assign(elem.style, data) if (elem.tagName.toLowerCase() == "select") { a.foreach(data, function (a, s) { elem[a] = s }) } else if (issvg) { Object.keys(data).forEach((e) => (elem[e] = data[e])) } else { Object.assign(elem, data) } return elem }, function ({ ifunset, end, args: [elem, data], maketype }) { ifunset([undefined, {}]) // Default value for data elem = maketype(elem, ["element", "string"]) // Ensure elem is an element or string data = maketype(data, ["object"]) // Ensure data is an object return end() } ) a.watchvar = newfunc( function watchvar(varname, onset, onget, obj = window) { obj = obj || window obj[`_${varname}`] = undefined obj[`${varname}`] = undefined Object.defineProperty(obj, varname, { configurable: false, get() { if (onget) return onget(obj[`_${varname}`]) return obj[`_${varname}`] }, set(value) { if (value === obj[`_${varname}`]) { return } var s = onset(value, obj[`_${varname}`]) if (s) obj[`_${varname}`] = value }, }) }, function ({ ifunset, end, args: [varname, onset, onget, obj], maketype, }) { ifunset([undefined, () => {}, () => {}, window]) // Default values varname = maketype(varname, ["string"]) // Ensure varname is a string onset = maketype(onset, ["function"]) // Ensure onset is a function onget = maketype(onget, ["function", "undefined"]) // Ensure onget is a function or undefined obj = maketype(obj, ["object", "undefined"]) // Ensure obj is an object or undefined return end() } ) a.randomizeorder = newfunc( function randomizeorder(arr) { arr = [...arr] var arr2 = [] var count = arr.length for (var i = 0; i < count; i++) { var idx = a.randfrom(0, arr.length - 1) arr2.push(arr[idx]) arr.splice(idx, 1) } return arr2 }, function ({ end, args: [arr], maketype }) { arr = maketype(arr, ["array"]) // Ensure arr is an array return end() } ) a.constrainvar = newfunc( function constrainvar(varname, min, max) { window[`_${varname}`] = undefined window[`${varname}`] = undefined Object.defineProperty(window, varname, { configurable: false, get() { return window[`_${varname}`] }, set(value) { if (value === window[`_${varname}`]) { return } if (value > max) value = max if (value < min) value = min window[`_${varname}`] = value }, }) }, function ({ ifunset, end, args: [varname, min, max], maketype }) { ifunset([undefined, -Infinity, Infinity]) // Default values for min and max varname = maketype(varname, ["string"]) // Ensure varname is a string min = maketype(min, ["number"]) // Ensure min is a number max = maketype(max, ["number"]) // Ensure max is a number return end() } ) a.isbetween = newfunc( function isbetween(z, x, c) { if (x == c) return false var big, small if (x > c) { big = x small = c } else { big = c small = x } return z > big && z < small }, function ({ args: [z, x, c], end, maketype }) { z = maketype(z, ["number"]) x = maketype(x, ["number"]) c = maketype(c, ["number"]) return end() } ) a.indexsof = newfunc( function indexsof(y, x) { var i = 0 var arr = [] y.split(x).forEach((e, k) => { i += e.length arr.push(i + k) }) arr.pop() return arr }, function ({ args: [y, x], end, maketype }) { y = maketype(y, ["string"]) // Ensure y is a string x = maketype(x, ["string"]) // Ensure x is a string return end() } ) a._export = newfunc( function _export() { var s = [] a.qsa("input, textarea").foreach((e) => { s.push({ path: a.csspath(e), value: escape(e.value), checked: e.checked, }) }) return JSON.stringify(s) }, function ({ end }) { return end() } ) a._import = newfunc( function _import(data) { data.forEach((e) => { var s = a.qs(e.path) s.checked = e.checked s.value = unescape(e.value) }) return data }, function ({ end, args: [data], maketype }) { data = maketype(data, ["array"]) return end() } ) a.popup = newfunc( function popup(data, x, y, w, h) { if (x || x === 0) { x = (screen.width / 100) * x y = (screen.height / 100) * y w = (screen.width / 100) * w h = (screen.height / 100) * h var win = open( "", "", `left=${x}, top=${y} width=${w},height=${h}` ) win.document.write(data) return win } else { var win = open("") win.document.write(data) return win } }, function ({ end, maketype, args: [data, x, y, w, h] }) { data = maketype(data, ["string"]) x = maketype(x, ["number"]) y = maketype(y, ["number"]) w = maketype(w, ["number"]) h = maketype(h, ["number"]) return end() } ) a.same = newfunc( function same(...a) { if (a.length == 1) a = a[0] return ( [...new Set(a.map((e) => JSON.stringify(e)))].length === 1 ) }, function ({ end }) { return end() } ) a.containsany = newfunc( function containsany(arr1, arr2) { return !!arr2.find((e) => arr1.includes(e)) }, function ({ end, args: [arr1, arr2], maketype }) { arr1 = maketype(arr1, ["string", "array"]) arr2 = maketype(arr2, ["string", "array"]) return end() } ) a.getprops = newfunc( function getprops(func, peramsonly) { return peramsonly ? getprops(func) .vars.map((e) => e.var) .filter((e) => e) : getprops(func) }, function ({ end, args: [func, peramsonly], maketype }) { func = maketype(func, ["function"]) // Ensure func is a function peramsonly = maketype(peramsonly, ["boolean", "undefined"]) // Ensure peramsonly is a boolean or undefined return end() } ) a.bodyload = newfunc( function bodyload() { return new Promise((resolve) => { if (document.body) resolve() var observer = new MutationObserver(function () { if (document.body) { resolve() observer.disconnect() } }) observer.observe(document.documentElement, { childList: true, }) }) }, function ({ end }) { return end() } ) a.repeat = newfunc( function repeat(func, count, delay, instantstart, waituntildone) { if (delay || waituntildone) return new Promise(async (resolve) => { if (delay) { var extra = 0 for (var i = 0; i < count; i++) { if (instantstart) waituntildone ? await func(i) : func(i) extra = await a.wait(delay - extra) if (!instantstart) waituntildone ? await func(i) : func(i) } resolve() } else for (var i = 0; i < count; i++) waituntildone ? await func(i) : func(i) resolve() }) for (var i = 0; i < count; i++) func(i) return }, function ({ end, args: [func, count, delay, instantstart, waituntildone], maketype, }) { func = maketype(func, ["function"]) // Ensure func is a function count = maketype(count, ["number"]) // Ensure count is a number delay = maketype(delay, ["number", "undefined"]) // Ensure delay is a number or undefined instantstart = maketype(instantstart, ["boolean", "undefined"]) // Ensure instantstart is a boolean or undefined waituntildone = maketype(waituntildone, [ "boolean", "undefined", ]) // Ensure waituntildone is a boolean or undefined return end() } ) a.repeatuntil = newfunc( function repeatuntil( func, donecheck, delay, instantstart, waituntildone ) { return new Promise(async (resolve) => { if (delay) { var extra = 0 var i = 0 while (!donecheck()) { i++ if (instantstart) { waituntildone ? await func(i) : func(i) } extra = await a.wait(delay - extra) if (!instantstart) { waituntildone ? await func(i) : func(i) } } resolve() } else { var i = 0 while (!donecheck()) { i++ waituntildone ? await func(i) : func(i) } resolve() } }) }, function ({ end, args: [func, donecheck, delay, instantstart, waituntildone], maketype, }) { func = maketype(func, ["function"]) // Ensure func is a function donecheck = maketype(donecheck, ["function"]) // Ensure donecheck is a function delay = maketype(delay, ["number", "undefined"]) // Ensure delay is a number or undefined instantstart = maketype(instantstart, ["boolean", "undefined"]) // Ensure instantstart is a boolean or undefined waituntildone = maketype(waituntildone, [ "boolean", "undefined", ]) // Ensure waituntildone is a boolean or undefined return end() } ) a.getfolderpath = newfunc( async function getfolderpath(folder) { async function parsedir(dir, x) { if (!x) { return [ { name: dir.name, inside: await parsedir(dir, true), type: "folder", handle: dir, }, ] } else var arr = [] for await (const [name, handle] of dir.entries()) { arr.push( a.gettype(handle, "filesystemdirectoryhandle") ? { type: "folder", inside: await parsedir(handle, true), name, handle, } : { type: "file", handle, name } ) } return arr } return parsedir(folder) }, function ({ end, args: [folder], maketype }) { folder = maketype(folder, ["filesystemdirectoryhandle"]) return end() } ) a.getfiles = newfunc( async function getfiles( oldway, multiple, accept = [], options = {} ) { const supportsFileSystemAccess = "showOpenFilePicker" in window && (() => { try { return window.self === window.top } catch { return false } })() if (!oldway) { if (!supportsFileSystemAccess) throw new Error("no access") let fileOrFiles = undefined try { const handles = await showOpenFilePicker({ types: [ { accept: { "*/*": accept, }, }, ], multiple, ...options, }) if (!multiple) { fileOrFiles = handles[0] } else { fileOrFiles = await Promise.all(handles) } } catch (err) { if (err.name !== "AbortError") { error(err.name, err.message) } } return fileOrFiles } return new Promise(async (resolve) => { await a.bodyload() const input = document.createElement("input") input.style.display = "none" input.type = "file" if (accept) input.accept = accept document.body.append(input) if (multiple) { input.multiple = true } input.addEventListener("change", () => { input.remove() resolve(multiple ? Array.from(input.files) : input.files[0]) }) if ("showPicker" in HTMLInputElement.prototype) { input.showPicker() } else { input.click() } }) }, function ({ end, args: [oldway, multiple, accept, options], maketype, ifunset, }) { ifunset(undefined, undefined, [], {}) oldway = maketype(oldway, ["boolean"]) // Ensure oldway is a boolean multiple = maketype(multiple, ["boolean"]) // Ensure multiple is a boolean accept = maketype(accept, ["array", "string", "undefined"]) // Ensure accept is an array or string options = maketype(options, ["object", "undefined"]) return end() } ) a.getfolder = newfunc( async function getfolder(write = false, options = {}) { const supportsFileSystemAccess = "showDirectoryPicker" in window && (() => { try { return window.self === window.top } catch { return false } })() if (!supportsFileSystemAccess) throw new Error("no access") try { return await showDirectoryPicker({ mode: write ? "readwrite" : "read", ...options, }) } catch (err) { if (err.name !== "AbortError") { error(err.name, err.message) } } return undefined }, function ({ end, args: [write, options], maketype, ifunset }) { ifunset(false, {}) write = maketype(write, ["boolean", "undefined"]) options = maketype(options, ["object", "undefined"]) return end() } ) a.map = newfunc( function map(arr, func) { var type = a.gettype(arr) if (type == "array") return arr.map(func) else if (type == "object") { var temparr = {} Reflect.ownKeys(arr).forEach((e, i) => { temparr = { ...temparr, ...func(e, arr[e], i), } }) return temparr } else { return [arr].map(func) } }, function ({ end, args: [arr, func], maketype }) { arr = maketype(arr, ["array", "object"]) // Ensure arr is an array or object func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.find = newfunc( function find(arr, func) { var type = a.gettype(arr) if (type == "array") return arr.find(func) else if (type == "object") { return Reflect.ownKeys(arr).find((e, i) => { return func(e, arr[e], i) }) } else { return [arr].find(func) } }, function ({ end, args: [arr, func], maketype }) { arr = maketype(arr, ["array", "object"]) // Ensure arr is an array or object func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.filteridx = newfunc( function filteridx(arr, func) { if (a.gettype(arr, "object")) arr = [arr] return a .map(arr, (e, i) => (func(e, i) ? i : undefined)) .filter((e) => e !== undefined) }, function ({ end, args: [arr, func], maketype }) { arr = maketype(arr, ["array", "object"]) // Ensure arr is an array or object func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.filter = newfunc( function filter(arr, func) { var type = a.gettype(arr) if (type == "array") return arr.filter(func) else if (type == "object") { var temparr = {} Reflect.ownKeys(arr).forEach((e, i) => { if (func(e, arr[e], i)) temparr = { ...temparr, [e]: arr[e], } }) return temparr } else { return [arr].filter(func) } }, function ({ end, args: [arr, func], maketype }) { arr = maketype(arr, ["array", "object"]) // Ensure arr is an array or object func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.unique = newfunc( function unique() /*object*/ { return last || (last = different.new()) }, function ({ end }) { return end() } ) a.tostring = newfunc( function tostring(e) { if (["object", "array"].includes(a.gettype(e))) return JSON.stringify(e) if (a.gettype(e, "element")) return a.csspath(e) return String(e) }, function ({ end, args: [e], maketype }) { e = maketype(e, ["any"]) // Ensure e can be any type return end() } ) a.toregex = newfunc( function toregex(d, s) { if (a.gettype(d, "array")) var temp = d if (s) var temp = [d, s] else if (String(d).match(/^\/(.*)\/(\w*)$/)) { var m = String(d).match(/^\/(.*)\/(\w*)$/) var temp = [m[1], m[2]] } else var temp = [String(d), ""] temp[1] = temp[1].toLowerCase() if (temp[1].includes("w")) { temp[1] = temp[1].replace("w", "") temp[0] = `(?<=[^a-z0-9]|^)${temp[0]}(?=[^a-z0-9]|$)` } return new RegExp( temp[0], temp[1].replaceAll(/(.)(?=.*\1)/g, "") ) }, function ({ end, args: [d, s], maketype }) { d = maketype(d, ["string", "array"]) // Ensure d is a string or array s = maketype(s, ["string", "undefined"]) // Ensure s is a string or undefined return end() } ) a.isregex = newfunc( function isregex(s) { if (a.gettype(s, "regex")) return true return ( /^\/.*(?<!\\)\/[gimusy]*$/.test(s) && !/^\/\*.*\*\/$/.test(s) ) }, function ({ end, args: [s], maketype }) { s = maketype(s, ["string"]) // Ensure s is a string return end() } ) a.ispressed = newfunc( function ispressed(e /*event*/, code) { code = code.toLowerCase() var temp = e.shiftKey == code.includes("shift") && e.altKey == code.includes("alt") && e.ctrlKey == code.includes("ctrl") && e.metaKey == code.includes("meta") && e.key.toLowerCase() == code.replaceAll(/alt|ctrl|shift|meta/g, "").trim() if (temp && !a) e.preventDefault() return temp }, function ({ end, args: [e, code], maketype }) { e = maketype(e, ["object"]) // Ensure e is an event object code = maketype(code, ["string"]) // Ensure code is a string return end() } ) a.controller_vibrate = newfunc( function controller_vibrate( pad, duration = 1000, strongMagnitude = 0, weakMagnitude = 0 ) { getpad(pad).vibrationActuator.playEffect("dual-rumble", { duration, strongMagnitude, weakMagnitude, }) return pad }, function ({ end, args: [pad, duration, strongMagnitude, weakMagnitude], maketype, }) { pad = maketype(pad, ["element"]) // Ensure pad is a valid element duration = maketype(duration, ["number"]) // Ensure duration is a number strongMagnitude = maketype(strongMagnitude, ["number"]) // Ensure strongMagnitude is a number weakMagnitude = maketype(weakMagnitude, ["number"]) // Ensure weakMagnitude is a number return end() } ) a.controller_getbutton = newfunc( function controller_getbutton(pad, button) { return button ? getpad(pad).buttons[button].value : getpad(pad).buttons.map((e) => e.value) }, function ({ end, args: [pad, button], maketype }) { pad = maketype(pad, ["element"]) // Ensure pad is a valid element button = maketype(button, ["number", "undefined"]) // Ensure button is a number or undefined return end() } ) a.controller_getaxes = newfunc( function controller_getaxes(pad, axes) { return axes ? getpad(pad).axes[axes] : getpad(pad).axes }, function ({ end, args: [pad, axes], maketype }) { pad = maketype(pad, ["element"]) // Ensure pad is a valid element axes = maketype(axes, ["number", "undefined"]) // Ensure axes is a number or undefined return end() } ) a.controller_exists = newfunc( function controller_exists(pad) { return pad === undefined ? getpad().filter((e) => e).length : !!getpad(pad) }, function ({ end, args: [pad], maketype }) { pad = maketype(pad, ["element", "undefined"]) // Ensure pad is a valid element or undefined return end() } ) a.readfile = newfunc( async function readfile(file, type = "Text") { return new Promise(function (done, error) { var f = new FileReader() f.onerror = error f.onload = () => done(type == "json" ? JSON.parse(f.result) : f.result) f["readAs" + (type == "json" ? "Text" : type)](file) }) }, function ({ end, args: [file, type], maketype, ifunset }) { ifunset(undefined, "Text") type = maketype(type, ["string", "undefined"]) // Ensure type is a string return end() } ) a.writefile = newfunc( async function writefile(file, text) { var f = await file.createWritable() await f.write(text) await f.close() return file }, function ({ end, args: [file, text], maketype }) { text = maketype(text, ["string"]) // Ensure text is a string return end() } ) a.getfileperms = newfunc( async function getfileperms(fileHandle, readWrite) { const options = {} if (readWrite) { options.mode = "readwrite" } try { return ( (await fileHandle.queryPermission(options)) === "granted" || (await fileHandle.requestPermission(options)) === "granted" ) } catch (e) { await a.waitforclick() return ( (await fileHandle.queryPermission(options)) === "granted" || (await fileHandle.requestPermission(options)) === "granted" ) } }, function ({ end, args: [fileHandle, readWrite], maketype }) { readWrite = maketype(readWrite, ["boolean", "undefined"]) // Ensure readWrite is a boolean or undefined return end() } ) a.waitforclick = newfunc( async function waitforclick(fileHandle, readWrite) { return new Promise((resolve) => { var listener = a.listen( window, "click", () => { a.unlisten(listener) resolve() }, true ) }) }, function ({ end, args: [] }) { return end() } ) a.indexeddb_set = newfunc( async function indexeddb_set(place, obj) { return new Promise((done, error) => place.put( obj, (e) => done(e), (e) => error(e) ) ) }, function ({ end, args: [place, obj], maketype }) { place = maketype(place, ["object"]) // Ensure place is an object obj = maketype(obj, ["object"]) // Ensure obj is an object return end() } ) a.indexeddb_get = newfunc( async function indexeddb_get(place, obj) { return new Promise((done, error) => place.get( obj, (e) => done(e), (e) => error(e) ) ) }, function ({ end, args: [place, obj], maketype }) { place = maketype(place, ["object"]) // Ensure place is an object obj = maketype(obj, ["string"]) // Ensure obj is a string return end() } ) a.indexeddb_getall = newfunc( async function indexeddb_getall(place) { return new Promise((done, error) => place.getAll( (e) => done(e), (e) => error(e) ) ) }, function ({ end, args: [place], maketype }) { place = maketype(place, ["object"]) // Ensure place is an object return end() } ) a.indexeddb_clearall = newfunc( async function indexeddb_clearall(place) { return new Promise((done, error) => place.clear( (e) => done(e), (e) => error(e) ) ) }, function ({ end, args: [place], maketype }) { place = maketype(place, ["object"]) // Ensure place is an object return end() } ) a.indexeddb_remove = newfunc( async function indexeddb_remove(place, obj) { return new Promise((done, error) => place.remove( obj, (e) => done(e), (e) => error(e) ) ) }, function ({ end, args: [place, obj], maketype }) { place = maketype(place, ["object"]) // Ensure place is an object obj = maketype(obj, ["string"]) // Ensure obj is a string return end() } ) a.indexeddb_setup = newfunc( async function indexeddb_setup(obj) { return new Promise((e) => { var x obj = { dbVersion: 1, storeName: "tempstorename", keyPath: "id", autoIncrement: true, ...obj, onStoreReady() { e(x) }, } if (!window.IDBStore) { ;(function (p, h, k) { "function" === typeof define ? define(h) : "undefined" !== typeof module && module.exports ? (module.exports = h()) : (k[p] = h()) })( "IDBStore", function () { function p(a, b) { var c, d for (c in b) (d = b[c]), d !== u[c] && d !== a[c] && (a[c] = d) return a } var h = function (a) { throw a }, k = function () {}, r = { storeName: "Store", storePrefix: "IDBWrapper-", dbVersion: 1, keyPath: "id", autoIncrement: !0, onStoreReady: function () {}, onError: h, indexes: [], implementationPreference: [ "indexedDB", "webkitIndexedDB", "mozIndexedDB", "shimIndexedDB", ], }, q = function (a, b) { "undefined" == typeof b && "function" == typeof a && (b = a) "[object Object]" != Object.prototype.toString.call(a) && (a = {}) for (var c in r) this[c] = "undefined" != typeof a[c] ? a[c] : r[c] this.dbName = this.storePrefix + this.storeName this.dbVersion = parseInt(this.dbVersion, 10) || 1 b && (this.onStoreReady = b) var d = "object" == typeof window ? window : self this.implementation = this.implementationPreference.filter(function ( a ) { return a in d })[0] this.idb = d[this.implementation] this.keyRange = d.IDBKeyRange || d.webkitIDBKeyRange || d.mozIDBKeyRange this.consts = { READ_ONLY: "readonly", READ_WRITE: "readwrite", VERSION_CHANGE: "versionchange", NEXT: "next", NEXT_NO_DUPLICATE: "nextunique", PREV: "prev", PREV_NO_DUPLICATE: "prevunique", } this.openDB() }, t = { constructor: q, version: "1.7.2", db: null, dbName: null, dbVersion: null, store: null, storeName: null, storePrefix: null, keyPath: null, autoIncrement: null, indexes: null, implementationPreference: null, implementation: "", onStoreReady: null, onError: null, _insertIdCount: 0, openDB: function () { var a = this.idb.open( this.dbName, this.dbVersion ), b = !1 a.onerror = function (a) { var b b = "error" in a.target ? "VersionError" == a.target.error.name : "errorCode" in a.target ? 12 == a.target.errorCode : !1 if (b) this.onError( Error( "The version number provided is lower than the existing one." ) ) else a.target.error ? (a = a.target.error) : ((b = "IndexedDB unknown error occurred when opening DB " + this.dbName + " version " + this.dbVersion), "errorCode" in a.target && (b += " with error code " + a.target.errorCode), (a = Error(b))), this.onError(a) }.bind(this) a.onsuccess = function (a) { if (!b) if (this.db) this.onStoreReady() else if ( ((this.db = a.target.result), "string" == typeof this.db.version) ) this.onError( Error( "The IndexedDB implementation in this browser is outdated. Please upgrade your browser." ) ) else if ( this.db.objectStoreNames.contains( this.storeName ) ) { this.store = this.db .transaction( [this.storeName], this.consts.READ_ONLY ) .objectStore(this.storeName) var d = Array.prototype.slice.call( this.getIndexList() ) this.indexes.forEach(function (a) { var c = a.name if (c) if ( (this.normalizeIndexData(a), this.hasIndex(c)) ) { var g = this.store.index(c) this.indexComplies(g, a) || ((b = !0), this.onError( Error( 'Cannot modify index "' + c + '" for current version. Please bump version number to ' + (this.dbVersion + 1) + "." ) )) d.splice(d.indexOf(c), 1) } else (b = !0), this.onError( Error( 'Cannot create new index "' + c + '" for current version. Please bump version number to ' + (this.dbVersion + 1) + "." ) ) else (b = !0), this.onError( Error( "Cannot create index: No index name given." ) ) }, this) d.length && ((b = !0), this.onError( Error( 'Cannot delete index(es) "' + d.toString() + '" for current version. Please bump version number to ' + (this.dbVersion + 1) + "." ) )) b || this.onStoreReady() } else this.onError( Error("Object store couldn't be created.") ) }.bind(this) a.onupgradeneeded = function (a) { this.db = a.target.result this.db.objectStoreNames.contains( this.storeName ) ? (this.store = a.target.transaction.objectStore( this.storeName )) : ((a = { autoIncrement: this.autoIncrement, }), null !== this.keyPath && (a.keyPath = this.keyPath), (this.store = this.db.createObjectStore( this.storeName, a ))) var d = Array.prototype.slice.call( this.getIndexList() ) this.indexes.forEach(function (a) { var c = a.name c || ((b = !0), this.onError( Error( "Cannot create index: No index name given." ) )) this.normalizeIndexData(a) if (this.hasIndex(c)) { var g = this.store.index(c) this.indexComplies(g, a) || (this.store.deleteIndex(c), this.store.createIndex(c, a.keyPath, { unique: a.unique, multiEntry: a.multiEntry, })) d.splice(d.indexOf(c), 1) } else this.store.createIndex(c, a.keyPath, { unique: a.unique, multiEntry: a.multiEntry, }) }, this) d.length && d.forEach(function (a) { this.store.deleteIndex(a) }, this) }.bind(this) }, deleteDatabase: function (a, b) { if (this.idb.deleteDatabase) { this.db.close() var c = this.idb.deleteDatabase(this.dbName) c.onsuccess = a c.onerror = b } else b( Error( "Browser does not support IndexedDB deleteDatabase!" ) ) }, put: function (a, b, c, d) { null !== this.keyPath && ((d = c), (c = b), (b = a)) d || (d = h) c || (c = k) var f = !1, e = null, g = this.db.transaction( [this.storeName], this.consts.READ_WRITE ) g.oncomplete = function () { ;(f ? c : d)(e) } g.onabort = d g.onerror = d null !== this.keyPath ? (this._addIdPropertyIfNeeded(b), (a = g.objectStore(this.storeName).put( (() => { function isFilesystemHandle(obj) { return ( obj && (obj instanceof FileSystemFileHandle || obj instanceof FileSystemDirectoryHandle) ) } function replaceProxies(obj) { if (isFilesystemHandle(obj)) { return obj } if ( typeof obj !== "object" || obj === null ) { return obj } if (Array.isArray(obj)) { return obj.map((item) => replaceProxies(item) ) } const result = {} for (const key in obj) { if (obj.hasOwnProperty(key)) { result[key] = replaceProxies( obj[key] ) } } return result } return replaceProxies(b) })() ))) : (a = g.objectStore(this.storeName).put(b, a)) a.onsuccess = function (a) { f = !0 e = a.target.result } a.onerror = d return g }, get: function (a, b, c) { c || (c = h) b || (b = k) var d = !1, f = null, e = this.db.transaction( [this.storeName], this.consts.READ_ONLY ) e.oncomplete = function () { ;(d ? b : c)(f) } e.onabort = c e.onerror = c a = e.objectStore(this.storeName).get(a) a.onsuccess = function (a) { d = !0 f = a.target.result } a.onerror = c return e }, remove: function (a, b, c) { c || (c = h) b || (b = k) var d = !1, f = null, e = this.db.transaction( [this.storeName], this.consts.READ_WRITE ) e.oncomplete = function () { ;(d ? b : c)(f) } e.onabort = c e.onerror = c a = e.objectStore(this.storeName)["delete"](a) a.onsuccess = function (a) { d = !0 f = a.target.result } a.onerror = c return e }, batch: function (a, b, c) { c || (c = h) b || (b = k) if ( "[object Array]" != Object.prototype.toString.call(a) ) c( Error( "dataArray argument must be of type Array." ) ) else if (0 === a.length) return b(!0) var d = a.length, f = !1, e = !1, g = this.db.transaction( [this.storeName], this.consts.READ_WRITE ) g.oncomplete = function () { ;(e ? b : c)(e) } g.onabort = c g.onerror = c var l = function () { d-- 0 !== d || f || (e = f = !0) } a.forEach(function (a) { var b = a.type, d = a.key, e = a.value a = function (a) { g.abort() f || ((f = !0), c(a, b, d)) } "remove" == b ? ((e = g .objectStore(this.storeName) ["delete"](d)), (e.onsuccess = l), (e.onerror = a)) : "put" == b && (null !== this.keyPath ? (this._addIdPropertyIfNeeded(e), (e = g .objectStore(this.storeName) .put(e))) : (e = g .objectStore(this.storeName) .put(e, d)), (e.onsuccess = l), (e.onerror = a)) }, this) return g }, putBatch: function (a, b, c) { a = a.map(function (a) { return { type: "put", value: a } }) return this.batch(a, b, c) }, upsertBatch: function (a, b, c, d) { "function" == typeof b && ((d = c = b), (b = {})) d || (d = h) c || (c = k) b || (b = {}) "[object Array]" != Object.prototype.toString.call(a) && d( Error( "dataArray argument must be of type Array." ) ) var f = b.keyField || this.keyPath, e = a.length, g = !1, l = !1, n = 0, m = this.db.transaction( [this.storeName], this.consts.READ_WRITE ) m.oncomplete = function () { l ? c(a) : d(!1) } m.onabort = d m.onerror = d var v = function (b) { a[n++][f] = b.target.result e-- 0 !== e || g || (l = g = !0) } a.forEach(function (a) { var b = a.key null !== this.keyPath ? (this._addIdPropertyIfNeeded(a), (a = m.objectStore(this.storeName).put(a))) : (a = m .objectStore(this.storeName) .put(a, b)) a.onsuccess = v a.onerror = function (a) { m.abort() g || ((g = !0), d(a)) } }, this) return m }, removeBatch: function (a, b, c) { a = a.map(function (a) { return { type: "remove", key: a } }) return this.batch(a, b, c) }, getBatch: function (a, b, c, d) { c || (c = h) b || (b = k) d || (d = "sparse") if ( "[object Array]" != Object.prototype.toString.call(a) ) c( Error( "keyArray argument must be of type Array." ) ) else if (0 === a.length) return b([]) var f = [], e = a.length, g = !1, l = null, n = this.db.transaction( [this.storeName], this.consts.READ_ONLY ) n.oncomplete = function () { ;(g ? b : c)(l) } n.onabort = c n.onerror = c var m = function (a) { a.target.result || "dense" == d ? f.push(a.target.result) : "sparse" == d && f.length++ e-- 0 === e && ((g = !0), (l = f)) } a.forEach(function (a) { a = n.objectStore(this.storeName).get(a) a.onsuccess = m a.onerror = function (a) { l = a c(a) n.abort() } }, this) return n }, getAll: function (a, b) { b || (b = h) a || (a = k) var c = this.db.transaction( [this.storeName], this.consts.READ_ONLY ), d = c.objectStore(this.storeName) d.getAll ? this._getAllNative(c, d, a, b) : this._getAllCursor(c, d, a, b) return c }, _getAllNative: function (a, b, c, d) { var f = !1, e = null a.oncomplete = function () { ;(f ? c : d)(e) } a.onabort = d a.onerror = d a = b.getAll() a.onsuccess = function (a) { f = !0 e = a.target.result } a.onerror = d }, _getAllCursor: function (a, b, c, d) { var f = [], e = !1, g = null a.oncomplete = function () { ;(e ? c : d)(g) } a.onabort = d a.onerror = d a = b.openCursor() a.onsuccess = function (a) { ;(a = a.target.result) ? (f.push(a.value), a["continue"]()) : ((e = !0), (g = f)) } a.onError = d }, clear: function (a, b) { b || (b = h) a || (a = k) var c = !1, d = null, f = this.db.transaction( [this.storeName], this.consts.READ_WRITE ) f.oncomplete = function () { ;(c ? a : b)(d) } f.onabort = b f.onerror = b var e = f.objectStore(this.storeName).clear() e.onsuccess = function (a) { c = !0 d = a.target.result } e.onerror = b return f }, _addIdPropertyIfNeeded: function (a) { "undefined" == typeof a[this.keyPath] && (a[this.keyPath] = this._insertIdCount++ + Date.now()) }, getIndexList: function () { return this.store.indexNames }, hasIndex: function (a) { return this.store.indexNames.contains(a) }, normalizeIndexData: function (a) { a.keyPath = a.keyPath || a.name a.unique = !!a.unique a.multiEntry = !!a.multiEntry }, indexComplies: function (a, b) { return ["keyPath", "unique", "multiEntry"].every( function (c) { if ( "multiEntry" == c && void 0 === a[c] && !1 === b[c] ) return !0 if ( "keyPath" == c && "[object Array]" == Object.prototype.toString.call(b[c]) ) { c = b.keyPath var d = a.keyPath if ("string" == typeof d) return c.toString() == d if ( ("function" != typeof d.contains && "function" != typeof d.indexOf) || d.length !== c.length ) return !1 for (var f = 0, e = c.length; f < e; f++) if ( !( (d.contains && d.contains(c[f])) || d.indexOf(-1 !== c[f]) ) ) return !1 return !0 } return b[c] == a[c] } ) }, iterate: function (a, b) { b = p( { index: null, order: "ASC", autoContinue: !0, filterDuplicates: !1, keyRange: null, writeAccess: !1, onEnd: null, onError: h, limit: Infinity, offset: 0, allowItemRejection: !1, }, b || {} ) var c = "desc" == b.order.toLowerCase() ? "PREV" : "NEXT" b.filterDuplicates && (c += "_NO_DUPLICATE") var d = !1, f = this.db.transaction( [this.storeName], this.consts[ b.writeAccess ? "READ_WRITE" : "READ_ONLY" ] ), e = f.objectStore(this.storeName) b.index && (e = e.index(b.index)) var g = 0 f.oncomplete = function () { if (d) if (b.onEnd) b.onEnd() else a(null) else b.onError(null) } f.onabort = b.onError f.onerror = b.onError c = e.openCursor(b.keyRange, this.consts[c]) c.onerror = b.onError c.onsuccess = function (c) { if ((c = c.target.result)) if (b.offset) c.advance(b.offset), (b.offset = 0) else { var e = a(c.value, c, f) ;(b.allowItemRejection && !1 === e) || g++ if (b.autoContinue) if (g + b.offset < b.limit) c["continue"]() else d = !0 } else d = !0 } return f }, query: function (a, b) { var c = [], d = 0 b = b || {} b.autoContinue = !0 b.writeAccess = !1 b.allowItemRejection = !!b.filter b.onEnd = function () { a(c, d) } return this.iterate(function (a) { d++ var e = b.filter ? b.filter(a) : !0 !1 !== e && c.push(a) return e }, b) }, count: function (a, b) { b = p({ index: null, keyRange: null }, b || {}) var c = b.onError || h, d = !1, f = null, e = this.db.transaction( [this.storeName], this.consts.READ_ONLY ) e.oncomplete = function () { ;(d ? a : c)(f) } e.onabort = c e.onerror = c var g = e.objectStore(this.storeName) b.index && (g = g.index(b.index)) g = g.count(b.keyRange) g.onsuccess = function (a) { d = !0 f = a.target.result } g.onError = c return e }, makeKeyRange: function (a) { var b = "undefined" != typeof a.lower, c = "undefined" != typeof a.upper, d = "undefined" != typeof a.only switch (!0) { case d: a = this.keyRange.only(a.only) break case b && c: a = this.keyRange.bound( a.lower, a.upper, a.excludeLower, a.excludeUpper ) break case b: a = this.keyRange.lowerBound( a.lower, a.excludeLower ) break case c: a = this.keyRange.upperBound( a.upper, a.excludeUpper ) break default: throw Error( 'Cannot create KeyRange. Provide one or both of "lower" or "upper" value, or an "only" value.' ) } return a }, }, u = {} q.prototype = t q.version = t.version return q }, unsafeWindow ) } x = new IDBStore(obj) }) }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [obj], }) { obj = maketype(obj, ["object"]) return end() } ) a.readfileslow = newfunc( function readfileslow( file, type = "Text", cb1 = (e) => e, cb2 = (e) => e ) { var fileSize = file.size var chunkSize = 64 * 1024 * 50 var offset = 0 var chunkReaderBlock = null var arr = [] var lastidx var readEventHandler = function (evt, idx) { if (evt.target.error == null) { arr.push([idx, evt.target.result]) cb1(a.rerange(arr.length, 0, lastidx, 0, 100)) if (arr.length === lastidx) cb2(arr.sort((e) => e[0]).map((e) => e[1])) } else { return error("Read error: " + evt.target.error) } } chunkReaderBlock = function (_offset, length, _file, idx) { var r = new FileReader() var blob = _file.slice(_offset, length + _offset) const zzz = idx + 1 r.onload = function (e) { readEventHandler(e, zzz - 1) } r["readAs" + type](blob) } let idx = 0 while (offset < fileSize) { idx++ chunkReaderBlock(offset, chunkSize, file, idx) offset += chunkSize } lastidx = idx }, function ({ end, args: [file, type, cb1, cb2], maketype }) { file = maketype(file, ["object"]) // Ensure file is an object type = maketype(type, ["string"]) // Ensure type is a string cb1 = maketype(cb1, ["function"]) // Ensure cb1 is a function cb2 = maketype(cb2, ["function"]) // Ensure cb2 is a function return end() } ) a.cbtoasync = newfunc( function cbtoasync(func, ...args) { return new Promise(function (resolve) { func(...args, resolve) }) }, function ({ end, args: [func, ...args], maketype }) { func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.asynctocb = newfunc( function asynctocb(func, ...args) { var cb = args.pop() return func(...args).then(cb) }, function ({ end, args: [func, ...args], maketype }) { func = maketype(func, ["function"]) // Ensure func is a function return end() } ) a.randstr = newfunc( function randstr({ lower = true, upper = false, number = false, symbol = false, length = 20, }) { var rand = "" a.repeat(() => { rand += a.randfrom( `${lower ? "asdfghjklzxcvbnmqwertyuiop" : ""}${ upper ? "ASDFGHJKLQWERTYUIOPZXCVBNM" : "" }${number ? "0123456789" : ""}${ symbol ? ",./;'[]-=\\`~!@#$%^&*()_+|{}:\"<>?" : "" }`.split("") ) }, length) return rand }, function ({ end, maketype, args: [options], ifunset }) { ifunset([ { lower: true, upper: false, number: false, symbol: false, length: 20, }, ]) options = maketype(options, ["object", "undefined"]) // Ensure options is an object return end() } ) a.toplaces = newfunc( function toplaces(num, pre, post = 0, func = Math.round) { num = String(num).split(".") if (num.length == 1) num.push("") if (pre !== undefined) { num[0] = num[0].substring(num[0].length - pre, num[0].length) while (num[0].length < pre) num[0] = "0" + num[0] } var temp = num[1].substring(post, post + 1) ?? 0 num[1] = num[1].substring(0, post) while (num[1].length < post) num[1] += "0" if (post > 0) { temp = func(num[1].at(-1) + "." + temp) num[1] = num[1].split("") num[1].pop() num[1].push(temp) num[1] = num[1].join("") num = num.join(".") } else num = num[0] return num }, function ({ end, maketype, args: [num, pre, post, func] }) { num = maketype(num, ["number"]) // Ensure num is a number pre = maketype(pre, ["number", "undefined"]) // Ensure pre is a number post = maketype(post, ["number"]) // Ensure post is a number func = maketype(func, ["function", "undefined"]) // Ensure func is a function or undefined return end() } ) a.fetch = newfunc( async function fetch(url, type = "text", ...args) { return await (await fetch(url, ...args))[type]() }, function ({ end, maketype, args: [url, type], makeenum }) { url = maketype(url, ["string"]) // Ensure url is a string type = maketype(type, ["string"]) // Ensure type is a string type = makeenum(type, ["text", "json"]) return end() } ) a.replaceall = newfunc( function replaceall(text, regex, replacewith) { return text.replaceAll( a.toregex(String(a.toregex(regex)) + "g"), replacewith ) }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [text, regex, replacewith], }) { text = maketype(text, ["string"]) regex = maketype(regex, ["regex"]) replacewith = maketype(replacewith, ["string"]) return end() } ) a.setrange = newfunc( function setrange(num, min, max) { return num < min ? min : num > max ? max : num }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [num, min, max], }) { num = maketype(num, ["number"]) min = maketype(min, ["number"]) max = maketype(max, ["number"]) return end() } ) a.ondrop = newfunc( function ondrop(obj) { if (!obj.types) obj.types = "all" obj.types = a.toarray(obj.types) if (!obj.func) throw new Error('object is missing "func"') var oldelem = obj.elem if (obj.elem) obj.elem = a.toelem(obj.elem, true) if (obj.elem && !a.gettype(obj.elem, "element")) throw new Error( `elem is not an elem, ${oldelem} -> ${obj.elem}` ) drops.push(obj) return obj }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [obj], }) { obj = maketype(obj, ["object"]) return end() } ) a.clamp = newfunc( function clamp(num, min, max) { if (min !== undefined && num < min) num = min if (max !== undefined && num > max) num = max return num }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [num, min, max], }) { ifunset(undefined, undefined, undefined) num = maketype(num, ["number"]) min = maketype(min, ["number", "undefined"]) max = maketype(max, ["number", "undefined"]) return end() } ) a.step = newfunc( function step(num, step) { return Math.round(num / step) * step }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [num, step], }) { num = maketype(num, ["number"]) step = maketype(step, ["number"]) return end() } ) a.download = newfunc( function download( data, //string|file|blob filename = "temp.txt", //string|undefined type = "text/plain", //string|undefined isurl = false //boolean|undefined ) { var url if (isurl) { url = data } else { if (a.gettype(data, "string")) var file = new Blob([data], { type, }) else if (a.gettype(data, ["file", "blob"])) { filename = data.name var file = data } url = URL.createObjectURL(file) } var link = document.createElement("a") link.href = url link.download = filename a.bodyload().then(() => { document.body.appendChild(link) link.click() document.body.removeChild(link) if (!isurl) URL.revokeObjectURL(url) }) }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [data, filename, type, isurl], }) { ifunset(undefined, "temp.txt", "text/plain", false) data = maketype(data, ["string", "blob", "file"]) filename = maketype(filename, ["string", "undefined"]) type = maketype(type, ["string", "undefined"]) isurl = maketype(isurl, ["boolean", "undefined"]) return end() } ) a.maketable = newfunc( function maketable(tableData) { const table = a.newelem("table", {}, []) var tbody var first = true for (var tableRow of tableData) { var tr = a.newelem("tr") if (tbody) { tbody.appendChild(tr) } else { table.appendChild(a.newelem("thead", {}, [tr])) table.appendChild((tbody = a.newelem("tbody"))) } for (var data of tableRow) { var type = first ? "th" : "td" if (data == null) { var elem = a.newelem(type, {}) } else if (a.gettype(data, "string")) { var elem = a.newelem(type, { innerHTML: data }) } else if (a.gettype(data, "object")) { var elem = a.newelem(type, data) } else if (a.gettype(data, "array")) { var elem = a.newelem(type, {}, data) } else if (a.gettype(data, "element")) { var elem = a.newelem(type, {}, [data]) } tr.appendChild(elem) } first = false } return table }, function ({ ifunset, gettype, end, maketype, makeenum, trymaketype, trymakeenum, trygettype, args: [tableData], }) { tableData = maketype(tableData, ["array"]) return end() } ) loadlib("libloader").savelib("allfuncs", a) })()