// ==UserScript==
// @name 深色模式
// @namespace https://greasyfork.org/zh-CN/users/1196880-ling2ling4
// @version 1.0.0
// @author Ling2Ling4
// @description 设置页面为深色模式
// @license AGPL-3.0-or-later
// @icon 
// @match *
// @run-at document-end
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_registerMenuCommand
// @compatible chrome
// @compatible edge
// @compatible firefox
// ==/UserScript==
(() => {
"use strict";
const keyBase = "ll_pageDarkMode_",
info = {
keyBase,
settingsArea: null,
isDarkMode: !1,
isCanRun: !0,
timer: null,
interval: 5e3,
settings: {
startTime: {
value: "0",
base: "0",
key: keyBase + "startTime",
valType: "string",
type: "基础设置",
desc: "深色模式的自动开启时间, 0表示关闭, 按照24小时制书写, 格式为 xx:xx, 如: 20:00",
compType: "textarea",
compH: "30px",
},
stopTime: {
value: "0",
base: "0",
key: keyBase + "stopTime",
valType: "string",
type: "基础设置",
desc: "深色模式的自动关闭时间, 0表示关闭, 按照24小时制书写",
compType: "textarea",
compH: "30px",
},
invert: {
value: 1,
base: 1,
key: keyBase + "invert",
valType: "number",
type: "颜色调整",
title: "颜色反转",
desc: "表示颜色反转的程度, 深色效果主要与该设置相关. 默认1, 范围0-1",
compType: "textarea",
compH: "30px",
},
brightness: {
value: 1,
base: 1,
key: keyBase + "brightness",
valType: "number",
type: "颜色调整",
title: "亮度",
desc: "表示亮度的大小. 默认1, 范围0-∞",
compType: "textarea",
compH: "30px",
},
contrast: {
value: 1,
base: 1,
key: keyBase + "contrast",
valType: "number",
type: "颜色调整",
title: "对比度",
desc: "表示对比度的强弱. 默认1, 范围0-∞",
compType: "textarea",
compH: "30px",
},
grayscale: {
value: 0,
base: 0,
key: keyBase + "grayscale",
valType: "number",
type: "颜色调整",
title: "灰度",
desc: "表示灰度的程度. 默认0, 范围0-1",
compType: "textarea",
compH: "30px",
},
hueRotate: {
value: 0,
base: 0,
key: keyBase + "hueRotate",
valType: "number",
type: "颜色调整",
title: "色调",
desc: "表示色调的旋转变化. 默认0, 范围0-360",
compType: "textarea",
compH: "30px",
},
saturate: {
value: 1,
base: 1,
key: keyBase + "saturate",
valType: "number",
type: "颜色调整",
title: "饱和度",
desc: "表示饱和度的高低. 默认1, 范围0-∞",
compType: "textarea",
compH: "30px",
},
sepia: {
value: 0,
base: 0,
key: keyBase + "sepia",
valType: "number",
type: "颜色调整",
title: "深褐色",
desc: "表示深褐色的程度. 默认0, 范围0-1",
compType: "textarea",
compH: "30px",
},
website: {
value: "*",
base: "*",
key: keyBase + "website",
valType: "string",
type: "其他设置",
desc: "以下网站可启用深色模式, 多个网站请换行书写, *表示所有网站都可启用\n【示例】*.bilibili.com/* 可匹配B站\n",
compType: "textarea",
compH: "90px",
},
notInvertNodes: {
value: "img, video",
base: "img, video",
key: keyBase + "notInvertNodes",
valType: "string",
type: "其他设置",
title: "不反转的元素",
desc: "不进行颜色反转的元素, 每项用 , 分隔, 可书写css选择器\n【可选】\nh1, h2, h3, h4, h5, h6, p, span, ul, li, i, svg\na, img, form, input, textarea, button, select, option, label, audio, video, ....",
compType: "textarea",
compH: "90px",
},
},
};
function setDarkMode(isDark = !0) {
if (info.isDarkMode === isDark) return;
info.isDarkMode = isDark;
const id = info.keyBase + "css";
let isAddCss = !1,
dom = document.head.querySelector("#" + id);
dom ||
((dom = document.createElement("style")), (dom.id = id), (isAddCss = !0)),
(dom.innerHTML = isDark
? (function getCssHtml() {
const settings = info.settings;
return `html {\n background-color:#fff;\n filter:invert(${
settings.invert.value
}) brightness(${settings.brightness.value}) contrast(${
settings.contrast.value
}) grayscale(${settings.grayscale.value}) hue-rotate(${
settings.hueRotate.value
}deg) saturate(${settings.saturate.value}) sepia(${
settings.sepia.value
});\n}\n${settings.notInvertNodes.value.replaceAll(
",",
","
)}{\n filter:invert(1);\n}`;
})()
: ""),
isAddCss && document.head.appendChild(dom),
console.log((isDark ? "开启" : "关闭") + "深色模式");
}
function setStartStopTimer() {
const settings = info.settings,
startTime = settings.startTime.value,
stopTime = settings.stopTime.value;
if (0 == +startTime && 0 == +stopTime) return;
const autoStartStop = () => {
const f = (function isNeedDarkMode() {
const settings = info.settings,
startTime = settings.startTime.value,
stopTime = settings.stopTime.value;
if (0 == +startTime && 0 == +stopTime) return -1;
const t = new Date(),
curT = 60 * t.getHours() + t.getMinutes();
let startT, stopT;
if (0 != +startTime) {
const tArr1 = startTime.trim().replace(":", ":").split(":");
startT = 60 * +tArr1[0] + +tArr1[1];
}
if (0 != +stopTime) {
const tArr2 = stopTime.trim().replace(":", ":").split(":");
stopT = 60 * +tArr2[0] + +tArr2[1];
}
if (0 == +startTime) return !(curT >= stopT) && -1;
if (0 == +stopTime) return curT >= startT || -1;
const f = (function isTimeInRange(t, startTime, stopTime, rangeLimit) {
const curH = t.getHours(),
curMin = t.getMinutes(),
startText = startTime.trim().replace(":", ":"),
stopText = stopTime.trim().replace(":", ":"),
tArr1 = startText.split(":"),
tArr2 = stopText.split(":"),
h1 = +tArr1[0],
h2 = +tArr2[0],
startT = 60 * h1 + +tArr1[1],
stopT = 60 * h2 + +tArr2[1],
curT = 60 * curH + curMin;
if (startT < stopT)
if (rangeLimit[0]) {
if (rangeLimit[1]) {
if (curT >= startT && curT <= stopT) return !0;
} else if (curT >= startT && curT < stopT) return !0;
} else if (rangeLimit[1]) {
if (curT > startT && curT <= stopT) return !0;
} else if (curT > startT && curT < stopT) return !0;
if (startT > stopT)
if (rangeLimit[0]) {
if (rangeLimit[1]) {
if (
(curT >= startT && curT < 1440) ||
(curT <= stopT && curT >= 0)
)
return !0;
} else if (
(curT >= startT && curT < 1440) ||
(curT < stopT && curT >= 0)
)
return !0;
} else if (rangeLimit[1]) {
if (
(curT > startT && curT < 1440) ||
(curT <= stopT && curT >= 0)
)
return !0;
} else if (
(curT > startT && curT < 1440) ||
(curT < stopT && curT >= 0)
)
return !0;
return !1;
})(t, startTime, stopTime, [1, 0]);
return f;
})();
-1 !== f && setDarkMode(f);
};
autoStartStop(),
info.timer && clearInterval(info.timer),
(info.timer = setInterval(autoStartStop, info.interval));
}
function verifyWebsite() {
const url = location.href,
f = info.settings.website.value
.trim()
.split("\n")
.some((item) =>
(function matchUrlWithWildcard(url, pattern) {
return new RegExp("^" + pattern.replace(/\*/g, ".*") + "$").test(
url
);
})(url, item)
);
return (
f &&
!info.isCanRun &&
GM_registerMenuCommand("切换深色模式", () => {
setDarkMode(!info.isDarkMode);
}),
(info.isCanRun = f),
f
);
}
const baseCfg = {
isEditing: !1,
hasSelectedPage: !1,
param: {
id: "ll_edit_wrap",
box: document.body,
classBase: "ll_edit_",
w: "500px",
h: "",
contentH: "450px",
bg: "rgba(0, 0, 0, 0.15)",
color: "#333",
fontSize: "15px",
fontFamily:
"PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif",
zIndex: 11e3,
resetTt: "重置所有设置为默认值",
isShowMenu: !1,
isScrollStyle: !0,
isResetBtn: !0,
isOnlyResetCurPage: !1,
showPage: void 0,
page: [],
},
},
cfg = {
version: "v1.1.2",
isEditing: baseCfg.isEditing,
hasSelectedPage: baseCfg.hasSelectedPage,
param: {},
tempParam: {},
allData: {},
baseData: {},
oldData: {},
controls: {},
doms: { page: [] },
editText: {},
callback: { resetBefore: null, confirmBefore: null, finished: null },
};
const css = function getCss() {
const param = cfg.param,
cBase = (param.page, param.classBase),
baseStart = `#${param.id} .${cBase}`,
fSize = param.fontSize ? param.fontSize : "14px",
css = `#${
param.id
} {\n position: fixed;\n left: 0;\n top: 0;\n width: 100%;\n height: 100%;\n z-index: ${
param.zIndex || 11e3
};\n background: ${
param.bg || "rgba(0, 0, 0, 0.12)"
};\n display: none;\n}\n${baseStart}box {\n letter-spacing: 1px;\n position: relative;\n width: ${
param.w || "450px"
};\n ${
param.h ? "max-height:" + param.h : ""
};\n margin: auto;\n color: ${
param.color || "#333"
};\n background: #fff;\n font-size: ${fSize};\n line-height: normal;\n font-family: ${
param.fontFamily ||
"PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif"
};\n border: 3px solid #dfedfe;\n border-radius: 10px;\n box-sizing: border-box;\n padding: 14px 8px 10px 15px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}menu {\n font-weight: bold;\n font-size: ${
parseInt(fSize) + 1
}px;\n display: flex;\n flex-wrap: wrap;\n gap: 0 8px;\n}\n${baseStart}menu-item {\n margin-bottom: 8px;\n border: 1px solid #dfedfe;\n color: #9ecaff;\n background: #eef6ff;\n border-radius: 6px;\n padding: 6px 10px;\n cursor: pointer;\n}\n${baseStart}menu-item:hover {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}menu-item.active {\n color: #65aaff;\n background: #dfedfe;\n border: 1px solid #dfedfe;\n}\n${baseStart}page-box {\n max-height: ${
param.contentH || ""
};\n padding-right: 7px;\n margin-bottom: 8px;\n overflow: hidden;\n overflow-y: auto;\n}\n${baseStart}page {\n display: none;\n}\n${baseStart}page.curPage {\n display: block;\n}\n${baseStart}comp {\n margin-bottom: 8px;\n}\n${baseStart}comp:last-child {\n margin-bottom: 2px;\n}\n${baseStart}tt {\n font-weight: bold;\n font-size: ${
parseInt(fSize) + 6
}px;\n margin-top: 4px;\n}\n${baseStart}tt2 {\n font-weight: bold;\n font-size: ${
parseInt(fSize) + 4
}px;\n margin-top: 3px;\n margin-bottom: 7px;\n}\n${baseStart}tt3 {\n font-weight: bold;\n font-size: ${
parseInt(fSize) + 2
}px;\n margin-top: 2px;\n margin-bottom: 6px;\n}\n${baseStart}desc {\n line-height: 1.5;\n}\n${baseStart}comp-tt {\n font-weight: bold;\n font-size: ${
parseInt(fSize) + 1
}px;\n line-height: 1.5;\n}\n${baseStart}comp-desc {\n line-height: 1.5;\n}\n${baseStart}rd-arr {\n line-height: 22px;\n}\n${baseStart}rd-arr label {\n margin-right: 6px;\n cursor: pointer;\n}\n${baseStart}rd-arr input {\n vertical-align: -2px;\n cursor: pointer;\n}\n${baseStart}rd-arr span {\n color: #666;\n margin-left: 2px;\n}\n#${
param.id
} textarea {\n width: 100%;\n max-width: 100%;\n max-height: 300px;\n border-radius: 6px;\n line-height: normal;\n padding: 5px 7px;\n outline-color: #cee4ff;\n border: 1px solid #aaa;\n box-sizing: border-box;\n font-size: ${
parseInt(fSize) - 2
}px;\n font-family: PingFang SC, HarmonyOS_Regular, Helvetica Neue, Microsoft YaHei, sans-serif;\n /* 保留空格 */\n white-space: pre-wrap;\n /* 允许词内换行 */\n word-break: break-all;\n letter-spacing: 1px;\n overflow: hidden;\n overflow-y: auto;\n}\n#${
param.id
} textarea::placeholder {\n color: #bbb;\n}\n${baseStart}ta-desc {\n margin-bottom: 3px;\n}\n${baseStart}btn-box {\n display: flex;\n justify-content: flex-end;\n}\n${baseStart}btn-box button {\n font-size: 16px;\n line-height: normal;\n color: #65aaff;\n background: #dfedfe;\n outline: none;\n border: none;\n border-radius: 6px;\n padding: 8px 16px;\n box-sizing: border-box;\n cursor: pointer;\n}\n${baseStart}btn-box .${cBase}reset-btn {\n position: absolute;\n left: 15px;\n bottom: 10px;\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}reset-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}cancel-btn {\n color: #888;\n background: #f4f4f4;\n margin-right: 15px;\n}\n${baseStart}btn-box .${cBase}cancel-btn:hover {\n color: #666;\n background: #eee;\n}\n${baseStart}btn-box .${cBase}confirm-btn {\n margin-right: 7px;\n}\n${baseStart}btn-box .${cBase}confirm-btn:hover {\n background: #cee4ff;\n}\n`;
return param.isScrollStyle
? css +
"\n.ll-scroll-style-1::-webkit-scrollbar,\n.ll-scroll-style-1 ::-webkit-scrollbar {\n width: 8px;\n}\n.ll-scroll-style-1-size-2::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-2::-webkit-scrollbar {\n width: 10px;\n}\n.ll-scroll-style-1-size-3::-webkit-scrollbar,\n.ll-scroll-style-1 .ll-scroll-style-1-size-3::-webkit-scrollbar {\n width: 12px;\n}\n.ll-scroll-style-1::-webkit-scrollbar-thumb,\n.ll-scroll-style-1 ::-webkit-scrollbar-thumb {\n border-radius: 10px;\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.05);\n opacity: 0.2;\n background: #daedff;\n}\n.ll-scroll-style-1::-webkit-scrollbar-track,\n.ll-scroll-style-1 ::-webkit-scrollbar-track {\n -webkit-box-shadow: inset 0 0 8px rgba(0, 0, 0, 0.08);\n border-radius: 0;\n background: #fff;\n border-radius: 5px;\n}"
: css;
};
const editArea_html = function getHTML() {
function getCompHTML({ info, active = "", id }) {
let type = info.type;
if (
((type = {
menuTitle: "mtt",
title: "tt",
title2: "tt2",
title3: "tt3",
desc: "ds",
radio: "rd",
checkbox: "cb",
textarea: "ta",
mtt: "mtt",
tt: "tt",
tt2: "tt2",
tt3: "tt3",
ds: "ds",
rd: "rd",
cb: "cb",
ta: "ta",
}[type]),
(id = 0 === id ? "0" : id || ""),
0 === info.value && (info.value = "0"),
!type)
)
return console.log("不存在的组件类型"), !1;
let title = "",
desc = "",
ctrlTt = "";
switch (
(["tt", "tt2", "tt3", "ds", "mtt"].includes(type) ||
((title = info.title
? `<div class="${cBase}comp-tt ${cBase}${type}-tt" title="${
info.tt || ""
}">${info.title}</div>`
: ""),
(desc = info.desc
? `<div class="${cBase}comp-desc ${cBase}${type}-desc">${info.desc}</div>`
: "")),
type)
) {
case "mtt":
return (
(info.value = info.value || ""),
info.value
? `<div class="${cBase}menu-item ${active || ""}" title="${
info.tt || ""
}">${info.value}</div>`
: ""
);
case "tt":
case "tt2":
case "tt3":
return (
(info.value = info.value || ""),
info.value
? `<div class="${cBase}${type} ${cBase}comp" title="${
info.tt || ""
}">${info.value}</div>`
: ""
);
case "ds":
return (
(info.value = info.value || ""),
info.value
? `<div class="${cBase}desc ${cBase}comp" title="${
info.descTt || ""
}">${info.value}</div>`
: ""
);
case "rd":
const name = info.name || info.id + new Date().getTime();
(ctrlTt = info.ctrlTt || ""),
ctrlTt && (ctrlTt = `title="${ctrlTt}"`);
let radio = `<div class="${cBase}rd ${cBase}rd-arr" ${ctrlTt}>`;
if (void 0 === info.value && info.radioList[0]) {
const obj = info.radioList[0];
info.value = void 0 === obj.value ? obj.text : obj.value;
}
return (
info.radioList.forEach((item, i) => {
void 0 === item.value && (info.radioList[i].value = item.text),
void 0 === item.text && (info.radioList[i].text = item.value);
const value = item.value;
let tt = item.tt || "";
tt && (tt = `title="${tt}"`);
let selected = "";
info.value + "" == item.value + "" && (selected = "checked"),
(radio += `<label ${tt}><input ${selected} type="radio" name="${name}" data-val="${value}" data-cpid="${id}"><span>${item.text}</span></label>`);
}),
(radio += "</div>"),
`<div class="${cBase}comp ${cBase}ctrl ${cBase}rd-box" data-type="${type}" data-cpid="${id}">${title}${desc}${radio}</div>`
);
case "cb":
const name2 = info.name || new Date().getTime();
if (
((ctrlTt = info.ctrlTt || ""),
ctrlTt && (ctrlTt = `title="${ctrlTt}"`),
void 0 === info.value && info.radioList[0])
) {
const obj = info.radioList[0];
info.value = void 0 === obj.value ? obj.text : obj.value;
}
let checkbox = `<div class="${cBase}cb ${cBase}rd-arr" ${ctrlTt}>`;
return (
info.radioList.forEach((item, i) => {
void 0 === item.value && (info.radioList[i].value = item.text),
void 0 === item.text && (info.radioList[i].text = item.value);
const value = item.value;
let tt = item.tt || "";
tt && (tt = `title="${tt}"`);
let selected = "";
info.value.includes(value) && (selected = "checked"),
(checkbox += `<label ${tt}><input ${selected} type="checkbox" name="${name2}" data-val="${value}" data-cpid="${id}"><span>${item.text}</span></label>`);
}),
(checkbox += "</div>"),
`<div class="${cBase}comp ${cBase}ctrl ${cBase}cb-box" data-type="${type}" data-cpid="${id}">${title}${desc}${checkbox}</div>`
);
case "ta":
const style = `style="${
info.width ? "width:" + info.width + ";" : ""
}${info.height ? "height:" + info.height + ";" : ""}${
info.fontSize ? "font-size:" + info.fontSize + ";" : ""
}${info.fontFamily ? "font-family:" + info.fontFamily + ";" : ""}"`,
textarea = `<textarea class="${cBase}ta" ${style} data-cpid="${id}" placeholder="${
info.ph || ""
}" title="${info.ctrlTt || "拖动右下角可调节宽高"}"></textarea>`;
return `<div class="${cBase}comp ${cBase}ctrl ${cBase}ta-box" data-type="${type}" data-cpid="${id}">${title}${desc}${textarea}</div>`;
}
}
const param = cfg.param,
page = param.page,
cBase = param.classBase,
isMenu = 1 !== page.length;
let menu = `<div class="${cBase}menu">`,
pageHTML = `<div class="${cBase}page-box ll-scroll-style-1 ll-scroll-style-1-size-2">`;
page.forEach((curPage, index) => {
let pgid = curPage.id || index;
(pgid += ""), (cfg.allData[pgid] = {}), (cfg.baseData[pgid] = {});
let pageFlag = "";
if (
(cfg.hasSelectedPage ||
((void 0 === param.showPage || pgid === param.showPage + "") &&
((pageFlag = "curPage"), (cfg.hasSelectedPage = !0))),
(pageHTML += `<div class="${cBase}page ${pageFlag}" data-pgid="${pgid}">`),
curPage.components)
) {
let compIndex = 0;
if (isMenu || param.isShowMenu) {
let curMenu = curPage.components.find(
(item) => "menuTitle" === item.type
);
curMenu || (curMenu = { type: "menuTitle", value: pgid }),
(menu += getCompHTML({
info: curMenu,
active: pageFlag ? "active" : "",
}));
}
curPage.components.forEach((item) => {
const cpid = item.id || compIndex;
"menuTitle" !== item.type &&
(pageHTML += getCompHTML({ info: item, id: cpid })),
["title", "title2", "title3", "desc", "menuTitle"].includes(
item.type
) ||
((item.base = void 0 === item.base ? item.value : item.base),
(cfg.allData[pgid][cpid] = item.value),
(cfg.baseData[pgid][cpid] = item.base),
compIndex++);
});
}
pageHTML += "</div>";
}),
(pageHTML += "</div>"),
isMenu || param.isShowMenu ? (menu += "</div>") : (menu = "");
const resetBtn = param.isResetBtn
? `<button class="${cBase}reset-btn" title="${
param.resetTt || "重置所有设置为默认值"
}">重置</button>`
: "",
btnBox = `<div class="${cBase}btn-box">\n${resetBtn}\n<button class="${cBase}cancel-btn">取 消</button>\n<button class="${cBase}confirm-btn">确 认</button>\n</div>`;
return `<div class="${cBase}box ll-scroll-style-1 ll-scroll-style-1-size-3" data-version="${cfg.version}">\n${menu}\n${pageHTML}\n${btnBox}\n</div>`;
};
let param = cfg.param;
const baseParam = baseCfg.param,
controls = cfg.controls,
doms = cfg.doms;
function createEditEle({
id = baseParam.id,
box = baseParam.box,
classBase = baseParam.classBase,
w = baseParam.w,
h = baseParam.h,
contentH = baseParam.contentH,
bg = baseParam.bg,
color = baseParam.color,
fontSize = baseParam.fontSize,
fontFamily = baseParam.fontFamily,
zIndex = baseParam.zIndex,
resetTt = baseParam.resetTt,
isShowMenu = baseParam.isShowMenu,
isScrollStyle = baseParam.isScrollStyle,
isResetBtn = baseParam.isResetBtn,
isOnlyResetCurPage = baseParam.isOnlyResetCurPage,
showPage = baseParam.showPage,
page = [],
} = {}) {
(cfg.isEditing = baseCfg.isEditing),
(cfg.hasSelectedPage = baseCfg.hasSelectedPage),
(cfg.param = { ...baseParam }),
(param = cfg.param),
(param.id = id),
(param.box = box),
(param.classBase = classBase),
(param.w = w),
(param.h = h),
(param.contentH = contentH),
(param.bg = bg),
(param.color = color),
(param.fontSize = fontSize),
(param.fontFamily = fontFamily),
(param.zIndex = zIndex),
(param.resetTt = resetTt),
(param.isShowMenu = isShowMenu),
(param.isScrollStyle = isScrollStyle),
(param.isResetBtn = isResetBtn),
(param.isOnlyResetCurPage = isOnlyResetCurPage),
(param.showPage = showPage),
(param.page = page);
const html = editArea_html();
return (
box.querySelector(`#${param.classBase}${param.id}-css`) ||
(function addCss(cssText, box = document.body, id = "") {
const style = document.createElement("style");
return (
id && (style.id = id),
box.appendChild(style),
(style.innerHTML = cssText),
style
);
})(css(), box, param.classBase + param.id + "-css"),
(doms.wrap = (function createEle({
className = "",
id = "",
title = "",
css,
box = document.body,
type = "div",
} = {}) {
const ele = document.createElement(type);
return (
id && (ele.id = id),
className && (ele.className = className),
title && (ele.title = title),
css && (ele.style.cssText = css),
box.appendChild(ele),
ele
);
})({ className: id, id })),
(doms.wrap.innerHTML = html),
(function getDoms() {
const cBase = param.classBase;
(doms.box = doms.wrap.querySelector(`.${cBase}box`)),
(doms.cancel = doms.box.querySelector(`.${cBase}cancel-btn`)),
(doms.confirm = doms.box.querySelector(`.${cBase}confirm-btn`));
const isMenu = 1 !== param.page.length;
(isMenu || param.isShowMenu) &&
((doms.menu = doms.box.querySelector(`.${cBase}menu`)),
(doms.menus = [].slice.call(
doms.menu.querySelectorAll(`.${cBase}menu-item`)
)));
const pages = [].slice.call(doms.box.querySelectorAll(`.${cBase}page`));
(doms.page = []),
param.isResetBtn &&
(doms.reset = doms.box.querySelector(`.${cBase}reset-btn`));
pages.forEach((curPage, index) => {
cfg.hasSelectedPage ||
(curPage.classList.add("curPage"),
(isMenu || param.isShowMenu) &&
doms.menus[0].classList.add("active"),
(cfg.hasSelectedPage = !0));
const page = {},
pgid = curPage.dataset.pgid;
(page.pgid = curPage.pgid = pgid),
(page.controls = [].slice.call(
curPage.querySelectorAll(`.${cBase}ctrl`)
)),
(page.ele = curPage),
doms.page.push(page),
(isMenu || param.isShowMenu) &&
(doms.menus[index].settingsPage = curPage);
const ctrls = {};
(controls[pgid] = ctrls),
page.controls.forEach((item, i) => {
const cpid = item.dataset.cpid,
cType = item.dataset.type;
let dom;
(item.cpid = cpid),
"rd" === cType || "cb" === cType
? ((dom = [].slice.call(item.querySelectorAll("input"))),
(dom.compType = cType))
: "ta" === cType &&
((dom = item.querySelector("textarea")),
(dom.compType = cType),
(dom.value = cfg.allData[pgid][cpid])),
(ctrls[cpid] = dom);
});
});
})(),
(function bindEvents() {
function menuHandle(e) {
const dom = e.target,
cBase = param.classBase;
if (dom.classList.contains(`${cBase}menu-item`)) {
const old = doms.menu.querySelector(".active");
old.classList.remove("active"),
old.settingsPage.classList.remove("curPage"),
dom.classList.add("active"),
dom.settingsPage.classList.add("curPage");
}
}
function cancelEdit(e) {
const cBase = param.classBase;
(e.target.className !== `${cBase}wrap` &&
e.target.className !== `${cBase}cancel-btn`) ||
(showEditArea(!1), setCompValue(cfg.oldData));
}
function confirmEdit() {
const callback = cfg.callback,
data = getAllData();
if (((cfg.allData = data), callback.confirmBefore)) {
let result;
const func = callback.confirmBefore;
if (
(Array.isArray(func)
? func.curFn
? ((result = func[curFn](data)), (func.curFn = null))
: func.forEach((fn) => {
result = fn(data);
})
: (result = func(data)),
!1 === result)
)
return;
}
if ((showEditArea(!1), callback.finished)) {
const func = callback.finished;
Array.isArray(func)
? func.curFn
? (func[curFn](data), (func.curFn = null))
: func.forEach((fn) => {
fn(data);
})
: func(data);
}
}
function resetEdit() {
const callback = cfg.callback,
data = getAllData();
if (callback.resetBefore) {
let result;
const func = callback.resetBefore;
if (
(Array.isArray(func)
? func.curFn
? ((result = func[curFn](data)), (func.curFn = null))
: func.forEach((fn) => {
result = fn(data);
})
: (result = func(data)),
!1 === result)
)
return;
}
!(function resetEditData(data = null) {
if (param.isResetBtn)
if (param.isOnlyResetCurPage) {
data = data || getAllData();
const curMenu = doms.menu.querySelector(".active");
(data[curMenu.innerText] = cfg.baseData[curMenu.innerText]),
setCompValue(data);
} else setCompValue(cfg.baseData);
})(data);
}
doms.menu && doms.menu.addEventListener("click", menuHandle),
doms.wrap.addEventListener("click", cancelEdit),
doms.cancel.addEventListener("click", cancelEdit),
doms.confirm.addEventListener("click", confirmEdit),
doms.reset && doms.reset.addEventListener("click", resetEdit);
})(),
cfg
);
}
function getAllData() {
function getCompItem(pgid, cpid) {
if (!controls[pgid]) return;
const ctrl = controls[pgid][cpid];
if (ctrl) {
if (!Array.isArray(ctrl)) return ctrl.value;
if ("rd" === ctrl.compType) {
return ctrl.find((item) => item.checked).dataset.val;
}
if ("cb" === ctrl.compType) {
return ctrl
.filter((item) => item.checked)
.map((item) => item.dataset.val);
}
}
}
const data = {};
if (0 === arguments.length) {
for (const key in controls) {
const page = controls[key];
data[key] = {};
for (const key2 in page) data[key][key2] = getCompItem(key, key2);
}
return data;
}
if (1 === arguments.length) {
const ctrls = arguments[0];
for (const pgid in ctrls) {
data[pgid] = {};
controls[pgid].forEach((cpid) => {
data[pgid][cpid] = getCompItem(pgid, cpid);
});
}
return cfg.allData;
}
return getCompItem(arguments[0], arguments[1]);
}
function setCompValue() {
function setCompItem(pgid, cpid, value) {
if (!controls[pgid]) return;
const ctrl = controls[pgid][cpid];
if (ctrl)
if (Array.isArray(ctrl)) {
if ("rd" === ctrl.compType) {
const selected = ctrl.find((item) => item.checked);
selected && (selected.checked = !1);
const select = ctrl.find((item) => item.dataset.val === value + "");
select && (select.checked = !0);
} else if ("cb" === ctrl.compType) {
if (
(ctrl
.filter((item) => item.checked)
.forEach((item) => {
item.checked = !1;
}),
Array.isArray(value))
)
value.forEach((val) => {
const select = ctrl.find(
(item) => item.dataset.val === val + ""
);
select && (select.checked = !0);
});
else {
const select = ctrl.find(
(item) => item.dataset.val === value + ""
);
select && (select.checked = !0);
}
}
} else ctrl.value = value;
}
if (1 === arguments.length) {
const data = arguments[0];
for (const key in data) {
const pageData = data[key];
for (const key2 in pageData) {
setCompItem(key, key2, pageData[key2]);
}
}
} else {
setCompItem(arguments[0], arguments[1], arguments[2]);
}
}
function showEditArea(isShow = !0, callback = null) {
if (
isShow &&
((cfg.oldData = getAllData()), "function" == typeof callback)
) {
if (!1 === callback(cfg.oldData)) return;
}
(cfg.isEditing = isShow),
(doms.wrap.style.display = isShow ? "block" : "none"),
isShow &&
!doms.box.style.top &&
(doms.box.style.top =
window.innerHeight / 2 - doms.box.clientHeight / 2 + "px"),
callback && (cfg.callback = callback);
}
function getValue({
base,
key,
valType = "string",
isReSet = !0,
getValue = null,
setValue = null,
getVal = null,
setVal = null,
} = {}) {
getValue && (getVal = getValue), setValue && (setVal = setValue);
let val = getVal ? getVal(key) : localStorage.getItem(key);
return (
void 0 !== base &&
null == val &&
((val = base),
isReSet &&
("string" != typeof base && (base = JSON.stringify(base)),
setVal ? setVal(key, base) : localStorage.setItem(key, base))),
(valType = valType.toLowerCase()),
"string" == typeof val
? "string" === valType
? val
: "boolean" === valType || "number" === valType
? JSON.parse(val)
: "object" === valType
? val
? JSON.parse(val)
: {}
: "array" === valType
? val
? JSON.parse(val)
: []
: val
: val
);
}
function getData(settings, getVal = null, setVal = null) {
(getVal = getVal || localStorage.getItem),
(setVal = setVal || localStorage.setItem);
for (const valName in settings) {
const setting = settings[valName];
setting.value = getValue({
base: setting.base,
key: setting.key,
valType: setting.valType,
getVal,
setVal,
});
}
return settings;
}
function setValue({
value,
base,
key,
verification = null,
getValue = null,
setValue = null,
getVal = null,
setVal = null,
} = {}) {
getValue && (getVal = getValue || localStorage.getItem),
setValue && (setVal = setValue || localStorage.getItem);
let newVal = value,
oldVal = getVal ? getVal(key) : localStorage.getItem(key);
return (
void 0 !== base &&
null == oldVal &&
((oldVal = base),
"string" != typeof base && (base = JSON.stringify(base)),
setVal ? setVal(key, base) : localStorage.setItem(key, base)),
null !== newVal &&
("function" != typeof verification ||
((newVal = verification(newVal, oldVal, base)), null !== newVal)) &&
newVal !== oldVal &&
("string" != typeof newVal && (newVal = JSON.stringify(newVal)),
setVal ? setVal(key, newVal) : localStorage.setItem(key, newVal),
!0)
);
}
function verify_notNull(newVal, oldVal) {
return newVal || oldVal;
}
function verify_time1(newVal, oldVal, base) {
const arr = newVal.trim().split(/:|:/);
if (2 === arr.length && 2 === arr[0].length && 2 === arr[1].length) {
const a = +arr[0],
b = +arr[1];
if (a >= 0 && a <= 24 && b >= 0 && b <= 59) return newVal;
}
return oldVal;
}
function getNumVerifyFn(min, max, rangeLimit = [1, 1]) {
return (newVal, oldVal, base) => {
if (!(newVal = +newVal) && 0 !== !newVal) return oldVal;
if (!1 !== min && !1 !== max) {
if (rangeLimit[0] && newVal >= min) {
if (rangeLimit[1] && newVal <= max) return newVal;
if (!rangeLimit[1] && newVal < max) return newVal;
}
if (!rangeLimit[0] && newVal > min) {
if (rangeLimit[1] && newVal <= max) return newVal;
if (!rangeLimit[1] && newVal < max) return newVal;
}
} else {
if (!1 === min) {
if (rangeLimit[1] && newVal <= max) return newVal;
if (!rangeLimit[1] && newVal < max) return newVal;
}
if (!1 === max) {
if (rangeLimit[0] && newVal >= min) return newVal;
if (!rangeLimit[0] && newVal > min) return newVal;
}
}
return oldVal;
};
}
function toPageObj({ settings, param = {}, otherPageName = "无分类" } = {}) {
param = { ...param };
const pageArr = [],
menuList = [];
let isOtherType = !1;
for (let key in settings) {
const item = settings[key];
item.type
? menuList.includes(item.type) || menuList.push(item.type)
: isOtherType || (isOtherType = !0);
}
return (
isOtherType && menuList.push(otherPageName),
menuList.forEach((menuTt) => {
const components = [],
page = { id: menuTt, components },
arr = [];
for (let key in settings) {
const item = settings[key];
menuTt === otherPageName
? item.type || arr.push(item)
: item.type === menuTt && arr.push(item);
}
arr.forEach((item) => {
let desc = item.desc || item.txt || "";
desc && (desc = desc.replaceAll("\n", "<br>").trim());
let comp,
base = item.base;
if (
(Array.isArray(base) && (base = base.join(", ")), item.groupTitle1)
) {
const comp = {
id: item.key + "-gTt1",
type: "title",
value: item.groupTitle1,
};
components.push(comp);
}
if (item.groupTitle2) {
const comp = {
id: item.key + "-gTt2",
type: "title2",
value: item.groupTitle2,
};
components.push(comp);
}
if (item.groupTitle3) {
const comp = {
id: item.key + "-gTt3",
type: "title3",
value: item.groupTitle3,
};
components.push(comp);
}
if (item.groupDesc) {
const comp = {
id: item.key + "-gDesc",
type: "desc",
value: item.groupDesc,
};
components.push(comp);
}
if (
(["menuTitle", "title", "desc", "title2", "title3"].includes(
item.compType
)
? ((comp = { ...item }),
(comp.type = comp.compType),
(comp.desc = desc))
: (comp = {
id: item.key,
type: item.compType,
tt: item.tt || "",
title: item.title || "",
desc,
descTt: item.descTt || "",
name: item.key,
value: item.value,
base: item.base,
}),
"textarea" === comp.type)
)
(comp.ph = base),
(comp.width = item.compW),
(comp.height = item.compH),
(comp.ctrlTt = "默认: " + base);
else if ("radio" === comp.type || "checkbox" === comp.type) {
let str = "默认: ";
if ("checkbox" === comp.type) {
let arr = item.base;
Array.isArray(arr) || (arr = arr.split(/,|,/)),
arr.forEach((val, i) => {
0 !== i && (str += ", "), (val = val.trim());
let valTxt = item.valueText[val];
void 0 === valTxt && (valTxt = val), (str += valTxt);
});
} else {
let val = item.valueText[item.base];
void 0 === val && (val = item.base), (str += val);
}
comp.ctrlTt = str;
}
if (item.valueText) {
comp.radioList = [];
for (let key in item.valueText) {
const rd = { text: item.valueText[key], value: key };
comp.radioList.push(rd);
}
}
components.push(comp);
}),
pageArr.push(page);
}),
(param.page = pageArr),
param
);
}
function showSettings() {
const settings = info.settings;
info.settingsArea = (function createEdit({
settings,
param = {},
oldEditCfg,
updateDataFn,
isNewEdit = !0,
isSyncOtherPage = !0,
otherPageName = "无分类",
} = {}) {
let oldSettings, curSettings;
updateDataFn &&
isSyncOtherPage &&
((oldSettings = JSON.stringify(settings)),
(settings = updateDataFn() || settings),
(curSettings = JSON.stringify(settings)));
const editInfo = { settings, param, otherPageName };
if (oldEditCfg) {
if (isNewEdit)
return (
oldEditCfg.doms.wrap.remove(), createEditEle(toPageObj(editInfo))
);
isSyncOtherPage &&
updateDataFn &&
oldSettings !== curSettings &&
(oldEditCfg.doms.wrap.remove(),
(oldEditCfg = createEditEle(toPageObj(editInfo)))),
isSyncOtherPage &&
!updateDataFn &&
(oldEditCfg.doms.wrap.remove(),
(oldEditCfg = createEditEle(toPageObj(editInfo))));
} else oldEditCfg = createEditEle(toPageObj(editInfo));
return oldEditCfg;
})({
settings,
param: {
bg: "rgba(0, 0, 0, 0)",
resetTt: "重置当前页的所有设置为默认值",
isOnlyResetCurPage: !0,
},
oldEditCfg: info.settingsArea,
updateDataFn: () => getData(settings, GM_getValue, GM_setValue),
});
showEditArea(!0, {
resetBefore: () => confirm("是否重置当前页的所有设置为默认值?"),
confirmBefore: () => {},
finished: (data) => {
console.log(data);
if (
!(function isValueChange() {
const curData = getAllData();
return JSON.stringify(curData) !== JSON.stringify(cfg.oldData);
})()
)
return;
for (const pageName in data) {
const page = data[pageName];
for (const key in page) {
const value = page[key];
let verifyFn;
const flag = key.replace(info.keyBase, ""),
item = settings[flag];
switch (key) {
case settings.startTime.key:
case settings.stopTime.key:
verifyFn = (newVal, oldVal, base) =>
0 == +newVal ? newVal : verify_time1(newVal, oldVal);
break;
case settings.invert.key:
verifyFn = getNumVerifyFn(0, 1);
break;
case settings.brightness.key:
case settings.contrast.key:
verifyFn = getNumVerifyFn(0, !1);
break;
case settings.grayscale.key:
verifyFn = getNumVerifyFn(0, 1);
break;
case settings.hueRotate.key:
verifyFn = getNumVerifyFn(0, 360);
break;
case settings.saturate.key:
verifyFn = getNumVerifyFn(0, !1);
break;
case settings.sepia.key:
verifyFn = getNumVerifyFn(0, 1);
break;
case settings.website.key:
case settings.notInvertNodes.key:
verifyFn = verify_notNull;
}
if (!item) return void console.log("设置的数据对应的对象获取失败");
setValue({
value,
base: item.base,
key,
verification: verifyFn,
getValue: GM_getValue,
setValue: GM_setValue,
});
}
}
getData(settings, GM_getValue, GM_setValue);
if (!verifyWebsite())
return (
setDarkMode(!1), void (info.timer && clearInterval(info.timer))
);
info.isDarkMode && (setDarkMode(!1), setDarkMode(!0)),
setStartStopTimer();
},
});
}
const settings = info.settings;
!(function main() {
getData(settings, GM_getValue, GM_setValue);
const f = verifyWebsite();
!(function registerMenu(f) {
f &&
GM_registerMenuCommand("切换深色模式", () => {
setDarkMode(!info.isDarkMode);
}),
GM_registerMenuCommand("设置", () => {
showSettings();
});
})(f),
f && setStartStopTimer();
})();
})();