index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585
  1. <script setup lang="ts">
  2. import {
  3. reactive,
  4. ref,
  5. unref,
  6. watch,
  7. computed,
  8. nextTick,
  9. useCssModule,
  10. getCurrentInstance
  11. } from "vue";
  12. import { find } from "lodash-unified";
  13. import { getConfig } from "/@/config";
  14. import { useRouter } from "vue-router";
  15. import panel from "../panel/index.vue";
  16. import { emitter } from "/@/utils/mitt";
  17. import { templateRef } from "@vueuse/core";
  18. import { debounce } from "/@/utils/debounce";
  19. import { themeColorsType } from "../../types";
  20. import { useAppStoreHook } from "/@/store/modules/app";
  21. import { shadeBgColor } from "../../theme/element-plus";
  22. import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
  23. import { storageLocal, storageSession } from "/@/utils/storage";
  24. import { useMultiTagsStoreHook } from "/@/store/modules/multiTags";
  25. import { createNewStyle, writeNewStyle } from "../../theme/element-plus";
  26. import { toggleTheme } from "@pureadmin/theme/dist/browser-utils";
  27. import dayIcon from "/@/assets/svg/day.svg?component";
  28. import darkIcon from "/@/assets/svg/dark.svg?component";
  29. const router = useRouter();
  30. const { isSelect } = useCssModule();
  31. const body = document.documentElement as HTMLElement;
  32. const instance =
  33. getCurrentInstance().appContext.app.config.globalProperties.$storage;
  34. const instanceConfig =
  35. getCurrentInstance().appContext.app.config.globalProperties.$config;
  36. let themeColors = ref<Array<themeColorsType>>([
  37. // 道奇蓝(默认)
  38. { color: "#1b2a47", themeColor: "default" },
  39. // 亮白色
  40. { color: "#ffffff", themeColor: "light" },
  41. // 猩红色
  42. { color: "#f5222d", themeColor: "dusk" },
  43. // 橙红色
  44. { color: "#fa541c", themeColor: "volcano" },
  45. // 金色
  46. { color: "#fadb14", themeColor: "yellow" },
  47. // 绿宝石
  48. { color: "#13c2c2", themeColor: "mingQing" },
  49. // 酸橙绿
  50. { color: "#52c41a", themeColor: "auroraGreen" },
  51. // 深粉色
  52. { color: "#eb2f96", themeColor: "pink" },
  53. // 深紫罗兰色
  54. { color: "#722ed1", themeColor: "saucePurple" }
  55. ]);
  56. const verticalRef = templateRef<HTMLElement | null>("verticalRef", null);
  57. const horizontalRef = templateRef<HTMLElement | null>("horizontalRef", null);
  58. const mixRef = templateRef<HTMLElement | null>("mixRef", null);
  59. let layoutTheme =
  60. ref(storageLocal.getItem("responsive-layout")) ||
  61. ref({
  62. layout: instanceConfig?.Layout ?? "vertical",
  63. theme: instanceConfig?.Theme ?? "default"
  64. });
  65. // body添加layout属性,作用于src/style/sidebar.scss
  66. if (unref(layoutTheme)) {
  67. let layout = unref(layoutTheme).layout;
  68. let theme = unref(layoutTheme).theme;
  69. toggleTheme({
  70. scopeName: `layout-theme-${theme}`
  71. });
  72. setLayoutModel(layout);
  73. }
  74. // 默认灵动模式
  75. const markValue = ref(instance.configure?.showModel ?? "smart");
  76. const logoVal = ref(instance.configure?.showLogo ?? true);
  77. const epThemeColor = ref(useEpThemeStoreHook().getEpThemeColor);
  78. const settings = reactive({
  79. greyVal: instance.configure.grey,
  80. weakVal: instance.configure.weak,
  81. tabsVal: instance.configure.hideTabs,
  82. showLogo: instance.configure.showLogo,
  83. showModel: instance.configure.showModel,
  84. multiTagsCache: instance.configure.multiTagsCache
  85. });
  86. const getThemeColorStyle = computed(() => {
  87. return color => {
  88. return { background: color };
  89. };
  90. });
  91. function storageConfigureChange<T>(key: string, val: T): void {
  92. const storageConfigure = instance.configure;
  93. storageConfigure[key] = val;
  94. instance.configure = storageConfigure;
  95. }
  96. function toggleClass(flag: boolean, clsName: string, target?: HTMLElement) {
  97. const targetEl = target || document.body;
  98. let { className } = targetEl;
  99. className = className.replace(clsName, "").trim();
  100. targetEl.className = flag ? `${className} ${clsName} ` : className;
  101. }
  102. // 灰色模式设置
  103. const greyChange = (value): void => {
  104. toggleClass(settings.greyVal, "html-grey", document.querySelector("html"));
  105. storageConfigureChange("grey", value);
  106. };
  107. // 色弱模式设置
  108. const weekChange = (value): void => {
  109. toggleClass(
  110. settings.weakVal,
  111. "html-weakness",
  112. document.querySelector("html")
  113. );
  114. storageConfigureChange("weak", value);
  115. };
  116. const tagsChange = () => {
  117. let showVal = settings.tabsVal;
  118. storageConfigureChange("hideTabs", showVal);
  119. emitter.emit("tagViewsChange", showVal);
  120. };
  121. const multiTagsCacheChange = () => {
  122. let multiTagsCache = settings.multiTagsCache;
  123. storageConfigureChange("multiTagsCache", multiTagsCache);
  124. useMultiTagsStoreHook().multiTagsCacheChange(multiTagsCache);
  125. };
  126. // 清空缓存并返回登录页
  127. function onReset() {
  128. router.push("/login");
  129. const { Grey, Weak, MultiTagsCache, EpThemeColor, Layout } = getConfig();
  130. useAppStoreHook().setLayout(Layout);
  131. useEpThemeStoreHook().setEpThemeColor(EpThemeColor);
  132. useMultiTagsStoreHook().multiTagsCacheChange(MultiTagsCache);
  133. toggleClass(Grey, "html-grey", document.querySelector("html"));
  134. toggleClass(Weak, "html-weakness", document.querySelector("html"));
  135. useMultiTagsStoreHook().handleTags("equal", [
  136. {
  137. path: "/welcome",
  138. parentPath: "/",
  139. meta: {
  140. title: "menus.hshome",
  141. icon: "home-filled",
  142. i18n: true
  143. }
  144. }
  145. ]);
  146. storageLocal.clear();
  147. storageSession.clear();
  148. }
  149. function onChange(label) {
  150. storageConfigureChange("showModel", label);
  151. emitter.emit("tagViewsShowModel", label);
  152. }
  153. // 侧边栏Logo
  154. function logoChange() {
  155. unref(logoVal)
  156. ? storageConfigureChange("showLogo", true)
  157. : storageConfigureChange("showLogo", false);
  158. emitter.emit("logoChange", unref(logoVal));
  159. }
  160. function setFalse(Doms): any {
  161. Doms.forEach(v => {
  162. toggleClass(false, isSelect, unref(v));
  163. });
  164. }
  165. watch(instance, ({ layout }) => {
  166. switch (layout["layout"]) {
  167. case "vertical":
  168. toggleClass(true, isSelect, unref(verticalRef));
  169. debounce(setFalse([horizontalRef]), 50);
  170. debounce(setFalse([mixRef]), 50);
  171. break;
  172. case "horizontal":
  173. toggleClass(true, isSelect, unref(horizontalRef));
  174. debounce(setFalse([verticalRef]), 50);
  175. debounce(setFalse([mixRef]), 50);
  176. break;
  177. case "mix":
  178. toggleClass(true, isSelect, unref(mixRef));
  179. debounce(setFalse([verticalRef]), 50);
  180. debounce(setFalse([horizontalRef]), 50);
  181. break;
  182. }
  183. });
  184. // 主题色 激活选择项
  185. const getThemeColor = computed(() => {
  186. return current => {
  187. if (
  188. current === layoutTheme.value.theme &&
  189. layoutTheme.value.theme !== "light"
  190. ) {
  191. return "#fff";
  192. } else if (
  193. current === layoutTheme.value.theme &&
  194. layoutTheme.value.theme === "light"
  195. ) {
  196. return "#1d2b45";
  197. } else {
  198. return "transparent";
  199. }
  200. };
  201. });
  202. // 设置导航模式
  203. function setLayoutModel(layout: string) {
  204. layoutTheme.value.layout = layout;
  205. window.document.body.setAttribute("layout", layout);
  206. instance.layout = {
  207. layout,
  208. theme: layoutTheme.value.theme,
  209. darkMode: instance.layout.darkMode,
  210. sidebarStatus: instance.layout.sidebarStatus,
  211. epThemeColor: instance.layout.epThemeColor
  212. };
  213. useAppStoreHook().setLayout(layout);
  214. }
  215. // 存放夜间主题切换前的导航主题色
  216. let tempLayoutThemeColor;
  217. // 设置导航主题色
  218. function setLayoutThemeColor(theme: string) {
  219. tempLayoutThemeColor = instance.layout.theme;
  220. layoutTheme.value.theme = theme;
  221. toggleTheme({
  222. scopeName: `layout-theme-${theme}`
  223. });
  224. instance.layout = {
  225. layout: useAppStoreHook().layout,
  226. theme,
  227. darkMode: dataTheme.value,
  228. sidebarStatus: instance.layout.sidebarStatus,
  229. epThemeColor: instance.layout.epThemeColor
  230. };
  231. if (theme === "default" || theme === "light") {
  232. setEpThemeColor(getConfig().EpThemeColor);
  233. } else {
  234. const colors = find(themeColors.value, { themeColor: theme });
  235. setEpThemeColor(colors.color);
  236. }
  237. }
  238. // 设置ep主题色
  239. const setEpThemeColor = (color: string) => {
  240. writeNewStyle(createNewStyle(color));
  241. useEpThemeStoreHook().setEpThemeColor(color);
  242. body.style.setProperty("--el-color-primary-active", shadeBgColor(color));
  243. };
  244. let dataTheme = ref<boolean>(instance.layout.darkMode);
  245. // 日间、夜间主题切换
  246. function dataThemeChange() {
  247. if (dataTheme.value) {
  248. body.setAttribute("data-theme", "dark");
  249. setLayoutThemeColor("light");
  250. } else {
  251. body.setAttribute("data-theme", "");
  252. tempLayoutThemeColor && setLayoutThemeColor(tempLayoutThemeColor);
  253. instance.layout = {
  254. layout: useAppStoreHook().layout,
  255. theme: instance.layout.theme,
  256. darkMode: dataTheme.value,
  257. sidebarStatus: instance.layout.sidebarStatus,
  258. epThemeColor: instance.layout.epThemeColor
  259. };
  260. }
  261. }
  262. //初始化项目配置
  263. nextTick(() => {
  264. settings.greyVal &&
  265. document.querySelector("html")?.setAttribute("class", "html-grey");
  266. settings.weakVal &&
  267. document.querySelector("html")?.setAttribute("class", "html-weakness");
  268. settings.tabsVal && tagsChange();
  269. writeNewStyle(createNewStyle(epThemeColor.value));
  270. dataThemeChange();
  271. });
  272. </script>
  273. <template>
  274. <panel>
  275. <el-divider>主题</el-divider>
  276. <el-switch
  277. v-model="dataTheme"
  278. inline-prompt
  279. class="pure-datatheme"
  280. :active-icon="dayIcon"
  281. :inactive-icon="darkIcon"
  282. @change="dataThemeChange"
  283. />
  284. <el-divider>导航栏模式</el-divider>
  285. <ul class="pure-theme">
  286. <el-tooltip class="item" content="左侧模式" placement="bottom">
  287. <li
  288. :class="layoutTheme.layout === 'vertical' ? $style.isSelect : ''"
  289. ref="verticalRef"
  290. @click="setLayoutModel('vertical')"
  291. >
  292. <div />
  293. <div />
  294. </li>
  295. </el-tooltip>
  296. <el-tooltip class="item" content="顶部模式" placement="bottom">
  297. <li
  298. :class="layoutTheme.layout === 'horizontal' ? $style.isSelect : ''"
  299. ref="horizontalRef"
  300. @click="setLayoutModel('horizontal')"
  301. >
  302. <div />
  303. <div />
  304. </li>
  305. </el-tooltip>
  306. <el-tooltip class="item" content="混合模式" placement="bottom">
  307. <li
  308. :class="layoutTheme.layout === 'mix' ? $style.isSelect : ''"
  309. ref="mixRef"
  310. @click="setLayoutModel('mix')"
  311. >
  312. <div />
  313. <div />
  314. </li>
  315. </el-tooltip>
  316. </ul>
  317. <el-divider v-show="!dataTheme">主题色</el-divider>
  318. <ul class="theme-color" v-show="!dataTheme">
  319. <li
  320. v-for="(item, index) in themeColors"
  321. :key="index"
  322. :style="getThemeColorStyle(item.color)"
  323. @click="setLayoutThemeColor(item.themeColor)"
  324. >
  325. <el-icon
  326. style="margin: 0.1em 0.1em 0 0"
  327. :size="17"
  328. :color="getThemeColor(item.themeColor)"
  329. >
  330. <IconifyIconOffline icon="check" />
  331. </el-icon>
  332. </li>
  333. </ul>
  334. <el-divider>界面显示</el-divider>
  335. <ul class="setting">
  336. <li v-show="!dataTheme">
  337. <span>灰色模式</span>
  338. <el-switch
  339. v-model="settings.greyVal"
  340. inline-prompt
  341. inactive-color="#a6a6a6"
  342. active-text="开"
  343. inactive-text="关"
  344. @change="greyChange"
  345. />
  346. </li>
  347. <li v-show="!dataTheme">
  348. <span>色弱模式</span>
  349. <el-switch
  350. v-model="settings.weakVal"
  351. inline-prompt
  352. inactive-color="#a6a6a6"
  353. active-text="开"
  354. inactive-text="关"
  355. @change="weekChange"
  356. />
  357. </li>
  358. <li>
  359. <span>隐藏标签页</span>
  360. <el-switch
  361. v-model="settings.tabsVal"
  362. inline-prompt
  363. inactive-color="#a6a6a6"
  364. active-text="开"
  365. inactive-text="关"
  366. @change="tagsChange"
  367. />
  368. </li>
  369. <li>
  370. <span>侧边栏Logo</span>
  371. <el-switch
  372. v-model="logoVal"
  373. inline-prompt
  374. :active-value="true"
  375. :inactive-value="false"
  376. inactive-color="#a6a6a6"
  377. active-text="开"
  378. inactive-text="关"
  379. @change="logoChange"
  380. />
  381. </li>
  382. <li>
  383. <span>标签页持久化</span>
  384. <el-switch
  385. v-model="settings.multiTagsCache"
  386. inline-prompt
  387. inactive-color="#a6a6a6"
  388. active-text="开"
  389. inactive-text="关"
  390. @change="multiTagsCacheChange"
  391. />
  392. </li>
  393. <li>
  394. <span>标签风格</span>
  395. <el-radio-group v-model="markValue" size="small" @change="onChange">
  396. <el-radio label="card">卡片</el-radio>
  397. <el-radio label="smart">灵动</el-radio>
  398. </el-radio-group>
  399. </li>
  400. </ul>
  401. <el-divider />
  402. <el-button
  403. type="danger"
  404. style="width: 90%; margin: 24px 15px"
  405. @click="onReset"
  406. >
  407. <IconifyIconOffline
  408. icon="fa-sign-out"
  409. width="15"
  410. height="15"
  411. style="margin-right: 4px"
  412. />
  413. 清空缓存并返回登录页</el-button
  414. >
  415. </panel>
  416. </template>
  417. <style scoped module>
  418. .isSelect {
  419. border: 2px solid var(--el-color-primary);
  420. }
  421. </style>
  422. <style lang="scss" scoped>
  423. .setting {
  424. width: 100%;
  425. li {
  426. display: flex;
  427. justify-content: space-between;
  428. align-items: center;
  429. margin: 25px;
  430. }
  431. }
  432. :deep(.el-divider__text) {
  433. font-size: 16px;
  434. font-weight: 700;
  435. }
  436. .pure-datatheme {
  437. width: 100%;
  438. height: 50px;
  439. text-align: center;
  440. display: block;
  441. padding-top: 25px;
  442. }
  443. .pure-theme {
  444. margin-top: 25px;
  445. width: 100%;
  446. height: 50px;
  447. display: flex;
  448. flex-wrap: wrap;
  449. justify-content: space-around;
  450. li {
  451. width: 18%;
  452. height: 45px;
  453. background: #f0f2f5;
  454. position: relative;
  455. overflow: hidden;
  456. cursor: pointer;
  457. border-radius: 4px;
  458. box-shadow: 0 1px 2.5px 0 rgb(0 0 0 / 18%);
  459. &:nth-child(1) {
  460. div {
  461. &:nth-child(1) {
  462. width: 30%;
  463. height: 100%;
  464. background: #1b2a47;
  465. }
  466. &:nth-child(2) {
  467. width: 70%;
  468. height: 30%;
  469. top: 0;
  470. right: 0;
  471. background: #fff;
  472. box-shadow: 0 0 1px #888;
  473. position: absolute;
  474. }
  475. }
  476. }
  477. &:nth-child(2) {
  478. div {
  479. &:nth-child(1) {
  480. width: 100%;
  481. height: 30%;
  482. background: #1b2a47;
  483. box-shadow: 0 0 1px #888;
  484. }
  485. }
  486. }
  487. &:nth-child(3) {
  488. div {
  489. &:nth-child(1) {
  490. width: 100%;
  491. height: 30%;
  492. background: #1b2a47;
  493. box-shadow: 0 0 1px #888;
  494. }
  495. &:nth-child(2) {
  496. width: 30%;
  497. height: 70%;
  498. bottom: 0;
  499. left: 0;
  500. background: #fff;
  501. box-shadow: 0 0 1px #888;
  502. position: absolute;
  503. }
  504. }
  505. }
  506. }
  507. }
  508. .theme-color {
  509. width: 100%;
  510. height: 40px;
  511. margin-top: 20px;
  512. display: flex;
  513. justify-content: center;
  514. li {
  515. float: left;
  516. width: 20px;
  517. height: 20px;
  518. margin-top: 8px;
  519. margin-right: 8px;
  520. font-weight: 700;
  521. text-align: center;
  522. border-radius: 2px;
  523. cursor: pointer;
  524. &:nth-child(2) {
  525. border: 1px solid #ddd;
  526. }
  527. }
  528. }
  529. </style>