From e597b6da4faa1f30c7b3479cdbb96ac5b4fbb0f5 Mon Sep 17 00:00:00 2001
From: buhuazhen <hua100783@gmail.com>
Date: 星期二, 17 三月 2026 11:44:20 +0800
Subject: [PATCH] feat(主题): 新增暗色模式支持

---
 src/settings.js                          |    4 +
 src/layout/components/Settings/index.vue |   28 +++++++
 src/store/modules/settings.js            |  172 ++++++++++++++++++++++++++----------------
 3 files changed, 137 insertions(+), 67 deletions(-)

diff --git a/src/layout/components/Settings/index.vue b/src/layout/components/Settings/index.vue
index b4796f3..917b28b 100644
--- a/src/layout/components/Settings/index.vue
+++ b/src/layout/components/Settings/index.vue
@@ -71,6 +71,24 @@
         />
       </span>
     </div>
+    <div class="drawer-item">
+      <span>鏄剧ず妯″紡</span>
+      <span class="comp-style">
+        <el-select
+          v-model="settingsStore.darkMode"
+          placeholder="璇烽�夋嫨"
+          style="width: 130px"
+          @change="darkModeChange"
+        >
+          <el-option
+            v-for="item in darkModeOptions"
+            :key="item.value"
+            :label="item.label"
+            :value="item.value"
+          />
+        </el-select>
+      </span>
+    </div>
     <el-divider />
 
     <h3 class="drawer-title">绯荤粺甯冨眬閰嶇疆</h3>
@@ -152,6 +170,11 @@
   "#8F4B28",
   "#4C0009",
 ]);
