MT论坛优化
// ==UserScript==
// @name MT论坛优化
// @namespace https://github.com/WhiteSevs/TamperMonkeyScript
// @version 2024.12.27
// @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://update.greasyfork.org/scripts/494167/1413255/CoverUMD.js
// @require https://fastly.jsdelivr.net/npm/@whitesev/utils@2.5.6/dist/index.umd.js
// @require https://fastly.jsdelivr.net/npm/@whitesev/domutils@1.4.8/dist/index.umd.js
// @require https://fastly.jsdelivr.net/npm/@whitesev/pops@1.9.6/dist/index.umd.js
// @require https://fastly.jsdelivr.net/npm/qmsg@1.2.8/dist/index.umd.js
// @require https://fastly.jsdelivr.net/npm/viewerjs@1.11.7/dist/viewer.min.js
// @require https://fastly.jsdelivr.net/npm/@highlightjs/cdn-assets@11.11.0/highlight.min.js
// @resource HljsCSS https://fastly.jsdelivr.net/npm/highlight.js@11.11.0/styles/github-dark.min.css
// @resource ViewerCSS https://fastly.jsdelivr.net/npm/viewerjs@1.11.7/dist/viewer.min.css
// @connect *
// @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-DfuMZFbd.js"(exports, module) {
var _a;
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 = {
ElementPlus: {
keyName: "ElementPlusResourceCSS",
url: "https://fastly.jsdelivr.net/npm/element-plus@latest/dist/index.min.css"
},
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]+)/,
/** 论坛账号的凭证 */
hash: /hash=(.+)&/,
/** 用户uid */
uid: /uid(=|-)(\d+)/,
/** 帖子内特殊字体格式 */
fontSpecial: /<font.*?>|<\/font>|<strike>|<strong>|<i>|<u>|align=".*?"|<br>[\s]*<br>[\s]*<br>/g,
/** 帖子链接的ptid参数 */
ptid: /&ptid=([\d]+)/i,
/** 帖子链接的pid参数 */
pid: /&pid=([\d]+)/i,
/** 链接的tid参数 */
tid: /&tid=([\d]+)/i
};
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 searchParamsData = {
operation: "qiandao",
format: "button",
formhash: formHash,
inajax: 1,
ajaxtarget: "midaben_sign"
};
let useFetch = Boolean(PopsPanel.getValue("mt-auto-sign-useFetch"));
let response = await httpx.get(
`/k_misign-sign.html?${utils.toSearchParamsStr(searchParamsData)}`,
{
fetch: useFetch,
headers: {
"User-Agent": utils.getRandomPCUA()
},
allowInterceptConfig: false
}
);
if (!response.status) {
log.error("签到:网络异常,请求失败");
Qmsg.error("签到:网络异常,请求失败");
return;
}
this.setSignTime();
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
});
this.clearSignTime();
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;
}
let $alert = pops.alert({
title: {
text: "未知签到内容",
position: "center"
},
content: {
text: "",
html: false
},
width: "88vw",
height: "400px"
});
let $content = $alert.$shadowRoot.querySelector(
".pops-alert-content"
);
$content.innerText = response.data.responseText;
}
};
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,
"自动把符合超链接格式的文字转为超链接"
)
]
}
]
},
{
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";
}
},
/**
* 信息界面,一般用于提示信息之类
*/
info: {
get width() {
return window.innerWidth < 350 ? "350px" : "350px";
},
get height() {
return window.innerHeight < 250 ? "250px" : "250px";
}
}
};
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: window.innerWidth > 500 ? "500px" : "88vw",
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;
}
/**
* 显示视图
*/
async showView() {
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(
$popsConfirm.$shadowRoot,
false,
await this.option.getAddData()
);
}
},
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 4px;
gap: 6px;
}
.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 4px;
}
.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();
for (let index = 0; index < allData.length; index++) {
await this.appendRuleItemElement(
$popsConfirm.$shadowRoot,
allData[index]
);
}
}
/**
* 解析弹窗内的各个元素
*/
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($shadowRoot, true, data, $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) {
const { $container } = this.parseViewElement($shadowRoot);
if (Array.isArray(data)) {
for (let index = 0; index < data.length; index++) {
const item = data[index];
$container.appendChild(
await this.createRuleItemElement(item, $shadowRoot)
);
}
} else {
$container.appendChild(
await this.createRuleItemElement(data, $shadowRoot)
);
}
await this.updateDeleteAllBtnText($shadowRoot);
}
/**
* 更新弹窗内容的元素
*/
async updateRuleContaienrElement($shadowRoot) {
this.clearContent($shadowRoot);
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);
}
}
/**
* 更新【清空所有】的按钮的文字
*/
async updateDeleteAllBtnText($shadowRoot) {
let data = await this.option.data();
this.setDeleteBtnText($shadowRoot, `清空所有(${data.length})`);
}
/**
* 显示编辑视图
* @param isEdit 是否是编辑状态
*/
showEditView($parentShadowRoot, isEdit, editData, $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("修改成功");
await this.updateRuleItemElement(
result.data,
$editRuleItemElement,
$parentShadowRoot
);
} else {
await this.appendRuleItemElement($parentShadowRoot, result.data);
}
} else {
if (isEdit) {
Qmsg.error("修改失败");
}
}
return result;
},
style: this.option.itemControls.edit.style
});
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();
});
});
},
/**
* 新增【最新发表】
*/
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'
);
}
}
};
PopsPanel.init();
MT.init();
}
});
require_entrance_001();
})(Qmsg, DOMUtils, Utils, pops, hljs, Viewer);