WhiteSevsUtils

一个好用的工具类

目前为 2024-03-15 提交的版本。查看 最新版本

此脚本不应直接安装,它是一个供其他脚本使用的外部库。如果您需要使用该库,请在脚本元属性加入:// @require https://update.gf.qytechs.cn/scripts/455186/1343163/WhiteSevsUtils.js

  1. /// <reference path="../库.ts/Utils.d.ts" />
  2.  
  3. /**
  4. * 方便好用的工具类
  5. * @copyright GPL-3.0-only
  6. * @author WhiteSev
  7. **/
  8. (function (global, factory) {
  9. /**
  10. * 不使用define
  11. * typeof define === "function" && define.amd
  12. * define(factory)
  13. */
  14. if (typeof exports === "object" && typeof module !== "undefined") {
  15. /* 适用于NodeJs或typeScript */
  16. module.exports = factory();
  17. } else {
  18. global = typeof globalThis !== "undefined" ? globalThis : global || self;
  19. /* 适用于浏览器中,且this对象是window,如果this是其它,那么会在其它对象下注册(不可用)对象 */
  20. global.Utils = factory(global.Utils);
  21. }
  22. })(typeof window !== "undefined" ? window : this, function (AnotherUtils) {
  23. const OriginPrototype = {
  24. Array: {
  25. isArray: Array.isArray,
  26. },
  27. Function: {
  28. hasOwnProperty: globalThis.Function.prototype.hasOwnProperty,
  29. apply: globalThis.Function.prototype.apply,
  30. call: globalThis.Function.prototype.call,
  31. },
  32. Object: {
  33. assign: globalThis.Object.assign,
  34. defineProperty: globalThis.Object.defineProperty,
  35. create: globalThis.Object.create,
  36. entries: globalThis.Object.entries,
  37. freeze: globalThis.Object.freeze,
  38. getOwnPropertySymbols: globalThis.Object.getOwnPropertySymbols,
  39. getOwnPropertyDescriptor: globalThis.Object.getOwnPropertyDescriptor,
  40. getPrototypeOf: globalThis.Object.getPrototypeOf,
  41. hasOwnProperty: globalThis.Object.hasOwnProperty,
  42. keys: globalThis.Object.keys,
  43. toString: globalThis.Object.toString,
  44. values: globalThis.Object.values,
  45. },
  46. Reflect: {
  47. deleteProperty: globalThis.Reflect.deleteProperty,
  48. get: globalThis.Reflect.get,
  49. has: globalThis.Reflect.has,
  50. set: globalThis.Reflect.set,
  51. },
  52. URL: {
  53. createObjectURL: globalThis.URL.createObjectURL,
  54. },
  55. fetch: globalThis.fetch,
  56. setTimeout: globalThis.setTimeout,
  57. clearTimeout: globalThis.clearTimeout,
  58. setInterval: globalThis.setInterval,
  59. clearInterval: globalThis.clearInterval,
  60. };
  61. /** @type {Utils} */
  62. const Utils = {};
  63. Utils.version = "2024-3-15";
  64.  
  65. Utils.assign = function (target = {}, source = {}, isAdd = false) {
  66. if (OriginPrototype.Array.isArray(source)) {
  67. let canTraverse = source.filter((item) => {
  68. return typeof item === "object";
  69. });
  70. if (!canTraverse.length) {
  71. return source;
  72. }
  73. }
  74. if (isAdd) {
  75. for (const sourceKeyName in source) {
  76. const targetKeyName = sourceKeyName;
  77. let targetValue = target[targetKeyName];
  78. let sourceValue = source[sourceKeyName];
  79. if (
  80. sourceKeyName in target &&
  81. typeof sourceValue === "object" &&
  82. !Utils.isDOM(sourceValue)
  83. ) {
  84. /* 源端的值是object类型,且不是元素节点 */
  85. target[sourceKeyName] = Utils.assign(targetValue, sourceValue, isAdd);
  86. continue;
  87. }
  88. target[sourceKeyName] = sourceValue;
  89. }
  90. } else {
  91. for (const targetKeyName in target) {
  92. if (targetKeyName in source) {
  93. let targetValue = target[targetKeyName];
  94. let sourceValue = source[targetKeyName];
  95. if (
  96. typeof sourceValue === "object" &&
  97. !Utils.isDOM(sourceValue) &&
  98. OriginPrototype.Object.keys(sourceValue).length
  99. ) {
  100. /* 源端的值是object类型,且不是元素节点 */
  101. target[targetKeyName] = Utils.assign(
  102. targetValue,
  103. sourceValue,
  104. isAdd
  105. );
  106. continue;
  107. }
  108. /* 直接赋值 */
  109. target[targetKeyName] = sourceValue;
  110. }
  111. }
  112. }
  113.  
  114. return target;
  115. };
  116.  
  117. Utils.ajaxHooker = function () {
  118. "use strict";
  119. const win = window.unsafeWindow || document.defaultView || window;
  120. const toString = OriginPrototype.Object.toString;
  121. const getDescriptor = OriginPrototype.Object.getOwnPropertyDescriptor;
  122. const hookFns = [];
  123. const realXhr = win.XMLHttpRequest;
  124. const realFetch = win.fetch;
  125. const resProto = win.Response.prototype;
  126. const xhrResponses = ["response", "responseText", "responseXML"];
  127. const fetchResponses = ["arrayBuffer", "blob", "formData", "json", "text"];
  128. const fetchInitProps = [
  129. "method",
  130. "headers",
  131. "body",
  132. "mode",
  133. "credentials",
  134. "cache",
  135. "redirect",
  136. "referrer",
  137. "referrerPolicy",
  138. "integrity",
  139. "keepalive",
  140. "signal",
  141. "priority",
  142. ];
  143. const xhrAsyncEvents = ["readystatechange", "load", "loadend"];
  144. let filter;
  145. function emptyFn() {}
  146. function errorFn(err) {
  147. console.error(err);
  148. }
  149. function defineProp(obj, prop, getter, setter) {
  150. OriginPrototype.Object.defineProperty(obj, prop, {
  151. configurable: true,
  152. enumerable: true,
  153. get: getter,
  154. set: setter,
  155. });
  156. }
  157. function readonly(obj, prop, value = obj[prop]) {
  158. defineProp(obj, prop, () => value, emptyFn);
  159. }
  160. function writable(obj, prop, value = obj[prop]) {
  161. OriginPrototype.Object.defineProperty(obj, prop, {
  162. configurable: true,
  163. enumerable: true,
  164. writable: true,
  165. value: value,
  166. });
  167. }
  168. function shouldFilter(type, url, method, async) {
  169. return (
  170. filter &&
  171. !filter.find((obj) => {
  172. switch (true) {
  173. case obj.type && obj.type !== type:
  174. case toString.call(obj.url) === "[object String]" &&
  175. !url.includes(obj.url):
  176. case toString.call(obj.url) === "[object RegExp]" &&
  177. !obj.url.test(url):
  178. case obj.method &&
  179. obj.method.toUpperCase() !== method.toUpperCase():
  180. case "async" in obj && obj.async !== async:
  181. return false;
  182. }
  183. return true;
  184. })
  185. );
  186. }
  187. function parseHeaders(obj) {
  188. const headers = {};
  189. switch (toString.call(obj)) {
  190. case "[object String]":
  191. for (const line of obj.trim().split(/[\r\n]+/)) {
  192. const parts = line.split(/\s*:\s*/);
  193. if (parts.length !== 2) continue;
  194. const lheader = parts[0].toLowerCase();
  195. if (lheader in headers) {
  196. headers[lheader] += ", " + parts[1];
  197. } else {
  198. headers[lheader] = parts[1];
  199. }
  200. }
  201. return headers;
  202. case "[object Headers]":
  203. for (const [key, val] of obj) {
  204. headers[key] = val;
  205. }
  206. return headers;
  207. case "[object Object]":
  208. return { ...obj };
  209. default:
  210. return headers;
  211. }
  212. }
  213. class AHRequest {
  214. constructor(request) {
  215. this.request = request;
  216. this.requestClone = { ...this.request };
  217. this.response = {};
  218. }
  219. waitForHookFns() {
  220. return Promise.all(
  221. hookFns.map((fn) => {
  222. try {
  223. return Promise.resolve(fn(this.request)).then(emptyFn, errorFn);
  224. } catch (err) {
  225. console.error(err);
  226. }
  227. })
  228. );
  229. }
  230. waitForResponseFn() {
  231. try {
  232. return Promise.resolve(this.request.response(this.response)).then(
  233. emptyFn,
  234. errorFn
  235. );
  236. } catch (err) {
  237. console.error(err);
  238. return Promise.resolve();
  239. }
  240. }
  241. waitForRequestKeys() {
  242. if (this.reqPromise) return this.reqPromise;
  243. const requestKeys = ["url", "method", "abort", "headers", "data"];
  244. return (this.reqPromise = this.waitForHookFns().then(() =>
  245. Promise.all(
  246. requestKeys.map((key) =>
  247. Promise.resolve(this.request[key]).then(
  248. (val) => (this.request[key] = val),
  249. (e) => (this.request[key] = this.requestClone[key])
  250. )
  251. )
  252. )
  253. ));
  254. }
  255. waitForResponseKeys() {
  256. if (this.resPromise) return this.resPromise;
  257. const responseKeys =
  258. this.request.type === "xhr" ? xhrResponses : fetchResponses;
  259. return (this.resPromise = this.waitForResponseFn().then(() =>
  260. Promise.all(
  261. responseKeys.map((key) => {
  262. const descriptor = getDescriptor(this.response, key);
  263. if (descriptor && "value" in descriptor) {
  264. return Promise.resolve(descriptor.value).then(
  265. (val) => (this.response[key] = val),
  266. (e) => delete this.response[key]
  267. );
  268. } else {
  269. delete this.response[key];
  270. }
  271. })
  272. )
  273. ));
  274. }
  275. }
  276. class XhrEvents {
  277. constructor() {
  278. this.events = {};
  279. }
  280. add(type, event) {
  281. if (type.startsWith("on")) {
  282. this.events[type] = typeof event === "function" ? event : null;
  283. } else {
  284. this.events[type] = this.events[type] || new Set();
  285. this.events[type].add(event);
  286. }
  287. }
  288. remove(type, event) {
  289. if (type.startsWith("on")) {
  290. this.events[type] = null;
  291. } else {
  292. this.events[type] && this.events[type].delete(event);
  293. }
  294. }
  295. _sIP() {
  296. this.ajaxHooker_isStopped = true;
  297. }
  298. trigger(e) {
  299. if (e.ajaxHooker_isTriggered || e.ajaxHooker_isStopped) return;
  300. e.stopImmediatePropagation = this._sIP;
  301. this.events[e.type] &&
  302. this.events[e.type].forEach((fn) => {
  303. !e.ajaxHooker_isStopped && fn.call(e.target, e);
  304. });
  305. this.events["on" + e.type] &&
  306. this.events["on" + e.type].call(e.target, e);
  307. e.ajaxHooker_isTriggered = true;
  308. }
  309. clone() {
  310. const eventsClone = new XhrEvents();
  311. for (const type in this.events) {
  312. if (type.startsWith("on")) {
  313. eventsClone.events[type] = this.events[type];
  314. } else {
  315. eventsClone.events[type] = new Set([...this.events[type]]);
  316. }
  317. }
  318. return eventsClone;
  319. }
  320. }
  321. const xhrMethods = {
  322. readyStateChange(e) {
  323. if (e.target.readyState === 4) {
  324. e.target.dispatchEvent(
  325. new CustomEvent("ajaxHooker_responseReady", { detail: e })
  326. );
  327. } else {
  328. e.target.__ajaxHooker.eventTrigger(e);
  329. }
  330. },
  331. asyncListener(e) {
  332. e.target.__ajaxHooker.eventTrigger(e);
  333. },
  334. setRequestHeader(header, value) {
  335. const ah = this.__ajaxHooker;
  336. ah.originalXhr.setRequestHeader(header, value);
  337. if (this.readyState !== 1) return;
  338. if (header in ah.headers) {
  339. ah.headers[header] += ", " + value;
  340. } else {
  341. ah.headers[header] = value;
  342. }
  343. },
  344. addEventListener(...args) {
  345. const ah = this.__ajaxHooker;
  346. if (xhrAsyncEvents.includes(args[0])) {
  347. ah.proxyEvents.add(args[0], args[1]);
  348. } else {
  349. ah.originalXhr.addEventListener(...args);
  350. }
  351. },
  352. removeEventListener(...args) {
  353. const ah = this.__ajaxHooker;
  354. if (xhrAsyncEvents.includes(args[0])) {
  355. ah.proxyEvents.remove(args[0], args[1]);
  356. } else {
  357. ah.originalXhr.removeEventListener(...args);
  358. }
  359. },
  360. open(method, url, async = true, ...args) {
  361. const ah = this.__ajaxHooker;
  362. ah.url = url.toString();
  363. ah.method = method.toUpperCase();
  364. ah.async = !!async;
  365. ah.openArgs = args;
  366. ah.headers = {};
  367. for (const key of xhrResponses) {
  368. ah.proxyProps[key] = {
  369. get: () => {
  370. const val = ah.originalXhr[key];
  371. ah.originalXhr.dispatchEvent(
  372. new CustomEvent("ajaxHooker_readResponse", {
  373. detail: { key, val },
  374. })
  375. );
  376. return val;
  377. },
  378. };
  379. }
  380. return ah.originalXhr.open(method, url, ...args);
  381. },
  382. sendFactory(realSend) {
  383. return function (data) {
  384. const ah = this.__ajaxHooker;
  385. const xhr = ah.originalXhr;
  386. if (xhr.readyState !== 1) return realSend.call(xhr, data);
  387. ah.eventTrigger = (e) => ah.proxyEvents.trigger(e);
  388. if (shouldFilter("xhr", ah.url, ah.method, ah.async)) {
  389. xhr.addEventListener(
  390. "ajaxHooker_responseReady",
  391. (e) => {
  392. ah.eventTrigger(e.detail);
  393. },
  394. { once: true }
  395. );
  396. return realSend.call(xhr, data);
  397. }
  398. const request = {
  399. type: "xhr",
  400. url: ah.url,
  401. method: ah.method,
  402. abort: false,
  403. headers: ah.headers,
  404. data: data,
  405. response: null,
  406. async: ah.async,
  407. };
  408. if (!ah.async) {
  409. const requestClone = { ...request };
  410. hookFns.forEach((fn) => {
  411. try {
  412. toString.call(fn) === "[object Function]" && fn(request);
  413. } catch (err) {
  414. console.error(err);
  415. }
  416. });
  417. for (const key in request) {
  418. if (toString.call(request[key]) === "[object Promise]") {
  419. request[key] = requestClone[key];
  420. }
  421. }
  422. xhr.open(request.method, request.url, ah.async, ...ah.openArgs);
  423. for (const header in request.headers) {
  424. xhr.setRequestHeader(header, request.headers[header]);
  425. }
  426. data = request.data;
  427. xhr.addEventListener(
  428. "ajaxHooker_responseReady",
  429. (e) => {
  430. ah.eventTrigger(e.detail);
  431. },
  432. { once: true }
  433. );
  434. realSend.call(xhr, data);
  435. if (toString.call(request.response) === "[object Function]") {
  436. const response = {
  437. finalUrl: xhr.responseURL,
  438. status: xhr.status,
  439. responseHeaders: parseHeaders(xhr.getAllResponseHeaders()),
  440. };
  441. for (const key of xhrResponses) {
  442. defineProp(
  443. response,
  444. key,
  445. () => {
  446. return (response[key] = ah.originalXhr[key]);
  447. },
  448. (val) => {
  449. if (toString.call(val) !== "[object Promise]") {
  450. delete response[key];
  451. response[key] = val;
  452. }
  453. }
  454. );
  455. }
  456. try {
  457. request.response(response);
  458. } catch (err) {
  459. console.error(err);
  460. }
  461. for (const key of xhrResponses) {
  462. ah.proxyProps[key] = { get: () => response[key] };
  463. }
  464. }
  465. return;
  466. }
  467. const req = new AHRequest(request);
  468. req.waitForRequestKeys().then(() => {
  469. if (request.abort) return;
  470. xhr.open(request.method, request.url, ...ah.openArgs);
  471. for (const header in request.headers) {
  472. xhr.setRequestHeader(header, request.headers[header]);
  473. }
  474. data = request.data;
  475. xhr.addEventListener(
  476. "ajaxHooker_responseReady",
  477. (e) => {
  478. if (typeof request.response !== "function")
  479. return ah.eventTrigger(e.detail);
  480. req.response = {
  481. finalUrl: xhr.responseURL,
  482. status: xhr.status,
  483. responseHeaders: parseHeaders(xhr.getAllResponseHeaders()),
  484. };
  485. for (const key of xhrResponses) {
  486. defineProp(
  487. req.response,
  488. key,
  489. () => {
  490. return (req.response[key] = ah.originalXhr[key]);
  491. },
  492. (val) => {
  493. delete req.response[key];
  494. req.response[key] = val;
  495. }
  496. );
  497. }
  498. const resPromise = req.waitForResponseKeys().then(() => {
  499. for (const key of xhrResponses) {
  500. if (!(key in req.response)) continue;
  501. ah.proxyProps[key] = {
  502. get: () => {
  503. const val = req.response[key];
  504. xhr.dispatchEvent(
  505. new CustomEvent("ajaxHooker_readResponse", {
  506. detail: { key, val },
  507. })
  508. );
  509. return val;
  510. },
  511. };
  512. }
  513. });
  514. xhr.addEventListener("ajaxHooker_readResponse", (e) => {
  515. const descriptor = getDescriptor(req.response, e.detail.key);
  516. if (!descriptor || "get" in descriptor) {
  517. req.response[e.detail.key] = e.detail.val;
  518. }
  519. });
  520. const eventsClone = ah.proxyEvents.clone();
  521. ah.eventTrigger = (event) =>
  522. resPromise.then(() => eventsClone.trigger(event));
  523. ah.eventTrigger(e.detail);
  524. },
  525. { once: true }
  526. );
  527. realSend.call(xhr, data);
  528. });
  529. };
  530. },
  531. };
  532. function fakeXhr() {
  533. const xhr = new realXhr();
  534. let ah = xhr.__ajaxHooker;
  535. let xhrProxy = xhr;
  536. if (!ah) {
  537. const proxyEvents = new XhrEvents();
  538. ah = xhr.__ajaxHooker = {
  539. headers: {},
  540. originalXhr: xhr,
  541. proxyProps: {},
  542. proxyEvents: proxyEvents,
  543. eventTrigger: (e) => proxyEvents.trigger(e),
  544. toJSON: emptyFn, // Converting circular structure to JSON
  545. };
  546. xhrProxy = new Proxy(xhr, {
  547. get(target, prop) {
  548. try {
  549. if (target === xhr) {
  550. if (prop in ah.proxyProps) {
  551. const descriptor = ah.proxyProps[prop];
  552. return descriptor.get ? descriptor.get() : descriptor.value;
  553. }
  554. if (typeof xhr[prop] === "function") return xhr[prop].bind(xhr);
  555. }
  556. } catch (err) {
  557. console.error(err);
  558. }
  559. return target[prop];
  560. },
  561. set(target, prop, value) {
  562. try {
  563. if (target === xhr && prop in ah.proxyProps) {
  564. const descriptor = ah.proxyProps[prop];
  565. descriptor.set
  566. ? descriptor.set(value)
  567. : (descriptor.value = value);
  568. } else {
  569. target[prop] = value;
  570. }
  571. } catch (err) {
  572. console.error(err);
  573. }
  574. return true;
  575. },
  576. });
  577. xhr.addEventListener("readystatechange", xhrMethods.readyStateChange);
  578. xhr.addEventListener("load", xhrMethods.asyncListener);
  579. xhr.addEventListener("loadend", xhrMethods.asyncListener);
  580. for (const evt of xhrAsyncEvents) {
  581. const onEvt = "on" + evt;
  582. ah.proxyProps[onEvt] = {
  583. get: () => proxyEvents.events[onEvt] || null,
  584. set: (val) => proxyEvents.add(onEvt, val),
  585. };
  586. }
  587. for (const method of [
  588. "setRequestHeader",
  589. "addEventListener",
  590. "removeEventListener",
  591. "open",
  592. ]) {
  593. ah.proxyProps[method] = { value: xhrMethods[method] };
  594. }
  595. }
  596. ah.proxyProps.send = { value: xhrMethods.sendFactory(xhr.send) };
  597. return xhrProxy;
  598. }
  599. function hookFetchResponse(response, req) {
  600. for (const key of fetchResponses) {
  601. response[key] = () =>
  602. new Promise((resolve, reject) => {
  603. if (key in req.response) return resolve(req.response[key]);
  604. resProto[key].call(response).then((res) => {
  605. req.response[key] = res;
  606. req.waitForResponseKeys().then(() => {
  607. resolve(key in req.response ? req.response[key] : res);
  608. });
  609. }, reject);
  610. });
  611. }
  612. }
  613. function fakeFetch(url, options = {}) {
  614. if (!url) return realFetch.call(win, url, options);
  615. let init = { ...options };
  616. if (toString.call(url) === "[object Request]") {
  617. init = {};
  618. for (const prop of fetchInitProps) init[prop] = url[prop];
  619. ObjectAssign(init, options);
  620. url = url.url;
  621. }
  622. url = url.toString();
  623. init.method = init.method || "GET";
  624. init.headers = init.headers || {};
  625. if (shouldFilter("fetch", url, init.method, true))
  626. return realFetch.call(win, url, init);
  627. const request = {
  628. type: "fetch",
  629. url: url,
  630. method: init.method.toUpperCase(),
  631. abort: false,
  632. headers: parseHeaders(init.headers),
  633. data: init.body,
  634. response: null,
  635. async: true,
  636. };
  637. const req = new AHRequest(request);
  638. return new Promise((resolve, reject) => {
  639. req
  640. .waitForRequestKeys()
  641. .then(() => {
  642. if (request.abort)
  643. return reject(new DOMException("aborted", "AbortError"));
  644. init.method = request.method;
  645. init.headers = request.headers;
  646. init.body = request.data;
  647. realFetch.call(win, request.url, init).then((response) => {
  648. if (typeof request.response === "function") {
  649. req.response = {
  650. finalUrl: response.url,
  651. status: response.status,
  652. responseHeaders: parseHeaders(response.headers),
  653. };
  654. hookFetchResponse(response, req);
  655. response.clone = () => {
  656. const resClone = resProto.clone.call(response);
  657. hookFetchResponse(resClone, req);
  658. return resClone;
  659. };
  660. }
  661. resolve(response);
  662. }, reject);
  663. })
  664. .catch((err) => {
  665. console.error(err);
  666. resolve(realFetch.call(win, url, init));
  667. });
  668. });
  669. }
  670. win.XMLHttpRequest = fakeXhr;
  671. OriginPrototype.Object.keys(realXhr).forEach(
  672. (key) => (fakeXhr[key] = realXhr[key])
  673. );
  674. fakeXhr.prototype = realXhr.prototype;
  675. win.fetch = fakeFetch;
  676. return {
  677. hook: (fn) => hookFns.push(fn),
  678. filter: (arr) => {
  679. filter = OriginPrototype.Array.isArray(arr) && arr;
  680. },
  681. protect: () => {
  682. readonly(win, "XMLHttpRequest", fakeXhr);
  683. readonly(win, "fetch", fakeFetch);
  684. },
  685. unhook: () => {
  686. writable(win, "XMLHttpRequest", realXhr);
  687. writable(win, "fetch", realFetch);
  688. },
  689. };
  690. };
  691.  
  692. Utils.canvasClickByPosition = function (
  693. canvasElement,
  694. clientX = 0,
  695. clientY = 0,
  696. view = globalThis
  697. ) {
  698. if (!canvasElement instanceof HTMLCanvasElement) {
  699. throw new Error(
  700. "Utils.canvasClickByPosition 参数canvasElement必须是canvas元素"
  701. );
  702. }
  703. clientX = parseInt(clientX);
  704. clientY = parseInt(clientY);
  705. const eventInit = {
  706. cancelBubble: true,
  707. cancelable: true,
  708. clientX: clientX,
  709. clientY: clientY,
  710. view: view,
  711. detail: 1,
  712. };
  713. canvas.dispatchEvent(new MouseEvent("mousedown", eventInit));
  714. canvas.dispatchEvent(new MouseEvent("mouseup", eventInit));
  715. };
  716.  
  717. Utils.checkUserClickInNode = function (element) {
  718. if (!Utils.isDOM(element)) {
  719. throw new Error(
  720. "Utils.checkUserClickInNode 参数 targetNode 必须为 Element|Node 类型"
  721. );
  722. }
  723. let mouseClickPosX = Number(window.event.clientX); /* 鼠标相对屏幕横坐标 */
  724. let mouseClickPosY = Number(window.event.clientY); /* 鼠标相对屏幕纵坐标 */
  725. let elementPosXLeft = Number(
  726. element.getBoundingClientRect().left
  727. ); /* 要检测的元素的相对屏幕的横坐标最左边 */
  728. let elementPosXRight = Number(
  729. element.getBoundingClientRect().right
  730. ); /* 要检测的元素的相对屏幕的横坐标最右边 */
  731. let elementPosYTop = Number(
  732. element.getBoundingClientRect().top
  733. ); /* 要检测的元素的相对屏幕的纵坐标最上边 */
  734. let elementPosYBottom = Number(
  735. element.getBoundingClientRect().bottom
  736. ); /* 要检测的元素的相对屏幕的纵坐标最下边 */
  737. let clickNodeHTML = window.event?.target?.innerHTML;
  738. if (
  739. mouseClickPosX >= elementPosXLeft &&
  740. mouseClickPosX <= elementPosXRight &&
  741. mouseClickPosY >= elementPosYTop &&
  742. mouseClickPosY <= elementPosYBottom
  743. ) {
  744. return true;
  745. } else if (clickNodeHTML && element.innerHTML.includes(clickNodeHTML)) {
  746. /* 这种情况是应对在界面中隐藏的元素,getBoundingClientRect获取的都是0 */
  747. return true;
  748. } else {
  749. return false;
  750. }
  751. };
  752.  
  753. Utils.cloneFormData = function (formData) {
  754. let clonedFormData = new FormData();
  755. for (let [key, value] of formData.entries()) {
  756. clonedFormData.append(key, value);
  757. }
  758. return clonedFormData;
  759. };
  760.  
  761. Utils.createOverload = function () {
  762. let fnMap = new Map();
  763. function overload(...args) {
  764. let key = args.map((it) => typeof it).join(",");
  765. let fn = fnMap.get(key);
  766. if (!fn) {
  767. throw new TypeError("没有找到对应的实现");
  768. }
  769. return fn.apply(this, args);
  770. }
  771. overload.addImpl = function (...args) {
  772. let fn = args.pop();
  773. if (typeof fn !== "function") {
  774. throw new TypeError("最后一个参数必须是函数");
  775. }
  776. let key = args.join(",");
  777. fnMap.set(key, fn);
  778. };
  779. return overload;
  780. };
  781.  
  782. Utils.deepClone = function (obj) {
  783. if (obj === void 0) return void 0;
  784. if (obj === null) return null;
  785. let clone = obj instanceof Array ? [] : {};
  786. for (const [key, value] of OriginPrototype.Object.entries(obj)) {
  787. clone[key] = typeof value === "object" ? deepClone(value) : value;
  788. }
  789. return clone;
  790. };
  791.  
  792. Utils.debounce = function (fn, delay = 0) {
  793. let timer = null;
  794. const context = this;
  795. return function (...args) {
  796. OriginPrototype.clearTimeout(timer);
  797. timer = OriginPrototype.setTimeout(function () {
  798. fn.apply(context, args);
  799. }, delay);
  800. };
  801. };
  802.  
  803. Utils.deleteParentNode = function (element, targetSelector) {
  804. if (element == null) {
  805. throw new Error("Utils.deleteParentNode 参数 target 不能为 null");
  806. }
  807. if (!Utils.isDOM(element)) {
  808. throw new Error(
  809. "Utils.deleteParentNode 参数 target 必须为 Node|HTMLElement 类型"
  810. );
  811. }
  812. if (typeof targetSelector !== "string") {
  813. throw new Error(
  814. "Utils.deleteParentNode 参数 targetSelector 必须为 string 类型"
  815. );
  816. }
  817. let result = false;
  818. let needRemoveDOM = element.closest(targetSelector);
  819. if (needRemoveDOM) {
  820. needRemoveDOM.remove();
  821. result = true;
  822. }
  823. return result;
  824. };
  825.  
  826. Utils.Dictionary = function () {
  827. class UtilsDictionary {
  828. items = {};
  829. /**
  830. * 检查是否有某一个键
  831. * @param {string} key 键
  832. * @returns {boolean}
  833. */
  834. has(key) {
  835. return this.items.hasOwnProperty(key);
  836. }
  837. /**
  838. * 检查已有的键中是否以xx开头
  839. * @param {string} key 需要匹配的键
  840. * @returns {boolean}
  841. */
  842. startsWith(key) {
  843. let allKeys = this.keys();
  844. for (const keyName of allKeys) {
  845. if (keyName.startsWith(key)) {
  846. return true;
  847. }
  848. }
  849. return false;
  850. }
  851. /**
  852. * 获取以xx开头的键的值
  853. * @param {string} key 需要匹配的键
  854. * @returns {any}
  855. */
  856. getStartsWith(key) {
  857. let allKeys = this.keys();
  858. for (const keyName of allKeys) {
  859. if (keyName.startsWith(key)) {
  860. return this.items[keyName];
  861. }
  862. }
  863. }
  864. /**
  865. * 为字典添加某一个值
  866. * @param {string} key 键
  867. * @param {any} val 值,默认为""
  868. */
  869. set(key, val = "") {
  870. if (key === void 0) {
  871. throw new Error("Utils.Dictionary().set 参数 key 不能为空");
  872. }
  873. this.items[key] = val;
  874. }
  875. /**
  876. * 删除某一个键
  877. * @param {string} key 键
  878. * @returns {boolean}
  879. */
  880. delete(key) {
  881. if (this.has(key)) {
  882. OriginPrototype.Reflect.deleteProperty(this.items, key);
  883. return true;
  884. }
  885. return false;
  886. }
  887. /**
  888. * 获取某个键的值
  889. * @param {string} key 键
  890. * @returns {any}
  891. */
  892. get(key) {
  893. return this.has(key) ? this.items[key] : void 0;
  894. }
  895. /**
  896. * 返回字典中的所有值
  897. * @returns {any[]}
  898. */
  899. values() {
  900. let resultList = [];
  901. for (let prop in this.items) {
  902. if (this.has(prop)) {
  903. resultList.push(this.items[prop]);
  904. }
  905. }
  906. return resultList;
  907. }
  908. /**
  909. * 清空字典
  910. */
  911. clear() {
  912. this.items = {};
  913. }
  914. /**
  915. * 获取字典的长度
  916. * @returns {number}
  917. */
  918. size() {
  919. return OriginPrototype.Object.keys(this.items).length;
  920. }
  921. /**
  922. * 获取字典所有的键
  923. */
  924. keys() {
  925. return OriginPrototype.Object.keys(this.items);
  926. }
  927. /**
  928. * 返回字典本身
  929. * @returns {object}
  930. */
  931. getItems() {
  932. return this.items;
  933. }
  934. /**
  935. * 合并另一个字典
  936. * @param {object} data 需要合并的字典
  937. */
  938. concat(data) {
  939. this.items = Utils.assign(this.items, data.getItems());
  940. }
  941. /**
  942. * 获取字典的长度,同this.size
  943. * @returns {number}
  944. */
  945. get length() {
  946. return this.size();
  947. }
  948. /**
  949. * 迭代器
  950. */
  951. get entries() {
  952. let that = this;
  953. return function* () {
  954. let itemKeys = OriginPrototype.Object.keys(that.getItems());
  955. for (const keyName of itemKeys) {
  956. yield [keyName, that.get(keyName)];
  957. }
  958. };
  959. }
  960. /**
  961. * 是否可遍历
  962. */
  963. get [Symbol.iterator]() {
  964. let that = this;
  965. return function () {
  966. return that.entries();
  967. };
  968. }
  969. /**
  970. * .toString()和.toLocaleString()输出的字符串
  971. */
  972. get [Symbol.toStringTag]() {
  973. return "UtilsDictionary";
  974. }
  975. }
  976. return new UtilsDictionary();
  977. };
  978.  
  979. Utils.dispatchEvent = function (element, eventName, details) {
  980. let eventNameList = [];
  981. if (typeof eventName === "string") {
  982. eventNameList = [eventName];
  983. }
  984. if (OriginPrototype.Array.isArray(eventName)) {
  985. eventNameList = [...eventName];
  986. }
  987. eventNameList.forEach((_eventName_) => {
  988. let event = new Event(_eventName_);
  989. if (details) {
  990. OriginPrototype.Object.assign(event, details);
  991. }
  992. element.dispatchEvent(event);
  993. });
  994. };
  995.  
  996. Utils.downloadBase64 = function (base64Data, fileName, isIFrame = false) {
  997. if (typeof base64Data !== "string") {
  998. throw new Error(
  999. "Utils.downloadBase64 参数 base64Data 必须为 string 类型"
  1000. );
  1001. }
  1002. if (typeof fileName !== "string") {
  1003. throw new Error("Utils.downloadBase64 参数 fileName 必须为 string 类型");
  1004. }
  1005. if (isIFrame) {
  1006. /* 使用iframe */
  1007. const iframeElement = document.createElement("iframe");
  1008. iframeElement.style.display = "none";
  1009. iframeElement.src = base64Data;
  1010. document.body.appendChild(iframeElement);
  1011. OriginPrototype.setTimeout(() => {
  1012. iframeElement.contentWindow.document.execCommand(
  1013. "SaveAs",
  1014. true,
  1015. fileName
  1016. );
  1017. document.body.removeChild(iframeElement);
  1018. }, 100);
  1019. } else {
  1020. /* 使用A标签 */
  1021. const linkElement = document.createElement("a");
  1022. linkElement.setAttribute("target", "_blank");
  1023. linkElement.download = fileName;
  1024. linkElement.href = base64Data;
  1025. linkElement.click();
  1026. }
  1027. };
  1028.  
  1029. Utils.findWebPageVisibleText = function (str = "", caseSensitive = false) {
  1030. let TRange = null;
  1031. let strFound;
  1032. if (window.find) {
  1033. /* CODE FOR BROWSERS THAT SUPPORT window.find */
  1034. strFound = self.find(str, caseSensitive, true, true, false);
  1035. if (strFound && self.getSelection && !self.getSelection().anchorNode) {
  1036. strFound = self.find(str, caseSensitive, true, true, false);
  1037. }
  1038. if (!strFound) {
  1039. strFound = self.find(str, 0, 1);
  1040. while (self.find(str, 0, 1)) continue;
  1041. }
  1042. } else if (navigator.appName.indexOf("Microsoft") != -1) {
  1043. /* EXPLORER-SPECIFIC CODE */
  1044. if (TRange != null) {
  1045. TRange.collapse(false);
  1046. strFound = TRange.findText(str);
  1047. if (strFound) TRange.select();
  1048. }
  1049. if (TRange == null || strFound == 0) {
  1050. TRange = self.document.body.createTextRange();
  1051. strFound = TRange.findText(str);
  1052. if (strFound) TRange.select();
  1053. }
  1054. } else if (navigator.appName == "Opera") {
  1055. alert("Opera browsers not supported, sorry...");
  1056. return;
  1057. }
  1058. return strFound ? true : false;
  1059. };
  1060.  
  1061. Utils.findElementsWithText = function* (element, text, filter) {
  1062. if (element.outerHTML.includes(text)) {
  1063. if (element.children.length === 0) {
  1064. let filterResult =
  1065. typeof filter === "function" ? filter(element) : false;
  1066. if (!filterResult) {
  1067. yield element;
  1068. }
  1069. } else {
  1070. let textElement = Array.from(element.childNodes).filter(
  1071. (ele) => ele.nodeType === Node.TEXT_NODE
  1072. );
  1073. for (let ele of textElement) {
  1074. if (ele.textContent.includes(text)) {
  1075. let filterResult =
  1076. typeof filter === "function" ? filter(element) : false;
  1077. if (!filterResult) {
  1078. yield ele;
  1079. }
  1080. }
  1081. }
  1082. }
  1083. }
  1084.  
  1085. for (let index = 0; index < element.children.length; index++) {
  1086. let childElement = element.children[index];
  1087. yield* Utils.findElementsWithText(childElement, text, filter);
  1088. }
  1089. };
  1090.  
  1091. Utils.findVisibleElement = function (element) {
  1092. let currentElement = element;
  1093. while (currentElement) {
  1094. let elementRect = currentElement.getBoundingClientRect();
  1095. if (Boolean(elementRect.length)) {
  1096. return currentElement;
  1097. }
  1098. currentElement = currentElement.parentElement;
  1099. }
  1100. return null;
  1101. };
  1102.  
  1103. Utils.formatByteToSize = function (byteSize, addType = true) {
  1104. byteSize = parseInt(byteSize);
  1105. if (isNaN(byteSize)) {
  1106. throw new Error("Utils.formatByteToSize 参数 byteSize 格式不正确");
  1107. }
  1108. let result = 0;
  1109. let resultType = "KB";
  1110. let sizeData = {};
  1111. sizeData.B = 1;
  1112. sizeData.KB = 1024;
  1113. sizeData.MB = sizeData.KB * sizeData.KB;
  1114. sizeData.GB = sizeData.MB * sizeData.KB;
  1115. sizeData.TB = sizeData.GB * sizeData.KB;
  1116. sizeData.PB = sizeData.TB * sizeData.KB;
  1117. sizeData.EB = sizeData.PB * sizeData.KB;
  1118. sizeData.ZB = sizeData.EB * sizeData.KB;
  1119. sizeData.YB = sizeData.ZB * sizeData.KB;
  1120. sizeData.BB = sizeData.YB * sizeData.KB;
  1121. sizeData.NB = sizeData.BB * sizeData.KB;
  1122. sizeData.DB = sizeData.NB * sizeData.KB;
  1123. for (let key in sizeData) {
  1124. result = byteSize / sizeData[key];
  1125. resultType = key;
  1126. if (sizeData.KB >= result) {
  1127. break;
  1128. }
  1129. }
  1130. result = result.toFixed(2);
  1131. result = addType ? result + resultType.toString() : parseFloat(result);
  1132. return result;
  1133. };
  1134.  
  1135. Utils.getNodeListValue = function () {
  1136. let resultArray = [];
  1137. for (let arg of arguments) {
  1138. let value = arg;
  1139. if (typeof arg === "function") {
  1140. /* 方法 */
  1141. value = arg();
  1142. }
  1143. if (value.length !== 0) {
  1144. resultArray = [...value];
  1145. break;
  1146. }
  1147. }
  1148. return resultArray;
  1149. };
  1150.  
  1151. Utils.getNonNullValue = function (...args) {
  1152. let resultValue = args[args.length - 1];
  1153. for (const argValue of args) {
  1154. if (Utils.isNotNull(argValue)) {
  1155. resultValue = argValue;
  1156. break;
  1157. }
  1158. }
  1159. return resultValue;
  1160. };
  1161.  
  1162. Utils.formatTime = function (
  1163. text = new Date(),
  1164. formatType = "yyyy-MM-dd HH:mm:ss"
  1165. ) {
  1166. let time = text == null ? new Date() : new Date(text);
  1167. /**
  1168. * 校验时间补0
  1169. * @param {number} timeNum
  1170. * @returns
  1171. */
  1172. function checkTime(timeNum) {
  1173. if (timeNum < 10) return "0" + timeNum;
  1174. return timeNum;
  1175. }
  1176.  
  1177. /**
  1178. * 时间制修改 24小时制转12小时制
  1179. * @param {number} hourNum 小时
  1180. * @returns
  1181. */
  1182. function timeSystemChange(hourNum) {
  1183. return hourNum > 12 ? hourNum - 12 : hourNum;
  1184. }
  1185.  
  1186. let timeRegexp = {
  1187. yyyy: time.getFullYear(),
  1188. /* 年 */
  1189. MM: checkTime(time.getMonth() + 1),
  1190. /* 月 */
  1191. dd: checkTime(time.getDate()),
  1192. /* 日 */
  1193. HH: checkTime(time.getHours()),
  1194. /* 时 (24小时制) */
  1195. hh: checkTime(timeSystemChange(time.getHours())),
  1196. /* 时 (12小时制) */
  1197. mm: checkTime(time.getMinutes()),
  1198. /* 分 */
  1199. ss: checkTime(time.getSeconds()),
  1200. /* 秒 */
  1201. };
  1202. OriginPrototype.Object.keys(timeRegexp).forEach(function (key) {
  1203. let replaecRegexp = new RegExp(key, "g");
  1204. formatType = formatType.replace(replaecRegexp, timeRegexp[key]);
  1205. });
  1206. return formatType;
  1207. };
  1208.  
  1209. Utils.formatToTimeStamp = function (text) {
  1210. /* 把字符串格式的时间(完整,包括日期和时间)格式化成时间 */
  1211. if (typeof text !== "string") {
  1212. throw new Error("Utils.formatToTimeStamp 参数 text 必须为 string 类型");
  1213. }
  1214. if (text.length === 8) {
  1215. /* 该字符串只有时分秒 */
  1216. let today = new Date();
  1217. text =
  1218. today.getFullYear() +
  1219. "-" +
  1220. (today.getMonth() + 1) +
  1221. "-" +
  1222. today.getDate() +
  1223. " " +
  1224. text;
  1225. }
  1226. text = text.substring(0, 19);
  1227. text = text.replace(/-/g, "/");
  1228. let timestamp = new Date(text).getTime();
  1229. return timestamp;
  1230. };
  1231.  
  1232. Utils.GBKEncoder = function () {
  1233. function handleText(text) {
  1234. text = text
  1235. .replace(/#(\d+)\$/g, function (a, b) {
  1236. return Array(+b + 3).join("#");
  1237. })
  1238. .replace(/#/g, "####")
  1239. .replace(/(\w\w):([\w#]+)(?:,|$)/g, function (a, hd, dt) {
  1240. return dt.replace(/../g, function (a) {
  1241. if (a != "##") {
  1242. return hd + a;
  1243. } else {
  1244. return a;
  1245. }
  1246. });
  1247. });
  1248. return text;
  1249. }
  1250. function handleHash() {
  1251. let index = 0;
  1252. data = data.match(/..../g);
  1253. for (let i = 0x81; i <= 0xfe; i++) {
  1254. for (let j = 0x40; j <= 0xfe; j++) {
  1255. U2Ghash[data[index++]] = (
  1256. "%" +
  1257. i.toString(16) +
  1258. "%" +
  1259. j.toString(16)
  1260. ).toUpperCase();
  1261. }
  1262. }
  1263. for (let key in U2Ghash) {
  1264. G2Uhash[U2Ghash[key]] = key;
  1265. }
  1266. }
  1267. function isAscii(unicode) {
  1268. return unicode <= 0x007f && unicode >= 0x0000;
  1269. }
  1270. let data = handleText(
  1271. "4e:020405060f12171f20212326292e2f313335373c40414244464a5155575a5b6263646567686a6b6c6d6e6f727475767778797a7b7c7d7f808182838485878a#909697999c9d9ea3aaafb0b1b4b6b7b8b9bcbdbec8cccfd0d2dadbdce0e2e6e7e9edeeeff1f4f8f9fafcfe,4f:00020304050607080b0c12131415161c1d212328292c2d2e31333537393b3e3f40414244454748494a4b4c525456616266686a6b6d6e7172757778797a7d8081828586878a8c8e909293959698999a9c9e9fa1a2a4abadb0b1b2b3b4b6b7b8b9babbbcbdbec0c1c2c6c7c8c9cbcccdd2d3d4d5d6d9dbe0e2e4e5e7ebecf0f2f4f5f6f7f9fbfcfdff,50:000102030405060708090a#0b0e1011131516171b1d1e20222324272b2f303132333435363738393b3d3f404142444546494a4b4d5051525354565758595b5d5e5f6061626364666768696a6b6d6e6f70717273747578797a7c7d818283848687898a8b8c8e8f909192939495969798999a9b9c9d9e9fa0a1a2a4a6aaabadaeafb0b1b3b4b5b6b7b8b9bcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdced0d1d2d3d4d5d7d8d9dbdcdddedfe0e1e2e3e4e5e8e9eaebeff0f1f2f4f6f7f8f9fafcfdfeff,51:00010203040508#090a0c0d0e0f1011131415161718191a1b1c1d1e1f2022232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e42474a4c4e4f5052535758595b5d5e5f606163646667696a6f727a7e7f838486878a8b8e8f90919394989a9d9e9fa1a3a6a7a8a9aaadaeb4b8b9babebfc1c2c3c5c8cacdced0d2d3d4d5d6d7d8d9dadcdedfe2e3e5e6e7e8e9eaeceef1f2f4f7fe,52:0405090b0c0f101314151c1e1f2122232526272a2c2f313234353c3e4445464748494b4e4f5253555758#595a5b5d5f6062636466686b6c6d6e7071737475767778797a7b7c7e808384858687898a8b8c8d8e8f91929495969798999a9ca4a5a6a7aeafb0b4b5b6b7b8b9babbbcbdc0c1c2c4c5c6c8cacccdcecfd1d3d4d5d7d9dadbdcdddee0e1e2e3e5e6e7e8e9eaebecedeeeff1f2f3f4f5f6f7f8fbfcfd,53:0102030407090a0b0c0e11121314181b1c1e1f2224252728292b2c2d2f3031323334353637383c3d404244464b4c4d505458595b5d65686a6c6d7276797b7c7d7e80818387888a8e8f#90919293949697999b9c9ea0a1a4a7aaabacadafb0b1b2b3b4b5b7b8b9babcbdbec0c3c4c5c6c7cecfd0d2d3d5dadcdddee1e2e7f4fafeff,54:000205070b1418191a1c2224252a303336373a3d3f4142444547494c4d4e4f515a5d5e5f6061636567696a6b6c6d6e6f7074797a7e7f8183858788898a8d919397989c9e9fa0a1a2a5aeb0b2b5b6b7b9babcbec3c5cacbd6d8dbe0e1e2e3e4ebeceff0f1f4f5f6f7f8f9fbfe,55:0002030405080a0b0c0d0e121315161718191a1c1d1e1f212526#28292b2d3234353638393a3b3d40424547484b4c4d4e4f515253545758595a5b5d5e5f60626368696b6f7071727374797a7d7f85868c8d8e9092939596979a9b9ea0a1a2a3a4a5a6a8a9aaabacadaeafb0b2b4b6b8babcbfc0c1c2c3c6c7c8cacbcecfd0d5d7d8d9dadbdee0e2e7e9edeef0f1f4f6f8f9fafbfcff,56:0203040506070a0b0d1011121314151617191a1c1d202122252628292a2b2e2f30333537383a3c3d3e404142434445464748494a4b4f5051525355565a5b5d5e5f6061#636566676d6e6f70727374757778797a7d7e7f80818283848788898a8b8c8d9091929495969798999a9b9c9d9e9fa0a1a2a4a5a6a7a8a9aaabacadaeb0b1b2b3b4b5b6b8b9babbbdbebfc0c1c2c3c4c5c6c7c8c9cbcccdcecfd0d1d2d3d5d6d8d9dce3e5e6e7e8e9eaeceeeff2f3f6f7f8fbfc,57:00010205070b0c0d0e0f101112131415161718191a1b1d1e202122242526272b313234353637383c3d3f414344454648494b52535455565859626365676c6e707172747578797a7d7e7f80#818788898a8d8e8f90919495969798999a9c9d9e9fa5a8aaacafb0b1b3b5b6b7b9babbbcbdbebfc0c1c4c5c6c7c8c9cacccdd0d1d3d6d7dbdcdee1e2e3e5e6e7e8e9eaebeceef0f1f2f3f5f6f7fbfcfeff,58:0103040508090a0c0e0f101213141617181a1b1c1d1f222325262728292b2c2d2e2f31323334363738393a3b3c3d3e3f4041424345464748494a4b4e4f505253555657595a5b5c5d5f6061626364666768696a6d6e6f707172737475767778797a7b7c7d7f82848687888a8b8c#8d8e8f909194959697989b9c9da0a1a2a3a4a5a6a7aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbdbebfc0c2c3c4c6c7c8c9cacbcccdcecfd0d2d3d4d6d7d8d9dadbdcdddedfe0e1e2e3e5e6e7e8e9eaedeff1f2f4f5f7f8fafbfcfdfeff,59:000103050608090a0b0c0e1011121317181b1d1e2021222326282c30323335363b3d3e3f404345464a4c4d505253595b5c5d5e5f616364666768696a6b6c6d6e6f70717275777a7b7c7e7f8085898b8c8e8f90919495989a9b9c9d9fa0a1a2a6#a7acadb0b1b3b4b5b6b7b8babcbdbfc0c1c2c3c4c5c7c8c9cccdcecfd5d6d9dbdedfe0e1e2e4e6e7e9eaebedeeeff0f1f2f3f4f5f6f7f8fafcfdfe,5a:00020a0b0d0e0f101214151617191a1b1d1e2122242627282a2b2c2d2e2f3033353738393a3b3d3e3f414243444547484b4c4d4e4f5051525354565758595b5c5d5e5f60616364656668696b6c6d6e6f7071727378797b7c7d7e808182838485868788898a8b8c8d8e8f9091939495969798999c9d9e9fa0a1a2a3a4a5a6a7a8a9abac#adaeafb0b1b4b6b7b9babbbcbdbfc0c3c4c5c6c7c8cacbcdcecfd0d1d3d5d7d9dadbdddedfe2e4e5e7e8eaecedeeeff0f2f3f4f5f6f7f8f9fafbfcfdfeff,5b:0001020304050607080a0b0c0d0e0f10111213141518191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303133353638393a3b3c3d3e3f4142434445464748494a4b4c4d4e4f52565e606167686b6d6e6f7274767778797b7c7e7f82868a8d8e90919294969fa7a8a9acadaeafb1b2b7babbbcc0c1c3c8c9cacbcdcecf#d1d4d5d6d7d8d9dadbdce0e2e3e6e7e9eaebecedeff1f2f3f4f5f6f7fdfe,5c:0002030507080b0c0d0e10121317191b1e1f2021232628292a2b2d2e2f303233353637434446474c4d5253545657585a5b5c5d5f62646768696a6b6c6d70727374757677787b7c7d7e808384858687898a8b8e8f9293959d9e9fa0a1a4a5a6a7a8aaaeafb0b2b4b6b9babbbcbec0c2c3c5c6c7c8c9cacccdcecfd0d1d3d4d5d6d7d8dadbdcdddedfe0e2e3e7e9ebeceeeff1f2f3f4f5f6f7f8f9fafcfdfeff,5d:00#01040508090a0b0c0d0f10111213151718191a1c1d1f2021222325282a2b2c2f3031323335363738393a3b3c3f4041424344454648494d4e4f5051525354555657595a5c5e5f6061626364656667686a6d6e7071727375767778797a7b7c7d7e7f8081838485868788898a8b8c8d8e8f9091929394959697989a9b9c9e9fa0a1a2a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b8b9babbbcbdbebfc0c1c2c3c4c6c7c8c9cacbcccecfd0d1d2d3d4d5d6d7d8d9dadcdfe0e3e4eaeced#f0f5f6f8f9fafbfcff,5e:000407090a0b0d0e1213171e1f20212223242528292a2b2c2f303233343536393a3e3f404143464748494a4b4d4e4f50515253565758595a5c5d5f60636465666768696a6b6c6d6e6f70717577797e8182838588898c8d8e92989b9da1a2a3a4a8a9aaabacaeafb0b1b2b4babbbcbdbfc0c1c2c3c4c5c6c7c8cbcccdcecfd0d4d5d7d8d9dadcdddedfe0e1e2e3e4e5e6e7e9ebecedeeeff0f1f2f3f5f8f9fbfcfd,5f:050607090c0d0e10121416191a1c1d1e21222324#282b2c2e30323334353637383b3d3e3f4142434445464748494a4b4c4d4e4f5154595a5b5c5e5f60636567686b6e6f72747576787a7d7e7f83868d8e8f919394969a9b9d9e9fa0a2a3a4a5a6a7a9abacafb0b1b2b3b4b6b8b9babbbebfc0c1c2c7c8cacbced3d4d5dadbdcdedfe2e3e5e6e8e9eceff0f2f3f4f6f7f9fafc,60:0708090b0c10111317181a1e1f2223242c2d2e3031323334363738393a3d3e404445464748494a4c4e4f5153545657585b5c5e5f606165666e71727475777e80#8182858687888a8b8e8f909193959798999c9ea1a2a4a5a7a9aaaeb0b3b5b6b7b9babdbebfc0c1c2c3c4c7c8c9cccdcecfd0d2d3d4d6d7d9dbdee1e2e3e4e5eaf1f2f5f7f8fbfcfdfeff,61:02030405070a0b0c1011121314161718191b1c1d1e21222528292a2c2d2e2f303132333435363738393a3b3c3d3e4041424344454647494b4d4f50525354565758595a5b5c5e5f606163646566696a6b6c6d6e6f717273747678797a7b7c7d7e7f808182838485868788898a8c8d8f9091929395#969798999a9b9c9e9fa0a1a2a3a4a5a6aaabadaeafb0b1b2b3b4b5b6b8b9babbbcbdbfc0c1c3c4c5c6c7c9cccdcecfd0d3d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e7e8e9eaebecedeeeff0f1f2f3f4f6f7f8f9fafbfcfdfe,62:00010203040507091314191c1d1e2023262728292b2d2f303132353638393a3b3c424445464a4f50555657595a5c5d5e5f6061626465687172747577787a7b7d818283858687888b8c8d8e8f9094999c9d9ea3a6a7a9aaadaeafb0b2b3b4b6b7b8babec0c1#c3cbcfd1d5dddee0e1e4eaebf0f2f5f8f9fafb,63:00030405060a0b0c0d0f10121314151718191c2627292c2d2e30313334353637383b3c3e3f40414447484a51525354565758595a5b5c5d60646566686a6b6c6f707273747578797c7d7e7f81838485868b8d9193949597999a9b9c9d9e9fa1a4a6abafb1b2b5b6b9bbbdbfc0c1c2c3c5c7c8cacbccd1d3d4d5d7d8d9dadbdcdddfe2e4e5e6e7e8ebeceeeff0f1f3f5f7f9fafbfcfe,64:0304060708090a0d0e111215161718191a1d1f222324#252728292b2e2f3031323335363738393b3c3e404243494b4c4d4e4f505153555657595a5b5c5d5f60616263646566686a6b6c6e6f70717273747576777b7c7d7e7f8081838688898a8b8c8d8e8f90939497989a9b9c9d9fa0a1a2a3a5a6a7a8aaabafb1b2b3b4b6b9bbbdbebfc1c3c4c6c7c8c9cacbcccfd1d3d4d5d6d9dadbdcdddfe0e1e3e5e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,65:01020304050607080a0b0c0d0e0f10111314151617191a1b1c1d1e1f2021#222324262728292a2c2d30313233373a3c3d404142434446474a4b4d4e5052535457585a5c5f606164656768696a6d6e6f7173757678797a7b7c7d7e7f8081828384858688898a8d8e8f92949596989a9d9ea0a2a3a6a8aaacaeb1b2b3b4b5b6b7b8babbbebfc0c2c7c8c9cacdd0d1d3d4d5d8d9dadbdcdddedfe1e3e4eaebf2f3f4f5f8f9fbfcfdfeff,66:0104050708090b0d1011121617181a1b1c1e2122232426292a2b2c2e3032333738393a3b3d3f40424445464748494a4d4e505158#595b5c5d5e6062636567696a6b6c6d7172737578797b7c7d7f808183858688898a8b8d8e8f909293949598999a9b9c9e9fa0a1a2a3a4a5a6a9aaabacadafb0b1b2b3b5b6b7b8babbbcbdbfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8dadedfe0e1e2e3e4e5e7e8eaebecedeeeff1f5f6f8fafbfd,67:010203040506070c0e0f1112131618191a1c1e20212223242527292e303233363738393b3c3e3f414445474a4b4d5254555758595a5b5d62636466676b6c6e717476#78797a7b7d8082838586888a8c8d8e8f9192939496999b9fa0a1a4a6a9acaeb1b2b4b9babbbcbdbebfc0c2c5c6c7c8c9cacbcccdced5d6d7dbdfe1e3e4e6e7e8eaebedeef2f5f6f7f8f9fafbfcfe,68:01020304060d1012141518191a1b1c1e1f20222324252627282b2c2d2e2f30313435363a3b3f474b4d4f52565758595a5b5c5d5e5f6a6c6d6e6f707172737578797a7b7c7d7e7f8082848788898a8b8c8d8e90919294959698999a9b9c9d9e9fa0a1a3a4a5a9aaabacaeb1b2b4b6b7b8#b9babbbcbdbebfc1c3c4c5c6c7c8cacccecfd0d1d3d4d6d7d9dbdcdddedfe1e2e4e5e6e7e8e9eaebecedeff2f3f4f6f7f8fbfdfeff,69:00020304060708090a0c0f11131415161718191a1b1c1d1e21222325262728292a2b2c2e2f313233353637383a3b3c3e4041434445464748494a4b4c4d4e4f50515253555658595b5c5f616264656768696a6c6d6f7072737475767a7b7d7e7f8183858a8b8c8e8f909192939697999a9d9e9fa0a1a2a3a4a5a6a9aaacaeafb0b2b3b5b6b8b9babcbd#bebfc0c2c3c4c5c6c7c8c9cbcdcfd1d2d3d5d6d7d8d9dadcdddee1e2e3e4e5e6e7e8e9eaebeceeeff0f1f3f4f5f6f7f8f9fafbfcfe,6a:000102030405060708090b0c0d0e0f10111213141516191a1b1c1d1e20222324252627292b2c2d2e30323334363738393a3b3c3f40414243454648494a4b4c4d4e4f515253545556575a5c5d5e5f60626364666768696a6b6c6d6e6f70727374757677787a7b7d7e7f81828385868788898a8b8c8d8f929394959698999a9b9c9d9e9fa1a2a3a4a5a6#a7a8aaadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,6b:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f252628292a2b2c2d2e2f303133343536383b3c3d3f4041424445484a4b4d4e4f5051525354555657585a5b5c5d5e5f606168696b6c6d6e6f7071727374757677787a7d7e7f808588#8c8e8f909194959798999c9d9e9fa0a2a3a4a5a6a7a8a9abacadaeafb0b1b2b6b8b9babbbcbdbec0c3c4c6c7c8c9caccced0d1d8dadcdddedfe0e2e3e4e5e6e7e8e9ecedeef0f1f2f4f6f7f8fafbfcfeff,6c:000102030408090a0b0c0e12171c1d1e2023252b2c2d31333637393a3b3c3e3f434445484b4c4d4e4f5152535658595a62636566676b6c6d6e6f71737577787a7b7c7f8084878a8b8d8e9192959697989a9c9d9ea0a2a8acafb0b4b5b6b7bac0c1c2c3c6c7c8cbcdcecfd1d2d8#d9dadcdddfe4e6e7e9ecedf2f4f9ff,6d:000203050608090a0d0f101113141516181c1d1f20212223242628292c2d2f30343637383a3f404244494c50555657585b5d5f6162646567686b6c6d707172737576797a7b7d7e7f8081838486878a8b8d8f9092969798999a9ca2a5acadb0b1b3b4b6b7b9babbbcbdbec1c2c3c8c9cacdcecfd0d2d3d4d5d7dadbdcdfe2e3e5e7e8e9eaedeff0f2f4f5f6f8fafdfeff,6e:0001020304060708090b0f12131518191b1c1e1f222627282a2c2e30313335#3637393b3c3d3e3f40414245464748494a4b4c4f5051525557595a5c5d5e606162636465666768696a6c6d6f707172737475767778797a7b7c7d8081828487888a8b8c8d8e91929394959697999a9b9d9ea0a1a3a4a6a8a9abacadaeb0b3b5b8b9bcbebfc0c3c4c5c6c8c9cacccdced0d2d6d8d9dbdcdde3e7eaebecedeeeff0f1f2f3f5f6f7f8fafbfcfdfeff,6f:000103040507080a0b0c0d0e101112161718191a1b1c1d1e1f212223252627282c2e303234353738393a3b3c3d3f404142#43444548494a4c4e4f5051525354555657595a5b5d5f60616364656768696a6b6c6f707173757677797b7d7e7f808182838586878a8b8f909192939495969798999a9b9d9e9fa0a2a3a4a5a6a8a9aaabacadaeafb0b1b2b4b5b7b8babbbcbdbebfc1c3c4c5c6c7c8cacbcccdcecfd0d3d4d5d6d7d8d9dadbdcdddfe2e3e4e5e6e7e8e9eaebecedf0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,70:000102030405060708090a0b0c0d0e0f1012131415161718191c1d1e1f2021222425262728292a#2b2c2d2e2f30313233343637383a3b3c3d3e3f404142434445464748494a4b4d4e505152535455565758595a5b5c5d5f606162636465666768696a6e7172737477797a7b7d818283848687888b8c8d8f90919397989a9b9e9fa0a1a2a3a4a5a6a7a8a9aab0b2b4b5b6babebfc4c5c6c7c9cbcccdcecfd0d1d2d3d4d5d6d7dadcdddee0e1e2e3e5eaeef0f1f2f3f4f5f6f8fafbfcfeff,71:0001020304050607080b0c0d0e0f111214171b1c1d1e1f2021222324252728292a2b2c2d2e323334#353738393a3b3c3d3e3f4041424344464748494b4d4f505152535455565758595a5b5d5f6061626365696a6b6c6d6f707174757677797b7c7e7f8081828385868788898b8c8d8e909192939596979a9b9c9d9ea1a2a3a4a5a6a7a9aaabadaeafb0b1b2b4b6b7b8babbbcbdbebfc0c1c2c4c5c6c7c8c9cacbcccdcfd0d1d2d3d6d7d8d9dadbdcdddedfe1e2e3e4e6e8e9eaebecedeff0f1f2f3f4f5f6f7f8fafbfcfdfeff,72:0001020304050708090a0b0c0d0e0f101112131415161718191a#1b1c1e1f2021222324252627292b2d2e2f3233343a3c3e40414243444546494a4b4e4f505153545557585a5c5e60636465686a6b6c6d707173747677787b7c7d828385868788898c8e9091939495969798999a9b9c9d9ea0a1a2a3a4a5a6a7a8a9aaabaeb1b2b3b5babbbcbdbebfc0c5c6c7c9cacbcccfd1d3d4d5d6d8dadb#95$,30:000102,00b702:c9c7,00a830:0305,2014ff5e20:162618191c1d,30:141508090a0b0c0d0e0f16171011,00:b1d7f7,22:362728110f2a2908371aa52520,231222:992b2e614c483d1d606e6f64651e3534,26:4240,00b020:3233,2103ff0400a4ff:e0e1,203000a7211626:0605,25:cbcfcec7c6a1a0b3b2,203b21:92909193,30:13#95$,21:70717273747576777879#4$,24:88898a8b8c8d8e8f909192939495969798999a9b7475767778797a7b7c7d7e7f808182838485868760616263646566676869##,32:20212223242526272829##,21:606162636465666768696a6b#97$,ff:010203e505060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142434445464748494a4b4c4d4e4f505152535455565758595a5b5c5de3#95$,30:4142434445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f707172737475767778797a7b7c7d7e7f808182838485868788898a8b8c8d8e8f90919293#106$a1a2a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6#103$,03:9192939495969798999a9b9c9d9e9fa0a1a3a4a5a6a7a8a9#6$b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c3c4c5c6c7c8c9#5$,fe:3536393a3f403d3e41424344##3b3c373831#3334#104$,04:10111213141501161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f#13$30313233343551363738393a3b3c3d3e3f404142434445464748494a4b4c4d4e4f#11$,02:cacbd9,20:13152535,21:050996979899,22:151f23526667bf,25:505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f7071727381828384858687#88898a8b8c8d8e8f939495bcbde2e3e4e5,2609229530:121d1e#9$,010100e101ce00e0011300e9011b00e8012b00ed01d000ec014d00f301d200f2016b00fa01d400f901:d6d8dadc,00:fcea,0251e7c701:4448,e7c802:61#2$,31:05060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f20212223242526272829#19$,30:212223242526272829,32a333:8e8f9c9d9ea1c4ced1d2d5,fe30ff:e2e4#,212132:31#,20:10#1$,30:fc9b9cfdfe069d9e,fe:494a4b4c4d4e4f50515254555657595a5b5c5d5e5f6061#626364656668696a6b,e7:e7e8e9eaebecedeeeff0f1f2f3,30:07#11$,25:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142434445464748494a4b#13$,72:dcdddfe2e3e4e5e6e7eaebf5f6f9fdfeff,73:00020405060708090b0c0d0f1011121418191a1f2023242627282d2f30323335363a3b3c3d404142434445464748#494a4b4c4e4f515354555658595a5b5c5d5e5f6162636465666768696a6b6e7071#92$72737475767778797a7b7c7d7f808182838586888a8c8d8f90929394959798999a9c9d9ea0a1a3a4a5a6a7a8aaacadb1b4b5b6b8b9bcbdbebfc1c3c4c5c6c7#cbccced2d3d4d5d6d7d8dadbdcdddfe1e2e3e4e6e8eaebeceeeff0f1f3f4f5f6f7#92$f8f9fafbfcfdfeff,74:0001020407080b0c0d0e1112131415161718191c1d1e1f2021232427292b2d2f31323738393a3b3d3e3f4042434445464748494a4b4c4d#4e4f505152535456585d606162636465666768696a6b6c6e6f717273747578797a#92$7b7c7d7f8284858688898a8c8d8f9192939495969798999a9b9d9fa0a1a2a3a4a5a6aaabacadaeafb0b1b2b3b4b5b6b7b8b9bbbcbdbebfc0c1c2c3c4c5c6c7#c8c9cacbcccdcecfd0d1d3d4d5d6d7d8d9dadbdddfe1e5e7e8e9eaebecedf0f1f2#92$f3f5f8f9fafbfcfdfe,75:0001020305060708090a0b0c0e1012141516171b1d1e202122232426272a2e3436393c3d3f414243444647494a4d5051525355565758#5d5e5f60616263646768696b6c6d6e6f7071737576777a7b7c7d7e808182848587#92$88898a8c8d8e909395989b9c9ea2a6a7a8a9aaadb6b7babbbfc0c1c6cbcccecfd0d1d3d7d9dadcdddfe0e1e5e9ecedeeeff2f3f5f6f7f8fafbfdfe,76:02040607#08090b0d0e0f11121314161a1c1d1e212327282c2e2f31323637393a3b3d414244#92$45464748494a4b4e4f50515253555758595a5b5d5f6061626465666768696a6c6d6e7071727374757677797a7c7f80818385898a8c8d8f9092949597989a9b#9c9d9e9fa0a1a2a3a5a6a7a8a9aaabacadafb0b3b5b6b7b8b9babbbcbdbec0c1c3,554a963f57c3632854ce550954c076:914c,853c77ee827e788d72319698978d6c285b894ffa630966975cb880fa684880ae660276ce51f9655671ac7ff1888450b2596561ca6fb382ad634c625253ed54277b06516b75a45df462d48dcb9776628a8019575d97387f627238767d67cf767e64464f708d2562dc7a17659173ed642c6273822c9881677f724862:6ecc,4f3474e3534a529e7eca90a65e2e6886699c81807ed168d278c5868c9551508d8c2482de80de53058912526576:c4c7c9cbccd3d5d9dadcdddee0e1e2e3e4e6e7e8e9eaebecedf0f3f5f6f7fafbfdff,77:00020305060a0c0e0f1011121314151617181b1c1d1e21232425272a2b#2c2e3031323334393b3d3e3f4244454648494a4b4c4d4e4f52535455565758595c,858496f94fdd582199715b9d62:b1a5,66b48c799c8d7206676f789160b253:5117,8f8880cc8d1d94a1500d72c8590760eb711988ab595482ef672c7b285d297ef7752d6cf58e668ff8903c9f3b6bd491197b145f7c78a784d6853d6b:d5d9d6,5e:0187,75f995ed655d5f:0ac5,8f9f58c181c2907f965b97ad8fb97f168d2c62414fbf53:d85e,8f:a8a9ab,904d68075f6a819888689cd6618b522b762a5f6c658c6fd26ee85bbe644851:75b0,67c44e1979c9997c70b377:5d5e5f606467696a6d6e6f7071727374757677787a7b7c818283868788898a8b8f90939495969798999a9b9c9d9ea1a3a4a6a8abadaeafb1b2b4b6b7b8b9ba#bcbec0c1c2c3c4c5c6c7c8c9cacbcccecfd0d1d2d3d4d5d6d8d9dadddedfe0e1e4,75c55e7673bb83e064ad62e894b56ce2535a52c3640f94c27b944f2f5e1b823681:168a,6e246cca9a736355535c54fa886557e04e0d5e036b657c3f90e8601664e6731c88c16750624d8d22776c8e2991c75f6983dc8521991053c286956b8b60:ede8,707f82:cd31,4ed36ca785cf64cd7cd969fd66f9834953957b564fa7518c6d4b5c428e6d63d253c983:2c36,67e578b4643d5bdf5c945dee8be762c667f48c7a640063ba8749998b8c177f2094f24ea7961098a4660c731677:e6e8eaeff0f1f2f4f5f7f9fafbfc,78:0304050607080a0b0e0f101315191b1e20212224282a2b2e2f31323335363d3f414243444648494a4b4d4f51535458595a#5b5c5e5f606162636465666768696f7071727374757678797a7b7d7e7f80818283,573a5c1d5e38957f507f80a05382655e7545553150218d856284949e671d56326f6e5de2543570928f66626f64a463a35f7b6f8890f481e38fb05c1866685ff16c8996488d81886c649179f057ce6a59621054484e587a0b60e96f848bda627f901e9a8b79e4540375f4630153196c608fdf5f1b9a70803b9f7f4f885c3a8d647fc565a570bd51:45b2,866b5d075ba062bd916c75748e0c7a2061017b794ec77ef877854e1181ed521d51fa6a7153a88e87950496cf6ec19664695a78:848586888a8b8f9092949596999d9ea0a2a4a6a8a9aaabacadaeafb5b6b7b8babbbcbdbfc0c2c3c4c6c7c8cccdcecfd1d2d3d6d7d8dadbdcdddedfe0e1e2e3#e4e5e6e7e9eaebedeeeff0f1f3f5f6f8f9fbfcfdfeff,79:00020304060708090a0b0c,784050a877d7641089e6590463e35ddd7a7f693d4f20823955984e3275ae7a975e:628a,95ef521b5439708a6376952457826625693f918755076df37eaf882262337ef075b5832878c196cc8f9e614874f78bcd6b64523a8d506b21806a847156f153064e:ce1b,51d17c97918b7c074fc38e7f7be17a9c64675d1450ac810676017cb96dec7fe067515b:58f8,78cb64:ae13,63:aa2b,9519642d8fbe7b5476296253592754466b7950a362345e266b864ee38d37888b5f85902e79:0d0e0f1011121415161718191a1b1c1d1f2021222325262728292a2b2c2d2e2f3031323335363738393d3f42434445474a4b4c4d4e4f505152545558596163#6466696a6b6c6e70717273747576797b7c7d7e7f8283868788898b8c8d8e909192,6020803d62c54e39535590f863b880c665e66c2e4f4660ee6de18bde5f3986cb5f536321515a83616863520063638e4850125c9b79775bfc52307a3b60bc905376d75f:b797,76848e6c706f767b7b4977aa51f3909358244f4e6ef48fea654c7b1b72c46da47fdf5ae162b55e95573084827b2c5e1d5f1f90127f1498a063826ec7789870b95178975b57ab75354f4375385e9760e659606dc06bbf788953fc96d551cb52016389540a94938c038dcc7239789f87768fed8c0d53e079:939495969798999b9c9d9e9fa0a1a2a3a4a5a6a8a9aaabacadaeafb0b1b2b4b5b6b7b8bcbfc2c4c5c7c8cacccecfd0d3d4d6d7d9dadbdcdddee0e1e2e5e8ea#eceef1f2f3f4f5f6f7f9fafcfeff,7a:0104050708090a0c0f10111213151618191b1c,4e0176ef53ee948998769f0e952d5b9a8ba24e:221c,51ac846361c252a8680b4f97606b51bb6d1e515c6296659796618c46901775d890fd77636bd272:8aec,8bfb583577798d4c675c9540809a5ea66e2159927aef77ed953b6bb565ad7f0e58065151961f5bf958a954288e726566987f56e4949d76fe9041638754c659:1a3a,579b8eb267358dfa8235524160f0581586fe5ce89e454fc4989d8bb95a2560765384627c904f9102997f6069800c513f80335c1499756d314e8c7a:1d1f21222425262728292a2b2c2d2e2f303132343536383a3e4041424344454748494a4b4c4d4e4f50525354555658595a5b5c5d5e5f606162636465666768#696a6b6c6d6e6f717273757b7c7d7e828587898a8b8c8e8f909394999a9b9ea1a2,8d3053d17f5a7b4f4f104e4f96006cd573d085e95e06756a7ffb6a0a77fe94927e4151e170e653cd8fd483038d2972af996d6cdb574a82b365b980aa623f963259a84eff8bbf7eba653e83f2975e556198de80a5532a8bfd542080ba5e9f6cb88d3982ac915a54296c1b52067eb7575f711a6c7e7c89594b4efd5fff61247caa4e305c0167ab87025cf0950b98ce75af70fd902251af7f1d8bbd594951e44f5b5426592b657780a45b7562:76c2,8f905e456c1f7b264f:0fd8,670d7a:a3a4a7a9aaabaeafb0b1b2b4b5b6b7b8b9babbbcbdbec0c1c2c3c4c5c6c7c8c9cacccdcecfd0d1d2d3d4d5d7d8dadbdcdde1e2e4e7e8e9eaebeceef0f1f2f3#f4f5f6f7f8fbfcfe,7b:0001020507090c0d0e1012131617181a1c1d1f21222327292d,6d:6eaa,798f88b15f17752b629a8f854fef91dc65a781:2f51,5e9c81508d74526f89868d4b590d50854ed8961c723681798d1f5bcc8ba3964459877f1a549056:760e,8be565396982949976d66e895e72751867:46d1,7aff809d8d76611f79c665628d635188521a94a27f38809b7eb25c976e2f67607bd9768b9ad8818f7f947cd5641e95507a3f54:4ae5,6b4c640162089e3d80f3759952729769845b683c86e496:0194,94ec4e2a54047ed968398ddf801566f45e9a7fb97b:2f303234353637393b3d3f404142434446484a4d4e535557595c5e5f61636465666768696a6b6c6d6f70737476787a7c7d7f81828384868788898a8b8c8e8f#9192939698999a9b9e9fa0a3a4a5aeafb0b2b3b5b6b7b9babbbcbdbebfc0c2c3c4,57c2803f68975de5653b529f606d9f9a4f9b8eac516c5bab5f135de96c5e62f18d21517194a952fe6c9f82df72d757a267848d2d591f8f9c83c754957b8d4f306cbd5b6459d19f1353e486ca9aa88c3780a16545987e56fa96c7522e74dc52505be1630289024e5662d0602a68fa51735b9851a089c27ba199867f5060ef704c8d2f51495e7f901b747089c4572d78455f529f9f95fa8f689b3c8be17678684267dc8d:ea35,523d8f8a6eda68cd950590ed56fd679c88f98fc754c87b:c5c8c9cacbcdcecfd0d2d4d5d6d7d8dbdcdedfe0e2e3e4e7e8e9ebecedeff0f2f3f4f5f6f8f9fafbfdff,7c:0001020304050608090a0d0e101112131415171819#1a1b1c1d1e20212223242528292b2c2d2e2f3031323334353637393a3b3c3d3e42,9ab85b696d776c264ea55bb39a87916361a890af97e9542b6db55bd251fd558a7f:55f0,64bc634d65f161be608d710a6c:5749,592f676d822a58d5568e8c6a6beb90dd597d801753f76d695475559d83:77cf,683879be548c4f55540876d28c8996026cb36db88d6b89109e648d3a563f9ed175d55f8872e0606854fc4ea86a2a886160528f7054c470d886799e3f6d2a5b8f5f187ea255894faf7334543c539a501954:0e7c,4e4e5ffd745a58f6846b80e1877472d07cca6e567c:434445464748494a4b4c4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f70717275767778797a7e7f8081828384858687#888a8b8c8d8e8f90939496999a9ba0a1a3a6a7a8a9abacadafb0b4b5b6b7b8babb,5f27864e552c62a44e926caa623782b154d7534e733e6ed1753b521253168bdd69d05f8a60006dee574f6b2273af68538fd87f13636260a3552475ea8c6271156da35ba65e7b8352614c9ec478fa87577c27768751f060f6714c66435e4c604d8c0e707063258f895fbd606286d456de6bc160946167534960e066668d3f79fd4f1a70e96c478b:b3f2,7ed88364660f5a5a9b426d:51f7,8c416d3b4f19706b83b7621660d1970d8d27797851fb57:3efa,673a75787a3d79ef7b957c:bfc0c2c3c4c6c9cbcecfd0d1d2d3d4d8dadbdddee1e2e3e4e5e6e7e9eaebecedeef0f1f2f3f4f5f6f7f9fafcfdfeff,7d:000102030405060708090b0c0d0e0f10#1112131415161718191a1b1c1d1e1f212324252628292a2c2d2e30313233343536,808c99658ff96fc08ba59e2159ec7ee97f095409678168d88f917c4d96c653ca602575be6c7253735ac97ea7632451e0810a5df184df628051805b634f0e796d524260b86d4e5b:c4c2,8b:a1b0,65e25fcc964559937e:e7aa,560967b759394f735bb652a0835a988a8d3e753294be50477a3c4ef767b69a7e5ac16b7c76d1575a5c167b3a95f4714e517c80a9827059787f04832768c067ec78:b177,62e363617b804fed526a51cf835069db92748d:f531,89c1952e7bad4ef67d:3738393a3b3c3d3e3f404142434445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6f70717273747576#78797a7b7c7d7e7f808182838485868788898a8b8c8d8e8f909192939495969798,506582305251996f6e:1085,6da75efa50f559dc5c066d466c5f7586848b686859568bb253209171964d854969127901712680f64ea490ca6d479a845a0756bc640594f077eb4fa5811a72e189d2997a7f347ede527f655991758f:7f83,53eb7a9663:eda5,768679f888579636622a52ab8282685467706377776b7aed6d017ed389e359d0621285c982a5754c501f4ecb75a58beb5c4a5dfe7b4b65a491d14eca6d25895f7d2795264ec58c288fdb9773664b79818fd170ec6d787d:999a9b9c9d9e9fa0a1a2a3a4a5a7a8a9aaabacadafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9#dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fa,5c3d52b283465162830e775b66769cb84eac60ca7c:beb3,7ecf4e958b66666f988897595883656c955c5f8475c997567a:dfde,51c070af7a9863ea7a767ea0739697ed4e4570784e5d915253a965:51e7,81fc8205548e5c31759a97a062d872d975bd5c459a7983ca5c40548077e94e3e6cae805a62d2636e5de851778ddd8e1e952f4ff153e560e770ac526763509e435a1f5026773753777ee26485652b628963985014723589c951b38bc07edd574783cc94a7519b541b5cfb7d:fbfcfdfeff,7e:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f30313233343536373839#3a3c3d3e3f40424344454648494a4b4c4d4e4f505152535455565758595a5b5c5d,4fca7ae36d5a90e19a8f55805496536154af5f0063e9697751ef6168520a582a52d8574e780d770b5eb761777ce062:5b97,4ea27095800362f770e49760577782db67ef68f578d5989779d158f354b353ef6e34514b523b5ba28bfe80af554357a660735751542d7a7a60505b5463a762a053e362635bc767af54ed7a9f82e691775e9388e4593857ae630e8de880ef57577b774fa95feb5bbd6b3e53217b5072c2684677:ff36,65f751b54e8f76d45cbf7aa58475594e9b4150807e:5e5f606162636465666768696a6b6c6d6e6f707172737475767778797a7b7c7d7e7f8081838485868788898a8b8c8d8e8f909192939495969798999a9c9d9e#aeb4bbbcd6e4ecf9,7f:0a101e37393b3c3d3e3f404143464748494a4b4c4d4e4f5253,998861276e8357646606634656f062:ec69,5ed39614578362c955878721814a8fa3556683b167658d5684dd5a6a680f62e67bee961151706f9c8c3063fd89c861d27f0670c26ee57405699472fc5eca90ce67176d6a635e52b3726280014f6c59e5916a70d96d9d52d24e5096f7956d857e78ca7d2f5121579264c2808b7c7b6cea68f1695e51b7539868a872819ece7bf172f879bb6f137406674e91cc9ca4793c83:8954,540f68174e3d538952b1783e5386522950884f:8bd0,7f:56595b5c5d5e6063646566676b6c6d6f7073757677787a7b7c7d7f8082838485868788898b8d8f9091929395969798999b9ca0a2a3a5a6a8a9aaabacadaeb1#b3b4b5b6b7babbbec0c2c3c4c6c7c8c9cbcdcfd0d1d2d3d6d7d9dadbdcdddee2e3,75e27acb7c926ca596b6529b748354e94fe9805483b28fde95705ec9601c6d9f5e18655b813894fe604b70bc7ec37cae51c968817cb1826f4e248f8691cf667e4eae8c0564a9804a50da759771ce5be58fbd6f664e86648295635ed66599521788c270c852a3730e7433679778f797164e3490bb9cde6dcb51db8d41541d62ce73b283f196f69f8494c34f367f9a51cc707596755cad988653e64ee46e9c740969b4786b998f7559521876246d4167f3516d9f99804b54997b3c7abf7f:e4e7e8eaebecedeff2f4f5f6f7f8f9fafdfeff,80:020708090a0e0f11131a1b1d1e1f2123242b2c2d2e2f303234393a3c3e404144454748494e4f505153555657#595b5c5d5e5f6061626364656667686b6c6d6e6f7072737475767778797a7b7c7d,9686578462e29647697c5a0464027bd36f0f964b82a6536298855e90708963b35364864f9c819e93788c97328d:ef42,9e7f6f5e79845f559646622e9a74541594dd4fa365c55c:6561,7f1586516c2f5f8b73876ee47eff5ce6631b5b6a6ee653754e7163a0756562a18f6e4f264ed16ca67eb68bba841d87ba7f57903b95237ba99aa188f8843d6d1b9a867edc59889ebb739b780186829a:6c82,561b541757cb4e709ea653568fc881097792999286ee6ee1851366fc61626f2b80:7e818285888a8d8e8f909192949597999ea3a6a7a8acb0b3b5b6b8b9bbc5c7c8c9cacbcfd0d1d2d3d4d5d8dfe0e2e3e6eef5f7f9fbfeff,81:000103040507080b#0c1517191b1c1d1f202122232425262728292a2b2d2e3033343537393a3b3c3d3f,8c298292832b76f26c135fd983bd732b8305951a6bdb77db94c6536f830251925e3d8c8c8d384e4873ab679a68859176970971646ca177095a9295416bcf7f8e66275bd059b95a9a95:e8f7,4eec84:0c99,6aac76df9530731b68a65b5f772f919a97617cdc8ff78c1c5f257c7379d889c56ccc871c5bc65e4268c977207ef551:954d,52c95a297f05976282d763cf778485d079d26e3a5e9959998511706d6c1162bf76bf654f60af95fd660e879f9e2394ed54:0d7d,8c2c647881:40414243444547494d4e4f525657585b5c5d5e5f6162636466686a6b6c6f727375767778818384858687898b8c8d8e90929394959697999a9e9fa0a1a2a4a5#a7a9abacadaeafb0b1b2b4b5b6b7b8b9bcbdbebfc4c5c7c8c9cbcdcecfd0d1d2d3,647986116a21819c78e864699b5462b9672b83ab58a89ed86cab6f205bde964c8c0b725f67d062c772614ea959c66bcd589366ae5e5552df6155672876ee776672677a4662ff54:ea50,94a090a35a1c7eb36c164e435976801059485357753796be56ca63208111607c95f96dd65462998151855ae980fd59ae9713502a6ce55c3c62df4f60533f817b90066eba852b62c85e7478be64b5637b5ff55a18917f9e1f5c3f634f80425b7d556e95:4a4d,6d8560a867e072de51dd5b8181:d4d5d6d7d8d9dadbdcdddedfe0e1e2e4e5e6e8e9ebeeeff0f1f2f5f6f7f8f9fafdff,82:030708090a0b0e0f111315161718191a1d2024252627292e323a3c3d3f#404142434546484a4c4d4e5051525354555657595b5c5d5e606162636465666769,62e76cde725b626d94ae7ebd81136d53519c5f04597452aa6012597366968650759f632a61e67cef8bfa54e66b279e256bb485d5545550766ca4556a8db4722c5e156015743662cd6392724c5f986e436d3e65006f5876d878d076fc7554522453db4e535e9e65c180:2ad6,629b5486522870ae888d8dd16ce1547880da57f988f48d54966a914d4f696c9b55b776c6783062a870f96f8e5f6d84ec68da787c7bf781a8670b9e4f636778b0576f7812973962:79ab,528874356bd782:6a6b6c6d71757677787b7c808183858687898c90939495969a9b9ea0a2a3a7b2b5b6babbbcbfc0c2c3c5c6c9d0d6d9dadde2e7e8e9eaecedeef0f2f3f5f6f8#fafcfdfeff,83:000a0b0d1012131618191d1e1f20212223242526292a2e3032373b3d,5564813e75b276ae533975de50fb5c418b6c7bc7504f72479a9798d86f0274e27968648777a562fc98918d2b54c180584e52576a82f9840d5e7351ed74f68bc45c4f57616cfc98875a4678349b448feb7c955256625194fa4ec68386846183e984b257d467345703666e6d668c3166dd7011671f6b3a6816621a59bb4e0351c46f0667d26c8f517668cb59476b6775665d0e81109f5065d779:4841,9a918d775c824e5e4f01542f5951780c56686c148fc45f036c:7de3,8bab639083:3e3f41424445484a4b4c4d4e5355565758595d6270717273747576797a7e7f808182838487888a8b8c8d8f909194959697999a9d9fa1a2a3a4a5a6a7acadae#afb5bbbebfc2c3c4c6c8c9cbcdced0d1d2d3d5d7d9dadbdee2e3e4e6e7e8ebeced,60706d3d7275626694:8ec5,53438fc17b7e4edf8c264e7e9ed494:b1b3,524d6f5c90636d458c3458115d4c6b:2049,67aa545b81547f8c589985375f3a62a26a47953965726084686577a74e544fa85de7979864ac7fd85ced4fcf7a8d520783044e14602f7a8394a64fb54eb279e6743452e482b964d279bd5bdd6c8197528f7b6c22503e537f6e0564ce66746c3060c598778bf75e86743c7a7779cb4e1890b174036c4256da914b6cc58d8b533a86c666f28eaf5c489a716e2083:eeeff3f4f5f6f7fafbfcfeff,84:0002050708090a10121314151617191a1b1e1f20212223292a2b2c2d2e2f30323334353637393a3b3e3f404142434445474849#4a4b4c4d4e4f505253545556585d5e5f606264656667686a6e6f70727477797b7c,53d65a369f8b8da353bb570898a76743919b6cc9516875ca62f372ac52:389d,7f3a7094763853749e4a69b7786e96c088d97fa471:36c3,518967d374e458e4651856b78ba9997662707ed560f970ed58ec4e:c1ba,5fcd97e74efb8ba45203598a7eab62544ecd65e5620e833884c98363878d71946eb65bb97ed2519763c967d480898339881551125b7a59828fb14e736c5d516589258f6f962e854a745e95:10f0,6da682e55f3164926d128428816e9cc3585e8d5b4e0953c184:7d7e7f8081838485868a8d8f90919293949596989a9b9d9e9fa0a2a3a4a5a6a7a8a9aaabacadaeb0b1b3b5b6b7bbbcbec0c2c3c5c6c7c8cbcccecfd2d4d5d7#d8d9dadbdcdee1e2e4e7e8e9eaebedeeeff1f2f3f4f5f6f7f8f9fafbfdfe,85:000102,4f1e6563685155d34e2764149a9a626b5ac2745f82726da968ee50e7838e7802674052396c997eb150bb5565715e7b5b665273ca82eb67495c715220717d886b95ea965564c58d6181b355846c5562477f2e58924f2455468d4f664c4e0a5c1a88f368a2634e7a0d70e7828d52fa97f65c1154e890b57ecd59628d4a86c782:0c0d,8d6664445c0461516d89793e8bbe78377533547b4f388eab6df15a207ec5795e6c885ba15a76751a80be614e6e1758f075:1f25,727253477ef385:030405060708090a0b0d0e0f101214151618191b1c1d1e2022232425262728292a2d2e2f303132333435363e3f404142444546474b4c4d4e4f505152535455#57585a5b5c5d5f60616263656667696a6b6c6d6e6f707173757677787c7d7f8081,770176db526980dc57235e08593172ee65bd6e7f8bd75c388671534177f362fe65f64ec098df86805b9e8bc653f277e24f7f5c4e9a7659cb5f0f793a58eb4e1667ff4e8b62ed8a93901d52bf662f55dc566c90024ed54f8d91ca99706c0f5e0260435ba489c68bd56536624b99965b:88ff,6388552e53d77626517d852c67a268b36b8a62928f9353d482126dd1758f4e668d4e5b70719f85af66:91d9,7f7287009ecd9f205c5e672f8ff06811675f620d7ad658855eb665706f3185:82838688898a8b8c8d8e909192939495969798999a9d9e9fa0a1a2a3a5a6a7a9abacadb1b2b3b4b5b6b8babbbcbdbebfc0c2c3c4c5c6c7c8cacbcccdced1d2#d4d6d7d8d9dadbdddedfe0e1e2e3e5e6e7e8eaebecedeeeff0f1f2f3f4f5f6f7f8,60555237800d6454887075295e05681362f4971c53cc723d8c016c3477617a0e542e77ac987a821c8bf47855671470c165af64955636601d79c153f84e1d6b7b80865bfa55e356db4f:3a3c,99725df3677e80386002988290015b8b8b:bcf5,641c825864de55fd82cf91654fd77d20901f7c9f50f358516eaf5bbf8bc980839178849c7b97867d96:8b8f,7ee59ad3788e5c817a57904296a7795f5b59635f7b0b84d168ad55067f2974107d2295016240584c4ed65b835979585485:f9fafcfdfe,86:0001020304060708090a0b0c0d0e0f10121314151718191a1b1c1d1e1f20212223242526282a2b2c2d2e2f3031323334353637393a3b3d3e3f40#4142434445464748494a4b4c525355565758595b5c5d5f6061636465666768696a,736d631e8e:4b0f,80ce82d462ac53f06cf0915e592a60016c70574d644a8d2a762b6ee9575b6a8075f06f6d8c:2d08,57666bef889278b363a253f970ad6c645858642a580268e0819b55107cd650188eba6dcc8d9f70eb638f6d9b6ed47ee68404684390036dd896768ba85957727985e4817e75bc8a8a68af52548e22951163d098988e44557c4f5366ff568f60d56d9552435c4959296dfb586b75:301c,606c82148146631167618fe2773a8d:f334,94c15e165385542c70c386:6d6f7072737475767778838485868788898e8f90919294969798999a9b9e9fa0a1a2a5a6abadaeb2b3b7b8b9bbbcbdbebfc1c2c3c5c8cccdd2d3d5d6d7dadc#dde0e1e2e3e5e6e7e8eaebeceff5f6f7fafbfcfdff,87:010405060b0c0e0f10111416,6c405ef7505c4ead5ead633a8247901a6850916e77b3540c94dc5f647ae5687663457b527edf75db507762955934900f51f879c37a8156fe5f9290146d825c60571f541051546e4d56e263a89893817f8715892a9000541e5c6f81c062:d658,81319e3596409a:6e7c,692d59a562d3553e631654c786d96d3c5a0374e6889c6b6a59168c4c5f2f6e7e73a9987d4e3870f75b8c7897633d665a769660cb5b9b5a494e0781556c6a738b4ea167897f515f8065fa671b5fd859845a0187:191b1d1f20242627282a2b2c2d2f303233353638393a3c3d404142434445464a4b4d4f505152545556585a5b5c5d5e5f6162666768696a6b6c6d6f71727375#7778797a7f8081848687898a8c8e8f90919294959698999a9b9c9d9ea0a1a2a3a4,5dcd5fae537197e68fdd684556f4552f60df4e3a6f4d7ef482c7840e59d44f:1f2a,5c3e7eac672a851a5473754f80c355829b4f4f4d6e2d8c135c096170536b761f6e29868a658795fb7eb9543b7a337d0a95ee55e17fc174ee631d87176da17a9d621165a1536763e16c835deb545c94a84e4c6c618bec5c4b65e0829c68a754:3e34,6b:cb66,4e9463425348821e4f:0dae,575e620a96fe6664726952:ffa1,609f8bef661471996790897f785277fd6670563b54389521727a87:a5a6a7a9aaaeb0b1b2b4b6b7b8b9bbbcbebfc1c2c3c4c5c7c8c9cccdcecfd0d4d5d6d7d8d9dadcdddedfe1e2e3e4e6e7e8e9ebecedeff0f1f2f3f4f5f6f7f8#fafbfcfdff,88:0001020405060708090b0c0d0e0f101112141718191a1c1d1e1f2023,7a00606f5e0c6089819d591560dc718470ef6eaa6c5072806a8488ad5e2d4e605ab3559c94e36d177cfb9699620f7ec6778e867e5323971e8f9666875ce14fa072ed4e0b53a6590f54136380952851484ed99c9c7ea454b88d248854823795f26d8e5f265acc663e966973:b02e,53bf817a99857fa15baa96:7750,7ebf76f853a2957699997bb189446e584e617fd479658be660f354cd4eab98795df76a6150cf54118c618427785d9704524a54ee56a395006d885bb56dc6665388:2425262728292a2b2c2d2e2f30313334353637383a3b3d3e3f414243464748494a4b4e4f505152535556585a5b5c5d5e5f6066676a6d6f717374757678797a#7b7c80838687898a8c8e8f90919394959798999a9b9d9e9fa0a1a3a5a6a7a8a9aa,5c0f5b5d6821809655787b11654869544e9b6b47874e978b534f631f643a90aa659c80c18c10519968b0537887f961c86c:c4fb,8c225c5185aa82af950c6b238f9b65b05f:fbc3,4fe18845661f8165732960fa51745211578b5f6290a2884c91925e78674f602759d351:44f6,80f853086c7996c4718a4f:11ee,7f9e673d55c5950879c088967ee3589f620c9700865a5618987b5f908bb884c4915753d965ed5e8f755c60647d6e5a7f7e:eaed,8f6955a75ba360ac65cb738488:acaeafb0b2b3b4b5b6b8b9babbbdbebfc0c3c4c7c8cacbcccdcfd0d1d3d6d7dadbdcdddee0e1e6e7e9eaebecedeeeff2f5f6f7fafbfdff,89:0001030405060708#090b0c0d0e0f1114151617181c1d1e1f20222324262728292c2d2e2f3132333537,9009766377297eda9774859b5b667a7496ea884052cb718f5faa65ec8be25bfb9a6f5de16b896c5b8b:adaf,900a8fc5538b62bc9e:262d,54404e2b82bd7259869c5d1688596daf96c554d14e9a8bb6710954bd960970df6df976d04e25781487125ca95ef68a00989c960e708e6cbf594463a9773c884d6f148273583071d5538c781a96c155015f6671305bb48c1a9a8c6b83592e9e2f79e76768626c4f6f75a17f8a6d0b96336c274ef075d2517b68376f3e908081705996747689:38393a3b3c3d3e3f40424345464748494a4b4c4d4e4f505152535455565758595a5b5c5d6061626364656768696a6b6c6d6e6f707172737475767778797a7c#7d7e808284858788898a8b8c8d8e8f909192939495969798999a9b9c9d9e9fa0a1,64475c2790657a918c2359da54ac8200836f898180006930564e8036723791ce51b64e5f987563964e1a53f666f3814b591c6db24e0058f9533b63d694f14f:9d0a,886398905937905779fb4eea80f075916c825b9c59e85f5d69058681501a5df24e5977e34ee5827a6291661390915c794ebf5f7981c69038808475ab4ea688d4610f6bc55fc64e4976ca6ea28b:e3ae,8c0a8bd15f027f:fccc,7ece83:356b,56e06bb797f3963459fb541f94f66deb5bc5996e5c395f15969089:a2a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c3cdd3d4d5d7d8d9dbdddfe0e1e2e4e7e8e9eaecedeef0f1f2f4f5f6f7f8f9fa#fbfcfdfeff,8a:01020304050608090a0b0c0d0e0f101112131415161718191a1b1c1d,537082f16a315a749e705e947f2883b984:2425,836787478fce8d6276c85f719896786c662054df62e54f6381c375c85eb896cd8e0a86f9548f6cf36d8c6c38607f52c775285e7d4f1860a05fe75c24753190ae94c072b96cb96e389149670953:cbf3,4f5191c98bf153c85e7c8fc26de44e8e76c26986865e611a82064f:59de,903e9c7c61096e:1d14,96854e885a3196e84e0e5c7f79b95b878bed7fbd738957df828b90c15401904755bb5cea5fa161086b3272f180b28a:891e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3f4041424344454647494a4b4c4d4e4f505152535455565758595a5b5c5d5e#5f606162636465666768696a6b6c6d6e6f7071727374757677787a7b7c7d7e7f80,6d745bd388d598848c6b9a6d9e336e0a51:a443,57a38881539f63f48f9556ed54585706733f6e907f188fdc82d1613f6028966266f07ea68d:8ac3,94a55cb37ca4670860a6960580184e9190e75300966851418fd08574915d665597f55b55531d78386742683d54c9707e5bb08f7d518d572854b1651266828d:5e43,810f846c906d7cdf51ff85fb67a365e96fa186a48e81566a90207682707671e58d2362e952196cfd8d3c600e589e618e66fe8d60624e55b36e23672d8f678a:81828384858687888b8c8d8e8f9091929495969798999a9b9c9d9e9fa0a1a2a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2#c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3,94e195f87728680569a8548b4e4d70b88bc86458658b5b857a84503a5be877bb6be18a797c986cbe76cf65a98f975d2d5c5586386808536062187ad96e5b7efd6a1f7ae05f706f335f20638c6da867564e085e108d264ed780c07634969c62db662d627e6cbc8d7571677f695146808753ec906e629854f286f08f998005951785178fd96d5973cd659f771f7504782781fb8d1e94884fa6679575b98bca9707632f9547963584b8632377415f8172f04e896014657462ef6b63653f8a:e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,8b:0001020304050608090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f20212223#24252728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142434445,5e2775c790d18bc1829d679d652f5431871877e580a281026c414e4b7ec7804c76f4690d6b966267503c4f84574063076b628dbe53ea65e87eb85fd763:1ab7,81:f3f4,7f6e5e1c5cd95236667a79e97a1a8d28709975d46ede6cbb7a924e2d76c55fe0949f88777ec879cd80bf91cd4ef24f17821f54685dde6d328bcc7ca58f7480985e1a549276b15b99663c9aa473e0682a86db6731732a8b:f8db,90107af970db716e62c477a956314e3b845767f152a986c08d2e94f87b518b:464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f6061626364656768696a6b6d6e6f707172737475767778797a7b7c7d7e7f80818283848586#8788898a8b8c8d8e8f909192939495969798999a9b9c9d9e9facb1bbc7d0ea,8c:091e,4f4f6ce8795d9a7b6293722a62fd4e1378168f6c64b08d5a7bc668695e8488c55986649e58ee72b6690e95258ffd8d5857607f008c0651c6634962d95353684c74228301914c55447740707c6d4a517954a88d4459ff6ecb6dc45b5c7d2b4ed47c7d6ed35b5081ea6e0d5b579b0368d58e2a5b977efc603b7eb590b98d70594f63cd79df8db3535265cf79568bc5963b7ec494bb7e825634918967007f6a5c0a907566285de64f5067de505a4f5c57505e:a7#3$,8c:38393a3b3c3d3e3f4042434445484a4b4d4e4f5051525354565758595b5c5d5e5f60636465666768696c6d6e6f707172747576777b7c7d7e7f808183848687#888b8d8e8f90919293959697999a9b9c9d9e9fa0a1a2a3a4a5a6a7a8a9aaabacad,4e:8d0c,51404e105eff53454e:15981e,9b325b6c56694e2879ba4e3f53154e47592d723b536e6c1056df80e499976bd3777e9f174e:369f,9f104e:5c6993,82885b5b556c560f4ec453:8d9da3a5ae,97658d5d53:1af5262e3e,8d5c53:6663,52:02080e2d333f404c5e615c,84af52:7d82819093,51827f544e:bbc3c9c2e8e1ebde,4f1b4ef34f:2264,4ef54f:2527092b5e67,65384f:5a5d,8c:aeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebec#edeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,8d:000102030405060708090a0b0c0d,4f:5f57323d76749189838f7e7baa7cac94e6e8eac5dae3dcd1dff8,50:294c,4ff350:2c0f2e2d,4ffe50:1c0c25287e4355484e6c7ba5a7a9bad6,510650:edece6ee,51:070b,4edd6c3d4f:5865ce,9fa06c467c74516e5dfd9ec999985181591452f9530d8a07531051eb591951554ea051564eb388:6ea4,4eb5811488d279805b3488037fb851:abb1bdbc,8d:0e0f101112131415161718191a1b1c205152575f6568696a6c6e6f717278797a7b7c7d7e7f808283868788898c8d8e8f90929395969798999a9b9c9d9ea0a1#a2a4a5a6a7a8a9aaabacadaeafb0b2b6b7b9bbbdc0c1c2c5c7c8c9cacdd0d2d3d4,51:c796a2a5,8b:a0a6a7aab4b5b7c2c3cbcfced2d3d4d6d8d9dcdfe0e4e8e9eef0f3f6f9fcff,8c:000204070c0f1112141516191b181d1f202125272a2b2e2f32333536,53:697a,96:1d2221312a3d3c4249545f676c7274888d97b0,90:979b9d99aca1b4b3b6ba,8d:d5d8d9dce0e1e2e5e6e7e9edeef0f1f2f4f6fcfeff,8e:00010203040607080b0d0e1011121315161718191a1b1c202124252627282b2d303233343637383b3c3e#3f4345464c4d4e4f505354555657585a5b5c5d5e5f60616263646567686a6b6e71,90:b8b0cfc5bed0c4c7d3e6e2dcd7dbebeffe,91:04221e23312f394346,520d594252:a2acadbe,54ff52:d0d6f0,53df71ee77cd5ef451:f5fc,9b2f53b65f01755a5def57:4ca9a1,58:7ebcc5d1,57:292c2a33392e2f5c3b4269856b867c7b686d7673ada48cb2cfa7b493a0d5d8dad9d2b8f4eff8e4dd,8e:73757778797a7b7d7e808283848688898a8b8c8d8e91929395969798999a9b9d9fa0a1a2a3a4a5a6a7a8a9aaadaeb0b1b3b4b5b6b7b8b9bbbcbdbebfc0c1c2#c3c4c5c6c7c8c9cacbcccdcfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4,58:0b0d,57:fded,58:001e194420656c81899a80,99a89f1961ff82:797d7f8f8aa8848e919799abb8beb0c8cae398b7aecbccc1a9b4a1aa9fc4cea4e1,830982:f7e4,83:0f07,82:dcf4d2d8,830c82:fbd3,83:111a061415,82:e0d5,83:1c515b5c08923c34319b5e2f4f47435f4017602d3a336665,8e:e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,8f:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f20212223#2425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f4041424344,83:681b696c6a6d6eb078b3b4a0aa939c857cb6a97db87b989ea8babcc1,840183:e5d8,580784:180b,83:ddfdd6,84:1c381106,83:d4df,84:0f03,83:f8f9eac5c0,842683:f0e1,84:5c515a597387887a89783c4669768c8e316dc1cdd0e6bdd3cabfbae0a1b9b497e5e3,850c750d853884f085:391f3a,8f:45464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f6061626364656a808c929da0a1a2a4a5a6a7aaacadaeafb2b3b4b5b7b8babbbcbfc0c3c6#c9cacbcccdcfd2d6d7dae0e1e3e7eceff1f2f4f5f6fafbfcfeff,90:07080c0e131518,85:563b,84:fffc,85:594868645e7a,77a285:43727ba4a8878f79ae9c85b9b7b0d3c1dcff,86:270529163c,5efe5f0859:3c41,803759:555a58,530f5c:22252c34,62:4c6a9fbbcadad7ee,632262f663:394b43adf6717a8eb46dac8a69aebcf2f8e0ffc4dece,645263:c6be,64:45410b1b200c26215e846d96,90:191c2324252728292a2b2c303132333437393a3d3f4043454648494a4b4c4e545556595a5c5d5e5f6061646667696a6b6c6f70717273767778797a7b7c7e81#84858687898a8c8d8e8f90929496989a9c9e9fa0a4a5a7a8a9abadb2b7bcbdbfc0,64:7ab7b899bac0d0d7e4e2,65:09252e,5f:0bd2,75195f1153:5ff1fde9e8fb,54:1216064b5253545643215759233282947771649a9b8476669dd0adc2b4d2a7a6d3d472a3d5bbbfccd9dadca9aaa4ddcfde,551b54e7552054fd551454f355:22230f11272a678fb5496d41553f503c,90:c2c3c6c8c9cbcccdd2d4d5d6d8d9dadedfe0e3e4e5e9eaeceef0f1f2f3f5f6f7f9fafbfcff,91:00010305060708090a0b0c0d0e0f1011121314151617181a1b1c#1d1f20212425262728292a2b2c2d2e30323334353637383a3b3c3d3e3f40414244,55:375675767733305c8bd283b1b988819f7ed6917bdfbdbe9499eaf7c9,561f55:d1ebecd4e6ddc4efe5f2f3cccde8f5e4,8f9456:1e080c012423,55fe56:00272d5839572c4d62595c4c548664716b7b7c8593afd4d7dde1f5ebf9ff,57:040a091c,5e:0f191411313b3c,91:454748515354555658595b5c5f606667686b6d737a7b7c808182838486888a8e8f939495969798999c9d9e9fa0a1a4a5a6a7a8a9abacb0b1b2b3b6b7b8b9bb#bcbdbebfc0c1c2c3c4c5c6c8cbd0d2d3d4d5d6d7d8d9dadbdddedfe0e1e2e3e4e5,5e:3744545b5e61,5c:8c7a8d9096889899919a9cb5a2bdacabb1a3c1b7c4d2e4cbe5,5d:020327262e241e061b583e343d6c5b6f5d6b4b4a697482999d,8c735d:b7c5,5f:73778287898c95999ca8adb5bc,88625f6172:adb0b4b7b8c3c1cecdd2e8efe9f2f4f7,730172f3730372fa91:e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,92:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f2021222324#25262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142434445,72fb73:1713210a1e1d152239252c3831504d57606c6f7e,821b592598e759:2402,99:636768696a6b6c74777d8084878a8d9091939495,5e:80918b96a5a0b9b5beb3,8d535e:d2d1dbe8ea,81ba5f:c4c9d6cf,60035fee60045f:e1e4fe,60:0506,5f:eaedf8,60:1935261b0f0d292b0a3f2178797b7a42,92:464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f7071727375767778797a7b7c7d7e7f808182838485#868788898a8b8c8d8f909192939495969798999a9b9c9d9e9fa0a1a2a3a4a5a6a7,60:6a7d969aad9d83928c9becbbb1ddd8c6dab4,61:20261523,60f461:000e2b4a75ac94a7b7d4f5,5fdd96b395:e9ebf1f3f5f6fcfe,96:030406080a0b0c0d0f12151617191a,4e2c723f62156c:35545c4aa38590948c6869747686a9d0d4adf7f8f1d7b2e0d6faebeeb1d3effe,92:a8a9aaabacadafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7e8#e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,93:00010203040506070809,6d:39270c43480704190e2b4d2e351a4f525433916f9ea05e93945c607c63,6e1a6d:c7c5de,6e0e6d:bfe0,6e116d:e6ddd9,6e166dab6e0c6dae6e:2b6e4e6bb25f865354322544dfb198e0,6f2d6e:e2a5a7bdbbb7d7b4cf8fc29f,6f:6246472415,6ef96f:2f364b742a0929898d8c78727c7ad1,93:0a0b0c0d0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3f40414243444546474849#4a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696b,6f:c9a7b9b6c2e1eedee0ef,70:1a231b39354f5e,5b:80849593a5b8,752f9a9e64345b:e4ee,89305bf08e478b078f:b6d3d5e5eee4e9e6f3e8,90:05040b26110d162135362d2f445152506858625b,66b990:747d8288838b,5f:50575658,5c3b54ab5c:5059,5b715c:6366,7fbc5f:2a292d,82745f3c9b3b5c6e59:81838da9aaa3,93:6c6d6e6f707172737475767778797a7b7c7d7e7f808182838485868788898a8b8c8d8e909192939495969798999a9b9c9d9e9fa0a1a2a3a4a5a6a7a8a9aaab#acadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cbcccd,59:97caab9ea4d2b2afd7be,5a:0506,59dd5a0859:e3d8f9,5a:0c09323411231340674a553c6275,80ec5a:aa9b777abeebb2d2d4b8e0e3f1d6e6d8dc,5b:091716323740,5c:151c,5b:5a6573515362,9a:7577787a7f7d808185888a90929396989b9c9d9fa0a2a3a5a7,7e:9fa1a3a5a8a9,93:cecfd0d1d2d3d4d5d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,94:000102030405060708090a0b0c0d#0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e,7e:adb0bec0c1c2c9cbccd0d4d7dbe0e1e8ebeeeff1f2,7f0d7e:f6fafbfe,7f:01020307080b0c0f111217191c1b1f212223242526272a2b2c2d2f3031323335,5e7a757f5ddb753e909573:8e91aea29fcfc2d1b7b3c0c9c8e5d9,987c740a73:e9e7debaf2,74:0f2a5b262528302e2c,94:2f303132333435363738393a3b3c3d3f404142434445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6c6d6e6f#707172737475767778797a7b7c7d7e7f8081828384919698c7cfd3d4dae6fb,95:1c20,74:1b1a415c575559776d7e9c8e8081878b9ea8a990a7d2ba,97:eaebec,67:4c535e4869a5876a7398a775a89ead8b777cf0,680967d8680a67:e9b0,680c67:d9b5dab3dd,680067:c3b8e2,680e67:c1fd,68:323360614e624464831d55664167403e4a4929b58f7477936bc2,696e68fc69:1f20,68f995:27333d43484b555a606e74757778797a7b7c7d7e808182838485868788898a8b8c8d8e8f909192939495969798999a9b9c9d9e9fa0a1a2a3a4a5a6a7a8a9aa#abacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacb,692468f069:0b0157,68e369:10713960425d846b80987834cc8788ce896663799ba7bbabadd4b1c1cadf95e08dff,6a2f69ed6a:171865,69f26a:443ea0505b358e793d28587c9190a997ab,73:3752,6b:8182878492938d9a9ba1aa,8f:6b6d71727375767877797a7c7e818284878b,95:cccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3e4e5e6e7ecff,96:0713181b1e20232425262728292b2c2d2f303738393a3e41434a4e4f5152535657#58595a5c5d5e606365666b6d6e6f70717378797a7b7c7d7e7f808182838487898a,8f:8d8e8f989a,8ece62:0b171b1f222125242c,81e774:eff4ff,75:0f1113,65:34eeeff0,66:0a19,677266:031500,708566:f71d34313635,800666:5f54414f56615777848ca79dbedbdce6e9,8d:3233363b3d4045464849474d5559,89:c7cacbcccecfd0d1,72:6e9f5d666f7e7f848b8d8f92,63:0832b0,96:8c8e91929395969a9b9d9e9fa0a1a2a3a4a5a6a8a9aaabacadaeafb1b2b4b5b7b8babbbfc2c3c8cacbd0d1d3d4d6d7d8d9dadbdcdddedfe1e2e3e4e5e6e7eb#ecedeef0f1f2f4f5f8fafbfcfdff,97:0203050a0b0c10111214151718191a1b1d1f20,64:3fd8,80046b:eaf3fdf5f9,6c:0507060d1518191a2129242a32,65:35556b,72:4d525630,8662521680:9f9c93bc,670a80:bdb1abadb4b7e7e8e9eadbc2c4d9cdd7,671080:ddebf1f4ed,81:0d0e,80:f2fc,671581128c5a81:361e2c1832484c5374595a7160697c7d6d67,584d5ab581:888291,6ed581:a3aacc,672681:cabb,97:2122232425262728292b2c2e2f3133343536373a3b3c3d3f404142434445464748494a4b4c4d4e4f5051545557585a5c5d5f63646667686a6b6c6d6e6f7071#72757778797a7b7d7e7f8081828384868788898a8c8e8f9093959697999a9b9c9d,81:c1a6,6b:243739434659,98:d1d2d3d5d9da,6bb35f406bc289f365909f5165:93bcc6c4c3ccced2d6,70:809c969dbbc0b7abb1e8ca,71:1013162f31735c6845724a787a98b3b5a8a0e0d4e7f9,72:1d28,706c71:1866b9,62:3e3d434849,79:3b4046495b5c535a6257606f677a858a9aa7b3,5f:d1d0,97:9e9fa1a2a4a5a6a7a8a9aaacaeb0b1b3b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1e2e3#e4e5e8eeeff0f1f2f4f7f8f9fafbfcfdfeff,98:000102030405060708090a0b0c0d0e,60:3c5d5a67415963ab,61:060d5da99dcbd1,620680:807f,6c:93f6,6dfc77:f6f8,78:0009171811,65ab78:2d1c1d393a3b1f3c252c23294e6d56572650474c6a9b939a879ca1a3b2b9a5d4d9c9ecf2,790578f479:13241e34,9f9b9e:f9fbfc,76f177:040d,76f977:07081a22192d263538505147435a68,98:0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142434445464748494a4b4c4d#4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e,77:62657f8d7d808c919fa0b0b5bd,75:3a404e4b485b727983,7f:58615f,8a487f:68747179817e,76:cde5,883294:8586878b8a8c8d8f909497959a9b9ca3a4abaaadacafb0b2b4b6b7b8b9babcbdbfc4c8c9cacbcccdced0d1d2d5d6d7d9d8dbdedfe0e2e4e5e7e8ea,98:6f70717273748b8e929599a3a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcfd0d4d6d7dbdcdde0e1e2e3e4#e5e6e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,99:0001020304050607,94:e9ebeeeff3f4f5f7f9fcfdff,95:03020607090a0d0e0f1213141516181b1d1e1f222a2b292c3132343637383c3e3f4235444546494c4e4f525354565758595b5e5f5d61626465666768696a6b6c6f7172733a,77:e7ec,96c979:d5ede3eb,7a065d477a:03021e14,99:08090a0b0c0e0f1112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2f303132333435363738393a3b3c3d3e3f40414243444546474849#4a4b4c4d4e4f50515253565758595a5b5c5d5e5f60616264667378797b7e828389,7a:393751,9ecf99a57a7076:888e9399a4,74:dee0,752c9e:202228292a2b2c3231363837393a3e414244464748494b4c4e5155575a5b5c5e63666768696a6b6c716d73,75:929496a09daca3b3b4b8c4b1b0c3c2d6cde3e8e6e4ebe7,760375:f1fcff,76:1000050c170a25181519,99:8c8e9a9b9c9d9e9fa0a1a2a3a4a6a7a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8#d9dadbdcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9,76:1b3c2220402d303f35433e334d5e545c566b6f,7fca7a:e6787980868895a6a0aca8adb3,88:6469727d7f82a2c6b7bcc9e2cee3e5f1,891a88:fce8fef0,89:2119131b0a342b3641667b,758b80e576:b2b4,77dc80:1214161c20222526272928310b3543464d526971,898398:788083,99:fafbfcfdfeff,9a:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738#393a3b3c3d3e3f404142434445464748494a4b4c4d4e4f50515253545556575859,98:898c8d8f949a9b9e9fa1a2a5a6,86:4d546c6e7f7a7c7ba88d8bac9da7a3aa93a9b6c4b5ceb0bab1afc9cfb4e9f1f2edf3d0,871386:def4dfd8d1,87:0307,86f887:080a0d09233b1e252e1a3e48343129373f82227d7e7b60704c6e8b53637c64596593afa8d2,9a:5a5b5c5d5e5f606162636465666768696a6b7283898d8e949599a6a9aaabacadaeafb2b3b4b5b9bbbdbebfc3c4c6c7c8c9cacdcecfd0d2d4d5d6d7d9dadbdc#dddee0e2e3e4e5e7e8e9eaeceef0f1f2f3f4f5f6f7f8fafcfdfeff,9b:000102040506,87:c68885ad9783abe5acb5b3cbd3bdd1c0cadbeae0ee,88:1613,87fe88:0a1b21393c,7f:36424445,82107a:fafd,7b:080304150a2b0f47382a192e31202524333e1e585a45754c5d606e7b62727190a6a7b8ac9da885aa9ca2abb4d1c1ccdddae5e6ea,7c0c7b:fefc,7c:0f160b,9b:07090a0b0c0d0e1011121415161718191a1b1c1d1e2021222425262728292a2b2c2d2e3031333435363738393a3d3e3f40464a4b4c4e50525355565758595a#5b5c5d5e5f606162636465666768696a6b6c6d6e6f707172737475767778797a7b,7c:1f2a26384140,81fe82:010204,81ec884482:2122232d2f282b383b33343e44494b4f5a5f68,88:7e8588d8df,895e7f:9d9fa7afb0b2,7c7c65497c:919d9c9ea2b2bcbdc1c7cccdc8c5d7e8,826e66a87f:bfced5e5e1e6e9eef3,7cf87d:77a6ae,7e:479b,9e:b8b4,8d:73849491b1676d,8c:4749,91:4a504e4f64,9b:7c7d7e7f808182838485868788898a8b8c8d8e8f909192939495969798999a9b9c9d9e9fa0a1a2a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9ba#bbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadb,91:626170696f7d7e7274798c85908d91a2a3aaadaeafb5b4ba,8c559e7e8d:b8eb,8e:055969,8d:b5bfbcbac4d6d7dadececfdbc6ecf7f8e3f9fbe4,8e098dfd8e:141d1f2c2e232f3a4039353d3149414251524a70767c6f74858f94909c9e,8c:78828a859894,659b89:d6dedadc,9b:dcdddedfe0e1e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,9c:000102030405060708090a0b0c0d0e0f101112131415161718191a#1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b,89:e5ebef,8a3e8b26975396:e9f3ef,97:0601080f0e2a2d303e,9f:808385868788898a8c,9efe9f:0b0d,96:b9bcbdced2,77bf96e092:8eaec8,93:3e6aca8f,94:3e6b,9c:7f8285868788,7a239c:8b8e90919294959a9b9e9fa0a1a2a3a5a6a7a8a9abadaeb0b1b2b3b4b5b6b7babbbcbdc4c5c6c7cacb3c3d3e3f404142434445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f707172737475767778797a#7b7d7e808384898a8c8f93969798999daaacafb9bebfc0c1c2c8c9d1d2dadbe0e1cccdcecfd0d3d4d5d7d8d9dcdddfe2,97:7c85919294afaba3b2b4,9a:b1b0b7,9e589a:b6babcc1c0c5c2cbccd1,9b:45434749484d51,98e899:0d2e5554,9a:dfe1e6efebfbedf9,9b:080f131f23,9e:bdbe,7e3b9e:8287888b92,93d69e:9d9fdbdcdde0dfe2e9e7e5eaef,9f:222c2f39373d3e44,9c:e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,9d:000102030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e1f2021#22232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f404142#92$434445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f707172737475767778797a7b7c7d7e7f8081#82838485868788898a8b8c8d8e8f909192939495969798999a9b9c9d9e9fa0a1a2#92$a3a4a5a6a7a8a9aaabacadaeafb0b1b2b3b4b5b6b7b8b9babbbcbdbebfc0c1c2c3c4c5c6c7c8c9cacbcccdcecfd0d1d2d3d4d5d6d7d8d9dadbdcdddedfe0e1#e2e3e4e5e6e7e8e9eaebecedeeeff0f1f2f3f4f5f6f7f8f9fafbfcfdfeff,9e:000102#92$030405060708090a0b0c0d0e0f101112131415161718191a1b1c1d1e24272e30343b3c404d5052535456595d5f606162656e6f727475767778797a7b7c7d80#8183848586898a8c8d8e8f90919495969798999a9b9c9ea0a1a2a3a4a5a7a8a9aa#92$abacadaeafb0b1b2b3b5b6b7b9babcbfc0c1c2c3c5c6c7c8cacbccd0d2d3d5d6d7d9dadee1e3e4e6e8ebecedeef0f1f2f3f4f5f6f7f8fafdff,9f:000102030405#060708090a0c0f1112141516181a1b1c1d1e1f21232425262728292a2b2d2e3031#92$3233343536383a3c3f4041424345464748494a4b4c4d4e4f52535455565758595a5b5c5d5e5f606162636465666768696a6b6c6d6e6f707172737475767778#797a7b7c7d7e81828d8e8f9091929394959697989c9d9ea1a2a3a4a5,f9:2c7995e7f1#92$,fa:0c0d0e0f111314181f20212324272829,e8:15161718191a1b1c1d1e1f202122232425262728292a2b2c2d2e2f303132333435363738393a3b3c3d3e3f40414243#4445464748494a4b4c4d4e4f505152535455565758595a5b5c5d5e5f6061626364"
  1272. );
  1273. let U2Ghash = {};
  1274. let G2Uhash = {};
  1275. handleHash();
  1276. /**
  1277. * 编码
  1278. * @param {string} str
  1279. * @returns {string}
  1280. */
  1281. this.encode = function (str) {
  1282. return [...str].reduce((result, val, i) => {
  1283. return result + toGBK(val);
  1284. }, "");
  1285. function toGBK(val) {
  1286. let result = "";
  1287. for (let i = 0; i < val.length; i++) {
  1288. const codePoint = val.codePointAt(i);
  1289. const code = String.fromCodePoint(codePoint);
  1290. let key = codePoint.toString(16);
  1291. key.length != 4 && (key = ("000" + key).match(/....$/)[0]);
  1292.  
  1293. /* Add up i by code.length */
  1294. i += code.length - 1;
  1295.  
  1296. /* If code is in ascii range */
  1297. if (isAscii(codePoint)) {
  1298. result += encodeURIComponent(code);
  1299. continue;
  1300. }
  1301.  
  1302. /* If Got encoded string from U2Ghash */
  1303. if (U2Ghash[key]) {
  1304. result += U2Ghash[key];
  1305. continue;
  1306. }
  1307.  
  1308. /*
  1309. If 2 or more char combines to one visible code,
  1310. or just this code is not in GBK
  1311. */
  1312. result += toGBK(`&#${codePoint};`);
  1313. }
  1314. return result;
  1315. }
  1316. };
  1317.  
  1318. /**
  1319. * 解码
  1320. * @param {string} str
  1321. * @returns {string}
  1322. */
  1323. this.decode = function (str) {
  1324. var GBKMatcher = /%[0-9A-F]{2}%[0-9A-F]{2}/;
  1325. var UTFMatcher = /%[0-9A-F]{2}/;
  1326. var gbk = true,
  1327. utf = true;
  1328. while (utf) {
  1329. gbk = str.match(GBKMatcher);
  1330. utf = str.match(UTFMatcher);
  1331. if (gbk && gbk in G2Uhash) {
  1332. str = str.replace(gbk, String.fromCharCode("0x" + G2Uhash[gbk]));
  1333. } else {
  1334. str = str.replace(utf, decodeURIComponent(utf));
  1335. }
  1336. }
  1337. return str;
  1338. };
  1339. return this;
  1340. };
  1341.  
  1342. Utils.getTransitionEndNameList = function () {
  1343. return [
  1344. "webkitTransitionEnd",
  1345. "mozTransitionEnd",
  1346. "MSTransitionEnd",
  1347. "otransitionend",
  1348. "transitionend",
  1349. ];
  1350. };
  1351.  
  1352. Utils.getAnimationEndNameList = function () {
  1353. return [
  1354. "webkitAnimationEnd",
  1355. "mozAnimationEnd",
  1356. "MSAnimationEnd",
  1357. "oanimationend",
  1358. "animationend",
  1359. ];
  1360. };
  1361.  
  1362. Utils.getArrayLastValue = function (targetObj) {
  1363. return targetObj[targetObj.length - 1];
  1364. };
  1365.  
  1366. Utils.getArrayRealValue = function () {
  1367. let result = null;
  1368. for (let arg of arguments) {
  1369. if (typeof arg === "function") {
  1370. /* 方法 */
  1371. arg = arg();
  1372. }
  1373. if (arg != null) {
  1374. result = arg;
  1375. break;
  1376. }
  1377. }
  1378. return result;
  1379. };
  1380.  
  1381. Utils.getDaysDifference = function (
  1382. timestamp1 = Date.now(),
  1383. timestamp2 = Date.now(),
  1384. type = "天"
  1385. ) {
  1386. type = type.trim();
  1387. if (timestamp1.toString().length === 10) {
  1388. timestamp1 = timestamp1 * 1000;
  1389. }
  1390. if (timestamp2.toString().length === 10) {
  1391. timestamp2 = timestamp2 * 1000;
  1392. }
  1393. let smallTimeStamp = timestamp1 > timestamp2 ? timestamp2 : timestamp1;
  1394. let bigTimeStamp = timestamp1 > timestamp2 ? timestamp1 : timestamp2;
  1395. let oneSecond = 1000; /* 一秒的毫秒数 */
  1396. let oneMinute = 60 * oneSecond; /* 一分钟的毫秒数 */
  1397. let oneHour = 60 * oneMinute; /* 一小时的毫秒数 */
  1398. let oneDay = 24 * oneHour; /* 一天的毫秒数 */
  1399. let oneMonth = 30 * oneDay; /* 一个月的毫秒数(30天) */
  1400. let oneYear = 12 * oneMonth; /* 一年的毫秒数 */
  1401. let bigDate = new Date(bigTimeStamp);
  1402. let smallDate = new Date(smallTimeStamp);
  1403. let remainderValue = 1;
  1404. if (type === "年") {
  1405. remainderValue = oneYear;
  1406. } else if (type === "月") {
  1407. remainderValue = oneMonth;
  1408. } else if (type === "天") {
  1409. remainderValue = oneDay;
  1410. } else if (type === "时") {
  1411. remainderValue = oneHour;
  1412. } else if (type === "分") {
  1413. remainderValue = oneMinute;
  1414. } else if (type === "秒") {
  1415. remainderValue = oneSecond;
  1416. }
  1417. let diffValue = Math.round(
  1418. Math.abs((bigDate - smallDate) / remainderValue)
  1419. );
  1420. if (type === "auto") {
  1421. let timeDifference = bigTimeStamp - smallTimeStamp;
  1422. diffValue = Math.floor(timeDifference / (24 * 3600 * 1000));
  1423. if (diffValue > 0) {
  1424. diffValue = diffValue + "天";
  1425. } else {
  1426. /* 计算出小时数 */
  1427. let leave1 =
  1428. timeDifference % (24 * 3600 * 1000); /* 计算天数后剩余的毫秒数 */
  1429. let hours = Math.floor(leave1 / (3600 * 1000));
  1430. if (hours > 0) {
  1431. diffValue = hours + "小时";
  1432. } else {
  1433. /* 计算相差分钟数 */
  1434. let leave2 = leave1 % (3600 * 1000); /* 计算小时数后剩余的毫秒数 */
  1435. let minutes = Math.floor(leave2 / (60 * 1000));
  1436. if (minutes > 0) {
  1437. diffValue = minutes + "分钟";
  1438. } else {
  1439. /* 计算相差秒数 */
  1440. let leave3 = leave2 % (60 * 1000); /* 计算分钟数后剩余的毫秒数 */
  1441. let seconds = Math.round(leave3 / 1000);
  1442. diffValue = seconds + "秒";
  1443. }
  1444. }
  1445. }
  1446. }
  1447. return diffValue;
  1448. };
  1449.  
  1450. Utils.getElementSelector = function (element) {
  1451. if (!element) return;
  1452. if (!element.parentElement) return;
  1453. /* 如果元素有id属性,则直接返回id选择器 */
  1454. if (element.id) return "#" + element.id;
  1455.  
  1456. /* 递归地获取父元素的选择器 */
  1457. let selector = Utils.getElementSelector(element.parentElement);
  1458. if (!selector) {
  1459. return element.tagName;
  1460. }
  1461. /* 如果有多个相同类型的兄弟元素,则需要添加索引 */
  1462. if (element.parentElement.querySelectorAll(element.tagName).length > 1) {
  1463. let index =
  1464. Array.prototype.indexOf.call(element.parentElement.children, element) +
  1465. 1;
  1466. selector +=
  1467. " > " + element.tagName.toLowerCase() + ":nth-child(" + index + ")";
  1468. } else {
  1469. selector += " > " + element.tagName.toLowerCase();
  1470. }
  1471. return selector;
  1472. };
  1473.  
  1474. Utils.getMaxValue = function () {
  1475. let result = [...arguments];
  1476. let newResult = [];
  1477. if (result.length > 1) {
  1478. if (
  1479. result.length === 2 &&
  1480. typeof result[0] === "object" &&
  1481. typeof result[1] === "function"
  1482. ) {
  1483. let data = result[0];
  1484. let handleDataFunc = result[1];
  1485. OriginPrototype.Object.keys(data).forEach((keyName) => {
  1486. newResult = [...newResult, handleDataFunc(keyName, data[keyName])];
  1487. });
  1488. } else {
  1489. result.forEach((item) => {
  1490. if (!isNaN(parseFloat(item))) {
  1491. newResult = [...newResult, parseFloat(item)];
  1492. }
  1493. });
  1494. }
  1495. return Math.max(...newResult);
  1496. } else if (result.length === 1) {
  1497. result[0].forEach((item) => {
  1498. if (!isNaN(parseFloat(item))) {
  1499. newResult = [...newResult, parseFloat(item)];
  1500. }
  1501. });
  1502. return Math.max(...newResult);
  1503. }
  1504. };
  1505.  
  1506. Utils.getMaxZIndex = function (deviation = 1) {
  1507. let nodeIndexList = [];
  1508. deviation = Number.isNaN(deviation) ? 1 : deviation;
  1509. document.querySelectorAll("*").forEach((element) => {
  1510. let nodeStyle = window.getComputedStyle(element);
  1511. /* 不对position为static和display为none的元素进行获取它们的z-index */
  1512. if (nodeStyle.position !== "static" && nodeStyle.display !== "none") {
  1513. nodeIndexList = nodeIndexList.concat(parseInt(nodeStyle.zIndex));
  1514. }
  1515. });
  1516. /* 过滤非Boolean类型 */
  1517. nodeIndexList = nodeIndexList.filter(Boolean);
  1518. let currentMaxZIndex = nodeIndexList.length
  1519. ? Math.max(...nodeIndexList)
  1520. : 0;
  1521. return currentMaxZIndex + deviation;
  1522. };
  1523.  
  1524. Utils.getMinValue = function () {
  1525. let result = [...arguments];
  1526. let newResult = [];
  1527. if (result.length > 1) {
  1528. if (
  1529. result.length === 2 &&
  1530. typeof result[0] === "object" &&
  1531. typeof result[1] === "function"
  1532. ) {
  1533. let data = result[0];
  1534. let handleDataFunc = result[1];
  1535. OriginPrototype.Object.keys(data).forEach((keyName) => {
  1536. newResult = [...newResult, handleDataFunc(keyName, data[keyName])];
  1537. });
  1538. } else {
  1539. result.forEach((item) => {
  1540. if (!isNaN(parseFloat(item))) {
  1541. newResult = [...newResult, parseFloat(item)];
  1542. }
  1543. });
  1544. }
  1545. return Math.min(...newResult);
  1546. } else if (result.length === 1) {
  1547. result[0].forEach((item) => {
  1548. if (!isNaN(parseFloat(item))) {
  1549. newResult = [...newResult, parseFloat(item)];
  1550. }
  1551. });
  1552. return Math.min(...newResult);
  1553. }
  1554. };
  1555.  
  1556. Utils.getRandomAndroidUA = function () {
  1557. let androidVersion = Utils.getRandomValue(10, 14);
  1558. let mobileNameList = [
  1559. "LDN-LX3",
  1560. "RNE-L03",
  1561. "ASUS_X00ID Build/NMF26F",
  1562. "WAS-LX3",
  1563. "PRA-LX3",
  1564. "MYA-L03",
  1565. "Moto G Play",
  1566. "Moto C Build/NRD90M.063",
  1567. "Redmi Note 4 Build/NRD90M",
  1568. "HUAWEI VNS-L21 Build/HUAWEIVNS-L21",
  1569. "VTR-L09",
  1570. "TRT-LX3",
  1571. "M2003J15SC Build/RP1A.200720.011; wv",
  1572. "MI 13 Build/OPR1.170623.027; wv",
  1573. ];
  1574. let randomMobile = Utils.getRandomValue(mobileNameList);
  1575. let chromeVersion1 = Utils.getRandomValue(110, 121);
  1576. let chromeVersion2 = Utils.getRandomValue(0, 0);
  1577. let chromeVersion3 = Utils.getRandomValue(2272, 6099);
  1578. let chromeVersion4 = Utils.getRandomValue(1, 218);
  1579. return `Mozilla/5.0 (Linux; Android ${androidVersion}; ${randomMobile}) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion1}.${chromeVersion2}.${chromeVersion3}.${chromeVersion4} Mobile Safari/537.36`;
  1580. };
  1581.  
  1582. Utils.getRandomValue = function () {
  1583. let result = [...arguments];
  1584. if (result.length > 1) {
  1585. if (
  1586. result.length === 2 &&
  1587. typeof result[0] === "number" &&
  1588. typeof result[1] === "number"
  1589. ) {
  1590. let leftNumber = result[0] > result[1] ? result[1] : result[0];
  1591. let rightNumber = result[0] > result[1] ? result[0] : result[1];
  1592. return (
  1593. Math.round(Math.random() * (rightNumber - leftNumber)) + leftNumber
  1594. );
  1595. } else {
  1596. return result[Math.floor(Math.random() * result.length)];
  1597. }
  1598. } else if (result.length === 1) {
  1599. let paramData = result[0];
  1600. if (OriginPrototype.Array.isArray(paramData)) {
  1601. return paramData[Math.floor(Math.random() * paramData.length)];
  1602. } else if (
  1603. typeof paramData === "object" &&
  1604. OriginPrototype.Object.keys(paramData).length > 0
  1605. ) {
  1606. let paramObjDataKey =
  1607. OriginPrototype.Object.keys(paramData)[
  1608. Math.floor(
  1609. Math.random() * OriginPrototype.Object.keys(paramData).length
  1610. )
  1611. ];
  1612. return paramData[paramObjDataKey];
  1613. } else {
  1614. return paramData;
  1615. }
  1616. }
  1617. };
  1618.  
  1619. Utils.getRandomPCUA = function () {
  1620. let chromeVersion1 = Utils.getRandomValue(110, 121);
  1621. let chromeVersion2 = Utils.getRandomValue(0, 0);
  1622. let chromeVersion3 = Utils.getRandomValue(2272, 6099);
  1623. let chromeVersion4 = Utils.getRandomValue(1, 218);
  1624. return `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion1}.${chromeVersion2}.${chromeVersion3}.${chromeVersion4} Safari/537.36`;
  1625. };
  1626.  
  1627. Utils.getReactObj = function (element) {
  1628. let result = {};
  1629. OriginPrototype.Object.keys(element).forEach((domPropsName) => {
  1630. if (domPropsName.startsWith("__react")) {
  1631. let propsName = domPropsName.replace(/__(.+)\$.+/i, "$1");
  1632. if (propsName in result) {
  1633. new Error("重复属性 " + domPropsName);
  1634. } else {
  1635. result[propsName] = element[domPropsName];
  1636. }
  1637. }
  1638. });
  1639. return result;
  1640. };
  1641.  
  1642. Utils.getSymbol = function (target, keyName) {
  1643. if (typeof target !== "object") {
  1644. throw new TypeError("target不是一个对象");
  1645. }
  1646. let objectsSymbols = OriginPrototype.Object.getOwnPropertySymbols(target);
  1647. if (typeof keyName === "string") {
  1648. let findSymbol = objectsSymbols.find((key) => {
  1649. return key.toString() === keyName;
  1650. });
  1651. if (findSymbol) {
  1652. return target[findSymbol];
  1653. }
  1654. } else if (typeof keyName === "symbol") {
  1655. let findSymbol = objectsSymbols.find((key) => {
  1656. return key === keyName;
  1657. });
  1658. if (findSymbol) {
  1659. return target[findSymbol];
  1660. }
  1661. } else {
  1662. let result = {};
  1663. objectsSymbols.forEach((item) => {
  1664. result[item] = target[item];
  1665. });
  1666. return result;
  1667. }
  1668. };
  1669.  
  1670. Utils.getTextLength = function (text) {
  1671. let encoder = new TextEncoder();
  1672. let bytes = encoder.encode(text);
  1673. return bytes.length;
  1674. };
  1675.  
  1676. Utils.getTextStorageSize = function (text, addType = true) {
  1677. return Utils.formatByteToSize(Utils.getTextLength(text), addType);
  1678. };
  1679.  
  1680. Utils.getThunderUrl = function (url) {
  1681. if (url == null) {
  1682. throw new TypeError("url不能为空");
  1683. }
  1684. if (typeof url !== "string") {
  1685. throw new TypeError("url必须是string类型");
  1686. }
  1687. if (url.trim() === "") {
  1688. throw new TypeError("url不能为空字符串或纯空格");
  1689. }
  1690. return `thunder://${globalThis.btoa("AA" + url + "ZZ")}`;
  1691. };
  1692.  
  1693. Utils.GM_addStyle = function (cssText) {
  1694. if (typeof cssText !== "string") {
  1695. throw new Error("Utils.GM_addStyle 参数cssText 必须为String类型");
  1696. }
  1697. let cssNode = document.createElement("style");
  1698. cssNode.setAttribute("type", "text/css");
  1699. cssNode.innerHTML = cssText;
  1700. if (document.head) {
  1701. /* 插入head最后 */
  1702. document.head.appendChild(cssNode);
  1703. } else if (document.body) {
  1704. /* 插入body后 */
  1705. document.body.appendChild(cssNode);
  1706. } else if (document.documentElement.childNodes.length === 0) {
  1707. /* 插入#html第一个元素后 */
  1708. document.documentElement.appendChild(cssNode);
  1709. } else {
  1710. /* 插入head前面 */
  1711. document.documentElement.insertBefore(
  1712. cssNode,
  1713. document.documentElement.childNodes[0]
  1714. );
  1715. }
  1716. return cssNode;
  1717. };
  1718.  
  1719. Utils.GM_Cookie = function () {
  1720. this.list = function (paramDetails = {}, callback = () => {}) {
  1721. /** @type {UtilsGMCookieListResult[]} */
  1722. let resultData = [];
  1723. try {
  1724. let details = {
  1725. url: globalThis.location.href,
  1726. domain: globalThis.location.hostname,
  1727. name: "",
  1728. path: "/",
  1729. };
  1730. details = Utils.assign(details, paramDetails);
  1731. let cookies = document.cookie.split(";");
  1732. cookies.forEach((item) => {
  1733. item = item.trimStart();
  1734. let itemName = item.split("=")[0];
  1735. let itemValue = item.replace(new RegExp("^" + itemName + "="), "");
  1736. let nameRegexp =
  1737. details.name instanceof RegExp
  1738. ? details.name
  1739. : new RegExp("^" + details.name, "g");
  1740. if (itemName.match(nameRegexp)) {
  1741. resultData.push({
  1742. domain: globalThis.location.hostname,
  1743. expirationDate: void 0,
  1744. hostOnly: true,
  1745. httpOnly: false,
  1746. name: itemName,
  1747. path: "/",
  1748. sameSite: "unspecified",
  1749. secure: true,
  1750. session: false,
  1751. value: itemValue,
  1752. });
  1753. return;
  1754. }
  1755. });
  1756. callback(resultData);
  1757. } catch (error) {
  1758. callback(resultData, error);
  1759. }
  1760. };
  1761.  
  1762. this.set = function (paramDetails = {}, callback = () => {}) {
  1763. try {
  1764. let details = {
  1765. url: window.location.href,
  1766. name: "",
  1767. value: "",
  1768. domain: window.location.hostname,
  1769. path: "/",
  1770. secure: true,
  1771. httpOnly: false,
  1772. /**
  1773. * Expires in 30 days
  1774. */
  1775. expirationDate: Math.floor(Date.now()) + 60 * 60 * 24 * 30,
  1776. };
  1777. details = Utils.assign(details, paramDetails);
  1778. let life = details.expirationDate
  1779. ? details.expirationDate
  1780. : Math.floor(Date.now()) + 60 * 60 * 24 * 30;
  1781. let cookieStr =
  1782. details.name +
  1783. "=" +
  1784. decodeURIComponent(details.value) +
  1785. ";expires=" +
  1786. new Date(life).toGMTString() +
  1787. "; path=/";
  1788. document.cookie = cookieStr;
  1789. callback();
  1790. } catch (error) {
  1791. callback(error);
  1792. }
  1793. };
  1794.  
  1795. this.delete = (paramDetails = {}, callback = () => {}) => {
  1796. try {
  1797. let details = {
  1798. url: window.location.href,
  1799. name: "",
  1800. firstPartyDomain: "",
  1801. };
  1802. details = Utils.assign(details, paramDetails);
  1803. let cookieStr =
  1804. details.name + "=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;";
  1805. document.cookie = cookieStr;
  1806. callback();
  1807. } catch (error) {
  1808. callback(error);
  1809. }
  1810. };
  1811. };
  1812.  
  1813. Utils.GM_Menu = function (details) {
  1814. /* 配置数据 */
  1815. let data = details.data || [];
  1816. const GM_Api = {
  1817. /**
  1818. * 获取存储的数据
  1819. * @type {GM_getValue}
  1820. */
  1821. getValue: details.GM_getValue,
  1822. /**
  1823. * 设置数据到存储
  1824. * @type {GM_setValue}
  1825. */
  1826. setValue: details.GM_setValue,
  1827. /**
  1828. * 注册(不可用)菜单
  1829. * @type {GM_registerMenuCommand}
  1830. */
  1831. registerMenuCommand: details.GM_registerMenuCommand,
  1832. /**
  1833. * 卸载菜单
  1834. * @type {GM_unregisterMenuCommand}
  1835. */
  1836. unregisterMenuCommand: details.GM_unregisterMenuCommand,
  1837. };
  1838. for (const keyName of OriginPrototype.Object.keys(GM_Api)) {
  1839. if (typeof GM_Api[keyName] !== "function") {
  1840. throw new Error(
  1841. `Utils.GM_Menu 请在脚本开头加上 @grant ${keyName},且传入该对象`
  1842. );
  1843. }
  1844. }
  1845. let that = this;
  1846. /**
  1847. * 注册(不可用)的菜单的映射信息
  1848. * @type {Map<number,UtilsGMMenuConstructorOptions>}
  1849. */
  1850. let menuIdMap = new Map();
  1851.  
  1852. /**
  1853. * 本地存储的键名
  1854. */
  1855. let LocalStorage_Key_Name = "GM_Menu_Local_Map";
  1856. /**
  1857. * 菜单enable为true的emoji
  1858. */
  1859. let Enable_True_Emoji = "✅";
  1860. /**
  1861. * 菜单enable为false的emoji
  1862. */
  1863. let Enable_False_Emoji = "❌";
  1864. /* 自动刷新网页,默认为true */
  1865. let Default_AutoReload =
  1866. typeof details.autoReload === "boolean" ? details.autoReload : true;
  1867. /**
  1868. * 菜单isStoreValue的默认值
  1869. */
  1870. let Default_IsStoreValue = true;
  1871. /**
  1872. * 获取本地存储菜单键值
  1873. * @param {string} key 键
  1874. * @returns {boolean}
  1875. */
  1876. function getLocalMenuData(key, defaultValue) {
  1877. let localData = GM_Api.getValue(LocalStorage_Key_Name, {});
  1878. return localData[key] == null ? defaultValue : localData[key];
  1879. }
  1880.  
  1881. /**
  1882. * 设置本地存储菜单键值
  1883. * @param {string} key 键
  1884. * @param {boolean} value 值
  1885. */
  1886. function setLocalMenuData(key, value) {
  1887. let localData = GM_Api.getValue(LocalStorage_Key_Name, {});
  1888. localData[key] = value;
  1889. GM_Api.setValue(LocalStorage_Key_Name, localData);
  1890. }
  1891.  
  1892. /**
  1893. * 对菜单数据进行处理
  1894. * @param { UtilsGMMenuConstructorOptions } menuOption
  1895. * @returns { any[] }
  1896. */
  1897. function handleMenuData(menuOption) {
  1898. /* 本地存储的键名 */
  1899. let menuLocalDataItemKey = menuOption.key;
  1900. /* 文本 */
  1901. let text = menuOption.text;
  1902. /* 菜单默认开启的状态 */
  1903. let defaultEnable = Boolean(
  1904. getLocalMenuData(menuLocalDataItemKey, menuOption.enable)
  1905. );
  1906. /* 油猴菜单上显示的文本 */
  1907. let showText = menuOption.showText(text, defaultEnable);
  1908. /* 用户点击后的回调 */
  1909. let defaultClickCallBack = menuOption.callback;
  1910.  
  1911. let menuOptions = {
  1912. /**
  1913. * 菜单的id
  1914. */
  1915. id: menuOption.id,
  1916. /**
  1917. * 点击菜单项后是否应关闭弹出菜单
  1918. */
  1919. autoClose: menuOption.autoClose,
  1920. /**
  1921. * 菜单项的可选访问键
  1922. */
  1923. accessKey: menuOption.accessKey,
  1924. /**
  1925. * 菜单项的鼠标悬浮上的工具提示
  1926. */
  1927. title: menuOption.title,
  1928. };
  1929. /* 点击菜单后触发callback后的网页是否刷新 */
  1930. menuOption.autoReload =
  1931. typeof menuOption.autoReload !== "boolean"
  1932. ? Default_AutoReload
  1933. : menuOption.autoReload;
  1934. menuOption.isStoreValue =
  1935. typeof menuOption.isStoreValue !== "boolean"
  1936. ? Default_IsStoreValue
  1937. : menuOption.isStoreValue;
  1938. /* 用户点击菜单后的回调函数 */
  1939. function clickCallBack(event) {
  1940. let localEnable = Boolean(
  1941. getLocalMenuData(menuLocalDataItemKey, defaultEnable)
  1942. );
  1943. if (menuOption.isStoreValue) {
  1944. setLocalMenuData(menuLocalDataItemKey, !localEnable);
  1945. }
  1946. if (typeof defaultClickCallBack === "function") {
  1947. defaultClickCallBack({
  1948. key: menuLocalDataItemKey,
  1949. enable: !localEnable,
  1950. oldEnable: localEnable,
  1951. event: event,
  1952. storeValue(_value_) {
  1953. setLocalMenuData(menuLocalDataItemKey, _value_);
  1954. },
  1955. });
  1956. }
  1957. /* 不刷新网页就刷新菜单 */
  1958. if (menuOption.autoReload) {
  1959. window.location.reload();
  1960. } else {
  1961. that.update();
  1962. }
  1963. }
  1964. let menuOptionsLength = OriginPrototype.Object.values(menuOptions).filter(
  1965. (_item_) => _item_ != null
  1966. ).length;
  1967.  
  1968. return [showText, clickCallBack];
  1969. /* 下面的暂不启用 */
  1970. if (menuOptionsLength === 0) {
  1971. return [showText, clickCallBack];
  1972. } else if (menuOptionsLength === 1 && menuOptions.accessKey != null) {
  1973. return [showText, clickCallBack, menuOptions["accessKey"]];
  1974. } else {
  1975. /* 这个是版本 > 4.20.6186才会有的选项 */
  1976. return [showText, clickCallBack, menuOptions];
  1977. }
  1978. }
  1979.  
  1980. /**
  1981. * 处理初始化配置
  1982. * @param { UtilsGMMenuConstructorOptions } menuOption
  1983. * @returns { UtilsGMMenuConstructorOptions }
  1984. */
  1985. function handleInitDetail(menuOption) {
  1986. menuOption.enable = Boolean(
  1987. getLocalMenuData(menuOption.key, menuOption.enable)
  1988. );
  1989. if (typeof menuOption.showText !== "function") {
  1990. menuOption.showText = function (menuText, menuEnable) {
  1991. return (
  1992. (menuEnable ? Enable_True_Emoji : Enable_False_Emoji) +
  1993. " " +
  1994. menuText
  1995. );
  1996. };
  1997. }
  1998. return menuOption;
  1999. }
  2000. /**
  2001. * 初始化数据
  2002. * @returns { UtilsGMMenuConstructorOptions[] }
  2003. */
  2004. function init() {
  2005. menuIdMap.clear();
  2006. data.forEach((dataItem, dataIndex) => {
  2007. dataItem = handleInitDetail(dataItem);
  2008. data[dataIndex].enable = dataItem.enable;
  2009. });
  2010. }
  2011.  
  2012. /**
  2013. * 注册(不可用)油猴菜单
  2014. * @param { ?UtilsGMMenuConstructorOptions[] } menuOptions 如果存在,使用它
  2015. */
  2016. function register(menuOptions) {
  2017. (menuOptions || data).forEach((menuOption) => {
  2018. let resultList = handleMenuData(menuOption);
  2019. data.id = GM_Api.registerMenuCommand(...resultList);
  2020. menuIdMap.set(data.id, menuOption);
  2021. });
  2022. }
  2023.  
  2024. /**
  2025. * 获取目标菜单配置
  2026. * @param {string} menuKey 菜单-键key
  2027. * @returns {UtilsGMMenuConstructorOptions}
  2028. */
  2029. function getTargetMenu(menuKey) {
  2030. return data.find((item) => item.key == menuKey);
  2031. }
  2032. /**
  2033. * 根据键值获取enable值
  2034. * @param {string} menuKey 菜单-键key
  2035. * @returns {boolean}
  2036. */
  2037. this.get = function (menuKey) {
  2038. return this.getEnable(menuKey);
  2039. };
  2040. /**
  2041. * 根据键值获取enable值
  2042. * @param {string} menuKey 菜单-键key
  2043. * @returns {boolean}
  2044. */
  2045. this.getEnable = function (menuKey) {
  2046. return getTargetMenu(menuKey).enable;
  2047. };
  2048. /**
  2049. * 根据键值获取text值
  2050. * @param {string} menuKey 菜单-键key
  2051. * @returns {string}
  2052. */
  2053. this.getText = function (menuKey) {
  2054. return getTargetMenu(menuKey).text;
  2055. };
  2056. /**
  2057. * 根据键值获取showText函数的值
  2058. * @param {string} menuKey 菜单-键key
  2059. * @returns {string}
  2060. */
  2061. this.getShowTextValue = function (menuKey) {
  2062. return getTargetMenu(menuKey).showText(
  2063. this.getText(menuKey),
  2064. this.get(menuKey)
  2065. );
  2066. };
  2067. /**
  2068. * 获取当前已注册(不可用)菜单的id
  2069. * @param {string} menuKey
  2070. * @returns {?number}
  2071. */
  2072. this.getMenuId = function (menuKey) {
  2073. let result = null;
  2074. menuIdMap.forEach((value, key) => {
  2075. if (value.key === menuKey) {
  2076. result = key;
  2077. return;
  2078. }
  2079. });
  2080. return result;
  2081. };
  2082. /**
  2083. * 根据键值获取accessKey值
  2084. * @param {string} menuKey 菜单-键key
  2085. * @returns {?string}
  2086. */
  2087. this.getAccessKey = function (menuKey) {
  2088. return getTargetMenu(menuKey).accessKey;
  2089. };
  2090. /**
  2091. * 根据键值获取autoClose值
  2092. * @param {string} menuKey 菜单-键key
  2093. * @returns {?boolean}
  2094. */
  2095. this.getAutoClose = function (menuKey) {
  2096. return getTargetMenu(menuKey).autoClose;
  2097. };
  2098. /**
  2099. * 根据键值获取autoReload值
  2100. * @param {string} menuKey 菜单-键key
  2101. * @returns {boolean}
  2102. */
  2103. this.getAutoReload = function (menuKey) {
  2104. return getTargetMenu(menuKey).autoReload;
  2105. };
  2106. /**
  2107. * 根据键值获取callback函数
  2108. * @param {string} menuKey 菜单-键key
  2109. * @returns {Function|undefined}
  2110. */
  2111. this.getCallBack = function (menuKey) {
  2112. return getTargetMenu(menuKey).callback;
  2113. };
  2114. /**
  2115. * 获取当enable为true时默认显示在菜单中前面的emoji图标
  2116. * @returns {string}
  2117. */
  2118. this.getEnableTrueEmoji = function () {
  2119. return Enable_True_Emoji;
  2120. };
  2121. /**
  2122. * 获取当enable为false时默认显示在菜单中前面的emoji图标
  2123. * @returns {string}
  2124. */
  2125. this.getEnableFalseEmoji = function () {
  2126. return Enable_False_Emoji;
  2127. };
  2128. /**
  2129. * 获取本地存储的菜单外部的键名
  2130. * @param {string} keyName
  2131. */
  2132. this.getLocalStorageKeyName = function () {
  2133. return LocalStorage_Key_Name;
  2134. };
  2135. /**
  2136. * 设置菜单的值
  2137. * @param {string} menuKey 菜单-键key
  2138. * @param {any} value 需要设置的值
  2139. */
  2140. this.setValue = function (menuKey, value) {
  2141. setLocalMenuData(menuKey, value);
  2142. };
  2143. /**
  2144. * 设置菜单的值
  2145. * @param {string} menuKey 菜单-键key
  2146. * @param {boolean} value 需要设置的值
  2147. */
  2148. this.setEnable = function (menuKey, value) {
  2149. this.setValue(menuKey, Boolean(value));
  2150. };
  2151. /**
  2152. * 设置当enable为true时默认显示在菜单中前面的emoji图标
  2153. * @param {string} emojiString
  2154. */
  2155. this.setEnableTrueEmoji = function (emojiString) {
  2156. if (typeof emojiString !== "string") {
  2157. throw new Error("参数emojiString必须是string类型");
  2158. }
  2159. Enable_True_Emoji = emojiString;
  2160. };
  2161. /**
  2162. * 设置当enable为false时默认显示在菜单中前面的emoji图标
  2163. * @param {string} emojiString
  2164. */
  2165. this.setEnableFalseEmoji = function (emojiString) {
  2166. if (typeof emojiString !== "string") {
  2167. throw new Error("参数emojiString必须是string类型");
  2168. }
  2169. Enable_False_Emoji = emojiString;
  2170. };
  2171. /**
  2172. * 设置本地存储的菜单外部的键名
  2173. * @param {string} keyName
  2174. */
  2175. this.setLocalStorageKeyName = function (keyName) {
  2176. if (typeof keyName !== "string") {
  2177. throw new Error("参数keyName必须是string类型");
  2178. }
  2179. LocalStorage_Key_Name = keyName;
  2180. };
  2181. /**
  2182. * 新增菜单数据
  2183. * @param {UtilsGMMenuConstructorOptions[]|UtilsGMMenuConstructorOptions} paramData
  2184. */
  2185. this.add = function (paramData) {
  2186. if (OriginPrototype.Array.isArray(paramData)) {
  2187. data = data.concat(paramData);
  2188. } else {
  2189. data.push(paramData);
  2190. }
  2191. this.update();
  2192. };
  2193. /**
  2194. * 更新菜单数据
  2195. * @param { UtilsGMMenuConstructorOptions[]|UtilsGMMenuConstructorOptions|undefined } options 数据
  2196. */
  2197. this.update = function (options) {
  2198. let optionsList = [];
  2199. if (OriginPrototype.Array.isArray(options)) {
  2200. /* 是数组 */
  2201. optionsList = optionsList.concat(options);
  2202. } else if (options != null) {
  2203. /* 是单个配置 */
  2204. optionsList.push(options);
  2205. }
  2206. optionsList.forEach((item) => {
  2207. let targetMenu = getTargetMenu(item.key);
  2208. if (targetMenu) {
  2209. OriginPrototype.Object.assign(targetMenu, item);
  2210. }
  2211. });
  2212. menuIdMap.forEach((value, key) => {
  2213. this.delete(key);
  2214. });
  2215. init();
  2216. register();
  2217. };
  2218. /**
  2219. * 根据已注册(不可用)菜单的id,来更新菜单配置,不会卸载菜单导致可能菜单选项可能会变化的情况
  2220. * 暂时用不到该api,因为需要油猴版本>5.0,其它的类似油猴的扩展可能没实现这个
  2221. * @param { UtilsGMMenuConstructorOptions[]|UtilsGMMenuConstructorOptions } menuOptions 配置
  2222. */
  2223. this.updateOptionsWithId = function (menuOptions) {
  2224. /**
  2225. * @type {UtilsGMMenuConstructorOptions[]}
  2226. */
  2227. let optionsList = [];
  2228. if (OriginPrototype.Array.isArray(menuOptions)) {
  2229. /* 是数组 */
  2230. optionsList = optionsList.concat(menuOptions);
  2231. } else if (menuOptions != null) {
  2232. /* 是单个配置 */
  2233. optionsList.push(menuOptions);
  2234. }
  2235. for (let option of optionsList) {
  2236. menuIdMap.forEach((value, key) => {
  2237. if (key === option.id) {
  2238. option = handleInitDetail(option);
  2239. let findDataIndex = data.findIndex(
  2240. (item) => item.key === value.key
  2241. );
  2242. if (findDataIndex !== -1) {
  2243. OriginPrototype.Object.assign(data[findDataIndex], option);
  2244. }
  2245. register([option]);
  2246. }
  2247. });
  2248. }
  2249. };
  2250. /**
  2251. * 卸载菜单
  2252. * @param {number} menuId 已注册(不可用)的菜单id
  2253. */
  2254. this.delete = function (menuId) {
  2255. GM_Api.unregisterMenuCommand(menuId);
  2256. };
  2257.  
  2258. this.update();
  2259. };
  2260.  
  2261. Utils.Hooks = function () {
  2262. this.initEnv = function () {
  2263. Function.prototype.hook = function (realFunc, hookFunc, context) {
  2264. let _context = null; //函数上下文
  2265. let _funcName = null; //函数名
  2266.  
  2267. _context = context || window;
  2268. _funcName = getFuncName(this);
  2269. _context["realFunc_" + _funcName] = this;
  2270.  
  2271. if (
  2272. _context[_funcName].prototype &&
  2273. _context[_funcName].prototype.isHooked
  2274. ) {
  2275. console.log("Already has been hooked,unhook first");
  2276. return false;
  2277. }
  2278. function getFuncName(fn) {
  2279. // 获取函数名
  2280. let strFunc = fn.toString();
  2281. let _regex = /function\s+(\w+)\s*\(/;
  2282. let patten = strFunc.match(_regex);
  2283. if (patten) {
  2284. return patten[1];
  2285. }
  2286. return "";
  2287. }
  2288. try {
  2289. eval(
  2290. "_context[_funcName] = function " +
  2291. _funcName +
  2292. "(){\n" +
  2293. "let args = Array.prototype.slice.call(arguments,0);\n" +
  2294. "let obj = this;\n" +
  2295. "hookFunc.apply(obj,args);\n" +
  2296. "return _context['realFunc_" +
  2297. _funcName +
  2298. "'].apply(obj,args);\n" +
  2299. "};"
  2300. );
  2301. _context[_funcName].prototype.isHooked = true;
  2302. return true;
  2303. } catch (e) {
  2304. console.log("Hook failed,check the params.");
  2305. return false;
  2306. }
  2307. };
  2308. Function.prototype.unhook = function (realFunc, funcName, context) {
  2309. let _context = null;
  2310. let _funcName = null;
  2311. _context = context || window;
  2312. _funcName = funcName;
  2313. if (!_context[_funcName].prototype.isHooked) {
  2314. console.log("No function is hooked on");
  2315. return false;
  2316. }
  2317. _context[_funcName] = _context["realFunc" + _funcName];
  2318. OriginPrototype.Reflect.deleteProperty(
  2319. _context,
  2320. "realFunc_" + _funcName
  2321. );
  2322. return true;
  2323. };
  2324. };
  2325. this.cleanEnv = function () {
  2326. if (OriginPrototype.Function.hasOwnProperty("hook")) {
  2327. OriginPrototype.Reflect.deleteProperty(unction.prototype, "hook");
  2328. }
  2329. if (OriginPrototype.Function.hasOwnProperty("unhook")) {
  2330. OriginPrototype.Reflect.deleteProperty(unction.prototype, "unhook");
  2331. }
  2332. return true;
  2333. };
  2334. };
  2335.  
  2336. Utils.Httpx = function (__xmlHttpRequest__) {
  2337. if (typeof __xmlHttpRequest__ !== "function") {
  2338. console.warn("Httpx未传入GM_xmlhttpRequest函数,强制默认使用fetch");
  2339. }
  2340. const GM_Api = {
  2341. /**
  2342. * @type {GM_xmlhttpRequest}
  2343. */
  2344. xmlHttpRequest: __xmlHttpRequest__,
  2345. };
  2346. /**
  2347. * @type {HttpxDetails}
  2348. */
  2349. let defaultDetails = {
  2350. url: void 0,
  2351. timeout: 5000,
  2352. async: false,
  2353. responseType: void 0,
  2354. headers: void 0,
  2355. data: void 0,
  2356. redirect: void 0,
  2357. cookie: void 0,
  2358. binary: void 0,
  2359. nocache: void 0,
  2360. revalidate: void 0,
  2361. context: void 0,
  2362. overrideMimeType: void 0,
  2363. anonymous: void 0,
  2364. fetch: void 0,
  2365. fetchInit: void 0,
  2366. user: void 0,
  2367. password: void 0,
  2368. onabort() {},
  2369. onerror() {},
  2370. ontimeout() {},
  2371. onloadstart() {},
  2372. onreadystatechange() {},
  2373. onprogress() {},
  2374. };
  2375. /**
  2376. * 输出请求配置
  2377. */
  2378. let LOG_DETAILS = false;
  2379.  
  2380. const HttpxRequestHook = {
  2381. /**
  2382. * 发送请求前的回调
  2383. * 如果返回false则阻止本次返回
  2384. * @param {HttpxDetails} details 当前的请求配置
  2385. */
  2386. beforeRequestCallBack(details) {},
  2387. };
  2388.  
  2389. const HttpxRequestDetails = {
  2390. /**
  2391. * 获取请求配置
  2392. * @param {HttpxMethod} method 当前请求方法,默认get
  2393. * @param {(...args: any[])=>void} resolve promise回调
  2394. * @param {HttpxDetails} details 请求配置
  2395. * @returns
  2396. */
  2397. getDetails(method, resolve, details) {
  2398. /**
  2399. * @type {HttpxDetails}
  2400. */
  2401. let result = {
  2402. url: details.url || defaultDetails.url,
  2403. method: (method || "GET").toString().toUpperCase(),
  2404. timeout: details.timeout || defaultDetails.timeout,
  2405. responseType: details.responseType || defaultDetails.responseType,
  2406. /* 对象使用深拷贝 */
  2407. headers: Utils.deepClone(defaultDetails.headers),
  2408. data: details.data || defaultDetails.data,
  2409. redirect: details.redirect || defaultDetails.redirect,
  2410. cookie: details.cookie || defaultDetails.cookie,
  2411. binary: details.binary || defaultDetails.binary,
  2412. nocache: details.nocache || defaultDetails.nocache,
  2413. revalidate: details.revalidate || defaultDetails.revalidate,
  2414. /* 对象使用深拷贝 */
  2415. context: Utils.deepClone(details.context || defaultDetails.context),
  2416. overrideMimeType:
  2417. details.overrideMimeType || defaultDetails.overrideMimeType,
  2418. anonymous: details.anonymous || defaultDetails.anonymous,
  2419. fetch: details.fetch || defaultDetails.fetch,
  2420. /* 对象使用深拷贝 */
  2421. fetchInit: Utils.deepClone(defaultDetails.fetchInit),
  2422. user: details.user || defaultDetails.user,
  2423. password: details.password || defaultDetails.password,
  2424. onabort(...args) {
  2425. HttpxCallBack.onAbort(details, resolve, args);
  2426. },
  2427. onerror(...args) {
  2428. HttpxCallBack.onError(details, resolve, args);
  2429. },
  2430. onloadstart(...args) {
  2431. HttpxCallBack.onLoadStart(details, args);
  2432. },
  2433. onprogress(...args) {
  2434. HttpxCallBack.onProgress(details, args);
  2435. },
  2436. onreadystatechange(...args) {
  2437. HttpxCallBack.onReadyStateChange(details, args);
  2438. },
  2439. ontimeout(...args) {
  2440. HttpxCallBack.onTimeout(details, resolve, args);
  2441. },
  2442. onload(...args) {
  2443. HttpxCallBack.onLoad(details, resolve, args);
  2444. },
  2445. };
  2446. if (typeof GM_Api.xmlHttpRequest !== "function") {
  2447. result.fetch = true;
  2448. }
  2449. if (typeof result.headers === "object") {
  2450. if (typeof details.headers === "object") {
  2451. OriginPrototype.Object.keys(details.headers).forEach(
  2452. (keyName, index) => {
  2453. if (
  2454. keyName in result.headers &&
  2455. details.headers[keyName] == null
  2456. ) {
  2457. /* 在默认的header中存在,且设置它新的值为空,那么就是默认的值 */
  2458. OriginPrototype.Reflect.deleteProperty(
  2459. result.headers,
  2460. keyName
  2461. );
  2462. } else {
  2463. result.headers[keyName] = details.headers[keyName];
  2464. }
  2465. }
  2466. );
  2467. } else {
  2468. /* details.headers为空 */
  2469. /* 不做处理 */
  2470. }
  2471. } else {
  2472. result.headers = details.headers;
  2473. }
  2474. if (typeof result.fetchInit === "object") {
  2475. /* 使用assign替换且添加 */
  2476. if (typeof details.fetchInit === "object") {
  2477. OriginPrototype.Object.keys(details.fetchInit).forEach(
  2478. (keyName, index) => {
  2479. if (
  2480. keyName in result.fetchInit &&
  2481. details.fetchInit[keyName] == null
  2482. ) {
  2483. /* 在默认的fetchInit中存在,且设置它新的值为空,那么就是默认的值 */
  2484. OriginPrototype.Reflect.deleteProperty(
  2485. result.fetchInit,
  2486. keyName
  2487. );
  2488. } else {
  2489. result.fetchInit[keyName] = details.fetchInit[keyName];
  2490. }
  2491. }
  2492. );
  2493. }
  2494. } else {
  2495. result.fetchInit = details.fetchInit;
  2496. }
  2497. return result;
  2498. },
  2499. /**
  2500. * 处理发送请求的details,去除值为undefined、空function的值
  2501. * @param {HttpxDetails} details
  2502. * @returns {HttpxDetails}
  2503. */
  2504. handle(details) {
  2505. OriginPrototype.Object.keys(details).forEach((keyName) => {
  2506. if (
  2507. details[keyName] == null ||
  2508. (details[keyName] instanceof Function &&
  2509. Utils.isNull(details[keyName]))
  2510. ) {
  2511. OriginPrototype.Reflect.deleteProperty(details, keyName);
  2512. return;
  2513. }
  2514. });
  2515. if (Utils.isNull(details.url)) {
  2516. throw new TypeError(`Utils.Httpx 参数 url不符合要求: ${details.url}`);
  2517. }
  2518. /* method值统一大写,兼容Via */
  2519. details.method = details.method.toUpperCase();
  2520. /* 判断是否是以http开头,否则主动加上origin */
  2521. try {
  2522. new URL(details.url);
  2523. } catch (error) {
  2524. if (details.url.startsWith("//")) {
  2525. details.url = globalThis.location.protocol + details.url;
  2526. } else if (details.url.startsWith("/")) {
  2527. details.url = globalThis.location.origin + details.url;
  2528. } else {
  2529. details.url = globalThis.location.origin + "/" + details.url;
  2530. }
  2531. }
  2532. return details;
  2533. },
  2534. /**
  2535. * 处理fetch的配置
  2536. * @param {HttpxDetails} details
  2537. */
  2538. handleFetchDetail(details) {
  2539. /**
  2540. * fetch的请求配置
  2541. * @type {RequestInit}
  2542. **/
  2543. let fetchRequestInit = {};
  2544. if (
  2545. (details.method === "GET" || details.method === "HEAD") &&
  2546. details.data != null
  2547. ) {
  2548. /* GET 或 HEAD 方法的请求不能包含 body 信息 */
  2549. OriginPrototype.Reflect.deleteProperty(details, "data");
  2550. }
  2551. /* 中止信号控制器 */
  2552. let abortController = new AbortController();
  2553. let signal = abortController.signal;
  2554. signal.onabort = () => {
  2555. details.onabort({
  2556. isFetch: true,
  2557. responseText: "",
  2558. response: null,
  2559. readyState: 4,
  2560. responseHeaders: "",
  2561. status: 0,
  2562. statusText: "",
  2563. error: "aborted",
  2564. });
  2565. };
  2566. fetchRequestInit.method = details.method ?? "GET";
  2567. fetchRequestInit.headers = details.headers;
  2568. fetchRequestInit.body = details.data;
  2569. fetchRequestInit.mode = "cors";
  2570. fetchRequestInit.credentials = "include";
  2571. fetchRequestInit.cache = "no-cache";
  2572. fetchRequestInit.redirect = "follow";
  2573. fetchRequestInit.referrerPolicy = "origin-when-cross-origin";
  2574. fetchRequestInit.signal = signal;
  2575. OriginPrototype.Object.assign(
  2576. fetchRequestInit,
  2577. details.fetchInit || {}
  2578. );
  2579. return {
  2580. fetchDetails: details,
  2581. fetchRequestInit: fetchRequestInit,
  2582. abortController: abortController,
  2583. };
  2584. },
  2585. };
  2586. const HttpxCallBack = {
  2587. /**
  2588. * onabort请求被取消-触发
  2589. * @param {HttpxDetails} details 配置
  2590. * @param {()=>void} resolve 回调
  2591. * @param {any[]} argumentsList 参数列表
  2592. */
  2593. onAbort(details, resolve, argumentsList) {
  2594. if ("onabort" in details) {
  2595. details.onabort.apply(this, argumentsList);
  2596. } else if ("onabort" in defaultDetails) {
  2597. defaultDetails.onabort.apply(this, argumentsList);
  2598. }
  2599. resolve({
  2600. status: false,
  2601. data: [...argumentsList],
  2602. msg: "请求被取消",
  2603. type: "onabort",
  2604. });
  2605. },
  2606.  
  2607. /**
  2608. * onerror请求异常-触发
  2609. * @param {HttpxDetails} details 配置
  2610. * @param {()=>void} resolve 回调
  2611. * @param {any[]} argumentsList 响应的参数列表
  2612. */
  2613. onError(details, resolve, argumentsList) {
  2614. if ("onerror" in details) {
  2615. details.onerror.apply(this, argumentsList);
  2616. } else if ("onerror" in defaultDetails) {
  2617. defaultDetails.onerror.apply(this, argumentsList);
  2618. }
  2619. let response = argumentsList;
  2620. if (response.length) {
  2621. response = response[0];
  2622. }
  2623. resolve({
  2624. status: false,
  2625. data: response,
  2626. details: details,
  2627. msg: "请求异常",
  2628. type: "onerror",
  2629. });
  2630. },
  2631. /**
  2632. * ontimeout请求超时-触发
  2633. * @param {HttpxDetails} details 配置
  2634. * @param {()=>void} resolve 回调
  2635. * @param {any[]} argumentsList 参数列表
  2636. */
  2637. onTimeout(details, resolve, argumentsList) {
  2638. if ("ontimeout" in details) {
  2639. details.ontimeout.apply(this, argumentsList);
  2640. } else if ("ontimeout" in defaultDetails) {
  2641. defaultDetails.ontimeout.apply(this, argumentsList);
  2642. }
  2643. resolve({
  2644. status: false,
  2645. data: [...argumentsList],
  2646. msg: "请求超时",
  2647. type: "ontimeout",
  2648. });
  2649. },
  2650.  
  2651. /**
  2652. * onloadstart请求开始-触发
  2653. * @param {HttpxDetails} details 配置
  2654. * @param {any[]} argumentsList 参数列表
  2655. */
  2656. onLoadStart(details, argumentsList) {
  2657. if ("onloadstart" in details) {
  2658. details.onloadstart.apply(this, argumentsList);
  2659. } else if ("onloadstart" in defaultDetails) {
  2660. defaultDetails.onloadstart.apply(this, argumentsList);
  2661. }
  2662. },
  2663. /**
  2664. * onload加载完毕-触发
  2665. * @param {HttpxDetails} details 请求的配置
  2666. * @param {()=>void} resolve 回调
  2667. * @param {...HttpxAsyncResultData[]} argumentsList 参数列表
  2668. */
  2669. onLoad(details, resolve, argumentsList) {
  2670. /* X浏览器会因为设置了responseType导致不返回responseText */
  2671. let response = argumentsList[0];
  2672. /* responseText为空,response不为空的情况 */
  2673. if (
  2674. Utils.isNull(response["responseText"]) &&
  2675. Utils.isNotNull(response["response"])
  2676. ) {
  2677. if (typeof response["response"] === "object") {
  2678. Utils.tryCatch().run(() => {
  2679. response["responseText"] = JSON.stringify(response["response"]);
  2680. });
  2681. } else {
  2682. response["responseText"] = response["response"];
  2683. }
  2684. }
  2685.  
  2686. /* response为空,responseText不为空的情况 */
  2687. if (
  2688. response["response"] == null &&
  2689. typeof response["responseText"] === "string" &&
  2690. response["responseText"].trim() !== ""
  2691. ) {
  2692. if (details.responseType === "json") {
  2693. response["response"] = Utils.toJSON(response["responseText"]);
  2694. } else if (details.responseType === "document") {
  2695. let parser = new DOMParser();
  2696. response["response"] = parser.parseFromString(
  2697. response["responseText"],
  2698. "text/html"
  2699. );
  2700. } else if (details.responseType === "arraybuffer") {
  2701. let encoder = new TextEncoder();
  2702. let arrayBuffer = encoder.encode(response["responseText"]);
  2703. response["response"] = arrayBuffer;
  2704. } else if (details.responseType === "blob") {
  2705. let encoder = new TextEncoder();
  2706. let arrayBuffer = encoder.encode(response["responseText"]);
  2707. response["response"] = new Blob([arrayBuffer]);
  2708. } else {
  2709. response["response"] = response["responseText"];
  2710. }
  2711. }
  2712. /* Stay扩展中没有finalUrl,对应的是responseURL */
  2713. if (response["finalUrl"] == null && response["responseURL"] != null) {
  2714. response["finalUrl"] = response["responseURL"];
  2715. }
  2716. /* 状态码2xx都是成功的 */
  2717. if (Math.floor(response.status / 100) === 2) {
  2718. resolve({
  2719. status: true,
  2720. data: response,
  2721. details: details,
  2722. msg: "请求完毕",
  2723. type: "onload",
  2724. });
  2725. } else {
  2726. HttpxCallBack.onError(details, resolve, argumentsList);
  2727. }
  2728. },
  2729. /**
  2730. * onprogress上传进度-触发
  2731. * @param {HttpxDetails} details 配置
  2732. * @param {any[]} argumentsList 参数列表
  2733. */
  2734. onProgress(details, argumentsList) {
  2735. if ("onprogress" in details) {
  2736. details.onprogress.apply(this, argumentsList);
  2737. } else if ("onprogress" in defaultDetails) {
  2738. defaultDetails.onprogress.apply(this, argumentsList);
  2739. }
  2740. },
  2741. /**
  2742. * onreadystatechange准备状态改变-触发
  2743. * @param {HttpxDetails} details 配置
  2744. * @param {any[]} argumentsList 参数列表
  2745. */
  2746. onReadyStateChange(details, argumentsList) {
  2747. if ("onreadystatechange" in details) {
  2748. details.onreadystatechange.apply(this, argumentsList);
  2749. } else if ("onreadystatechange" in defaultDetails) {
  2750. defaultDetails.onreadystatechange.apply(this, argumentsList);
  2751. }
  2752. },
  2753. };
  2754.  
  2755. const HttpxRequest = {
  2756. /**
  2757. * 发送请求
  2758. * @param {HttpxDetails} details
  2759. */
  2760. request(details) {
  2761. if (LOG_DETAILS) {
  2762. console.log("Httpx请求配置👇", details);
  2763. }
  2764. if (typeof HttpxRequestHook.beforeRequestCallBack === "function") {
  2765. let hookResult = HttpxRequestHook.beforeRequestCallBack(details);
  2766. if (typeof hookResult === "boolean" && !hookResult) {
  2767. return;
  2768. }
  2769. }
  2770. if (details.fetch) {
  2771. const { fetchDetails, fetchRequestInit, abortController } =
  2772. HttpxRequestDetails.handleFetchDetail(details);
  2773. this.fetch(fetchDetails, fetchRequestInit, abortController);
  2774. } else {
  2775. OriginPrototype.Reflect.deleteProperty(details, "fetchInit");
  2776. this.xmlHttpRequest(details);
  2777. }
  2778. },
  2779. /**
  2780. * 使用油猴函数GM_xmlhttpRequest发送请求
  2781. * @param {HttpxDetails} details
  2782. */
  2783. xmlHttpRequest(details) {
  2784. GM_Api.xmlHttpRequest(details);
  2785. },
  2786. /**
  2787. * 使用fetch发送请求
  2788. * @param {HttpxDetails} details
  2789. * @param {RequestInit} fetchRequestInit
  2790. * @param {AbortController} abortController
  2791. */
  2792. fetch(details, fetchRequestInit, abortController) {
  2793. fetch(details.url, fetchRequestInit)
  2794. .then(async (resp) => {
  2795. /**
  2796. * @type {HttpxAsyncResultData}
  2797. */
  2798. let httpxResponse = {
  2799. isFetch: true,
  2800. finalUrl: resp.url,
  2801. readyState: 4,
  2802. status: resp.status,
  2803. statusText: resp.statusText,
  2804. response: void 0,
  2805. responseFetchHeaders: resp.headers,
  2806. responseHeaders: "",
  2807. responseText: void 0,
  2808. responseType: details.responseType,
  2809. responseXML: void 0,
  2810. };
  2811. OriginPrototype.Object.assign(httpxResponse, details.context || {});
  2812.  
  2813. for (const [key, value] of resp.headers.entries()) {
  2814. httpxResponse.responseHeaders += `${key}: ${value}\n`;
  2815. }
  2816.  
  2817. /* 如果是流式传输,直接返回 */
  2818. if (
  2819. details.responseType === "stream" ||
  2820. (resp.headers.has("Content-Type") &&
  2821. resp.headers.get("Content-Type").includes("text/event-stream"))
  2822. ) {
  2823. httpxResponse["isStream"] = true;
  2824. httpxResponse.response = resp.body;
  2825. OriginPrototype.Reflect.deleteProperty(
  2826. httpxResponse,
  2827. "responseText"
  2828. );
  2829. OriginPrototype.Reflect.deleteProperty(
  2830. httpxResponse,
  2831. "responseXML"
  2832. );
  2833. details.onload(httpxResponse);
  2834. return;
  2835. }
  2836.  
  2837. /** 响应 */
  2838. let response = "";
  2839. /** 响应字符串 */
  2840. let responseText = "";
  2841. /** 响应xml文档 */
  2842. let responseXML = "";
  2843.  
  2844. let arrayBuffer = await resp.arrayBuffer;
  2845.  
  2846. let encoding = "utf-8";
  2847. if (resp.headers.has("Content-Type")) {
  2848. let charsetMatched = resp.headers
  2849. .get("Content-Type")
  2850. .match(/charset=(.+)/);
  2851. if (charsetMatched) {
  2852. encoding = charsetMatched[1];
  2853. }
  2854. }
  2855. let textDecoder = new TextDecoder(encoding);
  2856. responseText = textDecoder.decode(await resp.arrayBuffer());
  2857. response = responseText;
  2858.  
  2859. if (details.responseType === "arraybuffer") {
  2860. response = arrayBuffer;
  2861. } else if (details.responseType === "blob") {
  2862. response = new Blob([arrayBuffer]);
  2863. } else if (
  2864. details.responseType === "document" ||
  2865. details.responseType == null
  2866. ) {
  2867. let parser = new DOMParser();
  2868. response = parser.parseFromString(responseText, "text/html");
  2869. } else if (details.responseType === "json") {
  2870. response = Utils.toJSON(responseText);
  2871. }
  2872. let parser = new DOMParser();
  2873. responseXML = parser.parseFromString(responseText, "text/xml");
  2874.  
  2875. httpxResponse.response = response;
  2876. httpxResponse.responseText = responseText;
  2877. httpxResponse.responseXML = responseXML;
  2878.  
  2879. details.onload(httpxResponse);
  2880. })
  2881. .catch((err) => {
  2882. if (err.name === "AbortError") {
  2883. return;
  2884. }
  2885. details.onerror({
  2886. isFetch: true,
  2887. finalUrl: details.url,
  2888. readyState: 4,
  2889. status: 0,
  2890. statusText: "",
  2891. responseHeaders: "",
  2892. responseText: "",
  2893. error: err,
  2894. });
  2895. });
  2896. details.onloadstart({
  2897. isFetch: true,
  2898. finalUrl: details.url,
  2899. readyState: 1,
  2900. responseHeaders: "",
  2901. responseText: "",
  2902. status: 0,
  2903. statusText: "",
  2904. });
  2905. return {
  2906. abort() {
  2907. abortController.abort();
  2908. },
  2909. };
  2910. },
  2911. };
  2912.  
  2913. /**
  2914. * 覆盖当前配置
  2915. * @param {HttpxDetailsConfig} details
  2916. */
  2917. this.config = function (details = {}) {
  2918. if (
  2919. "logDetails" in details &&
  2920. typeof details["logDetails"] === "boolean"
  2921. ) {
  2922. LOG_DETAILS = details["logDetails"];
  2923. }
  2924.  
  2925. defaultDetails = Utils.assign(defaultDetails, details);
  2926. };
  2927.  
  2928. /**
  2929. * 修改xmlHttpRequest
  2930. * @param {Function} httpRequest 网络请求函数
  2931. */
  2932. this.setXMLHttpRequest = function (httpRequest) {
  2933. GM_Api.xmlHttpRequest = httpRequest;
  2934. };
  2935.  
  2936. /**
  2937. * GET 请求
  2938. * @param {...HttpxDetails|string} args
  2939. * @returns {Promise< HttpxAsyncResult >}
  2940. */
  2941. this.get = async function (...args) {
  2942. let details = {};
  2943. if (typeof args[0] === "string") {
  2944. details.url = args[0];
  2945. if (typeof args[1] === "object") {
  2946. details = args[1];
  2947. details.url = args[0];
  2948. }
  2949. } else {
  2950. details = args[0];
  2951. }
  2952. return new Promise((resolve) => {
  2953. let requestDetails = HttpxRequestDetails.getDetails(
  2954. "get",
  2955. resolve,
  2956. details
  2957. );
  2958. OriginPrototype.Reflect.deleteProperty(requestDetails, "onprogress");
  2959. requestDetails = HttpxRequestDetails.handle(requestDetails);
  2960. HttpxRequest.request(requestDetails);
  2961. });
  2962. };
  2963. /**
  2964. * POST 请求
  2965. * @param {...HttpxDetails|string} args
  2966. * @returns {Promise< HttpxAsyncResult >}
  2967. */
  2968. this.post = async function (...args) {
  2969. let details = {};
  2970. if (typeof args[0] === "string") {
  2971. details.url = args[0];
  2972. if (typeof args[1] === "object") {
  2973. details = args[1];
  2974. details.url = args[0];
  2975. }
  2976. } else {
  2977. details = args[0];
  2978. }
  2979. return new Promise((resolve) => {
  2980. let requestDetails = HttpxRequestDetails.getDetails(
  2981. "post",
  2982. resolve,
  2983. details
  2984. );
  2985. requestDetails = HttpxRequestDetails.handle(requestDetails);
  2986. HttpxRequest.request(requestDetails);
  2987. });
  2988. };
  2989. /**
  2990. * HEAD 请求
  2991. * @param {...HttpxDetails|string} args
  2992. * @returns {Promise< HttpxAsyncResult >}
  2993. */
  2994. this.head = async function (...args) {
  2995. let details = {};
  2996. if (typeof args[0] === "string") {
  2997. details.url = args[0];
  2998. if (typeof args[1] === "object") {
  2999. details = args[1];
  3000. details.url = args[0];
  3001. }
  3002. } else {
  3003. details = args[0];
  3004. }
  3005. return new Promise((resolve) => {
  3006. let requestDetails = HttpxRequestDetails.getDetails(
  3007. "head",
  3008. resolve,
  3009. details
  3010. );
  3011. OriginPrototype.Reflect.deleteProperty(requestDetails, "onprogress");
  3012. requestDetails = HttpxRequestDetails.handle(requestDetails);
  3013. HttpxRequest.request(requestDetails);
  3014. });
  3015. };
  3016.  
  3017. /**
  3018. * OPTIONS 请求
  3019. * @param {...HttpxDetails|string} args
  3020. * @returns {Promise< HttpxAsyncResult >}
  3021. */
  3022. this.options = async function (...args) {
  3023. let details = {};
  3024. if (typeof args[0] === "string") {
  3025. details.url = args[0];
  3026. if (typeof args[1] === "object") {
  3027. details = args[1];
  3028. details.url = args[0];
  3029. }
  3030. } else {
  3031. details = args[0];
  3032. }
  3033. return new Promise((resolve) => {
  3034. let requestDetails = HttpxRequestDetails.getDetails(
  3035. "options",
  3036. resolve,
  3037. details
  3038. );
  3039. OriginPrototype.Reflect.deleteProperty(requestDetails, "onprogress");
  3040. requestDetails = HttpxRequestDetails.handle(requestDetails);
  3041. HttpxRequest.request(requestDetails);
  3042. });
  3043. };
  3044.  
  3045. /**
  3046. * DELETE 请求
  3047. * @param {...HttpxDetails|string} args
  3048. * @returns {Promise< HttpxAsyncResult >}
  3049. */
  3050. this.delete = async function (...args) {
  3051. let details = {};
  3052. if (typeof args[0] === "string") {
  3053. details.url = args[0];
  3054. if (typeof args[1] === "object") {
  3055. details = args[1];
  3056. details.url = args[0];
  3057. }
  3058. } else {
  3059. details = args[0];
  3060. }
  3061. return new Promise((resolve) => {
  3062. let requestDetails = HttpxRequestDetails.getDetails(
  3063. "delete",
  3064. resolve,
  3065. details
  3066. );
  3067. OriginPrototype.Reflect.deleteProperty(requestDetails, "onprogress");
  3068. requestDetails = HttpxRequestDetails.handle(requestDetails);
  3069. HttpxRequest.request(requestDetails);
  3070. });
  3071. };
  3072.  
  3073. /**
  3074. * PUT 请求
  3075. * @param {...HttpxDetails|string} args
  3076. * @returns {Promise< HttpxAsyncResult >}
  3077. */
  3078. this.put = async function (...args) {
  3079. let details = {};
  3080. if (typeof args[0] === "string") {
  3081. details.url = args[0];
  3082. if (typeof args[1] === "object") {
  3083. details = args[1];
  3084. details.url = args[0];
  3085. }
  3086. } else {
  3087. details = args[0];
  3088. }
  3089. return new Promise((resolve) => {
  3090. let requestDetails = HttpxRequestDetails.getDetails(
  3091. "put",
  3092. resolve,
  3093. details
  3094. );
  3095. requestDetails = HttpxRequestDetails.handle(requestDetails);
  3096. HttpxRequest.request(requestDetails);
  3097. });
  3098. };
  3099. };
  3100.  
  3101. Utils.indexedDB = function (
  3102. dbName = "default_db",
  3103. storeName = "default_form",
  3104. dbVersion = 1
  3105. ) {
  3106. this.dbName = dbName;
  3107. /* websql的版本号,由于ios的问题,版本号的写法不一样 */
  3108. this.slqVersion = "1";
  3109. this.dbVersion = dbVersion;
  3110. this.storeName = storeName;
  3111. /* 监听IndexDB */
  3112. this.indexedDB =
  3113. window.indexedDB ||
  3114. window.mozIndexedDB ||
  3115. window.webkitIndexedDB ||
  3116. window.msIndexedDB;
  3117. if (!this.indexedDB) {
  3118. alert("很抱歉,您的浏览器不支持indexedDB");
  3119. }
  3120. /* 缓存数据库,避免同一个页面重复创建和销毁 */
  3121. this.db = {};
  3122. this.store = null;
  3123. this.errorCode = {
  3124. /* 错误码 */
  3125. success: {
  3126. code: 200,
  3127. msg: "操作成功",
  3128. },
  3129. error: {
  3130. code: 401,
  3131. msg: "操作失败",
  3132. },
  3133. open: { code: 91001, msg: "打开数据库失败" },
  3134. save: { code: 91002, msg: "保存数据失败" },
  3135. get: { code: 91003, msg: "获取数据失败" },
  3136. delete: { code: 91004, msg: "删除数据失败" },
  3137. deleteAll: { code: 91005, msg: "清空数据库失败" },
  3138. };
  3139. let that = this;
  3140. /**
  3141. * 创建 “表”
  3142. * @param {string} dbName 表名
  3143. * @returns
  3144. */
  3145. this.createStore = function (dbName) {
  3146. let txn, store;
  3147. if (that.indexedDB) {
  3148. /* 如果是支持IndexDB的 */
  3149. txn = that.db[dbName].transaction(that.storeName, "readwrite");
  3150. /* IndexDB的读写权限 */
  3151. store = txn.objectStore(that.storeName);
  3152. }
  3153. return store;
  3154. };
  3155. /**
  3156. * 打开数据库
  3157. * @param {function} callback 回调
  3158. * @param {string} dbName 数据库名
  3159. */
  3160. this.open = function (callback, dbName) {
  3161. /* 打开数据库 */
  3162. if (that.indexedDB) {
  3163. /* 如果支持IndexDB */
  3164. if (!that.db[dbName]) {
  3165. /* 如果缓存中没有,则进行数据库的创建或打开,提高效率 */
  3166. let request = that.indexedDB.open(dbName, that.dbVersion);
  3167. request.onerror = function (e) {
  3168. callback({
  3169. code: that.errorCode.open.code,
  3170. msg: that.errorCode.open.msg,
  3171. error: e,
  3172. });
  3173. };
  3174. request.onsuccess = function (e) {
  3175. if (!that.db[dbName]) {
  3176. that.db[dbName] = e.target.result;
  3177. }
  3178. let store = that.createStore(dbName);
  3179. callback(store);
  3180. };
  3181. request.onupgradeneeded = function (e) {
  3182. that.db[dbName] = e.target.result;
  3183. let store = that.db[dbName].createObjectStore(that.storeName, {
  3184. keyPath: "key",
  3185. });
  3186. store.transaction.oncomplete = function (event) {
  3187. callback(store);
  3188. };
  3189. };
  3190. } else {
  3191. /* 如果缓存中已经打开了数据库,就直接使用 */
  3192. let store = that.createStore(dbName);
  3193. callback(store);
  3194. }
  3195. }
  3196. };
  3197. /**
  3198. * 保存数据到数据库
  3199. * @param {string} key 数据key
  3200. * @param {any} value 数据值
  3201. * @returns {Promise< {
  3202. * code: number,
  3203. * msg: string,
  3204. * success: boolean
  3205. * }>}
  3206. */
  3207. this.save = async function (key, value) {
  3208. if (that.indexedDB) {
  3209. return new Promise((resolve, reject) => {
  3210. let dbName = that.dbName;
  3211. let inData = {
  3212. key: key,
  3213. value: value,
  3214. };
  3215. that.open(function (result) {
  3216. let error = result.hasOwnProperty("error");
  3217. if (error) {
  3218. resolve(result);
  3219. } else {
  3220. let request = result.put(inData);
  3221. request.onsuccess = function (e) {
  3222. /* 保存成功有success 字段 */
  3223. resolve({
  3224. code: that.errorCode.success.code,
  3225. msg: that.errorCode.success.msg,
  3226. success: true,
  3227. });
  3228. };
  3229. request.onerror = function (e) {
  3230. resolve({
  3231. code: that.errorCode.save.code,
  3232. msg: that.errorCode.save.msg,
  3233. error: e,
  3234. });
  3235. };
  3236. }
  3237. }, dbName);
  3238. });
  3239. }
  3240. };
  3241. /**
  3242. * 根据key获取值
  3243. * @param {string} key 数据key
  3244. * @returns {Promise< {
  3245. * code: number,
  3246. * msg: string,
  3247. * data: [...any],
  3248. * success: true
  3249. * }| {
  3250. * code: number,
  3251. * msg: string,
  3252. * error: Error,
  3253. * result: any,
  3254. * } >}
  3255. */
  3256. this.get = async function (key) {
  3257. return new Promise((resolve, reject) => {
  3258. let dbName = that.dbName;
  3259. if (that.indexedDB) {
  3260. that.open(function (result) {
  3261. /* 判断返回的数据中是否有error字段 */
  3262. let error = result.hasOwnProperty("error");
  3263. if (error) {
  3264. reject({
  3265. code: that.errorCode.open.get,
  3266. msg: that.errorCode.get.msg,
  3267. error: error,
  3268. result: result,
  3269. });
  3270. } else {
  3271. let request = result.get(key);
  3272. request.onsuccess = function (e) {
  3273. let result = e.target.result;
  3274. let data = result ? result.value : void 0;
  3275. resolve({
  3276. code: data
  3277. ? that.errorCode.success.code
  3278. : that.errorCode.error.code,
  3279. msg: data
  3280. ? that.errorCode.success.msg
  3281. : that.errorCode.error.msg,
  3282. data: data || [],
  3283. success: true,
  3284. });
  3285. };
  3286. request.onerror = function (e) {
  3287. reject({
  3288. code: that.errorCode.get.code,
  3289. msg: that.errorCode.get.msg,
  3290. result: result,
  3291. error: e,
  3292. });
  3293. };
  3294. }
  3295. }, dbName);
  3296. }
  3297. });
  3298. };
  3299. /**
  3300. * 正则获取数据
  3301. * @param {string} key 数据键
  3302. * @returns { Promise<{
  3303. * code: number,
  3304. * msg: string,
  3305. * data: [...any],
  3306. * success: true
  3307. * }|{
  3308. * code: number,
  3309. * msg: string,
  3310. * error: Error,
  3311. * result: any,
  3312. * }> }
  3313. */
  3314. this.regexpGet = async function (key) {
  3315. let list = [];
  3316. return new Promise((resolve, reject) => {
  3317. /* 正则查询 */
  3318. let dbName = that.dbName;
  3319. if (that.indexedDB) {
  3320. that.open(function (result) {
  3321. /* 判断返回的数据中是否有error字段 */
  3322. let error = result.hasOwnProperty("error");
  3323. if (error) {
  3324. reject({
  3325. code: that.errorCode.open.get,
  3326. msg: that.errorCode.get.msg,
  3327. error: error,
  3328. result: result,
  3329. });
  3330. } else {
  3331. let request = result.getAll();
  3332. request.onsuccess = function (e) {
  3333. let result = e.target.result;
  3334. if (result.length !== 0) {
  3335. result.forEach((item, index) => {
  3336. if (item["key"].match(key)) {
  3337. let concatList = item["value"];
  3338. concatList["key"] = item["key"];
  3339. list = [...list, concatList];
  3340. }
  3341. });
  3342. }
  3343. resolve({
  3344. code: that.errorCode.success.code,
  3345. msg: that.errorCode.success.msg,
  3346. data: list,
  3347. success: true,
  3348. });
  3349. };
  3350. request.onerror = function (e) {
  3351. reject({
  3352. code: that.errorCode.get.code,
  3353. msg: that.errorCode.get.msg,
  3354. result: result,
  3355. error: e,
  3356. });
  3357. };
  3358. }
  3359. }, dbName);
  3360. }
  3361. });
  3362. };
  3363. /**
  3364. * 删除数据
  3365. * @param {string} key 数据键
  3366. * @returns {Promise<{
  3367. * code: number,
  3368. * msg: string,
  3369. * success: true,
  3370. * }|{
  3371. * code: number,
  3372. * msg: string,
  3373. * error: Error,
  3374. * }>}
  3375. */
  3376. this.delete = async function (key) {
  3377. return new Promise((resolve, reject) => {
  3378. /* 根据key删除某条数据 */
  3379. let dbName = that.dbName;
  3380. if (that.indexedDB) {
  3381. that.open(function (result) {
  3382. let error = result.hasOwnProperty("error");
  3383. if (error) {
  3384. resolve(result);
  3385. } else {
  3386. let request = result.get(key);
  3387. request.onsuccess = function (e) {
  3388. let recode = e.target.result;
  3389. if (recode) {
  3390. request = result.delete(key);
  3391. }
  3392. resolve({
  3393. code: recode
  3394. ? that.errorCode.success.code
  3395. : that.errorCode.error.code,
  3396. msg: recode
  3397. ? that.errorCode.success.msg
  3398. : that.errorCode.error.msg,
  3399. success: true,
  3400. });
  3401. };
  3402. request.onerror = function (e) {
  3403. resolve({
  3404. code: that.errorCode.delete.code,
  3405. msg: that.errorCode.delete.msg,
  3406. error: e,
  3407. });
  3408. };
  3409. }
  3410. }, dbName);
  3411. }
  3412. });
  3413. };
  3414. /**
  3415. * 删除所有数据
  3416. * @returns {Promise<{
  3417. * code: number,
  3418. * msg: string,
  3419. * error: Error,
  3420. * result: any,
  3421. * }|{
  3422. * code: number,
  3423. * msg: string,
  3424. * success: true,
  3425. * }>}
  3426. */
  3427. this.deleteAll = async function () {
  3428. return new Promise((resolve, reject) => {
  3429. /* 清空数据库 */
  3430. let dbName = that.dbName;
  3431. if (that.indexedDB) {
  3432. that.open(function (result) {
  3433. let error = result.hasOwnProperty("error");
  3434. if (error) {
  3435. resolve({
  3436. code: that.errorCode.deleteAll.code,
  3437. msg: that.errorCode.deleteAll.msg,
  3438. error: error,
  3439. result: result,
  3440. });
  3441. } else {
  3442. result.clear();
  3443. resolve({
  3444. code: that.errorCode.success.code,
  3445. msg: that.errorCode.success.msg,
  3446. success: true,
  3447. });
  3448. }
  3449. }, dbName);
  3450. }
  3451. });
  3452. };
  3453. };
  3454.  
  3455. Utils.isDOM = function (target) {
  3456. return target instanceof Node;
  3457. };
  3458.  
  3459. Utils.isFullscreenEnabled = function () {
  3460. return !!(
  3461. document.fullscreenEnabled ||
  3462. document.webkitFullScreenEnabled ||
  3463. document.mozFullScreenEnabled ||
  3464. document.msFullScreenEnabled
  3465. );
  3466. };
  3467.  
  3468. Utils.isJQuery = function (target) {
  3469. let result = false;
  3470. if (typeof jQuery === "object" && target instanceof jQuery) {
  3471. result = true;
  3472. }
  3473. if (target == null) {
  3474. return false;
  3475. }
  3476. if (typeof target === "object") {
  3477. /* 也有种可能,这个jQuery对象是1.8.3版本的,页面中的jQuery是3.4.1版本的 */
  3478. let jQueryProps = [
  3479. "add",
  3480. "addBack",
  3481. "addClass",
  3482. "after",
  3483. "ajaxComplete",
  3484. "ajaxError",
  3485. "ajaxSend",
  3486. "ajaxStart",
  3487. "ajaxStop",
  3488. "ajaxSuccess",
  3489. "animate",
  3490. "append",
  3491. "appendTo",
  3492. "attr",
  3493. "before",
  3494. "bind",
  3495. "blur",
  3496. "change",
  3497. "children",
  3498. "clearQueue",
  3499. "click",
  3500. "clone",
  3501. "closest",
  3502. "constructor",
  3503. "contents",
  3504. "contextmenu",
  3505. "css",
  3506. "data",
  3507. "dblclick",
  3508. "delay",
  3509. "delegate",
  3510. "dequeue",
  3511. "each",
  3512. "empty",
  3513. "end",
  3514. "eq",
  3515. "extend",
  3516. "fadeIn",
  3517. "fadeOut",
  3518. "fadeTo",
  3519. "fadeToggle",
  3520. "filter",
  3521. "find",
  3522. "first",
  3523. "focus",
  3524. "focusin",
  3525. "focusout",
  3526. "get",
  3527. "has",
  3528. "hasClass",
  3529. "height",
  3530. "hide",
  3531. "hover",
  3532. "html",
  3533. "index",
  3534. "init",
  3535. "innerHeight",
  3536. "innerWidth",
  3537. "insertAfter",
  3538. "insertBefore",
  3539. "is",
  3540. "jquery",
  3541. "keydown",
  3542. "keypress",
  3543. "keyup",
  3544. "last",
  3545. "load",
  3546. "map",
  3547. "mousedown",
  3548. "mouseenter",
  3549. "mouseleave",
  3550. "mousemove",
  3551. "mouseout",
  3552. "mouseover",
  3553. "mouseup",
  3554. "next",
  3555. "nextAll",
  3556. "not",
  3557. "off",
  3558. "offset",
  3559. "offsetParent",
  3560. "on",
  3561. "one",
  3562. "outerHeight",
  3563. "outerWidth",
  3564. "parent",
  3565. "parents",
  3566. "position",
  3567. "prepend",
  3568. "prependTo",
  3569. "prev",
  3570. "prevAll",
  3571. "prevUntil",
  3572. "promise",
  3573. "prop",
  3574. "pushStack",
  3575. "queue",
  3576. "ready",
  3577. "remove",
  3578. "removeAttr",
  3579. "removeClass",
  3580. "removeData",
  3581. "removeProp",
  3582. "replaceAll",
  3583. "replaceWith",
  3584. "resize",
  3585. "scroll",
  3586. "scrollLeft",
  3587. "scrollTop",
  3588. "select",
  3589. "show",
  3590. "siblings",
  3591. "slice",
  3592. "slideDown",
  3593. "slideToggle",
  3594. "slideUp",
  3595. "sort",
  3596. "splice",
  3597. "text",
  3598. "toArray",
  3599. "toggle",
  3600. "toggleClass",
  3601. "trigger",
  3602. "triggerHandler",
  3603. "unbind",
  3604. "width",
  3605. "wrap",
  3606. ];
  3607. for (const jQueryPropsName of jQueryProps) {
  3608. if (!(jQueryPropsName in target)) {
  3609. result = false;
  3610. /* console.log(jQueryPropsName); */
  3611. break;
  3612. } else {
  3613. result = true;
  3614. }
  3615. }
  3616. }
  3617. return result;
  3618. };
  3619.  
  3620. Utils.isNativeFunc = function (target) {
  3621. return Boolean(
  3622. target.toString().match(/^function .*\(\) { \[native code\] }$/)
  3623. );
  3624. };
  3625.  
  3626. Utils.isNearBottom = function (nearValue = 50) {
  3627. var scrollTop = window.pageYOffset || document.documentElement.scrollTop;
  3628. var windowHeight =
  3629. window.innerHeight || document.documentElement.clientHeight;
  3630. var documentHeight = document.documentElement.scrollHeight;
  3631. return scrollTop + windowHeight >= documentHeight - nearValue;
  3632. };
  3633.  
  3634. Utils.isNotNull = function () {
  3635. return !Utils.isNull.apply(this, arguments);
  3636. };
  3637.  
  3638. Utils.isNull = function () {
  3639. let result = true;
  3640. let checkList = [...arguments];
  3641. for (const objItem of checkList) {
  3642. let itemResult = false;
  3643. switch (typeof objItem) {
  3644. case "undefined":
  3645. case "null":
  3646. itemResult = true;
  3647. break;
  3648. case "object":
  3649. /* object类型的也可能是null */
  3650. if (objItem == null) {
  3651. itemResult = true;
  3652. } else if (typeof objItem[Symbol.iterator] === "function") {
  3653. itemResult = objItem.length === 0;
  3654. } else {
  3655. itemResult = OriginPrototype.Object.keys(objItem).length === 0;
  3656. }
  3657. break;
  3658. case "number":
  3659. itemResult = objItem === 0;
  3660. break;
  3661. case "string":
  3662. itemResult =
  3663. objItem.trim() === "" ||
  3664. objItem === "null" ||
  3665. objItem === "undefined";
  3666. break;
  3667. case "boolean":
  3668. itemResult = !objItem;
  3669. break;
  3670. case "function":
  3671. let funcStr = objItem.toString().replace(/\s/g, "");
  3672. /* 排除()=>{}、(xxx="")=>{}、function(){}、function(xxx=""){} */
  3673. itemResult = Boolean(
  3674. funcStr.match(/^\(.*?\)=>\{\}$|^function.*?\(.*?\)\{\}$/)
  3675. );
  3676. break;
  3677. }
  3678. result = result && itemResult;
  3679. }
  3680.  
  3681. return result;
  3682. };
  3683.  
  3684. Utils.isPhone = function (userAgent = navigator.userAgent) {
  3685. return Boolean(/(iPhone|iPad|iPod|iOS|Android|Mobile)/i.test(userAgent));
  3686. };
  3687.  
  3688. Utils.isSameChars = function (str) {
  3689. if (typeof str !== "string") {
  3690. throw new TypeError("参数 str 必须是 string 类型");
  3691. }
  3692. if (str.length < 2) {
  3693. return false;
  3694. }
  3695. str = str.toLowerCase();
  3696. const charCount = {};
  3697. for (const char of str) {
  3698. if (char in charCount) {
  3699. charCount[char]++;
  3700. } else {
  3701. charCount[char] = 1;
  3702. }
  3703. }
  3704. if (OriginPrototype.Object.keys(charCount).length === 1) {
  3705. return true;
  3706. } else {
  3707. return false;
  3708. }
  3709. };
  3710.  
  3711. Utils.isThemeDark = function () {
  3712. return globalThis.matchMedia("(prefers-color-scheme: dark)").matches;
  3713. };
  3714.  
  3715. Utils.isVisible = function (element, inView = false) {
  3716. let needCheckDomList = [];
  3717. if (element instanceof Array || element instanceof NodeList) {
  3718. needCheckDomList = [...element];
  3719. } else {
  3720. needCheckDomList = [element];
  3721. }
  3722. let result = true;
  3723. for (const domItem of needCheckDomList) {
  3724. let domDisplay = window.getComputedStyle(domItem);
  3725. if (domDisplay.display === "none") {
  3726. result = false;
  3727. } else {
  3728. let domClientRect = domItem.getBoundingClientRect();
  3729. if (inView) {
  3730. let viewportWidth =
  3731. window.innerWidth || document.documentElement.clientWidth;
  3732. let viewportHeight =
  3733. window.innerHeight || document.documentElement.clientHeight;
  3734. result = !(
  3735. domClientRect.right < 0 ||
  3736. domClientRect.left > viewportWidth ||
  3737. domClientRect.bottom < 0 ||
  3738. domClientRect.top > viewportHeight
  3739. );
  3740. } else {
  3741. result = Boolean(domItem.getClientRects().length);
  3742. }
  3743. }
  3744. if (!result) {
  3745. /* 有一个不可见就退出循环 */
  3746. break;
  3747. }
  3748. }
  3749. return result;
  3750. };
  3751.  
  3752. Utils.isWebView_Via = function () {
  3753. let result = true;
  3754. if (typeof top.window.via === "object") {
  3755. for (const key in OriginPrototype.Object.values(top.window.via)) {
  3756. if (OriginPrototype.Object.hasOwnProperty.call(top.window.via, key)) {
  3757. let objValueFunc = top.window.via[key];
  3758. if (
  3759. typeof objValueFunc === "function" &&
  3760. Utils.isNativeFunc(objValueFunc)
  3761. ) {
  3762. result = true;
  3763. } else {
  3764. result = false;
  3765. break;
  3766. }
  3767. }
  3768. }
  3769. } else {
  3770. result = false;
  3771. }
  3772. return result;
  3773. };
  3774.  
  3775. Utils.isWebView_X = function () {
  3776. let result = true;
  3777. if (typeof top.window.mbrowser === "object") {
  3778. for (const key in OriginPrototype.Object.values(top.window.mbrowser)) {
  3779. if (
  3780. OriginPrototype.Object.hasOwnProperty.call(top.window.mbrowser, key)
  3781. ) {
  3782. let objValueFunc = top.window.mbrowser[key];
  3783. if (
  3784. typeof objValueFunc === "function" &&
  3785. Utils.isNativeFunc(objValueFunc)
  3786. ) {
  3787. result = true;
  3788. } else {
  3789. result = false;
  3790. break;
  3791. }
  3792. }
  3793. }
  3794. } else {
  3795. result = false;
  3796. }
  3797. return result;
  3798. };
  3799.  
  3800. Utils.parseObjectToArray = function (target) {
  3801. if (typeof target !== "object") {
  3802. throw new Error(
  3803. "Utils.parseObjectToArray 参数 target 必须为 object 类型"
  3804. );
  3805. }
  3806. let result = [];
  3807. OriginPrototype.Object.keys(target).forEach(function (keyName) {
  3808. result = result.concat(target[keyName]);
  3809. });
  3810. return result;
  3811. };
  3812.  
  3813. Utils.listenKeyboard = function (target, eventName = "keypress", callback) {
  3814. if (
  3815. typeof target !== "object" ||
  3816. (typeof target["addEventListener"] !== "function" &&
  3817. typeof target["removeEventListener"] !== "function")
  3818. ) {
  3819. throw new Error(
  3820. "Utils.listenKeyboard 参数 target 必须为 Window|HTMLElement 类型"
  3821. );
  3822. }
  3823. let keyEvent = function (event) {
  3824. let keyName = event.key || event.code;
  3825. let keyValue = event.charCode || event.keyCode || event.which;
  3826. let otherCodeList = [];
  3827. if (event.ctrlKey) {
  3828. otherCodeList.push("ctrl");
  3829. }
  3830. if (event.altKey) {
  3831. otherCodeList.push("alt");
  3832. }
  3833. if (event.metaKey) {
  3834. otherCodeList.push("meta");
  3835. }
  3836. if (event.shiftKey) {
  3837. otherCodeList.push("shift");
  3838. }
  3839. if (typeof callback === "function") {
  3840. callback(keyName, keyValue, otherCodeList, event);
  3841. }
  3842. };
  3843. target.addEventListener(eventName, keyEvent);
  3844. return {
  3845. removeListen() {
  3846. target.removeEventListener(eventName, keyEvent);
  3847. },
  3848. };
  3849. };
  3850.  
  3851. Utils.LockFunction = function (callback, context, delayTime = 0) {
  3852. let flag = false;
  3853. let that = this;
  3854. context = context || this;
  3855. /**
  3856. * 锁
  3857. */
  3858. this.lock = function () {
  3859. flag = true;
  3860. };
  3861. /**
  3862. * 解锁
  3863. */
  3864. this.unlock = function () {
  3865. OriginPrototype.setTimeout(() => {
  3866. flag = false;
  3867. }, delayTime);
  3868. };
  3869. /**
  3870. * 执行
  3871. */
  3872. this.run = async function (...args) {
  3873. if (flag) {
  3874. return;
  3875. }
  3876. that.lock();
  3877. await callback.apply(context, args);
  3878. that.unlock();
  3879. };
  3880. };
  3881.  
  3882. Utils.Log = function (
  3883. _GM_info_ = {
  3884. script: {
  3885. name: "Utils.Log",
  3886. },
  3887. },
  3888. console = globalThis.console
  3889. ) {
  3890. let msgColorDetails = [
  3891. "font-weight: bold; color: cornflowerblue",
  3892. "font-weight: bold; color: cornflowerblue",
  3893. "font-weight: bold; color: darkorange",
  3894. "font-weight: bold; color: cornflowerblue",
  3895. ];
  3896. /**
  3897. * @type {UtilsLogOptions}
  3898. */
  3899. let details = {
  3900. tag: true,
  3901. successColor: "#0000FF",
  3902. errorColor: "#FF0000",
  3903. infoColor: "0",
  3904. warnColor: "0",
  3905. debug: false,
  3906. autoClearConsole: false,
  3907. logMaxCount: 999,
  3908. };
  3909. let logCount = 0;
  3910. /**
  3911. * 解析Error的堆栈获取实际调用者的函数名及函数所在的位置
  3912. * @param {string[]} stack
  3913. * @returns {{
  3914. * name: string,
  3915. * position: string,
  3916. * }}
  3917. */
  3918. let parseErrorStack = function (stack) {
  3919. let result = {
  3920. name: "",
  3921. position: "",
  3922. };
  3923. for (let stackString of stack) {
  3924. stackString = stackString.trim();
  3925. let stackFunctionName = stackString.match(/^at[\s]+(.+?)[\s]+/i);
  3926. let stackFunctionNamePosition = stackString.match(
  3927. /^at[\s]+.+[\s]+\((.+?)\)/i
  3928. );
  3929. if (stackFunctionName == null) {
  3930. continue;
  3931. }
  3932. stackFunctionName = stackFunctionName[stackFunctionName.length - 1];
  3933. stackFunctionNamePosition =
  3934. stackFunctionNamePosition[stackFunctionNamePosition.length - 1];
  3935. if (
  3936. stackFunctionName === "" ||
  3937. stackFunctionName.match(
  3938. new RegExp(
  3939. "(^Utils.Log.|.<anonymous>$|^Function.each|^NodeList.forEach|^k.fn.init.each)",
  3940. "g"
  3941. )
  3942. )
  3943. ) {
  3944. continue;
  3945. } else {
  3946. result.name = stackFunctionName;
  3947. result.position = stackFunctionNamePosition;
  3948. break;
  3949. }
  3950. }
  3951. if (result.position === "") {
  3952. let lastStackString = stack[stack.length - 1].trim();
  3953. if (lastStackString.startsWith("at chrome-extension://")) {
  3954. let lastStackMatch = lastStackString.match(/^at[\s]+(.+)/);
  3955. if (lastStackMatch) {
  3956. result.position = lastStackMatch[lastStackMatch.length - 1];
  3957. }
  3958. }
  3959. }
  3960. if (result.position === "") {
  3961. result.position = stack[stack.length - 1]
  3962. .trim()
  3963. .replace(/^at[\s]*/g, "");
  3964. }
  3965. return result;
  3966. };
  3967. /**
  3968. * 待恢复的函数或对象
  3969. */
  3970. let recoveryList = [];
  3971. /**
  3972. * 检测清理控制台
  3973. * @this {Utils.Log}
  3974. */
  3975. let checkClearConsole = function () {
  3976. logCount++;
  3977. if (details.autoClearConsole && logCount > details.logMaxCount) {
  3978. console.clear();
  3979. logCount = 0;
  3980. }
  3981. };
  3982. /**
  3983. * 输出内容
  3984. * @param {any} msg 需要输出的内容
  3985. * @param {string} color 颜色
  3986. * @param {string|undefined} otherStyle 其它CSS
  3987. * @this {Utils.Log}
  3988. */
  3989. let printContent = function (msg, color, otherStyle) {
  3990. checkClearConsole.apply(this);
  3991. otherStyle = otherStyle || "";
  3992. let stackSplit = new Error().stack.split("\n");
  3993. stackSplit.splice(0, 2);
  3994. let { name: callerName, position: callerPosition } =
  3995. parseErrorStack(stackSplit);
  3996. if (typeof msg === "object") {
  3997. /* 要输出的内容是个对象 */
  3998. if (details.tag) {
  3999. if (OriginPrototype.Array.isArray(msg) && msg.length < 5) {
  4000. console.log(
  4001. `%c[${this.tag}%c-%c${callerName}%c]%c `,
  4002. ...msgColorDetails,
  4003. `color: ${color};${otherStyle}`,
  4004. ...msg
  4005. );
  4006. } else {
  4007. console.log(
  4008. `%c[${this.tag}%c-%c${callerName}%c]%c `,
  4009. ...msgColorDetails,
  4010. `color: ${color};${otherStyle}`,
  4011. msg
  4012. );
  4013. }
  4014. } else {
  4015. if (OriginPrototype.Array.isArray(msg) && msg.length < 5) {
  4016. console.log(...msg);
  4017. } else {
  4018. console.log(msg);
  4019. }
  4020. }
  4021. } else {
  4022. if (details.tag) {
  4023. console.log(
  4024. `%c[${this.tag}%c-%c${callerName}%c]%c ${msg}`,
  4025. ...msgColorDetails,
  4026. `color: ${color};${otherStyle}`
  4027. );
  4028. } else {
  4029. console.log(`%c${msg}`, `color: ${color};${otherStyle}`);
  4030. }
  4031. }
  4032. if (details.debug) {
  4033. /* 如果开启调试模式,输出堆栈位置 */
  4034. console.log(callerPosition);
  4035. }
  4036. };
  4037. /**
  4038. * 前面的TAG标志
  4039. */
  4040. this.tag = _GM_info_?.script?.name || "Utils.Log";
  4041. /**
  4042. * 控制台-普通输出
  4043. * @param {any} msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  4044. * @param {string|undefined} color 输出的颜色
  4045. * @param {string|undefined} otherStyle 其它CSS
  4046. */
  4047. this.info = function (msg, color = details.infoColor, otherStyle) {
  4048. printContent.call(this, msg, color, otherStyle);
  4049. };
  4050. /**
  4051. * 控制台-警告输出
  4052. * @param {any} msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  4053. * @param {string|undefined} color 输出的颜色
  4054. * @param {string|undefined} otherStyle 其它CSS
  4055. */
  4056. this.warn = function (
  4057. msg,
  4058. color = details.warnColor,
  4059. otherStyle = "background: #FEF6D5;padding: 4px 6px 4px 0px;"
  4060. ) {
  4061. printContent.call(this, msg, color, otherStyle);
  4062. };
  4063. /**
  4064. * 控制台-错误输出
  4065. * @param {any} msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  4066. * @param {string|undefined} color 输出的颜色
  4067. * @param {string|undefined} otherStyle 其它CSS
  4068. */
  4069. this.error = function (msg, color = details.errorColor, otherStyle) {
  4070. printContent.call(this, msg, color, otherStyle);
  4071. };
  4072. /**
  4073. * 控制台-成功输出
  4074. * @param {any} msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  4075. * @param {string|undefined} color 输出的颜色
  4076. * @param {string|undefined} otherStyle 其它CSS
  4077. */
  4078. this.success = function (msg, color = details.successColor, otherStyle) {
  4079. printContent.call(this, msg, color, otherStyle);
  4080. };
  4081. /**
  4082. * 控制台-输出表格
  4083. * @param {object[]} msg
  4084. * @param {string|undefined} color 输出的颜色
  4085. * @param {string|undefined} otherStyle 其它CSS
  4086. * @example
  4087. * log.table([{"名字":"example","值":"123"},{"名字":"example2","值":"345"}])
  4088. */
  4089. this.table = function (msg, color = details.infoColor, otherStyle = "") {
  4090. checkClearConsole.apply(this);
  4091. let stack = new Error().stack.split("\n");
  4092. stack.splice(0, 1);
  4093. let errorStackParse = parseErrorStack(stack);
  4094. let stackFunctionName = errorStackParse.name;
  4095. let stackFunctionNamePosition = errorStackParse.position;
  4096. let callerName = stackFunctionName;
  4097. console.log(
  4098. `%c[${this.tag}%c-%c${callerName}%c]%c`,
  4099. ...msgColorDetails,
  4100. `color: ${color};${otherStyle}`
  4101. );
  4102. console.table(msg);
  4103. if (details.debug) {
  4104. console.log(stackFunctionNamePosition);
  4105. }
  4106. };
  4107. /**
  4108. * 配置Log对象的颜色
  4109. * @param {UtilsLogOptions} paramDetails 配置信息
  4110. */
  4111. this.config = function (paramDetails) {
  4112. details = OriginPrototype.Object.assign(details, paramDetails);
  4113. };
  4114. /**
  4115. * 禁用输出
  4116. */
  4117. this.disable = function () {
  4118. let that = this;
  4119. OriginPrototype.Object.keys(this)
  4120. .filter((keyName) => Boolean(keyName.match(/info|error|success|table/)))
  4121. .forEach((keyName) => {
  4122. let value = {};
  4123. value[keyName] = that[keyName];
  4124. recoveryList = [...recoveryList, value];
  4125. that[keyName] = () => {};
  4126. });
  4127. };
  4128. /**
  4129. * 恢复输出
  4130. */
  4131. this.recovery = function () {
  4132. let that = this;
  4133. recoveryList.forEach((item) => {
  4134. let keyName = OriginPrototype.Object.keys(item);
  4135. that[keyName] = item[keyName];
  4136. });
  4137. recoveryList = [];
  4138. };
  4139. };
  4140.  
  4141. Utils.mergeArrayToString = function (data, handleFunc) {
  4142. if (!(data instanceof Array)) {
  4143. throw new Error("Utils.mergeArrayToString 参数 data 必须为 Array 类型");
  4144. }
  4145. let content = "";
  4146. if (typeof handleFunc === "function") {
  4147. data.forEach((item) => {
  4148. content += handleFunc(item);
  4149. });
  4150. } else if (typeof handleFunc === "string") {
  4151. data.forEach((item) => {
  4152. content += item[handleFunc];
  4153. });
  4154. } else {
  4155. data.forEach((item) => {
  4156. OriginPrototype.Object.values(item)
  4157. .filter((item2) => typeof item2 === "string")
  4158. .forEach((item3) => {
  4159. content += item3;
  4160. });
  4161. });
  4162. }
  4163. return content;
  4164. };
  4165.  
  4166. Utils.mutationObserver = function (target, observer_config) {
  4167. if (
  4168. !(target instanceof Node) &&
  4169. !(target instanceof NodeList) &&
  4170. !Utils.isJQuery(target)
  4171. ) {
  4172. throw new Error(
  4173. "Utils.mutationObserver 参数 target 必须为 Node|NodeList|jQuery类型"
  4174. );
  4175. }
  4176.  
  4177. let default_obverser_config = {
  4178. /* 监听到元素有反馈,需执行的函数 */
  4179. callback: () => {},
  4180. config: {
  4181. /**
  4182. * @type {boolean|undefined}
  4183. * + true 监听以 target 为根节点的整个子树。包括子树中所有节点的属性,而不仅仅是针对 target
  4184. * + false (默认) 不生效
  4185. */
  4186. subtree: void 0,
  4187. /**
  4188. * @type {boolean|undefined}
  4189. * + true 监听 target 节点中发生的节点的新增与删除(同时,如果 subtree 为 true,会针对整个子树生效)
  4190. * + false (默认) 不生效
  4191. */
  4192. childList: void 0,
  4193. /**
  4194. * @type {boolean|undefined}
  4195. * + true 观察所有监听的节点属性值的变化。默认值为 true,当声明了 attributeFilter 或 attributeOldValue
  4196. * + false (默认) 不生效
  4197. */
  4198. attributes: void 0,
  4199. /**
  4200. * 一个用于声明哪些属性名会被监听的数组。如果不声明该属性,所有属性的变化都将触发通知
  4201. * @type {[...string]|undefined}
  4202. */
  4203. attributeFilter: void 0,
  4204. /**
  4205. * @type {boolean|undefined}
  4206. * + true 记录上一次被监听的节点的属性变化;可查阅 MutationObserver 中的 Monitoring attribute values 了解关于观察属性变化和属性值记录的详情
  4207. * + false (默认) 不生效
  4208. */
  4209. attributeOldValue: void 0,
  4210. /**
  4211. * @type {boolean|undefined}
  4212. * + true 监听声明的 target 节点上所有字符的变化。默认值为 true,如果声明了 characterDataOldValue
  4213. * + false (默认) 不生效
  4214. */
  4215. characterData: void 0,
  4216. /**
  4217. * @type {boolean|undefined}
  4218. * + true 记录前一个被监听的节点中发生的文本变化
  4219. * + false (默认) 不生效
  4220. */
  4221. characterDataOldValue: void 0,
  4222. },
  4223. };
  4224. observer_config = Utils.assign(default_obverser_config, observer_config);
  4225. let MutationObserver =
  4226. window.MutationObserver ||
  4227. window.webkitMutationObserver ||
  4228. window.MozMutationObserver;
  4229. /** @type {MutationObserver} */
  4230. let mutationObserver = new MutationObserver(function (mutations, observer) {
  4231. observer_config?.callback(mutations, observer);
  4232. });
  4233. if (target instanceof Node) {
  4234. /* 传入的参数是节点元素 */
  4235. mutationObserver.observe(target, observer_config.config);
  4236. } else if (target instanceof NodeList) {
  4237. /* 传入的参数是节点元素数组 */
  4238. target.forEach((item) => {
  4239. mutationObserver.observe(item, observer_config.config);
  4240. });
  4241. } else if (Utils.isJQuery(target)) {
  4242. /* 传入的参数是jQuery对象 */
  4243. target.each((index, item) => {
  4244. mutationObserver.observe(item, observer_config.config);
  4245. });
  4246. } else {
  4247. /* 未知 */
  4248. console.error("Utils.mutationObserver 未知参数", arguments);
  4249. }
  4250. return mutationObserver;
  4251. };
  4252.  
  4253. Utils.noConflict = function () {
  4254. if (window.Utils) {
  4255. OriginPrototype.Reflect.deleteProperty(window, "Utils");
  4256. }
  4257. if (AnotherUtils) {
  4258. window.Utils = AnotherUtils;
  4259. }
  4260. return Utils;
  4261. };
  4262.  
  4263. Utils.noConflictFunc = function (
  4264. needReleaseObject,
  4265. needReleaseName,
  4266. functionNameList = [],
  4267. release = true
  4268. ) {
  4269. if (typeof needReleaseObject !== "object") {
  4270. throw new Error(
  4271. "Utils.noConflictFunc 参数 needReleaseObject 必须为 object 类型"
  4272. );
  4273. }
  4274. if (typeof needReleaseName !== "string") {
  4275. throw new Error(
  4276. "Utils.noConflictFunc 参数 needReleaseName 必须为 string 类型"
  4277. );
  4278. }
  4279. if (!OriginPrototype.Array.isArray(functionNameList)) {
  4280. throw new Error(
  4281. "Utils.noConflictFunc 参数 functionNameList 必须为 Array 类型"
  4282. );
  4283. }
  4284. let needReleaseKey = "__" + needReleaseName;
  4285. /**
  4286. * 复制对象
  4287. * @param {object} obj
  4288. * @returns {object}
  4289. */
  4290. function cloneObj(obj) {
  4291. let newObj = {};
  4292. if (obj instanceof Array) {
  4293. newObj = [];
  4294. }
  4295. for (let key in obj) {
  4296. let val = obj[key];
  4297. newObj[key] = typeof val === "object" ? cloneObj(val) : val;
  4298. }
  4299. return newObj;
  4300. }
  4301. /**
  4302. * 释放所有
  4303. */
  4304. function releaseAll() {
  4305. if (typeof window[needReleaseKey] !== "undefined") {
  4306. /* 已存在 */
  4307. return;
  4308. }
  4309. window[needReleaseKey] = cloneObj(needReleaseObject);
  4310. OriginPrototype.Object.values(needReleaseObject).forEach((value) => {
  4311. if (typeof value === "function") {
  4312. needReleaseObject[value.name] = () => {};
  4313. }
  4314. });
  4315. }
  4316. /**
  4317. * 释放单个
  4318. */
  4319. function releaseOne() {
  4320. Array.from(functionNameList).forEach((item) => {
  4321. OriginPrototype.Object.values(needReleaseObject).forEach((value) => {
  4322. if (typeof value === "function") {
  4323. if (typeof window[needReleaseKey] === "undefined") {
  4324. window[needReleaseKey] = {};
  4325. }
  4326. if (item === value.name) {
  4327. window[needReleaseKey][value.name] =
  4328. needReleaseObject[value.name];
  4329. needReleaseObject[value.name] = () => {};
  4330. }
  4331. }
  4332. });
  4333. });
  4334. }
  4335. /**
  4336. * 恢复所有
  4337. */
  4338. function recoveryAll() {
  4339. if (typeof window[needReleaseKey] === "undefined") {
  4340. /* 未存在 */
  4341. return;
  4342. }
  4343. OriginPrototype.Object.assign(needReleaseObject, window[needReleaseKey]);
  4344. OriginPrototype.Reflect.deleteProperty(window, "needReleaseKey");
  4345. }
  4346.  
  4347. /**
  4348. * 恢复单个
  4349. */
  4350. function recoveryOne() {
  4351. if (typeof window[needReleaseKey] === "undefined") {
  4352. /* 未存在 */
  4353. return;
  4354. }
  4355. Array.from(functionNameList).forEach((item) => {
  4356. if (window[needReleaseKey][item]) {
  4357. needReleaseObject[item] = window[needReleaseKey][item];
  4358. OriginPrototype.Reflect.deleteProperty(window[needReleaseKey], item);
  4359. if (
  4360. OriginPrototype.Object.keys(window[needReleaseKey]).length === 0
  4361. ) {
  4362. OriginPrototype.Reflect.deleteProperty(window, needReleaseKey);
  4363. }
  4364. }
  4365. });
  4366. }
  4367. if (release) {
  4368. /* 释放 */
  4369. if (functionNameList.length === 0) {
  4370. releaseAll();
  4371. } else {
  4372. /* 对单个进行操作 */
  4373. releaseOne();
  4374. }
  4375. } else {
  4376. /* 恢复 */
  4377. if (functionNameList.length === 0) {
  4378. recoveryAll();
  4379. } else {
  4380. /* 对单个进行操作 */
  4381. recoveryOne();
  4382. }
  4383. }
  4384. };
  4385.  
  4386. Utils.parseBase64ToBlob = function (dataUri) {
  4387. if (typeof dataUri !== "string") {
  4388. throw new Error(
  4389. "Utils.parseBase64ToBlob 参数 dataUri 必须为 string 类型"
  4390. );
  4391. }
  4392. let dataUriSplit = dataUri.split(","),
  4393. dataUriMime = dataUriSplit[0].match(/:(.*?);/)[1],
  4394. dataUriBase64Str = atob(dataUriSplit[1]),
  4395. dataUriLength = dataUriBase64Str.length,
  4396. u8arr = new Uint8Array(dataUriLength);
  4397. while (dataUriLength--) {
  4398. u8arr[dataUriLength] = dataUriBase64Str.charCodeAt(dataUriLength);
  4399. }
  4400. return new Blob([u8arr], {
  4401. type: dataUriMime,
  4402. });
  4403. };
  4404.  
  4405. Utils.parseBase64ToFile = function (dataUri, fileName = "example") {
  4406. if (typeof dataUri !== "string") {
  4407. throw new Error(
  4408. "Utils.parseBase64ToFile 参数 dataUri 必须为 string 类型"
  4409. );
  4410. }
  4411. if (typeof fileName !== "string") {
  4412. throw new Error(
  4413. "Utils.parseBase64ToFile 参数 fileName 必须为 string 类型"
  4414. );
  4415. }
  4416. let dataUriSplit = dataUri.split(","),
  4417. dataUriMime = dataUriSplit[0].match(/:(.*?);/)[1],
  4418. dataUriBase64Str = atob(dataUriSplit[1]),
  4419. dataUriLength = dataUriBase64Str.length,
  4420. u8arr = new Uint8Array(dataUriLength);
  4421. while (dataUriLength--) {
  4422. u8arr[dataUriLength] = dataUriBase64Str.charCodeAt(dataUriLength);
  4423. }
  4424. return new File([u8arr], fileName, {
  4425. type: dataUriMime,
  4426. });
  4427. };
  4428.  
  4429. Utils.parseInt = function (matchList = [], defaultValue = 0) {
  4430. if (matchList == null) {
  4431. return parseInt(defaultValue);
  4432. }
  4433. let parseValue = parseInt(matchList[matchList.length - 1]);
  4434. if (isNaN(parseValue)) {
  4435. parseValue = parseInt(defaultValue);
  4436. }
  4437. return parseValue;
  4438. };
  4439.  
  4440. Utils.parseBlobToFile = async function (blobUrl, fileName = "example") {
  4441. return new Promise((resolve, reject) => {
  4442. fetch(blobUrl)
  4443. .then((response) => response.blob())
  4444. .then((blob) => {
  4445. const file = new File([blob], fileName, { type: blob.type });
  4446. resolve(file);
  4447. })
  4448. .catch((error) => {
  4449. console.error("Error:", error);
  4450. reject(error);
  4451. });
  4452. });
  4453. };
  4454.  
  4455. Utils.parseCDATA = function (text = "") {
  4456. let result = "";
  4457. let cdataRegexp = /<\!\[CDATA\[([\s\S]*)\]\]>/;
  4458. let cdataMatch = cdataRegexp.exec(text.trim());
  4459. if (cdataMatch && cdataMatch.length > 1) {
  4460. result = cdataMatch[cdataMatch.length - 1];
  4461. }
  4462. return result;
  4463. };
  4464.  
  4465. Utils.parseFileToBase64 = async function (fileObj) {
  4466. let reader = new FileReader();
  4467. reader.readAsDataURL(fileObj);
  4468. return new Promise((resolve) => {
  4469. reader.onload = function (event) {
  4470. resolve(event.target.result);
  4471. };
  4472. });
  4473. };
  4474.  
  4475. Utils.parseFromString = function (text, mimeType = "text/html") {
  4476. let parser = new DOMParser();
  4477. return parser.parseFromString(text, mimeType);
  4478. };
  4479.  
  4480. Utils.preventEvent = function (element, eventNameList = []) {
  4481. function stopEvent(event) {
  4482. /* 阻止事件的默认行为发生。例如,当点击一个链接时,浏览器会默认打开链接的URL */
  4483. event?.preventDefault();
  4484. /* 停止事件的传播,阻止它继续向更上层的元素冒泡,事件将不会再传播给其他的元素 */
  4485. event?.stopPropagation();
  4486. /* 阻止事件传播,并且还能阻止元素上的其他事件处理程序被触发 */
  4487. event?.stopImmediatePropagation();
  4488. return false;
  4489. }
  4490. if (arguments.length === 1) {
  4491. return stopEvent(arguments[0]);
  4492. } else if (arguments.length === 2) {
  4493. if (typeof eventNameList === "string") {
  4494. eventNameList = [eventNameList];
  4495. }
  4496. eventNameList.forEach((eventName) => {
  4497. element.addEventListener(eventName, function (event) {
  4498. return stopEvent(event);
  4499. });
  4500. });
  4501. }
  4502. };
  4503.  
  4504. Utils.Progress = function (paramConfig) {
  4505. this.config = {
  4506. /**
  4507. * canvas元素节点
  4508. * @type {HTMLCanvasElement}
  4509. */
  4510. canvasNode: null,
  4511. /**
  4512. * 绘制角度
  4513. */
  4514. deg: 95,
  4515. /**
  4516. * 进度
  4517. */
  4518. progress: 0,
  4519. /**
  4520. * 绘制的线宽度
  4521. */
  4522. lineWidth: 10,
  4523. /**
  4524. * 绘制的背景颜色
  4525. */
  4526. lineBgColor: "#1e637c",
  4527. /**
  4528. * 绘制的线的颜色
  4529. */
  4530. lineColor: "#25deff",
  4531. /**
  4532. * 绘制的字体颜色
  4533. */
  4534. textColor: "#000000",
  4535. /**
  4536. * 绘制的字体大小(px)
  4537. */
  4538. fontSize: 22,
  4539. /**
  4540. * 绘制的圆的半径
  4541. */
  4542. circleRadius: 50,
  4543. /**
  4544. * 控制绘制的函数
  4545. */
  4546. draw: () => {},
  4547. };
  4548. this.config = Utils.assign(this.config, paramConfig);
  4549. if (!(this.config.canvasNode instanceof HTMLCanvasElement)) {
  4550. throw new Error(
  4551. "Utils.Progress 参数 canvasNode 必须是 HTMLCanvasElement"
  4552. );
  4553. }
  4554. /* 获取画笔 */
  4555. let ctx = this.config.canvasNode.getContext("2d");
  4556. /* 元素宽度 */
  4557. let width = this.config.canvasNode.width;
  4558. /* 元素高度 */
  4559. let height = this.config.canvasNode.height;
  4560.  
  4561. /* 清除锯齿 */
  4562. if (window.devicePixelRatio) {
  4563. this.config.canvasNode.style.width = width + "px";
  4564. this.config.canvasNode.style.height = height + "px";
  4565. this.config.canvasNode.height = height * window.devicePixelRatio;
  4566. this.config.canvasNode.width = width * window.devicePixelRatio;
  4567. ctx.scale(window.devicePixelRatio, window.devicePixelRatio);
  4568. }
  4569. /* 设置线宽 */
  4570. ctx.lineWidth = this.config.lineWidth;
  4571. /* 绘制 */
  4572. this.draw = function () {
  4573. let degActive = (this.config.progress * 360) / 100;
  4574. /* 清除画布 */
  4575. ctx.clearRect(0, 0, width, height);
  4576. /* 开始绘制底圆 */
  4577. ctx.beginPath();
  4578. ctx.arc(width / 2, height / 2, this.config.circleRadius, 1, 8);
  4579. ctx.strokeStyle = this.config.lineBgColor;
  4580. ctx.stroke();
  4581. /* 开始绘制动态圆 */
  4582. ctx.beginPath();
  4583. ctx.arc(
  4584. width / 2,
  4585. height / 2,
  4586. this.config.circleRadius,
  4587. -Math.PI / 2,
  4588. (degActive * Math.PI) / 180 - Math.PI / 2
  4589. );
  4590. ctx.strokeStyle = this.config.lineColor;
  4591. ctx.stroke();
  4592. /* 获取百分比 */
  4593. let txt = parseInt(this.config.progress) + "%";
  4594. ctx.font = this.config.fontSize + "px SimHei";
  4595. /* 获取文本宽度 */
  4596. let w = ctx.measureText(txt).width;
  4597. let h = this.config.fontSize / 2;
  4598. ctx.fillStyle = this.config.textColor;
  4599. ctx.fillText(txt, width / 2 - w / 2, height / 2 + h / 2);
  4600. }.bind(this);
  4601. };
  4602.  
  4603. Utils.registerTrustClickEvent = function (isTrustValue = true, filter) {
  4604. function trustEvent(event) {
  4605. return new Proxy(event, {
  4606. get: function (target, property) {
  4607. if (property === "isTrusted") {
  4608. return isTrustValue;
  4609. } else {
  4610. return Reflect.get(target, property);
  4611. }
  4612. },
  4613. });
  4614. }
  4615. if (filter == null) {
  4616. filter = function (typeName) {
  4617. return typeName === "click";
  4618. };
  4619. }
  4620. const originalListener = EventTarget.prototype.addEventListener;
  4621. EventTarget.prototype.addEventListener = function (...args) {
  4622. let type = args[0];
  4623. let callback = args[1];
  4624. let options = args[2];
  4625. if (filter(type)) {
  4626. if (typeof callback === "function") {
  4627. args[1] = function (event) {
  4628. callback.call(this, trustEvent(event));
  4629. };
  4630. } else if (typeof callback === "object" && "handleEvent" in callback) {
  4631. let oldHandleEvent = callback["handleEvent"];
  4632.  
  4633. args[1]["handleEvent"] = function (event) {
  4634. if (event == null) {
  4635. return;
  4636. }
  4637. try {
  4638. /* Proxy对象使用instanceof会报错 */
  4639. event instanceof Proxy;
  4640. oldHandleEvent.call(this, trustEvent(event));
  4641. } catch (error) {
  4642. event["isTrusted"] = isTrustValue;
  4643. }
  4644. };
  4645. }
  4646. }
  4647. return originalListener.apply(this, args);
  4648. };
  4649. };
  4650.  
  4651. Utils.reverseNumber = function (num) {
  4652. let reversedNum = 0;
  4653. let isNegative = false;
  4654.  
  4655. if (num < 0) {
  4656. isNegative = true;
  4657. num = Math.abs(num);
  4658. }
  4659.  
  4660. while (num > 0) {
  4661. reversedNum = reversedNum * 10 + (num % 10);
  4662. num = Math.floor(num / 10);
  4663. }
  4664.  
  4665. return isNegative ? -reversedNum : reversedNum;
  4666. };
  4667.  
  4668. Utils.selectElementText = function (
  4669. element,
  4670. childTextNode,
  4671. startIndex,
  4672. endIndex
  4673. ) {
  4674. let range = document.createRange();
  4675. range.selectNodeContents(element);
  4676. if (childTextNode) {
  4677. if (childTextNode.nodeType !== Node.TEXT_NODE) {
  4678. throw new TypeError("childTextNode必须是#text元素");
  4679. }
  4680. if (startIndex != null && endIndex != null) {
  4681. range.setStart(childTextNode, startIndex);
  4682. range.setEnd(childTextNode, endIndex);
  4683. }
  4684. }
  4685.  
  4686. let selection = globalThis.getSelection();
  4687. selection.removeAllRanges();
  4688. selection.addRange(range);
  4689. };
  4690.  
  4691. Utils.setClip = async function (data, info = "text/plain") {
  4692. if (typeof data === "object") {
  4693. if (data instanceof Element) {
  4694. data = data.outerHTML;
  4695. } else {
  4696. data = JSON.stringify(data);
  4697. }
  4698. } else if (typeof data !== "string") {
  4699. data = data.toString();
  4700. }
  4701. let textType = typeof info === "object" ? info.type : info;
  4702. if (textType.includes("html")) {
  4703. textType = "text/html";
  4704. } else {
  4705. textType = "text/plain";
  4706. }
  4707. let textBlob = new Blob([data], { type: textType });
  4708. let clipboardObject = navigator.clipboard;
  4709. /**
  4710. * 第二种方式进行复制
  4711. * @param {(value: any) => void} _resolve_ 回调
  4712. * @param {string} _copyText_ 复制的文字
  4713. */
  4714. function anotherCopy(_resolve_, _copyText_) {
  4715. let copyElement = document.createElement("textarea");
  4716. copyElement.value = _copyText_;
  4717. copyElement.setAttribute("type", "text");
  4718. copyElement.setAttribute("style", "opacity:0;position:absolute;");
  4719. copyElement.setAttribute("readonly", "readonly");
  4720. document.body.appendChild(copyElement);
  4721. copyElement.select();
  4722. document.execCommand("copy");
  4723. document.body.removeChild(copyElement);
  4724. _resolve_();
  4725. }
  4726. /**
  4727. * 运行复制
  4728. * @param {(value: any) => void} _resolve_ 回调
  4729. */
  4730. function runCopy(_resolve_) {
  4731. if (typeof ClipboardItem === "undefined") {
  4732. console.error("当前环境中不存在ClipboardItem对象,使用第二种方式");
  4733. anotherCopy(_resolve_, data);
  4734. } else if (clipboardObject) {
  4735. clipboardObject
  4736. .write([
  4737. new ClipboardItem({
  4738. [textType]: textBlob,
  4739. }),
  4740. ])
  4741. .then(() => {
  4742. _resolve_();
  4743. })
  4744. .catch((err) => {
  4745. console.error("复制失败,使用第二种方式,error👉", err);
  4746. anotherCopy(_resolve_, data);
  4747. });
  4748. } else {
  4749. anotherCopy(_resolve_, data);
  4750. }
  4751. }
  4752. return new Promise((resolve) => {
  4753. if (document.hasFocus()) {
  4754. runCopy(resolve);
  4755. } else {
  4756. window.addEventListener(
  4757. "focus",
  4758. () => {
  4759. runCopy(resolve);
  4760. },
  4761. { once: true }
  4762. );
  4763. }
  4764. });
  4765. };
  4766.  
  4767. Utils.setTimeout = async function (callback, delayTime = 0) {
  4768. if (typeof callback !== "function" && typeof callback !== "string") {
  4769. throw new TypeError(
  4770. "Utils.setTimeout 参数 callback 必须为 function|string 类型"
  4771. );
  4772. }
  4773. if (typeof delayTime !== "number") {
  4774. throw new TypeError("Utils.setTimeout 参数 delayTime 必须为 number 类型");
  4775. }
  4776. return new Promise((resolve) => {
  4777. OriginPrototype.setTimeout(() => {
  4778. resolve(Utils.tryCatch().run(callback));
  4779. }, delayTime);
  4780. });
  4781. };
  4782.  
  4783. Utils.sleep = async function (delayTime = 0) {
  4784. if (typeof delayTime !== "number") {
  4785. throw new Error("Utils.sleep 参数 delayTime 必须为 number 类型");
  4786. }
  4787. return new Promise((resolve) => {
  4788. OriginPrototype.setTimeout(() => {
  4789. resolve();
  4790. }, delayTime);
  4791. });
  4792. };
  4793.  
  4794. Utils.dragSlider = function (selector, offsetX = window.innerWidth) {
  4795. function initMouseEvent(eventName, offSetX, offSetY) {
  4796. let win = unsafeWindow || window;
  4797. let mouseEvent = document.createEvent("MouseEvents");
  4798. mouseEvent.initMouseEvent(
  4799. eventName,
  4800. true,
  4801. true,
  4802. win,
  4803. 0,
  4804. offSetX,
  4805. offSetY,
  4806. offSetX,
  4807. offSetY,
  4808. false,
  4809. false,
  4810. false,
  4811. false,
  4812. 0,
  4813. null
  4814. );
  4815. return mouseEvent;
  4816. }
  4817. let sliderElement =
  4818. typeof selector === "string"
  4819. ? document.querySelector(selector)
  4820. : selector;
  4821. if (
  4822. !(sliderElement instanceof Node) ||
  4823. !(sliderElement instanceof Element)
  4824. ) {
  4825. throw new Error("Utils.dragSlider 参数selector 必须为Node/Element类型");
  4826. }
  4827. let rect = sliderElement.getBoundingClientRect(),
  4828. x0 = rect.x || rect.left,
  4829. y0 = rect.y || rect.top,
  4830. x1 = x0 + offsetX,
  4831. y1 = y0;
  4832. sliderElement.dispatchEvent(initMouseEvent("mousedown", x0, y0));
  4833. sliderElement.dispatchEvent(initMouseEvent("mousemove", x1, y1));
  4834. sliderElement.dispatchEvent(initMouseEvent("mouseleave", x1, y1));
  4835. sliderElement.dispatchEvent(initMouseEvent("mouseout", x1, y1));
  4836. };
  4837.  
  4838. Utils.enterFullScreen = function (
  4839. element = document.documentElement,
  4840. options
  4841. ) {
  4842. try {
  4843. if (element.requestFullscreen) {
  4844. element.requestFullscreen(options);
  4845. } else if (element.webkitRequestFullScreen) {
  4846. element.webkitRequestFullScreen();
  4847. } else if (element.mozRequestFullScreen) {
  4848. element.mozRequestFullScreen();
  4849. } else if (element.msRequestFullscreen) {
  4850. element.msRequestFullscreen();
  4851. } else {
  4852. throw new TypeError("该浏览器不支持全屏API");
  4853. }
  4854. } catch (err) {
  4855. console.error(err);
  4856. }
  4857. };
  4858.  
  4859. Utils.exitFullScreen = function (element = document.documentElement) {
  4860. if (document.exitFullscreen) {
  4861. return document.exitFullscreen();
  4862. } else if (document.msExitFullscreen) {
  4863. return document.msExitFullscreen();
  4864. } else if (document.mozCancelFullScreen) {
  4865. return document.mozCancelFullScreen();
  4866. } else if (document.webkitCancelFullScreen) {
  4867. return document.webkitCancelFullScreen();
  4868. } else {
  4869. return new Promise((resolve, reject) => {
  4870. reject(new TypeError("该浏览器不支持全屏API"));
  4871. });
  4872. }
  4873. };
  4874.  
  4875. Utils.sortListByProperty = function (
  4876. data,
  4877. getPropertyValueFunc,
  4878. sortByDesc = true
  4879. ) {
  4880. if (
  4881. typeof getPropertyValueFunc !== "function" &&
  4882. typeof getPropertyValueFunc !== "string"
  4883. ) {
  4884. throw new Error(
  4885. "Utils.sortListByProperty 参数 getPropertyValueFunc 必须为 function|string 类型"
  4886. );
  4887. }
  4888. if (typeof sortByDesc !== "boolean") {
  4889. throw new Error(
  4890. "Utils.sortListByProperty 参数 sortByDesc 必须为 boolean 类型"
  4891. );
  4892. }
  4893. let getObjValue = function (obj) {
  4894. return typeof getPropertyValueFunc === "string"
  4895. ? obj[getPropertyValueFunc]
  4896. : getPropertyValueFunc(obj);
  4897. };
  4898. /**
  4899. * 排序方法
  4900. * @param {any} after_obj
  4901. * @param {any} before_obj
  4902. * @returns
  4903. */
  4904. let sortFunc = function (after_obj, before_obj) {
  4905. let beforeValue = getObjValue(before_obj); /* 前 */
  4906. let afterValue = getObjValue(after_obj); /* 后 */
  4907. if (sortByDesc) {
  4908. if (afterValue > beforeValue) {
  4909. return -1;
  4910. } else if (afterValue < beforeValue) {
  4911. return 1;
  4912. } else {
  4913. return 0;
  4914. }
  4915. } else {
  4916. if (afterValue < beforeValue) {
  4917. return -1;
  4918. } else if (afterValue > beforeValue) {
  4919. return 1;
  4920. } else {
  4921. return 0;
  4922. }
  4923. }
  4924. };
  4925. /**
  4926. * 排序元素方法
  4927. * @param {NodeList|jQuery} nodeList 元素列表
  4928. * @param {function} getNodeListFunc 获取元素列表的函数
  4929. */
  4930. let sortNodeFunc = function (nodeList, getNodeListFunc) {
  4931. let nodeListLength = nodeList.length;
  4932. for (let i = 0; i < nodeListLength - 1; i++) {
  4933. for (let j = 0; j < nodeListLength - 1 - i; j++) {
  4934. let beforeNode = nodeList[j];
  4935. let afterNode = nodeList[j + 1];
  4936. let beforeValue = getObjValue(beforeNode); /* 前 */
  4937. let afterValue = getObjValue(afterNode); /* 后 */
  4938. if (
  4939. (sortByDesc == true && beforeValue < afterValue) ||
  4940. (sortByDesc == false && beforeValue > afterValue)
  4941. ) {
  4942. /* 升序/降序 */
  4943. /* 相邻元素两两对比 */
  4944. let temp = beforeNode.nextElementSibling;
  4945. afterNode.after(beforeNode);
  4946. if (temp == null) {
  4947. /* 如果为空,那么是最后一个元素,使用append */
  4948. temp.parentNode.appendChild(afterNode);
  4949. } else {
  4950. /* 不为空,使用before */
  4951. temp.before(afterNode);
  4952. }
  4953. nodeList = getNodeListFunc();
  4954. }
  4955. }
  4956. }
  4957. };
  4958. let result = data;
  4959. let getDataFunc = null;
  4960. if (data instanceof Function) {
  4961. getDataFunc = data;
  4962. data = data();
  4963. }
  4964. if (OriginPrototype.Array.isArray(data)) {
  4965. data.sort(sortFunc);
  4966. } else if (data instanceof NodeList || Utils.isJQuery(data)) {
  4967. sortNodeFunc(data, getDataFunc);
  4968. result = getDataFunc();
  4969. } else {
  4970. throw new Error(
  4971. "Utils.sortListByProperty 参数 data 必须为 Array|NodeList|jQuery 类型"
  4972. );
  4973. }
  4974. return result;
  4975. };
  4976.  
  4977. Utils.stringToRegular = function (targetString, flags = "ig") {
  4978. let reg;
  4979. flags = flags.toLowerCase();
  4980. if (typeof targetString === "string") {
  4981. reg = new RegExp(
  4982. targetString.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&"),
  4983. flags
  4984. );
  4985. } else if (targetString instanceof RegExp) {
  4986. reg = targetString;
  4987. } else {
  4988. throw new Error(
  4989. "Utils.stringToRegular 参数targetString必须是string|Regexp类型"
  4990. );
  4991. }
  4992. return reg;
  4993. };
  4994.  
  4995. Utils.stringTitleToUpperCase = function (
  4996. targetString,
  4997. otherStrToLowerCase = false
  4998. ) {
  4999. let newTargetString = targetString.slice(0, 1).toUpperCase();
  5000. if (otherStrToLowerCase) {
  5001. newTargetString = newTargetString + targetString.slice(1).toLowerCase();
  5002. } else {
  5003. newTargetString = newTargetString + targetString.slice(1);
  5004. }
  5005. return newTargetString;
  5006. };
  5007.  
  5008. Utils.startsWith = function (target, searchString, position = 0) {
  5009. if (position > target.length) {
  5010. /* 超出目标字符串的长度 */
  5011. return false;
  5012. }
  5013. if (position !== 0) {
  5014. target = target.slice(position, target.length);
  5015. }
  5016. let searchStringRegexp = searchString;
  5017. if (typeof searchString === "string") {
  5018. searchStringRegexp = new RegExp(`^${searchString}`);
  5019. } else if (OriginPrototype.Array.isArray(searchString)) {
  5020. let flag = false;
  5021. for (const searcStr of searchString) {
  5022. if (!Utils.startsWith(target, searcStr, position)) {
  5023. flag = true;
  5024. break;
  5025. }
  5026. }
  5027. return flag;
  5028. }
  5029. return Boolean(target.match(searchStringRegexp));
  5030. };
  5031.  
  5032. Utils.stringTitleToLowerCase = function (
  5033. targetString,
  5034. otherStrToUpperCase = false
  5035. ) {
  5036. let newTargetString = targetString.slice(0, 1).toLowerCase();
  5037. if (otherStrToUpperCase) {
  5038. newTargetString = newTargetString + targetString.slice(1).toUpperCase();
  5039. } else {
  5040. newTargetString = newTargetString + targetString.slice(1);
  5041. }
  5042. return newTargetString;
  5043. };
  5044.  
  5045. Utils.toJSON = function (data, errorCallBack = () => {}) {
  5046. let result = {};
  5047. if (typeof data === "object") {
  5048. return data;
  5049. }
  5050. Utils.tryCatch()
  5051. .config({ log: false })
  5052. .error((error) => {
  5053. Utils.tryCatch()
  5054. .error(() => {
  5055. try {
  5056. result = window.eval("(" + data + ")");
  5057. } catch (error) {
  5058. errorCallBack(error);
  5059. }
  5060. })
  5061. .run(() => {
  5062. if (
  5063. data &&
  5064. /^[\],:{}\s]*$/.test(
  5065. data
  5066. .replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, "@")
  5067. .replace(
  5068. /"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g,
  5069. "]"
  5070. )
  5071. .replace(/(?:^|:|,)(?:\s*\[)+/g, "")
  5072. )
  5073. ) {
  5074. result = new Function("return " + data)();
  5075. } else {
  5076. errorCallBack(new Error("target is not a JSON"));
  5077. }
  5078. });
  5079. })
  5080. .run(() => {
  5081. data = data.trim();
  5082. result = JSON.parse(data);
  5083. });
  5084. return result;
  5085. };
  5086.  
  5087. Utils.toSearchParamsStr = function (obj) {
  5088. let searhParamsStr = "";
  5089. if (OriginPrototype.Array.isArray(obj)) {
  5090. obj.forEach((item) => {
  5091. if (searhParamsStr === "") {
  5092. searhParamsStr += Utils.toSearchParamsStr(item);
  5093. } else {
  5094. searhParamsStr += "&" + Utils.toSearchParamsStr(item);
  5095. }
  5096. });
  5097. } else {
  5098. searhParamsStr = new URLSearchParams(
  5099. OriginPrototype.Object.entries(obj)
  5100. ).toString();
  5101. }
  5102. return searhParamsStr;
  5103. };
  5104.  
  5105. Utils.tryCatch = function (...args) {
  5106. /* 定义变量和函数 */
  5107. let callbackFunction = null;
  5108. let context = null;
  5109. let handleError = null;
  5110. let defaultDetails = {
  5111. log: true,
  5112. };
  5113. /**
  5114. * @function tryCatchObj
  5115. * @description 空函数,用于链式调用。
  5116. */
  5117. function tryCatchObj() {}
  5118.  
  5119. /**
  5120. * 配置
  5121. * @param {{
  5122. * log: boolean
  5123. * }} paramDetails
  5124. */
  5125. tryCatchObj.config = function (paramDetails) {
  5126. defaultDetails = Utils.assign(defaultDetails, paramDetails);
  5127. return tryCatchObj;
  5128. };
  5129. /**
  5130. * 设置错误处理函数。
  5131. * @param {function|string} handler 错误处理函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  5132. * @returns {function} 返回 tryCatchObj 函数。
  5133. */
  5134. tryCatchObj.error = function (handler) {
  5135. handleError = handler;
  5136. return tryCatchObj;
  5137. };
  5138.  
  5139. /**
  5140. * 执行传入的函数并捕获其可能抛出的错误,并通过传入的错误处理函数进行处理。
  5141. * @param {function|string} callback 待执行函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  5142. * @param {object|null} __context__ 待执行函数的作用域,用于apply指定
  5143. * @returns {any|function} 如果函数有返回值,则返回该返回值;否则返回 tryCatchObj 函数以支持链式调用。
  5144. * @throws {Error} 如果传入参数不符合要求,则会抛出相应类型的错误。
  5145. */
  5146. tryCatchObj.run = function (callback, __context__) {
  5147. callbackFunction = callback;
  5148. context = __context__;
  5149. let result = executeTryCatch(callbackFunction, handleError, context);
  5150. return result !== void 0 ? result : tryCatchObj;
  5151. };
  5152.  
  5153. /**
  5154. * 执行传入的函数并捕获其可能抛出的错误,并通过传入的错误处理函数进行处理。
  5155. * @param {function|string} callback - 待执行函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  5156. * @param {function|string|null} handleErrorFunc - 错误处理函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  5157. * @param {object|null} funcThis - 待执行函数的作用域,用于apply指定
  5158. * @returns {any|undefined} - 如果函数有返回值,则返回该返回值;否则返回 undefined。
  5159. */
  5160. function executeTryCatch(callback, handleErrorFunc, funcThis) {
  5161. let result = void 0;
  5162. try {
  5163. if (typeof callback === "string") {
  5164. (function () {
  5165. eval(callback);
  5166. }).apply(funcThis, args);
  5167. } else {
  5168. result = callback.apply(funcThis, args);
  5169. }
  5170. } catch (error) {
  5171. if (defaultDetails.log) {
  5172. console.log(
  5173. `%c ${callback?.name ? callback?.name : callback + "出现错误"} `,
  5174. "color: #f20000"
  5175. );
  5176. console.log(`%c 错误原因:${error}`, "color: #f20000");
  5177. console.trace(callback);
  5178. }
  5179. if (handleErrorFunc) {
  5180. if (typeof handleErrorFunc === "string") {
  5181. result = function () {
  5182. return eval(handleErrorFunc);
  5183. }.apply(funcThis, [...args, error]);
  5184. } else {
  5185. result = handleErrorFunc.apply(funcThis, [...args, error]);
  5186. }
  5187. }
  5188. }
  5189. return result;
  5190. }
  5191.  
  5192. // 返回 tryCatchObj 函数
  5193. return tryCatchObj;
  5194. };
  5195.  
  5196. Utils.uniqueArray = function (
  5197. uniqueArrayData = [],
  5198. compareArrayData = [],
  5199. compareFun = (item, item2) => {
  5200. return item === item2;
  5201. }
  5202. ) {
  5203. return Array.from(uniqueArrayData).filter(
  5204. (item) =>
  5205. !Array.from(compareArrayData).some(function (item2) {
  5206. return compareFun(item, item2);
  5207. })
  5208. );
  5209. };
  5210.  
  5211. Utils.waitArrayLoopToEnd = function (data, handleFunc) {
  5212. if (typeof handleFunc !== "function" && typeof handleFunc !== "string") {
  5213. throw new Error(
  5214. "Utils.waitArrayLoopToEnd 参数 handleDataFunction 必须为 function|string 类型"
  5215. );
  5216. }
  5217. return Promise.all(
  5218. Array.from(data).map(async (item, index) => {
  5219. await Utils.tryCatch(index, item).run(handleFunc);
  5220. })
  5221. );
  5222. };
  5223.  
  5224. Utils.waitNode = async function (...nodeSelectors) {
  5225. /* 检查每个参数是否为字符串类型 */
  5226. for (let nodeSelector of nodeSelectors) {
  5227. if (typeof nodeSelector !== "string") {
  5228. throw new Error("Utils.waitNode 参数必须都是 string 类型");
  5229. }
  5230. }
  5231.  
  5232. return new Promise((resolve) => {
  5233. /* 防止触发第二次回调 */
  5234. let isReturn = false;
  5235.  
  5236. /**
  5237. * 检查所有选择器是否匹配到节点
  5238. * @param {MutationObserver} observer
  5239. */
  5240. let checkNodes = (observer) => {
  5241. let isFind = true;
  5242. let selectNodeArray = [];
  5243. for (let selector of nodeSelectors) {
  5244. let element = document.querySelector(selector);
  5245. if (!element) {
  5246. /* 没找到,直接退出循环 */
  5247. isFind = false;
  5248. break;
  5249. }
  5250. selectNodeArray.push(element);
  5251. }
  5252. if (isFind) {
  5253. isReturn = true;
  5254. observer?.disconnect();
  5255. /* 如果只有一个选择器,那么返回数组中存储的第一个 */
  5256. if (selectNodeArray.length === 1) {
  5257. resolve(selectNodeArray[0]);
  5258. } else {
  5259. resolve(selectNodeArray);
  5260. }
  5261. }
  5262. };
  5263.  
  5264. /* 在函数开始时检查节点是否已经存在 */
  5265. checkNodes();
  5266.  
  5267. /* 监听 DOM 的变化,直到至少有一个节点被匹配到 */
  5268. Utils.mutationObserver(document.documentElement, {
  5269. config: { subtree: true, childList: true, attributes: true },
  5270. callback: (mutations, observer) => {
  5271. if (isReturn) {
  5272. return;
  5273. }
  5274. checkNodes(observer);
  5275. },
  5276. });
  5277. });
  5278. };
  5279.  
  5280. Utils.waitNodeWithInterval = async function (
  5281. nodeSelectorsList = [],
  5282. maxTime = 0
  5283. ) {
  5284. /** @type {string[]} */
  5285. let nodeSelectors = [];
  5286. /* 检查每个参数是否为字符串类型 */
  5287. if (OriginPrototype.Array.isArray(nodeSelectorsList)) {
  5288. for (let nodeSelector of nodeSelectorsList) {
  5289. if (typeof nodeSelector !== "string") {
  5290. throw new Error(
  5291. "Utils.waitNodeWithInterval 参数nodeSelectorsList必须为 string[] 类型"
  5292. );
  5293. }
  5294. }
  5295. nodeSelectors = nodeSelectorsList;
  5296. } else {
  5297. nodeSelectors.push(nodeSelectorsList);
  5298. }
  5299.  
  5300. return new Promise((resolve) => {
  5301. /* 防止触发第二次回调 */
  5302. let isReturn = false;
  5303.  
  5304. /* 检查所有选择器是否匹配到节点 */
  5305. let checkNodes = (observer) => {
  5306. let isFind = true;
  5307. let selectNodeArray = [];
  5308. for (let selector of nodeSelectors) {
  5309. let element = document.querySelector(selector);
  5310. if (!element) {
  5311. /* 没找到,直接退出循环 */
  5312. isFind = false;
  5313. break;
  5314. }
  5315. selectNodeArray.push(element);
  5316. }
  5317. if (isFind) {
  5318. isReturn = true;
  5319. observer?.disconnect();
  5320. /* 如果只有一个选择器,那么返回数组中存储的第一个 */
  5321. if (selectNodeArray.length === 1) {
  5322. resolve(selectNodeArray[0]);
  5323. } else {
  5324. resolve(selectNodeArray);
  5325. }
  5326. }
  5327. };
  5328.  
  5329. /* 在函数开始时检查节点是否已经存在 */
  5330. checkNodes();
  5331.  
  5332. /* 监听 DOM 的变化,直到至少有一个节点被匹配到 */
  5333. let mutationObserver = Utils.mutationObserver(document.documentElement, {
  5334. config: { subtree: true, childList: true, attributes: true },
  5335. callback: (mutations, observer) => {
  5336. if (isReturn) {
  5337. return;
  5338. }
  5339. checkNodes(observer);
  5340. },
  5341. });
  5342. OriginPrototype.setTimeout(() => {
  5343. mutationObserver.disconnect();
  5344. }, maxTime);
  5345. });
  5346. };
  5347.  
  5348. Utils.waitAnyNode = async function (...nodeSelectors) {
  5349. /* 检查每个参数是否为字符串类型 */
  5350. for (let nodeSelector of nodeSelectors) {
  5351. if (typeof nodeSelector !== "string") {
  5352. throw new Error("Utils.waitNode 参数必须为 ...string 类型");
  5353. }
  5354. }
  5355.  
  5356. return new Promise((resolve) => {
  5357. /* 防止触发第二次回调 */
  5358. let isReturn = false;
  5359.  
  5360. /* 检查所有选择器是否存在任意匹配到元素 */
  5361. let checkNodes = (observer) => {
  5362. let selectNode = null;
  5363. for (let selector of nodeSelectors) {
  5364. selectNode = document.querySelector(selector);
  5365. if (selectNode) {
  5366. /* 找到,退出循环 */
  5367. break;
  5368. }
  5369. }
  5370. if (selectNode) {
  5371. isReturn = true;
  5372. observer?.disconnect();
  5373. resolve(selectNode);
  5374. }
  5375. };
  5376.  
  5377. /* 在函数开始时检查节点是否已经存在 */
  5378. checkNodes();
  5379.  
  5380. /* 监听 DOM 的变化,直到至少有一个节点被匹配到 */
  5381. Utils.mutationObserver(document.documentElement, {
  5382. config: { subtree: true, childList: true, attributes: true },
  5383. callback: (mutations, observer) => {
  5384. if (isReturn) {
  5385. return;
  5386. }
  5387. checkNodes(observer);
  5388. },
  5389. });
  5390. });
  5391. };
  5392.  
  5393. Utils.waitNodeList = async function (...nodeSelectors) {
  5394. /* 检查每个参数是否为字符串类型 */
  5395. for (let nodeSelector of nodeSelectors) {
  5396. if (typeof nodeSelector !== "string") {
  5397. throw new Error("Utils.waitNode 参数必须为 ...string 类型");
  5398. }
  5399. }
  5400.  
  5401. return new Promise((resolve) => {
  5402. /* 防止触发第二次回调 */
  5403. let isReturn = false;
  5404.  
  5405. /* 检查所有选择器是否匹配到节点 */
  5406. let checkNodes = (observer) => {
  5407. let isFind = true;
  5408. let selectNodes = [];
  5409. for (let selector of nodeSelectors) {
  5410. let nodeList = document.querySelectorAll(selector);
  5411. if (nodeList.length === 0) {
  5412. /* 没找到,直接退出循环 */
  5413. isFind = false;
  5414. break;
  5415. }
  5416. selectNodes.push(nodeList);
  5417. }
  5418. if (isFind) {
  5419. isReturn = true;
  5420. observer?.disconnect();
  5421. /* 如果只有一个选择器,那么返回第一个 */
  5422. if (selectNodes.length === 1) {
  5423. resolve(selectNodes[0]);
  5424. } else {
  5425. resolve(selectNodes);
  5426. }
  5427. }
  5428. };
  5429.  
  5430. /* 在函数开始时检查节点是否已经存在 */
  5431. checkNodes();
  5432.  
  5433. /* 监听 DOM 的变化,直到至少有一个节点被匹配到 */
  5434. Utils.mutationObserver(document.documentElement, {
  5435. config: { subtree: true, childList: true, attributes: true },
  5436. callback: (mutations, observer) => {
  5437. if (isReturn) {
  5438. return;
  5439. }
  5440. checkNodes(observer);
  5441. },
  5442. });
  5443. });
  5444. };
  5445.  
  5446. Utils.waitAnyNodeList = async function (...nodeSelectors) {
  5447. /* 检查每个参数是否为字符串类型 */
  5448. for (let nodeSelector of nodeSelectors) {
  5449. if (typeof nodeSelector !== "string") {
  5450. throw new Error("Utils.waitNode 参数必须为 ...string 类型");
  5451. }
  5452. }
  5453.  
  5454. return new Promise((resolve) => {
  5455. /* 防止触发第二次回调 */
  5456. let isReturn = false;
  5457.  
  5458. /* 检查所有选择器是否匹配到节点 */
  5459. let checkNodes = (observer) => {
  5460. let selectNodes = [];
  5461. for (let selector of nodeSelectors) {
  5462. selectNodes = document.querySelectorAll(selector);
  5463. if (selectNodes.length) {
  5464. /* 找到,退出循环 */
  5465. break;
  5466. }
  5467. }
  5468. if (selectNodes.length) {
  5469. isReturn = true;
  5470. observer?.disconnect();
  5471. resolve(selectNodes);
  5472. }
  5473. };
  5474.  
  5475. /* 在函数开始时检查节点是否已经存在 */
  5476. checkNodes();
  5477.  
  5478. /* 监听 DOM 的变化,直到至少有一个节点被匹配到 */
  5479. Utils.mutationObserver(document.documentElement, {
  5480. config: { subtree: true, childList: true, attributes: true },
  5481. callback: (mutations, observer) => {
  5482. if (isReturn) {
  5483. return;
  5484. }
  5485. checkNodes(observer);
  5486. },
  5487. });
  5488. });
  5489. };
  5490.  
  5491. Utils.waitProperty = async function (checkObj, checkPropertyName) {
  5492. return new Promise((resolve) => {
  5493. let obj = checkObj;
  5494. if (typeof checkObj === "function") {
  5495. obj = checkObj();
  5496. }
  5497. if (OriginPrototype.Object.hasOwnProperty.call(obj, checkPropertyName)) {
  5498. resolve(obj[checkPropertyName]);
  5499. } else {
  5500. OriginPrototype.Object.defineProperty(obj, checkPropertyName, {
  5501. set: function (value) {
  5502. try {
  5503. resolve(value);
  5504. } catch (error) {
  5505. console.error("Error setting property:", error);
  5506. }
  5507. },
  5508. });
  5509. }
  5510. });
  5511. };
  5512.  
  5513. Utils.waitPropertyByInterval = async function (
  5514. checkObj,
  5515. checkPropertyName,
  5516. intervalTimer = 250,
  5517. maxTime = -1
  5518. ) {
  5519. if (checkObj == null) {
  5520. throw new TypeError("checkObj 不能为空对象 ");
  5521. }
  5522. let isResolve = false;
  5523. return new Promise((resolve) => {
  5524. let interval = OriginPrototype.setInterval(() => {
  5525. let obj = checkObj;
  5526. if (typeof checkObj === "function") {
  5527. obj = checkObj();
  5528. }
  5529. if (
  5530. (typeof checkPropertyName === "function" && checkPropertyName(obj)) ||
  5531. OriginPrototype.Object.hasOwnProperty.call(obj, checkPropertyName)
  5532. ) {
  5533. isResolve = true;
  5534. OriginPrototype.clearInterval(interval);
  5535. resolve(obj[checkPropertyName]);
  5536. }
  5537. }, intervalTimer);
  5538. if (maxTime !== -1) {
  5539. OriginPrototype.setTimeout(() => {
  5540. if (!isResolve) {
  5541. OriginPrototype.clearInterval(interval);
  5542. resolve();
  5543. }
  5544. }, maxTime);
  5545. }
  5546. });
  5547. };
  5548.  
  5549. Utils.waitVueByInterval = async function (
  5550. element,
  5551. propertyName,
  5552. timer = 250,
  5553. maxTime = -1,
  5554. vueName = "__vue__"
  5555. ) {
  5556. if (element == null) {
  5557. throw new Error("Utils.waitVueByInterval 参数element 不能为空");
  5558. }
  5559. let flag = false;
  5560. await Utils.waitPropertyByInterval(
  5561. element,
  5562. function (targetElement) {
  5563. if (targetElement == null) {
  5564. return false;
  5565. }
  5566. if (!(vueName in targetElement)) {
  5567. return false;
  5568. }
  5569. if (propertyName == null) {
  5570. return true;
  5571. }
  5572. let vueObject = targetElement[vueName];
  5573. if (typeof propertyName === "string") {
  5574. if (propertyName in vueObject) {
  5575. flag = true;
  5576. return true;
  5577. }
  5578. } else {
  5579. /* Function */
  5580. if (propertyName(vueObject)) {
  5581. flag = true;
  5582. return true;
  5583. }
  5584. }
  5585. return false;
  5586. },
  5587. timer,
  5588. maxTime
  5589. );
  5590. return flag;
  5591. };
  5592.  
  5593. Utils.watchObject = function (
  5594. target,
  5595. propertyName,
  5596. getCallBack,
  5597. setCallBack
  5598. ) {
  5599. if (
  5600. typeof getCallBack !== "function" &&
  5601. typeof setCallBack !== "function"
  5602. ) {
  5603. return;
  5604. }
  5605.  
  5606. if (typeof getCallBack === "function") {
  5607. OriginPrototype.Object.defineProperty(target, propertyName, {
  5608. get() {
  5609. if (typeof getCallBack === "function") {
  5610. return getCallBack(value);
  5611. } else {
  5612. return target[propertyName];
  5613. }
  5614. },
  5615. });
  5616. } else if (typeof setCallBack === "function") {
  5617. OriginPrototype.Object.defineProperty(target, propertyName, {
  5618. set(value) {
  5619. if (typeof setCallBack === "function") {
  5620. setCallBack(value);
  5621. }
  5622. },
  5623. });
  5624. } else {
  5625. OriginPrototype.Object.defineProperty(target, propertyName, {
  5626. get() {
  5627. if (typeof getCallBack === "function") {
  5628. return getCallBack(value);
  5629. } else {
  5630. return target[propertyName];
  5631. }
  5632. },
  5633. set(value) {
  5634. if (typeof setCallBack === "function") {
  5635. setCallBack(value);
  5636. }
  5637. },
  5638. });
  5639. }
  5640. };
  5641.  
  5642. return Utils;
  5643. });

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址