@whitesev/utils 2.5.8 → 2.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.amd.js CHANGED
@@ -1026,6 +1026,454 @@ define((function () { 'use strict';
1026
1026
  })();
1027
1027
  };
1028
1028
 
1029
+ // ==UserScript==
1030
+ // @name ajaxHooker
1031
+ // @author cxxjackie
1032
+ // @version 1.2.4
1033
+ // @supportURL https://bbs.tampermonkey.net.cn/thread-3284-1-1.html
1034
+ // ==/UserScript==
1035
+
1036
+ const AjaxHooker1_2_4 = function () {
1037
+ return (function () {
1038
+ const win = window.unsafeWindow || document.defaultView || window;
1039
+ const hookFns = [];
1040
+ const realXhr = win.XMLHttpRequest;
1041
+ const resProto = win.Response.prototype;
1042
+ const toString = Object.prototype.toString;
1043
+ const realFetch = win.fetch;
1044
+ const xhrResponses = ["response", "responseText", "responseXML"];
1045
+ const fetchResponses = ["arrayBuffer", "blob", "formData", "json", "text"];
1046
+ const xhrAsyncEvents = ["readystatechange", "load", "loadend"];
1047
+ let filter;
1048
+ function emptyFn() {}
1049
+ function errorFn(err) {
1050
+ console.error(err);
1051
+ }
1052
+ function defineProp(obj, prop, getter, setter) {
1053
+ Object.defineProperty(obj, prop, {
1054
+ configurable: true,
1055
+ enumerable: true,
1056
+ get: getter,
1057
+ set: setter,
1058
+ });
1059
+ }
1060
+ function readonly(obj, prop, value = obj[prop]) {
1061
+ defineProp(obj, prop, () => value, emptyFn);
1062
+ }
1063
+ function writable(obj, prop, value = obj[prop]) {
1064
+ Object.defineProperty(obj, prop, {
1065
+ configurable: true,
1066
+ enumerable: true,
1067
+ writable: true,
1068
+ value: value,
1069
+ });
1070
+ }
1071
+ function toFilterObj(obj) {
1072
+ return {
1073
+ type: obj.type,
1074
+ url: obj.url,
1075
+ method: obj.method && obj.method.toUpperCase(),
1076
+ };
1077
+ }
1078
+ function shouldFilter(type, url, method) {
1079
+ return (
1080
+ filter &&
1081
+ !filter.find(
1082
+ (obj) =>
1083
+ (!obj.type || obj.type === type) &&
1084
+ (!obj.url ||
1085
+ (toString.call(obj.url) === "[object String]"
1086
+ ? url.includes(obj.url)
1087
+ : obj.url.test(url))) &&
1088
+ (!obj.method || obj.method === method.toUpperCase())
1089
+ )
1090
+ );
1091
+ }
1092
+ function lookupGetter(obj, prop) {
1093
+ let getter;
1094
+ let proto = obj;
1095
+ while (proto) {
1096
+ const descriptor = Object.getOwnPropertyDescriptor(proto, prop);
1097
+ getter = descriptor && descriptor.get;
1098
+ if (getter) break;
1099
+ proto = Object.getPrototypeOf(proto);
1100
+ }
1101
+ return getter ? getter.bind(obj) : emptyFn;
1102
+ }
1103
+ function waitForHookFns(request) {
1104
+ return Promise.all(
1105
+ hookFns.map((fn) => Promise.resolve(fn(request)).then(emptyFn, errorFn))
1106
+ );
1107
+ }
1108
+ function waitForRequestKeys(request, requestClone) {
1109
+ return Promise.all(
1110
+ ["url", "method", "abort", "headers", "data"].map((key) => {
1111
+ return Promise.resolve(request[key]).then(
1112
+ (val) => (request[key] = val),
1113
+ () => (request[key] = requestClone[key])
1114
+ );
1115
+ })
1116
+ );
1117
+ }
1118
+ function fakeEventSIP() {
1119
+ this.ajaxHooker_stopped = true;
1120
+ }
1121
+ function xhrDelegateEvent(e) {
1122
+ const xhr = e.target;
1123
+ e.stopImmediatePropagation = fakeEventSIP;
1124
+ xhr.__ajaxHooker.hookedEvents[e.type].forEach(
1125
+ (fn) => !e.ajaxHooker_stopped && fn.call(xhr, e)
1126
+ );
1127
+ const onEvent = xhr.__ajaxHooker.hookedEvents["on" + e.type];
1128
+ typeof onEvent === "function" && onEvent.call(xhr, e);
1129
+ }
1130
+ function xhrReadyStateChange(e) {
1131
+ if (e.target.readyState === 4) {
1132
+ e.target.dispatchEvent(
1133
+ new CustomEvent("ajaxHooker_responseReady", { detail: e })
1134
+ );
1135
+ } else {
1136
+ e.target.__ajaxHooker.delegateEvent(e);
1137
+ }
1138
+ }
1139
+ function xhrLoadAndLoadend(e) {
1140
+ e.target.__ajaxHooker.delegateEvent(e);
1141
+ }
1142
+ function fakeXhrOpen(method, url, ...args) {
1143
+ const ah = this.__ajaxHooker;
1144
+ ah.url = url.toString();
1145
+ ah.method = method.toUpperCase();
1146
+ ah.openArgs = args;
1147
+ ah.headers = {};
1148
+ return ah.originalMethods.open(method, url, ...args);
1149
+ }
1150
+ function fakeXhr() {
1151
+ const xhr = new realXhr();
1152
+ let ah = xhr.__ajaxHooker;
1153
+ if (!ah) {
1154
+ ah = xhr.__ajaxHooker = {
1155
+ headers: {},
1156
+ hookedEvents: {
1157
+ readystatechange: new Set(),
1158
+ load: new Set(),
1159
+ loadend: new Set(),
1160
+ },
1161
+ delegateEvent: xhrDelegateEvent,
1162
+ originalGetters: {},
1163
+ originalMethods: {},
1164
+ };
1165
+ xhr.addEventListener("readystatechange", xhrReadyStateChange);
1166
+ xhr.addEventListener("load", xhrLoadAndLoadend);
1167
+ xhr.addEventListener("loadend", xhrLoadAndLoadend);
1168
+ for (const key of xhrResponses) {
1169
+ ah.originalGetters[key] = lookupGetter(xhr, key);
1170
+ }
1171
+ for (const method of [
1172
+ "open",
1173
+ "setRequestHeader",
1174
+ "addEventListener",
1175
+ "removeEventListener",
1176
+ ]) {
1177
+ ah.originalMethods[method] = xhr[method].bind(xhr);
1178
+ }
1179
+ xhr.open = fakeXhrOpen;
1180
+ xhr.setRequestHeader = (header, value) => {
1181
+ ah.originalMethods.setRequestHeader(header, value);
1182
+ if (xhr.readyState === 1) {
1183
+ if (ah.headers[header]) {
1184
+ ah.headers[header] += ", " + value;
1185
+ } else {
1186
+ ah.headers[header] = value;
1187
+ }
1188
+ }
1189
+ };
1190
+ xhr.addEventListener = function (...args) {
1191
+ if (xhrAsyncEvents.includes(args[0])) {
1192
+ ah.hookedEvents[args[0]].add(args[1]);
1193
+ } else {
1194
+ ah.originalMethods.addEventListener(...args);
1195
+ }
1196
+ };
1197
+ xhr.removeEventListener = function (...args) {
1198
+ if (xhrAsyncEvents.includes(args[0])) {
1199
+ ah.hookedEvents[args[0]].delete(args[1]);
1200
+ } else {
1201
+ ah.originalMethods.removeEventListener(...args);
1202
+ }
1203
+ };
1204
+ xhrAsyncEvents.forEach((evt) => {
1205
+ const onEvt = "on" + evt;
1206
+ defineProp(
1207
+ xhr,
1208
+ onEvt,
1209
+ () => {
1210
+ return ah.hookedEvents[onEvt] || null;
1211
+ },
1212
+ (val) => {
1213
+ ah.hookedEvents[onEvt] = typeof val === "function" ? val : null;
1214
+ }
1215
+ );
1216
+ });
1217
+ }
1218
+ const realSend = xhr.send.bind(xhr);
1219
+ xhr.send = function (data) {
1220
+ if (xhr.readyState !== 1) return realSend(data);
1221
+ ah.delegateEvent = xhrDelegateEvent;
1222
+ xhrResponses.forEach((prop) => {
1223
+ delete xhr[prop]; // delete descriptor
1224
+ });
1225
+ if (shouldFilter("xhr", ah.url, ah.method)) {
1226
+ xhr.addEventListener("ajaxHooker_responseReady", (e) => {
1227
+ ah.delegateEvent(e.detail);
1228
+ });
1229
+ return realSend(data);
1230
+ }
1231
+ try {
1232
+ const request = {
1233
+ type: "xhr",
1234
+ url: ah.url,
1235
+ method: ah.method,
1236
+ abort: false,
1237
+ headers: ah.headers,
1238
+ data: data,
1239
+ response: null,
1240
+ };
1241
+ const requestClone = { ...request };
1242
+ waitForHookFns(request).then(() => {
1243
+ waitForRequestKeys(request, requestClone).then(() => {
1244
+ if (request.abort) return;
1245
+ ah.originalMethods.open(
1246
+ request.method,
1247
+ request.url,
1248
+ ...ah.openArgs
1249
+ );
1250
+ for (const header in request.headers) {
1251
+ ah.originalMethods.setRequestHeader(
1252
+ header,
1253
+ request.headers[header]
1254
+ );
1255
+ }
1256
+ data = request.data;
1257
+ xhr.addEventListener("ajaxHooker_responseReady", (e) => {
1258
+ try {
1259
+ if (typeof request.response === "function") {
1260
+ const arg = {
1261
+ finalUrl: xhr.responseURL,
1262
+ status: xhr.status,
1263
+ responseHeaders: {},
1264
+ };
1265
+ for (const line of xhr
1266
+ .getAllResponseHeaders()
1267
+ .trim()
1268
+ .split(/[\r\n]+/)) {
1269
+ const parts = line.split(/:\s*/);
1270
+ if (parts.length === 2) {
1271
+ const lheader = parts[0].toLowerCase();
1272
+ if (arg.responseHeaders[lheader]) {
1273
+ arg.responseHeaders[lheader] += ", " + parts[1];
1274
+ } else {
1275
+ arg.responseHeaders[lheader] = parts[1];
1276
+ }
1277
+ }
1278
+ }
1279
+ xhrResponses.forEach((prop) => {
1280
+ defineProp(
1281
+ arg,
1282
+ prop,
1283
+ () => {
1284
+ return (arg[prop] = ah.originalGetters[prop]());
1285
+ },
1286
+ (val) => {
1287
+ delete arg[prop];
1288
+ arg[prop] = val;
1289
+ }
1290
+ );
1291
+ defineProp(xhr, prop, () => {
1292
+ const val = ah.originalGetters[prop]();
1293
+ xhr.dispatchEvent(
1294
+ new CustomEvent("ajaxHooker_readResponse", {
1295
+ detail: { prop, val },
1296
+ })
1297
+ );
1298
+ return val;
1299
+ });
1300
+ });
1301
+ xhr.addEventListener("ajaxHooker_readResponse", (e) => {
1302
+ arg[e.detail.prop] = e.detail.val;
1303
+ });
1304
+ const resPromise = Promise.resolve(
1305
+ request.response(arg)
1306
+ ).then(() => {
1307
+ const task = [];
1308
+ xhrResponses.forEach((prop) => {
1309
+ const descriptor = Object.getOwnPropertyDescriptor(
1310
+ arg,
1311
+ prop
1312
+ );
1313
+ if (descriptor && "value" in descriptor) {
1314
+ task.push(
1315
+ Promise.resolve(descriptor.value).then((val) => {
1316
+ arg[prop] = val;
1317
+ defineProp(xhr, prop, () => {
1318
+ xhr.dispatchEvent(
1319
+ new CustomEvent("ajaxHooker_readResponse", {
1320
+ detail: { prop, val },
1321
+ })
1322
+ );
1323
+ return val;
1324
+ });
1325
+ }, emptyFn)
1326
+ );
1327
+ }
1328
+ });
1329
+ return Promise.all(task);
1330
+ }, errorFn);
1331
+ const eventsClone = {};
1332
+ xhrAsyncEvents.forEach((type) => {
1333
+ eventsClone[type] = new Set([...ah.hookedEvents[type]]);
1334
+ eventsClone["on" + type] = ah.hookedEvents["on" + type];
1335
+ });
1336
+ ah.delegateEvent = (event) =>
1337
+ resPromise.then(() => {
1338
+ event.stopImmediatePropagation = fakeEventSIP;
1339
+ eventsClone[event.type].forEach(
1340
+ (fn) =>
1341
+ !event.ajaxHooker_stopped && fn.call(xhr, event)
1342
+ );
1343
+ const onEvent = eventsClone["on" + event.type];
1344
+ typeof onEvent === "function" &&
1345
+ onEvent.call(xhr, event);
1346
+ });
1347
+ }
1348
+ } catch (err) {
1349
+ console.error(err);
1350
+ }
1351
+ ah.delegateEvent(e.detail);
1352
+ });
1353
+ realSend(data);
1354
+ });
1355
+ });
1356
+ } catch (err) {
1357
+ console.error(err);
1358
+ realSend(data);
1359
+ }
1360
+ };
1361
+ return xhr;
1362
+ }
1363
+ function hookFetchResponse(response, arg, callback) {
1364
+ fetchResponses.forEach((prop) => {
1365
+ response[prop] = () =>
1366
+ new Promise((resolve, reject) => {
1367
+ resProto[prop].call(response).then((res) => {
1368
+ if (prop in arg) {
1369
+ resolve(arg[prop]);
1370
+ } else {
1371
+ try {
1372
+ arg[prop] = res;
1373
+ Promise.resolve(callback(arg)).then(() => {
1374
+ if (prop in arg) {
1375
+ Promise.resolve(arg[prop]).then(
1376
+ (val) => resolve((arg[prop] = val)),
1377
+ () => resolve(res)
1378
+ );
1379
+ } else {
1380
+ resolve(res);
1381
+ }
1382
+ }, errorFn);
1383
+ } catch (err) {
1384
+ console.error(err);
1385
+ resolve(res);
1386
+ }
1387
+ }
1388
+ }, reject);
1389
+ });
1390
+ });
1391
+ }
1392
+ function fakeFetch(url, init) {
1393
+ if (url && typeof url.toString === "function") {
1394
+ url = url.toString();
1395
+ init = init || {};
1396
+ init.method = init.method || "GET";
1397
+ init.headers = init.headers || {};
1398
+ if (shouldFilter("fetch", url, init.method))
1399
+ return realFetch.call(win, url, init);
1400
+ const request = {
1401
+ type: "fetch",
1402
+ url: url,
1403
+ method: init.method.toUpperCase(),
1404
+ abort: false,
1405
+ headers: {},
1406
+ data: init.body,
1407
+ response: null,
1408
+ };
1409
+ if (toString.call(init.headers) === "[object Headers]") {
1410
+ for (const [key, val] of init.headers) {
1411
+ request.headers[key] = val;
1412
+ }
1413
+ } else {
1414
+ request.headers = { ...init.headers };
1415
+ }
1416
+ const requestClone = { ...request };
1417
+ return new Promise((resolve, reject) => {
1418
+ try {
1419
+ waitForHookFns(request).then(() => {
1420
+ waitForRequestKeys(request, requestClone).then(() => {
1421
+ if (request.abort) return reject("aborted");
1422
+ url = request.url;
1423
+ init.method = request.method;
1424
+ init.headers = request.headers;
1425
+ init.body = request.data;
1426
+ realFetch.call(win, url, init).then((response) => {
1427
+ if (typeof request.response === "function") {
1428
+ const arg = {
1429
+ finalUrl: response.url,
1430
+ status: response.status,
1431
+ responseHeaders: {},
1432
+ };
1433
+ for (const [key, val] of response.headers) {
1434
+ arg.responseHeaders[key] = val;
1435
+ }
1436
+ hookFetchResponse(response, arg, request.response);
1437
+ response.clone = () => {
1438
+ const resClone = resProto.clone.call(response);
1439
+ hookFetchResponse(resClone, arg, request.response);
1440
+ return resClone;
1441
+ };
1442
+ }
1443
+ resolve(response);
1444
+ }, reject);
1445
+ });
1446
+ });
1447
+ } catch (err) {
1448
+ console.error(err);
1449
+ return realFetch.call(win, url, init);
1450
+ }
1451
+ });
1452
+ } else {
1453
+ return realFetch.call(win, url, init);
1454
+ }
1455
+ }
1456
+ win.XMLHttpRequest = fakeXhr;
1457
+ Object.keys(realXhr).forEach((key) => (fakeXhr[key] = realXhr[key]));
1458
+ fakeXhr.prototype = realXhr.prototype;
1459
+ win.fetch = fakeFetch;
1460
+ return {
1461
+ hook: (fn) => hookFns.push(fn),
1462
+ filter: (arr) => {
1463
+ filter = Array.isArray(arr) && arr.map(toFilterObj);
1464
+ },
1465
+ protect: () => {
1466
+ readonly(win, "XMLHttpRequest", fakeXhr);
1467
+ readonly(win, "fetch", fakeFetch);
1468
+ },
1469
+ unhook: () => {
1470
+ writable(win, "XMLHttpRequest", realXhr);
1471
+ writable(win, "fetch", realFetch);
1472
+ },
1473
+ };
1474
+ })();
1475
+ };
1476
+
1029
1477
  class GMMenu {
1030
1478
  GM_Api = {
1031
1479
  /**
@@ -4659,10 +5107,19 @@ define((function () { 'use strict';
4659
5107
  * ajax劫持库,支持xhr和fetch劫持。
4660
5108
  * + 来源:https://bbs.tampermonkey.net.cn/thread-3284-1-1.html
4661
5109
  * + 作者:cxxjackie
4662
- * + 版本:1.4.1
5110
+ * + 版本:1.4.3
5111
+ * + 旧版本:1.2.4
4663
5112
  * + 文档:https://scriptcat.org/zh-CN/script-show-page/637/
5113
+ * @param useOldVersion 是否使用旧版本,默认false
4664
5114
  */
4665
- ajaxHooker = AjaxHooker;
5115
+ ajaxHooker = (useOldVersion = false) => {
5116
+ if (useOldVersion) {
5117
+ return AjaxHooker1_2_4();
5118
+ }
5119
+ else {
5120
+ return AjaxHooker();
5121
+ }
5122
+ };
4666
5123
  canvasClickByPosition(canvasElement, clientX = 0, clientY = 0, view = globalThis) {
4667
5124
  if (!(canvasElement instanceof HTMLCanvasElement)) {
4668
5125
  throw new Error("Utils.canvasClickByPosition 参数canvasElement必须是canvas元素");