+const darkModeOptions = ref([
+  { label: "璺熼殢绯荤粺", value: "auto" },
+  { label: "娴呰壊", value: "light" },
+  { label: "娣辫壊", value: "dark" },
+]);
 
 /** 鏄惁闇�瑕乼opnav */
 function topNavChange(val) {
@@ -164,6 +187,10 @@
 function themeChange(val) {
   settingsStore.theme = val;
   handleThemeStyle(val);
+}
+
+function darkModeChange(val) {
+  settingsStore.setDarkMode(val);
 }
 
 function handleTheme(val) {
@@ -181,6 +208,7 @@
     dynamicTitle: storeSettings.value.dynamicTitle,
     sideTheme: storeSettings.value.sideTheme,
     theme: storeSettings.value.theme,
+    darkMode: storeSettings.value.darkMode,
   };
   localStorage.setItem("layout-setting", JSON.stringify(layoutSetting));
   setTimeout(proxy.$modal.closeLoading(), 1000);
diff --git a/src/settings.js b/src/settings.js
index f944e2d..4a1c8bb 100644
--- a/src/settings.js
+++ b/src/settings.js
@@ -36,6 +36,10 @@
    * 鏄惁鏄剧ず鍔ㄦ�佹爣棰�
    */
   dynamicTitle: false,
+  /**
+   * 涓婚妯″紡 auto璺熼殢绯荤粺锛宭ight娴呰壊锛宒ark娣辫壊
+   */
+  darkMode: "auto",
 
   /**
    * @type {string | array} 'production' | ['production', 'development']
diff --git a/src/store/modules/settings.js b/src/store/modules/settings.js
index e526da0..3a5f6a2 100644
--- a/src/store/modules/settings.js
+++ b/src/store/modules/settings.js
@@ -1,67 +1,105 @@
-import defaultSettings from "@/settings";
-import { useDark, useToggle } from "@vueuse/core";
-import { useDynamicTitle } from "@/utils/dynamicTitle";
-
-const isDark = useDark();
-const toggleDark = useToggle(isDark);
-
-const {
-  sideTheme,
-  showSettings,
-  topNav,
-  tagsView,
-  fixedHeader,
-  sidebarLogo,
-  dynamicTitle,
-} = defaultSettings;
-
-const storageSetting = JSON.parse(localStorage.getItem("layout-setting")) || "";
-
-const useSettingsStore = defineStore("settings", {
-  state: () => ({
-    title: "",
-    theme: storageSetting.theme || "#002fa7",
-    sideTheme: storageSetting.sideTheme || sideTheme,
-    showSettings: showSettings,
-    topNav:
-      storageSetting.topNav === undefined ? topNav : storageSetting.topNav,
-    tagsView:
-      storageSetting.tagsView === undefined
-        ? tagsView
-        : storageSetting.tagsView,
-    fixedHeader:
-      storageSetting.fixedHeader === undefined
-        ? fixedHeader
-        : storageSetting.fixedHeader,
-    sidebarLogo:
-      storageSetting.sidebarLogo === undefined
-        ? sidebarLogo
-        : storageSetting.sidebarLogo,
-    dynamicTitle:
-      storageSetting.dynamicTitle === undefined
-        ? dynamicTitle
-        : storageSetting.dynamicTitle,
-    isDark: isDark.value,
-  }),
-  actions: {
-    // 淇敼甯冨眬璁剧疆
-    changeSetting(data) {
-      const { key, value } = data;
-      if (this.hasOwnProperty(key)) {
-        this[key] = value;
-      }
-    },
-    // 璁剧疆缃戦〉鏍囬
-    setTitle(title) {
-      this.title = title;
-      useDynamicTitle();
-    },
-    // 鍒囨崲鏆楅粦妯″紡
-    toggleTheme() {
-      this.isDark = !this.isDark;
-      toggleDark();
-    },
-  },
-});
-
-export default useSettingsStore;
+import defaultSettings from "@/settings";
+import { useColorMode, usePreferredDark } from "@vueuse/core";
+import { useDynamicTitle } from "@/utils/dynamicTitle";
+
+const preferredDark = usePreferredDark();
+const colorMode = useColorMode({
+  emitAuto: true,
+});
+
+const {
+  sideTheme,
+  showSettings,
+  topNav,
+  tagsView,
+  fixedHeader,
+  sidebarLogo,
+  dynamicTitle,
+  darkMode,
+} = defaultSettings;
+
+const storageSetting = JSON.parse(localStorage.getItem("layout-setting") || "{}");
+const defaultDarkMode = darkMode || "auto";
+const initialDarkMode = storageSetting.darkMode || defaultDarkMode;
+colorMode.value = initialDarkMode;
+const getIsDark = (mode) => mode === "dark" || (mode === "auto" && preferredDark.value);
+
+const useSettingsStore = defineStore("settings", () => {
+  const title = ref("");
+  const theme = ref(storageSetting.theme || "#002fa7");
+  const sideThemeValue = ref(storageSetting.sideTheme || sideTheme);
+  const showSettingsValue = ref(showSettings);
+  const topNavValue = ref(
+    storageSetting.topNav === undefined ? topNav : storageSetting.topNav
+  );
+  const tagsViewValue = ref(
+    storageSetting.tagsView === undefined ? tagsView : storageSetting.tagsView
+  );
+  const fixedHeaderValue = ref(
+    storageSetting.fixedHeader === undefined ? fixedHeader : storageSetting.fixedHeader
+  );
+  const sidebarLogoValue = ref(
+    storageSetting.sidebarLogo === undefined ? sidebarLogo : storageSetting.sidebarLogo
+  );
+  const dynamicTitleValue = ref(
+    storageSetting.dynamicTitle === undefined ? dynamicTitle : storageSetting.dynamicTitle
+  );
+  const darkModeValue = ref(initialDarkMode);
+  const isDark = computed(() => getIsDark(darkModeValue.value));
+
+  function changeSetting(data) {
+    const { key, value } = data;
+    const settingMap = {
+      title,
+      theme,
+      sideTheme: sideThemeValue,
+      showSettings: showSettingsValue,
+      topNav: topNavValue,
+      tagsView: tagsViewValue,
+      fixedHeader: fixedHeaderValue,
+      sidebarLogo: sidebarLogoValue,
+      dynamicTitle: dynamicTitleValue,
+      darkMode: darkModeValue,
+    };
+    if (Object.prototype.hasOwnProperty.call(settingMap, key)) {
+      settingMap[key].value = value;
+      if (key === "darkMode") {
+        colorMode.value = value;
+      }
+    }
+  }
+
+  function setTitle(value) {
+    title.value = value;
+    useDynamicTitle();
+  }
+
+  function setDarkMode(mode) {
+    darkModeValue.value = mode;
+    colorMode.value = mode;
+  }
+
+  function toggleTheme() {
+    setDarkMode(isDark.value ? "light" : "dark");
+  }
+
+  return {
+    title,
+    theme,
+    sideTheme: sideThemeValue,
+    showSettings: showSettingsValue,
+    topNav: topNavValue,
+    tagsView: tagsViewValue,
+    fixedHeader: fixedHeaderValue,
+    sidebarLogo: sidebarLogoValue,
+    dynamicTitle: dynamicTitleValue,
+    darkMode: darkModeValue,
+    isDark,
+    changeSetting,
+    setTitle,
+    setDarkMode,
+    toggleTheme,
+  };
+});
+
+export default useSettingsStore;

--
Gitblit v1.9.3