// ==UserScript== // @name ✨夜间模式助手 // @namespace https://github.com/521-baby/-Dark-Mode-Helper-Auto-Switch-Theme // @version 1.0.0 // @description ✨ 定时自动切换夜间模式 | 4种主题切换(夜间1,暗色2,护眼3,白天4)|自定义当前模式|自定义昼夜时间| 网站白名单 | 护眼模式 🛡️👁️ // @author 伏黑甚而 // @license MIT // @match *://*/* // @require https://unpkg.com/darkrule@1.0.4/dist/rule.min.js // @require https://unpkg.com/sweetalert2@10.16.6/dist/sweetalert2.min.js // @resource swalStyle https://unpkg.com/sweetalert2@10.16.6/dist/sweetalert2.min.css // @run-at document-start // @grant GM_getValue // @grant GM_setValue // @grant GM_registerMenuCommand // @grant GM_getResourceText // @icon  // @downloadURL https://update.greasyfork.icu/scripts/536592/%E2%9C%A8%E5%A4%9C%E9%97%B4%E6%A8%A1%E5%BC%8F%E5%8A%A9%E6%89%8B.user.js // @updateURL https://update.greasyfork.icu/scripts/536592/%E2%9C%A8%E5%A4%9C%E9%97%B4%E6%A8%A1%E5%BC%8F%E5%8A%A9%E6%89%8B.meta.js // ==/UserScript== ;(function () { 'use strict'; let util = { getValue(name) { return GM_getValue(name); }, setValue(name, value) { GM_setValue(name, value); }, addStyle(id, tag, css) { tag = tag || 'style'; let doc = document, styleDom = doc.getElementById(id); if (styleDom) return; let style = doc.createElement(tag); style.rel = 'stylesheet'; style.id = id; tag === 'style' ? style.innerHTML = css : style.href = css; doc.head.appendChild(style); }, hover(ele, fn1, fn2) { ele.onmouseenter = function () { //移入事件 fn1.call(ele); }; ele.onmouseleave = function () { //移出事件 fn2.call(ele); }; }, addThemeColor(color) { let doc = document, meta = doc.getElementsByName('theme-color')[0]; if (meta) return meta.setAttribute('content', color); let metaEle = doc.createElement('meta'); metaEle.name = 'theme-color'; metaEle.content = color; doc.head.appendChild(metaEle); }, getThemeColor() { let meta = document.getElementsByName('theme-color')[0]; if (meta) { return meta.content; } return '#ffffff'; }, removeElementById(eleId) { let ele = document.getElementById(eleId); ele && ele.parentNode.removeChild(ele); }, hasElementById(eleId) { return document.getElementById(eleId); }, filter: '-webkit-filter: url(#dark-mode-filter) !important; filter: url(#dark-mode-filter) !important;', reverseFilter: '-webkit-filter: url(#dark-mode-reverse-filter) !important; filter: url(#dark-mode-reverse-filter) !important;', firefoxFilter: `filter: url('data:image/svg+xml;utf8,#dark-mode-filter') !important;`, firefoxReverseFilter: `filter: url('data:image/svg+xml;utf8,#dark-mode-reverse-filter') !important;`, noneFilter: '-webkit-filter: none !important; filter: none !important;', }; // ===== 模式数据结构与切换逻辑 ===== const MODES = [ { key: 'night', name: '夜间', icon: 'moon', id: 1 }, { key: 'dark', name: '暗色', icon: 'eye', id: 2 }, { key: 'eye', name: '护眼', icon: 'palette', id: 3 }, { key: 'light', name: '白天', icon: 'sun', id: 4 } ]; function getEnabledModes() { const enableEye = util.getValue('bright_dark_mode'); const enableDark = util.getValue('menu_forcedToEnable'); // 从设置中获取暗色模式开关 let modes = [MODES[0]]; // 夜间 if (enableEye) modes.push(MODES[1]); if (enableDark) modes.push(MODES[2]); modes.push(MODES[3]); // 白天 return modes; } function getCurrentModeIndex() { const modes = getEnabledModes(); let current = util.getValue('menu_darkModeType') || 1; // 自动切换模式 const autoSwitch = util.getValue('menu_autoSwitch'); if (autoSwitch && autoSwitch.includes('|')) { const [dayMode, nightMode] = autoSwitch.split('|').map(Number); const [dayStart, nightStart] = (util.getValue('menu_customTime') || '6:00|18:00').split('|'); // 解析时间 function parseTime(str) { const [h, m] = str.split(':').map(Number); return h * 60 + (m || 0); } const now = new Date(); const nowMinutes = now.getHours() * 60 + now.getMinutes(); const dayStartMinutes = parseTime(dayStart); const nightStartMinutes = parseTime(nightStart); let isDay; if (dayStartMinutes < nightStartMinutes) { isDay = nowMinutes >= dayStartMinutes && nowMinutes < nightStartMinutes; } else { // 例如 18:00|6:00 这种跨天 isDay = nowMinutes >= dayStartMinutes || nowMinutes < nightStartMinutes; } current = isDay ? dayMode : nightMode; } let idx = modes.findIndex(m => m.id == current); if (idx === -1) idx = 0; return idx; } function setCurrentModeByIndex(idx) { const modes = getEnabledModes(); util.setValue('menu_darkModeType', modes[idx].id); } let main = { /** * 配置默认值 */ initValue() { let value = [{ name: 'dark_mode', value: 'light' }, { name: 'button_position', value: 'left' }, { name: 'button_size', value: 32 }, { name: 'exclude_list', value: ['youku.com', 'v.youku.com', 'www.douyu.com', 'www.iqiyi.com', 'vip.iqiyi.com', 'mail.qq.com', 'live.kuaishou.com'] }, { name: 'origin_theme_color', value: '#ffffff' }, { name: 'menu_autoRecognition', value: true }, { name: 'menu_forcedToEnable', value: false }, { name: 'menu_darkModeType', value: 2 }, { name: 'menu_customTime', value: '6:00|18:00' }, { name: 'menu_autoSwitch', value: '' }, { name: 'button_hidden', value: false }, { name: 'bright_dark_mode', value: true }]; value.forEach((v) => { util.getValue(v.name) === undefined && util.setValue(v.name, v.value); }); }, addExtraStyle() { try { return darkModeRule; } catch (e) { return ''; } }, createDarkFilter() { if (util.hasElementById('dark-mode-svg')) return; let svgDom = ''; let div = document.createElementNS('http://www.w3.org/1999/xhtml', 'div'); div.innerHTML = svgDom; let frag = document.createDocumentFragment(); while (div.firstChild) frag.appendChild(div.firstChild); document.head.appendChild(frag); }, createDarkStyle() { util.addStyle('dark-mode-style', 'style', ` @media screen { html { ${this.isFirefox() ? util.firefoxFilter : util.filter} scrollbar-color: #454a4d #202324; } /* Default Reverse rule */ img, video, iframe, canvas, :not(object):not(body) > embed, object, svg image, [style*="background:url"], [style*="background-image:url"], [style*="background: url"], [style*="background-image: url"], [background], twitterwidget, .sr-reader, .no-dark-mode, .sr-backdrop { ${this.isFirefox() ? util.firefoxReverseFilter : util.reverseFilter} } [style*="background:url"] *, [style*="background-image:url"] *, [style*="background: url"] *, [style*="background-image: url"] *, input, [background] *, img[src^="https://s0.wp.com/latex.php"], twitterwidget .NaturalImage-image { ${util.noneFilter} } /* Text contrast */ html { text-shadow: 0 0 0 !important; } /* Full screen */ .no-filter, :-webkit-full-screen, :-webkit-full-screen *, :-moz-full-screen, :-moz-full-screen *, :fullscreen, :fullscreen * { ${util.noneFilter} } ::-webkit-scrollbar { background-color: #202324; color: #aba499; } ::-webkit-scrollbar-thumb { background-color: #454a4d; } ::-webkit-scrollbar-thumb:hover { background-color: #575e62; } ::-webkit-scrollbar-thumb:active { background-color: #484e51; } ::-webkit-scrollbar-corner { background-color: #181a1b; } /* Page background */ html { background: #fff !important; } ${this.addExtraStyle()} } @media print { .no-print { display: none !important; } }`); }, setThemeColor() { util.setValue('origin_theme_color', util.getThemeColor()); }, enableDarkMode() { if (this.isFullScreen()) return; !this.isFirefox() && this.createDarkFilter(); this.createDarkStyle(); util.addThemeColor('#131313'); }, disableDarkMode() { util.removeElementById('dark-mode-svg'); util.removeElementById('dark-mode-style'); util.addThemeColor(util.getValue('origin_theme_color')); }, // 新增:根据当前模式动态切换样式 applyCurrentModeStyle() { const modes = getEnabledModes(); const idx = getCurrentModeIndex(); const mode = modes[idx]; // 清理所有样式 util.removeElementById('dark-mode-style'); util.removeElementById('dark-mode-svg'); // 夜间 if (mode.key === 'night') { this.enableDarkMode(); } else if (mode.key === 'light') { // 白天模式 = 亮度滤镜 let style_40 = (util.getValue('menu_customMode4') || '100|100').split('|'); let isDay = (new Date().getHours() >= 6 && new Date().getHours() < 18); let brightness = isDay ? style_40[0] : style_40[1]; util.addStyle('dark-mode-style', 'style', `html { filter: brightness(${brightness}% ) !important; background: #fff !important; }`); util.addThemeColor('#ffffff'); }else if (mode.key === 'dark') { // 护眼模式 = 亮度滤镜 let style_10 = (util.getValue('menu_customMode1') || '60|50').split('|'); let isDay = (new Date().getHours() >= 6 && new Date().getHours() < 18); let brightness = isDay ? style_10[0] : style_10[1]; util.addStyle('dark-mode-style', 'style', `html { filter: brightness(${brightness}% ) !important; background: #fff !important; }`); util.addThemeColor('#e6f7e6'); } else if (mode.key === 'eye') { // 暗色模式 = 亮度+暖色滤镜 let style_20 = (util.getValue('menu_customMode2') || '60|40|50|50').split('|'); let isDay = (new Date().getHours() >= 6 && new Date().getHours() < 18); let brightness = isDay ? style_20[0] : style_20[2]; let sepia = isDay ? style_20[1] : style_20[3]; util.addStyle('dark-mode-style', 'style', `html { filter: brightness(${brightness}% ) sepia(${sepia}% ) !important; background: #fff !important; }`); util.addThemeColor('#f5e6d6'); } }, addButton() { if (this.isTopWindow() && !util.getValue('button_hidden')) { let buttonSize = util.getValue('button_size'); let buttonPosition = util.getValue('button_position'); let svgSize = parseInt(buttonSize * 0.6); let buttonWidth = +buttonSize + 2; // SVG 图标定义 const icons = { moon: ``, sun: ``, eye: ``, palette: `` }; const modes = getEnabledModes(); const idx = getCurrentModeIndex(); const mode = modes[idx]; let html = `
生活不易,猪猪叹气 —— 赏口饲料,让我少气!🐷✨