// ==UserScript== // @name 深色模式 // @namespace https://greasyfork.org/zh-CN/users/1196880-ling2ling4 // @version 1.3.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 // @grant GM_notification // @noframes // @compatible chrome // @compatible edge // @compatible firefox // @downloadURL https://update.greasyfork.icu/scripts/503000/%E6%B7%B1%E8%89%B2%E6%A8%A1%E5%BC%8F.user.js // @updateURL https://update.greasyfork.icu/scripts/503000/%E6%B7%B1%E8%89%B2%E6%A8%A1%E5%BC%8F.meta.js // ==/UserScript== (() => { "use strict"; 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; }; } const keyBase = "ll_pageDarkMode_", info = { keyBase, settingsArea: null, isDarkMode: !1, isCanRun: !0, timer: null, interval: 5e3, cssDom: null, btnHoverTxt: "点击切换深色模式", otherSettings: { oldDarkMode: { value: !1, base: !1, key: keyBase + "oldDarkMode", valType: "boolean", }, }, settings: { btnPosition: { value: !1, base: !1, key: keyBase + "btnPosition", groupTitle3: "按钮设置", desc: "模式切换按钮的位置", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "左下", false: "右下" }, }, isHiddenBtn: { value: !1, base: !1, key: keyBase + "isHiddenBtn", desc: "是否隐藏按钮, 隐藏后鼠标移入时会重新显示", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "隐藏", false: "显示" }, }, btnSize: { value: "30", base: "30", key: keyBase + "btnSize", desc: "按钮的大小", type: "基础设置", valType: "number", compType: "textarea", verify: getNumVerifyFn(20, 60), }, isAutoStartStop: { value: !0, base: !0, key: keyBase + "isAutoStartStop", groupTitle3: "定时开关", 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", verify: (newVal, oldVal, base) => 0 == +newVal ? newVal : verify_time1(newVal, oldVal), }, stopTime: { value: "0", base: "0", key: keyBase + "stopTime", valType: "string", type: "基础设置", desc: "深色模式的自动关闭时间, 0表示关闭, 按照24小时制书写", compType: "textarea", verify: (newVal, oldVal, base) => 0 == +newVal ? newVal : verify_time1(newVal, oldVal), }, startStopWay: { value: !0, base: !0, key: keyBase + "startStopWay", desc: "定时开关深色模式的方式", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "仅在设定时刻进行开关", false: "根据设定时间段任意时刻都可开关", }, }, isShowTips: { value: !0, base: !0, key: keyBase + "isShowTips", desc: "定时开关时是否进行弹窗提示", type: "基础设置", valType: "boolean", compType: "radio", valueText: { true: "弹窗提示", false: "关闭弹窗" }, }, onlyColor: { value: !1, base: !1, key: keyBase + "onlyColor", desc: "深色模式下是否仅调整颜色而不使页面变成深色 (此时'颜色反转'设置将失效)", type: "颜色设置", valType: "boolean", compType: "radio", valueText: { true: "自定义颜色", false: "深色+自定义颜色" }, }, invert: { value: 1, base: 1, key: keyBase + "invert", valType: "number", type: "颜色设置", title: "颜色反转", desc: "颜色反转的程度, 深色效果主要与该设置相关. 默认1, 浏览器默认0, 取值范围0-1", compType: "textarea", verify: getNumVerifyFn(0, 1), }, brightness: { value: 0.9, base: 0.9, key: keyBase + "brightness", valType: "number", type: "颜色设置", title: "亮度", desc: "亮度的大小. 默认0.9, 浏览器默认1, 取值范围0-∞", compType: "textarea", verify: getNumVerifyFn(0, !1), }, contrast: { value: 1, base: 1, key: keyBase + "contrast", valType: "number", type: "颜色设置", title: "对比度", desc: "对比度的强弱. 默认1, 取值范围0-∞", compType: "textarea", verify: getNumVerifyFn(0, !1), }, grayscale: { value: 0, base: 0, key: keyBase + "grayscale", valType: "number", type: "颜色设置", title: "灰度", desc: "灰度的程度. 默认0, 取值范围0-1", compType: "textarea", verify: getNumVerifyFn(0, 1), }, hueRotate: { value: 0, base: 0, key: keyBase + "hueRotate", valType: "number", type: "颜色设置", title: "色调", desc: "色调的旋转变化. 默认0, 取值范围0-360", compType: "textarea", verify: getNumVerifyFn(0, 360), }, saturate: { value: 1, base: 1, key: keyBase + "saturate", valType: "number", type: "颜色设置", title: "饱和度", desc: "饱和度的高低. 默认1, 取值范围0-∞", compType: "textarea", verify: getNumVerifyFn(0, !1), }, sepia: { value: 0.2, base: 0.2, key: keyBase + "sepia", valType: "number", type: "颜色设置", title: "深褐色", desc: "深褐色的程度. 默认0.2, 浏览器默认0, 取值范围0-1", compType: "textarea", verify: getNumVerifyFn(0, 1), }, 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", title: "自动恢复显示模式的方式", desc: "左选项: 可使同一时间段内打开的每个页面都是相同显示模式\n右选项: 可使同一个页面打开后是上一次该页面的显示模式", type: "其他设置", valType: "boolean", compType: "radio", valueText: { true: "恢复上一次使用的显示模式", false: "恢复当前网页上一次的显示模式", }, }, website: { value: "*www.baidu.com*\n*www.bilibili.com*\n*message.bilibili.com*\n*space.bilibili.com*\n*weibo.com*\n*www.zhihu.com*\n*www.douyin.com*", base: "*www.baidu.com*\n*www.bilibili.com*\n*message.bilibili.com*\n*space.bilibili.com*\n*weibo.com*\n*www.zhihu.com*\n*www.douyin.com*", key: keyBase + "website", valType: "string", type: "其他设置", title: "应用的网站", desc: "以下网站可启用深色模式, 支持*通配符, 多个网站请换行书写, 仅书写*表示所有网站都可启用\n【示例】*www.bilibili.com* 可匹配B站", compType: "textarea", compH: "110px", }, onlyColorWebsite: { value: "", base: "", key: keyBase + "onlyColorWebsite", valType: "string", type: "其他设置", title: "不变为深色的网站", desc: '以下网站即使启用深色模式后也不会变为深色, 而是采用"自定义颜色"模式, 支持*通配符, 多个网站请换行书写', compType: "textarea", compH: "110px", }, noneInvertNodes: { value: '// B站\n.h .h-inner, .h-inner .avatar-container, bili-user-profile, .bili-im .avatar, .owner .to-top, #bilibili-player [role="comment"],\n// 百度\n#content_left h3.t, .cr-content [class*="opr-toplist"], .cr-content [class*="tag-common"]', base: '// B站\n.h .h-inner, .h-inner .avatar-container, bili-user-profile, .bili-im .avatar, .owner .to-top, #bilibili-player [role="comment"],\n// 百度\n#content_left h3.t, .cr-content [class*="opr-toplist"], .cr-content [class*="tag-common"]', key: keyBase + "noneInvertNodes", valType: "string", type: "其他设置", title: "不反转的元素", desc: "不进行颜色反转的元素, 每项用 , 分隔, 可书写css选择器. 以//开头表示行注释\n【可选】\nh1, h2, h3, h4, p, span, ul, li, i, svg, a, img, input, textarea, button, select, option, label, audio, video, ....", compType: "textarea", compH: "110px", verify: (newVal) => { "," === (newVal = newVal.trim().replaceAll(",", ","))[ newVal.length - 1 ] && (newVal = newVal.slice(0, -1)); return newVal .split("\n") .map((item, i) => { const t = item; return ( (item = item.trim()), 0 === i ? "/" === item[0] && "/" === item[1] ? item : t : "/" === item[0] && "/" === item[1] ? ",\n" + item : "\n" + t ); }) .join("") .replaceAll(", ,", ",") .replaceAll(",,", ","); }, }, }, }; function getCssHtml(isDark) { const settings = info.settings, r = parseInt(settings.btnSize.value / 5), btnCss = `#${info.keyBase}btn{\nbackground:#ffffff;padding:${ r - 2 }px;border-radius:${r}px;position:fixed;${ settings.btnPosition.value ? "left" : "right" }:-${ settings.btnSize.value / 2 }px;bottom:20px;z-index:1000;transition:ease 0.3s all,ease 0.5s 2s opacity;cursor:pointer;box-sizing:border-box;\n${ settings.isHiddenBtn.value ? "opacity:0" : "" }}\n#${info.keyBase}btn.dark{background:#ababab}\n#${ info.keyBase }btn:hover {${ settings.btnPosition.value ? "left:0" : "right:0" };bottom:20px;\n${ settings.isHiddenBtn.value ? "transition:ease 0.3s opacity;opacity:1" : "" }\n}\n #${info.keyBase}btn svg{display:block;fill:#addeee}\n #${ info.keyBase }btn.dark svg{fill:#ffffff}`; if (!isDark) return btnCss; const onlyColorflag = verifyWebsite(settings.onlyColorWebsite.value) || settings.onlyColor.value, invertText = onlyColorflag ? "" : `invert(${settings.invert.value})`, selectorArr = settings.noneInvertNodes.value .split("\n") .filter((item) => "/" !== (item = item.trim())[0] && "/" !== item[1]), nonoFilter = onlyColorflag ? "" : 'img,video,input,iframe,canvas,object,svg image,\n[style*="background:url"],\n[style*="background: url"],\n[style*="background-image:url"],\n[style*="background-image: url"],\n[background]{filter:invert(1)}', otherCss = onlyColorflag ? "" : `${selectorArr.join("")}{filter:invert(1)}`; return `html {\nbackground-color:#fff;\nfilter:${invertText} 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${nonoFilter}${otherCss}${btnCss}`; } function matchUrlWithWildcard(url, pattern) { return new RegExp("^" + pattern.replace(/\*/g, ".*") + "$").test(url); } function setValue_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 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 setDarkMode(isDark = !0, modeTxt = "") { if ( (((isDark) => { let dom = info.cssDom, isAdd = !1; if (!dom) { const id = info.keyBase + "css"; (dom = document.head.querySelector("#" + id)), dom || ((dom = document.createElement("style")), (dom.id = id), (info.cssDom = dom), (isAdd = !0)); } dom.isDark !== isDark && ((dom.innerHTML = getCssHtml(isDark)), (info.isDarkMode = isDark), (dom.isDark = isDark), isAdd && document.head.appendChild(dom)); })(isDark), document.body) ) document.body.appendChild(info.cssDom); else { const bodyObserver = new MutationObserver(() => { document.body && (bodyObserver.disconnect(), document.body.appendChild(info.cssDom)); }); bodyObserver.observe(document, { childList: !0, subtree: !0 }); } let logText, txt1; txt1 = isDark ? "开启" : "关闭"; const settings = info.settings; modeTxt || (settings.onlyColor.value && (modeTxt = `'${settings.onlyColor.valueText.true}'模式`), verifyWebsite(settings.onlyColorWebsite.value) && (modeTxt = `'${settings.onlyColor.valueText.true}'模式`)), (logText = `${txt1}${(modeTxt = modeTxt || "深色模式")}`), console.log(logText), setValue_setValue({ value: isDark, base: info.otherSettings.oldDarkMode.base, key: info.otherSettings.oldDarkMode.key, getValue: GM_getValue, setValue: GM_setValue, }), 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 && (settings.isShowTips.value && GM_notification({ title: "深色模式", text: `定时${f ? "开启" : "关闭"}深色模式`, timeout: 3e3, }), setDarkMode(f)); }; autoStartStop(), info.timer && clearInterval(info.timer), (info.timer = setInterval(autoStartStop, info.interval)); } function verifyWebsite(websiteText, url) { if (!websiteText) return !1; url = url || location.href; return websiteText .trim() .split("\n") .some((item) => matchUrlWithWildcard(url, item)); } function updateShow() { getData(info.settings, GM_getValue, GM_setValue), info.cssDom ? (info.cssDom.innerHTML = getCssHtml(info.isDarkMode)) : info.isDarkMode ? (setDarkMode(!1), setDarkMode(!0)) : setDarkMode(!1), createDarkModeBtn(), setStartStopTimer(); } function createDarkModeBtn() { const settings = info.settings; if (info.modeBtn) { const btn = info.modeBtn; return ( (btn.style.width = settings.btnSize.value + "px"), (btn.style.height = settings.btnSize.value + "px"), void (info.isDarkMode ? btn.classList.add("dark") : btn.classList.remove("dark")) ); } const html = `