// ==UserScript== // @name 深色模式 // @namespace https://greasyfork.org/zh-CN/users/1196880-ling2ling4 // @version 1.1.1 // @author Ling2Ling4 // @description 设置页面为深色模式, 可定时开关 // @license AGPL-3.0-or-later // @icon  // @match *://*/* // @run-at document-start // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @noframes // @compatible chrome // @compatible edge // @compatible firefox // @downloadURL none // ==/UserScript== (() => { "use strict"; const keyBase = "ll_pageDarkMode_", info = { keyBase, settingsArea: null, isDarkMode: !1, isCanRun: !0, timer: null, interval: 5e3, otherSettings: { oldDarkMode: { value: !1, base: !1, key: keyBase + "oldDarkMode", valType: "boolean", }, }, settings: { isAutoStartStop: { value: !0, base: !0, key: keyBase + "isAutoStartStop", desc: "是否开启定时开关功能", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "开启", false: "关闭" }, }, 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", }, startStopWay: { value: !0, base: !0, key: keyBase + "startStopWay", desc: "定时开关深色模式的方式", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "仅在设定时刻进行开关", false: "根据设定时间段任意时刻都可开关", }, }, 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", }, autoDarkMode: { value: !0, base: !0, key: keyBase + "autoDarkMode", desc: "是否自动恢复页面的深色模式 (刷新页面后将自动恢复之前的显示模式)", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "自动恢复", false: "手动开关" }, groupTitle3: "自动恢复", }, autoDarkModeWay: { value: !0, base: !0, key: keyBase + "autoDarkModeWay", desc: "自动恢复页面显示模式的方式\n左选项: 可确保同一时间段内打开的每个页面都是同一个显示模式\n右选项: 可确保同一个页面打开后是上一次该页面的显示模式", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "恢复上一次使用的显示模式", false: "恢复当前网页上一次的显示模式", }, }, website: { value: "*", base: "*", key: keyBase + "website", valType: "string", type: "其他设置", title: "应用的网站", desc: "以下网站可启用深色模式, 多个网站请换行书写, *表示所有网站都可启用\n【示例】*.bilibili.com/* 可匹配B站\n", compType: "textarea", compH: "90px", }, notInvertNodes: { value: "img, video, .h-inner, .avatar-container", base: "img, video, .h-inner, .avatar-container", 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 setValue({ value, base, key, verification = null, getValue = null, setValue = null, getVal = null, setVal = null, } = {}) { getValue && (getVal = getValue), setValue && (setVal = setValue); 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 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 ? "开启" : "关闭") + "深色模式"), setValue({ value: isDark, base: info.otherSettings.oldDarkMode.base, key: info.otherSettings.oldDarkMode.key, getValue: GM_getValue, setValue: GM_setValue, }), setValue({ value: isDark, base: info.otherSettings.oldDarkMode.base, key: info.otherSettings.oldDarkMode.key, }); } function setStartStopTimer() { const settings = info.settings; if (!settings.isAutoStartStop.value) return; const 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 (settings.startStopWay.value) return curT === startT || (curT !== stopT && -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 ? `
${info.title}
` : ""), (desc = info.desc ? `
${info.desc}
` : "")), type) ) { case "mtt": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "tt": case "tt2": case "tt3": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "ds": return ( (info.value = info.value || ""), info.value ? `
${info.value}
` : "" ); case "rd": const name = info.name || info.id + new Date().getTime(); (ctrlTt = info.ctrlTt || ""), ctrlTt && (ctrlTt = `title="${ctrlTt}"`); let radio = `
`; 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 += ``); }), (radio += "
"), `
${title}${desc}${radio}
` ); 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 = `
`; 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 += ``); }), (checkbox += "
"), `
${title}${desc}${checkbox}
` ); 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 = ``; return `
${title}${desc}${textarea}
`; } } const param = cfg.param, page = param.page, cBase = param.classBase, isMenu = 1 !== page.length; let menu = `
`, pageHTML = `
`; 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 += `
`), 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 += "
"; }), (pageHTML += "
"), isMenu || param.isShowMenu ? (menu += "
") : (menu = ""); const resetBtn = param.isResetBtn ? `` : "", btnBox = `
\n${resetBtn}\n\n\n
`; return `
\n${menu}\n${pageHTML}\n${btnBox}\n
`; }; 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), (box = box || document.body), (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 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", "
").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.isAutoStartStop.key: break; case settings.startTime.key: case settings.stopTime.key: verifyFn = (newVal, oldVal, base) => 0 == +newVal ? newVal : verify_time1(newVal, oldVal); break; case settings.startStopWay.key: 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.autoDarkMode.key: case settings.autoDarkModeWay.key: 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), (function getOtherData() { const settings = info.otherSettings, autoWay = info.settings.autoDarkModeWay.value; for (const valName in settings) { const setting = settings[valName]; setting.value = getValue({ base: setting.base, key: setting.key, valType: setting.valType, getVal: autoWay ? GM_getValue : null, setVal: autoWay ? GM_setValue : null, }); } })(); const f = verifyWebsite(); !(function registerMenu(f) { f && GM_registerMenuCommand("切换深色模式", () => { setDarkMode(!info.isDarkMode); }), GM_registerMenuCommand("设置", () => { showSettings(); }); })(f), f && (settings.autoDarkMode.value && info.otherSettings.oldDarkMode.value && setDarkMode(!0), setStartStopTimer()); })(); })();