365 lines
10 KiB
Vue
365 lines
10 KiB
Vue
<template>
|
|
<el-drawer v-model="showSettings" :withHeader="false" direction="rtl" size="300px">
|
|
<div class="setting-drawer-title">
|
|
<h3 class="drawer-title">{{ $t('layout.themeStyleSet') }}</h3>
|
|
</div>
|
|
<div class="setting-drawer-block-checbox">
|
|
<div class="item" @click="handleTheme('theme-dark')">
|
|
<img src="@/assets/images/dark.svg" alt="dark" />
|
|
<div v-if="sideTheme === 'theme-dark'" class="setting-drawer-block-checbox-selectIcon" style="display: block">
|
|
<el-icon><Check /></el-icon>
|
|
</div>
|
|
</div>
|
|
<div class="item" @click="handleTheme('theme-light')">
|
|
<img src="@/assets/images/light.svg" alt="light" />
|
|
<div v-if="sideTheme === 'theme-light'" class="setting-drawer-block-checbox-selectIcon" style="display: block">
|
|
<el-icon><Check /></el-icon>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="drawer-item">
|
|
<el-radio-group v-model="mode" size="small">
|
|
<el-radio label="dark">{{ $t('layout.darkMode') }}</el-radio>
|
|
<el-radio label="light">{{ $t('layout.lightMode') }}</el-radio>
|
|
<!-- <el-radio label="cafe">cafe</el-radio> -->
|
|
<!-- <el-radio label="contrast">contrast</el-radio> -->
|
|
</el-radio-group>
|
|
</div>
|
|
<!-- <div class="drawer-item">
|
|
<span>暗黑模式</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="isDark" class="mt-2" inline-prompt />
|
|
</span>
|
|
</div> -->
|
|
<!-- <h3 class="drawer-title">
|
|
{{ $t('layout.themeColor') }}
|
|
</h3> -->
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.themeColor') }}</span>
|
|
<span class="comp-style quick-color-wrap">
|
|
<!-- <span :style="{ 'background-color': item }" v-for="item in predefineColors" @change="themeChange(item)"></span> -->
|
|
<el-color-picker v-model="theme" :predefine="predefineColors" @change="themeChange" />
|
|
</span>
|
|
</div>
|
|
<el-divider />
|
|
|
|
<h3 class="drawer-title">{{ $t('layout.sysLayoutSet') }}</h3>
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.open') }} {{ $t('layout.topNav') }}</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="topNav" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.open') }} 标签页</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="tagsView" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.open') }} {{ $t('layout.bottomBar') }}</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="showFooter" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.openWatermark') }}</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="showWatermark" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
<!-- <div class="drawer-item">
|
|
<span>{{ $t('layout.fixed') }} Header</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="fixedHeader" class="drawer-switch" />
|
|
</span>
|
|
</div> -->
|
|
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.show') }} Logo</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="sidebarLogo" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
|
|
<div class="drawer-item">
|
|
<span>{{ $t('layout.dynamicTitle') }}</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="dynamicTitle" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
|
|
<div class="drawer-item">
|
|
<span>标签持久化</span>
|
|
<span class="comp-style">
|
|
<el-switch v-model="tabsPersist" class="drawer-switch" />
|
|
</span>
|
|
</div>
|
|
|
|
<el-divider />
|
|
|
|
<el-button type="primary" plain icon="DocumentAdd" @click="saveSetting">{{ $t('layout.saveConfig') }}</el-button>
|
|
<el-button plain icon="Refresh" @click="resetSetting">{{ $t('layout.resetConfig') }}</el-button>
|
|
</el-drawer>
|
|
</template>
|
|
|
|
<script setup>
|
|
import 'element-plus/theme-chalk/index.css'
|
|
import 'element-plus/theme-chalk/dark/css-vars.css'
|
|
import { useDark, useCycleList, useColorMode } from '@vueuse/core'
|
|
import { useDynamicTitle } from '@/utils/dynamicTitle'
|
|
import { getLightColor } from '@/utils/index'
|
|
import { getmark } from '@/utils/wartermark'
|
|
import useAppStore from '@/store/modules/app'
|
|
import useSettingsStore from '@/store/modules/settings'
|
|
import usePermissionStore from '@/store/modules/permission'
|
|
import useUserStore from '@/store/modules/user'
|
|
const { proxy } = getCurrentInstance()
|
|
const appStore = useAppStore()
|
|
const settingsStore = useSettingsStore()
|
|
const permissionStore = usePermissionStore()
|
|
const showSettings = ref(false)
|
|
const theme = ref(settingsStore.theme)
|
|
const sideTheme = ref(settingsStore.sideTheme)
|
|
const storeSettings = computed(() => settingsStore)
|
|
const predefineColors = ref(['#409EFF', '#ff4500', '#ff8c00', '#ffd700', '#90ee90', '#00ced1', '#1e90ff', '#c71585'])
|
|
const { setWatermark, removeWatermark } = getmark()
|
|
// 可以手动更改当前值 model.value = 'cafe'
|
|
const mode = useColorMode({
|
|
modes: {
|
|
// custom colors
|
|
contrast: 'dark contrast',
|
|
cafe: 'cafe'
|
|
}
|
|
})
|
|
const { next } = useCycleList(['light', 'dark', 'cafe', 'contrast'], { initialValue: mode })
|
|
// const isDark= useDark()
|
|
|
|
/** 是否需要topnav */
|
|
const topNav = computed({
|
|
get: () => storeSettings.value.topNav,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'topNav', value: val })
|
|
if (!val) {
|
|
appStore.toggleSideBarHide(false)
|
|
permissionStore.setSidebarRouters(permissionStore.defaultRoutes)
|
|
}
|
|
}
|
|
})
|
|
/** 是否需要tagview */
|
|
const tagsView = computed({
|
|
get: () => storeSettings.value.tagsView,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'tagsView', value: val })
|
|
}
|
|
})
|
|
/**是否需要固定头部 */
|
|
const fixedHeader = computed({
|
|
get: () => storeSettings.value.fixedHeader,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'fixedHeader', value: val })
|
|
}
|
|
})
|
|
// 是否显示底部
|
|
const showFooter = computed({
|
|
get: () => storeSettings.value.showFooter,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'showFooter', value: val })
|
|
}
|
|
})
|
|
/**是否需要侧边栏的logo */
|
|
const sidebarLogo = computed({
|
|
get: () => storeSettings.value.sidebarLogo,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'sidebarLogo', value: val })
|
|
}
|
|
})
|
|
/**是否需要侧边栏的动态网页的title */
|
|
const dynamicTitle = computed({
|
|
get: () => storeSettings.value.dynamicTitle,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'dynamicTitle', value: val })
|
|
// 动态设置网页标题
|
|
useDynamicTitle()
|
|
}
|
|
})
|
|
/**是否显示水印 */
|
|
const showWatermark = computed({
|
|
get: () => storeSettings.value.showWatermark,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'showWatermark', value: val })
|
|
changeWatermark()
|
|
}
|
|
})
|
|
|
|
/**标签持久化 */
|
|
const tabsPersist = computed({
|
|
get: () => storeSettings.value.tagsViewPersist,
|
|
set: (val) => {
|
|
settingsStore.changeSetting({ key: 'tagsViewPersist', value: val })
|
|
}
|
|
})
|
|
const changeWatermark = () => {
|
|
storeSettings.value.showWatermark ? setWatermark(useUserStore().userInfo.userName) : removeWatermark()
|
|
}
|
|
// 开启水印
|
|
changeWatermark()
|
|
// 监控主题颜色
|
|
watch(
|
|
() => theme,
|
|
(val) => {
|
|
themeChange(val.value)
|
|
},
|
|
{
|
|
immediate: true
|
|
}
|
|
)
|
|
watch(
|
|
() => sideTheme,
|
|
(val) => {
|
|
const body = document.documentElement
|
|
body.setAttribute('data-theme', '')
|
|
},
|
|
{
|
|
immediate: true
|
|
}
|
|
)
|
|
watch(
|
|
() => mode,
|
|
(val) => {
|
|
if (val.value === 'dark') {
|
|
handleTheme('')
|
|
}
|
|
},
|
|
{
|
|
immediate: true,
|
|
deep: true
|
|
}
|
|
)
|
|
/**
|
|
* 改变主题颜色
|
|
*/
|
|
function themeChange(val) {
|
|
settingsStore.changeSetting({ key: 'theme', value: val })
|
|
theme.value = val
|
|
// 设置element-plus ui主题
|
|
document.documentElement.style.setProperty('--el-color-primary', val)
|
|
|
|
// 颜色变浅
|
|
for (let i = 1; i <= 9; i++) {
|
|
document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, `${getLightColor(val, i / 10)}`)
|
|
}
|
|
}
|
|
function handleTheme(val) {
|
|
settingsStore.changeSetting({ key: 'sideTheme', value: val })
|
|
sideTheme.value = val
|
|
const body = document.documentElement
|
|
if (val == 'theme-black') body.setAttribute('data-theme', 'theme-black')
|
|
else body.removeAttribute('data-theme')
|
|
}
|
|
function saveSetting() {
|
|
proxy.$modal.loading('正在保存到本地,请稍候...')
|
|
let layoutSetting = {
|
|
topNav: storeSettings.value.topNav,
|
|
tagsView: storeSettings.value.tagsView,
|
|
fixedHeader: storeSettings.value.fixedHeader,
|
|
sidebarLogo: storeSettings.value.sidebarLogo,
|
|
dynamicTitle: storeSettings.value.dynamicTitle,
|
|
sideTheme: storeSettings.value.sideTheme,
|
|
theme: storeSettings.value.theme,
|
|
showFooter: storeSettings.value.showFooter,
|
|
showWatermark: storeSettings.value.showWatermark,
|
|
tagsViewPersist: storeSettings.value.tagsViewPersist
|
|
}
|
|
localStorage.setItem('layout-setting', JSON.stringify(layoutSetting))
|
|
setTimeout(proxy.$modal.closeLoading(), 100)
|
|
setTimeout('window.location.reload()', 200)
|
|
}
|
|
function resetSetting() {
|
|
proxy.$modal.loading('正在清除设置缓存并刷新,请稍候...')
|
|
localStorage.removeItem('layout-setting')
|
|
setTimeout('window.location.reload()', 1000)
|
|
}
|
|
function openSetting() {
|
|
showSettings.value = true
|
|
}
|
|
|
|
defineExpose({
|
|
openSetting
|
|
})
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.setting-drawer-title {
|
|
margin-bottom: 12px;
|
|
color: var(--base-text-color-rgba);
|
|
line-height: 22px;
|
|
font-weight: bold;
|
|
.drawer-title {
|
|
font-size: 14px;
|
|
}
|
|
}
|
|
.setting-drawer-block-checbox {
|
|
display: flex;
|
|
justify-content: flex-start;
|
|
align-items: center;
|
|
margin-top: 10px;
|
|
margin-bottom: 20px;
|
|
|
|
.item {
|
|
position: relative;
|
|
margin-right: 16px;
|
|
border-radius: 2px;
|
|
cursor: pointer;
|
|
|
|
img {
|
|
width: 48px;
|
|
height: 48px;
|
|
}
|
|
|
|
.custom-img {
|
|
width: 48px;
|
|
height: 38px;
|
|
border-radius: 5px;
|
|
box-shadow: 1px 1px 2px #898484;
|
|
}
|
|
|
|
.setting-drawer-block-checbox-selectIcon {
|
|
position: absolute;
|
|
top: 0;
|
|
right: 0;
|
|
width: 100%;
|
|
height: 100%;
|
|
padding-top: 15px;
|
|
padding-left: 24px;
|
|
color: #1890ff;
|
|
font-weight: 700;
|
|
font-size: 14px;
|
|
}
|
|
}
|
|
}
|
|
|
|
.drawer-item {
|
|
color: var(--base-text-color-rgba);
|
|
padding: 12px 0;
|
|
font-size: 14px;
|
|
|
|
.comp-style {
|
|
float: right;
|
|
margin: -3px 8px 0px 0px;
|
|
}
|
|
.quick-color-wrap {
|
|
display: flex;
|
|
align-items: center;
|
|
|
|
span {
|
|
width: 15px;
|
|
height: 15px;
|
|
margin-right: 10px;
|
|
cursor: pointer;
|
|
}
|
|
}
|
|
}
|
|
</style>
|