WhiteSevsUtils

一个好用的工具类

当前为 2024-06-11 提交的版本,查看 最新版本

此脚本不应直接安装。它是供其他脚本使用的外部库,要使用该库请加入元指令 // @require https://update.gf.qytechs.cn/scripts/455186/1392339/WhiteSevsUtils.js

  1. (function (global, factory) {
  2. typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
  3. typeof define === 'function' && define.amd ? define(factory) :
  4. (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Utils = factory());
  5. })(this, (function () { 'use strict';
  6.  
  7. class ColorConversion {
  8. /**
  9. * 判断是否是16进制颜色
  10. * @param str
  11. */
  12. isHex(str) {
  13. if (typeof str !== "string") {
  14. return false;
  15. }
  16. if (!str.match(/^(\#|)[0-9a-fA-F]{6}$/)) {
  17. return false;
  18. }
  19. return true;
  20. }
  21. /**
  22. * 16进制颜色转rgba
  23. *
  24. * #ff0000 转 rgba(123,123,123, 0.4)
  25. * @param hex
  26. * @param opacity
  27. */
  28. hexToRgba(hex, opacity) {
  29. if (!this.isHex(hex)) {
  30. // @ts-ignore
  31. throw new TypeError("输入错误的hex", hex);
  32. }
  33. return hex && hex.replace(/\s+/g, "").length === 7
  34. ? "rgba(" +
  35. parseInt("0x" + hex.slice(1, 3)) +
  36. "," +
  37. parseInt("0x" + hex.slice(3, 5)) +
  38. "," +
  39. parseInt("0x" + hex.slice(5, 7)) +
  40. "," +
  41. opacity +
  42. ")"
  43. : "";
  44. }
  45. /**
  46. * hex转rgb
  47. * @param str
  48. * @returns
  49. */
  50. hexToRgb(str) {
  51. if (!this.isHex(str)) {
  52. // @ts-ignore
  53. throw new TypeError("输入错误的hex", str);
  54. }
  55. /* replace替换查找的到的字符串 */
  56. str = str.replace("#", "");
  57. /* match得到查询数组 */
  58. let hxs = str.match(/../g);
  59. for (let index = 0; index < 3; index++) {
  60. // @ts-ignore
  61. hxs[index] = parseInt(hxs[index], 16);
  62. }
  63. return hxs;
  64. }
  65. /**
  66. * rgb转hex
  67. * @param redValue
  68. * @param greenValue
  69. * @param blueValue
  70. * @returns
  71. */
  72. rgbToHex(redValue, greenValue, blueValue) {
  73. /* 验证输入的rgb值是否合法 */
  74. let validPattern = /^\d{1,3}$/;
  75. if (!validPattern.test(redValue.toString()) ||
  76. !validPattern.test(greenValue.toString()) ||
  77. !validPattern.test(blueValue.toString()))
  78. throw new TypeError("输入错误的rgb颜色值");
  79. let hexs = [
  80. redValue.toString(16),
  81. greenValue.toString(16),
  82. blueValue.toString(16),
  83. ];
  84. for (let index = 0; index < 3; index++)
  85. if (hexs[index].length == 1)
  86. hexs[index] = "0" + hexs[index];
  87. return "#" + hexs.join("");
  88. }
  89. /**
  90. * 获取颜色变暗或亮
  91. * @param color 颜色
  92. * @param level 0~1.0
  93. * @returns
  94. */
  95. getDarkColor(color, level) {
  96. if (!this.isHex(color)) {
  97. // @ts-ignore
  98. throw new TypeError("输入错误的hex", color);
  99. }
  100. let rgbc = this.hexToRgb(color);
  101. for (let index = 0; index < 3; index++) {
  102. // @ts-ignore
  103. rgbc[index] = Math.floor(rgbc[index] * (1 - level));
  104. }
  105. // @ts-ignore
  106. return this.rgbToHex(rgbc[0], rgbc[1], rgbc[2]);
  107. }
  108. /**
  109. * 获取颜色变亮
  110. * @param color 颜色
  111. * @param level 0~1.0
  112. * @returns
  113. */
  114. getLightColor(color, level) {
  115. if (!this.isHex(color)) {
  116. // @ts-ignore
  117. throw new TypeError("输入错误的hex", color);
  118. }
  119. let rgbc = this.hexToRgb(color);
  120. for (let index = 0; index < 3; index++) {
  121. // @ts-ignore
  122. rgbc[index] = Math.floor((255 - rgbc[index]) * level + rgbc[index]);
  123. }
  124. // @ts-ignore
  125. return this.rgbToHex(rgbc[0], rgbc[1], rgbc[2]);
  126. }
  127. }
  128.  
  129. class GBKEncoder {
  130. #data = [];
  131. #U2Ghash = {};
  132. #G2Uhash = {};
  133. constructor() {
  134. let dataText = this.handleText("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");
  135. let index = 0;
  136. this.#data = dataText.match(/..../g);
  137. for (let i = 0x81; i <= 0xfe; i++) {
  138. for (let j = 0x40; j <= 0xfe; j++) {
  139. this.#U2Ghash[this.#data[index++]] = ("%" +
  140. i.toString(16) +
  141. "%" +
  142. j.toString(16)).toUpperCase();
  143. }
  144. }
  145. for (let key in this.#U2Ghash) {
  146. this.#G2Uhash[this.#U2Ghash[key]] = key;
  147. }
  148. }
  149. handleText(text) {
  150. text = text
  151. .replace(/#(\d+)\$/g, function (a, b) {
  152. return Array(+b + 3).join("#");
  153. })
  154. .replace(/#/g, "####")
  155. .replace(/(\w\w):([\w#]+)(?:,|$)/g, function (a, hd, dt) {
  156. return dt.replace(/../g, function (a) {
  157. if (a != "##") {
  158. return hd + a;
  159. }
  160. else {
  161. return a;
  162. }
  163. });
  164. });
  165. return text;
  166. }
  167. isAscii(unicode) {
  168. return unicode <= 0x007f && unicode >= 0x0000;
  169. }
  170. /**
  171. * 编码
  172. * @param str
  173. */
  174. encode(str) {
  175. let that = this;
  176. return [...str].reduce((result, val, i) => {
  177. return result + toGBK(val);
  178. }, "");
  179. function toGBK(val) {
  180. let result = "";
  181. for (let i = 0; i < val.length; i++) {
  182. const codePoint = val.codePointAt(i);
  183. const code = String.fromCodePoint(codePoint);
  184. let key = codePoint.toString(16);
  185. key.length != 4 && (key = ("000" + key).match(/....$/)?.[0]);
  186. /* Add up i by code.length */
  187. i += code.length - 1;
  188. /* If code is in ascii range */
  189. if (that.isAscii(codePoint)) {
  190. result += encodeURIComponent(code);
  191. continue;
  192. }
  193. /* If Got encoded string from U2Ghash */
  194. if (that.#U2Ghash[key]) {
  195. result += that.#U2Ghash[key];
  196. continue;
  197. }
  198. /*
  199. If 2 or more char combines to one visible code,
  200. or just this code is not in GBK
  201. */
  202. result += toGBK(`&#${codePoint};`);
  203. }
  204. return result;
  205. }
  206. }
  207. /**
  208. * 解码
  209. * @param str
  210. */
  211. decode(str) {
  212. var GBKMatcher = /%[0-9A-F]{2}%[0-9A-F]{2}/;
  213. var UTFMatcher = /%[0-9A-F]{2}/;
  214. var utf = true;
  215. let that = this;
  216. while (utf) {
  217. let gbkMatch = str.match(GBKMatcher);
  218. let utfMatch = str.match(UTFMatcher);
  219. utf = Boolean(utfMatch);
  220. if (gbkMatch && gbkMatch in that.#G2Uhash) {
  221. str = str.replace(gbkMatch, String.fromCharCode(("0x" + that.#G2Uhash[gbkMatch])));
  222. }
  223. else {
  224. // @ts-ignore
  225. str = str.replace(utfMatch, decodeURIComponent(utfMatch));
  226. }
  227. }
  228. return str;
  229. }
  230. }
  231.  
  232. const UtilsCoreDefaultEnv = {
  233. document: document,
  234. window: window,
  235. globalThis: globalThis,
  236. self: self,
  237. top: top,
  238. };
  239. const UtilsCoreEnv = {
  240. document: document,
  241. window: window,
  242. globalThis: globalThis,
  243. self: self,
  244. top: top,
  245. };
  246. const UtilsCore = {
  247. init(option) {
  248. if (!option) {
  249. option = Object.assign({}, UtilsCoreDefaultEnv);
  250. }
  251. Object.assign(UtilsCoreEnv, option);
  252. },
  253. get document() {
  254. return UtilsCoreEnv.document;
  255. },
  256. get window() {
  257. return UtilsCoreEnv.window;
  258. },
  259. get globalThis() {
  260. return UtilsCoreEnv.globalThis;
  261. },
  262. get self() {
  263. return UtilsCoreEnv.self;
  264. },
  265. get top() {
  266. return UtilsCoreEnv.top;
  267. },
  268. };
  269.  
  270. class UtilsGMCookie {
  271. /**
  272. * 获取单个cookie
  273. * @param cookieName cookie名
  274. */
  275. get(cookieName) {
  276. if (typeof cookieName !== "string") {
  277. throw new TypeError("Utils.GMCookie.get 参数cookieName 必须为字符串");
  278. }
  279. let cookies = document.cookie.split(";");
  280. let findValue = void 0;
  281. for (const cookieItem of cookies) {
  282. let item = cookieItem.trim();
  283. let itemSplit = item.split("=");
  284. let itemName = itemSplit[0];
  285. itemSplit.splice(0, 1);
  286. let itemValue = decodeURIComponent(itemSplit.join(""));
  287. if (itemName === cookieName) {
  288. findValue = {
  289. domain: globalThis.location.hostname,
  290. expirationDate: null,
  291. hostOnly: true,
  292. httpOnly: false,
  293. name: cookieName,
  294. path: "/",
  295. sameSite: "unspecified",
  296. secure: true,
  297. session: false,
  298. value: itemValue,
  299. };
  300. break;
  301. }
  302. }
  303. return findValue;
  304. }
  305. /**
  306. * 获取多组Cookie
  307. * @param paramDetails 配置
  308. * @param callback 获取操作后的回调
  309. * + cookies object[]
  310. * + error string|undefined
  311. **/
  312. list(paramDetails, callback) {
  313. if (paramDetails == null) {
  314. throw new Error("Utils.GMCookie.list 参数不能为空");
  315. }
  316. let resultData = [];
  317. try {
  318. let details = {
  319. url: globalThis.location.href,
  320. domain: globalThis.location.hostname,
  321. name: "",
  322. path: "/",
  323. };
  324. details = utils.assign(details, paramDetails);
  325. let cookies = document.cookie.split(";");
  326. cookies.forEach((item) => {
  327. item = item.trim();
  328. let itemSplit = item.split("=");
  329. let itemName = itemSplit[0];
  330. itemSplit.splice(0, 1);
  331. let itemValue = decodeURIComponent(itemSplit.join(""));
  332. let nameRegexp = details.name instanceof RegExp
  333. ? details.name
  334. : new RegExp("^" + details.name, "g");
  335. if (itemName.match(nameRegexp)) {
  336. resultData.push({
  337. domain: globalThis.location.hostname,
  338. expirationDate: null,
  339. hostOnly: true,
  340. httpOnly: false,
  341. name: itemName,
  342. path: "/",
  343. sameSite: "unspecified",
  344. secure: true,
  345. session: false,
  346. value: itemValue,
  347. });
  348. }
  349. });
  350. if (typeof callback === "function") {
  351. callback(resultData);
  352. }
  353. }
  354. catch (error) {
  355. if (typeof callback === "function") {
  356. callback(resultData, error);
  357. }
  358. }
  359. }
  360. /**
  361. * 获取多组Cookie
  362. * @param paramDetails 配置
  363. **/
  364. getList(paramDetails) {
  365. if (paramDetails == null) {
  366. throw new Error("Utils.GMCookie.list 参数不能为空");
  367. }
  368. let resultData = [];
  369. let details = {
  370. url: globalThis.location.href,
  371. domain: globalThis.location.hostname,
  372. name: "",
  373. path: "/",
  374. };
  375. details = utils.assign(details, paramDetails);
  376. let cookies = document.cookie.split(";");
  377. cookies.forEach((item) => {
  378. item = item.trim();
  379. let itemSplit = item.split("=");
  380. let itemName = itemSplit[0];
  381. itemSplit.splice(0, 1);
  382. let itemValue = decodeURIComponent(itemSplit.join(""));
  383. let nameRegexp = details.name instanceof RegExp
  384. ? details.name
  385. : new RegExp("^" + details.name, "g");
  386. if (itemName.match(nameRegexp)) {
  387. resultData.push({
  388. domain: globalThis.location.hostname,
  389. expirationDate: null,
  390. hostOnly: true,
  391. httpOnly: false,
  392. name: itemName,
  393. path: "/",
  394. sameSite: "unspecified",
  395. secure: true,
  396. session: false,
  397. value: itemValue,
  398. });
  399. }
  400. });
  401. return resultData;
  402. }
  403. /**
  404. * 设置Cookie
  405. * @param paramDetails 配置
  406. * @param callback 设置操作后的回调(成功/失败)
  407. */
  408. set(paramDetails, callback = (error) => { }) {
  409. try {
  410. let details = {
  411. url: window.location.href,
  412. name: "",
  413. value: "",
  414. domain: window.location.hostname,
  415. path: "/",
  416. secure: true,
  417. httpOnly: false,
  418. /**
  419. * Expires in 30 days
  420. */
  421. expirationDate: Math.floor(Date.now()) + 60 * 60 * 24 * 30,
  422. };
  423. details = utils.assign(details, paramDetails);
  424. let life = details.expirationDate
  425. ? details.expirationDate
  426. : Math.floor(Date.now()) + 60 * 60 * 24 * 30;
  427. let cookieStr = details.name +
  428. "=" +
  429. decodeURIComponent(details.value) +
  430. ";expires=" +
  431. new Date(life).toGMTString() +
  432. "; path=/";
  433. document.cookie = cookieStr;
  434. callback();
  435. }
  436. catch (error) {
  437. callback(error);
  438. }
  439. }
  440. /**
  441. * 删除Cookie
  442. * @param paramDetails 配置
  443. * @param callback 删除操作后的回调(成功/失败)
  444. */
  445. delete(paramDetails, callback = (error) => { }) {
  446. try {
  447. let details = {
  448. url: window.location.href,
  449. name: "",
  450. // @ts-ignore
  451. firstPartyDomain: "",
  452. };
  453. details = utils.assign(details, paramDetails);
  454. let cookieStr = details.name + "=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;";
  455. document.cookie = cookieStr;
  456. callback();
  457. }
  458. catch (error) {
  459. callback(error);
  460. }
  461. }
  462. }
  463.  
  464. /// <reference path="./index.d.ts" />
  465. // @name ajaxHooker
  466. // @author cxxjackie
  467. // @version 1.4.2
  468. // @updateLog 修复了fetch请求的参数为Request类型时body类型不正确的bug。
  469. // @supportURL https://bbs.tampermonkey.net.cn/thread-3284-1-1.html
  470.  
  471. const AjaxHooker = function () {
  472. return function() {
  473. const version = '1.4.2';
  474. const hookInst = {
  475. hookFns: [],
  476. filters: []
  477. };
  478. const win = window.unsafeWindow || document.defaultView || window;
  479. let winAh = win.__ajaxHooker;
  480. const resProto = win.Response.prototype;
  481. const xhrResponses = ['response', 'responseText', 'responseXML'];
  482. const fetchResponses = ['arrayBuffer', 'blob', 'formData', 'json', 'text'];
  483. const fetchInitProps = ['method', 'headers', 'body', 'mode', 'credentials', 'cache', 'redirect',
  484. 'referrer', 'referrerPolicy', 'integrity', 'keepalive', 'signal', 'priority'];
  485. const xhrAsyncEvents = ['readystatechange', 'load', 'loadend'];
  486. const getType = ({}).toString.call.bind(({}).toString);
  487. const getDescriptor = Object.getOwnPropertyDescriptor.bind(Object);
  488. const emptyFn = () => {};
  489. const errorFn = e => console.error(e);
  490. function isThenable(obj) {
  491. return obj && ['object', 'function'].includes(typeof obj) && typeof obj.then === 'function';
  492. }
  493. function catchError(fn, ...args) {
  494. try {
  495. const result = fn(...args);
  496. if (isThenable(result)) return result.then(null, errorFn);
  497. return result;
  498. } catch (err) {
  499. console.error(err);
  500. }
  501. }
  502. function defineProp(obj, prop, getter, setter) {
  503. Object.defineProperty(obj, prop, {
  504. configurable: true,
  505. enumerable: true,
  506. get: getter,
  507. set: setter
  508. });
  509. }
  510. function readonly(obj, prop, value = obj[prop]) {
  511. defineProp(obj, prop, () => value, emptyFn);
  512. }
  513. function writable(obj, prop, value = obj[prop]) {
  514. Object.defineProperty(obj, prop, {
  515. configurable: true,
  516. enumerable: true,
  517. writable: true,
  518. value: value
  519. });
  520. }
  521. function parseHeaders(obj) {
  522. const headers = {};
  523. switch (getType(obj)) {
  524. case '[object String]':
  525. for (const line of obj.trim().split(/[\r\n]+/)) {
  526. const [header, value] = line.split(/\s*:\s*/);
  527. if (!header) break;
  528. const lheader = header.toLowerCase();
  529. headers[lheader] = lheader in headers ? `${headers[lheader]}, ${value}` : value;
  530. }
  531. break;
  532. case '[object Headers]':
  533. for (const [key, val] of obj) {
  534. headers[key] = val;
  535. }
  536. break;
  537. case '[object Object]':
  538. return {...obj};
  539. }
  540. return headers;
  541. }
  542. function stopImmediatePropagation() {
  543. this.ajaxHooker_isStopped = true;
  544. }
  545. class SyncThenable {
  546. then(fn) {
  547. fn && fn();
  548. return new SyncThenable();
  549. }
  550. }
  551. class AHRequest {
  552. constructor(request) {
  553. this.request = request;
  554. this.requestClone = {...this.request};
  555. }
  556. shouldFilter(filters) {
  557. const {type, url, method, async} = this.request;
  558. return filters.length && !filters.find(obj => {
  559. switch (true) {
  560. case obj.type && obj.type !== type:
  561. case getType(obj.url) === '[object String]' && !url.includes(obj.url):
  562. case getType(obj.url) === '[object RegExp]' && !obj.url.test(url):
  563. case obj.method && obj.method.toUpperCase() !== method.toUpperCase():
  564. case 'async' in obj && obj.async !== async:
  565. return false;
  566. }
  567. return true;
  568. });
  569. }
  570. waitForRequestKeys() {
  571. const requestKeys = ['url', 'method', 'abort', 'headers', 'data'];
  572. if (!this.request.async) {
  573. win.__ajaxHooker.hookInsts.forEach(({hookFns, filters}) => {
  574. if (this.shouldFilter(filters)) return;
  575. hookFns.forEach(fn => {
  576. if (getType(fn) === '[object Function]') catchError(fn, this.request);
  577. });
  578. requestKeys.forEach(key => {
  579. if (isThenable(this.request[key])) this.request[key] = this.requestClone[key];
  580. });
  581. });
  582. return new SyncThenable();
  583. }
  584. const promises = [];
  585. win.__ajaxHooker.hookInsts.forEach(({hookFns, filters}) => {
  586. if (this.shouldFilter(filters)) return;
  587. promises.push(Promise.all(hookFns.map(fn => catchError(fn, this.request))).then(() =>
  588. Promise.all(requestKeys.map(key => Promise.resolve(this.request[key]).then(
  589. val => this.request[key] = val,
  590. () => this.request[key] = this.requestClone[key]
  591. )))
  592. ));
  593. });
  594. return Promise.all(promises);
  595. }
  596. waitForResponseKeys(response) {
  597. const responseKeys = this.request.type === 'xhr' ? xhrResponses : fetchResponses;
  598. if (!this.request.async) {
  599. if (getType(this.request.response) === '[object Function]') {
  600. catchError(this.request.response, response);
  601. responseKeys.forEach(key => {
  602. if ('get' in getDescriptor(response, key) || isThenable(response[key])) {
  603. delete response[key];
  604. }
  605. });
  606. }
  607. return new SyncThenable();
  608. }
  609. return Promise.resolve(catchError(this.request.response, response)).then(() =>
  610. Promise.all(responseKeys.map(key => {
  611. const descriptor = getDescriptor(response, key);
  612. if (descriptor && 'value' in descriptor) {
  613. return Promise.resolve(descriptor.value).then(
  614. val => response[key] = val,
  615. () => delete response[key]
  616. );
  617. } else {
  618. delete response[key];
  619. }
  620. }))
  621. );
  622. }
  623. }
  624. const proxyHandler = {
  625. get(target, prop) {
  626. const descriptor = getDescriptor(target, prop);
  627. if (descriptor && !descriptor.configurable && !descriptor.writable && !descriptor.get) return target[prop];
  628. const ah = target.__ajaxHooker;
  629. if (ah && ah.proxyProps) {
  630. if (prop in ah.proxyProps) {
  631. const pDescriptor = ah.proxyProps[prop];
  632. if ('get' in pDescriptor) return pDescriptor.get();
  633. if (typeof pDescriptor.value === 'function') return pDescriptor.value.bind(ah);
  634. return pDescriptor.value;
  635. }
  636. if (typeof target[prop] === 'function') return target[prop].bind(target);
  637. }
  638. return target[prop];
  639. },
  640. set(target, prop, value) {
  641. const descriptor = getDescriptor(target, prop);
  642. if (descriptor && !descriptor.configurable && !descriptor.writable && !descriptor.set) return true;
  643. const ah = target.__ajaxHooker;
  644. if (ah && ah.proxyProps && prop in ah.proxyProps) {
  645. const pDescriptor = ah.proxyProps[prop];
  646. pDescriptor.set ? pDescriptor.set(value) : (pDescriptor.value = value);
  647. } else {
  648. target[prop] = value;
  649. }
  650. return true;
  651. }
  652. };
  653. class XhrHooker {
  654. constructor(xhr) {
  655. const ah = this;
  656. Object.assign(ah, {
  657. originalXhr: xhr,
  658. proxyXhr: new Proxy(xhr, proxyHandler),
  659. resThenable: new SyncThenable(),
  660. proxyProps: {},
  661. proxyEvents: {}
  662. });
  663. xhr.addEventListener('readystatechange', e => {
  664. if (ah.proxyXhr.readyState === 4 && ah.request && typeof ah.request.response === 'function') {
  665. const response = {
  666. finalUrl: ah.proxyXhr.responseURL,
  667. status: ah.proxyXhr.status,
  668. responseHeaders: parseHeaders(ah.proxyXhr.getAllResponseHeaders())
  669. };
  670. const tempValues = {};
  671. for (const key of xhrResponses) {
  672. try {
  673. tempValues[key] = ah.originalXhr[key];
  674. } catch (err) {}
  675. defineProp(response, key, () => {
  676. return response[key] = tempValues[key];
  677. }, val => {
  678. delete response[key];
  679. response[key] = val;
  680. });
  681. }
  682. ah.resThenable = new AHRequest(ah.request).waitForResponseKeys(response).then(() => {
  683. for (const key of xhrResponses) {
  684. ah.proxyProps[key] = {get: () => {
  685. if (!(key in response)) response[key] = tempValues[key];
  686. return response[key];
  687. }};
  688. }
  689. });
  690. }
  691. ah.dispatchEvent(e);
  692. });
  693. xhr.addEventListener('load', e => ah.dispatchEvent(e));
  694. xhr.addEventListener('loadend', e => ah.dispatchEvent(e));
  695. for (const evt of xhrAsyncEvents) {
  696. const onEvt = 'on' + evt;
  697. ah.proxyProps[onEvt] = {
  698. get: () => ah.proxyEvents[onEvt] || null,
  699. set: val => ah.addEvent(onEvt, val)
  700. };
  701. }
  702. for (const method of ['setRequestHeader', 'addEventListener', 'removeEventListener', 'open', 'send']) {
  703. ah.proxyProps[method] = {value: ah[method]};
  704. }
  705. }
  706. toJSON() {} // Converting circular structure to JSON
  707. addEvent(type, event) {
  708. if (type.startsWith('on')) {
  709. this.proxyEvents[type] = typeof event === 'function' ? event : null;
  710. } else {
  711. if (typeof event === 'object' && event !== null) event = event.handleEvent;
  712. if (typeof event !== 'function') return;
  713. this.proxyEvents[type] = this.proxyEvents[type] || new Set();
  714. this.proxyEvents[type].add(event);
  715. }
  716. }
  717. removeEvent(type, event) {
  718. if (type.startsWith('on')) {
  719. this.proxyEvents[type] = null;
  720. } else {
  721. if (typeof event === 'object' && event !== null) event = event.handleEvent;
  722. this.proxyEvents[type] && this.proxyEvents[type].delete(event);
  723. }
  724. }
  725. dispatchEvent(e) {
  726. e.stopImmediatePropagation = stopImmediatePropagation;
  727. defineProp(e, 'target', () => this.proxyXhr);
  728. this.proxyEvents[e.type] && this.proxyEvents[e.type].forEach(fn => {
  729. this.resThenable.then(() => !e.ajaxHooker_isStopped && fn.call(this.proxyXhr, e));
  730. });
  731. if (e.ajaxHooker_isStopped) return;
  732. const onEvent = this.proxyEvents['on' + e.type];
  733. onEvent && this.resThenable.then(onEvent.bind(this.proxyXhr, e));
  734. }
  735. setRequestHeader(header, value) {
  736. this.originalXhr.setRequestHeader(header, value);
  737. if (this.originalXhr.readyState !== 1) return;
  738. const headers = this.request.headers;
  739. headers[header] = header in headers ? `${headers[header]}, ${value}` : value;
  740. }
  741. addEventListener(...args) {
  742. if (xhrAsyncEvents.includes(args[0])) {
  743. this.addEvent(args[0], args[1]);
  744. } else {
  745. this.originalXhr.addEventListener(...args);
  746. }
  747. }
  748. removeEventListener(...args) {
  749. if (xhrAsyncEvents.includes(args[0])) {
  750. this.removeEvent(args[0], args[1]);
  751. } else {
  752. this.originalXhr.removeEventListener(...args);
  753. }
  754. }
  755. open(method, url, async = true, ...args) {
  756. this.request = {
  757. type: 'xhr',
  758. url: url.toString(),
  759. method: method.toUpperCase(),
  760. abort: false,
  761. headers: {},
  762. data: null,
  763. response: null,
  764. async: !!async
  765. };
  766. this.openArgs = args;
  767. this.resThenable = new SyncThenable();
  768. ['responseURL', 'readyState', 'status', 'statusText', ...xhrResponses].forEach(key => {
  769. delete this.proxyProps[key];
  770. });
  771. return this.originalXhr.open(method, url, async, ...args);
  772. }
  773. send(data) {
  774. const ah = this;
  775. const xhr = ah.originalXhr;
  776. const request = ah.request;
  777. if (!request) return xhr.send(data);
  778. request.data = data;
  779. new AHRequest(request).waitForRequestKeys().then(() => {
  780. if (request.abort) {
  781. if (typeof request.response === 'function') {
  782. Object.assign(ah.proxyProps, {
  783. responseURL: {value: request.url},
  784. readyState: {value: 4},
  785. status: {value: 200},
  786. statusText: {value: 'OK'}
  787. });
  788. xhrAsyncEvents.forEach(evt => xhr.dispatchEvent(new Event(evt)));
  789. }
  790. } else {
  791. xhr.open(request.method, request.url, request.async, ...ah.openArgs);
  792. for (const header in request.headers) {
  793. xhr.setRequestHeader(header, request.headers[header]);
  794. }
  795. xhr.send(request.data);
  796. }
  797. });
  798. }
  799. }
  800. function fakeXHR() {
  801. const xhr = new winAh.realXHR();
  802. if ('__ajaxHooker' in xhr) console.warn('检测到不同版本的ajaxHooker,可能发生冲突!');
  803. xhr.__ajaxHooker = new XhrHooker(xhr);
  804. return xhr.__ajaxHooker.proxyXhr;
  805. }
  806. fakeXHR.prototype = win.XMLHttpRequest.prototype;
  807. Object.keys(win.XMLHttpRequest).forEach(key => fakeXHR[key] = win.XMLHttpRequest[key]);
  808. function fakeFetch(url, options = {}) {
  809. if (!url) return winAh.realFetch.call(win, url, options);
  810. return new Promise(async (resolve, reject) => {
  811. const init = {};
  812. if (getType(url) === '[object Request]') {
  813. for (const prop of fetchInitProps) init[prop] = url[prop];
  814. if (url.body) init.body = await url.arrayBuffer();
  815. url = url.url;
  816. }
  817. url = url.toString();
  818. Object.assign(init, options);
  819. init.method = init.method || 'GET';
  820. init.headers = init.headers || {};
  821. const request = {
  822. type: 'fetch',
  823. url: url,
  824. method: init.method.toUpperCase(),
  825. abort: false,
  826. headers: parseHeaders(init.headers),
  827. data: init.body,
  828. response: null,
  829. async: true
  830. };
  831. const req = new AHRequest(request);
  832. await req.waitForRequestKeys();
  833. if (request.abort) {
  834. if (typeof request.response === 'function') {
  835. const response = {
  836. finalUrl: request.url,
  837. status: 200,
  838. responseHeaders: {}
  839. };
  840. await req.waitForResponseKeys(response);
  841. const key = fetchResponses.find(k => k in response);
  842. let val = response[key];
  843. if (key === 'json' && typeof val === 'object') {
  844. val = catchError(JSON.stringify.bind(JSON), val);
  845. }
  846. const res = new Response(val, {
  847. status: 200,
  848. statusText: 'OK'
  849. });
  850. defineProp(res, 'type', () => 'basic');
  851. defineProp(res, 'url', () => request.url);
  852. resolve(res);
  853. } else {
  854. reject(new DOMException('aborted', 'AbortError'));
  855. }
  856. return;
  857. }
  858. init.method = request.method;
  859. init.headers = request.headers;
  860. init.body = request.data;
  861. winAh.realFetch.call(win, request.url, init).then(res => {
  862. if (typeof request.response === 'function') {
  863. const response = {
  864. finalUrl: res.url,
  865. status: res.status,
  866. responseHeaders: parseHeaders(res.headers)
  867. };
  868. fetchResponses.forEach(key => res[key] = function() {
  869. if (key in response) return Promise.resolve(response[key]);
  870. return resProto[key].call(this).then(val => {
  871. response[key] = val;
  872. return req.waitForResponseKeys(response).then(() => key in response ? response[key] : val);
  873. });
  874. });
  875. }
  876. resolve(res);
  877. }, reject);
  878. });
  879. }
  880. function fakeFetchClone() {
  881. const descriptors = Object.getOwnPropertyDescriptors(this);
  882. const res = winAh.realFetchClone.call(this);
  883. Object.defineProperties(res, descriptors);
  884. return res;
  885. }
  886. winAh = win.__ajaxHooker = winAh || {
  887. version, fakeXHR, fakeFetch, fakeFetchClone,
  888. realXHR: win.XMLHttpRequest,
  889. realFetch: win.fetch,
  890. realFetchClone: resProto.clone,
  891. hookInsts: new Set()
  892. };
  893. if (winAh.version !== version) console.warn('检测到不同版本的ajaxHooker,可能发生冲突!');
  894. win.XMLHttpRequest = winAh.fakeXHR;
  895. win.fetch = winAh.fakeFetch;
  896. resProto.clone = winAh.fakeFetchClone;
  897. winAh.hookInsts.add(hookInst);
  898. return {
  899. hook: fn => hookInst.hookFns.push(fn),
  900. filter: arr => {
  901. if (Array.isArray(arr)) hookInst.filters = arr;
  902. },
  903. protect: () => {
  904. readonly(win, 'XMLHttpRequest', winAh.fakeXHR);
  905. readonly(win, 'fetch', winAh.fakeFetch);
  906. readonly(resProto, 'clone', winAh.fakeFetchClone);
  907. },
  908. unhook: () => {
  909. winAh.hookInsts.delete(hookInst);
  910. if (!winAh.hookInsts.size) {
  911. writable(win, 'XMLHttpRequest', winAh.realXHR);
  912. writable(win, 'fetch', winAh.realFetch);
  913. writable(resProto, 'clone', winAh.realFetchClone);
  914. delete win.__ajaxHooker;
  915. }
  916. }
  917. };
  918. }();
  919. };
  920.  
  921. class GMMenu {
  922. GM_Api = {
  923. /**
  924. * 获取存储的数据
  925. */
  926. getValue: null,
  927. /**
  928. * 设置数据到存储
  929. */
  930. setValue: null,
  931. /**
  932. * 注册(不可用)菜单
  933. */
  934. registerMenuCommand: null,
  935. /**
  936. * 卸载菜单
  937. */
  938. unregisterMenuCommand: null,
  939. };
  940. MenuHandle = {
  941. context: this,
  942. $data: {
  943. /**
  944. * 菜单数据
  945. */
  946. data: [],
  947. /**
  948. * 本地存储的键名
  949. */
  950. key: "GM_Menu_Local_Map",
  951. },
  952. $default: {
  953. /** 自动刷新网页,默认为true */
  954. autoReload: true,
  955. /**
  956. * 菜单isStoreValue的默认值
  957. */
  958. isStoreValue: true,
  959. },
  960. $emoji: {
  961. /**
  962. * 菜单enable为true的emoji
  963. */
  964. success: "✅",
  965. /**
  966. * 菜单enable为false的emoji
  967. */
  968. error: "❌",
  969. },
  970. /**
  971. * 初始化数据
  972. */
  973. init() {
  974. for (let index = 0; index < this.$data.data.length; index++) {
  975. let menuOption = this.$data.data[index]["data"];
  976. menuOption.enable = Boolean(this.getLocalMenuData(menuOption.key, menuOption.enable));
  977. if (typeof menuOption.showText !== "function") {
  978. menuOption.showText = (menuText, menuEnable) => {
  979. if (menuEnable) {
  980. return this.$emoji.success + " " + menuText;
  981. }
  982. else {
  983. return this.$emoji.error + " " + menuText;
  984. }
  985. };
  986. }
  987. }
  988. },
  989. /**
  990. * 注册(不可用)油猴菜单
  991. * @param menuOptions 如果存在,使用它
  992. */
  993. register(menuOptions) {
  994. let that = this;
  995. if (menuOptions == null) {
  996. throw new TypeError("register菜单数据不能为空");
  997. }
  998. if (!Array.isArray(menuOptions)) {
  999. menuOptions = [menuOptions];
  1000. }
  1001. for (let index = 0; index < menuOptions.length; index++) {
  1002. let cloneMenuOptionData = utils.deepClone(menuOptions[index].data);
  1003. const { showText, clickCallBack } = this.handleMenuData(cloneMenuOptionData);
  1004. let menuId = that.context.GM_Api.registerMenuCommand(showText, clickCallBack);
  1005. menuOptions[index].id = menuId;
  1006. cloneMenuOptionData.deleteMenu = function () {
  1007. that.context.GM_Api.unregisterMenuCommand(menuId);
  1008. };
  1009. Reflect.deleteProperty(menuOptions[index], "handleData");
  1010. menuOptions[index].handleData = cloneMenuOptionData;
  1011. }
  1012. },
  1013. /**
  1014. * 获取本地存储菜单键值
  1015. * @param {string} key 键
  1016. */
  1017. getLocalMenuData(key, defaultValue) {
  1018. let localData = this.context.GM_Api.getValue(this.$data.key, {});
  1019. if (key in localData) {
  1020. return localData[key];
  1021. }
  1022. else {
  1023. return defaultValue;
  1024. }
  1025. },
  1026. /**
  1027. * 设置本地存储菜单键值
  1028. * @param key 键
  1029. * @param value 值
  1030. */
  1031. setLocalMenuData(key, value) {
  1032. let localData = this.context.GM_Api.getValue(this.$data.key, {});
  1033. localData[key] = value;
  1034. this.context.GM_Api.setValue(this.$data.key, localData);
  1035. },
  1036. /**
  1037. * 处理初始化配置
  1038. * @param menuOption
  1039. */
  1040. handleInitDetail(menuOption) {
  1041. menuOption.enable = Boolean(this.getLocalMenuData(menuOption.key, menuOption.enable));
  1042. if (typeof menuOption.showText !== "function") {
  1043. menuOption.showText = (menuText, menuEnable) => {
  1044. if (menuEnable) {
  1045. return this.$emoji.success + " " + menuText;
  1046. }
  1047. else {
  1048. return this.$emoji.error + " " + menuText;
  1049. }
  1050. };
  1051. }
  1052. return menuOption;
  1053. },
  1054. /**
  1055. * 对菜单数据进行处理
  1056. * @param menuOption
  1057. */
  1058. handleMenuData(menuOption) {
  1059. let that = this;
  1060. let menuLocalDataItemKey = menuOption.key;
  1061. /* 菜单默认开启的状态 */
  1062. let defaultEnable = Boolean(this.getLocalMenuData(menuLocalDataItemKey, menuOption.enable));
  1063. /** 油猴菜单上显示的文本 */
  1064. let showText = menuOption.showText(menuOption.text, defaultEnable);
  1065. ({
  1066. /**
  1067. * 菜单的id
  1068. */
  1069. id: menuOption.id,
  1070. /**
  1071. * 点击菜单项后是否应关闭弹出菜单
  1072. */
  1073. autoClose: menuOption.autoClose,
  1074. /**
  1075. * 菜单项的可选访问键
  1076. */
  1077. accessKey: menuOption.accessKey,
  1078. /**
  1079. * 菜单项的鼠标悬浮上的工具提示
  1080. */
  1081. title: menuOption.title,
  1082. });
  1083. /* 点击菜单后触发callback后的网页是否刷新 */
  1084. menuOption.autoReload =
  1085. typeof menuOption.autoReload !== "boolean"
  1086. ? this.$default.autoReload
  1087. : menuOption.autoReload;
  1088. /* 点击菜单后触发callback后的网页是否存储值 */
  1089. menuOption.isStoreValue =
  1090. typeof menuOption.isStoreValue !== "boolean"
  1091. ? this.$default.isStoreValue
  1092. : menuOption.isStoreValue;
  1093. /**
  1094. * 用户点击菜单后的回调函数
  1095. * @param event
  1096. */
  1097. function clickCallBack(event) {
  1098. let localEnable = Boolean(that.getLocalMenuData(menuLocalDataItemKey, defaultEnable));
  1099. if (menuOption.isStoreValue) {
  1100. that.setLocalMenuData(menuLocalDataItemKey, !localEnable);
  1101. }
  1102. if (typeof menuOption.callback === "function") {
  1103. menuOption.callback({
  1104. key: menuLocalDataItemKey,
  1105. enable: !localEnable,
  1106. oldEnable: localEnable,
  1107. event: event,
  1108. storeValue(value) {
  1109. that.setLocalMenuData(menuLocalDataItemKey, value);
  1110. },
  1111. });
  1112. }
  1113. /* 不刷新网页就刷新菜单 */
  1114. if (menuOption.autoReload) {
  1115. window.location.reload();
  1116. }
  1117. else {
  1118. that.context.update();
  1119. }
  1120. }
  1121. return {
  1122. showText,
  1123. clickCallBack,
  1124. };
  1125. },
  1126. /**
  1127. * 获取目标菜单配置数据
  1128. * @param menuKey 菜单-键key
  1129. */
  1130. getMenuData(menuKey) {
  1131. return this.$data.data.find((item) => item.data.key === menuKey);
  1132. },
  1133. /**
  1134. * 获取目标菜单配置
  1135. * @param menuKey 菜单-键key
  1136. */
  1137. getMenuOption(menuKey) {
  1138. return this.$data.data.find((item) => item.data.key === menuKey)?.data;
  1139. },
  1140. /**
  1141. * 获取目标菜单处理后的配置
  1142. * @param menuKey 菜单-键key
  1143. */
  1144. getMenuHandledOption(menuKey) {
  1145. return this.$data.data.find((item) => item.handleData.key === menuKey)
  1146. ?.handleData;
  1147. },
  1148. };
  1149. constructor(details) {
  1150. this.GM_Api.getValue = details.GM_getValue;
  1151. this.GM_Api.setValue = details.GM_setValue;
  1152. this.GM_Api.registerMenuCommand = details.GM_registerMenuCommand;
  1153. this.GM_Api.unregisterMenuCommand = details.GM_unregisterMenuCommand;
  1154. this.MenuHandle.$default.autoReload =
  1155. typeof details.autoReload === "boolean" ? details.autoReload : true;
  1156. for (const keyName of Object.keys(this.GM_Api)) {
  1157. if (typeof this.GM_Api[keyName] !== "function") {
  1158. throw new Error(`Utils.GM_Menu 请在脚本开头加上 @grant ${keyName},且传入该对象`);
  1159. }
  1160. }
  1161. this.add(details?.data || []);
  1162. }
  1163. /**
  1164. * 新增菜单数据
  1165. * @param paramData
  1166. */
  1167. add(paramData) {
  1168. if (Array.isArray(paramData)) {
  1169. for (const _paramData of paramData) {
  1170. // @ts-ignore
  1171. this.MenuHandle.$data.data.push({
  1172. data: _paramData,
  1173. id: void 0,
  1174. });
  1175. }
  1176. }
  1177. else {
  1178. // @ts-ignore
  1179. this.MenuHandle.$data.data.push({
  1180. data: paramData,
  1181. id: void 0,
  1182. });
  1183. }
  1184. this.update();
  1185. }
  1186. /**
  1187. * 更新菜单数据
  1188. * @param options 数据
  1189. */
  1190. update(options) {
  1191. let optionsList = [];
  1192. if (Array.isArray(options)) {
  1193. optionsList = [...optionsList, ...options];
  1194. }
  1195. else if (options != null) {
  1196. optionsList = [...optionsList, options];
  1197. }
  1198. optionsList.forEach((item) => {
  1199. let targetMenu = this.MenuHandle.getMenuOption(item.key);
  1200. if (targetMenu) {
  1201. Object.assign(targetMenu, item);
  1202. }
  1203. });
  1204. this.MenuHandle.$data.data.forEach((value) => {
  1205. if (value.handleData) {
  1206. value.handleData.deleteMenu();
  1207. }
  1208. });
  1209. this.MenuHandle.init();
  1210. this.MenuHandle.register(this.MenuHandle.$data.data);
  1211. }
  1212. /**
  1213. * 卸载菜单
  1214. * @param menuId 已注册(不可用)的菜单id
  1215. */
  1216. delete(menuId) {
  1217. this.GM_Api.unregisterMenuCommand(menuId);
  1218. }
  1219. /**
  1220. * 根据键值获取enable值
  1221. * @param menuKey 菜单-键key
  1222. */
  1223. get(menuKey) {
  1224. return this.getEnable(menuKey);
  1225. }
  1226. /**
  1227. * 根据键值获取enable值
  1228. * @param menuKey 菜单-键key
  1229. */
  1230. getEnable(menuKey) {
  1231. return this.MenuHandle.getMenuHandledOption(menuKey).enable;
  1232. }
  1233. /**
  1234. * 根据键值获取text值
  1235. * @param menuKey 菜单-键key
  1236. */
  1237. getText(menuKey) {
  1238. return this.MenuHandle.getMenuHandledOption(menuKey).text;
  1239. }
  1240. /**
  1241. * 根据键值获取showText函数的值
  1242. * @param menuKey 菜单-键key
  1243. */
  1244. getShowTextValue(menuKey) {
  1245. return this.MenuHandle.getMenuHandledOption(menuKey).showText(this.getText(menuKey), this.get(menuKey));
  1246. }
  1247. /**
  1248. * 获取当前已注册(不可用)菜单的id
  1249. * @param menuKey
  1250. */
  1251. getMenuId(menuKey) {
  1252. let result = null;
  1253. for (let index = 0; index < this.MenuHandle.$data.data.length; index++) {
  1254. const optionData = this.MenuHandle.$data.data[index];
  1255. if (optionData.handleData.key === menuKey) {
  1256. result = optionData.id;
  1257. break;
  1258. }
  1259. }
  1260. return result;
  1261. }
  1262. /**
  1263. * 根据键值获取accessKey值
  1264. * @param menuKey 菜单-键key
  1265. */
  1266. getAccessKey(menuKey) {
  1267. return this.MenuHandle.getMenuHandledOption(menuKey)?.accessKey;
  1268. }
  1269. /**
  1270. * 根据键值获取autoClose值
  1271. * @param menuKey 菜单-键key
  1272. */
  1273. getAutoClose(menuKey) {
  1274. return this.MenuHandle.getMenuHandledOption(menuKey)?.autoClose;
  1275. }
  1276. /**
  1277. * 根据键值获取autoReload值
  1278. * @param menuKey 菜单-键key
  1279. */
  1280. getAutoReload(menuKey) {
  1281. return this.MenuHandle.getMenuHandledOption(menuKey)?.autoReload;
  1282. }
  1283. /**
  1284. * 根据键值获取callback函数
  1285. * @param menuKey 菜单-键key
  1286. */
  1287. getCallBack(menuKey) {
  1288. return this.MenuHandle.getMenuHandledOption(menuKey)?.callback;
  1289. }
  1290. /**
  1291. * 获取当enable为true时默认显示在菜单中前面的emoji图标
  1292. */
  1293. getEnableTrueEmoji() {
  1294. return this.MenuHandle.$emoji.success;
  1295. }
  1296. /**
  1297. * 获取当enable为false时默认显示在菜单中前面的emoji图标
  1298. */
  1299. getEnableFalseEmoji() {
  1300. return this.MenuHandle.$emoji.error;
  1301. }
  1302. /**
  1303. * 获取本地存储的菜单外部的键名
  1304. * @param keyName
  1305. */
  1306. getLocalStorageKeyName() {
  1307. return this.MenuHandle.$data.key;
  1308. }
  1309. /**
  1310. * 设置菜单的值
  1311. * @param menuKey 菜单-键key
  1312. * @param value 需要设置的值
  1313. */
  1314. setValue(menuKey, value) {
  1315. this.MenuHandle.setLocalMenuData(menuKey, value);
  1316. }
  1317. /**
  1318. * 设置菜单的值
  1319. * @param menuKey 菜单-键key
  1320. * @param value 需要设置的值
  1321. */
  1322. setEnable(menuKey, value) {
  1323. this.setValue(menuKey, Boolean(value));
  1324. }
  1325. /**
  1326. * 设置当enable为true时默认显示在菜单中前面的emoji图标
  1327. * @param emojiString
  1328. */
  1329. setEnableTrueEmoji(emojiString) {
  1330. if (typeof emojiString !== "string") {
  1331. throw new Error("参数emojiString必须是string类型");
  1332. }
  1333. this.MenuHandle.$emoji.success = emojiString;
  1334. }
  1335. /**
  1336. * 设置当enable为false时默认显示在菜单中前面的emoji图标
  1337. * @param emojiString
  1338. */
  1339. setEnableFalseEmoji(emojiString) {
  1340. if (typeof emojiString !== "string") {
  1341. throw new Error("参数emojiString必须是string类型");
  1342. }
  1343. this.MenuHandle.$emoji.error = emojiString;
  1344. }
  1345. /**
  1346. * 设置本地存储的菜单外部的键名
  1347. * @param keyName
  1348. */
  1349. setLocalStorageKeyName(keyName) {
  1350. if (typeof keyName !== "string") {
  1351. throw new Error("参数keyName必须是string类型");
  1352. }
  1353. this.MenuHandle.$data.key = keyName;
  1354. }
  1355. }
  1356.  
  1357. class Hooks {
  1358. /**
  1359. * 在Function原型上添加自定义方法.hook和.unhook
  1360. */
  1361. initEnv() {
  1362. Function.prototype.hook = function (realFunc, hookFunc, context) {
  1363. let _context = null; //函数上下文
  1364. let _funcName = null; //函数名
  1365. _context = context || window;
  1366. _funcName = getFuncName(this);
  1367. _context["realFunc_" + _funcName] = this;
  1368. if (_context[_funcName].prototype &&
  1369. _context[_funcName].prototype.isHooked) {
  1370. console.log("Already has been hooked,unhook first");
  1371. return false;
  1372. }
  1373. function getFuncName(fn) {
  1374. // 获取函数名
  1375. let strFunc = fn.toString();
  1376. let _regex = /function\s+(\w+)\s*\(/;
  1377. let patten = strFunc.match(_regex);
  1378. if (patten) {
  1379. return patten[1];
  1380. }
  1381. return "";
  1382. }
  1383. try {
  1384. eval("_context[_funcName] = function " +
  1385. _funcName +
  1386. "(){\n" +
  1387. "let args = Array.prototype.slice.call(arguments,0);\n" +
  1388. "let obj = this;\n" +
  1389. "hookFunc.apply(obj,args);\n" +
  1390. "return _context['realFunc_" +
  1391. _funcName +
  1392. "'].apply(obj,args);\n" +
  1393. "};");
  1394. _context[_funcName].prototype.isHooked = true;
  1395. return true;
  1396. }
  1397. catch (e) {
  1398. console.log("Hook failed,check the params.");
  1399. return false;
  1400. }
  1401. };
  1402. Function.prototype.unhook = function (realFunc, funcName, context) {
  1403. let _context = null;
  1404. let _funcName = null;
  1405. _context = context || window;
  1406. _funcName = funcName;
  1407. if (!_context[_funcName].prototype.isHooked) {
  1408. console.log("No function is hooked on");
  1409. return false;
  1410. }
  1411. _context[_funcName] = _context["realFunc" + _funcName];
  1412. Reflect.deleteProperty(_context, "realFunc_" + _funcName);
  1413. return true;
  1414. };
  1415. }
  1416. /**
  1417. * 删除在Function原型上添加的自定义方法.hook和.unhook
  1418. */
  1419. cleanEnv() {
  1420. if (Function.prototype.hasOwnProperty("hook")) {
  1421. Reflect.deleteProperty(Function.prototype, "hook");
  1422. }
  1423. if (Function.prototype.hasOwnProperty("unhook")) {
  1424. Reflect.deleteProperty(Function.prototype, "unhook");
  1425. }
  1426. return true;
  1427. }
  1428. }
  1429.  
  1430. class Httpx {
  1431. GM_Api = {
  1432. xmlHttpRequest: null,
  1433. };
  1434. HttpxRequestHook = {
  1435. /**
  1436. * 发送请求前的回调
  1437. * 如果返回false则阻止本次返回
  1438. * @param details 当前的请求配置
  1439. */
  1440. beforeRequestCallBack(details) { },
  1441. };
  1442. HttpxRequestDetails = {
  1443. context: this,
  1444. /**
  1445. * 根据传入的参数处理获取details配置
  1446. */
  1447. handleBeforeRequestDetails(...args) {
  1448. let result = {};
  1449. if (typeof args[0] === "string") {
  1450. /* 传入的是url,details? */
  1451. let url = args[0];
  1452. result.url = url;
  1453. if (typeof args[1] === "object") {
  1454. /* 处理第二个参数details */
  1455. let details = args[1];
  1456. result = details;
  1457. result.url = url;
  1458. }
  1459. }
  1460. else {
  1461. /* 传入的是details */
  1462. result = args[0];
  1463. }
  1464. return result;
  1465. },
  1466. /**
  1467. * 获取请求配置
  1468. * @param method 当前请求方法,默认get
  1469. * @param resolve promise回调
  1470. * @param details 请求配置
  1471. */
  1472. getDetails(method, resolve, details) {
  1473. let that = this;
  1474. let result = {
  1475. url: details.url || this.context.#defaultDetails.url,
  1476. method: (method || "GET").toString().toUpperCase(),
  1477. timeout: details.timeout || this.context.#defaultDetails.timeout,
  1478. responseType: details.responseType || this.context.#defaultDetails.responseType,
  1479. /* 对象使用深拷贝 */
  1480. headers: utils.deepClone(this.context.#defaultDetails.headers),
  1481. data: details.data || this.context.#defaultDetails.data,
  1482. redirect: details.redirect || this.context.#defaultDetails.redirect,
  1483. cookie: details.cookie || this.context.#defaultDetails.cookie,
  1484. binary: details.binary || this.context.#defaultDetails.binary,
  1485. nocache: details.nocache || this.context.#defaultDetails.nocache,
  1486. revalidate: details.revalidate || this.context.#defaultDetails.revalidate,
  1487. /* 对象使用深拷贝 */
  1488. context: utils.deepClone(details.context || this.context.#defaultDetails.context),
  1489. overrideMimeType: details.overrideMimeType ||
  1490. this.context.#defaultDetails.overrideMimeType,
  1491. anonymous: details.anonymous || this.context.#defaultDetails.anonymous,
  1492. fetch: details.fetch || this.context.#defaultDetails.fetch,
  1493. /* 对象使用深拷贝 */
  1494. fetchInit: utils.deepClone(this.context.#defaultDetails.fetchInit),
  1495. user: details.user || this.context.#defaultDetails.user,
  1496. password: details.password || this.context.#defaultDetails.password,
  1497. onabort(...args) {
  1498. that.context.HttpxCallBack.onAbort(details, resolve, args);
  1499. },
  1500. onerror(...args) {
  1501. that.context.HttpxCallBack.onError(details, resolve, args);
  1502. },
  1503. onloadstart(...args) {
  1504. that.context.HttpxCallBack.onLoadStart(details, args);
  1505. },
  1506. onprogress(...args) {
  1507. that.context.HttpxCallBack.onProgress(details, args);
  1508. },
  1509. onreadystatechange(...args) {
  1510. that.context.HttpxCallBack.onReadyStateChange(details, args);
  1511. },
  1512. ontimeout(...args) {
  1513. that.context.HttpxCallBack.onTimeout(details, resolve, args);
  1514. },
  1515. onload(...args) {
  1516. that.context.HttpxCallBack.onLoad(details, resolve, args);
  1517. },
  1518. };
  1519. if (typeof this.context.GM_Api.xmlHttpRequest !== "function") {
  1520. result.fetch = true;
  1521. }
  1522. if (typeof result.headers === "object") {
  1523. if (typeof details.headers === "object") {
  1524. Object.keys(details.headers).forEach((keyName, index) => {
  1525. if (keyName in result.headers &&
  1526. details.headers?.[keyName] == null) {
  1527. /* 在默认的header中存在,且设置它新的值为空,那么就是默认的值 */
  1528. Reflect.deleteProperty(result.headers, keyName);
  1529. }
  1530. else {
  1531. result.headers[keyName] = details?.headers?.[keyName];
  1532. }
  1533. });
  1534. }
  1535. }
  1536. else {
  1537. result.headers = details.headers;
  1538. }
  1539. if (typeof result.fetchInit === "object") {
  1540. /* 使用assign替换且添加 */
  1541. if (typeof details.fetchInit === "object") {
  1542. Object.keys(details.fetchInit).forEach((keyName, index) => {
  1543. if (keyName in result.fetchInit &&
  1544. details.fetchInit[keyName] == null) {
  1545. /* 在默认的fetchInit中存在,且设置它新的值为空,那么就是默认的值 */
  1546. Reflect.deleteProperty(result.fetchInit, keyName);
  1547. }
  1548. else {
  1549. result.fetchInit[keyName] = details.fetchInit[keyName];
  1550. }
  1551. });
  1552. }
  1553. }
  1554. else {
  1555. result.fetchInit = details.fetchInit;
  1556. }
  1557. return result;
  1558. },
  1559. /**
  1560. * 处理发送请求的details,去除值为undefined、空function的值
  1561. * @param details
  1562. */
  1563. handle(details) {
  1564. Object.keys(details).forEach((keyName) => {
  1565. if (details[keyName] == null ||
  1566. (details[keyName] instanceof Function &&
  1567. utils.isNull(details[keyName]))) {
  1568. Reflect.deleteProperty(details, keyName);
  1569. return;
  1570. }
  1571. });
  1572. if (utils.isNull(details.url)) {
  1573. throw new TypeError(`Utils.Httpx 参数 url不符合要求: ${details.url}`);
  1574. }
  1575. /* method值统一大写,兼容Via */
  1576. details.method = details.method.toUpperCase();
  1577. /* 判断是否是以http开头,否则主动加上origin */
  1578. try {
  1579. new URL(details.url);
  1580. }
  1581. catch (error) {
  1582. if (details.url.startsWith("//")) {
  1583. details.url = globalThis.location.protocol + details.url;
  1584. }
  1585. else if (details.url.startsWith("/")) {
  1586. details.url = globalThis.location.origin + details.url;
  1587. }
  1588. else {
  1589. details.url = globalThis.location.origin + "/" + details.url;
  1590. }
  1591. }
  1592. return details;
  1593. },
  1594. /**
  1595. * 处理fetch的配置
  1596. * @param details
  1597. */
  1598. handleFetchDetail(details) {
  1599. /**
  1600. * fetch的请求配置
  1601. **/
  1602. let fetchRequestInit = {};
  1603. if ((details.method === "GET" || details.method === "HEAD") &&
  1604. details.data != null) {
  1605. /* GET 或 HEAD 方法的请求不能包含 body 信息 */
  1606. Reflect.deleteProperty(details, "data");
  1607. }
  1608. /* 中止信号控制器 */
  1609. let abortController = new AbortController();
  1610. let signal = abortController.signal;
  1611. signal.onabort = () => {
  1612. details.onabort({
  1613. isFetch: true,
  1614. responseText: "",
  1615. response: null,
  1616. readyState: 4,
  1617. responseHeaders: "",
  1618. status: 0,
  1619. statusText: "",
  1620. error: "aborted",
  1621. });
  1622. };
  1623. fetchRequestInit.method = details.method ?? "GET";
  1624. fetchRequestInit.headers = details.headers;
  1625. fetchRequestInit.body = details.data;
  1626. fetchRequestInit.mode = "cors";
  1627. fetchRequestInit.credentials = "include";
  1628. fetchRequestInit.cache = "no-cache";
  1629. fetchRequestInit.redirect = "follow";
  1630. fetchRequestInit.referrerPolicy = "origin-when-cross-origin";
  1631. fetchRequestInit.signal = signal;
  1632. Object.assign(fetchRequestInit, details.fetchInit || {});
  1633. return {
  1634. fetchDetails: details,
  1635. fetchRequestInit: fetchRequestInit,
  1636. abortController: abortController,
  1637. };
  1638. },
  1639. };
  1640. HttpxCallBack = {
  1641. context: this,
  1642. /**
  1643. * onabort请求被取消-触发
  1644. * @param details 配置
  1645. * @param resolve 回调
  1646. * @param argumentsList 参数列表
  1647. */
  1648. onAbort(details, resolve, argumentsList) {
  1649. if ("onabort" in details) {
  1650. details.onabort.apply(this, argumentsList);
  1651. }
  1652. else if ("onabort" in this.context.#defaultDetails) {
  1653. this.context.#defaultDetails.onabort.apply(this, argumentsList);
  1654. }
  1655. resolve({
  1656. status: false,
  1657. data: [...argumentsList],
  1658. msg: "请求被取消",
  1659. type: "onabort",
  1660. });
  1661. },
  1662. /**
  1663. * onerror请求异常-触发
  1664. * @param details 配置
  1665. * @param resolve 回调
  1666. * @param argumentsList 响应的参数列表
  1667. */
  1668. onError(details, resolve, argumentsList) {
  1669. if ("onerror" in details) {
  1670. details.onerror.apply(this, argumentsList);
  1671. }
  1672. else if ("onerror" in this.context.#defaultDetails) {
  1673. this.context.#defaultDetails.onerror.apply(this, argumentsList);
  1674. }
  1675. let response = argumentsList;
  1676. if (response.length) {
  1677. response = response[0];
  1678. }
  1679. resolve({
  1680. status: false,
  1681. data: response,
  1682. details: details,
  1683. msg: "请求异常",
  1684. type: "onerror",
  1685. });
  1686. },
  1687. /**
  1688. * ontimeout请求超时-触发
  1689. * @param details 配置
  1690. * @param resolve 回调
  1691. * @param argumentsList 参数列表
  1692. */
  1693. onTimeout(details, resolve, argumentsList) {
  1694. if ("ontimeout" in details) {
  1695. details.ontimeout.apply(this, argumentsList);
  1696. }
  1697. else if ("ontimeout" in this.context.#defaultDetails) {
  1698. this.context.#defaultDetails.ontimeout.apply(this, argumentsList);
  1699. }
  1700. resolve({
  1701. status: false,
  1702. data: [...argumentsList],
  1703. msg: "请求超时",
  1704. type: "ontimeout",
  1705. });
  1706. },
  1707. /**
  1708. * onloadstart请求开始-触发
  1709. * @param details 配置
  1710. * @param argumentsList 参数列表
  1711. */
  1712. onLoadStart(details, argumentsList) {
  1713. if ("onloadstart" in details) {
  1714. details.onloadstart.apply(this, argumentsList);
  1715. }
  1716. else if ("onloadstart" in this.context.#defaultDetails) {
  1717. this.context.#defaultDetails.onloadstart.apply(this, argumentsList);
  1718. }
  1719. },
  1720. /**
  1721. * onload加载完毕-触发
  1722. * @param details 请求的配置
  1723. * @param resolve 回调
  1724. * @param argumentsList 参数列表
  1725. */
  1726. onLoad(details, resolve, argumentsList) {
  1727. /* X浏览器会因为设置了responseType导致不返回responseText */
  1728. let Response = argumentsList[0];
  1729. /* responseText为空,response不为空的情况 */
  1730. if (utils.isNull(Response["responseText"]) &&
  1731. utils.isNotNull(Response["response"])) {
  1732. if (typeof Response["response"] === "object") {
  1733. utils.tryCatch().run(() => {
  1734. Response["responseText"] = JSON.stringify(Response["response"]);
  1735. });
  1736. }
  1737. else {
  1738. Response["responseText"] = Response["response"];
  1739. }
  1740. }
  1741. /* response为空,responseText不为空的情况 */
  1742. if (Response["response"] == null &&
  1743. typeof Response["responseText"] === "string" &&
  1744. Response["responseText"].trim() !== "") {
  1745. let newResponse = Response["responseText"];
  1746. if (details.responseType === "json") {
  1747. newResponse = utils.toJSON(Response["responseText"]);
  1748. }
  1749. else if (details.responseType === "document") {
  1750. let parser = new DOMParser();
  1751. newResponse = parser.parseFromString(Response["responseText"], "text/html");
  1752. }
  1753. else if (details.responseType === "arraybuffer") {
  1754. let encoder = new TextEncoder();
  1755. let arrayBuffer = encoder.encode(Response["responseText"]);
  1756. newResponse = arrayBuffer;
  1757. }
  1758. else if (details.responseType === "blob") {
  1759. let encoder = new TextEncoder();
  1760. let arrayBuffer = encoder.encode(Response["responseText"]);
  1761. newResponse = new Blob([arrayBuffer]);
  1762. }
  1763. else {
  1764. newResponse = Response["responseText"];
  1765. }
  1766. try {
  1767. Response["response"] = newResponse;
  1768. }
  1769. catch (error) {
  1770. console.warn("response 无法被覆盖");
  1771. }
  1772. }
  1773. /* Stay扩展中没有finalUrl,对应的是responseURL */
  1774. if (Response["finalUrl"] == null &&
  1775. Response["responseURL"] != null) {
  1776. Response["finalUrl"] = Response["responseURL"];
  1777. }
  1778. /* 状态码2xx都是成功的 */
  1779. if (Math.floor(Response.status / 100) === 2) {
  1780. resolve({
  1781. status: true,
  1782. data: Response,
  1783. details: details,
  1784. msg: "请求完毕",
  1785. type: "onload",
  1786. });
  1787. }
  1788. else {
  1789. this.context.HttpxCallBack.onError(details, resolve, argumentsList);
  1790. }
  1791. },
  1792. /**
  1793. * onprogress上传进度-触发
  1794. * @param details 配置
  1795. * @param argumentsList 参数列表
  1796. */
  1797. onProgress(details, argumentsList) {
  1798. if ("onprogress" in details) {
  1799. details.onprogress.apply(this, argumentsList);
  1800. }
  1801. else if ("onprogress" in this.context.#defaultDetails) {
  1802. this.context.#defaultDetails.onprogress.apply(this, argumentsList);
  1803. }
  1804. },
  1805. /**
  1806. * onreadystatechange准备状态改变-触发
  1807. * @param details 配置
  1808. * @param argumentsList 参数列表
  1809. */
  1810. onReadyStateChange(details, argumentsList) {
  1811. if ("onreadystatechange" in details) {
  1812. details.onreadystatechange.apply(this, argumentsList);
  1813. }
  1814. else if ("onreadystatechange" in this.context.#defaultDetails) {
  1815. this.context.#defaultDetails.onreadystatechange.apply(this, argumentsList);
  1816. }
  1817. },
  1818. };
  1819. HttpxRequest = {
  1820. context: this,
  1821. /**
  1822. * 发送请求
  1823. * @param details
  1824. */
  1825. request(details) {
  1826. if (this.context.#LOG_DETAILS) {
  1827. console.log("Httpx请求配置👇", details);
  1828. }
  1829. if (typeof this.context.HttpxRequestHook.beforeRequestCallBack ===
  1830. "function") {
  1831. let hookResult = this.context.HttpxRequestHook.beforeRequestCallBack(details);
  1832. if (typeof hookResult === "boolean" && !hookResult) {
  1833. return;
  1834. }
  1835. }
  1836. if (details.fetch) {
  1837. const { fetchDetails, fetchRequestInit, abortController } = this.context.HttpxRequestDetails.handleFetchDetail(details);
  1838. this.fetch(fetchDetails, fetchRequestInit, abortController);
  1839. }
  1840. else {
  1841. Reflect.deleteProperty(details, "fetchInit");
  1842. this.xmlHttpRequest(details);
  1843. }
  1844. },
  1845. /**
  1846. * 使用油猴函数GM_xmlhttpRequest发送请求
  1847. * @param details
  1848. */
  1849. xmlHttpRequest(details) {
  1850. this.context.GM_Api.xmlHttpRequest(details);
  1851. },
  1852. /**
  1853. * 使用fetch发送请求
  1854. * @param details
  1855. * @param fetchRequestInit
  1856. * @param abortController
  1857. */
  1858. fetch(details, fetchRequestInit, abortController) {
  1859. fetch(details.url, fetchRequestInit)
  1860. .then(async (resp) => {
  1861. /**
  1862. * @type {HttpxAsyncResultData}
  1863. */
  1864. let httpxResponse = {
  1865. isFetch: true,
  1866. finalUrl: resp.url,
  1867. readyState: 4,
  1868. status: resp.status,
  1869. statusText: resp.statusText,
  1870. response: void 0,
  1871. responseFetchHeaders: resp.headers,
  1872. responseHeaders: "",
  1873. responseText: void 0,
  1874. responseType: details.responseType,
  1875. responseXML: void 0,
  1876. };
  1877. Object.assign(httpxResponse, details.context || {});
  1878. for (const [key, value] of resp.headers.entries()) {
  1879. httpxResponse.responseHeaders += `${key}: ${value}\n`;
  1880. }
  1881. /* 如果是流式传输,直接返回 */
  1882. if (details.responseType === "stream" ||
  1883. (resp.headers.has("Content-Type") &&
  1884. resp.headers.get("Content-Type").includes("text/event-stream"))) {
  1885. httpxResponse["isStream"] = true;
  1886. httpxResponse.response = resp.body;
  1887. Reflect.deleteProperty(httpxResponse, "responseText");
  1888. Reflect.deleteProperty(httpxResponse, "responseXML");
  1889. details.onload(httpxResponse);
  1890. return;
  1891. }
  1892. /** 响应 */
  1893. let response = "";
  1894. /** 响应字符串 */
  1895. let responseText = "";
  1896. /** 响应xml文档 */
  1897. let responseXML = "";
  1898. let arrayBuffer = await resp.arrayBuffer;
  1899. let encoding = "utf-8";
  1900. if (resp.headers.has("Content-Type")) {
  1901. let charsetMatched = resp.headers
  1902. .get("Content-Type")
  1903. ?.match(/charset=(.+)/);
  1904. if (charsetMatched) {
  1905. encoding = charsetMatched[1];
  1906. }
  1907. }
  1908. let textDecoder = new TextDecoder(encoding);
  1909. responseText = textDecoder.decode(await resp.arrayBuffer());
  1910. response = responseText;
  1911. if (details.responseType === "arraybuffer") {
  1912. response = arrayBuffer;
  1913. }
  1914. else if (details.responseType === "blob") {
  1915. response = new Blob([arrayBuffer]);
  1916. }
  1917. else if (details.responseType === "document" ||
  1918. details.responseType == null) {
  1919. let parser = new DOMParser();
  1920. response = parser.parseFromString(responseText, "text/html");
  1921. }
  1922. else if (details.responseType === "json") {
  1923. response = utils.toJSON(responseText);
  1924. }
  1925. let parser = new DOMParser();
  1926. responseXML = parser.parseFromString(responseText, "text/xml");
  1927. httpxResponse.response = response;
  1928. httpxResponse.responseText = responseText;
  1929. httpxResponse.responseXML = responseXML;
  1930. details.onload(httpxResponse);
  1931. })
  1932. .catch((err) => {
  1933. if (err.name === "AbortError") {
  1934. return;
  1935. }
  1936. details.onerror({
  1937. isFetch: true,
  1938. finalUrl: details.url,
  1939. readyState: 4,
  1940. status: 0,
  1941. statusText: "",
  1942. responseHeaders: "",
  1943. responseText: "",
  1944. error: err,
  1945. });
  1946. });
  1947. details.onloadstart({
  1948. isFetch: true,
  1949. finalUrl: details.url,
  1950. readyState: 1,
  1951. responseHeaders: "",
  1952. responseText: "",
  1953. status: 0,
  1954. statusText: "",
  1955. });
  1956. return {
  1957. abort() {
  1958. abortController.abort();
  1959. },
  1960. };
  1961. },
  1962. };
  1963. /**
  1964. * 默认配置
  1965. */
  1966. #defaultDetails = {
  1967. url: void 0,
  1968. timeout: 5000,
  1969. async: false,
  1970. responseType: void 0,
  1971. headers: void 0,
  1972. data: void 0,
  1973. redirect: void 0,
  1974. cookie: void 0,
  1975. binary: void 0,
  1976. nocache: void 0,
  1977. revalidate: void 0,
  1978. context: void 0,
  1979. overrideMimeType: void 0,
  1980. anonymous: void 0,
  1981. fetch: void 0,
  1982. fetchInit: void 0,
  1983. user: void 0,
  1984. password: void 0,
  1985. onabort() { },
  1986. onerror() { },
  1987. ontimeout() { },
  1988. onloadstart() { },
  1989. onreadystatechange() { },
  1990. onprogress() { },
  1991. };
  1992. /**
  1993. * 当前使用请求时,输出请求的配置
  1994. */
  1995. #LOG_DETAILS = false;
  1996. constructor(__xmlHttpRequest__) {
  1997. if (typeof __xmlHttpRequest__ !== "function") {
  1998. console.warn("Httpx未传入GM_xmlhttpRequest函数或传入的GM_xmlhttpRequest不是Function,强制使用window.fetch");
  1999. }
  2000. this.GM_Api.xmlHttpRequest = __xmlHttpRequest__;
  2001. }
  2002. /**
  2003. * 覆盖当前配置
  2004. * @param details
  2005. */
  2006. config(details = {}) {
  2007. if ("logDetails" in details && typeof details["logDetails"] === "boolean") {
  2008. this.#LOG_DETAILS = details["logDetails"];
  2009. }
  2010. this.#defaultDetails = utils.assign(this.#defaultDetails, details);
  2011. }
  2012. /**
  2013. * 修改xmlHttpRequest
  2014. * @param httpRequest 网络请求函数
  2015. */
  2016. setXMLHttpRequest(httpRequest) {
  2017. this.GM_Api.xmlHttpRequest = httpRequest;
  2018. }
  2019. /**
  2020. * GET 请求
  2021. */
  2022. async get(...args) {
  2023. let that = this;
  2024. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2025. return new Promise((resolve) => {
  2026. let requestDetails = that.HttpxRequestDetails.getDetails("get", resolve, details);
  2027. Reflect.deleteProperty(requestDetails, "onprogress");
  2028. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2029. that.HttpxRequest.request(requestDetails);
  2030. });
  2031. }
  2032. /**
  2033. * POST 请求
  2034. */
  2035. async post(...args) {
  2036. let that = this;
  2037. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2038. return new Promise((resolve) => {
  2039. let requestDetails = that.HttpxRequestDetails.getDetails("post", resolve, details);
  2040. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2041. that.HttpxRequest.request(requestDetails);
  2042. });
  2043. }
  2044. /**
  2045. * HEAD 请求
  2046. */
  2047. async head(...args) {
  2048. let that = this;
  2049. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2050. return new Promise((resolve) => {
  2051. let requestDetails = that.HttpxRequestDetails.getDetails("head", resolve, details);
  2052. Reflect.deleteProperty(requestDetails, "onprogress");
  2053. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2054. that.HttpxRequest.request(requestDetails);
  2055. });
  2056. }
  2057. /**
  2058. * OPTIONS 请求
  2059. */
  2060. async options(...args) {
  2061. let that = this;
  2062. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2063. return new Promise((resolve) => {
  2064. let requestDetails = that.HttpxRequestDetails.getDetails("options", resolve, details);
  2065. Reflect.deleteProperty(requestDetails, "onprogress");
  2066. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2067. that.HttpxRequest.request(requestDetails);
  2068. });
  2069. }
  2070. /**
  2071. * DELETE 请求
  2072. */
  2073. async delete(...args) {
  2074. let that = this;
  2075. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2076. return new Promise((resolve) => {
  2077. let requestDetails = that.HttpxRequestDetails.getDetails("delete", resolve, details);
  2078. Reflect.deleteProperty(requestDetails, "onprogress");
  2079. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2080. that.HttpxRequest.request(requestDetails);
  2081. });
  2082. }
  2083. /**
  2084. * PUT 请求
  2085. */
  2086. async put(...args) {
  2087. let that = this;
  2088. let details = this.HttpxRequestDetails.handleBeforeRequestDetails(...args);
  2089. return new Promise((resolve) => {
  2090. let requestDetails = that.HttpxRequestDetails.getDetails("put", resolve, details);
  2091. requestDetails = that.HttpxRequestDetails.handle(requestDetails);
  2092. that.HttpxRequest.request(requestDetails);
  2093. });
  2094. }
  2095. }
  2096.  
  2097. class indexedDB {
  2098. #dbName;
  2099. #storeName;
  2100. #dbVersion;
  2101. /* websql的版本号,由于ios的问题,版本号的写法不一样 */
  2102. #slqVersion = "1";
  2103. /* 监听IndexDB */
  2104. #indexedDB = window.indexedDB ||
  2105. window.mozIndexedDB ||
  2106. window.webkitIndexedDB ||
  2107. window.msIndexedDB;
  2108. /* 缓存数据库,避免同一个页面重复创建和销毁 */
  2109. #db = {};
  2110. #store = null;
  2111. #errorCode = {
  2112. /* 错误码 */
  2113. success: {
  2114. code: 200,
  2115. msg: "操作成功",
  2116. },
  2117. error: {
  2118. code: 401,
  2119. msg: "操作失败",
  2120. },
  2121. open: { code: 91001, msg: "打开数据库失败" },
  2122. save: { code: 91002, msg: "保存数据失败" },
  2123. get: { code: 91003, msg: "获取数据失败" },
  2124. delete: { code: 91004, msg: "删除数据失败" },
  2125. deleteAll: { code: 91005, msg: "清空数据库失败" },
  2126. regexpGet: { code: 91006, msg: "正则获取数据失败" },
  2127. };
  2128. /**
  2129. * @param dbName 数据存储名,默认为:default_db
  2130. * @param storeName 表名,默认为:default_form
  2131. * @param dbVersion indexDB的版本号,默认为:1
  2132. */
  2133. constructor(dbName = "default_db", storeName = "default_form", dbVersion = 1) {
  2134. this.#dbName = dbName;
  2135. this.#storeName = storeName;
  2136. this.#dbVersion = dbVersion;
  2137. if (!this.#indexedDB) {
  2138. alert("很抱歉,您的浏览器不支持indexedDB");
  2139. throw new TypeError("很抱歉,您的浏览器不支持indexedDB");
  2140. }
  2141. }
  2142. /**
  2143. * 创建 “表”
  2144. * @param dbName 表名
  2145. */
  2146. createStore(dbName) {
  2147. let txn, store;
  2148. txn = this.#db[dbName].transaction(this.#storeName, "readwrite");
  2149. /* IndexDB的读写权限 */
  2150. store = txn.objectStore(this.#storeName);
  2151. this.#store = store;
  2152. return store;
  2153. }
  2154. /**
  2155. * 打开数据库
  2156. * @param callback 回调
  2157. * @param dbName 数据库名
  2158. */
  2159. open(callback, dbName) {
  2160. let that = this;
  2161. /* 打开数据库 */
  2162. /* 如果支持IndexDB */
  2163. if (!that.#db[dbName]) {
  2164. /* 如果缓存中没有,则进行数据库的创建或打开,提高效率 */
  2165. let request = that.#indexedDB.open(dbName, that.#dbVersion);
  2166. request.onerror = function (event) {
  2167. callback({
  2168. code: that.#errorCode.open.code,
  2169. msg: that.#errorCode.open.msg,
  2170. event: event,
  2171. }, false);
  2172. };
  2173. request.onsuccess = function (event) {
  2174. if (!that.#db[dbName]) {
  2175. let target = event.target;
  2176. that.#db[dbName] = target.result;
  2177. }
  2178. let store = that.createStore(dbName);
  2179. callback(store, true);
  2180. };
  2181. request.onupgradeneeded = function (event) {
  2182. let target = event.target;
  2183. that.#db[dbName] = target.result;
  2184. let store = that.#db[dbName].createObjectStore(that.#storeName, {
  2185. keyPath: "key",
  2186. });
  2187. store.transaction.oncomplete = function (event) {
  2188. callback(store, true);
  2189. };
  2190. };
  2191. }
  2192. else {
  2193. /* 如果缓存中已经打开了数据库,就直接使用 */
  2194. let store = that.createStore(dbName);
  2195. callback(store, true);
  2196. }
  2197. }
  2198. /**
  2199. * 保存数据到数据库
  2200. * @param key 数据key
  2201. * @param value 数据值
  2202. */
  2203. async save(key, value) {
  2204. let that = this;
  2205. return new Promise((resolve) => {
  2206. let dbName = that.#dbName;
  2207. let inData = {
  2208. key: key,
  2209. value: value,
  2210. };
  2211. that.open(function (idbStore, success) {
  2212. if (!success) {
  2213. resolve({
  2214. success: false,
  2215. code: that.#errorCode.save.code,
  2216. msg: that.#errorCode.save.msg,
  2217. });
  2218. }
  2219. else {
  2220. idbStore = idbStore;
  2221. let request = idbStore.put(inData);
  2222. request.onsuccess = function (event) {
  2223. /* 保存成功有success 字段 */
  2224. event.target;
  2225. resolve({
  2226. success: true,
  2227. code: that.#errorCode.success.code,
  2228. msg: that.#errorCode.success.msg,
  2229. event: event,
  2230. });
  2231. };
  2232. request.onerror = function (event) {
  2233. event.target;
  2234. resolve({
  2235. success: false,
  2236. code: that.#errorCode.save.code,
  2237. msg: that.#errorCode.save.msg,
  2238. event: event,
  2239. });
  2240. };
  2241. }
  2242. }, dbName);
  2243. });
  2244. }
  2245. /**
  2246. * 根据key获取值
  2247. * @param key 数据key
  2248. */
  2249. async get(key) {
  2250. let that = this;
  2251. return new Promise((resolve) => {
  2252. let dbName = that.#dbName;
  2253. that.open(function (idbStore, success) {
  2254. /* 判断返回的数据中是否有error字段 */
  2255. if (!success) {
  2256. resolve({
  2257. success: false,
  2258. code: that.#errorCode.get.code,
  2259. msg: that.#errorCode.get.msg,
  2260. data: void 0,
  2261. });
  2262. }
  2263. else {
  2264. idbStore = idbStore;
  2265. let request = idbStore.get(key);
  2266. request.onsuccess = function (event) {
  2267. let target = event.target;
  2268. let result = target.result;
  2269. /* result 返回的是 {key: string, value: any} */
  2270. /* 键值对存储 */
  2271. let data = result ? result.value : void 0;
  2272. if (data) {
  2273. resolve({
  2274. success: true,
  2275. code: that.#errorCode.success.code,
  2276. msg: that.#errorCode.success.msg,
  2277. data: data,
  2278. event: event,
  2279. result: result,
  2280. });
  2281. }
  2282. else {
  2283. resolve({
  2284. success: false,
  2285. code: that.#errorCode.error.code,
  2286. msg: that.#errorCode.error.msg,
  2287. data: void 0,
  2288. event: event,
  2289. result: result,
  2290. });
  2291. }
  2292. };
  2293. request.onerror = function (event) {
  2294. event.target;
  2295. resolve({
  2296. success: false,
  2297. code: that.#errorCode.get.code,
  2298. msg: that.#errorCode.get.msg,
  2299. data: void 0,
  2300. event: event,
  2301. });
  2302. };
  2303. }
  2304. }, dbName);
  2305. });
  2306. }
  2307. /**
  2308. * 正则获取数据
  2309. * @param key 数据键
  2310. */
  2311. async regexpGet(key) {
  2312. let list = [];
  2313. let that = this;
  2314. return new Promise((resolve) => {
  2315. /* 正则查询 */
  2316. let dbName = that.#dbName;
  2317. that.open(function (idbStore, success) {
  2318. /* 判断返回的数据中是否有error字段 */
  2319. if (!success) {
  2320. resolve({
  2321. success: false,
  2322. code: that.#errorCode.regexpGet.code,
  2323. msg: that.#errorCode.regexpGet.msg,
  2324. data: [],
  2325. });
  2326. }
  2327. else {
  2328. idbStore = idbStore;
  2329. let request = idbStore.getAll();
  2330. request.onsuccess = function (event) {
  2331. let target = event.target;
  2332. let result = target.result;
  2333. if (result.length !== 0) {
  2334. result.forEach((item, index) => {
  2335. if (item["key"].match(key)) {
  2336. let concatList = item["value"];
  2337. concatList["key"] = item["key"];
  2338. list = [...list, concatList];
  2339. }
  2340. });
  2341. }
  2342. resolve({
  2343. success: true,
  2344. code: that.#errorCode.success.code,
  2345. msg: that.#errorCode.success.msg,
  2346. data: list,
  2347. event: event,
  2348. });
  2349. };
  2350. request.onerror = function (event) {
  2351. event.target;
  2352. resolve({
  2353. success: false,
  2354. code: that.#errorCode.get.code,
  2355. msg: that.#errorCode.get.msg,
  2356. data: [],
  2357. event: event,
  2358. });
  2359. };
  2360. }
  2361. }, dbName);
  2362. });
  2363. }
  2364. /**
  2365. * 删除数据
  2366. * @param {string} key 数据键
  2367. */
  2368. async delete(key) {
  2369. let that = this;
  2370. return new Promise((resolve) => {
  2371. /* 根据key删除某条数据 */
  2372. let dbName = that.#dbName;
  2373. that.open(function (idbStore, success) {
  2374. if (!success) {
  2375. resolve({
  2376. success: false,
  2377. code: that.#errorCode.delete.code,
  2378. msg: that.#errorCode.delete.msg,
  2379. });
  2380. }
  2381. else {
  2382. idbStore = idbStore;
  2383. let request = idbStore.get(key);
  2384. request.onsuccess = function (event) {
  2385. let target = event.target;
  2386. let recode = target.result;
  2387. if (recode) {
  2388. /* 成功 */
  2389. request = idbStore.delete(key);
  2390. resolve({
  2391. success: true,
  2392. code: that.#errorCode.success.code,
  2393. msg: that.#errorCode.success.msg,
  2394. });
  2395. }
  2396. else {
  2397. resolve({
  2398. success: false,
  2399. code: that.#errorCode.error.code,
  2400. msg: that.#errorCode.error.msg,
  2401. });
  2402. }
  2403. };
  2404. request.onerror = function (event) {
  2405. event.target;
  2406. resolve({
  2407. success: false,
  2408. code: that.#errorCode.delete.code,
  2409. msg: that.#errorCode.delete.msg,
  2410. event: event,
  2411. });
  2412. };
  2413. }
  2414. }, dbName);
  2415. });
  2416. }
  2417. /**
  2418. * 删除所有数据
  2419. */
  2420. async deleteAll() {
  2421. let that = this;
  2422. return new Promise((resolve) => {
  2423. /* 清空数据库 */
  2424. let dbName = that.#dbName;
  2425. that.open(function (idbStore, success) {
  2426. if (!success) {
  2427. resolve({
  2428. success: false,
  2429. code: that.#errorCode.deleteAll.code,
  2430. msg: that.#errorCode.deleteAll.msg,
  2431. });
  2432. }
  2433. else {
  2434. idbStore = idbStore;
  2435. idbStore.clear();
  2436. resolve({
  2437. success: true,
  2438. code: that.#errorCode.success.code,
  2439. msg: that.#errorCode.success.msg,
  2440. });
  2441. }
  2442. }, dbName);
  2443. });
  2444. }
  2445. }
  2446.  
  2447. class LockFunction {
  2448. #flag = false;
  2449. #delayTime = 0;
  2450. #callback;
  2451. #context;
  2452. lock;
  2453. unlock;
  2454. run;
  2455. isLock;
  2456. constructor(callback, context, delayTime) {
  2457. let that = this;
  2458. this.#callback = callback;
  2459. if (typeof context === "number") {
  2460. this.#delayTime = context;
  2461. this.#context = utils;
  2462. }
  2463. else {
  2464. this.#delayTime = delayTime;
  2465. this.#context = context;
  2466. }
  2467. /**
  2468. * 锁
  2469. */
  2470. this.lock = function () {
  2471. that.#flag = true;
  2472. };
  2473. /**
  2474. * 解锁
  2475. */
  2476. this.unlock = function () {
  2477. setTimeout(() => {
  2478. that.#flag = false;
  2479. }, that.#delayTime);
  2480. };
  2481. /**
  2482. * 判断是否被锁
  2483. */
  2484. this.isLock = function () {
  2485. return that.#flag;
  2486. };
  2487. /**
  2488. * 执行
  2489. */
  2490. this.run = async function (...args) {
  2491. if (that.isLock()) {
  2492. return;
  2493. }
  2494. that.lock();
  2495. await that.#callback.apply(that.#context, args);
  2496. that.unlock();
  2497. };
  2498. }
  2499. }
  2500.  
  2501. class Log {
  2502. /** 是否禁用输出的flag */
  2503. #disable = false;
  2504. /** 前面的TAG标志 */
  2505. tag = "Utils.Log";
  2506. /* 使用的console函数 */
  2507. #console = null;
  2508. /* 当前输出的数量 */
  2509. #logCount = 0;
  2510. /* 配置 */
  2511. #details = {
  2512. tag: true,
  2513. successColor: "#0000FF",
  2514. errorColor: "#FF0000",
  2515. infoColor: "0",
  2516. warnColor: "0",
  2517. debug: false,
  2518. autoClearConsole: false,
  2519. logMaxCount: 999,
  2520. };
  2521. #msgColorDetails = [
  2522. "font-weight: bold; color: cornflowerblue",
  2523. "font-weight: bold; color: cornflowerblue",
  2524. "font-weight: bold; color: darkorange",
  2525. "font-weight: bold; color: cornflowerblue",
  2526. ];
  2527. /**
  2528. * @param _GM_info_ 油猴管理器的API GM_info,或者是一个对象,如{"script":{name:"Utils.Log"}},或者直接是一个字符串
  2529. * @param console 可指定console对象为unsafeWindow下的console或者是油猴window下的console
  2530. */
  2531. constructor(_GM_info_, console = globalThis.console) {
  2532. if (typeof _GM_info_ === "string") {
  2533. this.tag = _GM_info_;
  2534. }
  2535. else if (typeof _GM_info_ === "object" &&
  2536. typeof _GM_info_?.script?.name === "string") {
  2537. this.tag = _GM_info_.script.name;
  2538. }
  2539. this.#console = console;
  2540. }
  2541. /**
  2542. * 解析Error的堆栈获取实际调用者的函数名及函数所在的位置
  2543. * @param stack
  2544. */
  2545. parseErrorStack(stack) {
  2546. let result = {
  2547. name: "",
  2548. position: "",
  2549. };
  2550. for (let stackString of stack) {
  2551. stackString = stackString.trim();
  2552. let stackFunctionNameMatch = stackString.match(/^at[\s]+(.+?)[\s]+/i);
  2553. let stackFunctionNamePositionMatch = stackString.match(/^at[\s]+.+[\s]+\((.+?)\)/i);
  2554. if (stackFunctionNameMatch == null) {
  2555. continue;
  2556. }
  2557. if (stackFunctionNamePositionMatch == null) {
  2558. continue;
  2559. }
  2560. /* 获取最后一个,因为第一个是包含了at */
  2561. let stackFunctionName = stackFunctionNameMatch[stackFunctionNameMatch.length - 1];
  2562. let stackFunctionNamePosition = stackFunctionNamePositionMatch[stackFunctionNamePositionMatch.length - 1];
  2563. if (stackFunctionName === "" ||
  2564. stackFunctionName.match(/^(Utils\.|)Log(\.|)|.<anonymous>$|^Function.each|^NodeList.forEach|^k.fn.init.each/g)) {
  2565. continue;
  2566. }
  2567. else {
  2568. result.name = stackFunctionName;
  2569. result.position = stackFunctionNamePosition;
  2570. break;
  2571. }
  2572. }
  2573. if (result.position === "") {
  2574. let lastStackString = stack[stack.length - 1].trim();
  2575. if (lastStackString.startsWith("at chrome-extension://")) {
  2576. let lastStackMatch = lastStackString.match(/^at[\s]+(.+)/);
  2577. if (lastStackMatch) {
  2578. result.position = lastStackMatch[lastStackMatch.length - 1];
  2579. }
  2580. }
  2581. }
  2582. if (result.position === "") {
  2583. result.position = stack[stack.length - 1].trim().replace(/^at[\s]*/g, "");
  2584. }
  2585. return result;
  2586. }
  2587. /**
  2588. * 检测清理控制台
  2589. */
  2590. checkClearConsole() {
  2591. this.#logCount++;
  2592. if (this.#details.autoClearConsole &&
  2593. this.#logCount > this.#details.logMaxCount) {
  2594. this.#console.clear();
  2595. this.#logCount = 0;
  2596. }
  2597. }
  2598. /**
  2599. * 输出内容
  2600. * @param msg 需要输出的内容
  2601. * @param color 颜色
  2602. * @param otherStyle 其它CSS
  2603. */
  2604. printContent(msg, color, otherStyle) {
  2605. this.checkClearConsole();
  2606. otherStyle = otherStyle || "";
  2607. let stackSplit = new Error().stack.split("\n");
  2608. stackSplit.splice(0, 2);
  2609. let { name: callerName, position: callerPosition } = this.parseErrorStack(stackSplit);
  2610. let tagName = this.tag;
  2611. let that = this;
  2612. function consoleMsg(_msg_) {
  2613. if (typeof _msg_ === "string") {
  2614. that.#console.log(`%c[${tagName}%c-%c${callerName}%c]%c %s`, ...that.#msgColorDetails, `color: ${color};${otherStyle}`, _msg_);
  2615. }
  2616. else if (typeof _msg_ === "number") {
  2617. that.#console.log(`%c[${tagName}%c-%c${callerName}%c]%c %d`, ...that.#msgColorDetails, `color: ${color};${otherStyle}`, _msg_);
  2618. }
  2619. else if (typeof _msg_ === "object") {
  2620. that.#console.log(`%c[${tagName}%c-%c${callerName}%c]%c %o`, ...that.#msgColorDetails, `color: ${color};${otherStyle}`, _msg_);
  2621. }
  2622. else {
  2623. that.#console.log(_msg_);
  2624. }
  2625. }
  2626. if (Array.isArray(msg)) {
  2627. msg.forEach((item) => {
  2628. consoleMsg(item);
  2629. });
  2630. }
  2631. else {
  2632. consoleMsg(msg);
  2633. }
  2634. if (this.#details.debug) {
  2635. /* 如果开启调试模式,输出堆栈位置 */
  2636. this.#console.log(callerPosition);
  2637. }
  2638. }
  2639. /**
  2640. * 控制台-普通输出
  2641. * @param msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  2642. * @param color 输出的颜色
  2643. * @param otherStyle 其它CSS
  2644. */
  2645. info(msg, color = this.#details.infoColor, otherStyle) {
  2646. if (this.#disable)
  2647. return;
  2648. this.printContent.call(this, msg, color, otherStyle);
  2649. }
  2650. /**
  2651. * 控制台-警告输出
  2652. * @param msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  2653. * @param color 输出的颜色
  2654. * @param otherStyle 其它CSS
  2655. */
  2656. warn(msg, color = this.#details.warnColor, otherStyle = "background: #FEF6D5;padding: 4px 6px 4px 0px;") {
  2657. if (this.#disable)
  2658. return;
  2659. this.printContent.call(this, msg, color, otherStyle);
  2660. }
  2661. /**
  2662. * 控制台-错误输出
  2663. * @param msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  2664. * @param color 输出的颜色
  2665. * @param otherStyle 其它CSS
  2666. */
  2667. error(msg, color = this.#details.errorColor, otherStyle) {
  2668. if (this.#disable)
  2669. return;
  2670. this.printContent.call(this, msg, color, otherStyle);
  2671. }
  2672. /**
  2673. * 控制台-成功输出
  2674. * @param msg 需要输出的内容,如果想输出多个,修改成数组,且数组内的长度最大值为4个
  2675. * @param color 输出的颜色
  2676. * @param otherStyle 其它CSS
  2677. */
  2678. success(msg, color = this.#details.successColor, otherStyle) {
  2679. if (this.#disable)
  2680. return;
  2681. this.printContent.call(this, msg, color, otherStyle);
  2682. }
  2683. /**
  2684. * 控制台-输出表格
  2685. * @param msg
  2686. * @param color 输出的颜色
  2687. * @param otherStyle 其它CSS
  2688. * @example
  2689. * log.table([{"名字":"example","值":"123"},{"名字":"example2","值":"345"}])
  2690. */
  2691. table(msg, color = this.#details.infoColor, otherStyle = "") {
  2692. if (this.#disable)
  2693. return;
  2694. this.checkClearConsole();
  2695. let stack = new Error().stack.split("\n");
  2696. stack.splice(0, 1);
  2697. let errorStackParse = this.parseErrorStack(stack);
  2698. let stackFunctionName = errorStackParse.name;
  2699. let stackFunctionNamePosition = errorStackParse.position;
  2700. let callerName = stackFunctionName;
  2701. this.#console.log(`%c[${this.tag}%c-%c${callerName}%c]%c`, ...this.#msgColorDetails, `color: ${color};${otherStyle}`);
  2702. this.#console.table(msg);
  2703. if (this.#details.debug) {
  2704. this.#console.log(stackFunctionNamePosition);
  2705. }
  2706. }
  2707. /**
  2708. * 配置Log对象的颜色
  2709. * @param paramDetails 配置信息
  2710. */
  2711. config(paramDetails) {
  2712. this.#details = Object.assign(this.#details, paramDetails);
  2713. }
  2714. /** 禁用输出 */
  2715. disable() {
  2716. this.#disable = true;
  2717. }
  2718. /** 恢复输出 */
  2719. recovery() {
  2720. this.#disable = false;
  2721. }
  2722. }
  2723.  
  2724. class Progress {
  2725. #config = {
  2726. /**
  2727. * canvas元素节点
  2728. */
  2729. canvasNode: null,
  2730. /**
  2731. * 绘制角度
  2732. */
  2733. deg: 95,
  2734. /**
  2735. * 进度
  2736. */
  2737. progress: 0,
  2738. /**
  2739. * 绘制的线宽度
  2740. */
  2741. lineWidth: 10,
  2742. /**
  2743. * 绘制的背景颜色
  2744. */
  2745. lineBgColor: "#1e637c",
  2746. /**
  2747. * 绘制的线的颜色
  2748. */
  2749. lineColor: "#25deff",
  2750. /**
  2751. * 绘制的字体颜色
  2752. */
  2753. textColor: "#000000",
  2754. /**
  2755. * 绘制的字体大小(px)
  2756. */
  2757. fontSize: 22,
  2758. /**
  2759. * 绘制的圆的半径
  2760. */
  2761. circleRadius: 50,
  2762. };
  2763. #ctx = null;
  2764. #width = null;
  2765. #height = null;
  2766. /**
  2767. *
  2768. * @param paramConfig 配置信息
  2769. */
  2770. constructor(paramConfig) {
  2771. this.#config = utils.assign(this.#config, paramConfig);
  2772. if (!(this.#config.canvasNode instanceof HTMLCanvasElement)) {
  2773. throw new Error("Utils.Progress 参数 canvasNode 必须是 HTMLCanvasElement");
  2774. }
  2775. this.init();
  2776. }
  2777. /**
  2778. * 初始化
  2779. */
  2780. init() {
  2781. /* 获取画笔 */
  2782. let ctx = this.#config.canvasNode.getContext("2d");
  2783. if (ctx == null) {
  2784. throw new Error("Utils.Progress 获取画笔失败");
  2785. }
  2786. this.#ctx = ctx;
  2787. /* 元素宽度 */
  2788. this.#width = this.#config.canvasNode.width;
  2789. /* 元素高度 */
  2790. this.#height = this.#config.canvasNode.height;
  2791. /* 清除锯齿 */
  2792. if (window.devicePixelRatio) {
  2793. this.#config.canvasNode.style.width = this.#width + "px";
  2794. this.#config.canvasNode.style.height = this.#height + "px";
  2795. this.#config.canvasNode.height = this.#height * window.devicePixelRatio;
  2796. this.#config.canvasNode.width = this.#width * window.devicePixelRatio;
  2797. this.#ctx.scale(window.devicePixelRatio, window.devicePixelRatio);
  2798. }
  2799. /* 设置线宽 */
  2800. this.#ctx.lineWidth = this.#config.lineWidth;
  2801. }
  2802. /**
  2803. * 绘制
  2804. */
  2805. draw() {
  2806. let degActive = (this.#config.progress * 360) / 100;
  2807. /* 清除画布 */
  2808. this.#ctx.clearRect(0, 0, this.#width, this.#height);
  2809. /* 开始绘制底圆 */
  2810. this.#ctx.beginPath();
  2811. this.#ctx.arc(this.#width / 2, this.#height / 2, this.#config.circleRadius, 1, 8);
  2812. this.#ctx.strokeStyle = this.#config.lineBgColor;
  2813. this.#ctx.stroke();
  2814. /* 开始绘制动态圆 */
  2815. this.#ctx.beginPath();
  2816. this.#ctx.arc(this.#width / 2, this.#height / 2, this.#config.circleRadius, -Math.PI / 2, (degActive * Math.PI) / 180 - Math.PI / 2);
  2817. this.#ctx.strokeStyle = this.#config.lineColor;
  2818. this.#ctx.stroke();
  2819. /* 获取百分比 */
  2820. let txt = parseInt(this.#config.progress.toString()) + "%";
  2821. this.#ctx.font = this.#config.fontSize + "px SimHei";
  2822. /* 获取文本宽度 */
  2823. let w = this.#ctx.measureText(txt).width;
  2824. let h = this.#config.fontSize / 2;
  2825. this.#ctx.fillStyle = this.#config.textColor;
  2826. this.#ctx.fillText(txt, this.#width / 2 - w / 2, this.#height / 2 + h / 2);
  2827. }
  2828. }
  2829.  
  2830. const TryCatch = function (...args) {
  2831. /* 定义变量和函数 */
  2832. let callbackFunction = null;
  2833. let context = null;
  2834. let handleError = (error) => { };
  2835. let defaultDetails = {
  2836. log: true,
  2837. };
  2838. const TryCatchCore = {
  2839. /**
  2840. *
  2841. * @param paramDetails 配置
  2842. * @returns
  2843. */
  2844. config(paramDetails) {
  2845. defaultDetails = utils.assign(defaultDetails, paramDetails);
  2846. return TryCatchCore;
  2847. },
  2848. /**
  2849. * 处理错误
  2850. * @param handler
  2851. */
  2852. error(handler) {
  2853. // @ts-ignore
  2854. handleError = handler;
  2855. return TryCatchCore;
  2856. },
  2857. /**
  2858. * 执行传入的函数并捕获其可能抛出的错误,并通过传入的错误处理函数进行处理。
  2859. * @param callback 待执行函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  2860. * @param __context__ 待执行函数的作用域,用于apply指定
  2861. * @returns 如果函数有返回值,则返回该返回值;否则返回 tryCatchObj 函数以支持链式调用。
  2862. * @throws {Error} 如果传入参数不符合要求,则会抛出相应类型的错误。
  2863. */
  2864. run(callback, __context__) {
  2865. callbackFunction = callback;
  2866. context = __context__ || this;
  2867. let result = executeTryCatch(callbackFunction, handleError, context);
  2868. // @ts-ignore
  2869. return result !== void 0 ? result : TryCatchCore;
  2870. },
  2871. };
  2872. /**
  2873. * 执行传入的函数并捕获其可能抛出的错误,并通过传入的错误处理函数进行处理。
  2874. * @param callback - 待执行函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  2875. * @param handleErrorFunc - 错误处理函数,可以是 function 或者 string 类型。如果是 string 类型,则会被当做代码进行执行。
  2876. * @param funcThis - 待执行函数的作用域,用于apply指定
  2877. * @returns 如果函数有返回值,则返回该返回值;否则返回 undefined。
  2878. */
  2879. function executeTryCatch(callback, handleErrorFunc, funcThis) {
  2880. let result = void 0;
  2881. try {
  2882. if (typeof callback === "string") {
  2883. (function () {
  2884. eval(callback);
  2885. }).apply(funcThis, args);
  2886. }
  2887. else {
  2888. result = callback.apply(funcThis, args);
  2889. }
  2890. }
  2891. catch (error) {
  2892. if (defaultDetails.log) {
  2893. callback = callback;
  2894. console.log(`%c ${callback?.name ? callback?.name : callback + "出现错误"} `, "color: #f20000");
  2895. console.log(`%c 错误原因:${error}`, "color: #f20000");
  2896. console.trace(callback);
  2897. }
  2898. if (handleErrorFunc) {
  2899. if (typeof handleErrorFunc === "string") {
  2900. result = function () {
  2901. return eval(handleErrorFunc);
  2902. // @ts-ignore
  2903. }.apply(funcThis, [...args, error]);
  2904. }
  2905. else {
  2906. result = handleErrorFunc.apply(funcThis, [...args, error]);
  2907. }
  2908. }
  2909. }
  2910. return result;
  2911. }
  2912. return TryCatchCore;
  2913. };
  2914.  
  2915. class UtilsDictionary {
  2916. #items = {};
  2917. constructor() { }
  2918. /**
  2919. * 检查是否有某一个键
  2920. * @param key 键
  2921. */
  2922. has(key) {
  2923. return this.#items.hasOwnProperty(key);
  2924. }
  2925. /**
  2926. * 检查已有的键中是否以xx开头
  2927. * @param key 需要匹配的键
  2928. */
  2929. startsWith(key) {
  2930. let allKeys = this.keys();
  2931. for (const keyName of allKeys) {
  2932. if (keyName.startsWith(key)) {
  2933. return true;
  2934. }
  2935. }
  2936. return false;
  2937. }
  2938. /**
  2939. * 获取以xx开头的键的值
  2940. * @param key 需要匹配的键
  2941. */
  2942. getStartsWith(key) {
  2943. let allKeys = this.keys();
  2944. let result = null;
  2945. for (const keyName of allKeys) {
  2946. if (keyName.startsWith(key)) {
  2947. result = this.#items[keyName];
  2948. break;
  2949. }
  2950. }
  2951. return result;
  2952. }
  2953. /**
  2954. * 为字典添加某一个值
  2955. * @param key 键
  2956. * @param val 值,默认为""
  2957. */
  2958. set(key, val) {
  2959. if (key === void 0) {
  2960. throw new Error("Utils.Dictionary().set 参数 key 不能为空");
  2961. }
  2962. this.#items[key] = val;
  2963. }
  2964. /**
  2965. * 删除某一个键
  2966. * @param key 键
  2967. */
  2968. delete(key) {
  2969. if (this.has(key)) {
  2970. Reflect.deleteProperty(this.#items, key);
  2971. return true;
  2972. }
  2973. return false;
  2974. }
  2975. /**
  2976. * 获取某个键的值
  2977. * @param key 键
  2978. */
  2979. get(key) {
  2980. return this.has(key) ? this.getItems()[key] : void 0;
  2981. }
  2982. /**
  2983. * 返回字典中的所有值
  2984. */
  2985. values() {
  2986. let resultList = [];
  2987. for (let prop in this.getItems()) {
  2988. if (this.has(prop)) {
  2989. resultList.push(this.getItems()[prop]);
  2990. }
  2991. }
  2992. return resultList;
  2993. }
  2994. /**
  2995. * 清空字典
  2996. */
  2997. clear() {
  2998. this.#items = void 0;
  2999. this.#items = {};
  3000. }
  3001. /**
  3002. * 获取字典的长度
  3003. */
  3004. size() {
  3005. return Object.keys(this.getItems()).length;
  3006. }
  3007. /**
  3008. * 获取字典所有的键
  3009. */
  3010. keys() {
  3011. return Object.keys(this.getItems());
  3012. }
  3013. /**
  3014. * 返回字典本身
  3015. */
  3016. getItems() {
  3017. return this.#items;
  3018. }
  3019. /**
  3020. * 合并另一个字典
  3021. * @param data 需要合并的字典
  3022. */
  3023. concat(data) {
  3024. this.#items = utils.assign(this.#items, data.getItems());
  3025. }
  3026. forEach(callbackfn) {
  3027. for (const key in this.getItems()) {
  3028. callbackfn(this.get(key), key, this.getItems());
  3029. }
  3030. }
  3031. /**
  3032. * 获取字典的长度,同this.size
  3033. */
  3034. get length() {
  3035. return this.size();
  3036. }
  3037. /**
  3038. * 迭代器
  3039. */
  3040. get entries() {
  3041. let that = this;
  3042. return function* () {
  3043. let itemKeys = Object.keys(that.getItems());
  3044. for (const keyName of itemKeys) {
  3045. yield [keyName, that.get(keyName)];
  3046. }
  3047. };
  3048. }
  3049. /**
  3050. * 是否可遍历
  3051. */
  3052. get [Symbol.iterator]() {
  3053. let that = this;
  3054. return function () {
  3055. return that.entries();
  3056. };
  3057. }
  3058. }
  3059.  
  3060. /// <reference path="./ajaxHooker/index.d.ts" />
  3061. class Utils {
  3062. constructor(option) {
  3063. UtilsCore.init(option);
  3064. }
  3065. /** 版本号 */
  3066. version = "2024.6.11";
  3067. addStyle(cssText) {
  3068. if (typeof cssText !== "string") {
  3069. throw new Error("Utils.addStyle 参数cssText 必须为String类型");
  3070. }
  3071. let cssNode = document.createElement("style");
  3072. cssNode.setAttribute("type", "text/css");
  3073. cssNode.innerHTML = cssText;
  3074. if (document.head) {
  3075. /* 插入head最后 */
  3076. document.head.appendChild(cssNode);
  3077. }
  3078. else if (document.body) {
  3079. /* 插入body后 */
  3080. document.body.appendChild(cssNode);
  3081. }
  3082. else if (document.documentElement.childNodes.length === 0) {
  3083. /* 插入#html第一个元素后 */
  3084. document.documentElement.appendChild(cssNode);
  3085. }
  3086. else {
  3087. /* 插入head前面 */
  3088. document.documentElement.insertBefore(cssNode, document.documentElement.childNodes[0]);
  3089. }
  3090. return cssNode;
  3091. }
  3092. assign(target = {}, source = {}, isAdd = false) {
  3093. let UtilsContext = this;
  3094. if (Array.isArray(source)) {
  3095. let canTraverse = source.filter((item) => {
  3096. return typeof item === "object";
  3097. });
  3098. if (!canTraverse.length) {
  3099. return source;
  3100. }
  3101. }
  3102. if (isAdd) {
  3103. for (const sourceKeyName in source) {
  3104. const targetKeyName = sourceKeyName;
  3105. let targetValue = target[targetKeyName];
  3106. let sourceValue = source[sourceKeyName];
  3107. if (sourceKeyName in target &&
  3108. typeof sourceValue === "object" &&
  3109. !UtilsContext.isDOM(sourceValue)) {
  3110. /* 源端的值是object类型,且不是元素节点 */
  3111. target[sourceKeyName] = UtilsContext.assign(targetValue, sourceValue, isAdd);
  3112. continue;
  3113. }
  3114. target[sourceKeyName] = sourceValue;
  3115. }
  3116. }
  3117. else {
  3118. for (const targetKeyName in target) {
  3119. if (targetKeyName in source) {
  3120. let targetValue = target[targetKeyName];
  3121. let sourceValue = source[targetKeyName];
  3122. if (typeof sourceValue === "object" &&
  3123. !UtilsContext.isDOM(sourceValue) &&
  3124. Object.keys(sourceValue).length) {
  3125. /* 源端的值是object类型,且不是元素节点 */
  3126. target[targetKeyName] = UtilsContext.assign(targetValue, sourceValue, isAdd);
  3127. continue;
  3128. }
  3129. /* 直接赋值 */
  3130. target[targetKeyName] = sourceValue;
  3131. }
  3132. }
  3133. }
  3134. return target;
  3135. }
  3136. async asyncReplaceAll(string, pattern, asyncFn) {
  3137. let UtilsContext = this;
  3138. if (typeof string !== "string") {
  3139. throw new TypeError("string必须是字符串");
  3140. }
  3141. if (typeof asyncFn !== "function") {
  3142. throw new TypeError("asyncFn必须是函数");
  3143. }
  3144. let reg;
  3145. if (typeof pattern === "string") {
  3146. reg = new RegExp(UtilsContext.parseStringToRegExpString(pattern), "g");
  3147. }
  3148. else if (pattern instanceof RegExp) {
  3149. if (!pattern.global) {
  3150. throw new TypeError("pattern必须是全局匹配");
  3151. }
  3152. reg = new RegExp(pattern);
  3153. }
  3154. else {
  3155. throw new TypeError("pattern必须是正则对象");
  3156. }
  3157. let result = [];
  3158. let match;
  3159. let lastIndex = 0;
  3160. while ((match = reg.exec(string)) !== null) {
  3161. /* 异步获取匹配对应的字符串 */
  3162. const item = asyncFn(match[0]);
  3163. /* 获取该匹配项和上一个匹配项的中间的字符串 */
  3164. const prefix = string.slice(lastIndex, match.index);
  3165. lastIndex = match.index + match[0].length;
  3166. result.push(item);
  3167. result.push(prefix);
  3168. }
  3169. result.push(string.slice(lastIndex));
  3170. /* 等待所有异步完成 */
  3171. result = await Promise.all(result);
  3172. return result.join("");
  3173. }
  3174. /**
  3175. * ajax劫持库,支持xhr和fetch劫持。
  3176. * + 来源:https://bbs.tampermonkey.net.cn/thread-3284-1-1.html
  3177. * + 作者:cxxjackie
  3178. * + 版本:1.4.1
  3179. * + 文档:https://scriptcat.org/zh-CN/script-show-page/637/
  3180. */
  3181. ajaxHooker = AjaxHooker;
  3182. canvasClickByPosition(canvasElement, clientX = 0, clientY = 0, view = globalThis) {
  3183. if (!(canvasElement instanceof HTMLCanvasElement)) {
  3184. throw new Error("Utils.canvasClickByPosition 参数canvasElement必须是canvas元素");
  3185. }
  3186. clientX = parseInt(clientX.toString());
  3187. clientY = parseInt(clientY.toString());
  3188. const eventInit = {
  3189. cancelBubble: true,
  3190. cancelable: true,
  3191. clientX: clientX,
  3192. clientY: clientY,
  3193. // @ts-ignore
  3194. view: view,
  3195. detail: 1,
  3196. };
  3197. canvasElement.dispatchEvent(new MouseEvent("mousedown", eventInit));
  3198. canvasElement.dispatchEvent(new MouseEvent("mouseup", eventInit));
  3199. }
  3200. checkUserClickInNode(element) {
  3201. let UtilsContext = this;
  3202. if (!UtilsContext.isDOM(element)) {
  3203. throw new Error("Utils.checkUserClickInNode 参数 targetNode 必须为 Element|Node 类型");
  3204. }
  3205. let mouseClickPosX = Number(window.event.clientX.toString()); /* 鼠标相对屏幕横坐标 */
  3206. let mouseClickPosY = Number(window.event.clientY.toString()); /* 鼠标相对屏幕纵坐标 */
  3207. let elementPosXLeft = Number(element.getBoundingClientRect().left); /* 要检测的元素的相对屏幕的横坐标最左边 */
  3208. let elementPosXRight = Number(element.getBoundingClientRect().right); /* 要检测的元素的相对屏幕的横坐标最右边 */
  3209. let elementPosYTop = Number(element.getBoundingClientRect().top); /* 要检测的元素的相对屏幕的纵坐标最上边 */
  3210. let elementPosYBottom = Number(element.getBoundingClientRect().bottom); /* 要检测的元素的相对屏幕的纵坐标最下边 */
  3211. let clickNodeHTML = window.event.target.innerHTML;
  3212. if (mouseClickPosX >= elementPosXLeft &&
  3213. mouseClickPosX <= elementPosXRight &&
  3214. mouseClickPosY >= elementPosYTop &&
  3215. mouseClickPosY <= elementPosYBottom) {
  3216. return true;
  3217. }
  3218. else if (clickNodeHTML &&
  3219. element.innerHTML.includes(clickNodeHTML)) {
  3220. /* 这种情况是应对在界面中隐藏的元素,getBoundingClientRect获取的都是0 */
  3221. return true;
  3222. }
  3223. else {
  3224. return false;
  3225. }
  3226. }
  3227. cloneFormData(formData) {
  3228. let clonedFormData = new FormData();
  3229. for (let [key, value] of formData.entries()) {
  3230. clonedFormData.append(key, value);
  3231. }
  3232. return clonedFormData;
  3233. }
  3234. createOverload() {
  3235. let fnMap = new Map();
  3236. function overload(...args) {
  3237. let key = args.map((it) => typeof it).join(",");
  3238. let fn = fnMap.get(key);
  3239. if (!fn) {
  3240. throw new TypeError("没有找到对应的实现");
  3241. }
  3242. return fn.apply(this, args);
  3243. }
  3244. overload.addImpl = function (...args) {
  3245. let fn = args.pop();
  3246. if (typeof fn !== "function") {
  3247. throw new TypeError("最后一个参数必须是函数");
  3248. }
  3249. let key = args.join(",");
  3250. fnMap.set(key, fn);
  3251. };
  3252. return overload;
  3253. }
  3254. /**
  3255. * 颜色转换
  3256. * @returns
  3257. */
  3258. ColorConversion = ColorConversion;
  3259. deepClone(obj) {
  3260. let UtilsContext = this;
  3261. if (obj === void 0)
  3262. return void 0;
  3263. if (obj === null)
  3264. return null;
  3265. let clone = obj instanceof Array ? [] : {};
  3266. for (const [key, value] of Object.entries(obj)) {
  3267. clone[key] =
  3268. typeof value === "object" ? UtilsContext.deepClone(value) : value;
  3269. }
  3270. return clone;
  3271. }
  3272. debounce(fn, delay = 0) {
  3273. let timer = null;
  3274. const context = this;
  3275. return function (...args) {
  3276. clearTimeout(timer);
  3277. timer = setTimeout(function () {
  3278. fn.apply(context, args);
  3279. }, delay);
  3280. };
  3281. }
  3282. deleteParentNode(element, targetSelector) {
  3283. let UtilsContext = this;
  3284. if (element == null) {
  3285. return;
  3286. }
  3287. if (!UtilsContext.isDOM(element)) {
  3288. throw new Error("Utils.deleteParentNode 参数 target 必须为 Node|HTMLElement 类型");
  3289. }
  3290. if (typeof targetSelector !== "string") {
  3291. throw new Error("Utils.deleteParentNode 参数 targetSelector 必须为 string 类型");
  3292. }
  3293. let result = false;
  3294. let needRemoveDOM = element.closest(targetSelector);
  3295. if (needRemoveDOM) {
  3296. needRemoveDOM.remove();
  3297. result = true;
  3298. }
  3299. return result;
  3300. }
  3301. /**
  3302. * 字典
  3303. * @example
  3304. * let dictionary = new Utils.Dictionary();
  3305. * let dictionary2 = new Utils.Dictionary();
  3306. * dictionary.set("test","111");
  3307. * dictionary.get("test");
  3308. * > '111'
  3309. * dictionary.has("test");
  3310. * > true
  3311. * dictionary.concat(dictionary2);
  3312. **/
  3313. Dictionary = UtilsDictionary;
  3314. dispatchEvent(element, eventName, details) {
  3315. let eventNameList = [];
  3316. if (typeof eventName === "string") {
  3317. eventNameList = [eventName];
  3318. }
  3319. if (Array.isArray(eventName)) {
  3320. eventNameList = [...eventName];
  3321. }
  3322. eventNameList.forEach((_eventName_) => {
  3323. let event = new Event(_eventName_);
  3324. if (details) {
  3325. Object.assign(event, details);
  3326. }
  3327. element.dispatchEvent(event);
  3328. });
  3329. }
  3330. downloadBase64(base64Data, fileName, isIFrame = false) {
  3331. if (typeof base64Data !== "string") {
  3332. throw new Error("Utils.downloadBase64 参数 base64Data 必须为 string 类型");
  3333. }
  3334. if (typeof fileName !== "string") {
  3335. throw new Error("Utils.downloadBase64 参数 fileName 必须为 string 类型");
  3336. }
  3337. if (isIFrame) {
  3338. /* 使用iframe */
  3339. const iframeElement = document.createElement("iframe");
  3340. iframeElement.style.display = "none";
  3341. iframeElement.src = base64Data;
  3342. document.body.appendChild(iframeElement);
  3343. setTimeout(() => {
  3344. iframeElement.contentWindow.document.execCommand("SaveAs", true, fileName);
  3345. document.body.removeChild(iframeElement);
  3346. }, 100);
  3347. }
  3348. else {
  3349. /* 使用A标签 */
  3350. const linkElement = document.createElement("a");
  3351. linkElement.setAttribute("target", "_blank");
  3352. linkElement.download = fileName;
  3353. linkElement.href = base64Data;
  3354. linkElement.click();
  3355. }
  3356. }
  3357. findWebPageVisibleText(str = "", caseSensitive = false) {
  3358. let TRange = null;
  3359. let strFound;
  3360. if (UtilsCore.globalThis.find) {
  3361. /* CODE FOR BROWSERS THAT SUPPORT window.find */
  3362. let windowFind = UtilsCore.self.find;
  3363. strFound = windowFind(str, caseSensitive, true, true, false);
  3364. if (strFound && self.getSelection && !self.getSelection().anchorNode) {
  3365. strFound = windowFind(str, caseSensitive, true, true, false);
  3366. }
  3367. if (!strFound) {
  3368. strFound = windowFind(str, 0, 1);
  3369. while (windowFind(str, 0, 1))
  3370. continue;
  3371. }
  3372. }
  3373. else if (navigator.appName.indexOf("Microsoft") != -1) {
  3374. /* EXPLORER-SPECIFIC CODE */
  3375. if (TRange != null) {
  3376. TRange = TRange;
  3377. TRange.collapse(false);
  3378. strFound = TRange.findText(str);
  3379. if (strFound)
  3380. TRange.select();
  3381. }
  3382. if (TRange == null || strFound == 0) {
  3383. TRange = UtilsCore.self.document.body.createTextRange();
  3384. strFound = TRange.findText(str);
  3385. if (strFound)
  3386. TRange.select();
  3387. }
  3388. }
  3389. else if (navigator.appName == "Opera") {
  3390. alert("Opera browsers not supported, sorry...");
  3391. return;
  3392. }
  3393. return strFound ? true : false;
  3394. }
  3395. *findElementsWithText(element, text, filter) {
  3396. let that = this;
  3397. if (element.outerHTML.includes(text)) {
  3398. if (element.children.length === 0) {
  3399. let filterResult = typeof filter === "function" ? filter(element) : false;
  3400. if (!filterResult) {
  3401. yield element;
  3402. }
  3403. }
  3404. else {
  3405. let textElement = Array.from(element.childNodes).filter((ele) => ele.nodeType === Node.TEXT_NODE);
  3406. for (let ele of textElement) {
  3407. if (ele.textContent.includes(text)) {
  3408. let filterResult = typeof filter === "function" ? filter(element) : false;
  3409. if (!filterResult) {
  3410. yield ele;
  3411. }
  3412. }
  3413. }
  3414. }
  3415. }
  3416. for (let index = 0; index < element.children.length; index++) {
  3417. let childElement = element.children[index];
  3418. yield* that.findElementsWithText(childElement, text, filter);
  3419. }
  3420. }
  3421. /**
  3422. * 判断该元素是否可见,如果不可见,向上找它的父元素直至找到可见的元素
  3423. * @param element
  3424. * @example
  3425. * let visibleElement = Utils.findVisibleElement(document.querySelector("a.xx"));
  3426. * > <HTMLElement>
  3427. */
  3428. findVisibleElement(element) {
  3429. let currentElement = element;
  3430. while (currentElement) {
  3431. let elementRect = currentElement.getBoundingClientRect();
  3432. if (Boolean(elementRect.length)) {
  3433. return currentElement;
  3434. }
  3435. currentElement = currentElement.parentElement;
  3436. }
  3437. return null;
  3438. }
  3439. formatByteToSize(byteSize, addType = true) {
  3440. byteSize = parseInt(byteSize.toString());
  3441. if (isNaN(byteSize)) {
  3442. throw new Error("Utils.formatByteToSize 参数 byteSize 格式不正确");
  3443. }
  3444. let result = 0;
  3445. let resultType = "KB";
  3446. let sizeData = {};
  3447. sizeData.B = 1;
  3448. sizeData.KB = 1024;
  3449. sizeData.MB = sizeData.KB * sizeData.KB;
  3450. sizeData.GB = sizeData.MB * sizeData.KB;
  3451. sizeData.TB = sizeData.GB * sizeData.KB;
  3452. sizeData.PB = sizeData.TB * sizeData.KB;
  3453. sizeData.EB = sizeData.PB * sizeData.KB;
  3454. sizeData.ZB = sizeData.EB * sizeData.KB;
  3455. sizeData.YB = sizeData.ZB * sizeData.KB;
  3456. sizeData.BB = sizeData.YB * sizeData.KB;
  3457. sizeData.NB = sizeData.BB * sizeData.KB;
  3458. sizeData.DB = sizeData.NB * sizeData.KB;
  3459. for (let key in sizeData) {
  3460. result = byteSize / sizeData[key];
  3461. resultType = key;
  3462. if (sizeData.KB >= result) {
  3463. break;
  3464. }
  3465. }
  3466. result = result.toFixed(2);
  3467. result = addType
  3468. ? result + resultType.toString()
  3469. : parseFloat(result.toString());
  3470. return result;
  3471. }
  3472. getNodeListValue(...args) {
  3473. let resultArray = [];
  3474. for (let arg of args) {
  3475. let value = arg;
  3476. if (typeof arg === "function") {
  3477. /* 方法 */
  3478. value = arg();
  3479. }
  3480. if (value.length !== 0) {
  3481. resultArray = [...value];
  3482. break;
  3483. }
  3484. }
  3485. return resultArray;
  3486. }
  3487. getNonNullValue(...args) {
  3488. let resultValue = args[args.length - 1];
  3489. let UtilsContext = this;
  3490. for (const argValue of args) {
  3491. if (UtilsContext.isNotNull(argValue)) {
  3492. resultValue = argValue;
  3493. break;
  3494. }
  3495. }
  3496. return resultValue;
  3497. }
  3498. formatTime(text = new Date(), formatType = "yyyy-MM-dd HH:mm:ss") {
  3499. let time = text == null ? new Date() : new Date(text);
  3500. /**
  3501. * 校验时间补0
  3502. * @param timeNum
  3503. * @returns
  3504. */
  3505. function checkTime(timeNum) {
  3506. if (timeNum < 10)
  3507. return "0" + timeNum;
  3508. return timeNum;
  3509. }
  3510. /**
  3511. * 时间制修改 24小时制转12小时制
  3512. * @param hourNum 小时
  3513. * @returns
  3514. */
  3515. function timeSystemChange(hourNum) {
  3516. return hourNum > 12 ? hourNum - 12 : hourNum;
  3517. }
  3518. let timeRegexp = {
  3519. yyyy: time.getFullYear(),
  3520. /* 年 */
  3521. MM: checkTime(time.getMonth() + 1),
  3522. /* 月 */
  3523. dd: checkTime(time.getDate()),
  3524. /* 日 */
  3525. HH: checkTime(time.getHours()),
  3526. /* 时 (24小时制) */
  3527. hh: checkTime(timeSystemChange(time.getHours())),
  3528. /* 时 (12小时制) */
  3529. mm: checkTime(time.getMinutes()),
  3530. /* 分 */
  3531. ss: checkTime(time.getSeconds()),
  3532. /* 秒 */
  3533. };
  3534. Object.keys(timeRegexp).forEach(function (key) {
  3535. let replaecRegexp = new RegExp(key, "g");
  3536. formatType = formatType.replace(replaecRegexp, timeRegexp[key]);
  3537. });
  3538. return formatType;
  3539. }
  3540. formatToTimeStamp(text) {
  3541. /* 把字符串格式的时间(完整,包括日期和时间)格式化成时间 */
  3542. if (typeof text !== "string") {
  3543. throw new Error("Utils.formatToTimeStamp 参数 text 必须为 string 类型");
  3544. }
  3545. if (text.length === 8) {
  3546. /* 该字符串只有时分秒 */
  3547. let today = new Date();
  3548. text =
  3549. today.getFullYear() +
  3550. "-" +
  3551. (today.getMonth() + 1) +
  3552. "-" +
  3553. today.getDate() +
  3554. " " +
  3555. text;
  3556. }
  3557. text = text.substring(0, 19);
  3558. text = text.replace(/-/g, "/");
  3559. let timestamp = new Date(text).getTime();
  3560. return timestamp;
  3561. }
  3562. /**
  3563. * gbk格式的url编码,来自https://gf.qytechs.cn/zh-CN/scripts/427726-gbk-url-js
  3564. * @example
  3565. * let gbkEncoder = new Utils.GBKEncoder();
  3566. * gbkEncoder.encode("测试");
  3567. * > '%B2%E2%CA%D4'
  3568. * gbkEncoder.decode("%B2%E2%CA%D4");
  3569. * > 测试
  3570. */
  3571. GBKEncoder = GBKEncoder;
  3572. /**
  3573. * 获取 transitionend 的在各个浏览器的兼容名
  3574. */
  3575. getTransitionEndNameList() {
  3576. return [
  3577. "webkitTransitionEnd",
  3578. "mozTransitionEnd",
  3579. "MSTransitionEnd",
  3580. "otransitionend",
  3581. "transitionend",
  3582. ];
  3583. }
  3584. /**
  3585. * 获取 animationend 的在各个浏览器的兼容名
  3586. */
  3587. getAnimationEndNameList() {
  3588. return [
  3589. "webkitAnimationEnd",
  3590. "mozAnimationEnd",
  3591. "MSAnimationEnd",
  3592. "oanimationend",
  3593. "animationend",
  3594. ];
  3595. }
  3596. getArrayLastValue(targetObj) {
  3597. return targetObj[targetObj.length - 1];
  3598. }
  3599. getArrayRealValue(...args) {
  3600. let result = null;
  3601. for (let arg of args) {
  3602. if (typeof arg === "function") {
  3603. /* 方法 */
  3604. arg = arg();
  3605. }
  3606. if (arg != null) {
  3607. result = arg;
  3608. break;
  3609. }
  3610. }
  3611. return result;
  3612. }
  3613. getDaysDifference(timestamp1 = Date.now(), timestamp2 = Date.now(), type = "天") {
  3614. type = type.trim();
  3615. if (timestamp1.toString().length === 10) {
  3616. timestamp1 = timestamp1 * 1000;
  3617. }
  3618. if (timestamp2.toString().length === 10) {
  3619. timestamp2 = timestamp2 * 1000;
  3620. }
  3621. let smallTimeStamp = timestamp1 > timestamp2 ? timestamp2 : timestamp1;
  3622. let bigTimeStamp = timestamp1 > timestamp2 ? timestamp1 : timestamp2;
  3623. let oneSecond = 1000; /* 一秒的毫秒数 */
  3624. let oneMinute = 60 * oneSecond; /* 一分钟的毫秒数 */
  3625. let oneHour = 60 * oneMinute; /* 一小时的毫秒数 */
  3626. let oneDay = 24 * oneHour; /* 一天的毫秒数 */
  3627. let oneMonth = 30 * oneDay; /* 一个月的毫秒数(30天) */
  3628. let oneYear = 12 * oneMonth; /* 一年的毫秒数 */
  3629. let bigDate = new Date(bigTimeStamp);
  3630. let smallDate = new Date(smallTimeStamp);
  3631. let remainderValue = 1;
  3632. if (type === "年") {
  3633. remainderValue = oneYear;
  3634. }
  3635. else if (type === "月") {
  3636. remainderValue = oneMonth;
  3637. }
  3638. else if (type === "天") {
  3639. remainderValue = oneDay;
  3640. }
  3641. else if (type === "时") {
  3642. remainderValue = oneHour;
  3643. }
  3644. else if (type === "分") {
  3645. remainderValue = oneMinute;
  3646. }
  3647. else if (type === "秒") {
  3648. remainderValue = oneSecond;
  3649. }
  3650. let diffValue = Math.round(Math.abs((bigDate - smallDate) / remainderValue));
  3651. if (type === "auto") {
  3652. let timeDifference = bigTimeStamp - smallTimeStamp;
  3653. diffValue = Math.floor(timeDifference / (24 * 3600 * 1000));
  3654. if (diffValue > 0) {
  3655. diffValue = diffValue + "天";
  3656. }
  3657. else {
  3658. /* 计算出小时数 */
  3659. let leave1 = timeDifference % (24 * 3600 * 1000); /* 计算天数后剩余的毫秒数 */
  3660. let hours = Math.floor(leave1 / (3600 * 1000));
  3661. if (hours > 0) {
  3662. diffValue = hours + "小时";
  3663. }
  3664. else {
  3665. /* 计算相差分钟数 */
  3666. let leave2 = leave1 % (3600 * 1000); /* 计算小时数后剩余的毫秒数 */
  3667. let minutes = Math.floor(leave2 / (60 * 1000));
  3668. if (minutes > 0) {
  3669. diffValue = minutes + "分钟";
  3670. }
  3671. else {
  3672. /* 计算相差秒数 */
  3673. let leave3 = leave2 % (60 * 1000); /* 计算分钟数后剩余的毫秒数 */
  3674. let seconds = Math.round(leave3 / 1000);
  3675. diffValue = seconds + "秒";
  3676. }
  3677. }
  3678. }
  3679. }
  3680. return diffValue;
  3681. }
  3682. getElementSelector(element) {
  3683. let UtilsContext = this;
  3684. // @ts-ignore
  3685. if (!element)
  3686. return;
  3687. // @ts-ignore
  3688. if (!element.parentElement)
  3689. return;
  3690. /* 如果元素有id属性,则直接返回id选择器 */
  3691. if (element.id)
  3692. return "#" + element.id;
  3693. /* 递归地获取父元素的选择器 */
  3694. let selector = UtilsContext.getElementSelector(element.parentElement);
  3695. if (!selector) {
  3696. return element.tagName.toLowerCase();
  3697. }
  3698. /* 如果有多个相同类型的兄弟元素,则需要添加索引 */
  3699. if (element.parentElement.querySelectorAll(element.tagName).length > 1) {
  3700. let index = Array.prototype.indexOf.call(element.parentElement.children, element) +
  3701. 1;
  3702. selector +=
  3703. " > " + element.tagName.toLowerCase() + ":nth-child(" + index + ")";
  3704. }
  3705. else {
  3706. selector += " > " + element.tagName.toLowerCase();
  3707. }
  3708. return selector;
  3709. }
  3710. /**
  3711. * 获取最大值
  3712. * @example
  3713. * Utils.getMaxValue([{1:123},{2:345},{3:456}],(index,value)=>{return parseInt(index)})
  3714. * > 2
  3715. */
  3716. getMaxValue(...args) {
  3717. let result = [...args];
  3718. let newResult = [];
  3719. if (result.length === 0) {
  3720. // @ts-ignore
  3721. return;
  3722. }
  3723. if (result.length > 1) {
  3724. if (result.length === 2 &&
  3725. typeof result[0] === "object" &&
  3726. typeof result[1] === "function") {
  3727. let data = result[0];
  3728. let handleDataFunc = result[1];
  3729. Object.keys(data).forEach((keyName) => {
  3730. newResult = [...newResult, handleDataFunc(keyName, data[keyName])];
  3731. });
  3732. }
  3733. else {
  3734. result.forEach((item) => {
  3735. if (!isNaN(parseFloat(item))) {
  3736. newResult = [...newResult, parseFloat(item)];
  3737. }
  3738. });
  3739. }
  3740. return Math.max(...newResult);
  3741. }
  3742. else {
  3743. result[0].forEach((item) => {
  3744. if (!isNaN(parseFloat(item))) {
  3745. newResult = [...newResult, parseFloat(item)];
  3746. }
  3747. });
  3748. return Math.max(...newResult);
  3749. }
  3750. }
  3751. getMaxZIndex(deviation = 1) {
  3752. let nodeIndexList = [];
  3753. deviation = Number.isNaN(deviation) ? 1 : deviation;
  3754. document.querySelectorAll("*").forEach((element) => {
  3755. let nodeStyle = window.getComputedStyle(element);
  3756. /* 不对position为static和display为none的元素进行获取它们的z-index */
  3757. if (nodeStyle.position !== "static" && nodeStyle.display !== "none") {
  3758. nodeIndexList = nodeIndexList.concat(parseInt(nodeStyle.zIndex));
  3759. }
  3760. });
  3761. /* 过滤非Boolean类型 */
  3762. nodeIndexList = nodeIndexList.filter(Boolean);
  3763. let currentMaxZIndex = nodeIndexList.length
  3764. ? Math.max(...nodeIndexList)
  3765. : 0;
  3766. return currentMaxZIndex + deviation;
  3767. }
  3768. getMinValue(...args) {
  3769. let result = [...args];
  3770. let newResult = [];
  3771. if (result.length === 0) {
  3772. // @ts-ignore
  3773. return;
  3774. }
  3775. if (result.length > 1) {
  3776. if (result.length === 2 &&
  3777. typeof result[0] === "object" &&
  3778. typeof result[1] === "function") {
  3779. let data = result[0];
  3780. let handleDataFunc = result[1];
  3781. Object.keys(data).forEach((keyName) => {
  3782. newResult = [...newResult, handleDataFunc(keyName, data[keyName])];
  3783. });
  3784. }
  3785. else {
  3786. result.forEach((item) => {
  3787. if (!isNaN(parseFloat(item))) {
  3788. newResult = [...newResult, parseFloat(item)];
  3789. }
  3790. });
  3791. }
  3792. return Math.min(...newResult);
  3793. }
  3794. else {
  3795. result[0].forEach((item) => {
  3796. if (!isNaN(parseFloat(item))) {
  3797. newResult = [...newResult, parseFloat(item)];
  3798. }
  3799. });
  3800. return Math.min(...newResult);
  3801. }
  3802. }
  3803. getRandomAndroidUA() {
  3804. let UtilsContext = this;
  3805. let mobileNameList = [
  3806. "LDN-LX3",
  3807. "RNE-L03",
  3808. "ASUS_X00ID Build/NMF26F",
  3809. "WAS-LX3",
  3810. "PRA-LX3",
  3811. "MYA-L03",
  3812. "Moto G Play",
  3813. "Moto C Build/NRD90M.063",
  3814. "Redmi Note 4 Build/NRD90M",
  3815. "HUAWEI VNS-L21 Build/HUAWEIVNS-L21",
  3816. "VTR-L09",
  3817. "TRT-LX3",
  3818. "M2003J15SC Build/RP1A.200720.011; wv",
  3819. "MI 13 Build/OPR1.170623.027; wv",
  3820. ];
  3821. let androidVersion = UtilsContext.getRandomValue(12, 14);
  3822. let randomMobile = UtilsContext.getRandomValue(mobileNameList);
  3823. let chromeVersion1 = UtilsContext.getRandomValue(115, 125);
  3824. let chromeVersion2 = UtilsContext.getRandomValue(0, 0);
  3825. let chromeVersion3 = UtilsContext.getRandomValue(2272, 6099);
  3826. let chromeVersion4 = UtilsContext.getRandomValue(1, 218);
  3827. return `Mozilla/5.0 (Linux; Android ${androidVersion}; ${randomMobile}) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion1}.${chromeVersion2}.${chromeVersion3}.${chromeVersion4} Mobile Safari/537.36`;
  3828. }
  3829. getRandomValue(...args) {
  3830. let result = [...args];
  3831. if (result.length > 1) {
  3832. if (result.length === 2 &&
  3833. typeof result[0] === "number" &&
  3834. typeof result[1] === "number") {
  3835. let leftNumber = result[0] > result[1] ? result[1] : result[0];
  3836. let rightNumber = result[0] > result[1] ? result[0] : result[1];
  3837. return (Math.round(Math.random() * (rightNumber - leftNumber)) + leftNumber);
  3838. }
  3839. else {
  3840. return result[Math.floor(Math.random() * result.length)];
  3841. }
  3842. }
  3843. else if (result.length === 1) {
  3844. let paramData = result[0];
  3845. if (Array.isArray(paramData)) {
  3846. return paramData[Math.floor(Math.random() * paramData.length)];
  3847. }
  3848. else if (typeof paramData === "object" &&
  3849. Object.keys(paramData).length > 0) {
  3850. let paramObjDataKey = Object.keys(paramData)[Math.floor(Math.random() * Object.keys(paramData).length)];
  3851. return paramData[paramObjDataKey];
  3852. }
  3853. else {
  3854. return paramData;
  3855. }
  3856. }
  3857. }
  3858. /**
  3859. * 获取随机的电脑端User-Agent
  3860. * + Mozilla/5.0:以前用于Netscape浏览器,目前大多数浏览器UA都会带有
  3861. * + Windows NT 13:代表Window11系统
  3862. * + Windows NT 10.0:代表Window10系统
  3863. * + Windows NT 6.1:代表windows7系统
  3864. * + WOW64:Windows-on-Windows 64-bit,32位的应用程序运行于此64位处理器上
  3865. * + Win64:64位
  3866. * + AppleWebKit/537.36:浏览器内核
  3867. * + KHTML:HTML排版引擎
  3868. * + like Gecko:这不是Geckeo 浏览器,但是运行起来像Geckeo浏览器
  3869. * + Chrome/106.0.5068.19:Chrome版本号
  3870. * + Safari/537.36:宣称自己是Safari?
  3871. * @returns 返回随机字符串
  3872. * @example
  3873. * Utils.getRandomPCUA();
  3874. * > 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/106.0.5068.19 Safari/537.36'
  3875. **/
  3876. getRandomPCUA() {
  3877. let UtilsContext = this;
  3878. let chromeVersion1 = UtilsContext.getRandomValue(115, 126);
  3879. let chromeVersion2 = UtilsContext.getRandomValue(0, 0);
  3880. let chromeVersion3 = UtilsContext.getRandomValue(2272, 6099);
  3881. let chromeVersion4 = UtilsContext.getRandomValue(1, 218);
  3882. return `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${chromeVersion1}.${chromeVersion2}.${chromeVersion3}.${chromeVersion4} Safari/537.36`;
  3883. }
  3884. /**
  3885. * 获取元素上的使用React框架的实例属性,目前包括reactFiber、reactProps、reactEvents、reactEventHandlers、reactInternalInstance
  3886. * @param element 需要获取的目标元素
  3887. * @returns
  3888. * @example
  3889. * Utils.getReactObj(document.querySelector("input"))?.reactProps?.onChange({target:{value:"123"}});
  3890. */
  3891. getReactObj(element) {
  3892. let result = {};
  3893. Object.keys(element).forEach((domPropsName) => {
  3894. if (domPropsName.startsWith("__react")) {
  3895. let propsName = domPropsName.replace(/__(.+)\$.+/i, "$1");
  3896. if (propsName in result) ;
  3897. else {
  3898. result[propsName] = element[domPropsName];
  3899. }
  3900. }
  3901. });
  3902. return result;
  3903. }
  3904. /**
  3905. * 获取对象上的Symbol属性,如果没设置keyName,那么返回一个对象,对象是所有遍历到的Symbol对象
  3906. * @param target 目标对象
  3907. * @param keyName (可选)Symbol名或者Symbol对象
  3908. */
  3909. getSymbol(target, keyName) {
  3910. if (typeof target !== "object") {
  3911. throw new TypeError("target不是一个对象");
  3912. }
  3913. let objectsSymbols = Object.getOwnPropertySymbols(target);
  3914. if (typeof keyName === "string") {
  3915. let findSymbol = objectsSymbols.find((key) => {
  3916. return key.toString() === keyName;
  3917. });
  3918. if (findSymbol) {
  3919. return target[findSymbol];
  3920. }
  3921. }
  3922. else if (typeof keyName === "symbol") {
  3923. let findSymbol = objectsSymbols.find((key) => {
  3924. return key === keyName;
  3925. });
  3926. if (findSymbol) {
  3927. return target[findSymbol];
  3928. }
  3929. }
  3930. else {
  3931. let result = {};
  3932. objectsSymbols.forEach((item) => {
  3933. result[item] = target[item];
  3934. });
  3935. return result;
  3936. }
  3937. }
  3938. /**
  3939. * 获取文本的字符长度
  3940. * @param text
  3941. * @example
  3942. * Utils.getTextLength("测试文本")
  3943. * > 12
  3944. */
  3945. getTextLength(text) {
  3946. let encoder = new TextEncoder();
  3947. let bytes = encoder.encode(text);
  3948. return bytes.length;
  3949. }
  3950. getTextStorageSize(text, addType = true) {
  3951. let UtilsContext = this;
  3952. return UtilsContext.formatByteToSize(UtilsContext.getTextLength(text), addType);
  3953. }
  3954. getThunderUrl(url) {
  3955. if (url == null) {
  3956. throw new TypeError("url不能为空");
  3957. }
  3958. if (typeof url !== "string") {
  3959. throw new TypeError("url必须是string类型");
  3960. }
  3961. if (url.trim() === "") {
  3962. throw new TypeError("url不能为空字符串或纯空格");
  3963. }
  3964. return `thunder://${globalThis.btoa("AA" + url + "ZZ")}`;
  3965. }
  3966. /**
  3967. * 对于GM_cookie的兼容写法,当无法使用GM_cookie时可以使用这个,但是并不完全兼容,有些写不出来且限制了httponly是无法访问的
  3968. * @example
  3969. let GM_cookie = new Utils.GM_Cookie();
  3970. GM_cookie.list({name:"xxx_cookie_xxx"},function(cookies,error){
  3971. if (!error) {
  3972. console.log(cookies);
  3973. console.log(cookies.value);
  3974. } else {
  3975. console.error(error);
  3976. }
  3977. });
  3978. GM_cookie.set({name:"xxx_cookie_test_xxx",value:"这是Cookie测试值"},function(error){
  3979. if (error) {
  3980. console.error(error);
  3981. } else {
  3982. console.log('Cookie set successfully.');
  3983. }
  3984. })
  3985. GM_cookie.delete({name:"xxx_cookie_test_xxx"},function(error){
  3986. if (error) {
  3987. console.error(error);
  3988. } else {
  3989. console.log('Cookie set successfully.');
  3990. }
  3991. })
  3992. **/
  3993. GM_Cookie = UtilsGMCookie;
  3994. /**
  3995. * 注册(不可用)油猴菜单,要求本地存储的键名不能存在其它键名`GM_Menu_Local_Map`会冲突/覆盖
  3996. * @example
  3997. let GM_Menu = new Utils.GM_Menu({
  3998. data: [
  3999. {
  4000. menu_key: "menu_key",
  4001. text: "测试按钮",
  4002. enable: true,
  4003. accessKey: "a",
  4004. autoClose: false,
  4005. showText(text, enable) {
  4006. return "[" + (enable ? "√" : "×") + "]" + text;
  4007. },
  4008. callback(data) {
  4009. console.log("点击菜单,值修改为", data.enable);
  4010. },
  4011. },
  4012. ],
  4013. autoReload: false,
  4014. GM_getValue,
  4015. GM_setValue,
  4016. GM_registerMenuCommand,
  4017. GM_unregisterMenuCommand,
  4018. });
  4019. // 获取某个菜单项的值
  4020. GM_Menu.get("menu_key");
  4021. > true
  4022. // 获取某个菜单项的开启/关闭后显示的文本
  4023. GM_Menu.getShowTextValue("menu_key");
  4024. > √测试按钮
  4025. // 添加键为menu_key2的菜单项
  4026. GM_Menu.add({
  4027. key:"menu_key2",
  4028. text: "测试按钮2",
  4029. enable: false,
  4030. showText(text,enable){
  4031. return "[" + (enable ? "√" : "×") + "]" + text;
  4032. },
  4033. callback(data){
  4034. console.log("点击菜单,值修改为",data.enable);
  4035. }
  4036. });
  4037. // 使用数组的方式添加多个菜单,如menu_key3、menu_key4
  4038. GM_Menu.add([
  4039. {
  4040. key:"menu_key3",
  4041. text: "测试按钮3",
  4042. enable: false,
  4043. showText(text,enable){
  4044. return "[" + (enable ? "√" : "×") + "]" + text;
  4045. },
  4046. callback(data){
  4047. console.log("点击菜单,值修改为",data.enable);
  4048. }
  4049. },
  4050. {
  4051. key:"menu_key4",
  4052. text: "测试按钮4",
  4053. enable: false,
  4054. showText(text,enable){
  4055. return "[" + (enable ? "√" : "×") + "]" + text;
  4056. },
  4057. callback(data){
  4058. console.log("点击菜单,值修改为",data.enable);
  4059. }
  4060. }
  4061. ]);
  4062. // 更新键为menu_key的显示文字和点击回调
  4063. GM_Menu.update({
  4064. menu_key:{
  4065. text: "更新后的测试按钮",
  4066. enable: true,
  4067. showText(text,enable){
  4068. return "[" + (enable ? "√" : "×") + "]" + text;
  4069. },
  4070. callback(data){
  4071. console.log("点击菜单更新后的测试按钮,新值修改为",data.enable);
  4072. }
  4073. }
  4074. });
  4075. // 删除键为menu_key的菜单
  4076. GM_Menu.delete("menu_key");
  4077. **/
  4078. GM_Menu = GMMenu;
  4079. /**
  4080. * 基于Function prototype,能够勾住和释放任何函数
  4081. *
  4082. * .hook
  4083. * + realFunc {string} 用于保存原始函数的函数名称,用于unHook
  4084. * + hookFunc {string} 替换的hook函数
  4085. * + context {object} 目标函数所在对象,用于hook非window对象下的函数,如String.protype.slice,carInstance1
  4086. * + methodName {string} 匿名函数需显式传入目标函数名eg:this.Begin = function(){....};}
  4087. *
  4088. * .unhook
  4089. * + realFunc {string} 用于保存原始函数的函数名称,用于unHook
  4090. * + funcName {string} 被Hook的函数名称
  4091. * + context {object} 目标函数所在对象,用于hook非window对象下的函数,如String.protype.slice,carInstance1
  4092. * @example
  4093. let hook = new Utils.Hooks();
  4094. hook.initEnv();
  4095. function myFunction(){
  4096. console.log("我自己需要执行的函数");
  4097. }
  4098. function testFunction(){
  4099. console.log("正常执行的函数");
  4100. }
  4101. testFunction.hook(testFunction,myFunction,window);
  4102. **/
  4103. Hooks = Hooks;
  4104. /**
  4105. * 为减少代码量和回调,把GM_xmlhttpRequest封装
  4106. * 文档地址: https://www.tampermonkey.net/documentation.php?ext=iikm
  4107. * 其中onloadstart、onprogress、onreadystatechange是回调形式,onabort、ontimeout、onerror可以设置全局回调函数
  4108. * @param _GM_xmlHttpRequest_ 油猴中的GM_xmlhttpRequest
  4109. * @example
  4110. let httpx = new Utils.Httpx(GM_xmlhttpRequest);
  4111. let postResp = await httpx.post({
  4112. url:url,
  4113. data:JSON.stringify({
  4114. test:1
  4115. }),
  4116. timeout: 5000
  4117. });
  4118. console.log(postResp);
  4119. > {
  4120. status: true,
  4121. data: {responseText: "...", response: xxx,...},
  4122. msg: "请求完毕",
  4123. type: "onload",
  4124. }
  4125. if(postResp === "onload" && postResp.status){
  4126. // onload
  4127. }else if(postResp === "ontimeout"){
  4128. // ontimeout
  4129. }
  4130. * @example
  4131. // 也可以先配置全局参数
  4132. let httpx = new Utils.Httpx(GM_xmlhttpRequest);
  4133. httpx.config({
  4134. timeout: 5000,
  4135. async: false,
  4136. responseType: "html",
  4137. redirect: "follow",
  4138. })
  4139. // 优先级为 默认details < 全局details < 单独的details
  4140. */
  4141. Httpx = Httpx;
  4142. /**
  4143. * 浏览器端的indexedDB操作封装
  4144. * @example
  4145. let db = new Utils.indexedDB('web_DB', 'nav_text')
  4146. let data = {name:'管理员', roleId: 1, type: 1};
  4147. db.save('list',data).then((resolve)=>{
  4148. console.log(resolve,'存储成功')
  4149. })
  4150. db.get('list').then((resolve)=>{
  4151. console.log(resolve,'查询成功')
  4152. })
  4153. db.getPaging('list',20,10).then((resolve)=>{
  4154. console.log(resolve,'查询分页偏移第20,一共10行成功');
  4155. })
  4156. db.delete('list').then(resolve=>{
  4157. console.log(resolve,'删除成功---->>>>>>name')
  4158. })
  4159. db.deleteAll().then(resolve=>{
  4160. console.log(resolve,'清除数据库---->>>>>>name')
  4161. })
  4162. **/
  4163. indexedDB = indexedDB;
  4164. isNativeFunc(target) {
  4165. return Boolean(target.toString().match(/^function .*\(\) { \[native code\] }$/));
  4166. }
  4167. isNearBottom(nearValue = 50) {
  4168. var scrollTop = window.pageYOffset || document.documentElement.scrollTop;
  4169. var windowHeight = window.innerHeight || document.documentElement.clientHeight;
  4170. var documentHeight = document.documentElement.scrollHeight;
  4171. return scrollTop + windowHeight >= documentHeight - nearValue;
  4172. }
  4173. isDOM(target) {
  4174. return target instanceof Node;
  4175. }
  4176. isFullscreenEnabled() {
  4177. return !!(UtilsCore.document.fullscreenEnabled ||
  4178. UtilsCore.document.webkitFullScreenEnabled ||
  4179. UtilsCore.document.mozFullScreenEnabled ||
  4180. UtilsCore.document.msFullScreenEnabled);
  4181. }
  4182. isJQuery(target) {
  4183. let result = false;
  4184. // @ts-ignore
  4185. if (typeof jQuery === "object" && target instanceof jQuery) {
  4186. result = true;
  4187. }
  4188. if (target == null) {
  4189. return false;
  4190. }
  4191. if (typeof target === "object") {
  4192. /* 也有种可能,这个jQuery对象是1.8.3版本的,页面中的jQuery是3.4.1版本的 */
  4193. let jQueryProps = [
  4194. "add",
  4195. "addBack",
  4196. "addClass",
  4197. "after",
  4198. "ajaxComplete",
  4199. "ajaxError",
  4200. "ajaxSend",
  4201. "ajaxStart",
  4202. "ajaxStop",
  4203. "ajaxSuccess",
  4204. "animate",
  4205. "append",
  4206. "appendTo",
  4207. "attr",
  4208. "before",
  4209. "bind",
  4210. "blur",
  4211. "change",
  4212. "children",
  4213. "clearQueue",
  4214. "click",
  4215. "clone",
  4216. "closest",
  4217. "constructor",
  4218. "contents",
  4219. "contextmenu",
  4220. "css",
  4221. "data",
  4222. "dblclick",
  4223. "delay",
  4224. "delegate",
  4225. "dequeue",
  4226. "each",
  4227. "empty",
  4228. "end",
  4229. "eq",
  4230. "extend",
  4231. "fadeIn",
  4232. "fadeOut",
  4233. "fadeTo",
  4234. "fadeToggle",
  4235. "filter",
  4236. "find",
  4237. "first",
  4238. "focus",
  4239. "focusin",
  4240. "focusout",
  4241. "get",
  4242. "has",
  4243. "hasClass",
  4244. "height",
  4245. "hide",
  4246. "hover",
  4247. "html",
  4248. "index",
  4249. "init",
  4250. "innerHeight",
  4251. "innerWidth",
  4252. "insertAfter",
  4253. "insertBefore",
  4254. "is",
  4255. "jquery",
  4256. "keydown",
  4257. "keypress",
  4258. "keyup",
  4259. "last",
  4260. "load",
  4261. "map",
  4262. "mousedown",
  4263. "mouseenter",
  4264. "mouseleave",
  4265. "mousemove",
  4266. "mouseout",
  4267. "mouseover",
  4268. "mouseup",
  4269. "next",
  4270. "nextAll",
  4271. "not",
  4272. "off",
  4273. "offset",
  4274. "offsetParent",
  4275. "on",
  4276. "one",
  4277. "outerHeight",
  4278. "outerWidth",
  4279. "parent",
  4280. "parents",
  4281. "position",
  4282. "prepend",
  4283. "prependTo",
  4284. "prev",
  4285. "prevAll",
  4286. "prevUntil",
  4287. "promise",
  4288. "prop",
  4289. "pushStack",
  4290. "queue",
  4291. "ready",
  4292. "remove",
  4293. "removeAttr",
  4294. "removeClass",
  4295. "removeData",
  4296. "removeProp",
  4297. "replaceAll",
  4298. "replaceWith",
  4299. "resize",
  4300. "scroll",
  4301. "scrollLeft",
  4302. "scrollTop",
  4303. "select",
  4304. "show",
  4305. "siblings",
  4306. "slice",
  4307. "slideDown",
  4308. "slideToggle",
  4309. "slideUp",
  4310. "sort",
  4311. "splice",
  4312. "text",
  4313. "toArray",
  4314. "toggle",
  4315. "toggleClass",
  4316. "trigger",
  4317. "triggerHandler",
  4318. "unbind",
  4319. "width",
  4320. "wrap",
  4321. ];
  4322. for (const jQueryPropsName of jQueryProps) {
  4323. if (!(jQueryPropsName in target)) {
  4324. result = false;
  4325. /* console.log(jQueryPropsName); */
  4326. break;
  4327. }
  4328. else {
  4329. result = true;
  4330. }
  4331. }
  4332. }
  4333. return result;
  4334. }
  4335. isPhone(userAgent = navigator.userAgent) {
  4336. return Boolean(/(iPhone|iPad|iPod|iOS|Android|Mobile)/i.test(userAgent));
  4337. }
  4338. isSameChars(targetStr, coefficient = 1) {
  4339. if (typeof targetStr !== "string") {
  4340. throw new TypeError("参数 str 必须是 string 类型");
  4341. }
  4342. if (targetStr.length < 2) {
  4343. return false;
  4344. }
  4345. targetStr = targetStr.toLowerCase();
  4346. const targetCharMap = {};
  4347. let targetStrLength = 0;
  4348. for (const char of targetStr) {
  4349. if (Reflect.has(targetCharMap, char)) {
  4350. targetCharMap[char]++;
  4351. }
  4352. else {
  4353. targetCharMap[char] = 1;
  4354. }
  4355. targetStrLength++;
  4356. }
  4357. let result = false;
  4358. for (const char in targetCharMap) {
  4359. if (targetCharMap[char] / targetStrLength >= coefficient) {
  4360. result = true;
  4361. break;
  4362. }
  4363. }
  4364. return result;
  4365. }
  4366. isNotNull(...args) {
  4367. let UtilsContext = this;
  4368. return !UtilsContext.isNull.apply(this, args);
  4369. }
  4370. isNull(...args) {
  4371. let result = true;
  4372. let checkList = [...args];
  4373. for (const objItem of checkList) {
  4374. let itemResult = false;
  4375. if (objItem === null || objItem === undefined) {
  4376. itemResult = true;
  4377. }
  4378. else {
  4379. switch (typeof objItem) {
  4380. case "object":
  4381. if (typeof objItem[Symbol.iterator] === "function") {
  4382. /* 可迭代 */
  4383. itemResult = objItem.length === 0;
  4384. }
  4385. else {
  4386. itemResult = Object.keys(objItem).length === 0;
  4387. }
  4388. break;
  4389. case "number":
  4390. itemResult = objItem === 0;
  4391. break;
  4392. case "string":
  4393. itemResult =
  4394. objItem.trim() === "" ||
  4395. objItem === "null" ||
  4396. objItem === "undefined";
  4397. break;
  4398. case "boolean":
  4399. itemResult = !objItem;
  4400. break;
  4401. case "function":
  4402. let funcStr = objItem.toString().replace(/\s/g, "");
  4403. /* 排除()=>{}、(xxx="")=>{}、function(){}、function(xxx=""){} */
  4404. itemResult = Boolean(funcStr.match(/^\(.*?\)=>\{\}$|^function.*?\(.*?\)\{\}$/));
  4405. break;
  4406. }
  4407. }
  4408. result = result && itemResult;
  4409. }
  4410. return result;
  4411. }
  4412. isThemeDark() {
  4413. return globalThis.matchMedia("(prefers-color-scheme: dark)").matches;
  4414. }
  4415. isVisible(element, inView = false) {
  4416. let needCheckDomList = [];
  4417. if (element instanceof Array || element instanceof NodeList) {
  4418. element = element;
  4419. needCheckDomList = [...element];
  4420. }
  4421. else {
  4422. needCheckDomList = [element];
  4423. }
  4424. let result = true;
  4425. for (const domItem of needCheckDomList) {
  4426. let domDisplay = window.getComputedStyle(domItem);
  4427. if (domDisplay.display === "none") {
  4428. result = false;
  4429. }
  4430. else {
  4431. let domClientRect = domItem.getBoundingClientRect();
  4432. if (inView) {
  4433. let viewportWidth = window.innerWidth || document.documentElement.clientWidth;
  4434. let viewportHeight = window.innerHeight || document.documentElement.clientHeight;
  4435. result = !(domClientRect.right < 0 ||
  4436. domClientRect.left > viewportWidth ||
  4437. domClientRect.bottom < 0 ||
  4438. domClientRect.top > viewportHeight);
  4439. }
  4440. else {
  4441. result = Boolean(domItem.getClientRects().length);
  4442. }
  4443. }
  4444. if (!result) {
  4445. /* 有一个不可见就退出循环 */
  4446. break;
  4447. }
  4448. }
  4449. return result;
  4450. }
  4451. isWebView_Via() {
  4452. let result = true;
  4453. let UtilsContext = this;
  4454. if (typeof UtilsCore.top.window.via === "object") {
  4455. for (const key in Object.values(UtilsCore.top.window.via)) {
  4456. if (Reflect.has(UtilsCore.top.window.via, key)) {
  4457. let objValueFunc = UtilsCore.top.window.via[key];
  4458. if (typeof objValueFunc === "function" &&
  4459. UtilsContext.isNativeFunc(objValueFunc)) {
  4460. result = true;
  4461. }
  4462. else {
  4463. result = false;
  4464. break;
  4465. }
  4466. }
  4467. }
  4468. }
  4469. else {
  4470. result = false;
  4471. }
  4472. return result;
  4473. }
  4474. isWebView_X() {
  4475. let result = true;
  4476. let UtilsContext = this;
  4477. if (typeof UtilsCore.top.window.mbrowser === "object") {
  4478. for (const key in Object.values(UtilsCore.top.window.mbrowser)) {
  4479. if (Reflect.has(UtilsCore.top.window.mbrowser, key)) {
  4480. let objValueFunc = UtilsCore.top.window.mbrowser[key];
  4481. if (typeof objValueFunc === "function" &&
  4482. UtilsContext.isNativeFunc(objValueFunc)) {
  4483. result = true;
  4484. }
  4485. else {
  4486. result = false;
  4487. break;
  4488. }
  4489. }
  4490. }
  4491. }
  4492. else {
  4493. result = false;
  4494. }
  4495. return result;
  4496. }
  4497. parseObjectToArray(target) {
  4498. if (typeof target !== "object") {
  4499. throw new Error("Utils.parseObjectToArray 参数 target 必须为 object 类型");
  4500. }
  4501. let result = [];
  4502. Object.keys(target).forEach(function (keyName) {
  4503. result = result.concat(target[keyName]);
  4504. });
  4505. return result;
  4506. }
  4507. listenKeyboard(target, eventName = "keypress", callback) {
  4508. if (typeof target !== "object" ||
  4509. (typeof target["addEventListener"] !== "function" &&
  4510. typeof target["removeEventListener"] !== "function")) {
  4511. throw new Error("Utils.listenKeyboard 参数 target 必须为 Window|HTMLElement 类型");
  4512. }
  4513. let keyEvent = function (event) {
  4514. let keyName = event.key || event.code;
  4515. let keyValue = event.charCode || event.keyCode || event.which;
  4516. let otherCodeList = [];
  4517. if (event.ctrlKey) {
  4518. otherCodeList.push("ctrl");
  4519. }
  4520. if (event.altKey) {
  4521. otherCodeList.push("alt");
  4522. }
  4523. if (event.metaKey) {
  4524. otherCodeList.push("meta");
  4525. }
  4526. if (event.shiftKey) {
  4527. otherCodeList.push("shift");
  4528. }
  4529. if (typeof callback === "function") {
  4530. callback(keyName, keyValue.toString(), otherCodeList, event);
  4531. }
  4532. };
  4533. target.addEventListener(eventName, keyEvent);
  4534. return {
  4535. removeListen() {
  4536. target.removeEventListener(eventName, keyEvent);
  4537. },
  4538. };
  4539. }
  4540. /**
  4541. * 自动锁对象,用于循环判断运行的函数,在循环外new后使用,注意,如果函数内部存在异步操作,需要使用await
  4542. * @example
  4543. let lock = new Utils.LockFunction(()=>{console.log(1)}))
  4544. lock.run();
  4545. > 1
  4546. * @example
  4547. let lock = new Utils.LockFunction(()=>{console.log(1)}),true) -- 异步操作
  4548. await lock.run();
  4549. > 1
  4550. **/
  4551. LockFunction = LockFunction;
  4552. /**
  4553. * 日志对象
  4554. * @param _GM_info_ 油猴管理器的API GM_info,或者是一个对象,如{"script":{name:"Utils.Log"}}
  4555. * @example
  4556. let log = new Utils.Log(GM_info);
  4557. log.info("普通输出");
  4558. > 普通输出
  4559. log.success("成功输出");
  4560. > 成功输出
  4561. log.error("错误输出");
  4562. > 错误输出
  4563. log.warn("警告输出");
  4564. > 警告输出
  4565. log.tag = "自定义tag信息";
  4566. log.info("自定义info的颜色","#e0e0e0");
  4567. > 自定义info的颜色
  4568. log.config({
  4569. successColor: "#31dc02",
  4570. errorColor: "#e02d2d",
  4571. infoColor: "black",
  4572. })
  4573. log.success("颜色为#31dc02");
  4574. > 颜色为#31dc02
  4575. */
  4576. Log = Log;
  4577. mergeArrayToString(data, handleFunc) {
  4578. if (!(data instanceof Array)) {
  4579. throw new Error("Utils.mergeArrayToString 参数 data 必须为 Array 类型");
  4580. }
  4581. let content = "";
  4582. if (typeof handleFunc === "function") {
  4583. data.forEach((item) => {
  4584. content += handleFunc(item);
  4585. });
  4586. }
  4587. else if (typeof handleFunc === "string") {
  4588. data.forEach((item) => {
  4589. content += item[handleFunc];
  4590. });
  4591. }
  4592. else {
  4593. data.forEach((item) => {
  4594. Object.values(item)
  4595. .filter((item2) => typeof item2 === "string")
  4596. .forEach((item3) => {
  4597. content += item3;
  4598. });
  4599. });
  4600. }
  4601. return content;
  4602. }
  4603. mutationObserver(target, observer_config) {
  4604. let UtilsContext = this;
  4605. if (!(target instanceof Node) &&
  4606. !(target instanceof NodeList) &&
  4607. !UtilsContext.isJQuery(target)) {
  4608. throw new Error("Utils.mutationObserver 参数 target 必须为 Node|NodeList|jQuery类型");
  4609. }
  4610. let default_obverser_config = {
  4611. /* 监听到元素有反馈,需执行的函数 */
  4612. callback: () => { },
  4613. config: {
  4614. /**
  4615. * + true 监听以 target 为根节点的整个子树。包括子树中所有节点的属性,而不仅仅是针对 target
  4616. * + false (默认) 不生效
  4617. */
  4618. subtree: void 0,
  4619. /**
  4620. * + true 监听 target 节点中发生的节点的新增与删除(同时,如果 subtree 为 true,会针对整个子树生效)
  4621. * + false (默认) 不生效
  4622. */
  4623. childList: void 0,
  4624. /**
  4625. * + true 观察所有监听的节点属性值的变化。默认值为 true,当声明了 attributeFilter 或 attributeOldValue
  4626. * + false (默认) 不生效
  4627. */
  4628. attributes: void 0,
  4629. /**
  4630. * 一个用于声明哪些属性名会被监听的数组。如果不声明该属性,所有属性的变化都将触发通知
  4631. */
  4632. attributeFilter: void 0,
  4633. /**
  4634. * + true 记录上一次被监听的节点的属性变化;可查阅 MutationObserver 中的 Monitoring attribute values 了解关于观察属性变化和属性值记录的详情
  4635. * + false (默认) 不生效
  4636. */
  4637. attributeOldValue: void 0,
  4638. /**
  4639. * + true 监听声明的 target 节点上所有字符的变化。默认值为 true,如果声明了 characterDataOldValue
  4640. * + false (默认) 不生效
  4641. */
  4642. characterData: void 0,
  4643. /**
  4644. * + true 记录前一个被监听的节点中发生的文本变化
  4645. * + false (默认) 不生效
  4646. */
  4647. characterDataOldValue: void 0,
  4648. },
  4649. immediate: false,
  4650. };
  4651. observer_config = UtilsContext.assign(default_obverser_config, observer_config);
  4652. let windowMutationObserver = window.MutationObserver ||
  4653. UtilsCore.window.webkitMutationObserver ||
  4654. UtilsCore.window.MozMutationObserver;
  4655. let mutationObserver = new windowMutationObserver(function (mutations, observer) {
  4656. observer_config?.callback(mutations, observer);
  4657. });
  4658. if (target instanceof Node) {
  4659. /* 传入的参数是节点元素 */
  4660. mutationObserver.observe(target, observer_config.config);
  4661. }
  4662. else if (target instanceof NodeList) {
  4663. /* 传入的参数是节点元素数组 */
  4664. target.forEach((item) => {
  4665. mutationObserver.observe(item, observer_config.config);
  4666. });
  4667. }
  4668. else if (UtilsContext.isJQuery(target)) {
  4669. /* 传入的参数是jQuery对象 */
  4670. target.each((index, item) => {
  4671. mutationObserver.observe(item, observer_config.config);
  4672. });
  4673. }
  4674. else {
  4675. /* 未知 */
  4676. console.error("Utils.mutationObserver 未知参数", arguments);
  4677. }
  4678. if (observer_config.immediate) {
  4679. /* 主动触发一次 */
  4680. observer_config.callback([], mutationObserver);
  4681. }
  4682. return mutationObserver;
  4683. }
  4684. /**
  4685. * 去除全局window下的Utils,返回控制权
  4686. * @example
  4687. * let utils = Utils.noConflict();
  4688. * > ...
  4689. */
  4690. noConflict = function () {
  4691. if (UtilsCore.window.Utils) {
  4692. Reflect.deleteProperty(UtilsCore.window, "Utils");
  4693. }
  4694. UtilsCore.window.Utils = utils;
  4695. return utils;
  4696. };
  4697. noConflictFunc(needReleaseObject, needReleaseName, functionNameList = [], release = true) {
  4698. let UtilsContext = this;
  4699. if (typeof needReleaseObject !== "object") {
  4700. throw new Error("Utils.noConflictFunc 参数 needReleaseObject 必须为 object 类型");
  4701. }
  4702. if (typeof needReleaseName !== "string") {
  4703. throw new Error("Utils.noConflictFunc 参数 needReleaseName 必须为 string 类型");
  4704. }
  4705. if (!Array.isArray(functionNameList)) {
  4706. throw new Error("Utils.noConflictFunc 参数 functionNameList 必须为 Array 类型");
  4707. }
  4708. let needReleaseKey = "__" + needReleaseName;
  4709. /**
  4710. * 释放所有
  4711. */
  4712. function releaseAll() {
  4713. if (typeof UtilsCore.window[needReleaseKey] !== "undefined") {
  4714. /* 已存在 */
  4715. return;
  4716. }
  4717. UtilsCore.window[needReleaseKey] =
  4718. UtilsContext.deepClone(needReleaseObject);
  4719. Object.values(needReleaseObject).forEach((value) => {
  4720. if (typeof value === "function") {
  4721. needReleaseObject[value.name] = () => { };
  4722. }
  4723. });
  4724. }
  4725. /**
  4726. * 释放单个
  4727. */
  4728. function releaseOne() {
  4729. Array.from(functionNameList).forEach((item) => {
  4730. Object.values(needReleaseObject).forEach((value) => {
  4731. if (typeof value === "function") {
  4732. if (typeof UtilsCore.window[needReleaseKey] === "undefined") {
  4733. UtilsCore.window[needReleaseKey] = {};
  4734. }
  4735. if (item === value.name) {
  4736. UtilsCore.window[needReleaseKey][value.name] = needReleaseObject[value.name];
  4737. needReleaseObject[value.name] = () => { };
  4738. }
  4739. }
  4740. });
  4741. });
  4742. }
  4743. /**
  4744. * 恢复所有
  4745. */
  4746. function recoveryAll() {
  4747. if (typeof UtilsCore.window[needReleaseKey] === "undefined") {
  4748. /* 未存在 */
  4749. return;
  4750. }
  4751. Object.assign(needReleaseObject, UtilsCore.window[needReleaseKey]);
  4752. Reflect.deleteProperty(UtilsCore.window, "needReleaseKey");
  4753. }
  4754. /**
  4755. * 恢复单个
  4756. */
  4757. function recoveryOne() {
  4758. if (typeof UtilsCore.window[needReleaseKey] === "undefined") {
  4759. /* 未存在 */
  4760. return;
  4761. }
  4762. Array.from(functionNameList).forEach((item) => {
  4763. if (UtilsCore.window[needReleaseKey][item]) {
  4764. needReleaseObject[item] = UtilsCore.window[needReleaseKey][item];
  4765. Reflect.deleteProperty(UtilsCore.window[needReleaseKey], item);
  4766. if (Object.keys(UtilsCore.window[needReleaseKey]).length === 0) {
  4767. Reflect.deleteProperty(window, needReleaseKey);
  4768. }
  4769. }
  4770. });
  4771. }
  4772. if (release) {
  4773. /* 释放 */
  4774. if (functionNameList.length === 0) {
  4775. releaseAll();
  4776. }
  4777. else {
  4778. /* 对单个进行操作 */
  4779. releaseOne();
  4780. }
  4781. }
  4782. else {
  4783. /* 恢复 */
  4784. if (functionNameList.length === 0) {
  4785. recoveryAll();
  4786. }
  4787. else {
  4788. /* 对单个进行操作 */
  4789. recoveryOne();
  4790. }
  4791. }
  4792. }
  4793. parseBase64ToBlob(dataUri) {
  4794. if (typeof dataUri !== "string") {
  4795. throw new Error("Utils.parseBase64ToBlob 参数 dataUri 必须为 string 类型");
  4796. }
  4797. let dataUriSplit = dataUri.split(","), dataUriMime = dataUriSplit[0].match(/:(.*?);/)[1], dataUriBase64Str = atob(dataUriSplit[1]), dataUriLength = dataUriBase64Str.length, u8arr = new Uint8Array(dataUriLength);
  4798. while (dataUriLength--) {
  4799. u8arr[dataUriLength] = dataUriBase64Str.charCodeAt(dataUriLength);
  4800. }
  4801. return new Blob([u8arr], {
  4802. type: dataUriMime,
  4803. });
  4804. }
  4805. parseBase64ToFile(dataUri, fileName = "example") {
  4806. if (typeof dataUri !== "string") {
  4807. throw new Error("Utils.parseBase64ToFile 参数 dataUri 必须为 string 类型");
  4808. }
  4809. if (typeof fileName !== "string") {
  4810. throw new Error("Utils.parseBase64ToFile 参数 fileName 必须为 string 类型");
  4811. }
  4812. let dataUriSplit = dataUri.split(","), dataUriMime = dataUriSplit[0].match(/:(.*?);/)[1], dataUriBase64Str = atob(dataUriSplit[1]), dataUriLength = dataUriBase64Str.length, u8arr = new Uint8Array(dataUriLength);
  4813. while (dataUriLength--) {
  4814. u8arr[dataUriLength] = dataUriBase64Str.charCodeAt(dataUriLength);
  4815. }
  4816. return new File([u8arr], fileName, {
  4817. type: dataUriMime,
  4818. });
  4819. }
  4820. parseInt(matchList = [], defaultValue = 0) {
  4821. if (matchList == null) {
  4822. return parseInt(defaultValue.toString());
  4823. }
  4824. let parseValue = parseInt(matchList[matchList.length - 1]);
  4825. if (isNaN(parseValue)) {
  4826. parseValue = parseInt(defaultValue.toString());
  4827. }
  4828. return parseValue;
  4829. }
  4830. async parseBlobToFile(blobUrl, fileName = "example") {
  4831. return new Promise((resolve, reject) => {
  4832. fetch(blobUrl)
  4833. .then((response) => response.blob())
  4834. .then((blob) => {
  4835. const file = new File([blob], fileName, { type: blob.type });
  4836. resolve(file);
  4837. })
  4838. .catch((error) => {
  4839. console.error("Error:", error);
  4840. reject(error);
  4841. });
  4842. });
  4843. }
  4844. parseCDATA(text = "") {
  4845. let result = "";
  4846. let cdataRegexp = /<\!\[CDATA\[([\s\S]*)\]\]>/;
  4847. let cdataMatch = cdataRegexp.exec(text.trim());
  4848. if (cdataMatch && cdataMatch.length > 1) {
  4849. result = cdataMatch[cdataMatch.length - 1];
  4850. }
  4851. return result;
  4852. }
  4853. async parseFileToBase64(fileObj) {
  4854. let reader = new FileReader();
  4855. reader.readAsDataURL(fileObj);
  4856. return new Promise((resolve) => {
  4857. reader.onload = function (event) {
  4858. resolve(event.target.result);
  4859. };
  4860. });
  4861. }
  4862. parseFromString(text, mimeType = "text/html") {
  4863. let parser = new DOMParser();
  4864. return parser.parseFromString(text, mimeType);
  4865. }
  4866. parseStringToRegExpString(text) {
  4867. if (typeof text !== "string") {
  4868. throw new TypeError("string必须是字符串");
  4869. }
  4870. let regString = text.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&");
  4871. return regString;
  4872. }
  4873. preventEvent(element, eventNameList = [], capture) {
  4874. function stopEvent(event) {
  4875. /* 阻止事件的默认行为发生。例如,当点击一个链接时,浏览器会默认打开链接的URL */
  4876. event?.preventDefault();
  4877. /* 停止事件的传播,阻止它继续向更上层的元素冒泡,事件将不会再传播给其他的元素 */
  4878. event?.stopPropagation();
  4879. /* 阻止事件传播,并且还能阻止元素上的其他事件处理程序被触发 */
  4880. event?.stopImmediatePropagation();
  4881. return false;
  4882. }
  4883. if (arguments.length === 1) {
  4884. /* 直接阻止事件 */
  4885. return stopEvent(arguments[0]);
  4886. }
  4887. else {
  4888. /* 添加对应的事件来阻止触发 */
  4889. if (typeof eventNameList === "string") {
  4890. eventNameList = [eventNameList];
  4891. }
  4892. eventNameList.forEach((eventName) => {
  4893. element.addEventListener(eventName, stopEvent, {
  4894. capture: Boolean(capture),
  4895. });
  4896. });
  4897. }
  4898. }
  4899. /**
  4900. * 在canvas元素节点上绘制进度圆圈
  4901. * @example
  4902. let progress = new Utils.Process({canvasNode:document.querySelector("canvas")});
  4903. progress.draw();
  4904. * **/
  4905. Progress = Progress;
  4906. registerTrustClickEvent(isTrustValue = true, filter) {
  4907. function trustEvent(event) {
  4908. return new Proxy(event, {
  4909. get: function (target, property) {
  4910. if (property === "isTrusted") {
  4911. return isTrustValue;
  4912. }
  4913. else {
  4914. return Reflect.get(target, property);
  4915. }
  4916. },
  4917. });
  4918. }
  4919. if (filter == null) {
  4920. filter = function (typeName) {
  4921. return typeName === "click";
  4922. };
  4923. }
  4924. const originalListener = EventTarget.prototype.addEventListener;
  4925. EventTarget.prototype.addEventListener = function (...args) {
  4926. let type = args[0];
  4927. let callback = args[1];
  4928. args[2];
  4929. if (filter(type)) {
  4930. if (typeof callback === "function") {
  4931. args[1] = function (event) {
  4932. callback.call(this, trustEvent(event));
  4933. };
  4934. }
  4935. else if (typeof callback === "object" &&
  4936. "handleEvent" in callback) {
  4937. let oldHandleEvent = callback["handleEvent"];
  4938. args[1]["handleEvent"] = function (event) {
  4939. if (event == null) {
  4940. return;
  4941. }
  4942. try {
  4943. /* Proxy对象使用instanceof会报错 */
  4944. event instanceof Proxy;
  4945. oldHandleEvent.call(this, trustEvent(event));
  4946. }
  4947. catch (error) {
  4948. // @ts-ignore
  4949. event["isTrusted"] = isTrustValue;
  4950. }
  4951. };
  4952. }
  4953. }
  4954. return originalListener.apply(this, args);
  4955. };
  4956. }
  4957. reverseNumber(num) {
  4958. let reversedNum = 0;
  4959. let isNegative = false;
  4960. if (num < 0) {
  4961. isNegative = true;
  4962. num = Math.abs(num);
  4963. }
  4964. while (num > 0) {
  4965. reversedNum = reversedNum * 10 + (num % 10);
  4966. num = Math.floor(num / 10);
  4967. }
  4968. return isNegative ? -reversedNum : reversedNum;
  4969. }
  4970. selectElementText(element, childTextNode, startIndex, endIndex) {
  4971. let range = document.createRange();
  4972. range.selectNodeContents(element);
  4973. if (childTextNode) {
  4974. if (childTextNode.nodeType !== Node.TEXT_NODE) {
  4975. throw new TypeError("childTextNode必须是#text元素");
  4976. }
  4977. if (startIndex != null && endIndex != null) {
  4978. range.setStart(childTextNode, startIndex);
  4979. range.setEnd(childTextNode, endIndex);
  4980. }
  4981. }
  4982. let selection = globalThis.getSelection();
  4983. if (selection) {
  4984. selection.removeAllRanges();
  4985. selection.addRange(range);
  4986. }
  4987. }
  4988. setClip(data, info = {
  4989. type: "text",
  4990. mimetype: "text/plain",
  4991. }) {
  4992. if (typeof data === "object") {
  4993. if (data instanceof Element) {
  4994. data = data.outerHTML;
  4995. }
  4996. else {
  4997. data = JSON.stringify(data);
  4998. }
  4999. }
  5000. else if (typeof data !== "string") {
  5001. data = data.toString();
  5002. }
  5003. let textType = typeof info === "object" ? info.type : info;
  5004. if (textType.includes("html")) {
  5005. textType = "text/html";
  5006. }
  5007. else {
  5008. textType = "text/plain";
  5009. }
  5010. class UtilsClipboard {
  5011. #resolve;
  5012. #copyData;
  5013. #copyDataType;
  5014. constructor(resolve, copyData, copyDataType) {
  5015. this.#resolve = resolve;
  5016. this.#copyData = copyData;
  5017. this.#copyDataType = copyDataType;
  5018. }
  5019. async init() {
  5020. let copyStatus = false;
  5021. await this.requestClipboardPermission();
  5022. if (this.hasClipboard() &&
  5023. (this.hasClipboardWrite() || this.hasClipboardWriteText())) {
  5024. try {
  5025. copyStatus = await this.copyDataByClipboard();
  5026. }
  5027. catch (error) {
  5028. console.error("复制失败,使用第二种方式,error👉", error);
  5029. copyStatus = this.copyTextByTextArea();
  5030. }
  5031. }
  5032. else {
  5033. copyStatus = this.copyTextByTextArea();
  5034. }
  5035. this.#resolve(copyStatus);
  5036. this.destroy();
  5037. }
  5038. destroy() {
  5039. // @ts-ignore
  5040. this.#resolve = null;
  5041. // @ts-ignore
  5042. this.#copyData = null;
  5043. // @ts-ignore
  5044. this.#copyDataType = null;
  5045. }
  5046. isText() {
  5047. return this.#copyDataType.includes("text");
  5048. }
  5049. hasClipboard() {
  5050. return navigator?.clipboard != null;
  5051. }
  5052. hasClipboardWrite() {
  5053. return navigator?.clipboard?.write != null;
  5054. }
  5055. hasClipboardWriteText() {
  5056. return navigator?.clipboard?.writeText != null;
  5057. }
  5058. /**
  5059. * 使用textarea和document.execCommand("copy")来复制文字
  5060. */
  5061. copyTextByTextArea() {
  5062. try {
  5063. let copyElement = document.createElement("textarea");
  5064. copyElement.value = this.#copyData;
  5065. copyElement.setAttribute("type", "text");
  5066. copyElement.setAttribute("style", "opacity:0;position:absolute;");
  5067. copyElement.setAttribute("readonly", "readonly");
  5068. document.body.appendChild(copyElement);
  5069. copyElement.select();
  5070. document.execCommand("copy");
  5071. document.body.removeChild(copyElement);
  5072. return true;
  5073. }
  5074. catch (error) {
  5075. console.error("复制失败,error👉", error);
  5076. return false;
  5077. }
  5078. }
  5079. /**
  5080. * 申请剪贴板权限
  5081. * @returns {Promise<boolean>}
  5082. */
  5083. requestClipboardPermission() {
  5084. return new Promise((resolve, reject) => {
  5085. if (navigator.permissions && navigator.permissions.query) {
  5086. navigator.permissions
  5087. .query({
  5088. // @ts-ignore
  5089. name: "clipboard-write",
  5090. })
  5091. .then((permissionStatus) => {
  5092. resolve(true);
  5093. })
  5094. .catch(
  5095. /** @param {TypeError} error */
  5096. (error) => {
  5097. console.error([
  5098. "申请剪贴板权限失败,尝试直接写入👉",
  5099. error.message ?? error.name ?? error.stack,
  5100. ]);
  5101. resolve(false);
  5102. });
  5103. }
  5104. else {
  5105. resolve(false);
  5106. }
  5107. });
  5108. }
  5109. /**
  5110. * 使用clipboard直接写入数据到剪贴板
  5111. */
  5112. copyDataByClipboard() {
  5113. return new Promise((resolve, reject) => {
  5114. if (this.isText()) {
  5115. /* 只复制文字 */
  5116. navigator.clipboard
  5117. .writeText(this.#copyData)
  5118. .then(() => {
  5119. resolve(true);
  5120. })
  5121. .catch((error) => {
  5122. reject(error);
  5123. });
  5124. }
  5125. else {
  5126. /* 可复制对象 */
  5127. let textBlob = new Blob([this.#copyData], {
  5128. type: this.#copyDataType,
  5129. });
  5130. navigator.clipboard
  5131. .write([
  5132. new ClipboardItem({
  5133. [this.#copyDataType]: textBlob,
  5134. }),
  5135. ])
  5136. .then(() => {
  5137. resolve(true);
  5138. })
  5139. .catch((error) => {
  5140. reject(error);
  5141. });
  5142. }
  5143. });
  5144. }
  5145. }
  5146. return new Promise((resolve) => {
  5147. const utilsClipboard = new UtilsClipboard(resolve, data, textType);
  5148. if (document.hasFocus()) {
  5149. utilsClipboard.init();
  5150. }
  5151. else {
  5152. window.addEventListener("focus", () => {
  5153. utilsClipboard.init();
  5154. }, { once: true });
  5155. }
  5156. });
  5157. }
  5158. setTimeout(callback, delayTime = 0) {
  5159. let UtilsContext = this;
  5160. if (typeof callback !== "function" && typeof callback !== "string") {
  5161. throw new TypeError("Utils.setTimeout 参数 callback 必须为 function|string 类型");
  5162. }
  5163. if (typeof delayTime !== "number") {
  5164. throw new TypeError("Utils.setTimeout 参数 delayTime 必须为 number 类型");
  5165. }
  5166. return new Promise((resolve) => {
  5167. setTimeout(() => {
  5168. resolve(UtilsContext.tryCatch().run(callback));
  5169. }, delayTime);
  5170. });
  5171. }
  5172. sleep(delayTime = 0) {
  5173. if (typeof delayTime !== "number") {
  5174. throw new Error("Utils.sleep 参数 delayTime 必须为 number 类型");
  5175. }
  5176. return new Promise((resolve) => {
  5177. setTimeout(() => {
  5178. resolve(void 0);
  5179. }, delayTime);
  5180. });
  5181. }
  5182. dragSlider(selector, offsetX = UtilsCore.window.innerWidth) {
  5183. function initMouseEvent(eventName, offSetX, offSetY) {
  5184. let win = unsafeWindow || window;
  5185. let mouseEvent = document.createEvent("MouseEvents");
  5186. mouseEvent.initMouseEvent(eventName, true, true, win, 0, offSetX, offSetY, offSetX, offSetY, false, false, false, false, 0, null);
  5187. return mouseEvent;
  5188. }
  5189. let sliderElement = typeof selector === "string"
  5190. ? document.querySelector(selector)
  5191. : selector;
  5192. if (!(sliderElement instanceof Node) ||
  5193. !(sliderElement instanceof Element)) {
  5194. throw new Error("Utils.dragSlider 参数selector 必须为Node/Element类型");
  5195. }
  5196. let rect = sliderElement.getBoundingClientRect(), x0 = rect.x || rect.left, y0 = rect.y || rect.top, x1 = x0 + offsetX, y1 = y0;
  5197. sliderElement.dispatchEvent(initMouseEvent("mousedown", x0, y0));
  5198. sliderElement.dispatchEvent(initMouseEvent("mousemove", x1, y1));
  5199. sliderElement.dispatchEvent(initMouseEvent("mouseleave", x1, y1));
  5200. sliderElement.dispatchEvent(initMouseEvent("mouseout", x1, y1));
  5201. }
  5202. enterFullScreen(element = UtilsCore.document.documentElement, options) {
  5203. try {
  5204. if (element.requestFullscreen) {
  5205. element.requestFullscreen(options);
  5206. }
  5207. else if (element.webkitRequestFullScreen) {
  5208. element.webkitRequestFullScreen();
  5209. }
  5210. else if (element.mozRequestFullScreen) {
  5211. element.mozRequestFullScreen();
  5212. }
  5213. else if (element.msRequestFullscreen) {
  5214. element.msRequestFullscreen();
  5215. }
  5216. else {
  5217. throw new TypeError("该浏览器不支持全屏API");
  5218. }
  5219. }
  5220. catch (err) {
  5221. console.error(err);
  5222. }
  5223. }
  5224. exitFullScreen(element = document.documentElement) {
  5225. if (UtilsCore.document.exitFullscreen) {
  5226. return UtilsCore.document.exitFullscreen();
  5227. }
  5228. else if (UtilsCore.document.msExitFullscreen) {
  5229. return UtilsCore.document.msExitFullscreen();
  5230. }
  5231. else if (UtilsCore.document.mozCancelFullScreen) {
  5232. return UtilsCore.document.mozCancelFullScreen();
  5233. }
  5234. else if (UtilsCore.document.webkitCancelFullScreen) {
  5235. return UtilsCore.document.webkitCancelFullScreen();
  5236. }
  5237. else {
  5238. return new Promise((resolve, reject) => {
  5239. reject(new TypeError("该浏览器不支持全屏API"));
  5240. });
  5241. }
  5242. }
  5243. sortListByProperty(data, getPropertyValueFunc, sortByDesc = true) {
  5244. let UtilsContext = this;
  5245. if (typeof getPropertyValueFunc !== "function" &&
  5246. typeof getPropertyValueFunc !== "string") {
  5247. throw new Error("Utils.sortListByProperty 参数 getPropertyValueFunc 必须为 function|string 类型");
  5248. }
  5249. if (typeof sortByDesc !== "boolean") {
  5250. throw new Error("Utils.sortListByProperty 参数 sortByDesc 必须为 boolean 类型");
  5251. }
  5252. let getObjValue = function (obj) {
  5253. return typeof getPropertyValueFunc === "string"
  5254. ? obj[getPropertyValueFunc]
  5255. : getPropertyValueFunc(obj);
  5256. };
  5257. /**
  5258. * 排序方法
  5259. * @param {any} after_obj
  5260. * @param {any} before_obj
  5261. * @returns
  5262. */
  5263. let sortFunc = function (after_obj, before_obj) {
  5264. let beforeValue = getObjValue(before_obj); /* 前 */
  5265. let afterValue = getObjValue(after_obj); /* 后 */
  5266. if (sortByDesc) {
  5267. if (afterValue > beforeValue) {
  5268. return -1;
  5269. }
  5270. else if (afterValue < beforeValue) {
  5271. return 1;
  5272. }
  5273. else {
  5274. return 0;
  5275. }
  5276. }
  5277. else {
  5278. if (afterValue < beforeValue) {
  5279. return -1;
  5280. }
  5281. else if (afterValue > beforeValue) {
  5282. return 1;
  5283. }
  5284. else {
  5285. return 0;
  5286. }
  5287. }
  5288. };
  5289. /**
  5290. * 排序元素方法
  5291. * @param nodeList 元素列表
  5292. * @param getNodeListFunc 获取元素列表的函数
  5293. */
  5294. let sortNodeFunc = function (nodeList, getNodeListFunc) {
  5295. let nodeListLength = nodeList.length;
  5296. for (let i = 0; i < nodeListLength - 1; i++) {
  5297. for (let j = 0; j < nodeListLength - 1 - i; j++) {
  5298. let beforeNode = nodeList[j];
  5299. let afterNode = nodeList[j + 1];
  5300. let beforeValue = getObjValue(beforeNode); /* 前 */
  5301. let afterValue = getObjValue(afterNode); /* 后 */
  5302. if ((sortByDesc == true && beforeValue < afterValue) ||
  5303. (sortByDesc == false && beforeValue > afterValue)) {
  5304. /* 升序/降序 */
  5305. /* 相邻元素两两对比 */
  5306. let temp = beforeNode.nextElementSibling;
  5307. afterNode.after(beforeNode);
  5308. if (temp == null) {
  5309. /* 如果为空,那么是最后一个元素,使用append */
  5310. temp.parentNode.appendChild(afterNode);
  5311. }
  5312. else {
  5313. /* 不为空,使用before */
  5314. temp.before(afterNode);
  5315. }
  5316. nodeList = getNodeListFunc();
  5317. }
  5318. }
  5319. }
  5320. };
  5321. let result = data;
  5322. let getDataFunc = null;
  5323. if (data instanceof Function) {
  5324. getDataFunc = data;
  5325. data = data();
  5326. }
  5327. if (Array.isArray(data)) {
  5328. data.sort(sortFunc);
  5329. }
  5330. else if (data instanceof NodeList || UtilsContext.isJQuery(data)) {
  5331. sortNodeFunc(data, getDataFunc);
  5332. result = getDataFunc();
  5333. }
  5334. else {
  5335. throw new Error("Utils.sortListByProperty 参数 data 必须为 Array|NodeList|jQuery 类型");
  5336. }
  5337. return result;
  5338. }
  5339. stringToRegular(targetString, flags = "ig") {
  5340. let reg;
  5341. // @ts-ignore
  5342. flags = flags.toLowerCase();
  5343. if (typeof targetString === "string") {
  5344. reg = new RegExp(targetString.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&"), flags);
  5345. }
  5346. else if (targetString instanceof RegExp) {
  5347. reg = targetString;
  5348. }
  5349. else {
  5350. throw new Error("Utils.stringToRegular 参数targetString必须是string|Regexp类型");
  5351. }
  5352. return reg;
  5353. }
  5354. stringTitleToUpperCase(targetString, otherStrToLowerCase = false) {
  5355. let newTargetString = targetString.slice(0, 1).toUpperCase();
  5356. if (otherStrToLowerCase) {
  5357. newTargetString = newTargetString + targetString.slice(1).toLowerCase();
  5358. }
  5359. else {
  5360. newTargetString = newTargetString + targetString.slice(1);
  5361. }
  5362. return newTargetString;
  5363. }
  5364. startsWith(target, searchString, position = 0) {
  5365. let UtilsContext = this;
  5366. if (position > target.length) {
  5367. /* 超出目标字符串的长度 */
  5368. return false;
  5369. }
  5370. if (position !== 0) {
  5371. target = target.slice(position, target.length);
  5372. }
  5373. let searchStringRegexp = searchString;
  5374. if (typeof searchString === "string") {
  5375. searchStringRegexp = new RegExp(`^${searchString}`);
  5376. }
  5377. else if (Array.isArray(searchString)) {
  5378. let flag = false;
  5379. for (const searcStr of searchString) {
  5380. if (!UtilsContext.startsWith(target, searcStr, position)) {
  5381. flag = true;
  5382. break;
  5383. }
  5384. }
  5385. return flag;
  5386. }
  5387. return Boolean(target.match(searchStringRegexp));
  5388. }
  5389. stringTitleToLowerCase(targetString, otherStrToUpperCase = false) {
  5390. let newTargetString = targetString.slice(0, 1).toLowerCase();
  5391. if (otherStrToUpperCase) {
  5392. newTargetString = newTargetString + targetString.slice(1).toUpperCase();
  5393. }
  5394. else {
  5395. newTargetString = newTargetString + targetString.slice(1);
  5396. }
  5397. return newTargetString;
  5398. }
  5399. toJSON(data, errorCallBack) {
  5400. let UtilsContext = this;
  5401. let result = {};
  5402. if (typeof data === "object") {
  5403. return data;
  5404. }
  5405. UtilsContext.tryCatch()
  5406. .config({ log: false })
  5407. .error((error) => {
  5408. UtilsContext.tryCatch()
  5409. .error(() => {
  5410. try {
  5411. result = UtilsCore.window.eval("(" + data + ")");
  5412. }
  5413. catch (error2) {
  5414. if (typeof errorCallBack === "function") {
  5415. errorCallBack(error2);
  5416. }
  5417. }
  5418. })
  5419. .run(() => {
  5420. if (data &&
  5421. /^[\],:{}\s]*$/.test(data
  5422. .replace(/\\(?:["\\\/bfnrt]|u[0-9a-fA-F]{4})/g, "@")
  5423. .replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, "]")
  5424. .replace(/(?:^|:|,)(?:\s*\[)+/g, ""))) {
  5425. result = new Function("return " + data)();
  5426. }
  5427. else {
  5428. if (typeof errorCallBack === "function") {
  5429. errorCallBack(new Error("target is not a JSON"));
  5430. }
  5431. }
  5432. });
  5433. })
  5434. .run(() => {
  5435. data = data.trim();
  5436. result = JSON.parse(data);
  5437. });
  5438. return result;
  5439. }
  5440. toSearchParamsStr(obj) {
  5441. let UtilsContext = this;
  5442. let searhParamsStr = "";
  5443. if (Array.isArray(obj)) {
  5444. obj.forEach((item) => {
  5445. if (searhParamsStr === "") {
  5446. searhParamsStr += UtilsContext.toSearchParamsStr(item);
  5447. }
  5448. else {
  5449. searhParamsStr += "&" + UtilsContext.toSearchParamsStr(item);
  5450. }
  5451. });
  5452. }
  5453. else {
  5454. searhParamsStr = new URLSearchParams(Object.entries(obj)).toString();
  5455. }
  5456. return searhParamsStr;
  5457. }
  5458. /**
  5459. * 提供一个封装了 try-catch 的函数,可以执行传入的函数并捕获其可能抛出的错误,并通过传入的错误处理函数进行处理。
  5460. * @example
  5461. * Utils.tryCatch().error().run(()=>{console.log(1)});
  5462. * > 1
  5463. * @example
  5464. * Utils.tryCatch().config({log:true}).error((error)=>{console.log(error)}).run(()=>{throw new Error('测试错误')});
  5465. * > ()=>{throw new Error('测试错误')}出现错误
  5466. */
  5467. tryCatch = TryCatch;
  5468. uniqueArray(uniqueArrayData = [], compareArrayData = [], compareFun = (item, item2) => {
  5469. // @ts-ignore
  5470. return item === item2;
  5471. }) {
  5472. return Array.from(uniqueArrayData).filter((item) => !Array.from(compareArrayData).some(function (item2) {
  5473. return compareFun(item, item2);
  5474. }));
  5475. }
  5476. waitArrayLoopToEnd(data, handleFunc) {
  5477. let UtilsContext = this;
  5478. if (typeof handleFunc !== "function" && typeof handleFunc !== "string") {
  5479. throw new Error("Utils.waitArrayLoopToEnd 参数 handleDataFunction 必须为 function|string 类型");
  5480. }
  5481. return Promise.all(Array.from(data).map(async (item, index) => {
  5482. await UtilsContext.tryCatch(index, item).run(handleFunc);
  5483. }));
  5484. }
  5485. waitNode(...args) {
  5486. // 过滤掉undefined
  5487. args = args.filter((arg) => arg !== void 0);
  5488. let that = this;
  5489. // 选择器
  5490. let selector = args[0];
  5491. // 父元素(监听的元素)
  5492. let parent = UtilsCore.document;
  5493. // 超时时间
  5494. let timeout = 0;
  5495. if (typeof args[0] !== "string" && !Array.isArray(args[0])) {
  5496. throw new TypeError("Utils.waitNode 第一个参数必须是string|string[]");
  5497. }
  5498. if (args.length === 1) ;
  5499. else if (args.length === 2) {
  5500. let secondParam = args[1];
  5501. if (typeof secondParam === "number") {
  5502. // "div",10000
  5503. timeout = secondParam;
  5504. }
  5505. else if (typeof secondParam === "object" &&
  5506. secondParam instanceof Node) {
  5507. // "div",document
  5508. parent = secondParam;
  5509. }
  5510. else {
  5511. throw new TypeError("Utils.waitNode 第二个参数必须是number|Node");
  5512. }
  5513. }
  5514. else if (args.length === 3) {
  5515. // "div",document,10000
  5516. // 第二个参数,parent
  5517. let secondParam = args[1];
  5518. // 第三个参数,timeout
  5519. let thirdParam = args[2];
  5520. if (typeof secondParam === "object" && secondParam instanceof Node) {
  5521. parent = secondParam;
  5522. if (typeof thirdParam === "number") {
  5523. timeout = thirdParam;
  5524. }
  5525. else {
  5526. throw new TypeError("Utils.waitNode 第三个参数必须是number");
  5527. }
  5528. }
  5529. else {
  5530. throw new TypeError("Utils.waitNode 第二个参数必须是Node");
  5531. }
  5532. }
  5533. else {
  5534. throw new TypeError("Utils.waitNode 参数个数错误");
  5535. }
  5536. return new Promise((resolve) => {
  5537. function getNode() {
  5538. if (Array.isArray(selector)) {
  5539. let result = [];
  5540. for (let index = 0; index < selector.length; index++) {
  5541. let node = parent.querySelector(selector[index]);
  5542. if (node) {
  5543. result.push(node);
  5544. }
  5545. }
  5546. if (result.length === selector.length) {
  5547. return result;
  5548. }
  5549. }
  5550. else {
  5551. return parent.querySelector(selector);
  5552. }
  5553. }
  5554. let node = getNode();
  5555. if (node) {
  5556. resolve(node);
  5557. return;
  5558. }
  5559. let observer = that.mutationObserver(parent, {
  5560. config: {
  5561. subtree: true,
  5562. childList: true,
  5563. attributes: true,
  5564. },
  5565. callback() {
  5566. let node = getNode();
  5567. if (node) {
  5568. // 取消观察器
  5569. observer.disconnect();
  5570. resolve(node);
  5571. return;
  5572. }
  5573. },
  5574. });
  5575. if (timeout > 0) {
  5576. setTimeout(() => {
  5577. // 取消观察器
  5578. observer.disconnect();
  5579. resolve(null);
  5580. }, timeout);
  5581. }
  5582. });
  5583. }
  5584. waitAnyNode(...args) {
  5585. // 过滤掉undefined
  5586. args = args.filter((arg) => arg !== void 0);
  5587. let that = this;
  5588. // 选择器
  5589. let selectorList = args[0];
  5590. // 父元素(监听的元素)
  5591. let parent = UtilsCore.document;
  5592. // 超时时间
  5593. let timeout = 0;
  5594. if (typeof args[0] !== "object" && !Array.isArray(args[0])) {
  5595. throw new TypeError("Utils.waitAnyNode 第一个参数必须是string[]");
  5596. }
  5597. if (args.length === 1) ;
  5598. else if (args.length === 2) {
  5599. let secondParam = args[1];
  5600. if (typeof secondParam === "number") {
  5601. // "div",10000
  5602. timeout = secondParam;
  5603. }
  5604. else if (typeof secondParam === "object" &&
  5605. secondParam instanceof Node) {
  5606. // "div",document
  5607. parent = secondParam;
  5608. }
  5609. else {
  5610. throw new TypeError("Utils.waitAnyNode 第二个参数必须是number|Node");
  5611. }
  5612. }
  5613. else if (args.length === 3) {
  5614. // "div",document,10000
  5615. // 第二个参数,parent
  5616. let secondParam = args[1];
  5617. // 第三个参数,timeout
  5618. let thirdParam = args[2];
  5619. if (typeof secondParam === "object" && secondParam instanceof Node) {
  5620. parent = secondParam;
  5621. if (typeof thirdParam === "number") {
  5622. timeout = thirdParam;
  5623. }
  5624. else {
  5625. throw new TypeError("Utils.waitAnyNode 第三个参数必须是number");
  5626. }
  5627. }
  5628. else {
  5629. throw new TypeError("Utils.waitAnyNode 第二个参数必须是Node");
  5630. }
  5631. }
  5632. else {
  5633. throw new TypeError("Utils.waitAnyNode 参数个数错误");
  5634. }
  5635. let promiseList = selectorList.map((selector) => {
  5636. return that.waitNode(selector, parent, timeout);
  5637. });
  5638. return Promise.any(promiseList);
  5639. }
  5640. waitNodeList(...args) {
  5641. // 过滤掉undefined
  5642. args = args.filter((arg) => arg !== void 0);
  5643. let that = this;
  5644. // 选择器数组
  5645. let selector = args[0];
  5646. // 父元素(监听的元素)
  5647. let parent = UtilsCore.document;
  5648. // 超时时间
  5649. let timeout = 0;
  5650. if (typeof args[0] !== "string" && !Array.isArray(args[0])) {
  5651. throw new TypeError("Utils.waitNodeList 第一个参数必须是string|string[]");
  5652. }
  5653. if (args.length === 1) ;
  5654. else if (args.length === 2) {
  5655. let secondParam = args[1];
  5656. if (typeof secondParam === "number") {
  5657. // "div",10000
  5658. timeout = secondParam;
  5659. }
  5660. else if (typeof secondParam === "object" &&
  5661. secondParam instanceof Node) {
  5662. // "div",document
  5663. parent = secondParam;
  5664. }
  5665. else {
  5666. throw new TypeError("Utils.waitNodeList 第二个参数必须是number|Node");
  5667. }
  5668. }
  5669. else if (args.length === 3) {
  5670. // "div",document,10000
  5671. // 第二个参数,parent
  5672. let secondParam = args[1];
  5673. // 第三个参数,timeout
  5674. let thirdParam = args[2];
  5675. if (typeof secondParam === "object" && secondParam instanceof Node) {
  5676. parent = secondParam;
  5677. if (typeof thirdParam === "number") {
  5678. timeout = thirdParam;
  5679. }
  5680. else {
  5681. throw new TypeError("Utils.waitNodeList 第三个参数必须是number");
  5682. }
  5683. }
  5684. else {
  5685. throw new TypeError("Utils.waitNodeList 第二个参数必须是Node");
  5686. }
  5687. }
  5688. else {
  5689. throw new TypeError("Utils.waitNodeList 参数个数错误");
  5690. }
  5691. return new Promise((resolve) => {
  5692. function getNodeList() {
  5693. if (Array.isArray(selector)) {
  5694. let result = [];
  5695. for (let index = 0; index < selector.length; index++) {
  5696. let nodeList = parent.querySelectorAll(selector[index]);
  5697. if (nodeList.length) {
  5698. result.push(nodeList);
  5699. }
  5700. }
  5701. if (result.length === selector.length) {
  5702. return result;
  5703. }
  5704. }
  5705. else {
  5706. let nodeList = parent.querySelectorAll(selector);
  5707. if (nodeList.length) {
  5708. return nodeList;
  5709. }
  5710. }
  5711. }
  5712. let nodeList = getNodeList();
  5713. if (nodeList) {
  5714. resolve(nodeList);
  5715. return;
  5716. }
  5717. let observer = that.mutationObserver(parent, {
  5718. config: {
  5719. subtree: true,
  5720. childList: true,
  5721. attributes: true,
  5722. },
  5723. callback() {
  5724. let node = getNodeList();
  5725. if (node) {
  5726. // 取消观察器
  5727. observer.disconnect();
  5728. resolve(node);
  5729. return;
  5730. }
  5731. },
  5732. });
  5733. if (timeout > 0) {
  5734. setTimeout(() => {
  5735. // 取消观察器
  5736. observer.disconnect();
  5737. resolve(null);
  5738. }, timeout);
  5739. }
  5740. });
  5741. }
  5742. waitAnyNodeList(...args) {
  5743. // 过滤掉undefined
  5744. args = args.filter((arg) => arg !== void 0);
  5745. let that = this;
  5746. // 选择器数组
  5747. let selectorList = args[0];
  5748. // 父元素(监听的元素)
  5749. let parent = UtilsCore.document;
  5750. // 超时时间
  5751. let timeout = 0;
  5752. if (!Array.isArray(args[0])) {
  5753. throw new TypeError("Utils.waitAnyNodeList 第一个参数必须是string[]");
  5754. }
  5755. if (args.length === 1) ;
  5756. else if (args.length === 2) {
  5757. let secondParam = args[1];
  5758. if (typeof secondParam === "number") {
  5759. // "div",10000
  5760. timeout = secondParam;
  5761. }
  5762. else if (typeof secondParam === "object" &&
  5763. secondParam instanceof Node) {
  5764. // "div",document
  5765. parent = secondParam;
  5766. }
  5767. else {
  5768. throw new TypeError("Utils.waitAnyNodeList 第二个参数必须是number|Node");
  5769. }
  5770. }
  5771. else if (args.length === 3) {
  5772. // "div",document,10000
  5773. // 第二个参数,parent
  5774. let secondParam = args[1];
  5775. // 第三个参数,timeout
  5776. let thirdParam = args[2];
  5777. if (typeof secondParam === "object" && secondParam instanceof Node) {
  5778. parent = secondParam;
  5779. if (typeof thirdParam === "number") {
  5780. timeout = thirdParam;
  5781. }
  5782. else {
  5783. throw new TypeError("Utils.waitAnyNodeList 第三个参数必须是number");
  5784. }
  5785. }
  5786. else {
  5787. throw new TypeError("Utils.waitAnyNodeList 第二个参数必须是Node");
  5788. }
  5789. }
  5790. else {
  5791. throw new TypeError("Utils.waitAnyNodeList 参数个数错误");
  5792. }
  5793. let promiseList = selectorList.map((selector) => {
  5794. return that.waitNodeList(selector, parent, timeout);
  5795. });
  5796. return Promise.any(promiseList);
  5797. }
  5798. waitProperty(checkObj, checkPropertyName) {
  5799. return new Promise((resolve) => {
  5800. let obj = checkObj;
  5801. if (typeof checkObj === "function") {
  5802. obj = checkObj();
  5803. }
  5804. if (Reflect.has(obj, checkPropertyName)) {
  5805. resolve(obj[checkPropertyName]);
  5806. }
  5807. else {
  5808. Object.defineProperty(obj, checkPropertyName, {
  5809. set: function (value) {
  5810. try {
  5811. resolve(value);
  5812. }
  5813. catch (error) {
  5814. console.error("Error setting property:", error);
  5815. }
  5816. },
  5817. });
  5818. }
  5819. });
  5820. }
  5821. waitPropertyByInterval(checkObj, checkPropertyName, intervalTimer = 250, maxTime = -1) {
  5822. if (checkObj == null) {
  5823. throw new TypeError("checkObj 不能为空对象 ");
  5824. }
  5825. let isResolve = false;
  5826. return new Promise((resolve, reject) => {
  5827. let interval = setInterval(() => {
  5828. let obj = checkObj;
  5829. if (typeof checkObj === "function") {
  5830. obj = checkObj();
  5831. }
  5832. if (typeof obj !== "object") {
  5833. return;
  5834. }
  5835. if (obj == null) {
  5836. return;
  5837. }
  5838. if ((typeof checkPropertyName === "function" && checkPropertyName(obj)) ||
  5839. Reflect.has(obj, checkPropertyName)) {
  5840. isResolve = true;
  5841. clearInterval(interval);
  5842. resolve(obj[checkPropertyName]);
  5843. }
  5844. }, intervalTimer);
  5845. if (maxTime !== -1) {
  5846. setTimeout(() => {
  5847. if (!isResolve) {
  5848. clearInterval(interval);
  5849. reject();
  5850. }
  5851. }, maxTime);
  5852. }
  5853. });
  5854. }
  5855. async waitVueByInterval(element, propertyName, timer = 250, maxTime = -1, vueName = "__vue__") {
  5856. if (element == null) {
  5857. throw new Error("Utils.waitVueByInterval 参数element 不能为空");
  5858. }
  5859. let flag = false;
  5860. let UtilsContext = this;
  5861. try {
  5862. await UtilsContext.waitPropertyByInterval(element, function (targetElement) {
  5863. if (targetElement == null) {
  5864. return false;
  5865. }
  5866. if (!(vueName in targetElement)) {
  5867. return false;
  5868. }
  5869. if (propertyName == null) {
  5870. return true;
  5871. }
  5872. let vueObject = targetElement[vueName];
  5873. if (typeof propertyName === "string") {
  5874. if (propertyName in vueObject) {
  5875. flag = true;
  5876. return true;
  5877. }
  5878. }
  5879. else {
  5880. /* Function */
  5881. if (propertyName(vueObject)) {
  5882. flag = true;
  5883. return true;
  5884. }
  5885. }
  5886. return false;
  5887. }, timer, maxTime);
  5888. }
  5889. catch (error) {
  5890. return flag;
  5891. }
  5892. return flag;
  5893. }
  5894. watchObject(target, propertyName, getCallBack, setCallBack) {
  5895. if (typeof getCallBack !== "function" &&
  5896. typeof setCallBack !== "function") {
  5897. return;
  5898. }
  5899. if (typeof getCallBack === "function") {
  5900. Object.defineProperty(target, propertyName, {
  5901. get() {
  5902. if (typeof getCallBack === "function") {
  5903. return getCallBack(target[propertyName]);
  5904. }
  5905. else {
  5906. return target[propertyName];
  5907. }
  5908. },
  5909. });
  5910. }
  5911. else if (typeof setCallBack === "function") {
  5912. Object.defineProperty(target, propertyName, {
  5913. set(value) {
  5914. if (typeof setCallBack === "function") {
  5915. setCallBack(value);
  5916. }
  5917. },
  5918. });
  5919. }
  5920. else {
  5921. Object.defineProperty(target, propertyName, {
  5922. get() {
  5923. if (typeof getCallBack === "function") {
  5924. return getCallBack(target[propertyName]);
  5925. }
  5926. else {
  5927. return target[propertyName];
  5928. }
  5929. },
  5930. set(value) {
  5931. if (typeof setCallBack === "function") {
  5932. setCallBack(value);
  5933. }
  5934. },
  5935. });
  5936. }
  5937. }
  5938. /**
  5939. * 创建一个新的Utils实例
  5940. * @param option
  5941. * @returns
  5942. */
  5943. createUtils(option) {
  5944. return new Utils(option);
  5945. }
  5946. /**
  5947. * 将对象转换为FormData
  5948. * @param data 待转换的对象
  5949. * @param isEncode 是否对值为string进行编码转换(encodeURIComponent)
  5950. */
  5951. toFormData(data, isEncode = false) {
  5952. const formData = new FormData();
  5953. Object.keys(data).forEach((key) => {
  5954. let value = data[key];
  5955. if (typeof value === "number") {
  5956. value = value.toString();
  5957. }
  5958. if (isEncode && typeof value === "string") {
  5959. value = encodeURIComponent(value);
  5960. }
  5961. if (value instanceof File) {
  5962. formData.append(key, value, value.name);
  5963. }
  5964. else {
  5965. formData.append(key, value);
  5966. }
  5967. });
  5968. return formData;
  5969. }
  5970. }
  5971. let utils = new Utils();
  5972.  
  5973. return utils;
  5974.  
  5975. }));
  5976. //# sourceMappingURL=index.umd.js.map

QingJ © 2025

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