index.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. <template>
  2. <div>
  3. <!-- <div class="m-setting-fix">
  4. <div class="item">
  5. <div class="item-child" @click="operator(1)">
  6. <el-icon size="30" color="#3698fd" style="margin-bottom: 8px"><brush /></el-icon>
  7. 主题配置
  8. </div>
  9. <div class="item-child item-child2" @click="operator(2)">
  10. <el-icon size="30" color="#3698fd" style="margin-bottom: 8px"><Link /></el-icon>
  11. 源码拷贝
  12. </div>
  13. </div>
  14. </div> -->
  15. <el-drawer
  16. v-model="drawer" title="主题配置" size="300px">
  17. <div class="theme-item">
  18. <label>导航栏布局</label>
  19. <el-select
  20. v-model="layout"
  21. placeholder="请选择"
  22. style="width: 150px"
  23. @change="(val) => changeSwitch('mode',val)"
  24. >
  25. <el-option label="纵向" value="vertical"></el-option>
  26. <el-option label="横向" value="horizontal"></el-option>
  27. <el-option label="分栏" value="columns"></el-option>
  28. </el-select>
  29. </div>
  30. <div class="theme-item">
  31. <label>主题颜色</label>
  32. <el-color-picker v-model="primary" :predefine="predefineColor" @change="changePrimary" />
  33. </div>
  34. <div class="theme-item">
  35. <label>暗黑模式</label>
  36. <switch-dark></switch-dark>
  37. </div>
  38. <div class="theme-item">
  39. <label>灰色模式</label>
  40. <el-switch v-model="gray" @change="(val) => changeGrayWeak('gray',val)" />
  41. </div>
  42. <div class="theme-item">
  43. <label>色弱模式</label>
  44. <el-switch v-model="weak" @change="(val) => changeGrayWeak('weak',val)" />
  45. </div>
  46. <div class="theme-item">
  47. <label>标签栏</label>
  48. <el-switch v-model="showTag" @change="(val) => changeSwitch('showTag',val)" />
  49. </div>
  50. <div class="theme-item">
  51. <label>侧边栏 Logo</label>
  52. <el-switch v-model="showLogo" @change="(val) => changeSwitch('showLogo',val)" />
  53. </div>
  54. <div class="theme-item">
  55. <label>保持一个子菜单的展开</label>
  56. <el-switch v-model="uniqueOpened" @change="(val) => changeSwitch('uniqueOpened',val)" />
  57. </div>
  58. <div class="theme-item">
  59. <label>固定header</label>
  60. <el-switch v-model="fixedHeader" @change="(val) => changeSwitch('fixedHeader',val)" />
  61. </div>
  62. </el-drawer>
  63. </div>
  64. </template>
  65. <script lang="ts" setup>
  66. import {computed, ref,watch} from 'vue'
  67. import {ElMessage} from "element-plus";
  68. import {openLoading,closeLoading} from "@/utils/element"
  69. import SwitchDark from '@/components/SwitchDark/index.vue'
  70. import {PRIMARY_COLOR} from "@/config/index";
  71. import {useSettingStore} from "@/store/modules/setting"
  72. const SettingStore = useSettingStore()
  73. const layout = ref(SettingStore.themeConfig.mode)
  74. const showTag = ref(SettingStore.themeConfig.showTag)
  75. const showLogo = ref(SettingStore.themeConfig.showLogo)
  76. const uniqueOpened = ref(SettingStore.themeConfig.uniqueOpened)
  77. const primary = ref(SettingStore.themeConfig.primary)
  78. const fixedHeader = ref(SettingStore.themeConfig.fixedHeader)
  79. const gray = ref(SettingStore.themeConfig.gray)
  80. const weak = ref(SettingStore.themeConfig.weak)
  81. const drawer = computed({
  82. get() {
  83. return SettingStore.themeConfig.showSetting;
  84. },
  85. set() {
  86. changeSwitch('showSetting',!SettingStore.themeConfig.showSetting)
  87. }
  88. })
  89. // 预定义主题颜色
  90. const predefineColor = [
  91. '#409EFF', '#1890ff', '#304156','#212121','#11a983', '#13c2c2', '#6959CD', '#f5222d'
  92. ];
  93. const operator = (type) => {
  94. switch (type) {
  95. case 1:
  96. drawer.value = true
  97. return
  98. case 2:
  99. window.open('https://github.com/zouzhibin/vue-admin-perfect')
  100. return
  101. }
  102. }
  103. // 进行配置
  104. const changeSwitch = (key,val) => {
  105. SettingStore.setThemeConfig({key, val})
  106. if(key==='mode'){
  107. openLoading()
  108. setTimeout(()=>{
  109. closeLoading()
  110. },600)
  111. }
  112. }
  113. // 监听布局变化
  114. watch(
  115. () => layout.value,
  116. () => {
  117. const body = document.body as HTMLElement;
  118. body.setAttribute("class", `layout-${layout.value}`);
  119. },
  120. { immediate: true }
  121. );
  122. // 修改主题颜色
  123. const changePrimary = (val)=>{
  124. if (!val) {
  125. primary.value = val = PRIMARY_COLOR;
  126. ElMessage({ type: "success", message: `主题颜色已重置为 ${PRIMARY_COLOR}` });
  127. }
  128. document.documentElement.style.setProperty("--el-color-primary", val);
  129. changeSwitch('primary',val)
  130. }
  131. // 修改灰色模式
  132. const changeGrayWeak = (type,val)=>{
  133. const body = document.documentElement as HTMLElement;
  134. if (!val) return body.setAttribute("style", "");
  135. if (type === "gray") body.setAttribute("style", "filter: grayscale(1)");
  136. if (type === "weak") body.setAttribute("style", "filter: invert(80%)");
  137. changeSwitch(type,val)
  138. }
  139. </script>
  140. <style lang="scss" scoped>
  141. ::v-deep(.el-drawer__header){
  142. border-bottom: 1px solid #ebeef5;
  143. padding: 15px 20px 14px;
  144. margin-bottom: 0;
  145. }
  146. .m-setting-fix {
  147. position: fixed;
  148. top: 50%;
  149. right: 0;
  150. z-index: 999;
  151. padding: 10px 0 0 0;
  152. margin: 0;
  153. text-align: center;
  154. cursor: pointer;
  155. background: #fff;
  156. border: 1px solid #dcdfe6;
  157. border-top-left-radius: 5.5px;
  158. border-bottom-left-radius: 5.5px;
  159. box-shadow: 0 0 50px 0 rgb(82 63 105 / 15%);
  160. transform: translateY(-50%);
  161. .item {
  162. display: flex;
  163. flex-direction: column;
  164. align-items: center;
  165. justify-content: center;
  166. padding: 0 8px 10px 10px;
  167. margin: 0;
  168. list-style: none;
  169. }
  170. .item-child {
  171. color: #3698fd;
  172. width: 60px;
  173. height: 60px;
  174. /*padding-top: 10px;*/
  175. text-align: center;
  176. display: flex;
  177. flex-direction: column;
  178. background: #f6f8f9;
  179. align-items: center;
  180. justify-content: center;
  181. border-radius: 5.5px;
  182. font-size: 12px;
  183. background: #ebf5ff;
  184. transition: color 0.15s ease, background-color 0.15s ease, border-color 0.15s ease,
  185. box-shadow 0.15s ease;
  186. }
  187. .item-child2 {
  188. margin-top: 10px;
  189. color: #b37feb;
  190. background: #f7f2fd;
  191. transition: color 0.15s ease, background-color 0.15s ease, border-color 0.15s ease,
  192. box-shadow 0.15s ease;
  193. }
  194. }
  195. :deep(.el-drawer__title) {
  196. font-weight: bold;
  197. color: black;
  198. }
  199. .theme-item {
  200. width: 100%;
  201. display: flex;
  202. margin-bottom: 15px;
  203. align-items: center;
  204. font-size: 14px;
  205. color: black;
  206. justify-content: space-between;
  207. }
  208. </style>