自动主题切换

根据用户设定时间段, 自动切换已适配网站的黑白主题

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         自动主题切换
// @namespace    airbash/Rocy-June/AutoDarkMode
// @homepage     https://github.com/AirBashX/UserScript
// @version      25.07.18.01
// @description  根据用户设定时间段, 自动切换已适配网站的黑白主题
// @author       airbash / Rocy-June
// @match        *://*/*
// @icon         
// @run-at       document-body
// @grant        unsafeWindow
// @grant        GM_registerMenuCommand
// @grant        GM_unregisterMenuCommand
// @grant        GM_setValue
// @grant        GM_getValue
// @license      GPL-3.0
// ==/UserScript==

(function () {
  "use strict";

  // 调试模式开关
  const DEBUG = false;
  // 调试模式: 是否在报错时中断脚本运行
  const DEBUG_INTERRUPT_ON_ERROR = false;
  // 调试模式: 是否强制切换主题
  let DEBUG_FORCE_TOGGLE = false;

  // 日志函数
  const log = console.log.bind(console, "[AutoDarkMode Script]");
  const warn = console.warn.bind(console, "[AutoDarkMode Script]");
  const error = console.error.bind(console, "[AutoDarkMode Script]");
  const debug = (() =>
    DEBUG
      ? console.error.bind(console, "[AutoDarkMode Script] [Debug Mode]")
      : () => {})();

  debug("Debug mode is on");

  // 失败检测计数器
  let fail_count = 0;
  // 失败检查时间间隔, 单位: 毫秒
  let fail_check_time = 10000;

  const sim_events = {
    pointer_down: () =>
      new PointerEvent("pointerdown", {
        bubbles: true,
        cancelable: true,
        pointerType: "mouse",
      }),
    mouse_over: () =>
      new MouseEvent("mouseover", {
        bubbles: true,
        cancelable: true,
        relatedTarget: null,
      }),
    mouse_leave: () =>
      new MouseEvent("mouseleave", {
        bubbles: true,
        cancelable: true,
        relatedTarget: null,
      }),
    mouse_enter: () =>
      new MouseEvent("mouseenter", {
        bubbles: true,
        cancelable: true,
        relatedTarget: null,
      }),
    key: (is_down, key, modifiers = {}) => {
      const key_codes = {
        control: "ControlLeft",
        alt: "AltLeft",
        shift: "ShiftLeft",
        meta: "MetaLeft",
        enter: "Enter",
      };

      const key_code =
        key_codes[key.toLowerCase()] || `Key${key.toUpperCase()}`;

      const {
        ctrl = false,
        meta = false,
        shift = false,
        alt = false,
        bubbles = true,
        cancelable = true,
      } = modifiers;

      return new KeyboardEvent(is_down ? "keydown" : "keyup", {
        key: key.toUpperCase(),
        code: key_code,
        ctrlKey: ctrl,
        metaKey: meta,
        shiftKey: shift,
        altKey: alt,
        bubbles,
        cancelable,
      });
    },
  };

  log("脚本开始运行");

  // 设置对象
  const settings = {
    // 默认明亮主题起始时间
    light_time: "08:00",
    // 默认黑夜主题起始时间
    dark_time: "18:00",
    // 油猴菜单: 明亮主题设置菜单ID
    light_menu_id: null,
    // 油猴菜单: 黑夜主题设置菜单ID
    dark_menu_id: null,
    // 油猴菜单: Debug强制切换菜单ID
    debug_toggle_id: null,

    // 默认快速检查时间 (Debug 模式且开启报错中断时将延长)
    fast_check_default_time: DEBUG && DEBUG_INTERRUPT_ON_ERROR ? 3000 : 200,
    // 快速检查过后的常态检查时间 (Debug 模式且开启报错中断时将延长)
    after_check_default_time:
      DEBUG && DEBUG_INTERRUPT_ON_ERROR ? 3600000 : 10000,
  };

  // 初始化设置
  initSettings();

  // 所有适配的站点
  /* 
    {
      "domain (站点域名匹配, 防止后续匹配项过多, 提高加载速度)": [{
        url: 匹配站点正则,
        check: 检查当前主题函数(返回 "dark" 或 "light"),
        toggle: 切换主题函数(可选, 为 null 时 toLight 和 toDark 函数会被调用),
        toLight: 切换到明亮主题函数(可选, toggle 为 null 时必需设定),
        toDark: 切换到黑夜主题函数(可选, toggle 为 null 时必需设定),
        fastCheckTime: 未成功检查 / 切换主题前的检查主题间隔时间(可选, 为 null 时默认 1 秒),
        afterCheckTime: 无需切换或切换成功后, 再次检查主题间隔时间(可选, 为 null 时默认 10 秒),
      }]
    }

    tip: 
      如果新增站点有新的顶级域名, 
      记得同步增加到 sites 下方的 domain_suffixes  
  */
  const sites = {
    // Pixiv
    "pixiv.net": [
      {
        url: /^https?:\/\/.*?pixiv\.net.*/,
        check: () =>
          html().getAttribute("data-theme") === "dark" ? "dark" : "light",
        toLight: () => {
          $single(".gtm-darkmode-toggle-on-user-menu-to-light").click();
        },
        toDark: () => {
          $single(".gtm-darkmode-toggle-on-user-menu-to-dark").click();
        },
      },
    ],
    // 风车动漫
    "fengchedmp.com": [
      {
        url: /^https?:\/\/.*?fengchedmp\.com.*/,
        check: () =>
          $single("#cssFile").getAttribute("href").includes("black")
            ? "dark"
            : "light",
        toLight: () => {
          $single("i.icon-rijian").click();
        },
        toDark: () => {
          $single("i.icon-yejian").click();
        },
      },
    ],
    // Wikipedia
    "wikipedia.org": [
      {
        url: /^https?:\/\/.*?wikipedia\.org.*/,
        check: () =>
          html().classList.contains("skin-theme-clientpref-night")
            ? "dark"
            : "light",
        toLight: () => {
          $single("#skin-client-pref-skin-theme-value-day").click();
        },
        toDark: () => {
          $single("#skin-client-pref-skin-theme-value-night").click();
        },
      },
    ],
    // ChatGPT
    "chatgpt.com": [
      {
        url: /^https?:\/\/.*?chatgpt\.com.*/,
        check: () => (html().classList.contains("dark") ? "dark" : "light"),
        toLight: async () => {
          $single("nav>:last-child>div").dispatchEvent(
            sim_events.pointer_down()
          );
          await $singleAsyncClick(
            "div[data-radix-popper-content-wrapper]>div>div>div:nth-child(4)"
          );
          (
            await $singleAsync("div[data-testid=modal-settings]")
          ).style.opacity = 0;
          $single(
            "div.absolute>div.fixed>div>div[role=dialog]>div>div>div:nth-child(2)>section>div>div>div>button"
          ).click();
          await $singleAsyncClick(
            "div[data-radix-popper-content-wrapper]>div>div>div:nth-child(3)"
          );
          $single("button[data-testid=close-button]").click();
        },
        toDark: async () => {
          $single("nav>:last-child>div").dispatchEvent(
            sim_events.pointer_down()
          );
          await $singleAsyncClick(
            "div[data-radix-popper-content-wrapper]>div>div>div:nth-child(4)"
          );
          (
            await $singleAsync("div[data-testid=modal-settings]")
          ).style.opacity = 0;
          $single(
            "div.absolute>div.fixed>div>div[role=dialog]>div>div>div:nth-child(2)>section>div>div>div>button"
          ).click();
          await $singleAsyncClick(
            "div[data-radix-popper-content-wrapper]>div>div>div:nth-child(2)"
          );
          $single("button[data-testid=close-button]").click();
        },
        load: () => {
          document.addEventListener("visibilitychange", function () {
            if (document.visibilityState === "visible") {
              let modal_settings = $single("div[data-testid=modal-settings]");
              if (!modal_settings) {
                return;
              }

              $single("button[data-testid=close-button]").click();
            }
          });
        },
      },
    ],
    // GitHub
    "github.com": [
      {
        url: /^https?:\/\/.*?github\.com.*/,
        check: () =>
          html().getAttribute("data-color-mode") === "dark" ? "dark" : "light",
        toLight: async () => {
          document.body.dispatchEvent(
            sim_events.key(true, "k", { ctrl: true, shift: true })
          );
          const command_container = $single("#command-palette-pjax-container");
          command_container.style.opacity = 0;

          const command_palette = $single(
            "input[aria-controls=command-palette-page-stack]"
          );
          command_palette.value = ">switch theme";

          (
            await $singleAsync("div[role=listbox]>command-palette-item>span")
          ).click();

          let flag = await waitForAsync(() => {
            return $single("svg[aria-label=Loading]").parentNode.hasAttribute(
              "hidden"
            );
          });
          if (!flag) {
            throw new Error("Waiting for load ready failed");
          }

          command_palette.value = "default light";

          (
            await $selectSingleAsync(
              "div[role=listbox]>command-palette-item>span",
              (e) => {
                return e.innerText.toLowerCase().includes("default light");
              }
            )
          ).click();
          command_container.style.opacity = 1;
        },
        toDark: async () => {
          document.body.dispatchEvent(
            sim_events.key(true, "k", { ctrl: true, shift: true })
          );
          const command_container = $single("#command-palette-pjax-container");
          command_container.style.opacity = 0;

          const command_palette = $single(
            "input[aria-controls=command-palette-page-stack]"
          );
          command_palette.value = ">switch theme";

          (
            await $singleAsync("div[role=listbox]>command-palette-item>span")
          ).click();

          let flag = await waitForAsync(() => {
            return $single("svg[aria-label=Loading]").parentNode.hasAttribute(
              "hidden"
            );
          });
          if (!flag) {
            throw new Error("Waiting for load ready failed");
          }

          command_palette.value = "default dark";

          (
            await $selectSingleAsync(
              "div[role=listbox]>command-palette-item>span",
              (e) => {
                return e.innerText.toLowerCase().includes("default dark");
              }
            )
          ).click();
          command_container.style.opacity = 1;
        },
      },
    ],
    // Bilibili
    "bilibili.com": [
      {
        url: /^https?:\/\/.*?bilibili\.com.*/,
        check: () =>
          html().classList.contains("night-mode") ? "dark" : "light",
        toLight: async () => {
          let flag = await waitForTimerAsync(
            () => $single("li.v-popover-wrap.header-avatar-wrap") != null,
            200,
            10000
          );
          if (!flag) {
            throw new Error("Waiting for links-item created failed");
          }

          let avatar = $single("li.v-popover-wrap.header-avatar-wrap");
          avatar.dispatchEvent(sim_events.mouse_enter());
          await nextTick();
          avatar.dispatchEvent(sim_events.mouse_leave());

          flag = await waitForTimerAsync(
            () => $all(".links-item").length > 0,
            200,
            10000
          );
          if (!flag) {
            throw new Error("Waiting for links-item created failed");
          }

          let links_item = $all(".links-item");
          let single_link_item = $single(
            links_item[links_item.length - 1],
            ".single-link-item"
          );
          single_link_item.dispatchEvent(sim_events.mouse_enter());

          flag = await waitForAsync(() =>
            single_link_item.nextElementSibling?.classList.contains("v-popover")
          );
          if (!flag) {
            throw new Error("Waiting for popover created failed");
          }

          let popover = single_link_item.nextElementSibling;
          let options = $all(popover, ".single-link-item.sub-link-item");
          options[1].click();
        },
        toDark: async () => {
          let avatar = $single("li.v-popover-wrap.header-avatar-wrap");
          avatar.dispatchEvent(sim_events.mouse_enter());
          await nextTick();
          avatar.dispatchEvent(sim_events.mouse_leave());

          let flag = await waitForTimerAsync(
            () => $all(".links-item").length > 0,
            200,
            10000
          );
          if (!flag) {
            throw new Error("Waiting for links-item created failed");
          }

          let links_item = $all(".links-item");
          let single_link_item = $single(
            links_item[links_item.length - 1],
            ".single-link-item"
          );
          single_link_item.dispatchEvent(sim_events.mouse_enter());

          flag = await waitForAsync(() =>
            single_link_item.nextElementSibling?.classList.contains("v-popover")
          );
          if (!flag) {
            throw new Error("Waiting for popover created failed");
          }

          let popover = single_link_item.nextElementSibling;
          let options = $all(popover, ".single-link-item.sub-link-item");
          options[0].click();
        },
        load: async () => {
          document.addEventListener("visibilitychange", async function () {
            if (document.visibilityState === "visible") {
              await nextTick();

              let popover = $single(
                "li.v-popover-wrap.header-avatar-wrap>.v-popover"
              );
              let flag = true;
              while (flag) {
                if (getComputedStyle(popover).display === "none") {
                  flag = false;
                }

                popover.dispatchEvent(sim_events.mouse_leave());

                await nextFrame();
              }
            }
          });
        },
      },
    ],
    // DeepSeek
    "deepseek.com": [
      {
        url: /^https?:\/\/.*?deepseek\.com.*/,
        check: () =>
          document.body.classList.contains("dark") ? "dark" : "light",
        toLight: async () => {
          $single(
            "#root>div>div>div:nth-child(2)>div>div>div:last-child>div:last-child"
          ).click();
          (
            await $singleAsync(
              ".ds-floating-position-wrapper .ds-dropdown-menu .ds-dropdown-menu-option:nth-child(2)"
            )
          ).click();
          const theme_select = await $singleAsync(
            ".ds-modal-wrapper .ds-modal .ds-modal-content .ds-flex:nth-child(3) select"
          );
          theme_select.value = "light";
          theme_select.dispatchEvent(new Event("change", { bubbles: true }));
          $single(
            ".ds-modal-wrapper .ds-modal .ds-modal-content .ds-icon-button"
          ).click();
        },
        toDark: async () => {
          $single(
            "#root>div>div>div:nth-child(2)>div>div>div:last-child>div:last-child"
          ).click();
          (
            await $singleAsync(
              ".ds-floating-position-wrapper .ds-dropdown-menu .ds-dropdown-menu-option:nth-child(2)"
            )
          ).click();
          const theme_select = await $singleAsync(
            ".ds-modal-wrapper .ds-modal .ds-modal-content .ds-flex:nth-child(3) select"
          );
          theme_select.value = "dark";
          theme_select.dispatchEvent(new Event("change", { bubbles: true }));
          $single(
            ".ds-modal-wrapper .ds-modal .ds-modal-content .ds-icon-button"
          ).click();
        },
      },
    ],
  };

  const domain_suffixes = [
    ".com.cn",
    ".com",
    ".org",
    ".net",
    ".edu",
    ".gov",
    ".cn",
  ];

  // 匹配到的域名设置
  const no_www_domain = getRootDomain(location.hostname);

  const domain_setting = sites[no_www_domain];
  if (!domain_setting) {
    log(`未找到适配的站点: ${no_www_domain}`);
    return;
  }

  // 匹配到的网站设置
  const site_setting = domain_setting.find((s) => s.url.test(location.href));
  if (!site_setting) {
    log(`未找到当前页面的适配操作`);
    return;
  }

  // 加载完成后开始检查主题
  addEventListener("load", async () => {
    // 调试模式向 window 注册调试函数
    if (DEBUG) {
      warn("向 window 注册调试函数");

      const debug_functions = [
        ["$log", log],
        ["$warn", warn],
        ["$error", error],
        ["$debug", debug],
        ["$single", $single],
        ["$singleAsync", $singleAsync],
        ["$singleAsyncClick", $singleAsyncClick],
        ["$singleTimerAsync", $singleTimerAsync],
        ["$selectSingleAsync", $selectSingleAsync],
        ["$all", $all],
      ];

      for (const [name, func] of debug_functions) {
        unsafeWindow[name] = func;
      }

      warn("调试函数注册完毕");
    }

    site_setting.load?.();

    const timer = new Timer(
      intervalFunc,
      site_setting.afterCheckTime || settings.after_check_default_time
    );

    await intervalFunc();

    timer.start();

    document.addEventListener("visibilitychange", async function () {
      if (document.visibilityState === "visible") {
        log("页面可见, 重新开始检查主题");

        await checkAndChangeTheme();
      }
    });

    async function intervalFunc() {
      try {
        await checkAndChangeTheme();

        fail_count = 0;
        timer.delay =
          site_setting.afterCheckTime || settings.after_check_default_time;

        log("检查/操作完成, 切换到慢速模式");
      } catch (ex) {
        if (DEBUG) {
          if (DEBUG_INTERRUPT_ON_ERROR) {
            debug("检查/操作失败, 中断脚本运行", ex);
            return;
          }

          debug("检查/操作失败", ex);
        }

        fail_count++;
        if (
          fail_count >=
          fail_check_time /
            (site_setting.fastCheckTime || settings.fast_check_default_time)
        ) {
          timer.delay =
            site_setting.afterCheckTime || settings.after_check_default_time;
          error("失败超出时长限制, 切换到慢速模式");
          return;
        }

        timer.delay =
          site_setting.fastCheckTime || settings.fast_check_default_time;
        error("检查/操作失败, 切换到高速模式", ex);
      }
    }
  });

  // 查找元素, 用法类似于 jQuery
  function $single(arg1, arg2) {
    if (arguments.length == 2) {
      return $single2(arg1, arg2);
    }
    return $single1(arg1);
  }
  function $single1(selector) {
    return document.querySelector(selector);
  }
  function $single2(dom, selector) {
    return dom.querySelector(selector);
  }

  /*
    类似 jQuery 的元素查找函数

    适用于连续操作中可能影响用户操作体验的场景, 该方式会尽可能在渲染的同一帧内找到该 DOM 元素,
    以缩短元素变更对用户交互的干扰时间 (如弹窗在显示帧立即点击关闭按钮)
    
    与 singleTimerAsync 不同的是, 本函数每一帧都会进行查询, 因此可能对性能造成一定影响
    如果操作不会显著影响用户体验, 则推荐使用 singleTimerAsync
  */
  async function $singleAsync(selector, timeout = 1000) {
    const start = Date.now();
    while (Date.now() - start < timeout) {
      const ele = $single(selector);
      if (ele) {
        return ele;
      }

      await nextFrame();
    }

    throw new Error("Timeout");
  }

  async function $singleAsyncClick(select, timeout = 1000) {
    const ele = await $singleAsync(select, timeout);
    ele?.click();

    return ele;
  }

  /*
    类似 jQuery 的元素查找函数
    
    适用于不会改变页面结构、或不影响用户操作体验的场景
    使用自定义 Timer 以固定间隔查询元素, 性能开销较小, 但查找速度相对较慢

    与 singleAsync 不同的是, 该函数通过节流方式降低性能消耗,
    更适合异步检查某些静态区域元素是否已加载完成等情况
  */
  function $singleTimerAsync(selector, interval = 100, timeout = 1000) {
    return promise(async (resolve, reject) => {
      const start = Date.now();
      const timer = new Timer(() => {
        const ele = $single(selector);
        if (ele) {
          resolve(ele);
          timer.stop();
          return;
        }
        if (Date.now() - start >= timeout) {
          reject(new Error("Timeout"));
          timer.stop();
          return;
        }
      }, interval);
      timer.start(true);
    });
  }

  // 用于查找无法单纯使用 CSS 选择器的复杂元素
  async function $selectSingleAsync(selector, filter, timeout = 1000) {
    const start = Date.now();
    while (Date.now() - start < timeout) {
      const eles = $all(selector);
      if (eles.length) {
        let selected = Array.from(eles).filter(filter);
        if (selected.length) {
          return selected[0];
        }
      }

      await nextFrame();
    }

    throw new Error("Timeout");
  }

  function $all(arg1, arg2) {
    if (arguments.length == 2) {
      return $all2(arg1, arg2);
    }
    return $all1(arg1);
  }
  function $all1(selector) {
    return document.querySelectorAll(selector);
  }
  function $all2(dom, selector) {
    return dom.querySelectorAll(selector);
  }

  function html() {
    return document.documentElement;
  }

  // 初始化设置
  function initSettings() {
    log("初始化设置");

    settings.light_time = GM_getValue("light_time", settings.light_time);
    settings.dark_time = GM_getValue("dark_time", settings.dark_time);

    refreshMenuCommand();
  }

  // 获取根域名
  function getRootDomain(domain) {
    for (const suffix of domain_suffixes) {
      if (!domain.endsWith(suffix)) {
        continue;
      }

      const index = domain.lastIndexOf(".", suffix.length);
      return domain.substring(index + 1);
    }
  }

  // 检查当前时间是否需要切换主题
  async function checkAndChangeTheme() {
    log("开始检查主题");

    const light_minutes = timeToMinutes(settings.light_time);
    const dark_minutes = timeToMinutes(settings.dark_time);

    let now = nowMinutes();

    const current_theme = site_setting.check();
    log(`当前主题:${current_theme}`);

    if (DEBUG && DEBUG_FORCE_TOGGLE) {
      debug("强制切换主题");
      if (
        now >= light_minutes &&
        now < dark_minutes &&
        current_theme === "light"
      ) {
        now = dark_minutes;
        debug("将当前时间设置为黑夜时间");
      } else if (
        (now >= dark_minutes || now < light_minutes) &&
        current_theme === "dark"
      ) {
        now = light_minutes;
        debug("将当前时间设置为明亮时间");
      }
    }

    if (now >= light_minutes && now < dark_minutes) {
      if (current_theme === "light") {
        log("当前时间为明亮主题时间, 无需切换");
        return;
      }
      if (site_setting.toggle) {
        log("切换到明亮主题 - toggle");
        await site_setting.toggle();
        log("切换完成");
        return;
      }
      if (site_setting.toLight) {
        log("切换到明亮主题 - toLight");
        await site_setting.toLight();
        log("切换完成");
        return;
      }

      error("切换到明亮主题 - 未指定切换函数");
    } else {
      if (current_theme === "dark") {
        log("当前时间为黑夜主题时间, 无需切换");
        return;
      }
      if (site_setting.toggle) {
        log("切换到黑夜主题 - toggle");
        await site_setting.toggle();
        log("切换完成");
        return;
      }
      if (site_setting.toDark) {
        log("切换到黑夜主题 - toDark");
        await site_setting.toDark();
        log("切换完成");
        return;
      }

      error("切换到黑夜主题 - 未指定切换函数");
    }
  }

  // 刷新菜单
  function refreshMenuCommand() {
    log("刷新脚本菜单");

    if (settings.light_menu_id) {
      GM_unregisterMenuCommand(settings.light_menu_id);
    }
    if (settings.dark_menu_id) {
      GM_unregisterMenuCommand(settings.dark_menu_id);
    }
    if (settings.debug_toggle_id) {
      GM_unregisterMenuCommand(settings.debug_toggle_id);
    }

    settings.light_menu_id = GM_registerMenuCommand(
      `设置明亮时间 (${settings.light_time})`,
      () => setTimePrompt("light_time", "明亮时间")
    );
    settings.dark_menu_id = GM_registerMenuCommand(
      `设置黑夜时间 (${settings.dark_time})`,
      () => setTimePrompt("dark_time", "黑夜时间")
    );
    if (DEBUG) {
      settings.debug_toggle_id = GM_registerMenuCommand(
        `调试模式: 强制切换主题`,
        async () => {
          DEBUG_FORCE_TOGGLE = true;
          try {
            await checkAndChangeTheme();
          } catch (ex) {
            error("强制切换主题失败", ex);
          } finally {
            DEBUG_FORCE_TOGGLE = false;
          }
        }
      );
    }
  }

  // 设置时间提示框
  function setTimePrompt(key, label) {
    log(`设置${label}提示框`);

    const old_val = GM_getValue(key, settings[key]);

    log(`旧${label}:${old_val}`);

    const new_val = prompt(`设置${label}(格式 HH:mm):`, old_val);
    log(`用户输入: ${new_val}`);
    if (!new_val) {
      return;
    }
    if (!new_val || !/^([0-1][0-9]|2[0-3])[::]([0-5][0-9])$/.test(new_val)) {
      alert('格式不正确, 时间格式为 "08:00"');
      return;
    }
    const standard_new_val = new_val.replace(":", ":");

    log(`新${label}:${standard_new_val}`);

    const tmp = standard_new_val;
    settings[key] = standard_new_val;
    if (
      timeToMinutes(settings.light_time) > timeToMinutes(settings.dark_time)
    ) {
      log("黑夜时间不能早于明亮时间");

      settings[key] = tmp;
      alert("黑夜时间不能早于明亮时间");
      return;
    }

    GM_setValue(key, standard_new_val);

    log(`${label} 已设置为:${standard_new_val}`);

    alert(`${label} 已设置为:${standard_new_val}`);

    refreshMenuCommand();
  }

  // 将时间字符串转为分钟
  function timeToMinutes(time) {
    const [hour, minute] = time.split(":").map(Number);
    return hour * 60 + minute;
  }

  // 获取当前时间的分钟数
  function nowMinutes() {
    const now = new Date();
    return now.getHours() * 60 + now.getMinutes();
  }

  // 将函数加入下一轮事件循环
  function nextTick(func) {
    return Promise.resolve().then(() => {
      func?.();
    });
  }

  // 等待下一帧
  function nextFrame(func) {
    return new Promise((resolve) => {
      requestAnimationFrame((timestamp) => {
        func?.(timestamp);
        resolve(timestamp);
      });
    });
  }

  // 异步函数返回 Promise
  function promise(func) {
    return new Promise((resolve, reject) => {
      try {
        func(resolve, reject);
      } catch (ex) {
        reject(ex);
      }
    });
  }

  // 等待指定时间
  function waitTimeAsync(time) {
    return new Promise((resolve) => {
      setTimeout(() => {
        resolve();
      }, time);
    });
  }

  /*
    等待某个元素达到某个状态

    适用于连续操作中可能影响用户操作体验的场景, 该方式会在渲染的每一帧内检测该 DOM 元素的状态,
    以缩短元素变更对用户交互的干扰时间 (如弹窗在显示帧立即点击关闭按钮)
    
    与 waitForTimerAsync 不同的是, 本函数每一帧都会进行查询, 因此可能对性能造成一定影响
    如果操作不会显著影响用户体验, 则推荐使用 waitForTimerAsync
  */
  async function waitForAsync(detector, timeout = 1000) {
    if (!detector) throw new Error("detector can not be null.");

    const start = Date.now();
    while (Date.now() - start < timeout) {
      if (detector()) {
        return true;
      }

      await nextFrame();
    }

    return false;
  }

  /* 等待某个元素达到某个状态
    
    适用于不会改变页面结构、或不影响用户操作体验的场景
    使用自定义 Timer 以固定间隔查询元素, 性能开销较小, 但查找速度相对较慢

    与 waitForAsync 不同的是, 该函数通过节流方式降低性能消耗,
    更适合异步检查某些静态区域元素是否已加载完成等情况
  */
  function waitForTimerAsync(detector, interval = 100, timeout = 1000) {
    if (!detector) throw new Error("detector can not be null.");

    return promise(async (resolve, reject) => {
      const start = Date.now();
      const timer = new Timer(() => {
        if (detector()) {
          resolve(true);
          timer.stop();
          return;
        }
        if (Date.now() - start >= timeout) {
          reject(new Error("Timeout"));
          timer.stop();
          return;
        }
      }, interval);
      timer.start(true);
    });
  }

  // 自建定时器
  class Timer {
    #timer = null;
    #func = null;
    delay = 0;

    constructor(func, delay) {
      this.#func = func;
      this.delay = delay;
    }

    start(immediate = false) {
      this.stop();

      let start_new = async () => {
        if (immediate) {
          await this.#func();
        }

        this.#timer = setTimeout(async () => {
          await this.#func();
          this.start();
        }, this.delay);
      };
      start_new();
    }

    stop() {
      clearTimeout(this.#timer);
    }
  }
})();