您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
MT论坛效果增强,如自动签到、自动展开帖子、用户状态查看、美化导航、动态头像上传、最新发表、评论过滤器等
当前为
// ==UserScript== // @name MT论坛优化 // @namespace https://github.com/WhiteSevs/TamperMonkeyScript // @version 2025.2.18 // @author WhiteSevs // @description MT论坛效果增强,如自动签到、自动展开帖子、用户状态查看、美化导航、动态头像上传、最新发表、评论过滤器等 // @license GPL-3.0-only // @icon  // @supportURL https://github.com/WhiteSevs/TamperMonkeyScript/issues // @match *://bbs.binmt.cc/* // @exclude /^http(s|)://bbs.binmt.cc/uc_server.*$/ // @require https://fastly.jsdelivr.net/gh/WhiteSevs/TamperMonkeyScript@86be74b83fca4fa47521cded28377b35e1d7d2ac/lib/CoverUMD/index.js // @require https://fastly.jsdelivr.net/npm/@whitesev/[email protected]/dist/index.umd.js // @require https://fastly.jsdelivr.net/npm/@whitesev/[email protected]/dist/index.umd.js // @require https://fastly.jsdelivr.net/npm/@whitesev/[email protected]/dist/index.umd.js // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/index.umd.js // @require https://fastly.jsdelivr.net/npm/[email protected]/dist/viewer.min.js // @require https://fastly.jsdelivr.net/npm/@highlightjs/[email protected]/highlight.min.js // @resource HljsCSS https://fastly.jsdelivr.net/npm/[email protected]/styles/github-dark.min.css // @resource ViewerCSS https://fastly.jsdelivr.net/npm/[email protected]/dist/viewer.min.css // @connect * // @grant GM.cookie // @grant GM_addStyle // @grant GM_deleteValue // @grant GM_getResourceText // @grant GM_getValue // @grant GM_info // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_unregisterMenuCommand // @grant GM_xmlhttpRequest // @grant unsafeWindow // @run-at document-start // ==/UserScript== (t=>{function n(d){if(typeof d!="string")throw new TypeError("cssText must be a string");let e=document.createElement("style");return e.setAttribute("type","text/css"),e.innerHTML=d,document.head?document.head.appendChild(e):document.body?document.body.appendChild(e):document.documentElement.childNodes.length===0?document.documentElement.appendChild(e):document.documentElement.insertBefore(e,document.documentElement.childNodes[0]),e}if(typeof GM_addStyle=="function"){GM_addStyle(t);return}n(t)})(" .pls .avatar img,.avtm img{border-radius:10%}.pls .avatar img{--avatar-size: 90px;width:var(--avatar-size);height:var(--avatar-size)} "); (function (Qmsg, DOMUtils, Utils, pops, hljs, Viewer) { 'use strict'; var __defProp = Object.defineProperty; var __getOwnPropNames = Object.getOwnPropertyNames; var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value; var __commonJS = (cb, mod) => function __require() { return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports; }; var __publicField = (obj, key, value) => __defNormalProp(obj, key + "" , value); var require_entrance_001 = __commonJS({ "entrance-DcS4z3Fu.js"(exports, module) { var _a; var _GM = /* @__PURE__ */ (() => typeof GM != "undefined" ? GM : void 0)(); var _GM_deleteValue = /* @__PURE__ */ (() => typeof GM_deleteValue != "undefined" ? GM_deleteValue : void 0)(); var _GM_getResourceText = /* @__PURE__ */ (() => typeof GM_getResourceText != "undefined" ? GM_getResourceText : void 0)(); var _GM_getValue = /* @__PURE__ */ (() => typeof GM_getValue != "undefined" ? GM_getValue : void 0)(); var _GM_info = /* @__PURE__ */ (() => typeof GM_info != "undefined" ? GM_info : void 0)(); var _GM_registerMenuCommand = /* @__PURE__ */ (() => typeof GM_registerMenuCommand != "undefined" ? GM_registerMenuCommand : void 0)(); var _GM_setValue = /* @__PURE__ */ (() => typeof GM_setValue != "undefined" ? GM_setValue : void 0)(); var _GM_unregisterMenuCommand = /* @__PURE__ */ (() => typeof GM_unregisterMenuCommand != "undefined" ? GM_unregisterMenuCommand : void 0)(); var _GM_xmlhttpRequest = /* @__PURE__ */ (() => typeof GM_xmlhttpRequest != "undefined" ? GM_xmlhttpRequest : void 0)(); var _unsafeWindow = /* @__PURE__ */ (() => typeof unsafeWindow != "undefined" ? unsafeWindow : void 0)(); var _monkeyWindow = /* @__PURE__ */ (() => window)(); const HttpxCookieManager = { $data: { /** 是否启用 */ get enable() { return PopsPanel.getValue("httpx-use-cookie-enable"); }, /** 是否使用document.cookie */ get useDocumentCookie() { return PopsPanel.getValue("httpx-use-document-cookie"); }, cookieRule: [] }, /** * 补充cookie末尾分号 */ fixCookieSplit(str) { if (utils.isNotNull(str) && !str.trim().endsWith(";")) { str += ";"; } return str; }, /** * 合并两个cookie */ concatCookie(targetCookie, newCookie) { if (utils.isNull(targetCookie)) { return newCookie; } targetCookie = targetCookie.trim(); newCookie = newCookie.trim(); targetCookie = this.fixCookieSplit(targetCookie); if (newCookie.startsWith(";")) { newCookie = newCookie.substring(1); } return targetCookie.concat(newCookie); }, /** * 处理cookie * @param details * @returns */ handle(details) { if (details.fetch) { return; } if (!this.$data.enable) { return; } let ownCookie = ""; let url = details.url; if (url.startsWith("//")) { url = window.location.protocol + url; } let urlObj = new URL(url); if (this.$data.useDocumentCookie && urlObj.hostname.endsWith( window.location.hostname.split(".").slice(-2).join(".") )) { ownCookie = this.concatCookie(ownCookie, document.cookie.trim()); } for (let index = 0; index < this.$data.cookieRule.length; index++) { let rule = this.$data.cookieRule[index]; if (urlObj.hostname.match(rule.hostname)) { let cookie = PopsPanel.getValue(rule.key); if (utils.isNull(cookie)) { break; } ownCookie = this.concatCookie(ownCookie, cookie); } } if (utils.isNotNull(ownCookie)) { if (details.headers && details.headers["Cookie"]) { details.headers.Cookie = this.concatCookie( details.headers.Cookie, ownCookie ); } else { details.headers["Cookie"] = ownCookie; } log.info(["Httpx => 设置cookie:", details]); } if (details.headers && details.headers.Cookie != null && utils.isNull(details.headers.Cookie)) { delete details.headers.Cookie; } } }; const CommonUtil = { /** * 添加屏蔽CSS * @param args * @example * addBlockCSS("") * addBlockCSS("","") * addBlockCSS(["",""]) */ addBlockCSS(...args) { let selectorList = []; if (args.length === 0) { return; } if (args.length === 1 && typeof args[0] === "string" && args[0].trim() === "") { return; } args.forEach((selector) => { if (Array.isArray(selector)) { selectorList = selectorList.concat(selector); } else { selectorList.push(selector); } }); return addStyle(`${selectorList.join(",\n")}{display: none !important;}`); }, /** * 设置GM_getResourceText的style内容 * @param resourceMapData 资源数据 * @example * setGMResourceCSS({ * keyName: "ViewerCSS", * url: "https://example.com/example.css", * }) */ setGMResourceCSS(resourceMapData) { let cssText = typeof _GM_getResourceText === "function" ? _GM_getResourceText(resourceMapData.keyName) : ""; if (typeof cssText === "string" && cssText) { addStyle(cssText); } else { CommonUtil.loadStyleLink(resourceMapData.url); } }, /** * 添加<link>标签 * @param url * @example * loadStyleLink("https://example.com/example.css") */ async loadStyleLink(url) { let $link = document.createElement("link"); $link.rel = "stylesheet"; $link.type = "text/css"; $link.href = url; domUtils.ready(() => { document.head.appendChild($link); }); }, /** * 添加<script>标签 * @param url * @example * loadStyleLink("https://example.com/example.js") */ async loadScript(url) { let $script = document.createElement("script"); $script.src = url; return new Promise((resolve) => { $script.onload = () => { resolve(null); }; (document.head || document.documentElement).appendChild($script); }); }, /** * 将url修复,例如只有search的链接修复为完整的链接 * * 注意:不包括http转https * @param url 需要修复的链接 * @example * 修复前:`/xxx/xxx?ss=ssss` * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss` * @example * 修复前:`//xxx/xxx?ss=ssss` * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss` * @example * 修复前:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss` * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss` * @example * 修复前:`xxx/xxx?ss=ssss` * 修复后:`https://xxx.xxx.xxx/xxx/xxx?ss=ssss` */ fixUrl(url) { url = url.trim(); if (url.match(/^http(s|):\/\//i)) { return url; } else { if (!url.startsWith("/")) { url += "/"; } url = window.location.origin + url; return url; } }, /** * http转https * @param url 需要修复的链接 * @example * 修复前: * 修复后: * @example * 修复前: * 修复后: */ fixHttps(url) { if (url.startsWith("https://")) { return url; } if (!url.startsWith("http://")) { return url; } let urlObj = new URL(url); urlObj.protocol = "https:"; return urlObj.toString(); } }; const GM_RESOURCE_MAP = { Viewer: { keyName: "ViewerCSS", url: "https://fastly.jsdelivr.net/npm/viewerjs@latest/dist/viewer.min.css" }, Hljs: { keyName: "HljsCSS", url: "https://fastly.jsdelivr.net/npm/highlight.js@latest/styles/github-dark.min.css" } }; (function(global, factory) { if (typeof exports === "object" && typeof module !== "undefined") { module.exports = factory(); } else { global = typeof globalThis !== "undefined" ? globalThis : global || self; global.Watermark = factory(global.Watermark); } })(typeof window !== "undefined" ? window : void 0, function(AnotherWatermark) { let Watermark = function() { }; CanvasRenderingContext2D.prototype.letterSpacingText = function(text, x, y, letterSpacing) { var context = this; var canvas = context.canvas; if (!letterSpacing && canvas) { letterSpacing = parseFloat(window.getComputedStyle(canvas).letterSpacing); } if (!letterSpacing) { return this.fillText(text, x, y); } var arrText = text.split(""); var align = context.textAlign || "left"; var originWidth = context.measureText(text).width; var actualWidth = originWidth + letterSpacing * (arrText.length - 1); if (align == "center") { x = x - actualWidth / 2; } else if (align == "right") { x = x - actualWidth; } context.textAlign = "left"; arrText.forEach(function(letter) { var letterWidth = context.measureText(letter).width; context.fillText(letter, x, y); x = x + letterWidth + letterSpacing; }); context.textAlign = align; }; CanvasRenderingContext2D.prototype.wrapText = function(text, x, y, maxWidth, lineHeight, stroke) { if (typeof text != "string" || typeof x != "number" || typeof y != "number") { return; } var context = this; var canvas = context.canvas; if (typeof maxWidth == "undefined") { maxWidth = canvas && canvas.width || 300; } if (typeof lineHeight == "undefined") { lineHeight = canvas && parseInt(window.getComputedStyle(canvas).lineHeight) || parseInt(window.getComputedStyle(document.body).lineHeight); } var arrText = text.split(""); var line = ""; for (var n = 0; n < arrText.length; n++) { var testLine = line + arrText[n]; var metrics = context.measureText(testLine); var testWidth = metrics.width; if (testWidth > maxWidth && n > 0) { if (stroke) { context.strokeText(line, x, y, canvas.width); } else { context.fillText(line, x, y); } line = arrText[n]; y += lineHeight; } else { line = testLine; } } if (stroke) { context.strokeText(line, x, y, canvas.width); } else { context.fillText(line, x, y); } }; CanvasRenderingContext2D.prototype.fillTextVertical = function(text, x, y) { var context = this; context.canvas; var arrText = text.split(""); var arrWidth = arrText.map(function(letter) { return context.measureText(letter).width; }); var align = context.textAlign; var baseline = context.textBaseline; if (align == "left") { x = x + Math.max.apply(null, arrWidth) / 2; } else if (align == "right") { x = x - Math.max.apply(null, arrWidth) / 2; } if (baseline == "bottom" || baseline == "alphabetic" || baseline == "ideographic") { y = y - arrWidth[0] / 2; } else if (baseline == "top" || baseline == "hanging") { y = y + arrWidth[0] / 2; } context.textAlign = "center"; context.textBaseline = "middle"; arrText.forEach(function(letter, index) { var letterWidth = arrWidth[index]; var code = letter.charCodeAt(0); if (code <= 256) { context.translate(x, y); context.rotate(90 * Math.PI / 180); context.translate(-x, -y); } else if (index > 0 && text.charCodeAt(index - 1) < 256) { y = y + arrWidth[index - 1] / 2; } context.fillText(letter, x, y); context.setTransform(1, 0, 0, 1, 0, 0); var letterWidth = arrWidth[index]; y = y + letterWidth; }); context.textAlign = align; context.textBaseline = baseline; }; function loadFile(file) { let fileReader = new FileReader(); return new Promise((resolve) => { fileReader.onloadend = async function(event) { resolve(event); }; fileReader.readAsDataURL(file); }); } function loadImage(src) { let image = new Image(); return new Promise((resolve) => { image.onload = () => { resolve(image); }; image.src = src; }); } function checkInArrayByPos(arrayData, x, y) { let flag = false; Array.from(arrayData).forEach((item) => { if (item["x"] == x && item["y"] == y) { flag = true; return; } }); return flag; } function getRandValue(arr) { if (arr instanceof Array) { return arr[Math.floor(Math.random() * arr.length)]; } else { return arr; } } Watermark.prototype.setFile = function(file) { let that = this; return new Promise(async (resolve) => { try { var fileReader = await loadFile(file); await that.setImage(fileReader.target.result); resolve(true); } catch (error) { resolve(false); } }); }; Watermark.prototype.setImage = function(src) { this.dataUrl = src; let that = this; return new Promise(async (res) => { var image = await loadImage(src); that.sizes = { width: image.width, height: image.height }; var canvas = document.createElement("canvas"); canvas.width = that.sizes.width; canvas.height = that.sizes.height; var ctx = canvas.getContext("2d"); ctx.drawImage(image, 0, 0); image = null; that.canvas = canvas; res(true); }); }; Watermark.prototype.hasImage = function() { return !!this.dataUrl; }; Watermark.prototype.getSize = function() { return this.sizes; }; Watermark.prototype.clearMark = function() { let that = this; if (typeof that.canvas === "undefined") { return; } function _clearMark_() { var ctx = that.canvas.getContext("2d"); ctx.clearRect(0, 0, that.canvas.width, that.canvas.height); var w = that.canvas.width; var h = that.canvas.height; that.canvas.width = w; that.canvas.height = h; ctx.beginPath(); var image = new Image(); image.src = that.dataUrl; ctx.drawImage(image, 0, 0); image = null; } _clearMark_(); }; Watermark.prototype.addText = function(opts) { var options = { text: ["Call By waterMark.addText"], fontSize: "6vw", fontFamily: "Microsoft Yahei", color: "#000000", textAlign: "center", /* 描边 */ stroke: false, globalAlpha: 0.7, /* -360 ~ 360 */ rotateAngle: 50, /* 必须大于0 */ maxWidth: 100, /* 必须大于0 */ xMoveDistance: 30, /* 必须大于0 */ yMoveDistance: 30 }; for (let key in options) { if (typeof opts[key] !== "undefined") { options[key] = opts[key]; } } options.maxWidth = parseInt(options.maxWidth) > 0 ? options.maxWidth : 1; options.xMoveDistance = parseInt(options.xMoveDistance) > 0 ? options.xMoveDistance : 1; options.yMoveDistance = parseInt(options.yMoveDistance) > 0 ? options.yMoveDistance : 1; var ctx = this.canvas.getContext("2d"); var fontSize = options.fontSize; fontSize = fontSize.toString(); if (~fontSize.indexOf("vw")) { fontSize = (this.sizes.width / 100 * parseInt(fontSize)).toFixed(0); } fontSize = parseInt(fontSize); ctx.font = fontSize + "px " + options.fontFamily; ctx.fillStyle = options.color; ctx.textAlign = options.textAlign; ctx.globalAlpha = options.globalAlpha; let canvasWidth = this.sizes.width, canvasHeight = this.sizes.height; let rotateAngle = options.rotateAngle * Math.PI / 180; let xMoveDistance = options.xMoveDistance; let yMoveDistance = options.yMoveDistance; let maxWidth = options.maxWidth; let lineHeight = fontSize; let pos = []; for (var i = canvasWidth / 2; i < canvasWidth; i += xMoveDistance) { for (var j = canvasHeight / 2; j < canvasHeight; j += yMoveDistance) { if (!checkInArrayByPos(pos, i, j)) { pos = pos.concat({ x: i, y: j }); ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.translate(i, j); ctx.rotate(rotateAngle); ctx.wrapText( getRandValue(options.text), 0, 0, maxWidth, lineHeight, options.stroke ); } } for (var k = canvasHeight / 2; k > 0; k -= yMoveDistance) { if (!checkInArrayByPos(pos, i, k)) { pos = pos.concat({ x: i, y: k }); ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.translate(i, k); ctx.rotate(rotateAngle); ctx.wrapText( getRandValue(options.text), 0, 0, maxWidth, lineHeight, options.stroke ); } } } for (var i = canvasWidth / 2; i > 0; i -= xMoveDistance) { for (var j = canvasHeight / 2; j < canvasHeight; j += yMoveDistance) { if (!checkInArrayByPos(pos, i, j)) { pos = pos.concat({ x: i, y: j }); ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.translate(i, j); ctx.rotate(rotateAngle); ctx.wrapText( getRandValue(options.text), 0, 0, maxWidth, lineHeight, options.stroke ); } } for (var k = canvasHeight / 2; k > 0; k -= yMoveDistance) { if (!checkInArrayByPos(pos, i, k)) { pos = pos.concat({ x: i, y: k }); ctx.setTransform(1, 0, 0, 1, 0, 0); ctx.translate(i, k); ctx.rotate(rotateAngle); ctx.wrapText( getRandValue(options.text), 0, 0, maxWidth, lineHeight, options.stroke ); } } } }; Watermark.prototype.addPixelText = function(opts) { var options = { text: "像素文字水印", /* 像素文字 */ big: { fontSize: 150, fontFamily: "微软雅黑", textAlign: "center", rotateAngle: 0, /* 描边 */ stroke: false }, /* 绘制像素的文字 */ small: { fontSize: 10, fontFamily: "微软雅黑", color: "#000", textAlign: "center", globalAlpha: 0.7 } }; for (let key in options) { if (typeof opts[key] !== "undefined") { options[key] = opts[key]; } } var ctx = this.canvas.getContext("2d"); var tmpCanvas = document.createElement("canvas"); var tmpctx = tmpCanvas.getContext("2d"); tmpCanvas.width = this.sizes.width; tmpCanvas.height = this.sizes.height; tmpctx.font = options.big.fontSize + "px " + options.big.fontFamily; tmpctx.textAlign = options.big.textAlign; tmpctx.textBaseline = "middle"; tmpctx.translate(tmpCanvas.width / 2, tmpCanvas.height / 2); tmpctx.rotate(options.big.rotateAngle * Math.PI / 180); tmpctx.translate(-tmpCanvas.width / 2, -tmpCanvas.height / 2); if (options.big.stroke) { tmpctx.strokeText( options.text, tmpCanvas.width / 2, tmpCanvas.height / 2, tmpCanvas.width ); } else { tmpctx.fillText(options.text, tmpCanvas.width / 2, tmpCanvas.height / 2); } var textArray = options.text.split(""); var textPixleInfo = tmpctx.getImageData( 0, 0, tmpCanvas.width, tmpCanvas.height ); var pixelArray = []; for (var i = 0; i < tmpCanvas.height; i += options.small.fontSize) { for (var j = 0; j < tmpCanvas.width; j += options.small.fontSize) { var index = j + i * tmpCanvas.width; var a = textPixleInfo.data[index * 4 + 3]; if (a > 128) { pixelArray.push({ text: getRandValue(textArray), x: j, y: i }); } } } ctx.font = options.small.fontSize + "px " + options.small.fontFamily; ctx.fillStyle = options.small.color; ctx.textAlign = options.small.textAlign; ctx.textBaseline = "middle"; ctx.globalAlpha = options.small.globalAlpha; pixelArray.forEach((item) => { ctx.fillText(item.text, item.x, item.y); }); }; Watermark.prototype.addImage = function(opts) { if (opts.imageArray == null) { alert("参数缺少imageArray"); return false; } if (opts.imageArray.length === 0) { alert("参数imageArray不能为空"); return false; } let options = { imageArray: [], /* 里面为水印Image对象 */ width: 50, /* 必须大于0 */ height: 50, /* 必须大于0 */ globalAlpha: 0.5, rotateAngle: 0, xMoveDistance: 70, /* 必须大于0 */ yMoveDistance: 70 /* 必须大于0 */ }; for (let key in options) { if (typeof opts[key] !== "undefined") { options[key] = opts[key]; } } options.width = parseInt(options.width) > 0 ? options.width : 1; options.height = parseInt(options.height) > 0 ? options.height : 1; options.xMoveDistance = parseInt(options.xMoveDistance) > 0 ? options.xMoveDistance : 1; options.yMoveDistance = parseInt(options.yMoveDistance) > 0 ? options.yMoveDistance : 1; let ctx = this.canvas.getContext("2d"); let waterImageCanvasArray = []; let waterImageCanvasDiagonal = parseInt( Math.sqrt(options.width * options.width + options.height * options.height) ); let canvasWidth = this.sizes.width, canvasHeight = this.sizes.height; let rotateAngle = options.rotateAngle * Math.PI / 180; let xMoveDistance = options.xMoveDistance; let yMoveDistance = options.yMoveDistance; let centerDrawLeftPosX = canvasWidth / 2 - waterImageCanvasDiagonal / 2; let centerDrawLeftPosY = canvasHeight / 2 - waterImageCanvasDiagonal / 2; let waterDrawPosX = (waterImageCanvasDiagonal - options.width) / 2; let waterDrawPosY = (waterImageCanvasDiagonal - options.height) / 2; Array.from(options.imageArray).forEach((item) => { var waterImageCanvas = document.createElement("canvas"); var waterctx = waterImageCanvas.getContext("2d"); waterImageCanvas.width = waterImageCanvasDiagonal; waterImageCanvas.height = waterImageCanvasDiagonal; waterctx.globalAlpha = options.globalAlpha; waterctx.translate( waterImageCanvasDiagonal / 2, waterImageCanvasDiagonal / 2 ); waterctx.rotate(rotateAngle); waterctx.translate( -waterImageCanvasDiagonal / 2, -waterImageCanvasDiagonal / 2 ); waterctx.drawImage( item, waterDrawPosX, waterDrawPosY, options.width, options.height ); waterImageCanvasArray = waterImageCanvasArray.concat(waterImageCanvas); }); function randomArrayData(array_data) { return array_data[Math.floor(Math.random() * array_data.length)]; } ctx.setTransform(1, 0, 0, 1, 0, 0); let pos = []; for (let i = centerDrawLeftPosX; i < canvasWidth; i += xMoveDistance) { for (let j = centerDrawLeftPosY; j < canvasHeight; j += yMoveDistance) { if (!checkInArrayByPos(pos, i, j)) { pos = pos.concat({ x: i, y: j }); ctx.drawImage( randomArrayData(waterImageCanvasArray), i, j ); } } for (let k = centerDrawLeftPosY; k > -Math.abs(waterImageCanvasDiagonal); k -= yMoveDistance) { if (!checkInArrayByPos(pos, i, k)) { pos = pos.concat({ x: i, y: k }); ctx.drawImage(randomArrayData(waterImageCanvasArray), i, k); } } } for (let i = centerDrawLeftPosX; i > -Math.abs(waterImageCanvasDiagonal); i -= xMoveDistance) { for (let j = centerDrawLeftPosY; j < canvasHeight; j += yMoveDistance) { if (!checkInArrayByPos(pos, i, j)) { pos = pos.concat({ x: i, y: j }); ctx.drawImage(randomArrayData(waterImageCanvasArray), i, j); } } for (let k = centerDrawLeftPosY; k > -Math.abs(waterImageCanvasDiagonal); k -= yMoveDistance) { if (!checkInArrayByPos(pos, i, k)) { pos = pos.concat({ x: i, y: k }); ctx.drawImage(randomArrayData(waterImageCanvasArray), i, k); } } } }; Watermark.prototype.getPreview = function() { return this.dataUrl; }; Watermark.prototype.render = function(type) { type = type === "png" ? "png" : "jpeg"; return this.canvas.toDataURL("image/" + type); }; Watermark.prototype.renderBlob = function() { let that = this; return new Promise((res) => { that.canvas.toBlob(function(blob) { res(window.URL.createObjectURL(blob)); }); }); }; Watermark.prototype.noConflict = function() { if (window.Watermark) { delete window.Watermark; } if (AnotherWatermark) { window.Watermark = AnotherWatermark; } return Watermark; }; return Watermark; }); const _SCRIPT_NAME_ = "MT论坛优化"; const utils = Utils.noConflict(); const domUtils = DOMUtils.noConflict(); const __pops = pops; const log = new utils.Log( _GM_info, _unsafeWindow.console || _monkeyWindow.console ); const SCRIPT_NAME = ((_a = _GM_info == null ? void 0 : _GM_info.script) == null ? void 0 : _a.name) || _SCRIPT_NAME_; const DEBUG = false; log.config({ debug: DEBUG, logMaxCount: 1e3, autoClearConsole: true, tag: true }); Qmsg.config( Object.defineProperties( { html: true, autoClose: true, showClose: false }, { position: { get() { return PopsPanel.getValue("qmsg-config-position", "bottom"); } }, maxNums: { get() { return PopsPanel.getValue("qmsg-config-maxnums", 5); } }, showReverse: { get() { return PopsPanel.getValue("qmsg-config-showreverse", true); } }, zIndex: { get() { let maxZIndex = Utils.getMaxZIndex(); let popsMaxZIndex = pops.config.InstanceUtils.getPopsMaxZIndex().zIndex; return Utils.getMaxValue(maxZIndex, popsMaxZIndex) + 100; } } } ) ); const GM_Menu = new utils.GM_Menu({ GM_getValue: _GM_getValue, GM_setValue: _GM_setValue, GM_registerMenuCommand: _GM_registerMenuCommand, GM_unregisterMenuCommand: _GM_unregisterMenuCommand }); const httpx = new utils.Httpx(_GM_xmlhttpRequest); httpx.interceptors.request.use((data) => { HttpxCookieManager.handle(data); return data; }); httpx.interceptors.response.use(void 0, (data) => { log.error(["拦截器-请求错误", data]); if (data.type === "onabort") { Qmsg.warning("请求取消"); } else if (data.type === "onerror") { Qmsg.error("请求异常"); } else if (data.type === "ontimeout") { Qmsg.error("请求超时"); } else { Qmsg.error("其它错误"); } return data; }); httpx.config({ logDetails: DEBUG }); pops.GlobalConfig.setGlobalConfig({ mask: { enable: true, clickEvent: { toClose: false, toHide: false } } }); ({ Object: { defineProperty: _unsafeWindow.Object.defineProperty }, Function: { apply: _unsafeWindow.Function.prototype.apply, call: _unsafeWindow.Function.prototype.call }, Element: { appendChild: _unsafeWindow.Element.prototype.appendChild }, setTimeout: _unsafeWindow.setTimeout }); const addStyle = utils.addStyle.bind(utils); { CommonUtil.setGMResourceCSS(GM_RESOURCE_MAP.Viewer); } { CommonUtil.setGMResourceCSS(GM_RESOURCE_MAP.Hljs); } const $ = document.querySelector.bind(document); const $$ = document.querySelectorAll.bind(document); const KEY = "GM_Panel"; const ATTRIBUTE_INIT = "data-init"; const ATTRIBUTE_KEY = "data-key"; const ATTRIBUTE_DEFAULT_VALUE = "data-default-value"; const ATTRIBUTE_INIT_MORE_VALUE = "data-init-more-value"; const PROPS_STORAGE_API = "data-storage-api"; const UISwitch = function(text, key, defaultValue, clickCallBack, description, afterAddToUListCallBack) { let result = { text, type: "switch", description, attributes: {}, props: {}, getValue() { return Boolean( this.props[PROPS_STORAGE_API].get(key, defaultValue) ); }, callback(event, __value) { let value = Boolean(__value); log.success(`${value ? "开启" : "关闭"} ${text}`); this.props[PROPS_STORAGE_API].set(key, value); }, afterAddToUListCallBack }; Reflect.set(result.attributes, ATTRIBUTE_KEY, key); Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue); Reflect.set(result.props, PROPS_STORAGE_API, { get(key2, defaultValue2) { return PopsPanel.getValue(key2, defaultValue2); }, set(key2, value) { PopsPanel.setValue(key2, value); } }); return result; }; const UISelect = function(text, key, defaultValue, data, callback, description) { let selectData = []; if (typeof data === "function") { selectData = data(); } else { selectData = data; } let result = { text, type: "select", description, attributes: {}, props: {}, getValue() { return this.props[PROPS_STORAGE_API].get(key, defaultValue); }, callback(event, isSelectedValue, isSelectedText) { let value = isSelectedValue; log.info(`选择:${isSelectedText}`); this.props[PROPS_STORAGE_API].set(key, value); if (typeof callback === "function") { callback(event, value, isSelectedText); } }, data: selectData }; Reflect.set(result.attributes, ATTRIBUTE_KEY, key); Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue); Reflect.set(result.props, PROPS_STORAGE_API, { get(key2, defaultValue2) { return PopsPanel.getValue(key2, defaultValue2); }, set(key2, value) { PopsPanel.setValue(key2, value); } }); return result; }; const UIButton = function(text, description, buttonText, buttonIcon, buttonIsRightIcon, buttonIconIsLoading, buttonType, clickCallBack, afterAddToUListCallBack) { let result = { text, type: "button", description, buttonIcon, buttonIsRightIcon, buttonIconIsLoading, buttonType, buttonText, callback(event) { if (typeof clickCallBack === "function") { clickCallBack(event); } }, afterAddToUListCallBack }; return result; }; const UIOwn = function(getLiElementCallBack, initConfig, props, afterAddToUListCallBack) { let result = { type: "own", attributes: {}, props, getLiElementCallBack, afterAddToUListCallBack }; Reflect.set(result.attributes, ATTRIBUTE_INIT, () => { return false; }); return result; }; const MTRegExp = { /** 论坛账号的凭证 */ formhash: /formhash=([0-9a-zA-Z]+)/, /** 用户uid */ uid: /uid(=|-)(\d+)/, /** 帖子内特殊字体格式 */ fontSpecial: /<font.*?>|<\/font>|<strike>|<strong>|<i>|<u>|align=".*?"|<br>[\s]*<br>[\s]*<br>/g }; const MTUtils = { /** * 根据UID获取小|中|大头像 * @param uid * @param size */ getAvatar: (uid, size = "middle") => { return `/uc_server/avatar.php?uid=${uid}&size=${size}&ts=1`; }, /** * 获取当前已登录(不可用)的用户的uid */ getCurrentUID() { let discuz_uid = _unsafeWindow.discuz_uid; if (typeof discuz_uid === "string") { return discuz_uid; } let $exit = $('.sidenv_exit a[href*="uid-"]') || $('#comiis_key a[href*="uid-"]'); if ($exit) { let uidMatch = $exit.href.match(/uid=([0-9]+)/); if (uidMatch) { return uidMatch[uidMatch.length - 1]; } } }, /** * 获取账号的formhash */ getFormHash() { let $inputFormHashList = Array.from( (top || globalThis).document.querySelectorAll( "input[name=formhash]" ) ); for (let index = 0; index < $inputFormHashList.length; index++) { const $input = $inputFormHashList[index]; let formHash = $input.value; if (formHash) { return formHash; } } let $anchorFormHashList = Array.from( (top || globalThis).document.querySelectorAll( 'a[href*="formhash="]' ) ); for (let index = 0; index < $anchorFormHashList.length; index++) { const $anchorFormHash = $anchorFormHashList[index]; let anchorFormHashMatch = $anchorFormHash.href.match(MTRegExp.formhash); if (anchorFormHashMatch) { let formHash = anchorFormHashMatch[anchorFormHashMatch.length - 1]; if (formHash) { return formHash; } } } }, /** * 检测环境模板 */ envIsMobile() { return ( // @ts-ignore (_unsafeWindow.STYLEID || // @ts-ignore window.STYLEID || // @ts-ignore typeof STYLEID !== "undefined" && STYLEID) === "3" ); }, /** * 获取帖子id * @param url */ getThreadId: (url) => { let urlMatch = url.match(/thread-([\d]+)-|&tid=([\d]+)/i); if (urlMatch) { let forumIdList = urlMatch.filter(Boolean); let forumId = forumIdList[forumIdList.length - 1]; return forumId; } }, /** * 获取板块?id */ getForumId(url) { let urlMatch = url.match(/&fid=([\d]+)/i); if (urlMatch) { return urlMatch[urlMatch.length - 1]; } }, /** * 获取发布id */ getPostId(url) { let urlMatch = url.match(/&pid=([\d]+)/i); if (urlMatch) { return urlMatch[urlMatch.length - 1]; } }, /** * 获取回复id */ getRepquote(url) { let urlMatch = url.match(/&repquote=([\d]+)/i); if (urlMatch) { return urlMatch[urlMatch.length - 1]; } } }; const MTDyncmicAvatar = { $upload: { small: false, middle: false, big: false }, $data: { /** * 图片文件最大大小 */ avatarInfo: { maxSize: 2097152, small: { width: 48, height: 48 }, middle: { width: 120, height: 120 }, big: { width: 200, height: 250 } } }, $el: { $smallUpload: null, $middleUpload: null, $bigUpload: null, $smallStatus: null, $middleStatus: null, $bigStatus: null }, $avatar: { get small() { return MTDyncmicAvatar.$el.$smallUpload.files[0]; }, get middle() { return MTDyncmicAvatar.$el.$middleUpload.files[0]; }, get big() { return MTDyncmicAvatar.$el.$bigUpload.files[0]; } }, init() { this.showView(); }, showView() { const that = this; let $confirm = __pops.confirm({ title: { text: "修改头像", position: "center" }, content: { text: ( /*html*/ ` <div class="avatar-container"> <p class="avatar-tip">1. 小头像(图片宽高限制最大尺寸:48×48)</p> <p class="avatar-upload-status" data-type="small">🤡请先上传图片</p> <input type="file" class="avatar-upload" data-type="small" data-maxwidth="48" data-maxheight="48" accept="image/*"> </div> <div class="avatar-container"> <p class="avatar-tip">2. 中头像(图片宽高限制最大尺寸:120×120)</p> <p class="avatar-upload-status" data-type="middle">🤡请先上传图片</p> <input type="file" class="avatar-upload" data-type="middle" data-maxwidth="120" data-maxheight="120" accept="image/*"> </div> <div class="avatar-container"> <p class="avatar-tip">3. 大头像(图片宽高限制最大尺寸:200×250)</p> <p class="avatar-upload-status" data-type="big">🤡请先上传图片</p> <input type="file" class="avatar-upload" data-type="big" data-maxwidth="200" data-maxheight="250" accept="image/*"> </div> ` ), html: true }, btn: { ok: { text: "上传", callback: async () => { if (!that.$upload.small) { Qmsg.error("请上传小头像"); return; } if (!that.$upload.middle) { Qmsg.error("请上传中头像"); return; } if (!that.$upload.big) { Qmsg.error("请上传大头像"); return; } let $loading = Qmsg.loading("正在处理数据中..."); try { let uploadUrl = await this.getUploadUrl(); if (uploadUrl == null) { return; } let formhash = MTUtils.getFormHash(); if (formhash == null) { Qmsg.error("获取formhash失败"); return; } let avatarInfo = { big: { base64: await utils.parseFileToBase64(this.$avatar.big) }, middle: { base64: await utils.parseFileToBase64(this.$avatar.middle) }, small: { base64: await utils.parseFileToBase64(this.$avatar.small) } }; Object.keys(avatarInfo).forEach((keyName) => { let value = avatarInfo[keyName]; value.base64 = value.base64.substring( value.base64.indexOf(",") + 1 ); }); let formData = new FormData(); formData.append("Filedata", this.$avatar.big || ""); formData.append("confirm", "确定"); formData.append("avatar1", avatarInfo.big.base64); formData.append("avatar2", avatarInfo.middle.base64); formData.append("avatar3", avatarInfo.small.base64); formData.append("formhash", formhash); log.info(`头像的base64字符串`, avatarInfo); let response = await httpx.post(uploadUrl, { data: formData, processData: false, headers: { Accept: "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8,application/signed-exchange;v=b3;q=0.9", "User-Agent": utils.getRandomPCUA(), Host: window.location.hostname, Origin: window.location.origin, Referer: `${window.location.origin}/home.php?mod=spacecp&ac=avatar` } }); if (!response.status) { return; } if (response.data.responseText.indexOf( "window.parent.postMessage('success','*')" ) != -1) { $confirm.close(); Qmsg.success("上传成功"); } else { log.error("上传失败", response); Qmsg.error(response.data.responseText, { timeout: 6e3, isHTML: false, html: false }); } } catch (error) { log.error(error); } finally { $loading.close(); } } } }, mask: { enable: true }, width: window.innerWidth > 500 ? "500px" : "88vw", height: window.innerHeight > 500 ? "500px" : "80vh", style: ( /*css*/ ` .avatar-container{ display: flex; width: -webkit-fill-available; width: -moz-available; margin: 6px 10px; flex-direction: column; } .avatar-tip{ float: left; font-weight: bold; } .avatar-upload-status { padding: 0px; padding-left: 10px; font-weight: bold; width: -webkit-fill-available; text-align: left; font-size: small; } .avatar-upload-status[data-success="false"]{ color: red; } .avatar-upload-status[data-success="true"]{ color: green; } .avatar-upload { margin: 20px 0px; } ` ) }); this.$el.$smallUpload = $confirm.$shadowRoot.querySelector( ".avatar-upload[data-type='small']" ); this.$el.$middleUpload = $confirm.$shadowRoot.querySelector( ".avatar-upload[data-type='middle']" ); this.$el.$bigUpload = $confirm.$shadowRoot.querySelector( ".avatar-upload[data-type='big']" ); this.$el.$smallStatus = $confirm.$shadowRoot.querySelector( ".avatar-upload-status[data-type='small']" ); this.$el.$middleStatus = $confirm.$shadowRoot.querySelector( ".avatar-upload-status[data-type='middle']" ); this.$el.$bigStatus = $confirm.$shadowRoot.querySelector( ".avatar-upload-status[data-type='big']" ); this.setUploadChangeEvent( this.$el.$smallUpload, this.$el.$smallStatus, this.$data.avatarInfo.small, () => { this.$upload.small = true; } ); this.setUploadChangeEvent( this.$el.$middleUpload, this.$el.$middleStatus, this.$data.avatarInfo.middle, () => { this.$upload.middle = true; } ); this.setUploadChangeEvent( this.$el.$bigUpload, this.$el.$bigStatus, this.$data.avatarInfo.big, () => { this.$upload.big = true; } ); }, /** * 设置文件改变事件 */ setUploadChangeEvent($file, $status, sizeInfo, successCallBack) { domUtils.on($file, "change", (event) => { var _a2; if (!((_a2 = $file.files) == null ? void 0 : _a2.length)) { return; } domUtils.text($status, "🤡获取文件信息中..."); $status.removeAttribute("data-success"); let uploadImageFile = $file.files[0]; let fileSize = uploadImageFile.size; let $image = new Image(); let reader = new FileReader(); reader.readAsDataURL(uploadImageFile); reader.onload = function(response) { $image.src = response.target.result; $image.onload = function() { if ($image.width > sizeInfo.width || $image.height > sizeInfo.height) { $file.value = ""; $status.setAttribute("data-success", "false"); domUtils.text( $status, `🤡校验失败 ==> 图片尺寸不符合,宽:${$image.width} 高:${$image.height}` ); return; } if (fileSize > MTDyncmicAvatar.$data.avatarInfo.maxSize) { $file.value = ""; $status.setAttribute("data-success", "false"); domUtils.text( $status, `🤡校验失败 ==> 图片大小不符合:${fileSize}byte,限制最大:${MTDyncmicAvatar.$data.avatarInfo.maxSize}byte` ); return; } $status.setAttribute("data-success", "true"); domUtils.text( $status, `🤣 通过 宽:${$image.width} 高:${$image.height} 大小(byte):${fileSize}` ); successCallBack(); }; }; }); }, /** * 获取上传地址 */ async getUploadUrl() { let response = await httpx.get("/home.php?mod=spacecp&ac=avatar", { headers: { "User-Agent": utils.getRandomPCUA() } }); if (!response.status) { return; } if (utils.isNull(response.data.responseText)) { Qmsg.error("动态头像:获取上传地址失败"); return; } let dataMatch = response.data.responseText.match( /var[\s]*data[\s]*=[\s]*"(.+?)"/ ); if (dataMatch == null || dataMatch.length != 2) { Qmsg.error("动态头像:获取变量data失败"); return; } let data = dataMatch[dataMatch.length - 1]; let data_split = data.split(","); let uploadUrl = data_split[data_split.indexOf("src") + 1].replace( "images/camera.swf?inajax=1", "index.php?m=user&a=rectavatar&base64=yes" ); log.info(`上传地址:` + uploadUrl); return uploadUrl; } }; const MTAutoSignIn = { $key: { signTime: "mt-sign-time" }, init() { this.sign(); }, /** * 判断今日是否已签到 */ todayIsSign() { let signTime = this.getSignTime(); if (signTime == null) { return false; } if (utils.formatTime(Date.now(), "yyyyMMdd") !== utils.formatTime(signTime, "yyyyMMdd")) { return false; } return true; }, /** * 设置签到时间 */ setSignTime() { _GM_setValue(this.$key.signTime, Date.now()); }, /** * 获取签到时间 */ getSignTime() { return _GM_getValue(this.$key.signTime); }, /** * 清空签到信息 */ clearSignTime() { _GM_deleteValue(this.$key.signTime); }, /** * 检测是否登录(不可用) */ checkLogin() { if (MTUtils.envIsMobile()) { let mobile_login_exitBtn = $( "a[href*='member.php?mod=logging&action=logout']" ); return Boolean(mobile_login_exitBtn); } else { let pc_login = $("#comiis_key"); return Boolean(pc_login); } }, /** * 签到 */ async sign() { let formHash = MTUtils.getFormHash(); if (formHash == null) { if ($("#comiis_picshowbox")) { log.info("当前为评论区的看图模式 "); return; } log.error("自动签到:获取账号formhash失败"); _GM_deleteValue("mt_sign"); Qmsg.error({ content: "自动签到:获取账号formhash失败" }); return; } if (this.todayIsSign()) { log.info("今日已签到"); return; } let useFetch = Boolean(PopsPanel.getValue("mt-auto-sign-useFetch")); let userAgent = utils.getRandomPCUA(); let signSuccessCallBack = () => { this.setSignTime(); }; let signFailedCallBack = () => { this.clearSignTime(); }; let unknownSignContentCallback = (content) => { let $alert = pops.alert({ title: { text: "未知签到内容", position: "center" }, content: { text: "", html: false }, width: "88vw", height: "300px" }); let $content = $alert.$shadowRoot.querySelector( ".pops-alert-content" ); $content.innerText = content; }; let sign_plugin = [ { id: "k_misign", async sign() { let searchParamsData = { operation: "qiandao", format: "button", formhash: formHash, inajax: 1, ajaxtarget: "midaben_sign" }; let response = await httpx.get( `/k_misign-sign.html?${utils.toSearchParamsStr(searchParamsData)}`, { fetch: useFetch, headers: { "User-Agent": userAgent }, allowInterceptConfig: false } ); if (!response.status) { Qmsg.error("签到:网络异常,请求失败", { consoleLogContent: true }); return; } signSuccessCallBack(); log.info("签到信息:", response); let responseText = response.data.responseText; let CDATA = utils.parseCDATA(responseText); let CDATAElement = domUtils.parseHTML( `<div>${CDATA}</div>`, true, false ); let content = domUtils.text(CDATAElement); if (content.includes("需要先登录(不可用)")) { Qmsg.error("签到:请先登录(不可用)账号", { timeout: 3e3, consoleLogContent: true }); signFailedCallBack(); return; } else if (content.includes("请稍后再试") || content.includes("您已经被列入黑名单") || content.includes("绑定手机号后才可以签到") || content.includes("您所在用户组不允许使用")) { Qmsg.error("签到:" + content, { timeout: 5e3 }); return; } else if (content.includes("今日已签") || content.includes("今日已经签到")) { Qmsg.info("签到:" + content); return; } else if (responseText.includes( "您当前的访问请求当中含有非法字符,已经被系统拒绝" )) { Qmsg.error( "签到: 您当前的访问请求当中含有非法字符,已经被系统拒绝", { timeout: 6e3 } ); return; } else if (useFetch && "location" in utils.toJSON(responseText)) { Qmsg.success("签到: 签到成功"); return; } let signIn_con = CDATAElement.querySelector(".con"); let signIn_line = CDATAElement.querySelector(".line"); if (signIn_con && signIn_line) { let conMatch = domUtils.text(signIn_con).match(/([0-9]+)金币/); let lineMatch = domUtils.text(signIn_line).match(/([0-9]+)/); let con = conMatch[conMatch.length - 1]; let line = lineMatch[lineMatch.length - 1]; log.success(`金币${con},排名${line}`); Qmsg.info( /*html*/ ` <div style="display: flex;${!MTUtils.envIsMobile() ? "padding: 20px;" : ""}"> <div style="align-self: center;margin-right: 20px;">签到</div> <div>排名 ${line}<br>金币 ${con}</div> </div>`, { timeout: 4e3, isHTML: true } ); return; } unknownSignContentCallback(responseText); } }, { id: "dsu_paulsign", async sign() { let searchParamsData = { id: "dsu_paulsign:sign", operation: "qiandao", infloat: 1, inajax: 1 }; let response = await httpx.post( `/plugin.php?${utils.toSearchParamsStr(searchParamsData)}`, { data: { formhash: formHash, qdxq: "kx", qdmode: 3, todaysay: "", fastreply: 0 }, processData: true, fetch: useFetch, headers: { "User-Agent": userAgent, "Content-Type": "application/x-www-form-urlencoded" }, allowInterceptConfig: false } ); if (!response.status) { Qmsg.error("签到:网络异常,请求失败", { consoleLogContent: true }); return; } signSuccessCallBack(); log.info("签到信息:", response); let responseText = response.data.responseText; if (responseText.includes("签到成功")) { Qmsg.success("签到:签到成功"); return; } if (responseText.includes("今日已经签到")) { Qmsg.info("签到:您今日已经签到,请明天再来!"); return; } unknownSignContentCallback(responseText); } } ]; for (let index = 0; index < sign_plugin.length; index++) { const signPluginItem = sign_plugin[index]; let checkResponse = await httpx.get( `/plugin.php?id=${signPluginItem.id}:sign`, { headers: { "User-Agent": utils.getRandomPCUA() }, allowInterceptConfig: false } ); if (!checkResponse.status) { log.error("签到:检查签到插件是否启用的请求失败", checkResponse); continue; } let checkDoc = domUtils.parseHTML( checkResponse.data.responseText, true, true ); if (checkDoc.querySelector("#messagetext")) { log.error(`插件:${signPluginItem.id} 未启用或不存在`); continue; } await signPluginItem.sign(); break; } } }; const Component_Common = { id: "component-common", title: "通用", forms: [ { text: "", type: "forms", forms: [ { text: "Toast配置", type: "deepMenu", forms: [ { text: "", type: "forms", forms: [ UISelect( "Toast位置", "qmsg-config-position", "bottom", [ { value: "topleft", text: "左上角" }, { value: "top", text: "顶部" }, { value: "topright", text: "右上角" }, { value: "left", text: "左边" }, { value: "center", text: "中间" }, { value: "right", text: "右边" }, { value: "bottomleft", text: "左下角" }, { value: "bottom", text: "底部" }, { value: "bottomright", text: "右下角" } ], (event, isSelectValue, isSelectText) => { log.info("设置当前Qmsg弹出位置" + isSelectText); }, "Toast显示在页面九宫格的位置" ), UISelect( "最多显示的数量", "qmsg-config-maxnums", 3, [ { value: 1, text: "1" }, { value: 2, text: "2" }, { value: 3, text: "3" }, { value: 4, text: "4" }, { value: 5, text: "5" } ], void 0, "限制Toast显示的数量" ), UISwitch( "逆序弹出", "qmsg-config-showreverse", false, void 0, "修改Toast弹出的顺序" ) ] } ] } // { // text: "Cookie配置", // type: "deepMenu", // forms: [ // { // text: "", // type: "forms", // forms: [ // UISwitch( // "启用", // "httpx-use-cookie-enable", // false, // void 0, // "启用后,将根据下面的配置进行添加cookie" // ), // UISwitch( // "使用document.cookie", // "httpx-use-document-cookie", // false, // void 0, // "自动根据请求的域名来设置对应的cookie" // ), // UITextArea( // "bbs.binmt.cc", // "httpx-cookie-bbs.binmt.cc", // "", // void 0, // void 0, // "Cookie格式:xxx=xxxx;xxx=xxxx" // ), // ], // }, // ], // }, ] }, { text: "", type: "forms", forms: [ { text: "功能", type: "deepMenu", forms: [ { text: "", type: "forms", forms: [ UISwitch( "新增【最新发表】", "mt-addLatestPostBtn", true, void 0, "便于快捷跳转" ), UISwitch( "超链接文字转换", "mt-link-text-to-hyperlink", true, void 0, "自动把符合超链接格式的文字转为超链接" ), UISwitch( "延长登录(不可用)Cookie过期时间", "mt-extend-cookie-expire", true, void 0, "减少频繁登录(不可用)账号的问题" ) ] } ] }, { type: "deepMenu", text: "自动签到", forms: [ { text: "", type: "forms", forms: [ UISwitch("启用", "mt-auto-sign", true, void 0, "自动请求签到"), UISwitch( "使用fetch请求", "mt-auto-sign-useFetch", false, void 0, "" ), UIButton( "签到信息", `上次签到时间:${MTAutoSignIn.getSignTime() == null ? "尚未签到" : Utils.formatTime(MTAutoSignIn.getSignTime())}`, "清空信息", void 0, void 0, void 0, "primary", (event) => { let $click = event.composedPath()[0]; let $desc = $click.closest("li").querySelector( ".pops-panel-item-left-desc-text" ); __pops.confirm({ title: { text: "提示 ", position: "center" }, content: { text: "<p>是否清空脚本签到记录的时间?</p>", html: true }, btn: { ok: { enable: true, callback: (event2) => { MTAutoSignIn.clearSignTime(); Qmsg.success("删除成功"); domUtils.text( $desc, `上次签到时间:${MTAutoSignIn.getSignTime() == null ? "尚未签到" : Utils.formatTime(MTAutoSignIn.getSignTime())}` ); event2.close(); } } }, mask: { enable: true }, width: "300px", height: "200px" }); } ) ] } ] }, { text: "头像", type: "deepMenu", forms: [ { text: "<a href='https://ezgif.com/resize' target='_blank'>Resize Image</a>", type: "forms", forms: [ UIOwn(($li) => { let $left = domUtils.createElement("div", { className: "pops-panel-item-left-text", innerHTML: ( /*html*/ ` <p class="pops-panel-item-left-main-text">头像(有缓存)</p> <p class="pops-panel-item-left-desc-text">小、中、大</p> ` ) }); let $right = domUtils.createElement("div", { className: "pops-panel-avatar-img", innerHTML: ( /*html*/ ` <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=small" class="avatar-img" data-size="small"> <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=middle" class="avatar-img" data-size="middle"> <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=big" class="avatar-img" data-size="big"> ` ) }); let $style = domUtils.createElement("style", { innerHTML: ( /*css*/ ` .avatar-img { width: 30px; height: 30px; border-radius: 50%; overflow: hidden; } ` ) }); $right.querySelector( ".avatar-img[data-size='small']" ); $right.querySelector( ".avatar-img[data-size='middle']" ); $right.querySelector( ".avatar-img[data-size='big']" ); $li.appendChild($left); $li.appendChild($right); $li.appendChild($style); return $li; }), UIOwn(($li) => { let $left = domUtils.createElement("div", { className: "pops-panel-item-left-text", innerHTML: ( /*html*/ ` <p class="pops-panel-item-left-main-text">头像</p> <p class="pops-panel-item-left-desc-text">小、中、大</p> ` ) }); let $right = domUtils.createElement("div", { className: "pops-panel-avatar-img", innerHTML: ( /*html*/ ` <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=small&ts=${Date.now()}" class="avatar-img" data-size="small"> <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=middle&ts=${Date.now()}" class="avatar-img" data-size="middle"> <img src="/uc_server/avatar.php?uid=${MTUtils.getCurrentUID()}&size=big&ts=${Date.now()}" class="avatar-img" data-size="big"> ` ) }); $li.appendChild($left); $li.appendChild($right); return $li; }), UIButton( "修改头像", `可以上传gif图片,注意图片最大限制为${Utils.formatByteToSize( MTDyncmicAvatar.$data.avatarInfo.maxSize )}`, "上传", void 0, false, false, "primary", () => { MTDyncmicAvatar.init(); } ) ] } ] } ] } ] }; const Component_ForumPost = { id: "component-forum-post", title: "帖子", forms: [ { type: "forms", text: "", forms: [ { text: "功能", type: "deepMenu", forms: [ { type: "forms", text: "", forms: [ UISwitch( "拦截附件", "mt-forum-post-interceptionAttachment", true, void 0, "点击附件时弹出提示框进行确认是否下载附件" ), UISwitch( "图片查看优化", "mt-forum-post-optimizationImagePreview", true, void 0, "使用Viewer查看图片" ), UISwitch( "自动加载下一页", "mt-forum-post-loadNextPageComment", true, void 0, "无缝预览下一页" ), UISwitch( "代码块优化", "mt-forum-post-codeQuoteOptimization", true, void 0, "自动检测代码块语言并设置关键字高亮" ) ] } ] }, { type: "deepMenu", text: "用户信息块", forms: [ { type: "forms", text: "", forms: [ UISwitch( "探测用户在线状态", "mt-forum-post-detectingUserOnlineStatus", false, void 0, "获取用户在线状态并在用户信息处显示状态表情" ), UISwitch( "显示用户等级", "mt-forum-post-showUserLevel", true, void 0, "在用户信息处显示当前用户的等级" ), UISwitch( "隐藏底部信息块", "mt-forum-post-hideBottomInfoBlock", false, void 0, "包括金币、好评、信誉等信息" ) ] } ] }, { type: "deepMenu", text: "右侧悬浮工具栏", forms: [ { type: "forms", text: "", forms: [ UISwitch( "新增【快捷收藏】", "mt-forum-post-quickCollentBtn", true, void 0, "在右侧悬浮工具栏添加【收藏】按钮,用于快捷收藏" ), UISwitch( "快捷回复优化", "mt-forum-post-quickReplyOptimization", true, void 0, "为快捷回复弹窗底部区域添加【一键空格】按钮" ) ] } ] } ] } ] }; const Component_Guide = { id: "component-guide", title: "导读", forms: [ { type: "forms", text: "", forms: [ UISwitch( "页面美化", "mt-guide-beautifyPage", true, void 0, "美化样式" ) ] } ] }; const PanelUISize = { /** * 一般设置界面的尺寸 */ setting: { get width() { return window.innerWidth < 550 ? "88vw" : "550px"; }, get height() { return window.innerHeight < 450 ? "70vh" : "450px"; } }, /** * 功能丰富,aside铺满了的设置界面,要稍微大一点 */ settingBig: { get width() { return window.innerWidth < 800 ? "92vw" : "800px"; }, get height() { return window.innerHeight < 600 ? "80vh" : "600px"; } } }; const PopsPanel = { /** 数据 */ $data: { __data: null, __oneSuccessExecMenu: null, __onceExec: null, __listenData: null, /** * 菜单项的默认值 */ get data() { if (PopsPanel.$data.__data == null) { PopsPanel.$data.__data = new utils.Dictionary(); } return PopsPanel.$data.__data; }, /** * 成功只执行了一次的项 */ get oneSuccessExecMenu() { if (PopsPanel.$data.__oneSuccessExecMenu == null) { PopsPanel.$data.__oneSuccessExecMenu = new utils.Dictionary(); } return PopsPanel.$data.__oneSuccessExecMenu; }, /** * 成功只执行了一次的项 */ get onceExec() { if (PopsPanel.$data.__onceExec == null) { PopsPanel.$data.__onceExec = new utils.Dictionary(); } return PopsPanel.$data.__onceExec; }, /** 脚本名,一般用在设置的标题上 */ get scriptName() { return SCRIPT_NAME; }, /** 菜单项的总值在本地数据配置的键名 */ key: KEY, /** 菜单项在attributes上配置的菜单键 */ attributeKeyName: ATTRIBUTE_KEY, /** 菜单项在attributes上配置的菜单默认值 */ attributeDefaultValueName: ATTRIBUTE_DEFAULT_VALUE }, /** 监听器 */ $listener: { /** * 值改变的监听器 */ get listenData() { if (PopsPanel.$data.__listenData == null) { PopsPanel.$data.__listenData = new utils.Dictionary(); } return PopsPanel.$data.__listenData; } }, init() { this.initPanelDefaultValue(); this.initExtensionsMenu(); }, /** 判断是否是顶层窗口 */ isTopWindow() { return _unsafeWindow.top === _unsafeWindow.self; }, /** 初始化进行注册(不可用)油猴菜单 */ initExtensionsMenu() { if (!this.isTopWindow()) { return; } GM_Menu.add([ { key: "show_pops_panel_setting", text: "⚙ 设置", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showPanel(); } } ]); }, /** 初始化菜单项的默认值保存到本地数据中 */ initPanelDefaultValue() { let that = this; function initDefaultValue(config) { if (!config.attributes) { return; } let needInitConfig = {}; let key = config.attributes[ATTRIBUTE_KEY]; if (key != null) { needInitConfig[key] = config.attributes[ATTRIBUTE_DEFAULT_VALUE]; } let __attr_init__ = config.attributes[ATTRIBUTE_INIT]; if (typeof __attr_init__ === "function") { let __attr_result__ = __attr_init__(); if (typeof __attr_result__ === "boolean" && !__attr_result__) { return; } } let initMoreValue = config.attributes[ATTRIBUTE_INIT_MORE_VALUE]; if (initMoreValue && typeof initMoreValue === "object") { Object.assign(needInitConfig, initMoreValue); } let needInitConfigList = Object.keys(needInitConfig); if (!needInitConfigList.length) { log.warn(["请先配置键", config]); return; } needInitConfigList.forEach((__key) => { let __defaultValue = needInitConfig[__key]; if (that.$data.data.has(__key)) { log.warn("请检查该key(已存在): " + __key); } that.$data.data.set(__key, __defaultValue); }); } function loopInitDefaultValue(configList) { for (let index = 0; index < configList.length; index++) { let configItem = configList[index]; initDefaultValue(configItem); let childForms = configItem.forms; if (childForms && Array.isArray(childForms)) { loopInitDefaultValue(childForms); } } } let contentConfigList = this.getPanelContentConfig(); for (let index = 0; index < contentConfigList.length; index++) { let leftContentConfigItem = contentConfigList[index]; if (!leftContentConfigItem.forms) { continue; } let rightContentConfigList = leftContentConfigItem.forms; if (rightContentConfigList && Array.isArray(rightContentConfigList)) { loopInitDefaultValue(rightContentConfigList); } } }, /** * 设置值 * @param key 键 * @param value 值 */ setValue(key, value) { let locaData = _GM_getValue(KEY, {}); let oldValue = locaData[key]; locaData[key] = value; _GM_setValue(KEY, locaData); if (this.$listener.listenData.has(key)) { this.$listener.listenData.get(key).callback(key, oldValue, value); } }, /** * 获取值 * @param key 键 * @param defaultValue 默认值 */ getValue(key, defaultValue) { let locaData = _GM_getValue(KEY, {}); let localValue = locaData[key]; if (localValue == null) { if (this.$data.data.has(key)) { return this.$data.data.get(key); } return defaultValue; } return localValue; }, /** * 删除值 * @param key 键 */ deleteValue(key) { let locaData = _GM_getValue(KEY, {}); let oldValue = locaData[key]; Reflect.deleteProperty(locaData, key); _GM_setValue(KEY, locaData); if (this.$listener.listenData.has(key)) { this.$listener.listenData.get(key).callback(key, oldValue, void 0); } }, /** * 监听调用setValue、deleteValue * @param key 需要监听的键 * @param callback */ addValueChangeListener(key, callback, option) { let listenerId = Math.random(); this.$listener.listenData.set(key, { id: listenerId, key, callback }); if (option) { if (option.immediate) { callback(key, this.getValue(key), this.getValue(key)); } } return listenerId; }, /** * 移除监听 * @param listenerId 监听的id */ removeValueChangeListener(listenerId) { let deleteKey = null; for (const [key, value] of this.$listener.listenData.entries()) { if (value.id === listenerId) { deleteKey = key; break; } } if (typeof deleteKey === "string") { this.$listener.listenData.delete(deleteKey); } else { console.warn("没有找到对应的监听器"); } }, /** * 主动触发菜单值改变的回调 * @param key 菜单键 * @param newValue 想要触发的新值,默认使用当前值 * @param oldValue 想要触发的旧值,默认使用当前值 */ triggerMenuValueChange(key, newValue, oldValue) { if (this.$listener.listenData.has(key)) { let listenData = this.$listener.listenData.get(key); if (typeof listenData.callback === "function") { let value = this.getValue(key); let __newValue = value; let __oldValue = value; if (typeof newValue !== "undefined" && arguments.length > 1) { __newValue = newValue; } if (typeof oldValue !== "undefined" && arguments.length > 2) { __oldValue = oldValue; } listenData.callback(key, __oldValue, __newValue); } } }, /** * 判断该键是否存在 * @param key 键 */ hasKey(key) { let locaData = _GM_getValue(KEY, {}); return key in locaData; }, /** * 自动判断菜单是否启用,然后执行回调 * @param key * @param callback 回调 * @param isReverse 逆反判断菜单启用 * @param checkEnableCallBack 自定义检测菜单的值,可自行决定是否强制启用菜单,true是启用菜单,false是不启用菜单 */ execMenu(key, callback, isReverse = false, checkEnableCallBack) { if (!(typeof key === "string" || typeof key === "object" && Array.isArray(key))) { throw new TypeError("key 必须是字符串或者字符串数组"); } let runKeyList = []; if (typeof key === "object" && Array.isArray(key)) { runKeyList = [...key]; } else { runKeyList.push(key); } let value = void 0; for (let index = 0; index < runKeyList.length; index++) { const runKey = runKeyList[index]; if (!this.$data.data.has(runKey)) { log.warn(`${key} 键不存在`); return; } let runValue = PopsPanel.getValue(runKey); if (isReverse) { runValue = !runValue; } if (typeof checkEnableCallBack === "function") { let checkResult = checkEnableCallBack(runKey, runValue); if (typeof checkResult === "boolean") { runValue = checkResult; } } if (!runValue) { break; } value = runValue; } if (value) { callback(value); } }, /** * 自动判断菜单是否启用,然后执行回调,只会执行一次 * @param key * @param callback 回调 * @param getValueFn 自定义处理获取当前值,值true是启用并执行回调,值false是不执行回调 * @param handleValueChangeFn 自定义处理值改变时的回调,值true是启用并执行回调,值false是不执行回调 * @param checkEnableCallBack 自定义检测菜单的值,可自行决定是否强制启用菜单,true是启用菜单,false是不启用菜单 */ execMenuOnce(key, callback, getValueFn, handleValueChangeFn, checkEnableCallBack) { if (typeof key !== "string") { throw new TypeError("key 必须是字符串"); } if (!this.$data.data.has(key)) { log.warn(`${key} 键不存在`); return; } if (this.$data.oneSuccessExecMenu.has(key)) { return; } this.$data.oneSuccessExecMenu.set(key, 1); let __getValue = () => { let localValue = PopsPanel.getValue(key); return typeof getValueFn === "function" ? getValueFn(key, localValue) : localValue; }; let resultStyleList = []; let dynamicPushStyleNode = ($style) => { let __value = __getValue(); let dynamicResultList = []; if ($style instanceof HTMLStyleElement) { dynamicResultList = [$style]; } else if (Array.isArray($style)) { dynamicResultList = [ ...$style.filter( (item) => item != null && item instanceof HTMLStyleElement ) ]; } if (__value) { resultStyleList = resultStyleList.concat(dynamicResultList); } else { for (let index = 0; index < dynamicResultList.length; index++) { let $css = dynamicResultList[index]; $css.remove(); dynamicResultList.splice(index, 1); index--; } } }; let checkMenuEnableCallBack = (currentValue) => { return typeof checkEnableCallBack === "function" ? checkEnableCallBack(key, currentValue) : currentValue; }; let changeCallBack = (currentValue) => { let resultList = []; if (checkMenuEnableCallBack(currentValue)) { let result = callback(currentValue, dynamicPushStyleNode); if (result instanceof HTMLStyleElement) { resultList = [result]; } else if (Array.isArray(result)) { resultList = [ ...result.filter( (item) => item != null && item instanceof HTMLStyleElement ) ]; } } for (let index = 0; index < resultStyleList.length; index++) { let $css = resultStyleList[index]; $css.remove(); resultStyleList.splice(index, 1); index--; } resultStyleList = [...resultList]; }; this.addValueChangeListener( key, (__key, oldValue, newValue) => { let __newValue = newValue; if (typeof handleValueChangeFn === "function") { __newValue = handleValueChangeFn(__key, newValue, oldValue); } changeCallBack(__newValue); } ); let value = __getValue(); if (value) { changeCallBack(value); } }, /** * 父子菜单联动,自动判断菜单是否启用,然后执行回调,只会执行一次 * @param key 菜单键 * @param childKey 子菜单键 * @param callback 回调 * @param replaceValueFn 用于修改mainValue,返回undefined则不做处理 */ execInheritMenuOnce(key, childKey, callback, replaceValueFn) { let that = this; const handleInheritValue = (key2, childKey2) => { let mainValue = that.getValue(key2); let childValue = that.getValue(childKey2); if (typeof replaceValueFn === "function") { let changedMainValue = replaceValueFn(mainValue, childValue); if (changedMainValue != null) { return changedMainValue; } } return mainValue; }; this.execMenuOnce( key, callback, () => { return handleInheritValue(key, childKey); }, () => { return handleInheritValue(key, childKey); } ); this.execMenuOnce( childKey, () => { }, () => false, () => { this.triggerMenuValueChange(key); return false; } ); }, /** * 根据自定义key只执行一次 * @param key 自定义key */ onceExec(key, callback) { if (typeof key !== "string") { throw new TypeError("key 必须是字符串"); } if (this.$data.onceExec.has(key)) { return; } callback(); this.$data.onceExec.set(key, 1); }, /** * 显示设置面板 */ showPanel() { __pops.panel({ title: { text: `${SCRIPT_NAME}-设置`, position: "center", html: false, style: "" }, content: this.getPanelContentConfig(), mask: { enable: true, clickEvent: { toClose: true, toHide: false } }, width: PanelUISize.setting.width, height: PanelUISize.setting.height, drag: true, only: true }); }, /** * 获取配置内容 */ getPanelContentConfig() { let configList = [ Component_Common, Component_ForumPost, Component_Guide ]; return configList; } }; const MTIdentifyLinks = () => { var clearLink, excludedTags, filter, linkMixInit, linkPack, linkify, observePage, observer, setLink, url_regexp, xpath; url_regexp = /((https?:\/\/|www\.)[\x21-\x7e]+[\w\/]|(\w[\w._-]+\.(com|cn|org|net|info|tv|cc))(\/[\x21-\x7e]*[\w\/])?|ed2k:\/\/[\x21-\x7e]+\|\/|thunder:\/\/[\x21-\x7e]+=)/gi; clearLink = function(a) { var b; a = null != (b = a.originalTarget) ? b : a.target; if (null != a && "a" === a.localName && -1 !== a.className.indexOf("texttolink") && (b = a.getAttribute("href"), 0 !== b.indexOf("http") && 0 !== b.indexOf("ed2k://") && 0 !== b.indexOf("thunder://"))) return a.setAttribute("href", "http://" + b); }; document.addEventListener("mouseover", clearLink); setLink = function(a) { if (typeof a != "object") { return; } if (null != a && typeof a.parentNode !== "undefined" && typeof a.parentNode.className !== "undefined" && typeof a.parentNode.className.indexOf === "function" && -1 === a.parentNode.className.indexOf("texttolink") && "#cdata-section" !== a.nodeName) { var b = a.textContent.replace( url_regexp, '<a href="$1" target="_blank" class="texttolink">$1</a>' ); if (a.textContent.length !== b.length) { var c = document.createElement("span"); c.innerHTML = b; console.log(`识别: ${c.querySelector("a")}`); return a.parentNode.replaceChild(c, a); } } }; excludedTags = "a svg canvas applet input button area pre embed frame frameset head iframe img option map meta noscript object script style textarea code".split( " " ); xpath = `//text()[not(ancestor::${excludedTags.join( ") and not(ancestor::" )})]`; filter = new RegExp(`^(${excludedTags.join("|")})$`, "i"); linkPack = function(a, b) { var c, d; if (b + 1e4 < a.snapshotLength) { var e = c = b; for (d = b + 1e4; b <= d ? c <= d : c >= d; e = b <= d ? ++c : --c) setLink(a.snapshotItem(e)); setTimeout(function() { return linkPack(a, b + 1e4); }, 15); } else for (e = c = b, d = a.snapshotLength; b <= d ? c <= d : c >= d; e = b <= d ? ++c : --c) setLink(a.snapshotItem(e)); }; linkify = function(a) { a = document.evaluate( xpath, a, null, XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE, null ); return linkPack(a, 0); }; observePage = function(a) { for (a = document.createTreeWalker( a, NodeFilter.SHOW_TEXT, { acceptNode: function(a2) { if (!filter.test(a2.parentNode.localName)) return NodeFilter.FILTER_ACCEPT; } }, false ); a.nextNode(); ) setLink(a.currentNode); }; observer = new window.MutationObserver(function(a) { var b, c; var d = 0; for (b = a.length; d < b; d++) { var e = a[d]; if ("childList" === e.type) { var g = e.addedNodes; var f = 0; for (c = g.length; f < c; f++) e = g[f], observePage(e); } } }); linkMixInit = function() { return linkify(document.body), observer.observe(document.body, { childList: true, subtree: true }); }; var clearlinkF = function(a) { var url = a.getAttribute("href"); if (0 !== url.indexOf("http") && 0 !== url.indexOf("ed2k://") && 0 !== url.indexOf("thunder://")) return a.setAttribute("href", "http://" + url); }, clearlinkE = function() { for (var a = document.getElementsByClassName("texttolink"), b = 0; b < a.length; b++) clearlinkF(a[b]); }; setTimeout(clearlinkE, 1500); setTimeout(linkMixInit, 100); }; const pathname = globalThis.location.pathname; const search = globalThis.location.search; new URLSearchParams(search); const Router = { /** * 克米签到页面 */ isKMiSign() { return pathname.startsWith("/k_misign-sign.html"); }, /** * 帖子 */ isPost() { return pathname.startsWith("/thread-") || pathname.startsWith("/forum.php") && search.startsWith("?mod=viewthread"); }, /** * 首页、精华 */ isPage() { return Boolean(pathname.match(/^\/page-([0-9]+).html/g)); }, /** * 导读链接 */ isGuide() { return pathname.startsWith("/forum.php") && search.startsWith("?mod=guide"); }, /** * 板块 */ isPlate() { return Boolean(pathname.match(/\/forum-[0-9]{1,2}-[0-9]{1,2}.html/g)); }, /** * 搜索页面 */ isSearch() { return pathname.startsWith("/search.php"); }, /** * 空间 */ isSpace() { return pathname.startsWith("/home.php") && search.startsWith("?mod=space"); }, /** * 我的个人空间 */ isMySpace() { return pathname.startsWith("/home.php") && search.startsWith("?mod=space&do=profile&mycenter"); }, /** * 个人空间页的@点进去 */ isSpaceWithAt() { return pathname.startsWith("/space-uid-"); }, /** * 社区列表 */ isForumList() { return pathname.startsWith("/forum.php") && search.startsWith("?forumlist"); }, /** * 消息提醒 */ isMessage() { return pathname.startsWith("/home.php") && search.startsWith("?mod=space&do=notice"); }, /** * 消息提醒列表 */ isMessageList() { return pathname.startsWith("/home.php") && search.startsWith("?mod=space&do=pm"); }, /** * 积分商城 */ isPointsMall() { return pathname.startsWith("/keke_integralmall-keke_integralmall.html") || pathname.startsWith("/plugin.php") && search.startsWith("?id=keke_integralmal"); }, /** * 帖子发布/回复页面 */ isPostPublish() { return pathname.startsWith("/forum.php") && search.startsWith("?mod=post"); }, /** * 投票页面 */ isPostPublish_voting() { return pathname.startsWith("/forum.php") && search.includes("&special=1") || search.includes("&fid=42"); }, /** * 帖子编辑页面 */ isPostPublish_edit() { return this.isPostPublish() && search.includes("&action=edit"); }, /** * 发帖页面,该页面是尚未存入草稿 */ isPostPublish_newthread() { return this.isPostPublish() && search.includes("&action=newthread"); }, /** * 回复编辑页面 */ isPostPublish_reply() { return this.isPostPublish() && search.includes("&action=reply"); } }; const MTForumPostRightToolBar = { init() { domUtils.ready(() => { PopsPanel.execMenuOnce("mt-forum-post-quickCollentBtn", () => { this.quickCollentBtn(); }); PopsPanel.execMenuOnce("mt-forum-post-quickReplyOptimization", () => { this.quickReplyOptimization(); }); }); }, /** * 【快捷收藏】 */ quickCollentBtn() { log.info(`【快捷收藏】`); utils.waitNode("#scrolltop", 1e4).then(($scrollTop) => { if (!$scrollTop) { return; } let formhash = MTUtils.getFormHash(); let threadId = MTUtils.getThreadId(window.location.href); let collectUrl = `/home.php?${utils.toSearchParamsStr({ mod: "spacecp", ac: "favorite", type: "thread", id: threadId, formhash, infloat: "yes", handlekey: "k_favorite", inajax: 1, ajaxtarget: "fwin_content_k_favorite" })}`; let $collect = document.createElement("span"); $collect.innerHTML = /*html*/ ` <a href="${collectUrl}" id="k_favorite" onclick="showWindow(this.id, this.href, 'get', 0);" onmouseover="this.title = $('favoritenumber').innerHTML + ' 人收藏'"> <img src="https://s1.ax1x.com/2020/04/29/JTk3lD.gif" height="26" width="26" style="position:absolute;top:10px;left:11px"> </a> `; domUtils.prepend($scrollTop, $collect); }); }, /** * 快捷回复优化 */ quickReplyOptimization() { utils.waitNode('#scrolltop a[title="快速回复"]', 1e4).then(($ele) => { if (!$ele) { return; } log.info(`快捷回复优化`); domUtils.on($ele, "click", function() { _unsafeWindow.showWindow("reply", $ele.href); log.info(`等待弹窗出现`); utils.waitNode("#moreconf", 1e4).then(($moreconf) => { if (!$moreconf) { return; } log.success(`弹出出现,添加按钮`); let $oneKeySpace = domUtils.createElement( "button", { innerText: "一键空格", type: "button", id: "insertspace2" }, { style: "float: left;" } ); domUtils.on($oneKeySpace, "click", (event) => { utils.preventEvent(event); domUtils.val( $("#postmessage"), domUtils.val($("#postmessage")) + " " ); }); domUtils.append($moreconf, $oneKeySpace); }); }); }); } }; const MTForumPost = { $flag: { isSetHljsCSS: false }, init() { MTForumPostRightToolBar.init(); PopsPanel.execMenuOnce("mt-forum-post-autoExpandContent", () => { return this.autoExpandContent(); }); PopsPanel.execMenuOnce("mt-forum-post-repairImageWidth", () => { return this.repairImageWidth(); }); PopsPanel.execMenuOnce("mt-forum-post-hideBottomInfoBlock", () => { return this.hideBottomInfoBlock(); }); domUtils.ready(() => { PopsPanel.execMenu("mt-forum-post-removeFontStyle", () => { this.removeFontStyle(); }); PopsPanel.execMenu("mt-forum-post-removeCommentFontStyle", () => { this.removeCommentFontStyle(); }); PopsPanel.execMenuOnce("mt-forum-post-loadNextPageComment", () => { this.loadNextPageComment(); }); PopsPanel.execMenuOnce("mt-forum-post-codeQuoteOptimization", () => { this.codeQuoteOptimization(); }); PopsPanel.execMenuOnce("mt-forum-post-optimizationImagePreview", () => { this.optimizationImagePreview(); }); PopsPanel.execMenuOnce("mt-forum-post-interceptionAttachment", () => { this.setAttachmentsClickTip(); }); PopsPanel.execMenu("mt-forum-post-detectingUserOnlineStatus", () => { this.detectingUserOnlineStatus(); }); PopsPanel.execMenu("mt-forum-post-showUserLevel", () => { this.showUserLevel(); }); }); }, /** * 自动展开帖子内容 */ autoExpandContent() { log.info(`自动展开帖子内容`); return addStyle( /*css*/ ` div.comiis_message.bg_f.view_one.b_b.cl.message>div.comiis_messages.comiis_aimg_show.cl{max-height:inherit!important;overflow-y:inherit!important;position:inherit!important} .comiis_lookfulltext_bg,.comiis_lookfulltext_key{display:none!important} ` ); }, /** * 修复图片宽度 */ repairImageWidth() { log.info(`修复图片宽度`); return addStyle( /*css*/ ` .comiis_messages img{ max-width: 100% !important; }` ); }, /** * 移除帖子字体效果 */ removeFontStyle() { let $messageTable = document.querySelector( ".comiis_a.comiis_message_table" ); if (!$messageTable) { return; } log.info(`移除帖子字体效果`); domUtils.html( $messageTable, domUtils.html($messageTable).replace(MTRegExp.fontSpecial, "") ); }, /** * 移除评论区的字体效果 */ removeCommentFontStyle() { var _a2; log.info(`移除评论区的字体效果`); let $fontList = $$("font"); let $postForumMainContent = ((_a2 = $(".comiis_postlist .comiis_postli")) == null ? void 0 : _a2.innerHTML) || ""; if ($postForumMainContent !== "") { $fontList.forEach(($font) => { if (!$postForumMainContent.includes($font.innerHTML)) { $font.removeAttribute("color"); $font.removeAttribute("style"); $font.removeAttribute("size"); } }); $$(".comiis_message.message").forEach(($message) => { if ($postForumMainContent.includes($message.innerHTML)) { $message.innerHTML = $message.innerHTML.replace( MTRegExp.fontSpecial, "" ); let $next = $message.nextElementSibling; if ($next && $next.localName === "strike") { $next.outerHTML = $next.outerHTML.replace(/^<strike>(\n|)/g, "").replace(/<\/strike>$/g, ""); } } }); } $$(".comiis_postli.comiis_list_readimgs.nfqsqi").forEach((item) => { let $parent = item.parentElement; if ($parent && $parent.localName === "strike") { $parent.outerHTML = $parent.outerHTML.replace(/^<strike>(\n|)/g, "").replace(/<\/strike>$/g, ""); } }); }, /** * 自动加载下一页评论 */ loadNextPageComment() { log.info(`自动加载下一页评论`); if (document.title.includes("提示信息 - MT论坛")) { return; } if ($$(".pgbtn").length == 0) { log.warn("没有找到下一页按钮"); return; } var getPageInfo = async function(url) { var _a2, _b; let response = await httpx.get(url, { fetch: true, allowInterceptConfig: false }); if (!response.status) { Qmsg.error("网络异常,请求下一页失败"); return; } var pageHTML = utils.parseFromString(response.data.responseText); var nextPageBtn = pageHTML.querySelector(".pgbtn a"); (_a2 = pageHTML.querySelector("#postlistreply")) == null ? void 0 : _a2.remove(); (_b = pageHTML.querySelector(".bm_h.comiis_snvbt")) == null ? void 0 : _b.remove(); return { url: nextPageBtn ? nextPageBtn.getAttribute("href") : null, postlist: pageHTML.querySelector("#postlist"), pgbtn: pageHTML.querySelector(".pgbtn"), pgs: pageHTML.querySelector(".pgs.mtm") }; }; var scrollEvent = async function() { var _a2, _b; var nextURL = $(".pgbtn a").getAttribute("href"); if (nextURL) { let pageInfo = await getPageInfo(nextURL); if (pageInfo) { if ((_b = (_a2 = pageInfo["postlist"]) == null ? void 0 : _a2.querySelector(".comiis_vrx")) == null ? void 0 : _b.querySelector(".km1")) { Object.keys(pageInfo).forEach((it) => { pageInfo[it] = null; }); log.warn( `检测到请求的本页内容中存在【楼主】标识,判断为重复页请求` ); } if (!pageInfo["url"] || pageInfo["url"] == nextURL) { log.error("最后一页,取消监听"); domUtils.off(document, ["scroll", "wheel"], lockFn.run); domUtils.remove(".pgbtn"); } if (pageInfo["postlist"]) { domUtils.append("#postlist", domUtils.html(pageInfo["postlist"])); } if (pageInfo["pgbtn"]) { domUtils.html(".pgbtn", domUtils.html(pageInfo["pgbtn"])); } if (pageInfo["pgs"]) { domUtils.html(".pgs.mtm", domUtils.html(pageInfo["pgs"])); } MTForumPost.init(); } } else { log.error("获取下一页元素失败"); } }; let lockFn = new utils.LockFunction(async () => { if (utils.isNearBottom()) { await scrollEvent(); } }); domUtils.on(document, ["scroll", "wheel"], lockFn.run); }, /** * 代码块优化 **/ codeQuoteOptimization() { log.info(`代码块优化`); function hljs_smali(hljs2) { var smali_instr_low_prio = [ "add", "and", "cmp", "cmpg", "cmpl", "const", "div", "double", "float", "goto", "if", "int", "long", "move", "mul", "neg", "new", "nop", "not", "or", "rem", "return", "shl", "shr", "sput", "sub", "throw", "ushr", "xor" ]; var smali_instr_high_prio = [ "aget", "aput", "array", "check", "execute", "fill", "filled", "goto/16", "goto/32", "iget", "instance", "invoke", "iput", "monitor", "packed", "sget", "sparse" ]; var smali_keywords = [ "transient", "constructor", "abstract", "final", "synthetic", "public", "private", "protected", "static", "bridge", "system" ]; return { aliases: ["smali"], contains: [ { className: "string", begin: '"', end: '"', relevance: 0 }, hljs2.COMMENT("#", "$", { relevance: 0 }), { className: "keyword", variants: [ { begin: "\\s*\\.end\\s[a-zA-Z0-9]*" }, { begin: "^[ ]*\\.[a-zA-Z]*", relevance: 0 }, { begin: "\\s:[a-zA-Z_0-9]*", relevance: 0 }, { begin: "\\s(" + smali_keywords.join("|") + ")" } ] }, { className: "built_in", variants: [ { begin: "\\s(" + smali_instr_low_prio.join("|") + ")\\s" }, { begin: "\\s(" + smali_instr_low_prio.join("|") + ")((\\-|/)[a-zA-Z0-9]+)+\\s", relevance: 10 }, { begin: "\\s(" + smali_instr_high_prio.join("|") + ")((\\-|/)[a-zA-Z0-9]+)*\\s", relevance: 10 } ] }, { className: "class", begin: "L[^(;:\n]*;", relevance: 0 }, { begin: "[vp][0-9]+" } ] }; } addStyle( /*css*/ ` .hljs{text-align:left} .hljs ol{margin:0 0 0 10px;padding:10px 10px} .hljs li{padding-left:10px;list-style-type:decimal-leading-zero;font-family:Monaco,Consolas,'Lucida Console','Courier New',serif;font-size:12px;line-height:1.8em} .hljs li:hover{background:#2c313c} .hljs li::marker{unicode-bidi:isolate;font-variant-numeric:tabular-nums;text-transform:none;text-indent:0!important;text-align:start!important;text-align-last:start!important} .hljs em[onclick^=copycode]{color:#fff;background:#246fff;margin:5px 10px;border-radius:3px;padding:0 5px;cursor:pointer;height:32px;line-height:32px;display:inline-flex} .hljs .code-select-language{height:32px;line-height:32px;font-size:14px;border:1px solid #5c5c5c;border-radius:5px;text-align:center;outline:0} ` ); hljs.registerLanguage("smali", hljs_smali); let lockFn = new utils.LockFunction( () => { function setElementHighlight(ele, language = "java") { if (!ele.oldValue) { ele.oldValue = ele.textContent; } ele.innerHTML = hljs.highlight(ele.oldValue, { language }).value.replace(/\\n$/gi, ""); } document.querySelectorAll("em[onclick^=copycode]").forEach((coypCodeElement) => { if (coypCodeElement.nextElementSibling && typeof coypCodeElement.nextElementSibling.className === "string" && coypCodeElement.nextElementSibling.className == "code-select-language") { return; } let codeLanguage = hljs.highlightAuto( domUtils.text( coypCodeElement.parentElement.querySelector( "div[id^=code]" ) ) ).language; let selectElement = document.createElement("select"); let selectLanguageList = hljs.listLanguages().sort(); selectLanguageList = selectLanguageList.concat("自动检测"); let selectInnerHTML = ""; selectLanguageList.forEach((languageName) => { if (languageName.startsWith("自动检测")) { selectInnerHTML += `<option data-value="${codeLanguage}" selected="selected">${languageName}(${codeLanguage})</option>`; } else { selectInnerHTML += `<option data-value="${languageName}">${languageName}</option>`; } }); selectElement.className = "code-select-language"; selectElement.innerHTML = selectInnerHTML; domUtils.on(selectElement, "change", () => { let changeCodeLanguage = selectElement.selectedOptions[0].getAttribute("data-value"); log.info("切换代码块语言: ", changeCodeLanguage); domUtils.parent(selectElement).querySelectorAll("li").forEach((liElement) => { setElementHighlight(liElement, changeCodeLanguage); }); }); utils.preventEvent(selectElement, "click"); utils.preventEvent(coypCodeElement, "click"); coypCodeElement.insertAdjacentElement("afterend", selectElement); utils.dispatchEvent(selectElement, "change"); }); let blockcodeElementList = document.querySelectorAll(".blockcode"); blockcodeElementList.forEach((ele) => ele.className = "hljs"); }, this, 500 ); utils.mutationObserver(document, { config: { subtree: true, childList: true }, callback: () => { lockFn.run(); } }); }, /** * 图片查看优化 */ optimizationImagePreview() { log.info(`图片查看优化`); let blackListNoViewIMG = [ { hostName: "avatar-bbs.mt2.cn", pathName: "*" }, { hostName: "cdn-bbs.mt2.cn", pathName: "^(/static(/|//)image|/template)" }, { hostName: window.location.hostname, pathName: "^(/static(/|//)image|/template)" }, { hostName: window.location.hostname, pathName: "/uc_server/avatar.php" } ]; function viewIMG(imgList = [], index = 0) { let viewerULNodeHTML = ""; imgList.forEach((item) => { viewerULNodeHTML += `<li><img data-src="${item}"></li>`; }); let viewerULNode = domUtils.createElement("ul", { innerHTML: viewerULNodeHTML }); let viewer = new Viewer(viewerULNode, { inline: false, url: "data-src", zIndex: utils.getMaxZIndex() + 100, hidden: () => { viewer.destroy(); } }); viewer.view(index); viewer.zoomTo(1); viewer.show(); } function handleImage() { document.querySelectorAll( "#postlist .comiis_vrx:not([data-isHandlingViewIMG])" ).forEach((item) => { item.setAttribute("data-isHandlingViewIMG", "true"); let clickShowIMGList = []; item.querySelectorAll("img").forEach(($img) => { let IMG_URL = $img.src; let IMG_URL_HOSTNAME = new URL(IMG_URL).hostname; let IMG_URL_PATHNAME = new URL(IMG_URL).pathname; let imgParentNode = $img.parentElement; if (imgParentNode.nodeName.toLowerCase() === "a" && imgParentNode.getAttribute("href") === IMG_URL) { imgParentNode.setAttribute("href", "javascript:;"); imgParentNode.removeAttribute("target"); } let isMatching = false; for (let item2 of blackListNoViewIMG) { if (IMG_URL_HOSTNAME.indexOf(item2["hostName"]) != -1 && IMG_URL_PATHNAME.match(item2["pathName"])) { isMatching = true; break; } } if (isMatching) { return; } clickShowIMGList = [...clickShowIMGList, IMG_URL]; $img.removeAttribute("onclick"); $img.setAttribute("onclick", ""); domUtils.on($img, "click", function() { log.info("点击图片", $img); let _index_ = clickShowIMGList.findIndex((_img_) => { return _img_ == IMG_URL; }); viewIMG(clickShowIMGList, _index_); }); }); }); } let lockFn = new utils.LockFunction(() => { handleImage(); }); utils.mutationObserver(document, { config: { subtree: true, childList: true }, callback: () => { lockFn.run(); } }); }, /** * 附件点击提醒 */ setAttachmentsClickTip() { log.info(`附件点击提醒`); function handleClick(item) { if (item.hasAttribute("href")) { let attachmentId = item.hasAttribute("id") ? item.id : item.parentElement.id; let attachmentURL = item.getAttribute("href"); let attachmentName = item.innerText; let attachmentMenu = document.querySelector( `#${attachmentId}_menu` ); if (attachmentMenu.innerText.indexOf("金币") === -1) { return; } console.log("发现附件", item); console.log("该附件是金币附件,拦截!"); item.setAttribute("data-href", attachmentURL); item.style.setProperty("cursor", "pointer"); item.removeAttribute("href"); item.innerText = "【已拦截】" + attachmentName; item.onclick = function() { __pops.confirm({ title: { text: "提示", position: "center" }, content: { text: ( /*html*/ `<br />确定花费2金币下载附件 <a style="color: #507daf !important;">${attachmentName}</a> ?<br /><br />` ), html: true }, btn: { ok: { callback: (details) => { window.open(attachmentURL, "_blank"); details.close(); } } }, mask: { enable: true }, width: "400px", height: "200px" }); }; } } utils.mutationObserver(document.documentElement, { callback: () => { document.querySelectorAll(".attnm a").forEach((item) => { handleClick(item); }); document.querySelectorAll(".comiis_attach a").forEach((item) => { handleClick(item); }); document.querySelectorAll("span[id*=attach_] a").forEach((item) => { handleClick(item); }); }, immediate: true, config: { childList: true, subtree: true } }); }, /** * 探测用户在线状态 */ async detectingUserOnlineStatus() { var _a2; log.info(`探测用户在线状态`); PopsPanel.onceExec("mt-forum-post-detectingUserOnlineStatus", () => { addStyle( /*css*/ ` .gm-user-status-icon{ border: 0 !important; float: right !important; display: block !important; width: 40px !important; height: 40px !important; } ` ); }); function getStatusIcon(isOffLine) { return domUtils.createElement("img", { className: "gm-user-status-icon", smilied: isOffLine ? "1353" : "1384", loading: "lazy", src: isOffLine ? "" : "" }); } function setAvatarOnlineStatus($ele, isOffLine) { let $icon = getStatusIcon(isOffLine); domUtils.prepend($ele, $icon); } let $favatarList = Array.from( $$(".pls.favatar:not([data-is-detectingUserOnlineStatus])") ); for (let index = 0; index < $favatarList.length; index++) { const $favatar = $favatarList[index]; let $kmfxx = $favatar.querySelector( ".comiis_o.cl a.kmfxx" ); if (!$kmfxx) { log.error("探测用户在线状态失败,未找到发消息按钮"); return; } $favatar.setAttribute("data-is-detectingUserOnlineStatus", "true"); let sendMessageUrl = $kmfxx.href; let response = await httpx.get(sendMessageUrl, { fetch: true, allowInterceptConfig: false }); if (!response.status) { log.error("探测用户在线状态,中止网络请求探测"); setAvatarOnlineStatus($favatar, true); return; } let doc = domUtils.parseHTML(response.data.responseText, true, true); let $flb = doc.querySelector(".flb"); if ($flb) { let statusText = (_a2 = domUtils.text($flb)) == null ? void 0 : _a2.trim(); let isOffLine = statusText.endsWith("……[离线]"); setAvatarOnlineStatus($favatar, isOffLine); } else { setAvatarOnlineStatus($favatar, true); } } }, /** * 显示用户等级 */ showUserLevel() { log.info(`显示用户等级`); $$(".pls.favatar:not([data-show-user-level])").forEach( ($userAvatar) => { $userAvatar.setAttribute("data-show-user-level", "true"); let userLevel = "0级"; let userInfo = $userAvatar.querySelector(".tns tr"); let currentLevelText = $userAvatar.querySelector("p em").innerText; let userLevelText = document.createElement("td"); userLevelText.setAttribute("style", "border-left: 1px solid #e3e3e3;"); switch (currentLevelText) { case "幼儿园": case "初级工程师": userLevel = "1级"; break; case "小学生": case "中级工程师": userLevel = "2级"; break; case "初中生": case "高级工程师": userLevel = "3级"; break; case "高中生": case "专家": userLevel = "4级"; break; case "大学生": case "高级专家": userLevel = "5级"; break; case "硕士生": case "资深专家": userLevel = "6级"; break; case "博士生": case "实习版主": case "版主": case "审核员": case "研究员": userLevel = "7级"; break; case "博士后": case "超级版主": case "网站编辑": case "高级研究员": case "荣誉开发者": userLevel = "8级"; break; case "管理员": case "信息监察员": case "资深研究员": userLevel = "9级"; break; } userLevelText.innerHTML = `<p><a class="dj">${userLevel}</a></p>Lv`; userInfo.appendChild(userLevelText); } ); }, /** * 隐藏底部信息块 */ hideBottomInfoBlock() { log.info(`隐藏底部信息块`); return addStyle( /*css*/ ` .pls .favatar>*:not([id^="userinfo"]+div){ display: none; } .pls .favatar>div[id^="userinfo"], .pls .favatar>div.tns{ display: block; } .t_f{ min-height: 100px !important; } ` ); } }; const MTGuide = { init() { domUtils.ready(() => { PopsPanel.execMenuOnce("mt-guide-beautifyPage", () => { return this.beautifyPage(); }); }); }, /** * 页面美化 */ beautifyPage() { log.info(`页面美化`); addStyle( /*css*/ ` .xst{font-size:15px} td.author_img{width:50px;padding:15px 0} td.author_img img{width:40px;height:40px;border-radius:50%} .list_author{margin-top:2px;color:#999;font-size:12px} .bankuai_tu_by a{color:#999!important} .bankuai_tu_by img{height:16px;margin:1px 1px 0 0;vertical-align:top} tbody a:hover{text-decoration:none;color:#3498db} .byg_th_align em+a{margin-right:5px} ` ); $$(".bm_c table tbody").forEach(($tbody) => { let $common = $tbody.querySelector("th.common"); let commonHTML = domUtils.html($common); let forumUrl = $common.querySelectorAll("a")[0].getAttribute("href"); let mid = null; let $uidAnchor = $tbody.querySelector("td.by>cite>a"); let uid = $uidAnchor.getAttribute("href").match(/uid-(\d+)/)[1]; let newHTML = ( /*html*/ ` <td class="author_img"> <a href="space-uid-${uid}.html" c="1" mid="${mid}"> <img src="${MTUtils.getAvatar(uid, "middle")}"> </a> </td> <th colspan="3" class="new byg_th"> <div class="byg_th_align"> <em>[${$tbody.querySelector("tr>td.by>a").outerHTML}]</em> ${commonHTML} </div> <div class="list_author cl"> <span class="z">作者: ${$tbody.querySelectorAll("td.by>cite>a")[0].innerHTML} ${$tbody.querySelectorAll("td.by>em>span")[0].innerHTML} </span> <span class="z pipe"> | </span> <span class="z">最后发表: ${$tbody.querySelectorAll("td.by>cite>a")[1].innerHTML} ${$tbody.querySelectorAll("td.by>em>a")[0].innerHTML} </span> <span class="y bankuai_tu_by"> <a href="${forumUrl}"> <img src="" />${$tbody.querySelectorAll("td.num>a")[0].innerText} </a> </span> <span class="y bankuai_tu_by" style="margin-right: 20px"> <img src="" />${$tbody.querySelectorAll("td.num>em")[0].innerText} </span> </div> </th> ` ); domUtils.html($tbody, newHTML); }); } }; const UIInput = function(text, key, defaultValue, description, changeCallBack, placeholder = "", isNumber, isPassword, afterAddToUListCallBack) { let result = { text, type: "input", isNumber: Boolean(isNumber), isPassword: Boolean(isPassword), props: {}, attributes: {}, description, afterAddToUListCallBack, getValue() { return this.props[PROPS_STORAGE_API].get(key, defaultValue); }, callback(event, value, valueAsNumber) { this.props[PROPS_STORAGE_API].set( key, isNumber ? valueAsNumber : value ); }, placeholder }; Reflect.set(result.attributes, ATTRIBUTE_KEY, key); Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue); Reflect.set(result.props, PROPS_STORAGE_API, { get(key2, defaultValue2) { return PopsPanel.getValue(key2, defaultValue2); }, set(key2, value) { PopsPanel.setValue(key2, value); } }); return result; }; const UITextArea = function(text, key, defaultValue, description, changeCallBack, placeholder = "", disabled) { let result = { text, type: "textarea", attributes: {}, props: {}, description, placeholder, disabled, getValue() { return this.props[PROPS_STORAGE_API].get(key, defaultValue); }, callback(event, value) { this.props[PROPS_STORAGE_API].set(key, value); } }; Reflect.set(result.attributes, ATTRIBUTE_KEY, key); Reflect.set(result.attributes, ATTRIBUTE_DEFAULT_VALUE, defaultValue); Reflect.set(result.props, PROPS_STORAGE_API, { get(key2, defaultValue2) { return PopsPanel.getValue(key2, defaultValue2); }, set(key2, value) { PopsPanel.setValue(key2, value); } }); return result; }; class RuleEditView { constructor(option) { __publicField(this, "option"); this.option = option; } /** * 显示视图 */ async showView() { var _a2; let $dialog = __pops.confirm({ title: { text: this.option.title, position: "center" }, content: { text: ( /*html*/ ` <form class="rule-form-container" onsubmit="return false"> <ul class="rule-form-ulist"> </ul> <input type="submit" style="display: none;" /> </form> ` ), html: true }, btn: utils.assign( { ok: { callback: async () => { await submitSaveOption(); } } }, this.option.btn || {}, true ), mask: { enable: true }, style: ( /*css*/ ` ${__pops.config.cssText.panelCSS} .rule-form-container { } .rule-form-container li{ display: flex; align-items: center; justify-content: space-between; padding: 5px 20px; gap: 10px; } .pops-panel-item-left-main-text{ max-width: 150px; } .pops-panel-item-right-text{ padding-left: 30px; } .pops-panel-item-right-text, .pops-panel-item-right-main-text{ text-overflow: ellipsis; overflow: hidden; white-space: nowrap; } ${((_a2 = this.option) == null ? void 0 : _a2.style) ?? ""} ` ), width: typeof this.option.width === "function" ? this.option.width() : window.innerWidth > 500 ? "500px" : "88vw", height: typeof this.option.height === "function" ? this.option.height() : window.innerHeight > 500 ? "500px" : "80vh" }); let $form = $dialog.$shadowRoot.querySelector( ".rule-form-container" ); $dialog.$shadowRoot.querySelector( "input[type=submit]" ); let $ulist = $dialog.$shadowRoot.querySelector(".rule-form-ulist"); let view = await this.option.getView(await this.option.data()); $ulist.appendChild(view); const submitSaveOption = async () => { let result = await this.option.onsubmit($form, await this.option.data()); if (!result.success) { return; } $dialog.close(); await this.option.dialogCloseCallBack(true); }; } } const MTCommentFilter = { $el: { isFilterElementHTML: [] }, $key: { STORAGE_KEY: "mt-post-comment-filter-rule" }, init() { this.registerMenu(); if (Router.isPost()) { let allData = this.getData(); if (!allData.enable) { return; } let lockFn = new utils.LockFunction(() => { this.runFilter(allData); }); utils.mutationObserver(document, { config: { subtree: true, childList: true }, callback: () => { lockFn.run(); } }); } }, /** * 注册(不可用)菜单 */ registerMenu() { GM_Menu.add({ key: "comment-filter", text: "⚙ 评论过滤器", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showView(); } }); }, /** * 执行过滤 */ runFilter(filterData) { let isBlackListUser = function(postForumInfo) { for (const userName of filterData["userBlackList"]) { if (userName == postForumInfo.userName || userName == postForumInfo.userUID) { log.success("评论过滤器:黑名单用户", postForumInfo); return true; } } return false; }; let isWhiteListUser = function(postForumInfo) { for (const userName of filterData["userWhiteList"]) { if (userName === postForumInfo.userName || userName === postForumInfo.userUID) { log.success("评论过滤器:白名单用户", postForumInfo); return true; } } return false; }; $$(".comiis_vrx").forEach((item) => { var _a2, _b, _c, _d, _e; if (item.querySelector(".plc .pti .authi .show")) { return; } let $name = item.querySelector(".pls .authi a"); let postForumInfo = { userName: ($name == null ? void 0 : $name.innerText) || "", userUID: ((_c = (_b = (_a2 = $name == null ? void 0 : $name.href) == null ? void 0 : _a2.match(MTRegExp.uid)) == null ? void 0 : _b[2]) == null ? void 0 : _c.trim()) || "", content: ((_e = (_d = item.querySelector(".plc td.t_f")) == null ? void 0 : _d.innerText) == null ? void 0 : _e.trim()) || "", // PC端无法实现 isAuthor: false }; if (isWhiteListUser(postForumInfo)) { return; } if (filterData["replyFlag"] && item.querySelector(".quote")) { let comiis_quote_Element = item.querySelector(".quote"); this.$el.isFilterElementHTML.push(comiis_quote_Element.outerHTML); comiis_quote_Element.remove(); } if (postForumInfo.isAuthor && !filterData["avatarFlag"]) { return; } if (isBlackListUser(postForumInfo)) { this.$el.isFilterElementHTML.push(item.outerHTML); item.remove(); return; } if (typeof filterData["minLength"] === "number" && filterData["minLength"] > postForumInfo.content.length) { return; } if (typeof filterData["keywordLength"] === "number" && filterData["keywordLength"] < postForumInfo.content.length) { return; } for (const keywordItem of filterData["keywords"]) { if (typeof keywordItem !== "string") { continue; } let keywordPattern = new RegExp(keywordItem); if (postForumInfo.content.match(keywordPattern)) { console.log("评论过滤器:", postForumInfo); this.$el.isFilterElementHTML.push(item.outerHTML); item.remove(); return; } } }); }, /** * 显示视图 */ showView() { const that = this; function generateStorageApi(data) { return { get(key, defaultValue) { let localData = that.getData(); let value = Reflect.get(localData, key, defaultValue); if (key === "keywords" || key === "userWhiteList" || key === "userBlackList") { value = value.join("\n"); } return value; }, set(key, value) { if (key === "keywords" || key === "userWhiteList" || key === "userBlackList") { value = value.split("\n").filter((item) => item.trim() != ""); } Reflect.set(data, key, value); that.setData(data); } }; } let popsPanelContentUtils = __pops.config.panelHandleContentUtils(); let view = new RuleEditView({ title: "评论过滤器", data: () => { return this.getData(); }, getView: (data) => { let $fragment = document.createDocumentFragment(); let enable_template = UISwitch("启用", "enable", true); Reflect.set( enable_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $enable = popsPanelContentUtils.createSectionContainerItem_switch( enable_template ); let replyFlag_template = UISwitch( "处理回复引用", "replyFlag", false, void 0, "移除引用" ); Reflect.set( replyFlag_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $replyFlag = popsPanelContentUtils.createSectionContainerItem_switch( replyFlag_template ); let avatarFlag_template = UISwitch("处理作者评论", "avatarFlag", false); Reflect.set( avatarFlag_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $avatarFlag = popsPanelContentUtils.createSectionContainerItem_switch( avatarFlag_template ); let viewthreadFlag_template = UISwitch( '处理从"搜索页面"或"我的帖子提醒页面"进入的网站', "viewthreadFlag", false ); Reflect.set( viewthreadFlag_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $viewthreadFlag = popsPanelContentUtils.createSectionContainerItem_switch( viewthreadFlag_template ); let minLength_template = UIInput( "匹配的评论内容长度最小值", "minLength", 5, "小于此长度的评论就算关键字匹配成功了也不会被排除", void 0, "", true ); Reflect.set( minLength_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $minLength = popsPanelContentUtils.createSectionContainerItem_input( minLength_template ); let keywordLength = UIInput( "匹配的评论内容长度最大值", "keywordLength", 8, "大于此长度的评论就算关键字匹配成功了也不会被排除", void 0, "", true ); Reflect.set( keywordLength.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $keywordLength = popsPanelContentUtils.createSectionContainerItem_input(keywordLength); let keywords_template = UITextArea( "评论关键字", "keywords", "", "多个评论关键字换行分割" ); Reflect.set( keywords_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $keywords = popsPanelContentUtils.createSectionContainerItem_textarea( keywords_template ); let userBlackList_template = UITextArea( "黑名单用户", "userBlackList", "", "多个用户换行分割" ); Reflect.set( userBlackList_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $userBlackList = popsPanelContentUtils.createSectionContainerItem_textarea( userBlackList_template ); let userWhiteList_template = UITextArea( "白名单用户", "userWhiteList", "", "多个用户换行分割" ); Reflect.set( userWhiteList_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $userWhiteList = popsPanelContentUtils.createSectionContainerItem_textarea( userWhiteList_template ); $fragment.append( $enable, $replyFlag, $avatarFlag, $viewthreadFlag, $minLength, $keywordLength, $keywords, $userBlackList, $userWhiteList ); return $fragment; }, btn: { merge: true, position: "space-between", ok: { enable: false }, cancel: { enable: true, text: "关闭" }, other: { enable: true, text: `查看已过滤(${this.$el.isFilterElementHTML.length})`, type: "primary", callback: (details, event) => { __pops.alert({ title: { text: "评论过滤器-已过滤", position: "center" }, content: { text: ( /*html*/ ` ${Array.from( document.querySelectorAll( 'link[rel="stylesheet"]' ) ).map((item) => item.outerHTML).join("\n")} ${this.$el.isFilterElementHTML.join("\n")} ` ), html: true }, mask: { enable: true }, style: ( /*css*/ ` .plhin{ width: 100%; } td.plc .pi{ height: auto; } ` ), width: "88vw", height: "80vh" }); } } }, dialogCloseCallBack(isSubmit) { }, onsubmit: ($form, data) => { return { success: true, data }; }, style: ( /*css*/ ` .pops-panel-item-left-desc-text{ line-height: normal; margin-top: 6px; font-size: 0.8em; color: rgb(108, 108, 108); } .pops-panel-item-left-main-text{ max-width: unset; } .pops-panel-textarea textarea{ height: 150px; } ` ) }); view.showView(); }, /** * 获取模板数据 */ getTemplateData() { return { enable: true, avatarFlag: false, replyFlag: false, viewthreadFlag: false, minLength: 5, keywordLength: 8, keywords: [], userBlackList: [], userWhiteList: [] }; }, /** * 获取数据 */ getData() { return _GM_getValue( this.$key.STORAGE_KEY, this.getTemplateData() ); }, /** * 设置数据 */ setData(data) { _GM_setValue(this.$key.STORAGE_KEY, data); } }; class RuleFilterView { constructor(option) { __publicField(this, "option"); this.option = option; } showView() { let $alert = __pops.alert({ title: { text: this.option.title, position: "center" }, content: { text: ( /*html*/ ` <div class="filter-container"></div> ` ) }, btn: { ok: { text: "关闭", type: "default" } }, mask: { enable: true }, width: window.innerWidth > 500 ? "350px" : "80vw", height: window.innerHeight > 500 ? "300px" : "70vh", style: ( /*css*/ ` .filter-container{ height: 100%; display: flex; flex-direction: column; gap: 20px; } .filter-container button{ text-wrap: wrap; padding: 8px; height: auto; text-align: left; } ` ) }); let $filterContainer = $alert.$shadowRoot.querySelector(".filter-container"); let $fragment = document.createDocumentFragment(); this.option.filterOption.forEach((filterOption) => { let $button = document.createElement("button"); $button.innerText = filterOption.name; let execFilterAndCloseDialog = async () => { let allRuleInfo = await this.option.getAllRuleInfo(); allRuleInfo.forEach(async (ruleInfo) => { let filterResult = await filterOption.filterCallBack(ruleInfo.data); if (!filterResult) { domUtils.hide(ruleInfo.$el, false); } else { domUtils.show(ruleInfo.$el, false); } }); if (typeof this.option.execFilterCallBack === "function") { await this.option.execFilterCallBack(); } $alert.close(); }; domUtils.on($button, "click", async (event) => { utils.preventEvent(event); if (typeof filterOption.callback === "function") { let result = await filterOption.callback( event, execFilterAndCloseDialog ); if (!result) { return; } } await execFilterAndCloseDialog(); }); $fragment.appendChild($button); }); $filterContainer.appendChild($fragment); } } class RuleView { constructor(option) { __publicField(this, "option"); this.option = option; } /** * 显示视图 * @param filterCallBack 返回值为false隐藏,true则不隐藏(不处理) */ async showView(filterCallBack) { var _a2, _b, _c, _d, _e, _f, _g, _h, _i; let $popsConfirm = __pops.confirm({ title: { text: this.option.title, position: "center" }, content: { text: ( /*html*/ ` <div class="rule-view-container"> </div> ` ), html: true }, btn: { merge: true, reverse: false, position: "space-between", ok: { enable: ((_c = (_b = (_a2 = this.option) == null ? void 0 : _a2.bottomControls) == null ? void 0 : _b.add) == null ? void 0 : _c.enable) || true, type: "primary", text: "添加", callback: async (event) => { this.showEditView( false, await this.option.getAddData(), $popsConfirm.$shadowRoot ); } }, close: { enable: true, callback(event) { $popsConfirm.close(); } }, cancel: { enable: ((_f = (_e = (_d = this.option) == null ? void 0 : _d.bottomControls) == null ? void 0 : _e.filter) == null ? void 0 : _f.enable) || false, type: "default", text: "过滤", callback: (details, event) => { var _a3, _b2, _c2, _d2, _e2, _f2, _g2; if (typeof ((_c2 = (_b2 = (_a3 = this.option) == null ? void 0 : _a3.bottomControls) == null ? void 0 : _b2.filter) == null ? void 0 : _c2.callback) === "function") { this.option.bottomControls.filter.callback(); } let getAllRuleElement = () => { return Array.from( $popsConfirm.$shadowRoot.querySelectorAll( ".rule-view-container .rule-item" ) ); }; let $button = event.target.closest(".pops-confirm-btn").querySelector(".pops-confirm-btn-cancel span"); if (domUtils.text($button).includes("取消")) { getAllRuleElement().forEach(($el) => { domUtils.show($el, false); }); domUtils.text($button, "过滤"); } else { let ruleFilterView = new RuleFilterView({ title: ((_e2 = (_d2 = this.option.bottomControls) == null ? void 0 : _d2.filter) == null ? void 0 : _e2.title) ?? "过滤规则", filterOption: ((_g2 = (_f2 = this.option.bottomControls) == null ? void 0 : _f2.filter) == null ? void 0 : _g2.option) || [], execFilterCallBack() { domUtils.text($button, "取消过滤"); }, getAllRuleInfo: () => { return getAllRuleElement().map(($el) => { return { data: this.parseRuleItemElement($el).data, $el }; }); } }); ruleFilterView.showView(); } } }, other: { enable: ((_i = (_h = (_g = this.option) == null ? void 0 : _g.bottomControls) == null ? void 0 : _h.clear) == null ? void 0 : _i.enable) || true, type: "xiaomi-primary", text: `清空所有(${(await this.option.data()).length})`, callback: (event) => { let $askDialog = __pops.confirm({ title: { text: "提示", position: "center" }, content: { text: "确定清空所有的数据?", html: false }, btn: { ok: { enable: true, callback: async (popsEvent) => { var _a3, _b2, _c2; log.success("清空所有"); if (typeof ((_c2 = (_b2 = (_a3 = this.option) == null ? void 0 : _a3.bottomControls) == null ? void 0 : _b2.clear) == null ? void 0 : _c2.callback) === "function") { this.option.bottomControls.clear.callback(); } let data = await this.option.data(); if (data.length) { Qmsg.error("清理失败"); return; } else { Qmsg.success("清理成功"); } await this.updateDeleteAllBtnText($popsConfirm.$shadowRoot); this.clearContent($popsConfirm.$shadowRoot); $askDialog.close(); } }, cancel: { text: "取消", enable: true } }, mask: { enable: true }, width: "300px", height: "200px" }); } } }, mask: { enable: true }, width: window.innerWidth > 500 ? "500px" : "88vw", height: window.innerHeight > 500 ? "500px" : "80vh", style: ( /*css*/ ` ${__pops.config.cssText.panelCSS} .rule-item{ display: flex; align-items: center; line-height: normal; font-size: 16px; padding: 4px 8px; gap: 8px; } .rule-name{ flex: 1; white-space: nowrap; text-overflow: ellipsis; overflow: hidden; } .rule-controls{ display: flex; align-items: center; text-overflow: ellipsis; overflow: hidden; white-space: nowrap; gap: 8px; padding: 0px; } .rule-controls-enable{ } .rule-controls-edit{ } .rule-controls-delete{ } .rule-controls-edit, .rule-controls-delete{ width: 16px; height: 16px; cursor: pointer; } ` ) }); let allData = await this.option.data(); let changeButtonText = false; for (let index = 0; index < allData.length; index++) { let item = allData[index]; let $ruleItemList = await this.appendRuleItemElement( $popsConfirm.$shadowRoot, item ); let flag = typeof filterCallBack === "function" ? filterCallBack(item) : true; if (!flag) { changeButtonText = true; $ruleItemList.forEach(($el) => { domUtils.hide($el, false); }); } } if (changeButtonText) { let $button = $popsConfirm.$shadowRoot.querySelector( ".pops-confirm-btn-cancel span" ); domUtils.text($button, "取消过滤"); } } /** * 解析弹窗内的各个元素 */ parseViewElement($shadowRoot) { let $container = $shadowRoot.querySelector( ".rule-view-container" ); let $deleteBtn = $shadowRoot.querySelector( ".pops-confirm-btn button.pops-confirm-btn-other" ); return { /** 容器 */ $container, /** 左下角的清空按钮 */ $deleteBtn }; } /** * 解析每一项的元素 */ parseRuleItemElement($ruleElement) { let $enable = $ruleElement.querySelector( ".rule-controls-enable" ); let $enableSwitch = $enable.querySelector(".pops-panel-switch"); let $enableSwitchInput = $enable.querySelector( ".pops-panel-switch__input" ); let $enableSwitchCore = $enable.querySelector( ".pops-panel-switch__core" ); let $edit = $ruleElement.querySelector(".rule-controls-edit"); let $delete = $ruleElement.querySelector( ".rule-controls-delete" ); return { /** 启用开关 */ $enable, /** 启用开关的container */ $enableSwitch, /** 启用开关的input */ $enableSwitchInput, /** 启用开关的core */ $enableSwitchCore, /** 编辑按钮 */ $edit, /** 删除按钮 */ $delete, /** 存储在元素上的数据 */ data: Reflect.get($ruleElement, "data-rule") }; } /** * 创建一条规则元素 */ async createRuleItemElement(data, $shadowRoot) { let name = await this.option.getDataItemName(data); let $ruleItem = domUtils.createElement("div", { className: "rule-item", innerHTML: ( /*html*/ ` <div class="rule-name">${name}</div> <div class="rule-controls"> <div class="rule-controls-enable"> <div class="pops-panel-switch"> <input class="pops-panel-switch__input" type="checkbox"> <span class="pops-panel-switch__core"> <div class="pops-panel-switch__action"> </div> </span> </div> </div> <div class="rule-controls-edit"> ${__pops.config.iconSVG.edit} </div> <div class="rule-controls-delete"> ${__pops.config.iconSVG.delete} </div> </div> ` ) }); Reflect.set($ruleItem, "data-rule", data); let switchCheckedClassName = "pops-panel-switch-is-checked"; const { $enable, $enableSwitch, $enableSwitchCore, $enableSwitchInput, $delete, $edit } = this.parseRuleItemElement($ruleItem); if (this.option.itemControls.enable.enable) { domUtils.on($enableSwitchCore, "click", async (event) => { let isChecked = false; if ($enableSwitch.classList.contains(switchCheckedClassName)) { $enableSwitch.classList.remove(switchCheckedClassName); isChecked = false; } else { $enableSwitch.classList.add(switchCheckedClassName); isChecked = true; } $enableSwitchInput.checked = isChecked; await this.option.itemControls.enable.callback(data, isChecked); }); if (await this.option.itemControls.enable.getEnable(data)) { $enableSwitch.classList.add(switchCheckedClassName); } } else { $enable.remove(); } if (this.option.itemControls.edit.enable) { domUtils.on($edit, "click", (event) => { utils.preventEvent(event); this.showEditView(true, data, $shadowRoot, $ruleItem, (newData) => { data = null; data = newData; }); }); } else { $edit.remove(); } if (this.option.itemControls.delete.enable) { domUtils.on($delete, "click", (event) => { utils.preventEvent(event); let $askDialog = __pops.confirm({ title: { text: "提示", position: "center" }, content: { text: "确定删除该条数据?", html: false }, btn: { ok: { enable: true, callback: async (popsEvent) => { log.success("删除数据"); let flag = await this.option.itemControls.delete.deleteCallBack( data ); if (flag) { Qmsg.success("成功删除该数据"); $ruleItem.remove(); await this.updateDeleteAllBtnText($shadowRoot); $askDialog.close(); } else { Qmsg.error("删除该数据失败"); } } }, cancel: { text: "取消", enable: true } }, mask: { enable: true }, width: "300px", height: "200px" }); }); } else { $delete.remove(); } return $ruleItem; } /** * 添加一个规则元素 */ async appendRuleItemElement($shadowRoot, data) { let { $container } = this.parseViewElement($shadowRoot); let $ruleItem = []; let iteratorData = Array.isArray(data) ? data : [data]; for (let index = 0; index < iteratorData.length; index++) { let item = iteratorData[index]; let $item = await this.createRuleItemElement(item, $shadowRoot); $container.appendChild($item); $ruleItem.push($item); } await this.updateDeleteAllBtnText($shadowRoot); return $ruleItem; } /** * 更新弹窗内容的元素 */ async updateRuleContaienrElement($shadowRoot) { this.clearContent($shadowRoot); const { $container } = this.parseViewElement($shadowRoot); let data = await this.option.data(); await this.appendRuleItemElement($shadowRoot, data); await this.updateDeleteAllBtnText($shadowRoot); } /** * 更新规则元素 */ async updateRuleItemElement(data, $oldRuleItem, $shadowRoot) { let $newRuleItem = await this.createRuleItemElement(data, $shadowRoot); $oldRuleItem.after($newRuleItem); $oldRuleItem.remove(); } /** * 清空内容 */ clearContent($shadowRoot) { const { $container } = this.parseViewElement($shadowRoot); domUtils.html($container, ""); } /** * 设置删除按钮的文字 */ setDeleteBtnText($shadowRoot, text, isHTML = false) { const { $deleteBtn } = this.parseViewElement($shadowRoot); if (isHTML) { domUtils.html($deleteBtn, text); } else { domUtils.text($deleteBtn, text); } } /** * 更新【清空所有】的按钮的文字 * @param $shadowRoot */ async updateDeleteAllBtnText($shadowRoot) { let data = await this.option.data(); this.setDeleteBtnText($shadowRoot, `清空所有(${data.length})`); } /** * 显示编辑视图 * @param isEdit 是否是编辑状态 * @param editData 编辑的数据 */ showEditView(isEdit, editData, $parentShadowRoot, $editRuleItemElement, updateDataCallBack) { let dialogCloseCallBack = async (isSubmit) => { if (isSubmit) ; else { if (!isEdit) { await this.option.deleteData(editData); } if (typeof updateDataCallBack === "function") { let newData = await this.option.getData(editData); updateDataCallBack(newData); } } }; let editView = new RuleEditView({ title: isEdit ? "编辑" : "添加", data: () => { return editData; }, dialogCloseCallBack, getView: async (data) => { return await this.option.itemControls.edit.getView(data, isEdit); }, btn: { ok: { enable: true, text: isEdit ? "修改" : "添加" }, cancel: { callback: async (detail, event) => { detail.close(); await dialogCloseCallBack(false); } }, close: { callback: async (detail, event) => { detail.close(); await dialogCloseCallBack(false); } } }, onsubmit: async ($form, data) => { let result = await this.option.itemControls.edit.onsubmit( $form, isEdit, data ); if (result.success) { if (isEdit) { Qmsg.success("修改成功"); $parentShadowRoot && await this.updateRuleItemElement( result.data, $editRuleItemElement, $parentShadowRoot ); } else { $parentShadowRoot && await this.appendRuleItemElement( $parentShadowRoot, result.data ); } } else { if (isEdit) { Qmsg.error("修改失败"); } } return result; }, style: this.option.itemControls.edit.style, width: this.option.itemControls.edit.width, height: this.option.itemControls.edit.height }); editView.showView(); } } const MTProductListingReminder = { $key: { STORAGE_KEY: "mt-productListingReminder-rule" }, init() { this.registerMenu(); this.runRule(); }, /** * 注册(不可用)菜单 */ registerMenu() { GM_Menu.add({ key: "product-reminder", text: "⚙ 商品上架提醒", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showView(); } }); }, /** * 执行规则 */ async runRule() { async function getCurrentProduct() { let response = await httpx.get( "/keke_integralmall-keke_integralmall.html", { allowInterceptConfig: false, headers: { "User-Agent": utils.getRandomAndroidUA() } } ); if (!response.status) { Qmsg.error("【积分商城】获取数据失败"); return; } let productInfoList = []; let doc = domUtils.parseHTML(response.data.responseText, true, true); doc.querySelectorAll(".task-list-wrapper li.col-xs-12").forEach(($taskList) => { var _a2, _b; productInfoList.push({ name: domUtils.text( $taskList.querySelector( ".mall-info a > *:first-child" ) ) || domUtils.text( $taskList.querySelector(".mall-info a") ), price: domUtils.text( $taskList.querySelector( ".mall-info span.discount-price i" ) ), endTime: domUtils.text( $taskList.querySelector( ".mall-info #time_hz span.time" ) ), remainingQuantity: parseInt( ((_b = (_a2 = $taskList.querySelector(".mall-info .mall-count .count-r")) == null ? void 0 : _a2.innerText) == null ? void 0 : _b.replace(/仅剩|件/gi, "")) || "0" ) }); }); return productInfoList; } if (Router.isPointsMall()) { return; } let allData = this.getData(); if (!allData.length) { return; } let productList = await getCurrentProduct(); if (!productList) { return; } for (const productItem of productList) { for (const reminderOption of allData) { if (reminderOption.enable && productItem["name"].match( new RegExp(reminderOption["productName"], "i") ) && !isNaN(productItem["remainingQuantity"]) && productItem["remainingQuantity"] > 0) { log.success(`成功匹配对应商品`, reminderOption, productItem); __pops.confirm({ title: { text: "积分商城提醒", position: "center" }, content: { text: ( /*html*/ `<br /> 您设置的商品已上架在积分商城中,当前售价 ${productItem["price"]}金币,仅剩${productItem["remainingQuantity"]}件,是否前往购买? <a style="color: red !important;">(如需关闭提醒,请删除该关键字)</a> <br />` ), html: true }, btn: { merge: true, position: "space-between", other: { enable: true, type: "danger", text: "删除提醒", callback: () => { let status = this.deleteData(reminderOption); if (status) { Qmsg.success("删除成功"); } else { Qmsg.error("删除失败"); } } }, ok: { text: "前往购买", callback: () => { window.location.href = `${window.location.origin}/keke_integralmall-keke_integralmall.html`; } } }, width: "300px", height: "300px" }); return; } } } }, /** * 获取模板数据 */ getTemplateData() { return { uuid: utils.generateUUID(), enable: true, name: "", productName: "" }; }, /** * 显示视图 */ showView() { let popsPanelContentUtils = __pops.config.panelHandleContentUtils(); function generateStorageApi(data) { return { get(key, defaultValue) { return data[key] ?? defaultValue; }, set(key, value) { data[key] = value; } }; } let ruleView = new RuleView({ title: "商品上架提醒", data: () => { return this.getData(); }, getAddData: () => { return this.getTemplateData(); }, getDataItemName: (data) => { return data["name"]; }, updateData: (data) => { return this.updateData(data); }, deleteData: (data) => { return this.deleteData(data); }, getData: (data) => { let allData = this.getData(); let findValue = allData.find((item) => item.uuid === data.uuid); return findValue ?? data; }, itemControls: { enable: { enable: true, getEnable(data) { return data.enable; }, callback: (data, enable) => { data.enable = enable; this.updateData(data); } }, edit: { enable: true, getView: (data, isEdit) => { let $fragment = document.createDocumentFragment(); if (!isEdit) { data = this.getTemplateData(); } let enable_template = UISwitch("启用", "enable", true); Reflect.set( enable_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $enable = popsPanelContentUtils.createSectionContainerItem_switch( enable_template ); let name_template = UIInput( "规则名称", "name", "", "", void 0, "必填" ); Reflect.set( name_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $name = popsPanelContentUtils.createSectionContainerItem_input( name_template ); let productName_template = UIInput( "商品名", "productName", "", "", void 0, "可正则,需手动转义" ); Reflect.set( productName_template.props, PROPS_STORAGE_API, generateStorageApi(data) ); let $productName = popsPanelContentUtils.createSectionContainerItem_input( productName_template ); $fragment.append($enable, $name, $productName); return $fragment; }, onsubmit: ($form, isEdit, editData) => { let $ulist_li = $form.querySelectorAll( ".rule-form-ulist > li" ); let data = this.getTemplateData(); if (isEdit) { data.uuid = editData.uuid; } $ulist_li.forEach(($li) => { let formConfig = Reflect.get($li, "__formConfig__"); let attrs = Reflect.get(formConfig, "attributes"); let storageApi = Reflect.get($li, PROPS_STORAGE_API); let key = Reflect.get(attrs, ATTRIBUTE_KEY); let defaultValue = Reflect.get(attrs, ATTRIBUTE_DEFAULT_VALUE); let value = storageApi.get(key, defaultValue); if (Reflect.has(data, key)) { Reflect.set(data, key, value); } else { log.error(`${key}不在数据中`); } }); if (data.name.trim() === "") { Qmsg.error("规则名称不能为空"); return { success: false, data }; } if (isEdit) { return { success: this.updateData(data), data }; } else { return { success: this.addData(data), data }; } } }, delete: { enable: true, deleteCallBack: (data) => { return this.deleteData(data); } } } }); ruleView.showView(); }, /** * 获取数据 */ getData() { return _GM_getValue(this.$key.STORAGE_KEY, []); }, /** * 设置数据 * @param data */ setData(data) { _GM_setValue(this.$key.STORAGE_KEY, data); }, /** * 添加数据 * @param data */ addData(data) { let localData = this.getData(); let findIndex = localData.findIndex((item) => item.uuid == data.uuid); if (findIndex === -1) { localData.push(data); _GM_setValue(this.$key.STORAGE_KEY, localData); return true; } else { return false; } }, /** * 更新数据 * @param data */ updateData(data) { let localData = this.getData(); let index = localData.findIndex((item) => item.uuid == data.uuid); let updateFlag = false; if (index !== -1) { updateFlag = true; localData[index] = data; } this.setData(localData); return updateFlag; }, /** * 删除数据 * @param data */ deleteData(data) { let localData = this.getData(); let index = localData.findIndex((item) => item.uuid == data.uuid); let deleteFlag = false; if (index !== -1) { deleteFlag = true; localData.splice(index, 1); } this.setData(localData); return deleteFlag; }, /** * 清空数据 */ clearData() { _GM_deleteValue(this.$key.STORAGE_KEY); } }; const blackHomeCSS = ".pops-confirm-content {\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n.blackhome-user-filter input {\r\n width: -moz-available;\r\n width: -webkit-fill-available;\r\n height: 30px;\r\n margin: 8px 20px;\r\n border: 0;\r\n border-bottom: 1px solid;\r\n text-overflow: ellipsis;\r\n overflow: hidden;\r\n white-space: nowrap;\r\n}\r\n.blackhome-user-filter input:focus-within {\r\n outline: none;\r\n}\r\n.blackhome-user-list {\r\n flex: 1;\r\n overflow-y: auto;\r\n}\r\n.blackhome-user-list .blackhome-user-item {\r\n margin: 15px 10px;\r\n padding: 10px;\r\n border-radius: 8px;\r\n box-shadow: 0 0 0.6rem #c8d0e7, -0.2rem -0.2rem 0.5rem #fff;\r\n}\r\n.blackhome-user {\r\n display: flex;\r\n}\r\n.blackhome-user img {\r\n width: 45px;\r\n height: 45px;\r\n border-radius: 45px;\r\n}\r\n.blackhome-user-info {\r\n margin-left: 10px;\r\n}\r\n.blackhome-user-info p:nth-child(1) {\r\n margin-bottom: 5px;\r\n}\r\n.blackhome-user-info p:nth-child(2) {\r\n font-size: 14px;\r\n}\r\n.blackhome-user-action {\r\n display: flex;\r\n margin: 10px 0;\r\n}\r\n.blackhome-user-action p:nth-child(1),\r\n.blackhome-user-action p:nth-child(2) {\r\n border: 1px solid red;\r\n color: red;\r\n border-radius: 4px;\r\n padding: 2px 4px;\r\n font-weight: 500;\r\n font-size: 14px;\r\n place-self: center;\r\n}\r\n.blackhome-user-action p:nth-child(2) {\r\n border: 1px solid #ff4b4b;\r\n color: #ff4b4b;\r\n margin-left: 8px;\r\n}\r\n.blackhome-user-uuid {\r\n border: 1px solid #ff7600;\r\n color: #ff7600;\r\n border-radius: 4px;\r\n padding: 2px 4px;\r\n font-weight: 500;\r\n font-size: 14px;\r\n width: fit-content;\r\n width: -moz-fit-content;\r\n margin: 10px 0;\r\n}\r\n.blackhome-operator {\r\n padding: 10px;\r\n background-color: #efefef;\r\n border-radius: 6px;\r\n}\r\n.blackhome-operator-user {\r\n display: flex;\r\n}\r\n.blackhome-operator-user img {\r\n width: 35px;\r\n height: 35px;\r\n border-radius: 35px;\r\n}\r\n.blackhome-operator-user p {\r\n align-self: center;\r\n margin-left: 10px;\r\n}\r\n.blackhome-operator-user-info {\r\n margin: 10px 0;\r\n font-weight: 500;\r\n}\r\n\r\n@media screen and (min-width: 800px) {\r\n .blackhome-user-list {\r\n display: flex;\r\n flex-wrap: wrap;\r\n }\r\n .blackhome-user-list .blackhome-user-item {\r\n flex: 1 1 250px;\r\n max-width: calc(50% - 10px - 10px);\r\n }\r\n}\r\n"; const MTBlackHome = { $data: { cid: "" }, init() { this.registerMenu(); }, /** * 注册(不可用)菜单 */ registerMenu() { GM_Menu.add({ key: "black-home", text: "⚙ 小黑屋", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showBlackHome(); } }); }, /** * 显示小黑屋dialog */ async showBlackHome() { let $loading = Qmsg.loading("正在获取小黑屋名单中..."); let blackListInfo = await this.getBlackListInfo(""); $loading.close(); if (!blackListInfo) { return; } if (blackListInfo.data.length === 0) { Qmsg.error("获取小黑屋名单为空"); return; } this.$data.cid = blackListInfo.next_cid; let $confirm = __pops.confirm({ title: { text: "小黑屋名单", position: "center" }, content: { text: ( /*html*/ ` <div class="blackhome-user-filter"> <input placeholder="过滤用户名/操作人员/UID(可正则)"> </div> <div class="blackhome-user-list"></div> ` ), html: true }, btn: { ok: { text: "下一页", callback: async () => { let $loading2 = Qmsg.loading("正在获取小黑屋名单中..."); log.info("下一页的cid: ", this.$data.cid); let nextBlackListInfo2 = await this.getBlackListInfo(this.$data.cid); $loading2.close(); if (!nextBlackListInfo2) { return; } this.$data.cid = nextBlackListInfo2.next_cid; nextBlackListInfo2.data.forEach((item) => { let $item = this.createListViewItem(item); $list.appendChild($item); }); if (nextBlackListInfo2.data.length === 0) { Qmsg.error("获取小黑屋名单为空"); } else { Qmsg.success(`成功获取 ${nextBlackListInfo2.data.length}条数据`); } } }, cancel: { text: "关闭" } }, width: PanelUISize.settingBig.width, height: PanelUISize.settingBig.height, style: blackHomeCSS, mask: { enable: true } }); let $list = $confirm.$shadowRoot.querySelector( ".blackhome-user-list" ); let $filterInput = $confirm.$shadowRoot.querySelector( ".blackhome-user-filter input" ); blackListInfo.data.forEach((item) => { let $item = this.createListViewItem(item); $list.appendChild($item); }); Qmsg.success(`成功获取 ${blackListInfo.data.length}条数据`); let isSeaching = false; domUtils.on( $filterInput, ["input", "propertychange"], utils.debounce(() => { let inputText = $filterInput.value.trim(); if (isSeaching) { return; } isSeaching = true; if (inputText == "") { $confirm.$shadowRoot.querySelectorAll(".blackhome-user-item").forEach((item) => { item.removeAttribute("style"); }); isSeaching = false; return; } $confirm.$shadowRoot.querySelectorAll(".blackhome-user-item").forEach((item) => { if (item.getAttribute("data-name").match(new RegExp(inputText, "ig")) || item.getAttribute("data-uid").trim().match(new RegExp(inputText, "ig")) || item.getAttribute("data-operator").match(new RegExp(inputText, "ig"))) { item.removeAttribute("style"); } else { item.setAttribute("style", "display:none;"); } }); isSeaching = false; }) ); let nextBlackListInfo = await this.getBlackListInfo(this.$data.cid); if (!nextBlackListInfo) { return; } this.$data.cid = nextBlackListInfo.next_cid; }, /** * 获取小黑屋名单信息 */ async getBlackListInfo(cid = "") { let searchParamsData = { mod: "misc", action: "showdarkroom", cid, ajaxdata: "json" }; let response = await httpx.get( `/forum.php?${utils.toSearchParamsStr(searchParamsData)}`, { headers: { "User-Agent": utils.getRandomPCUA() }, responseType: "json" } ); if (!response.status) { return; } let data = utils.toJSON(response.data.responseText); let cidSplit = data["message"].split("|"); let next_cid = cidSplit[cidSplit.length - 1]; let blackListData = utils.parseObjectToArray(data["data"]); let new_blackListData = []; let new_blackListData_noTime = []; blackListData.forEach((item) => { let date = item["dateline"].match( /([0-9]{4}-[0-9]{1,2}-[0-9]{1,2}[\s]*[0-9]{1,2}:[0-9]{1,2})/g ); if (date == null) { let _time_ = parseInt((Date.now() / 1e3).toString()); let _time_after_count_ = 0; let sec_data = item["dateline"].match(/([0-9]+|半)[\s\S]*秒前/); let min_data = item["dateline"].match(/([0-9]+|半)[\s\S]*分钟前/); let hour_data = item["dateline"].match(/([0-9]+|半)[\s\S]*小时前/); let yesterday_time_data = item["dateline"].match( /昨天[\s\S]*(\d{2}):(\d{2})/ ); let before_yesterday_time_data = item["dateline"].match( /前天[\s\S]*(\d{2}):(\d{2})/ ); let day_data = item["dateline"].match(/([0-9]+|半)[\s\S]*天前/); if (sec_data) { sec_data = sec_data[sec_data.length - 1]; sec_data = sec_data.replace(/半/g, 0.5); sec_data = parseFloat(sec_data); _time_after_count_ = _time_ - sec_data; } else if (min_data) { min_data = min_data[min_data.length - 1]; min_data = min_data.replace(/半/g, 0.5); min_data = parseFloat(min_data); _time_after_count_ = _time_ - min_data * 60; } else if (hour_data) { hour_data = hour_data[hour_data.length - 1]; hour_data = hour_data.replace(/半/g, 0.5); hour_data = parseFloat(hour_data); _time_after_count_ = _time_ - hour_data * 60 * 60; } else if (yesterday_time_data) { let yesterday_hour_data = yesterday_time_data[1]; let yesterday_min_data = yesterday_time_data[2]; _time_after_count_ = _time_ - 86400 - parseInt(yesterday_hour_data) * 3600 - parseInt(yesterday_min_data) * 60; } else if (before_yesterday_time_data) { let before_yesterday_hour_data = before_yesterday_time_data[1]; let before_yesterday_min_data = before_yesterday_time_data[2]; _time_after_count_ = _time_ - 86400 * 2 - parseInt(before_yesterday_hour_data) * 3600 - parseInt(before_yesterday_min_data) * 60; } else if (day_data) { day_data = day_data[day_data.length - 1]; day_data = day_data.replace(/半/g, 0.5); day_data = parseFloat(day_data); _time_after_count_ = _time_ - day_data * 60 * 60 * 24; } item["time"] = parseInt(_time_after_count_.toString()) * 1e3; new_blackListData = new_blackListData.concat(item); return; } else { date = date[0]; } item["time"] = utils.formatToTimeStamp(date); new_blackListData = new_blackListData.concat(item); }); utils.sortListByProperty(new_blackListData, "time"); utils.sortListByProperty(new_blackListData_noTime, "time", false); new_blackListData = new_blackListData.concat(new_blackListData_noTime); return { next_cid, data: new_blackListData }; }, /** * 创建黑名单节点 */ createListViewItem(userInfo) { let $item = domUtils.createElement( "div", { className: "blackhome-user-item", innerHTML: ( /*html*/ ` <div class="blackhome-user-avatar"> <div class="blackhome-user"> <img src="${MTUtils.getAvatar( userInfo["uid"], "big" )}" loading="lazy"> <div class="blackhome-user-info"> <p>${userInfo["username"]}</p> <p>${userInfo["dateline"]}</p> </div> </div> <div class="blackhome-user-action"> <p>${userInfo["action"]}</p> <p>到期: ${userInfo["groupexpiry"]}</p> </div> <div class="blackhome-user-uuid">UID: ${userInfo["uid"]}</div> <div class="blackhome-operator"> <div class="blackhome-operator-user"> <img loading="lazy" src="${MTUtils.getAvatar( userInfo["operatorid"], "big" )}"> <p>${userInfo["operator"]}</p> </div> <div class="blackhome-operator-user-info"> ${userInfo["reason"]} </div> </div> </div> ` ) }, { "data-name": userInfo.username, "data-uid": userInfo.uid, "data-operator": userInfo.operator, "data-operator-uid": userInfo.operatorid } ); domUtils.on( $item, "click", ".blackhome-user img", function() { window.open( `home.php?mod=space&uid=${userInfo.uid}&do=profile`, "_blank" ); } ); domUtils.on( $item, "click", ".blackhome-operator-user img", function() { window.open( `home.php?mod=space&uid=${userInfo.operatorid}&do=profile`, "_blank" ); } ); return $item; } }; const onlineUserCSS = '.pops-alert-content {\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n.pops-alert-content > .online-user-info {\r\n text-align: center;\r\n padding: 0px 6px;\r\n}\r\n.online-user-filter input {\r\n width: -webkit-fill-available;\r\n width: -moz-available;\r\n height: 30px;\r\n margin: 8px 20px;\r\n border: 0;\r\n border-bottom: 1px solid;\r\n}\r\n.online-user-filter input:focus-within {\r\n outline: none;\r\n}\r\n.online-user-list {\r\n flex: 1;\r\n overflow-y: auto;\r\n}\r\n.online-user-list li {\r\n margin: 18px 0;\r\n}\r\n.online-user {\r\n display: flex;\r\n margin: 2px 20px;\r\n align-items: center;\r\n}\r\n.online-user img[data-avatar] {\r\n width: 45px;\r\n height: 45px;\r\n border-radius: 45px;\r\n}\r\n.online-user-list .online-user-info {\r\n margin: 2px 14px;\r\n}\r\n.online-user-list .online-user-info p[data-name] {\r\n margin-bottom: 4px;\r\n}\r\n.online-user-list .online-user-info span[data-sf] {\r\n border-radius: 4px;\r\n padding: 2px 4px;\r\n font-weight: 500;\r\n font-size: 14px;\r\n}\r\n.online-user-list .online-user-info span[data-uid] {\r\n border: 1px solid #ff7600;\r\n color: #ff7600;\r\n border-radius: 4px;\r\n padding: 2px 4px;\r\n font-weight: 500;\r\n font-size: 14px;\r\n width: fit-content;\r\n width: -moz-fit-content;\r\n margin: 10px 0;\r\n}\r\n.online-user-list .online-user-info span[data-sf="会员"] {\r\n color: #88b500;\r\n border: 1px solid #88b500;\r\n}\r\n.online-user-list .online-user-info span[data-sf="版主"] {\r\n color: #2db5e3;\r\n border: 1px solid #2db5e3;\r\n}\r\n.online-user-list .online-user-info span[data-sf="超级版主"] {\r\n color: #e89e38;\r\n border: 1px solid #e89e38;\r\n}\r\n.online-user-list .online-user-info span[data-sf="管理员"] {\r\n color: #ff5416;\r\n border: 1px solid #ff5416;\r\n}\r\n\r\n@media screen and (min-width: 800px) {\r\n .online-user-list {\r\n display: flex;\r\n flex-wrap: wrap;\r\n }\r\n .online-user-list .online-item {\r\n flex: 1 1 250px;\r\n }\r\n}\r\n'; const MTOnlineUser = { $data: {}, init() { this.registerMenu(); }, /** * 注册(不可用)菜单 */ registerMenu() { GM_Menu.add({ key: "online-user", text: "⚙ 在线用户", autoReload: false, isStoreValue: false, showText(text) { return text; }, callback: () => { this.showOnlineUser(); } }); }, /** * 显示在线用户dialog */ async showOnlineUser() { let $loading = Qmsg.loading("正在获取在线用户名单中..."); let onlineUserInfo = await this.getOnlineUserListInfo(); $loading.close(); if (!onlineUserInfo) { return; } let $alert = __pops.alert({ title: { text: "在线用户", position: "center" }, content: { text: ( /*html*/ ` <div class="online-user-info">${onlineUserInfo.totalOnline} 人在线 - ${onlineUserInfo.onlineUser} 会员${onlineUserInfo.invisibleUser == 0 ? "" : `(${onlineUserInfo.invisibleUser}隐身)`} - ${onlineUserInfo.noRegisterUser} 位游客</div> <div class="online-user-filter"> <input placeholder="过滤用户名/身份/UID(可正则)"></div> <div class="online-user-list"></div> ` ), html: true }, btn: { ok: { text: "关闭", type: "default" } }, width: PanelUISize.settingBig.width, height: PanelUISize.settingBig.height, style: onlineUserCSS, mask: { enable: true } }); let $list = $alert.$shadowRoot.querySelector(".online-user-list"); let $filterInput = $alert.$shadowRoot.querySelector( ".online-user-filter input" ); onlineUserInfo.data.forEach((item) => { let $item = this.createListViewItem(item); $list.appendChild($item); }); Qmsg.success(`成功获取 ${onlineUserInfo.data.length}条数据`); let isSeaching = false; DOMUtils.on( $filterInput, ["propertychange", "input"], utils.debounce(() => { let inputText = $filterInput.value.trim(); if (isSeaching) { return; } isSeaching = true; if (inputText == "") { $alert.$shadowRoot.querySelectorAll(".online-user-list .online-item").forEach((item) => { item.removeAttribute("style"); }); isSeaching = false; return; } $alert.$shadowRoot.querySelectorAll(".online-user-list .online-item").forEach((item) => { if (item.getAttribute("data-name").match(new RegExp(inputText, "ig")) || item.getAttribute("data-sf").match(new RegExp(inputText, "ig")) || item.getAttribute("data-uid").match(new RegExp(inputText, "ig"))) { item.removeAttribute("style"); } else { item.setAttribute("style", "display:none;"); } }); isSeaching = false; }) ); }, /** * 获取在线用户名单信息 */ async getOnlineUserListInfo() { let searchParamsData = { showoldetails: "yes" }; let response = await httpx.get( `/forum.php?${utils.toSearchParamsStr(searchParamsData)}`, { headers: { "User-Agent": utils.getRandomPCUA() } } ); if (!response.status) { return; } let pageHTML = utils.parseFromString( response.data.responseText, "text/html" ); let result = { data: [], totalOnline: 0, onlineUser: 0, noRegisterUser: 0, invisibleUser: 0 }; let onlineList = pageHTML.querySelectorAll("#onlinelist ul li"); onlineList.forEach((item) => { let uid = item.querySelector("a").getAttribute("href").match("uid-(.+?).html")[1]; let avatar = MTUtils.getAvatar(uid, "middle"); let name = item.querySelector("a").innerText; let sf = ""; let space = item.querySelector("a").getAttribute("href"); let memberSrc = item.querySelector("img").src; if (memberSrc.indexOf("online_member") != -1) { sf = "会员"; } else if (memberSrc.indexOf("online_moderator") != -1) { sf = "版主"; } else if (memberSrc.indexOf("online_supermod") != -1) { sf = "超级版主"; } else if (memberSrc.indexOf("online_admin") != -1) { sf = "管理员"; } else { sf = "未知身份"; } result.data.push({ uid, avatar, name, sf, space }); }); let onlineInfo = pageHTML.querySelector("#online div.bm_h span.xs1").textContent; result.totalOnline = utils.parseInt( onlineInfo.match(/([0-9]*)\s*人在线/i), 0 ); result.onlineUser = utils.parseInt( onlineInfo.match(/([0-9]*)\s*会员/i), 0 ); result.noRegisterUser = utils.parseInt( onlineInfo.match(/([0-9]*)\s*位游客/i), 0 ); result.invisibleUser = utils.parseInt( onlineInfo.match(/([0-9]*)\s*隐身/i), 0 ); return result; }, /** * 创建在线用户节点 */ createListViewItem(userInfo) { let $item = DOMUtils.createElement( "div", { className: "online-item", innerHTML: ( /*html*/ ` <div class="online-user"> <img data-avatar src="${userInfo["avatar"]}" loading="lazy" class="online-user-avatar"> <div class="online-user-info"> <p data-name>${userInfo["name"]}</p> <span data-sf="${userInfo["sf"]}">${userInfo["sf"]}</span> <span data-uid>UID: ${userInfo["uid"]}</span> </div> </div> ` ) }, { "data-name": userInfo.name, "data-uid": userInfo.uid, "data-sf": userInfo.sf } ); DOMUtils.on($item, "click", ".online-user-avatar", (event) => { utils.preventEvent(event); window.open( `home.php?mod=space&uid=${userInfo.uid}&do=profile`, "_blank" ); }); return $item; } }; const MT = { $flag: { showUserUID_initCSS: false }, init() { PopsPanel.onceExec("mt-MTCommentFilter", () => { MTCommentFilter.init(); }); if (Router.isPost()) { log.info(`Router: 帖子`); MTForumPost.init(); } else if (Router.isGuide()) { log.info(`Router: 导读`); MTGuide.init(); } else { log.error(`Router: 未适配的链接 ==> ` + window.location.href); } domUtils.ready(() => { PopsPanel.onceExec("mt-MTProductListingReminder", () => { MTProductListingReminder.init(); }); PopsPanel.onceExec("mt-blackHome", () => { MTBlackHome.init(); }); PopsPanel.onceExec("mt-onlineUser", () => { MTOnlineUser.init(); }); PopsPanel.execMenuOnce("mt-link-text-to-hyperlink", () => { MTIdentifyLinks(); }); PopsPanel.execMenuOnce("mt-addLatestPostBtn", () => { this.addLatestPostBtn(); }); PopsPanel.execMenu("mt-auto-sign", () => { MTAutoSignIn.init(); }); PopsPanel.execMenu("mt-extend-cookie-expire", () => { this.extendCookieExpire(); }); }); }, /** * 新增【最新发表】 */ addLatestPostBtn() { log.info(`新增【最新发表】`); domUtils.append( "#comiis_nv .wp.comiis_nvbox.cl ul", /*html*/ ` <li id="latest_publication"> <a href="/forum.php?mod=guide&view=newthread" hidefocus="true" title="最新发表">最新发表</a> </li> ` ); if (window.location.href.includes("/forum.php?mod=guide&view=newthread")) { domUtils.removeClass("#mn_forum_10", "a"); domUtils.css( "#latest_publication a", "background", 'url("") repeat-x 50% -50px' ); } }, /** * 延长cookie有效期 */ async extendCookieExpire() { log.info(`延长cookie有效期`); let cookieList = await _GM.cookie.list({}); let needExtendCookieNameList = [ "_auth", "_saltkey", "_client_created", "_client_token" ]; cookieList.forEach(async (cookieItem) => { if (cookieItem.session) { return; } let expireTime = cookieItem.expirationDate; let nowTime = Date.now() / 1e3; if (expireTime < nowTime) { return; } let _30days = 60 * 60 * 24 * 30; if (expireTime - nowTime > _30days) { return; } let flag = needExtendCookieNameList.find( (it) => cookieItem.name.endsWith(it) ); if (!flag) { return; } _GM.cookie.set({ name: cookieItem.name, value: cookieItem.value, expirationDate: cookieItem.expirationDate + _30days }).then(() => { log.info(`延长Cookie +30天成功:${cookieItem.name}`); }).catch(() => { log.error(`延长Cookie +30天失败:${cookieItem.name}`); }); }); } }; PopsPanel.init(); MT.init(); } }); require_entrance_001(); })(Qmsg, DOMUtils, Utils, pops, hljs, Viewer);
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址