'admin-23.04.11:发布v2.4.33版本,更新内容查看CHANGELOG.md'
This commit is contained in:
parent
8de54a844b
commit
8189fec5b4
15
CHANGELOG.md
15
CHANGELOG.md
@ -2,6 +2,19 @@
|
|||||||
|
|
||||||
🎉🎉🔥 `vue-next-admin` 基于 vue3.x 、Typescript、vite、Element plus 等,适配手机、平板、pc 的后台开源免费模板库(vue2.x 请切换 vue-prev-admin 分支)
|
🎉🎉🔥 `vue-next-admin` 基于 vue3.x 、Typescript、vite、Element plus 等,适配手机、平板、pc 的后台开源免费模板库(vue2.x 请切换 vue-prev-admin 分支)
|
||||||
|
|
||||||
|
## 2.4.33
|
||||||
|
|
||||||
|
`2023.04.11`
|
||||||
|
|
||||||
|
- 🌟 更新 依赖更新最新版本
|
||||||
|
- 🎉 新增 `/make/tableDemo` 中添加打印、图片预览功能
|
||||||
|
- 🐞 修复 菜单收起时(isCollapse),图标不居中问题
|
||||||
|
- 🐞 修复 演示 `权限管理 -> 前端控制 -> 页面权限` 切换不生效,感谢群友@傲世盛唐
|
||||||
|
- 🐞 修复 `"typescript": "5.x"` 中 `tsconfig.json`,`compilerOptions.suppressImplicitAnyIndexErrors` 弃用问题 [TypeScript/issues/51909](https://github.com/microsoft/TypeScript/issues/51909)、[suppressImplicitAnyIndexErrors](https://www.typescriptlang.org/tsconfig#suppressImplicitAnyIndexErrors)
|
||||||
|
- 🎨 合并 [!47cdn 打包支持 pnpm,消除无 external 的报错](https://gitee.com/lyt-top/vue-next-admin/commit/8de54a844bb54468d0bdccca158bf9bcb449f270),感谢[@yujiacheng](https://gitee.com/YujiaCheng1996)
|
||||||
|
- 🎯 优化 `layout/navBars/breadcrumb` 文件夹名称改成 `layout/navBars/topBar` 更易理解(`可全局替换`),感谢群友@傲世盛唐
|
||||||
|
- 🎯 优化 `layout/navBars/topBar/user.vue` 组件,`UserNews` 点击消息图标触发范围,改用 [element plus Popover 气泡卡片 虚拟触发方式](https://element-plus.org/zh-CN/component/popover.html#%E8%99%9A%E6%8B%9F%E8%A7%A6%E5%8F%91),防止点击消息通知背景色时不触发 `Popover` 弹出框
|
||||||
|
|
||||||
## 2.4.32
|
## 2.4.32
|
||||||
|
|
||||||
💔💔💔 图片不显示问题(README.md、演示中使用的图片,[vue-next-admin-images](https://gitee.com/lyt-top/vue-next-admin-images)),通过网站 [https://www.hd-r.cn/](https://www.hd-r.cn/) 转在线链接,如若侵权请联系作者 qq:1105290566
|
💔💔💔 图片不显示问题(README.md、演示中使用的图片,[vue-next-admin-images](https://gitee.com/lyt-top/vue-next-admin-images)),通过网站 [https://www.hd-r.cn/](https://www.hd-r.cn/) 转在线链接,如若侵权请联系作者 qq:1105290566
|
||||||
@ -150,7 +163,7 @@
|
|||||||
- 🎯 优化 图标选择器 icon type 类型为 all 时,类型 ali、ele、awe 回显问题
|
- 🎯 优化 图标选择器 icon type 类型为 all 时,类型 ali、ele、awe 回显问题
|
||||||
- 🎯 优化 去掉开发环境 i18n 控制台警告,页面代码:[i18n/index.ts](https://gitee.com/lyt-top/vue-next-admin/blob/master/src/i18n/index.ts)
|
- 🎯 优化 去掉开发环境 i18n 控制台警告,页面代码:[i18n/index.ts](https://gitee.com/lyt-top/vue-next-admin/blob/master/src/i18n/index.ts)
|
||||||
- 🎯 优化 `NextLoading.start()` 方法,防止第一次进入界面时出现短暂空白
|
- 🎯 优化 `NextLoading.start()` 方法,防止第一次进入界面时出现短暂空白
|
||||||
- 🎯 优化 地址栏有参数退出登录,再次登录不跳之前界面问题 `src/layout/navBars/breadcrumb/user.vue`
|
- 🎯 优化 地址栏有参数退出登录,再次登录不跳之前界面问题 `src/layout/navBars/topBar/user.vue`
|
||||||
- 🎯 优化 `SvgIcon` 组件,防止 `开启 Tagsview 图标` 时,`tagsView 右键菜单关闭` 报错问题,工作流不可连线、全屏时关闭按钮消失问题
|
- 🎯 优化 `SvgIcon` 组件,防止 `开启 Tagsview 图标` 时,`tagsView 右键菜单关闭` 报错问题,工作流不可连线、全屏时关闭按钮消失问题
|
||||||
- 🎯 优化 [如果 url 中有中文等特殊字符,第一次切换该 tab 时 keep-alive 失效#I55JS7](https://gitee.com/lyt-top/vue-next-admin/issues/I55JS7),感谢[yuyong1566](https://gitee.com/yuyong1566)
|
- 🎯 优化 [如果 url 中有中文等特殊字符,第一次切换该 tab 时 keep-alive 失效#I55JS7](https://gitee.com/lyt-top/vue-next-admin/issues/I55JS7),感谢[yuyong1566](https://gitee.com/yuyong1566)
|
||||||
- 🎯 优化 [wangEditor](https://www.wangeditor.com/) 更新到 v5,[vue3 版本线上示例中 wangeditor 富文本编辑器 demo 实例,无法换行#I5565B](https://gitee.com/lyt-top/vue-next-admin/issues/I5565B),感谢@[jenchih](https://gitee.com/jenchih)
|
- 🎯 优化 [wangEditor](https://www.wangeditor.com/) 更新到 v5,[vue3 版本线上示例中 wangeditor 富文本编辑器 demo 实例,无法换行#I5565B](https://gitee.com/lyt-top/vue-next-admin/issues/I5565B),感谢@[jenchih](https://gitee.com/jenchih)
|
||||||
|
851
package-lock.json
generated
851
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
24
package.json
24
package.json
@ -1,11 +1,11 @@
|
|||||||
{
|
{
|
||||||
"name": "vue-next-admin",
|
"name": "vue-next-admin",
|
||||||
"version": "2.4.32",
|
"version": "2.4.33",
|
||||||
"description": "vue3 vite next admin template",
|
"description": "vue3 vite next admin template",
|
||||||
"author": "lyt_20201208",
|
"author": "lyt_20201208",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite --force",
|
"dev": "vite",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"lint-fix": "eslint --fix --ext .js --ext .jsx --ext .vue src/"
|
"lint-fix": "eslint --fix --ext .js --ext .jsx --ext .vue src/"
|
||||||
},
|
},
|
||||||
@ -13,19 +13,19 @@
|
|||||||
"@element-plus/icons-vue": "^2.1.0",
|
"@element-plus/icons-vue": "^2.1.0",
|
||||||
"@wangeditor/editor": "^5.1.23",
|
"@wangeditor/editor": "^5.1.23",
|
||||||
"@wangeditor/editor-for-vue": "^5.1.12",
|
"@wangeditor/editor-for-vue": "^5.1.12",
|
||||||
"axios": "^1.3.4",
|
"axios": "^1.3.5",
|
||||||
"countup.js": "^2.6.0",
|
"countup.js": "^2.6.0",
|
||||||
"cropperjs": "^1.5.13",
|
"cropperjs": "^1.5.13",
|
||||||
"echarts": "^5.4.2",
|
"echarts": "^5.4.2",
|
||||||
"echarts-gl": "^2.0.9",
|
"echarts-gl": "^2.0.9",
|
||||||
"echarts-wordcloud": "^2.1.0",
|
"echarts-wordcloud": "^2.1.0",
|
||||||
"element-plus": "^2.3.1",
|
"element-plus": "^2.3.3",
|
||||||
"js-cookie": "^3.0.1",
|
"js-cookie": "^3.0.1",
|
||||||
"js-table2excel": "^1.0.3",
|
"js-table2excel": "^1.0.3",
|
||||||
"jsplumb": "^2.15.6",
|
"jsplumb": "^2.15.6",
|
||||||
"mitt": "^3.0.0",
|
"mitt": "^3.0.0",
|
||||||
"nprogress": "^0.2.0",
|
"nprogress": "^0.2.0",
|
||||||
"pinia": "^2.0.33",
|
"pinia": "^2.0.34",
|
||||||
"print-js": "^1.6.0",
|
"print-js": "^1.6.0",
|
||||||
"qrcodejs2-fixes": "^0.0.2",
|
"qrcodejs2-fixes": "^0.0.2",
|
||||||
"qs": "^6.11.1",
|
"qs": "^6.11.1",
|
||||||
@ -40,23 +40,23 @@
|
|||||||
"vue-router": "^4.1.6"
|
"vue-router": "^4.1.6"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^18.15.10",
|
"@types/node": "^18.15.11",
|
||||||
"@types/nprogress": "^0.2.0",
|
"@types/nprogress": "^0.2.0",
|
||||||
"@types/sortablejs": "^1.15.1",
|
"@types/sortablejs": "^1.15.1",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.56.0",
|
"@typescript-eslint/eslint-plugin": "^5.58.0",
|
||||||
"@typescript-eslint/parser": "^5.56.0",
|
"@typescript-eslint/parser": "^5.58.0",
|
||||||
"@vitejs/plugin-vue": "^4.1.0",
|
"@vitejs/plugin-vue": "^4.1.0",
|
||||||
"@vue/compiler-sfc": "^3.2.47",
|
"@vue/compiler-sfc": "^3.2.47",
|
||||||
"eslint": "^8.36.0",
|
"eslint": "^8.38.0",
|
||||||
"eslint-plugin-vue": "^9.10.0",
|
"eslint-plugin-vue": "^9.10.0",
|
||||||
"prettier": "^2.8.7",
|
"prettier": "^2.8.7",
|
||||||
"sass": "^1.60.0",
|
"sass": "^1.61.0",
|
||||||
"typescript": "^5.0.2",
|
"typescript": "^5.0.4",
|
||||||
"vite": "^4.2.1",
|
"vite": "^4.2.1",
|
||||||
"vite-plugin-cdn-import": "^0.3.5",
|
"vite-plugin-cdn-import": "^0.3.5",
|
||||||
"vite-plugin-compression": "^0.5.1",
|
"vite-plugin-compression": "^0.5.1",
|
||||||
"vite-plugin-vue-setup-extend-plus": "^0.1.0",
|
"vite-plugin-vue-setup-extend-plus": "^0.1.0",
|
||||||
"vue-eslint-parser": "^9.1.0"
|
"vue-eslint-parser": "^9.1.1"
|
||||||
},
|
},
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"> 1%",
|
"> 1%",
|
||||||
|
@ -23,8 +23,8 @@ import setIntroduction from '/@/utils/setIconfont';
|
|||||||
|
|
||||||
// 引入组件
|
// 引入组件
|
||||||
const LockScreen = defineAsyncComponent(() => import('/@/layout/lockScreen/index.vue'));
|
const LockScreen = defineAsyncComponent(() => import('/@/layout/lockScreen/index.vue'));
|
||||||
const Setings = defineAsyncComponent(() => import('/@/layout/navBars/breadcrumb/setings.vue'));
|
const Setings = defineAsyncComponent(() => import('/@/layout/navBars/topBar/setings.vue'));
|
||||||
const CloseFull = defineAsyncComponent(() => import('/@/layout/navBars/breadcrumb/closeFull.vue'));
|
const CloseFull = defineAsyncComponent(() => import('/@/layout/navBars/topBar/closeFull.vue'));
|
||||||
const Upgrade = defineAsyncComponent(() => import('/@/layout/upgrade/index.vue'));
|
const Upgrade = defineAsyncComponent(() => import('/@/layout/upgrade/index.vue'));
|
||||||
const Sponsors = defineAsyncComponent(() => import('/@/layout/sponsors/index.vue'));
|
const Sponsors = defineAsyncComponent(() => import('/@/layout/sponsors/index.vue'));
|
||||||
|
|
||||||
|
@ -22,7 +22,14 @@
|
|||||||
>
|
>
|
||||||
<template v-slot="scope">
|
<template v-slot="scope">
|
||||||
<template v-if="item.type === 'image'">
|
<template v-if="item.type === 'image'">
|
||||||
<img :src="scope.row[item.key]" :width="item.width" :height="item.height" />
|
<el-image
|
||||||
|
:style="{ width: `${item.width}px`, height: `${item.height}px` }"
|
||||||
|
:src="scope.row[item.key]"
|
||||||
|
:zoom-rate="1.2"
|
||||||
|
:preview-src-list="[scope.row[item.key]]"
|
||||||
|
preview-teleported
|
||||||
|
fit="cover"
|
||||||
|
/>
|
||||||
</template>
|
</template>
|
||||||
<template v-else>
|
<template v-else>
|
||||||
{{ scope.row[item.key] }}
|
{{ scope.row[item.key] }}
|
||||||
@ -56,6 +63,7 @@
|
|||||||
>
|
>
|
||||||
</el-pagination>
|
</el-pagination>
|
||||||
<div class="table-footer-tool">
|
<div class="table-footer-tool">
|
||||||
|
<SvgIcon name="iconfont icon-dayin" :size="19" title="打印" @click="onPrintTable" />
|
||||||
<SvgIcon name="iconfont icon-yunxiazai_o" :size="22" title="导出" @click="onImportTable" />
|
<SvgIcon name="iconfont icon-yunxiazai_o" :size="22" title="导出" @click="onImportTable" />
|
||||||
<SvgIcon name="iconfont icon-shuaxin" :size="22" title="刷新" @click="onRefreshTable" />
|
<SvgIcon name="iconfont icon-shuaxin" :size="22" title="刷新" @click="onRefreshTable" />
|
||||||
<el-popover
|
<el-popover
|
||||||
@ -103,6 +111,7 @@
|
|||||||
<script setup lang="ts" name="netxTable">
|
<script setup lang="ts" name="netxTable">
|
||||||
import { reactive, computed, nextTick, ref } from 'vue';
|
import { reactive, computed, nextTick, ref } from 'vue';
|
||||||
import { ElMessage } from 'element-plus';
|
import { ElMessage } from 'element-plus';
|
||||||
|
import printJs from 'print-js';
|
||||||
import table2excel from 'js-table2excel';
|
import table2excel from 'js-table2excel';
|
||||||
import Sortable from 'sortablejs';
|
import Sortable from 'sortablejs';
|
||||||
import { storeToRefs } from 'pinia';
|
import { storeToRefs } from 'pinia';
|
||||||
@ -126,6 +135,11 @@ const props = defineProps({
|
|||||||
type: Object,
|
type: Object,
|
||||||
default: () => {},
|
default: () => {},
|
||||||
},
|
},
|
||||||
|
// 打印标题
|
||||||
|
printName: {
|
||||||
|
type: String,
|
||||||
|
default: () => '',
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
// 定义子组件向父组件传值/事件
|
// 定义子组件向父组件传值/事件
|
||||||
@ -193,6 +207,37 @@ const pageReset = () => {
|
|||||||
state.page.pageSize = 10;
|
state.page.pageSize = 10;
|
||||||
emit('pageChange', state.page);
|
emit('pageChange', state.page);
|
||||||
};
|
};
|
||||||
|
// 打印
|
||||||
|
const onPrintTable = () => {
|
||||||
|
// https://printjs.crabbly.com/#documentation
|
||||||
|
// 自定义打印
|
||||||
|
let tableTh = '';
|
||||||
|
let tableTrTd = '';
|
||||||
|
let tableTd: any = {};
|
||||||
|
// 表头
|
||||||
|
props.header.forEach((v) => {
|
||||||
|
tableTh += `<th class="table-th">${v.title}</th>`;
|
||||||
|
});
|
||||||
|
// 表格内容
|
||||||
|
props.data.forEach((val, key) => {
|
||||||
|
if (!tableTd[key]) tableTd[key] = [];
|
||||||
|
props.header.forEach((v) => {
|
||||||
|
if (v.type === 'text') {
|
||||||
|
tableTd[key].push(`<td class="table-th table-center">${val[v.key]}</td>`);
|
||||||
|
} else if (v.type === 'image') {
|
||||||
|
tableTd[key].push(`<td class="table-th table-center"><img src="${val[v.key]}" style="width:${v.width}px;height:${v.height}px;"/></td>`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
tableTrTd += `<tr>${tableTd[key].join('')}</tr>`;
|
||||||
|
});
|
||||||
|
// 打印
|
||||||
|
printJs({
|
||||||
|
printable: `<div style=display:flex;flex-direction:column;text-align:center><h3>${props.printName}</h3></div><table border=1 cellspacing=0><tr>${tableTh}${tableTrTd}</table>`,
|
||||||
|
type: 'raw-html',
|
||||||
|
css: ['//at.alicdn.com/t/c/font_2298093_rnp72ifj3ba.css', '//unpkg.com/element-plus/dist/index.css'],
|
||||||
|
style: `@media print{.mb15{margin-bottom:15px;}.el-button--small i.iconfont{font-size: 12px !important;margin-right: 5px;}}; .table-th{word-break: break-all;white-space: pre-wrap;}.table-center{text-align: center;}`,
|
||||||
|
});
|
||||||
|
};
|
||||||
// 导出
|
// 导出
|
||||||
const onImportTable = () => {
|
const onImportTable = () => {
|
||||||
if (state.selectlist.length <= 0) return ElMessage.warning('请先选择要导出的数据');
|
if (state.selectlist.length <= 0) return ElMessage.warning('请先选择要导出的数据');
|
||||||
@ -211,7 +256,7 @@ const onSetTable = () => {
|
|||||||
animation: 150,
|
animation: 150,
|
||||||
onEnd: () => {
|
onEnd: () => {
|
||||||
const headerList: EmptyObjectType[] = [];
|
const headerList: EmptyObjectType[] = [];
|
||||||
sortable.toArray().forEach((val) => {
|
sortable.toArray().forEach((val: string) => {
|
||||||
props.header.forEach((v) => {
|
props.header.forEach((v) => {
|
||||||
if (v.key === val) headerList.push({ ...v });
|
if (v.key === val) headerList.push({ ...v });
|
||||||
});
|
});
|
||||||
|
@ -146,6 +146,12 @@ watch(
|
|||||||
if (layoutAsideScrollbarRef.value) layoutAsideScrollbarRef.value.update();
|
if (layoutAsideScrollbarRef.value) layoutAsideScrollbarRef.value.update();
|
||||||
}
|
}
|
||||||
if (layout === 'classic' && isClassicSplitMenu) return false;
|
if (layout === 'classic' && isClassicSplitMenu) return false;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
// 监听用户权限切换,用于演示 `权限管理 -> 前端控制 -> 页面权限` 权限切换不生效
|
||||||
|
watch(
|
||||||
|
() => routesList.value,
|
||||||
|
() => {
|
||||||
setFilterRoutes();
|
setFilterRoutes();
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -11,7 +11,7 @@ import { storeToRefs } from 'pinia';
|
|||||||
import { useThemeConfig } from '/@/stores/themeConfig';
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
|
||||||
// 引入组件
|
// 引入组件
|
||||||
const BreadcrumbIndex = defineAsyncComponent(() => import('/@/layout/navBars/breadcrumb/index.vue'));
|
const BreadcrumbIndex = defineAsyncComponent(() => import('/@/layout/navBars/topBar/index.vue'));
|
||||||
const TagsView = defineAsyncComponent(() => import('/@/layout/navBars/tagsView/tagsView.vue'));
|
const TagsView = defineAsyncComponent(() => import('/@/layout/navBars/tagsView/tagsView.vue'));
|
||||||
|
|
||||||
// 定义变量内容
|
// 定义变量内容
|
||||||
|
146
src/layout/navBars/topBar/breadcrumb.vue
Normal file
146
src/layout/navBars/topBar/breadcrumb.vue
Normal file
@ -0,0 +1,146 @@
|
|||||||
|
<template>
|
||||||
|
<div v-if="isShowBreadcrumb" class="layout-navbars-breadcrumb">
|
||||||
|
<SvgIcon
|
||||||
|
class="layout-navbars-breadcrumb-icon"
|
||||||
|
:name="themeConfig.isCollapse ? 'ele-Expand' : 'ele-Fold'"
|
||||||
|
:size="16"
|
||||||
|
@click="onThemeConfigChange"
|
||||||
|
/>
|
||||||
|
<el-breadcrumb class="layout-navbars-breadcrumb-hide">
|
||||||
|
<transition-group name="breadcrumb">
|
||||||
|
<el-breadcrumb-item v-for="(v, k) in state.breadcrumbList" :key="!v.meta.tagsViewName ? v.meta.title : v.meta.tagsViewName">
|
||||||
|
<span v-if="k === state.breadcrumbList.length - 1" class="layout-navbars-breadcrumb-span">
|
||||||
|
<SvgIcon :name="v.meta.icon" class="layout-navbars-breadcrumb-iconfont" v-if="themeConfig.isBreadcrumbIcon" />
|
||||||
|
<div v-if="!v.meta.tagsViewName">{{ $t(v.meta.title) }}</div>
|
||||||
|
<div v-else>{{ v.meta.tagsViewName }}</div>
|
||||||
|
</span>
|
||||||
|
<a v-else @click.prevent="onBreadcrumbClick(v)">
|
||||||
|
<SvgIcon :name="v.meta.icon" class="layout-navbars-breadcrumb-iconfont" v-if="themeConfig.isBreadcrumbIcon" />{{ $t(v.meta.title) }}
|
||||||
|
</a>
|
||||||
|
</el-breadcrumb-item>
|
||||||
|
</transition-group>
|
||||||
|
</el-breadcrumb>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumb">
|
||||||
|
import { reactive, computed, onMounted } from 'vue';
|
||||||
|
import { onBeforeRouteUpdate, useRoute, useRouter } from 'vue-router';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const route = useRoute();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<BreadcrumbState>({
|
||||||
|
breadcrumbList: [],
|
||||||
|
routeSplit: [],
|
||||||
|
routeSplitFirst: '',
|
||||||
|
routeSplitIndex: 1,
|
||||||
|
});
|
||||||
|
|
||||||
|
// 动态设置经典、横向布局不显示
|
||||||
|
const isShowBreadcrumb = computed(() => {
|
||||||
|
initRouteSplit(route.path);
|
||||||
|
const { layout, isBreadcrumb } = themeConfig.value;
|
||||||
|
if (layout === 'classic' || layout === 'transverse') return false;
|
||||||
|
else return isBreadcrumb ? true : false;
|
||||||
|
});
|
||||||
|
// 面包屑点击时
|
||||||
|
const onBreadcrumbClick = (v: RouteItem) => {
|
||||||
|
const { redirect, path } = v;
|
||||||
|
if (redirect) router.push(redirect);
|
||||||
|
else router.push(path);
|
||||||
|
};
|
||||||
|
// 展开/收起左侧菜单点击
|
||||||
|
const onThemeConfigChange = () => {
|
||||||
|
themeConfig.value.isCollapse = !themeConfig.value.isCollapse;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 存储布局配置
|
||||||
|
const setLocalThemeConfig = () => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
};
|
||||||
|
// 处理面包屑数据
|
||||||
|
const getBreadcrumbList = (arr: RouteItems) => {
|
||||||
|
arr.forEach((item: RouteItem) => {
|
||||||
|
state.routeSplit.forEach((v: string, k: number, arrs: string[]) => {
|
||||||
|
if (state.routeSplitFirst === item.path) {
|
||||||
|
state.routeSplitFirst += `/${arrs[state.routeSplitIndex]}`;
|
||||||
|
state.breadcrumbList.push(item);
|
||||||
|
state.routeSplitIndex++;
|
||||||
|
if (item.children) getBreadcrumbList(item.children);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 当前路由字符串切割成数组,并删除第一项空内容
|
||||||
|
const initRouteSplit = (path: string) => {
|
||||||
|
if (!themeConfig.value.isBreadcrumb) return false;
|
||||||
|
state.breadcrumbList = [routesList.value[0]];
|
||||||
|
state.routeSplit = path.split('/');
|
||||||
|
state.routeSplit.shift();
|
||||||
|
state.routeSplitFirst = `/${state.routeSplit[0]}`;
|
||||||
|
state.routeSplitIndex = 1;
|
||||||
|
getBreadcrumbList(routesList.value);
|
||||||
|
if (route.name === 'home' || (route.name === 'notFound' && state.breadcrumbList.length > 1)) state.breadcrumbList.shift();
|
||||||
|
if (state.breadcrumbList.length > 0)
|
||||||
|
state.breadcrumbList[state.breadcrumbList.length - 1].meta.tagsViewName = other.setTagsViewNameI18n(<RouteToFrom>route);
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
initRouteSplit(route.path);
|
||||||
|
});
|
||||||
|
// 路由更新时
|
||||||
|
onBeforeRouteUpdate((to) => {
|
||||||
|
initRouteSplit(to.path);
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb {
|
||||||
|
flex: 1;
|
||||||
|
height: inherit;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
.layout-navbars-breadcrumb-icon {
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 18px;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
height: 100%;
|
||||||
|
width: 40px;
|
||||||
|
opacity: 0.8;
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-navbars-breadcrumb-span {
|
||||||
|
display: flex;
|
||||||
|
opacity: 0.7;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
.layout-navbars-breadcrumb-iconfont {
|
||||||
|
font-size: 14px;
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
:deep(.el-breadcrumb__separator) {
|
||||||
|
opacity: 0.7;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
:deep(.el-breadcrumb__inner a, .el-breadcrumb__inner.is-link) {
|
||||||
|
font-weight: unset !important;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
&:hover {
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
53
src/layout/navBars/topBar/closeFull.vue
Normal file
53
src/layout/navBars/topBar/closeFull.vue
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-close-full" v-if="isTagsViewCurrenFull">
|
||||||
|
<div class="layout-navbars-close-full-icon">
|
||||||
|
<SvgIcon name="ele-Close" :title="$t('message.tagsView.closeFullscreen')" @click="onCloseFullscreen" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutCloseFull">
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const stores = useTagsViewRoutes();
|
||||||
|
const { isTagsViewCurrenFull } = storeToRefs(stores);
|
||||||
|
|
||||||
|
// 关闭当前全屏
|
||||||
|
const onCloseFullscreen = () => {
|
||||||
|
stores.setCurrenFullscreen(false);
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-close-full {
|
||||||
|
position: fixed;
|
||||||
|
z-index: 9999999999;
|
||||||
|
right: -30px;
|
||||||
|
top: -30px;
|
||||||
|
.layout-navbars-close-full-icon {
|
||||||
|
width: 60px;
|
||||||
|
height: 60px;
|
||||||
|
border-radius: 100%;
|
||||||
|
cursor: pointer;
|
||||||
|
background: rgba(0, 0, 0, 0.1);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
position: relative;
|
||||||
|
:deep(i) {
|
||||||
|
position: absolute;
|
||||||
|
left: 10px;
|
||||||
|
top: 35px;
|
||||||
|
color: #333333;
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
:deep(i) {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
107
src/layout/navBars/topBar/index.vue
Normal file
107
src/layout/navBars/topBar/index.vue
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-index">
|
||||||
|
<Logo v-if="setIsShowLogo" />
|
||||||
|
<Breadcrumb />
|
||||||
|
<Horizontal :menuList="state.menuList" v-if="isLayoutTransverse" />
|
||||||
|
<User />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumbIndex">
|
||||||
|
import { defineAsyncComponent, computed, reactive, onMounted, onUnmounted } from 'vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useRoutesList } from '/@/stores/routesList';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import mittBus from '/@/utils/mitt';
|
||||||
|
|
||||||
|
// 引入组件
|
||||||
|
const Breadcrumb = defineAsyncComponent(() => import('/@/layout/navBars/topBar/breadcrumb.vue'));
|
||||||
|
const User = defineAsyncComponent(() => import('/@/layout/navBars/topBar/user.vue'));
|
||||||
|
const Logo = defineAsyncComponent(() => import('/@/layout/logo/index.vue'));
|
||||||
|
const Horizontal = defineAsyncComponent(() => import('/@/layout/navMenu/horizontal.vue'));
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const stores = useRoutesList();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { routesList } = storeToRefs(stores);
|
||||||
|
const route = useRoute();
|
||||||
|
const state = reactive({
|
||||||
|
menuList: [] as RouteItems,
|
||||||
|
});
|
||||||
|
|
||||||
|
// 设置 logo 显示/隐藏
|
||||||
|
const setIsShowLogo = computed(() => {
|
||||||
|
let { isShowLogo, layout } = themeConfig.value;
|
||||||
|
return (isShowLogo && layout === 'classic') || (isShowLogo && layout === 'transverse');
|
||||||
|
});
|
||||||
|
// 设置是否显示横向导航菜单
|
||||||
|
const isLayoutTransverse = computed(() => {
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
return layout === 'transverse' || (isClassicSplitMenu && layout === 'classic');
|
||||||
|
});
|
||||||
|
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
||||||
|
const setFilterRoutes = () => {
|
||||||
|
let { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
if (layout === 'classic' && isClassicSplitMenu) {
|
||||||
|
state.menuList = delClassicChildren(filterRoutesFun(routesList.value));
|
||||||
|
const resData = setSendClassicChildren(route.path);
|
||||||
|
mittBus.emit('setSendClassicChildren', resData);
|
||||||
|
} else {
|
||||||
|
state.menuList = filterRoutesFun(routesList.value);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 设置了分割菜单时,删除底下 children
|
||||||
|
const delClassicChildren = <T extends ChilType>(arr: T[]): T[] => {
|
||||||
|
arr.map((v: T) => {
|
||||||
|
if (v.children) delete v.children;
|
||||||
|
});
|
||||||
|
return arr;
|
||||||
|
};
|
||||||
|
// 路由过滤递归函数
|
||||||
|
const filterRoutesFun = <T extends RouteItem>(arr: T[]): T[] => {
|
||||||
|
return arr
|
||||||
|
.filter((item: T) => !item.meta?.isHide)
|
||||||
|
.map((item: T) => {
|
||||||
|
item = Object.assign({}, item);
|
||||||
|
if (item.children) item.children = filterRoutesFun(item.children);
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 传送当前子级数据到菜单中
|
||||||
|
const setSendClassicChildren = (path: string) => {
|
||||||
|
const currentPathSplit = path.split('/');
|
||||||
|
let currentData: MittMenu = { children: [] };
|
||||||
|
filterRoutesFun(routesList.value).map((v: RouteItem, k: number) => {
|
||||||
|
if (v.path === `/${currentPathSplit[1]}`) {
|
||||||
|
v['k'] = k;
|
||||||
|
currentData['item'] = { ...v };
|
||||||
|
currentData['children'] = [{ ...v }];
|
||||||
|
if (v.children) currentData['children'] = v.children;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return currentData;
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
setFilterRoutes();
|
||||||
|
mittBus.on('getBreadcrumbIndexSetFilterRoutes', () => {
|
||||||
|
setFilterRoutes();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 页面卸载时
|
||||||
|
onUnmounted(() => {
|
||||||
|
mittBus.off('getBreadcrumbIndexSetFilterRoutes', () => {});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-index {
|
||||||
|
height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
background: var(--next-bg-topBar);
|
||||||
|
border-bottom: 1px solid var(--next-border-color-light);
|
||||||
|
}
|
||||||
|
</style>
|
125
src/layout/navBars/topBar/search.vue
Normal file
125
src/layout/navBars/topBar/search.vue
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-search-dialog">
|
||||||
|
<el-dialog v-model="state.isShowSearch" destroy-on-close :show-close="false">
|
||||||
|
<template #footer>
|
||||||
|
<el-autocomplete
|
||||||
|
v-model="state.menuQuery"
|
||||||
|
:fetch-suggestions="menuSearch"
|
||||||
|
:placeholder="$t('message.user.searchPlaceholder')"
|
||||||
|
ref="layoutMenuAutocompleteRef"
|
||||||
|
@select="onHandleSelect"
|
||||||
|
:fit-input-width="true"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<el-icon class="el-input__icon">
|
||||||
|
<ele-Search />
|
||||||
|
</el-icon>
|
||||||
|
</template>
|
||||||
|
<template #default="{ item }">
|
||||||
|
<div>
|
||||||
|
<SvgIcon :name="item.meta.icon" class="mr5" />
|
||||||
|
{{ $t(item.meta.title) }}
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-autocomplete>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumbSearch">
|
||||||
|
import { reactive, ref, nextTick } from 'vue';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const storesTagsViewRoutes = useTagsViewRoutes();
|
||||||
|
const { tagsViewRoutes } = storeToRefs(storesTagsViewRoutes);
|
||||||
|
const layoutMenuAutocompleteRef = ref();
|
||||||
|
const { t } = useI18n();
|
||||||
|
const router = useRouter();
|
||||||
|
const state = reactive<SearchState>({
|
||||||
|
isShowSearch: false,
|
||||||
|
menuQuery: '',
|
||||||
|
tagsViewList: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
// 搜索弹窗打开
|
||||||
|
const openSearch = () => {
|
||||||
|
state.menuQuery = '';
|
||||||
|
state.isShowSearch = true;
|
||||||
|
initTageView();
|
||||||
|
nextTick(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
layoutMenuAutocompleteRef.value.focus();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 搜索弹窗关闭
|
||||||
|
const closeSearch = () => {
|
||||||
|
state.isShowSearch = false;
|
||||||
|
};
|
||||||
|
// 菜单搜索数据过滤
|
||||||
|
const menuSearch = (queryString: string, cb: Function) => {
|
||||||
|
let results = queryString ? state.tagsViewList.filter(createFilter(queryString)) : state.tagsViewList;
|
||||||
|
cb(results);
|
||||||
|
};
|
||||||
|
// 菜单搜索过滤
|
||||||
|
const createFilter = (queryString: string) => {
|
||||||
|
return (restaurant: RouteItem) => {
|
||||||
|
return (
|
||||||
|
restaurant.path.toLowerCase().indexOf(queryString.toLowerCase()) > -1 ||
|
||||||
|
restaurant.meta!.title!.toLowerCase().indexOf(queryString.toLowerCase()) > -1 ||
|
||||||
|
t(restaurant.meta!.title!).indexOf(queryString.toLowerCase()) > -1
|
||||||
|
);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
// 初始化菜单数据
|
||||||
|
const initTageView = () => {
|
||||||
|
if (state.tagsViewList.length > 0) return false;
|
||||||
|
tagsViewRoutes.value.map((v: RouteItem) => {
|
||||||
|
if (!v.meta?.isHide) state.tagsViewList.push({ ...v });
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 当前菜单选中时
|
||||||
|
const onHandleSelect = (item: RouteItem) => {
|
||||||
|
let { path, redirect } = item;
|
||||||
|
if (item.meta?.isLink && !item.meta?.isIframe) window.open(item.meta?.isLink);
|
||||||
|
else if (redirect) router.push(redirect);
|
||||||
|
else router.push(path);
|
||||||
|
closeSearch();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 暴露变量
|
||||||
|
defineExpose({
|
||||||
|
openSearch,
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-search-dialog {
|
||||||
|
position: relative;
|
||||||
|
:deep(.el-dialog) {
|
||||||
|
.el-dialog__header,
|
||||||
|
.el-dialog__body {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
.el-dialog__footer {
|
||||||
|
width: 100%;
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
top: -53vh;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-autocomplete) {
|
||||||
|
width: 560px;
|
||||||
|
position: absolute;
|
||||||
|
top: 150px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
826
src/layout/navBars/topBar/setings.vue
Normal file
826
src/layout/navBars/topBar/setings.vue
Normal file
@ -0,0 +1,826 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-breadcrumb-seting">
|
||||||
|
<el-drawer
|
||||||
|
:title="$t('message.layout.configTitle')"
|
||||||
|
v-model="getThemeConfig.isDrawer"
|
||||||
|
direction="rtl"
|
||||||
|
destroy-on-close
|
||||||
|
size="260px"
|
||||||
|
@close="onDrawerClose"
|
||||||
|
>
|
||||||
|
<el-scrollbar class="layout-breadcrumb-seting-bar">
|
||||||
|
<!-- 全局主题 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.oneTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">primary</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.primary" size="default" @change="onColorPickerChange"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsDark') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isIsDark" size="small" @change="onAddDarkChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 顶栏设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.twoTopTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoTopBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.topBar" size="default" @change="onBgColorPickerChange('topBar')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoTopBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.topBarColor" size="default" @change="onBgColorPickerChange('topBarColor')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt10">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsTopBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTopBarColorGradual" size="small" @change="onTopBarGradualChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 菜单设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.twoMenuTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoMenuBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.menuBar" size="default" @change="onBgColorPickerChange('menuBar')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoMenuBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker v-model="getThemeConfig.menuBarColor" size="default" @change="onBgColorPickerChange('menuBarColor')"> </el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoMenuBarActiveColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker
|
||||||
|
v-model="getThemeConfig.menuBarActiveColor"
|
||||||
|
size="default"
|
||||||
|
show-alpha
|
||||||
|
@change="onBgColorPickerChange('menuBarActiveColor')"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsMenuBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isMenuBarColorGradual" size="small" @change="onMenuBarGradualChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 分栏设置 -->
|
||||||
|
<el-divider content-position="left" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">{{
|
||||||
|
$t('message.layout.twoColumnsTitle')
|
||||||
|
}}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoColumnsMenuBar') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker
|
||||||
|
v-model="getThemeConfig.columnsMenuBar"
|
||||||
|
size="default"
|
||||||
|
@change="onBgColorPickerChange('columnsMenuBar')"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
>
|
||||||
|
</el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoColumnsMenuBarColor') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-color-picker
|
||||||
|
v-model="getThemeConfig.columnsMenuBarColor"
|
||||||
|
size="default"
|
||||||
|
@change="onBgColorPickerChange('columnsMenuBarColor')"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
>
|
||||||
|
</el-color-picker>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsColumnsMenuBarColorGradual') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isColumnsMenuBarColorGradual"
|
||||||
|
size="small"
|
||||||
|
@change="onColumnsMenuBarGradualChange"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.twoIsColumnsMenuHoverPreload') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isColumnsMenuHoverPreload"
|
||||||
|
size="small"
|
||||||
|
@change="onColumnsMenuHoverPreloadChange"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns'"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 界面设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.threeTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex" :style="{ opacity: getThemeConfig.layout === 'transverse' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsCollapse') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isCollapse"
|
||||||
|
:disabled="getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="onThemeConfigChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout === 'transverse' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsUniqueOpened') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isUniqueOpened"
|
||||||
|
:disabled="getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsFixedHeader') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isFixedHeader" size="small" @change="onIsFixedHeaderChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout !== 'classic' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsClassicSplitMenu') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isClassicSplitMenu"
|
||||||
|
:disabled="getThemeConfig.layout !== 'classic'"
|
||||||
|
size="small"
|
||||||
|
@change="onClassicSplitMenuChange"
|
||||||
|
>
|
||||||
|
</el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeIsLockScreen') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isLockScreen" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt11">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.threeLockScreenTime') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-input-number
|
||||||
|
v-model="getThemeConfig.lockScreenTime"
|
||||||
|
controls-position="right"
|
||||||
|
:min="1"
|
||||||
|
:max="9999"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
>
|
||||||
|
</el-input-number>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 界面显示 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.fourTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsShowLogo') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isShowLogo" size="small" @change="onIsShowLogoChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="layout-breadcrumb-seting-bar-flex mt15"
|
||||||
|
:style="{ opacity: getThemeConfig.layout === 'classic' || getThemeConfig.layout === 'transverse' ? 0.5 : 1 }"
|
||||||
|
>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsBreadcrumb') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isBreadcrumb"
|
||||||
|
:disabled="getThemeConfig.layout === 'classic' || getThemeConfig.layout === 'transverse'"
|
||||||
|
size="small"
|
||||||
|
@change="onIsBreadcrumbChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsBreadcrumbIcon') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isBreadcrumbIcon" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsTagsview') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTagsview" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsTagsviewIcon') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isTagsviewIcon" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsCacheTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isCacheTagsView" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: state.isMobile ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsSortableTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch
|
||||||
|
v-model="getThemeConfig.isSortableTagsView"
|
||||||
|
:disabled="state.isMobile ? true : false"
|
||||||
|
size="small"
|
||||||
|
@change="onSortableTagsViewChange"
|
||||||
|
></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsShareTagsView') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isShareTagsView" size="small" @change="onShareTagsViewChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsFooter') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isFooter" size="small" @change="setLocalThemeConfig"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsGrayscale') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isGrayscale" size="small" @change="onAddFilterChange('grayscale')"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsInvert') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isInvert" size="small" @change="onAddFilterChange('invert')"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourIsWartermark') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-switch v-model="getThemeConfig.isWartermark" size="small" @change="onWartermarkChange"></el-switch>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt14">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fourWartermarkText') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-input v-model="getThemeConfig.wartermarkText" size="default" style="width: 90px" @input="onWartermarkTextInput"></el-input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 其它设置 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.fiveTitle') }}</el-divider>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveTagsStyle') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select v-model="getThemeConfig.tagsStyle" placeholder="请选择" size="default" style="width: 90px" @change="setLocalThemeConfig">
|
||||||
|
<el-option label="风格1" value="tags-style-one"></el-option>
|
||||||
|
<el-option label="风格4" value="tags-style-four"></el-option>
|
||||||
|
<el-option label="风格5" value="tags-style-five"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveAnimation') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select v-model="getThemeConfig.animation" placeholder="请选择" size="default" style="width: 90px" @change="setLocalThemeConfig">
|
||||||
|
<el-option label="slide-right" value="slide-right"></el-option>
|
||||||
|
<el-option label="slide-left" value="slide-left"></el-option>
|
||||||
|
<el-option label="opacitys" value="opacitys"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveColumnsAsideStyle') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select
|
||||||
|
v-model="getThemeConfig.columnsAsideStyle"
|
||||||
|
placeholder="请选择"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns' ? true : false"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
>
|
||||||
|
<el-option label="圆角" value="columns-round"></el-option>
|
||||||
|
<el-option label="卡片" value="columns-card"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex mt15 mb27" :style="{ opacity: getThemeConfig.layout !== 'columns' ? 0.5 : 1 }">
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-label">{{ $t('message.layout.fiveColumnsAsideLayout') }}</div>
|
||||||
|
<div class="layout-breadcrumb-seting-bar-flex-value">
|
||||||
|
<el-select
|
||||||
|
v-model="getThemeConfig.columnsAsideLayout"
|
||||||
|
placeholder="请选择"
|
||||||
|
size="default"
|
||||||
|
style="width: 90px"
|
||||||
|
:disabled="getThemeConfig.layout !== 'columns' ? true : false"
|
||||||
|
@change="setLocalThemeConfig"
|
||||||
|
>
|
||||||
|
<el-option label="水平" value="columns-horizontal"></el-option>
|
||||||
|
<el-option label="垂直" value="columns-vertical"></el-option>
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 布局切换 -->
|
||||||
|
<el-divider content-position="left">{{ $t('message.layout.sixTitle') }}</el-divider>
|
||||||
|
<div class="layout-drawer-content-flex">
|
||||||
|
<!-- defaults 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('defaults')">
|
||||||
|
<section class="el-container el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'defaults' }">
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'defaults' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixDefaults') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- classic 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('classic')">
|
||||||
|
<section class="el-container is-vertical el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'classic' }">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<section class="el-container">
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'classic' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixClassic') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- transverse 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('transverse')">
|
||||||
|
<section class="el-container is-vertical el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'transverse' }">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<section class="el-container">
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'transverse' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixTransverse') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- columns 布局 -->
|
||||||
|
<div class="layout-drawer-content-item" @click="onSetLayout('columns')">
|
||||||
|
<section class="el-container el-circular" :class="{ 'drawer-layout-active': getThemeConfig.layout === 'columns' }">
|
||||||
|
<aside class="el-aside-dark" style="width: 10px"></aside>
|
||||||
|
<aside class="el-aside" style="width: 20px"></aside>
|
||||||
|
<section class="el-container is-vertical">
|
||||||
|
<header class="el-header" style="height: 10px"></header>
|
||||||
|
<main class="el-main"></main>
|
||||||
|
</section>
|
||||||
|
</section>
|
||||||
|
<div class="layout-tips-warp" :class="{ 'layout-tips-warp-active': getThemeConfig.layout === 'columns' }">
|
||||||
|
<div class="layout-tips-box">
|
||||||
|
<p class="layout-tips-txt">{{ $t('message.layout.sixColumns') }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="copy-config">
|
||||||
|
<el-alert :title="$t('message.layout.tipText')" type="warning" :closable="false"> </el-alert>
|
||||||
|
<el-button size="default" class="copy-config-btn" type="primary" ref="copyConfigBtnRef" @click="onCopyConfigClick">
|
||||||
|
<el-icon class="mr5">
|
||||||
|
<ele-CopyDocument />
|
||||||
|
</el-icon>
|
||||||
|
{{ $t('message.layout.copyText') }}
|
||||||
|
</el-button>
|
||||||
|
<el-button size="default" class="copy-config-btn-reset" type="info" @click="onResetConfigClick">
|
||||||
|
<el-icon class="mr5">
|
||||||
|
<ele-RefreshRight />
|
||||||
|
</el-icon>
|
||||||
|
{{ $t('message.layout.resetText') }}
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
</el-scrollbar>
|
||||||
|
</el-drawer>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumbSeting">
|
||||||
|
import { nextTick, onUnmounted, onMounted, computed, reactive } from 'vue';
|
||||||
|
import { ElMessage } from 'element-plus';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import { useChangeColor } from '/@/utils/theme';
|
||||||
|
import { verifyAndSpace } from '/@/utils/toolsValidate';
|
||||||
|
import { Local } from '/@/utils/storage';
|
||||||
|
import Watermark from '/@/utils/watermark';
|
||||||
|
import commonFunction from '/@/utils/commonFunction';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import mittBus from '/@/utils/mitt';
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const { locale } = useI18n();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const { copyText } = commonFunction();
|
||||||
|
const { getLightColor, getDarkColor } = useChangeColor();
|
||||||
|
const state = reactive({
|
||||||
|
isMobile: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
// 获取布局配置信息
|
||||||
|
const getThemeConfig = computed(() => {
|
||||||
|
return themeConfig.value;
|
||||||
|
});
|
||||||
|
// 1、全局主题
|
||||||
|
const onColorPickerChange = () => {
|
||||||
|
if (!getThemeConfig.value.primary) return ElMessage.warning('全局主题 primary 颜色值不能为空');
|
||||||
|
// 颜色加深
|
||||||
|
document.documentElement.style.setProperty('--el-color-primary-dark-2', `${getDarkColor(getThemeConfig.value.primary, 0.1)}`);
|
||||||
|
document.documentElement.style.setProperty('--el-color-primary', getThemeConfig.value.primary);
|
||||||
|
// 颜色变浅
|
||||||
|
for (let i = 1; i <= 9; i++) {
|
||||||
|
document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, `${getLightColor(getThemeConfig.value.primary, i / 10)}`);
|
||||||
|
}
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏
|
||||||
|
const onBgColorPickerChange = (bg: string) => {
|
||||||
|
document.documentElement.style.setProperty(`--next-bg-${bg}`, themeConfig.value[bg]);
|
||||||
|
if (bg === 'menuBar') {
|
||||||
|
document.documentElement.style.setProperty(`--next-bg-menuBar-light-1`, getLightColor(getThemeConfig.value.menuBar, 0.05));
|
||||||
|
}
|
||||||
|
onTopBarGradualChange();
|
||||||
|
onMenuBarGradualChange();
|
||||||
|
onColumnsMenuBarGradualChange();
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 顶栏背景渐变
|
||||||
|
const onTopBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-navbars-breadcrumb-index', getThemeConfig.value.isTopBarColorGradual, getThemeConfig.value.topBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 菜单背景渐变
|
||||||
|
const onMenuBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-container .el-aside', getThemeConfig.value.isMenuBarColorGradual, getThemeConfig.value.menuBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 分栏菜单背景渐变
|
||||||
|
const onColumnsMenuBarGradualChange = () => {
|
||||||
|
setGraduaFun('.layout-container .layout-columns-aside', getThemeConfig.value.isColumnsMenuBarColorGradual, getThemeConfig.value.columnsMenuBar);
|
||||||
|
};
|
||||||
|
// 2、菜单 / 顶栏 --> 背景渐变函数
|
||||||
|
const setGraduaFun = (el: string, bool: boolean, color: string) => {
|
||||||
|
nextTick(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
let els = document.querySelector(el);
|
||||||
|
if (!els) return false;
|
||||||
|
document.documentElement.style.setProperty('--el-menu-bg-color', document.documentElement.style.getPropertyValue('--next-bg-menuBar'));
|
||||||
|
if (bool) els.setAttribute('style', `background:linear-gradient(to bottom , ${color}, ${getLightColor(color, 0.5)})`);
|
||||||
|
else els.setAttribute('style', ``);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
}, 300);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 2、分栏设置 ->
|
||||||
|
const onColumnsMenuHoverPreloadChange = () => {
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 菜单水平折叠
|
||||||
|
const onThemeConfigChange = () => {
|
||||||
|
setDispatchThemeConfig();
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 固定 Header
|
||||||
|
const onIsFixedHeaderChange = () => {
|
||||||
|
getThemeConfig.value.isFixedHeaderChange = getThemeConfig.value.isFixedHeader ? false : true;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 3、界面设置 --> 经典布局分割菜单
|
||||||
|
const onClassicSplitMenuChange = () => {
|
||||||
|
getThemeConfig.value.isBreadcrumb = false;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
mittBus.emit('getBreadcrumbIndexSetFilterRoutes');
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 侧边栏 Logo
|
||||||
|
const onIsShowLogoChange = () => {
|
||||||
|
getThemeConfig.value.isShowLogoChange = getThemeConfig.value.isShowLogo ? false : true;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 面包屑 Breadcrumb
|
||||||
|
const onIsBreadcrumbChange = () => {
|
||||||
|
if (getThemeConfig.value.layout === 'classic') {
|
||||||
|
getThemeConfig.value.isClassicSplitMenu = false;
|
||||||
|
}
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启 TagsView 拖拽
|
||||||
|
const onSortableTagsViewChange = () => {
|
||||||
|
mittBus.emit('openOrCloseSortable');
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启 TagsView 共用
|
||||||
|
const onShareTagsViewChange = () => {
|
||||||
|
mittBus.emit('openShareTagsView');
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 灰色模式/色弱模式
|
||||||
|
const onAddFilterChange = (attr: string) => {
|
||||||
|
if (attr === 'grayscale') {
|
||||||
|
if (getThemeConfig.value.isGrayscale) getThemeConfig.value.isInvert = false;
|
||||||
|
} else {
|
||||||
|
if (getThemeConfig.value.isInvert) getThemeConfig.value.isGrayscale = false;
|
||||||
|
}
|
||||||
|
const cssAttr =
|
||||||
|
attr === 'grayscale' ? `grayscale(${getThemeConfig.value.isGrayscale ? 1 : 0})` : `invert(${getThemeConfig.value.isInvert ? '80%' : '0%'})`;
|
||||||
|
const appEle = document.body;
|
||||||
|
appEle.setAttribute('style', `filter: ${cssAttr}`);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 深色模式
|
||||||
|
const onAddDarkChange = () => {
|
||||||
|
const body = document.documentElement as HTMLElement;
|
||||||
|
if (getThemeConfig.value.isIsDark) body.setAttribute('data-theme', 'dark');
|
||||||
|
else body.setAttribute('data-theme', '');
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 开启水印
|
||||||
|
const onWartermarkChange = () => {
|
||||||
|
getThemeConfig.value.isWartermark ? Watermark.set(getThemeConfig.value.wartermarkText) : Watermark.del();
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 4、界面显示 --> 水印文案
|
||||||
|
const onWartermarkTextInput = (val: string) => {
|
||||||
|
getThemeConfig.value.wartermarkText = verifyAndSpace(val);
|
||||||
|
if (getThemeConfig.value.wartermarkText === '') return false;
|
||||||
|
if (getThemeConfig.value.isWartermark) Watermark.set(getThemeConfig.value.wartermarkText);
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 5、布局切换
|
||||||
|
const onSetLayout = (layout: string) => {
|
||||||
|
Local.set('oldLayout', layout);
|
||||||
|
if (getThemeConfig.value.layout === layout) return false;
|
||||||
|
if (layout === 'transverse') getThemeConfig.value.isCollapse = false;
|
||||||
|
getThemeConfig.value.layout = layout;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
initLayoutChangeFun();
|
||||||
|
};
|
||||||
|
// 设置布局切换函数
|
||||||
|
const initLayoutChangeFun = () => {
|
||||||
|
onBgColorPickerChange('menuBar');
|
||||||
|
onBgColorPickerChange('menuBarColor');
|
||||||
|
onBgColorPickerChange('menuBarActiveColor');
|
||||||
|
onBgColorPickerChange('topBar');
|
||||||
|
onBgColorPickerChange('topBarColor');
|
||||||
|
onBgColorPickerChange('columnsMenuBar');
|
||||||
|
onBgColorPickerChange('columnsMenuBarColor');
|
||||||
|
};
|
||||||
|
// 关闭弹窗时,初始化变量。变量用于处理 layoutScrollbarRef.value.update() 更新滚动条高度
|
||||||
|
const onDrawerClose = () => {
|
||||||
|
getThemeConfig.value.isFixedHeaderChange = false;
|
||||||
|
getThemeConfig.value.isShowLogoChange = false;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
setLocalThemeConfig();
|
||||||
|
};
|
||||||
|
// 布局配置弹窗打开
|
||||||
|
const openDrawer = () => {
|
||||||
|
getThemeConfig.value.isDrawer = true;
|
||||||
|
};
|
||||||
|
// 触发 store 布局配置更新
|
||||||
|
const setDispatchThemeConfig = () => {
|
||||||
|
setLocalThemeConfig();
|
||||||
|
setLocalThemeConfigStyle();
|
||||||
|
};
|
||||||
|
// 存储布局配置
|
||||||
|
const setLocalThemeConfig = () => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
Local.set('themeConfig', getThemeConfig.value);
|
||||||
|
};
|
||||||
|
// 存储布局配置全局主题样式(html根标签)
|
||||||
|
const setLocalThemeConfigStyle = () => {
|
||||||
|
Local.set('themeConfigStyle', document.documentElement.style.cssText);
|
||||||
|
};
|
||||||
|
// 一键复制配置
|
||||||
|
const onCopyConfigClick = () => {
|
||||||
|
let copyThemeConfig = Local.get('themeConfig');
|
||||||
|
copyThemeConfig.isDrawer = false;
|
||||||
|
copyText(JSON.stringify(copyThemeConfig)).then(() => {
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 一键恢复默认
|
||||||
|
const onResetConfigClick = () => {
|
||||||
|
Local.clear();
|
||||||
|
window.location.reload();
|
||||||
|
// @ts-ignore
|
||||||
|
Local.set('version', __NEXT_VERSION__);
|
||||||
|
};
|
||||||
|
// 初始化菜单样式等
|
||||||
|
const initSetStyle = () => {
|
||||||
|
// 2、菜单 / 顶栏 --> 顶栏背景渐变
|
||||||
|
onTopBarGradualChange();
|
||||||
|
// 2、菜单 / 顶栏 --> 菜单背景渐变
|
||||||
|
onMenuBarGradualChange();
|
||||||
|
// 2、菜单 / 顶栏 --> 分栏菜单背景渐变
|
||||||
|
onColumnsMenuBarGradualChange();
|
||||||
|
};
|
||||||
|
onMounted(() => {
|
||||||
|
nextTick(() => {
|
||||||
|
// 判断当前布局是否不相同,不相同则初始化当前布局的样式,防止监听窗口大小改变时,布局配置logo、菜单背景等部分布局失效问题
|
||||||
|
if (!Local.get('frequency')) initLayoutChangeFun();
|
||||||
|
Local.set('frequency', 1);
|
||||||
|
// 监听窗口大小改变,非默认布局,设置成默认布局(适配移动端)
|
||||||
|
mittBus.on('layoutMobileResize', (res: LayoutMobileResize) => {
|
||||||
|
getThemeConfig.value.layout = res.layout;
|
||||||
|
getThemeConfig.value.isDrawer = false;
|
||||||
|
initLayoutChangeFun();
|
||||||
|
state.isMobile = other.isMobile();
|
||||||
|
});
|
||||||
|
setTimeout(() => {
|
||||||
|
// 默认样式
|
||||||
|
onColorPickerChange();
|
||||||
|
// 灰色模式
|
||||||
|
if (getThemeConfig.value.isGrayscale) onAddFilterChange('grayscale');
|
||||||
|
// 色弱模式
|
||||||
|
if (getThemeConfig.value.isInvert) onAddFilterChange('invert');
|
||||||
|
// 深色模式
|
||||||
|
if (getThemeConfig.value.isIsDark) onAddDarkChange();
|
||||||
|
// 开启水印
|
||||||
|
onWartermarkChange();
|
||||||
|
// 语言国际化
|
||||||
|
if (Local.get('themeConfig')) locale.value = Local.get('themeConfig').globalI18n;
|
||||||
|
// 初始化菜单样式等
|
||||||
|
initSetStyle();
|
||||||
|
}, 100);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
onUnmounted(() => {
|
||||||
|
mittBus.off('layoutMobileResize', () => {});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 暴露变量
|
||||||
|
defineExpose({
|
||||||
|
openDrawer,
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-breadcrumb-seting-bar {
|
||||||
|
height: calc(100vh - 50px);
|
||||||
|
padding: 0 15px;
|
||||||
|
:deep(.el-scrollbar__view) {
|
||||||
|
overflow-x: hidden !important;
|
||||||
|
}
|
||||||
|
.layout-breadcrumb-seting-bar-flex {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
&-label {
|
||||||
|
flex: 1;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-drawer-content-flex {
|
||||||
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
align-content: flex-start;
|
||||||
|
margin: 0 -5px;
|
||||||
|
.layout-drawer-content-item {
|
||||||
|
width: 50%;
|
||||||
|
height: 70px;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
position: relative;
|
||||||
|
padding: 5px;
|
||||||
|
.el-container {
|
||||||
|
height: 100%;
|
||||||
|
.el-aside-dark {
|
||||||
|
background-color: var(--next-color-seting-header);
|
||||||
|
}
|
||||||
|
.el-aside {
|
||||||
|
background-color: var(--next-color-seting-aside);
|
||||||
|
}
|
||||||
|
.el-header {
|
||||||
|
background-color: var(--next-color-seting-header);
|
||||||
|
}
|
||||||
|
.el-main {
|
||||||
|
background-color: var(--next-color-seting-main);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.el-circular {
|
||||||
|
border-radius: 2px;
|
||||||
|
overflow: hidden;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
.drawer-layout-active {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
.layout-tips-warp,
|
||||||
|
.layout-tips-warp-active {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
position: absolute;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
border-radius: 100%;
|
||||||
|
padding: 4px;
|
||||||
|
.layout-tips-box {
|
||||||
|
transition: inherit;
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
z-index: 9;
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
border-radius: 100%;
|
||||||
|
.layout-tips-txt {
|
||||||
|
transition: inherit;
|
||||||
|
position: relative;
|
||||||
|
top: 5px;
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 1;
|
||||||
|
letter-spacing: 2px;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: var(--el-color-primary-light-5);
|
||||||
|
text-align: center;
|
||||||
|
transform: rotate(30deg);
|
||||||
|
left: -1px;
|
||||||
|
background-color: var(--next-color-seting-main);
|
||||||
|
width: 32px;
|
||||||
|
height: 17px;
|
||||||
|
line-height: 17px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.layout-tips-warp-active {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-box {
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-txt {
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
background-color: var(--next-color-seting-main) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
.el-circular {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
border: 1px solid;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
}
|
||||||
|
.layout-tips-warp {
|
||||||
|
transition: all 0.3s ease-in-out;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-box {
|
||||||
|
transition: inherit;
|
||||||
|
border-color: var(--el-color-primary);
|
||||||
|
.layout-tips-txt {
|
||||||
|
transition: inherit;
|
||||||
|
color: var(--el-color-primary) !important;
|
||||||
|
background-color: var(--next-color-seting-main) !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.copy-config {
|
||||||
|
margin: 10px 0;
|
||||||
|
.copy-config-btn {
|
||||||
|
width: 100%;
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
.copy-config-btn-reset {
|
||||||
|
width: 100%;
|
||||||
|
margin: 10px 0 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
269
src/layout/navBars/topBar/user.vue
Normal file
269
src/layout/navBars/topBar/user.vue
Normal file
@ -0,0 +1,269 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-user pr15" :style="{ flex: layoutUserFlexNum }">
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" trigger="click" @command="onComponentSizeChange">
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon">
|
||||||
|
<i class="iconfont icon-ziti" :title="$t('message.user.title0')"></i>
|
||||||
|
</div>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="large" :disabled="state.disabledSize === 'large'">{{ $t('message.user.dropdownLarge') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="default" :disabled="state.disabledSize === 'default'">{{ $t('message.user.dropdownDefault') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="small" :disabled="state.disabledSize === 'small'">{{ $t('message.user.dropdownSmall') }}</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" trigger="click" @command="onLanguageChange">
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon">
|
||||||
|
<i
|
||||||
|
class="iconfont"
|
||||||
|
:class="state.disabledI18n === 'en' ? 'icon-fuhao-yingwen' : 'icon-fuhao-zhongwen'"
|
||||||
|
:title="$t('message.user.title1')"
|
||||||
|
></i>
|
||||||
|
</div>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="zh-cn" :disabled="state.disabledI18n === 'zh-cn'">简体中文</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="en" :disabled="state.disabledI18n === 'en'">English</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="zh-tw" :disabled="state.disabledI18n === 'zh-tw'">繁體中文</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon" @click="onSearchClick">
|
||||||
|
<el-icon :title="$t('message.user.title2')">
|
||||||
|
<ele-Search />
|
||||||
|
</el-icon>
|
||||||
|
</div>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon" @click="onLayoutSetingClick">
|
||||||
|
<i class="icon-skin iconfont" :title="$t('message.user.title3')"></i>
|
||||||
|
</div>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon" ref="userNewsBadgeRef" v-click-outside="onUserNewsClick">
|
||||||
|
<el-badge :is-dot="true">
|
||||||
|
<el-icon :title="$t('message.user.title4')">
|
||||||
|
<ele-Bell />
|
||||||
|
</el-icon>
|
||||||
|
</el-badge>
|
||||||
|
</div>
|
||||||
|
<el-popover
|
||||||
|
ref="userNewsRef"
|
||||||
|
:virtual-ref="userNewsBadgeRef"
|
||||||
|
placement="bottom"
|
||||||
|
trigger="click"
|
||||||
|
transition="el-zoom-in-top"
|
||||||
|
virtual-triggering
|
||||||
|
:width="300"
|
||||||
|
:persistent="false"
|
||||||
|
>
|
||||||
|
<UserNews />
|
||||||
|
</el-popover>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-icon mr10" @click="onScreenfullClick">
|
||||||
|
<i
|
||||||
|
class="iconfont"
|
||||||
|
:title="state.isScreenfull ? $t('message.user.title6') : $t('message.user.title5')"
|
||||||
|
:class="!state.isScreenfull ? 'icon-fullscreen' : 'icon-tuichuquanping'"
|
||||||
|
></i>
|
||||||
|
</div>
|
||||||
|
<el-dropdown :show-timeout="70" :hide-timeout="50" @command="onHandleCommandClick">
|
||||||
|
<span class="layout-navbars-breadcrumb-user-link">
|
||||||
|
<img :src="userInfos.photo" class="layout-navbars-breadcrumb-user-link-photo mr5" />
|
||||||
|
{{ userInfos.userName === '' ? 'common' : userInfos.userName }}
|
||||||
|
<el-icon class="el-icon--right">
|
||||||
|
<ele-ArrowDown />
|
||||||
|
</el-icon>
|
||||||
|
</span>
|
||||||
|
<template #dropdown>
|
||||||
|
<el-dropdown-menu>
|
||||||
|
<el-dropdown-item command="/home">{{ $t('message.user.dropdown1') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="wareHouse">{{ $t('message.user.dropdown6') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/personal">{{ $t('message.user.dropdown2') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/404">{{ $t('message.user.dropdown3') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item command="/401">{{ $t('message.user.dropdown4') }}</el-dropdown-item>
|
||||||
|
<el-dropdown-item divided command="logOut">{{ $t('message.user.dropdown5') }}</el-dropdown-item>
|
||||||
|
</el-dropdown-menu>
|
||||||
|
</template>
|
||||||
|
</el-dropdown>
|
||||||
|
<Search ref="searchRef" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumbUser">
|
||||||
|
import { defineAsyncComponent, ref, unref, computed, reactive, onMounted } from 'vue';
|
||||||
|
import { useRouter } from 'vue-router';
|
||||||
|
import { ElMessageBox, ElMessage, ClickOutside as vClickOutside } from 'element-plus';
|
||||||
|
import screenfull from 'screenfull';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import { storeToRefs } from 'pinia';
|
||||||
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { useThemeConfig } from '/@/stores/themeConfig';
|
||||||
|
import other from '/@/utils/other';
|
||||||
|
import mittBus from '/@/utils/mitt';
|
||||||
|
import { Session, Local } from '/@/utils/storage';
|
||||||
|
|
||||||
|
// 引入组件
|
||||||
|
const UserNews = defineAsyncComponent(() => import('/@/layout/navBars/topBar/userNews.vue'));
|
||||||
|
const Search = defineAsyncComponent(() => import('/@/layout/navBars/topBar/search.vue'));
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const userNewsRef = ref();
|
||||||
|
const userNewsBadgeRef = ref();
|
||||||
|
const { locale, t } = useI18n();
|
||||||
|
const router = useRouter();
|
||||||
|
const stores = useUserInfo();
|
||||||
|
const storesThemeConfig = useThemeConfig();
|
||||||
|
const { userInfos } = storeToRefs(stores);
|
||||||
|
const { themeConfig } = storeToRefs(storesThemeConfig);
|
||||||
|
const searchRef = ref();
|
||||||
|
const state = reactive({
|
||||||
|
isScreenfull: false,
|
||||||
|
disabledI18n: 'zh-cn',
|
||||||
|
disabledSize: 'large',
|
||||||
|
});
|
||||||
|
|
||||||
|
// 设置分割样式
|
||||||
|
const layoutUserFlexNum = computed(() => {
|
||||||
|
let num: string | number = '';
|
||||||
|
const { layout, isClassicSplitMenu } = themeConfig.value;
|
||||||
|
const layoutArr: string[] = ['defaults', 'columns'];
|
||||||
|
if (layoutArr.includes(layout) || (layout === 'classic' && !isClassicSplitMenu)) num = '1';
|
||||||
|
else num = '';
|
||||||
|
return num;
|
||||||
|
});
|
||||||
|
// 全屏点击时
|
||||||
|
const onScreenfullClick = () => {
|
||||||
|
if (!screenfull.isEnabled) {
|
||||||
|
ElMessage.warning('暂不不支持全屏');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
screenfull.toggle();
|
||||||
|
screenfull.on('change', () => {
|
||||||
|
if (screenfull.isFullscreen) state.isScreenfull = true;
|
||||||
|
else state.isScreenfull = false;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// 消息通知点击时
|
||||||
|
const onUserNewsClick = () => {
|
||||||
|
unref(userNewsRef).popperRef?.delayHide?.();
|
||||||
|
};
|
||||||
|
// 布局配置 icon 点击时
|
||||||
|
const onLayoutSetingClick = () => {
|
||||||
|
mittBus.emit('openSetingsDrawer');
|
||||||
|
};
|
||||||
|
// 下拉菜单点击时
|
||||||
|
const onHandleCommandClick = (path: string) => {
|
||||||
|
if (path === 'logOut') {
|
||||||
|
ElMessageBox({
|
||||||
|
closeOnClickModal: false,
|
||||||
|
closeOnPressEscape: false,
|
||||||
|
title: t('message.user.logOutTitle'),
|
||||||
|
message: t('message.user.logOutMessage'),
|
||||||
|
showCancelButton: true,
|
||||||
|
confirmButtonText: t('message.user.logOutConfirm'),
|
||||||
|
cancelButtonText: t('message.user.logOutCancel'),
|
||||||
|
buttonSize: 'default',
|
||||||
|
beforeClose: (action, instance, done) => {
|
||||||
|
if (action === 'confirm') {
|
||||||
|
instance.confirmButtonLoading = true;
|
||||||
|
instance.confirmButtonText = t('message.user.logOutExit');
|
||||||
|
setTimeout(() => {
|
||||||
|
done();
|
||||||
|
setTimeout(() => {
|
||||||
|
instance.confirmButtonLoading = false;
|
||||||
|
}, 300);
|
||||||
|
}, 700);
|
||||||
|
} else {
|
||||||
|
done();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.then(async () => {
|
||||||
|
// 清除缓存/token等
|
||||||
|
Session.clear();
|
||||||
|
// 使用 reload 时,不需要调用 resetRoute() 重置路由
|
||||||
|
window.location.reload();
|
||||||
|
})
|
||||||
|
.catch(() => {});
|
||||||
|
} else if (path === 'wareHouse') {
|
||||||
|
window.open('https://gitee.com/lyt-top/vue-next-admin');
|
||||||
|
} else {
|
||||||
|
router.push(path);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
// 菜单搜索点击
|
||||||
|
const onSearchClick = () => {
|
||||||
|
searchRef.value.openSearch();
|
||||||
|
};
|
||||||
|
// 组件大小改变
|
||||||
|
const onComponentSizeChange = (size: string) => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
themeConfig.value.globalComponentSize = size;
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
initI18nOrSize('globalComponentSize', 'disabledSize');
|
||||||
|
window.location.reload();
|
||||||
|
};
|
||||||
|
// 语言切换
|
||||||
|
const onLanguageChange = (lang: string) => {
|
||||||
|
Local.remove('themeConfig');
|
||||||
|
themeConfig.value.globalI18n = lang;
|
||||||
|
Local.set('themeConfig', themeConfig.value);
|
||||||
|
locale.value = lang;
|
||||||
|
other.useTitle();
|
||||||
|
initI18nOrSize('globalI18n', 'disabledI18n');
|
||||||
|
};
|
||||||
|
// 初始化组件大小/i18n
|
||||||
|
const initI18nOrSize = (value: string, attr: string) => {
|
||||||
|
(<any>state)[attr] = Local.get('themeConfig')[value];
|
||||||
|
};
|
||||||
|
// 页面加载时
|
||||||
|
onMounted(() => {
|
||||||
|
if (Local.get('themeConfig')) {
|
||||||
|
initI18nOrSize('globalComponentSize', 'disabledSize');
|
||||||
|
initI18nOrSize('globalI18n', 'disabledI18n');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-user {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
&-link {
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
white-space: nowrap;
|
||||||
|
&-photo {
|
||||||
|
width: 25px;
|
||||||
|
height: 25px;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&-icon {
|
||||||
|
padding: 0 10px;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
height: 50px;
|
||||||
|
line-height: 50px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
&:hover {
|
||||||
|
background: var(--next-color-user-hover);
|
||||||
|
i {
|
||||||
|
display: inline-block;
|
||||||
|
animation: logoAnimation 0.3s ease-in-out;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-dropdown) {
|
||||||
|
color: var(--next-bg-topBarColor);
|
||||||
|
}
|
||||||
|
:deep(.el-badge) {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
:deep(.el-badge__content.is-fixed) {
|
||||||
|
top: 12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
107
src/layout/navBars/topBar/userNews.vue
Normal file
107
src/layout/navBars/topBar/userNews.vue
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
<template>
|
||||||
|
<div class="layout-navbars-breadcrumb-user-news">
|
||||||
|
<div class="head-box">
|
||||||
|
<div class="head-box-title">{{ $t('message.user.newTitle') }}</div>
|
||||||
|
<div class="head-box-btn" v-if="state.newsList.length > 0" @click="onAllReadClick">{{ $t('message.user.newBtn') }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="content-box">
|
||||||
|
<template v-if="state.newsList.length > 0">
|
||||||
|
<div class="content-box-item" v-for="(v, k) in state.newsList" :key="k">
|
||||||
|
<div>{{ v.label }}</div>
|
||||||
|
<div class="content-box-msg">
|
||||||
|
{{ v.value }}
|
||||||
|
</div>
|
||||||
|
<div class="content-box-time">{{ v.time }}</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<el-empty :description="$t('message.user.newDesc')" v-else></el-empty>
|
||||||
|
</div>
|
||||||
|
<div class="foot-box" @click="onGoToGiteeClick" v-if="state.newsList.length > 0">{{ $t('message.user.newGo') }}</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup lang="ts" name="layoutBreadcrumbUserNews">
|
||||||
|
import { reactive } from 'vue';
|
||||||
|
|
||||||
|
// 定义变量内容
|
||||||
|
const state = reactive({
|
||||||
|
newsList: [
|
||||||
|
{
|
||||||
|
label: '关于版本发布的通知',
|
||||||
|
value: 'vue-next-admin,基于 vue3 + CompositionAPI + typescript + vite + element plus,正式发布时间:2021年02月28日!',
|
||||||
|
time: '2020-12-08',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '关于学习交流的通知',
|
||||||
|
value: 'QQ群号码 665452019,欢迎小伙伴入群学习交流探讨!',
|
||||||
|
time: '2020-12-08',
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
// 全部已读点击
|
||||||
|
const onAllReadClick = () => {
|
||||||
|
state.newsList = [];
|
||||||
|
};
|
||||||
|
// 前往通知中心点击
|
||||||
|
const onGoToGiteeClick = () => {
|
||||||
|
window.open('https://gitee.com/lyt-top/vue-next-admin');
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style scoped lang="scss">
|
||||||
|
.layout-navbars-breadcrumb-user-news {
|
||||||
|
.head-box {
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 1px solid var(--el-border-color-lighter);
|
||||||
|
box-sizing: border-box;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
justify-content: space-between;
|
||||||
|
height: 35px;
|
||||||
|
align-items: center;
|
||||||
|
.head-box-btn {
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
font-size: 13px;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.8;
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.content-box {
|
||||||
|
font-size: 13px;
|
||||||
|
.content-box-item {
|
||||||
|
padding-top: 12px;
|
||||||
|
&:last-of-type {
|
||||||
|
padding-bottom: 12px;
|
||||||
|
}
|
||||||
|
.content-box-msg {
|
||||||
|
color: var(--el-text-color-secondary);
|
||||||
|
margin-top: 5px;
|
||||||
|
margin-bottom: 5px;
|
||||||
|
}
|
||||||
|
.content-box-time {
|
||||||
|
color: var(--el-text-color-secondary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.foot-box {
|
||||||
|
height: 35px;
|
||||||
|
color: var(--el-color-primary);
|
||||||
|
font-size: 13px;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.8;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-top: 1px solid var(--el-border-color-lighter);
|
||||||
|
&:hover {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:deep(.el-empty__description p) {
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
@ -53,7 +53,7 @@ export async function initBackEndControlRoutes() {
|
|||||||
// 添加动态路由
|
// 添加动态路由
|
||||||
await setAddRoute();
|
await setAddRoute();
|
||||||
// 设置路由到 pinia routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
// 设置路由到 pinia routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
await setFilterMenuAndCacheTagsViewRoutes();
|
setFilterMenuAndCacheTagsViewRoutes();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -32,7 +32,7 @@ export async function initFrontEndControlRoutes() {
|
|||||||
// 添加动态路由
|
// 添加动态路由
|
||||||
await setAddRoute();
|
await setAddRoute();
|
||||||
// 设置递归过滤有权限的路由到 pinia routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
// 设置递归过滤有权限的路由到 pinia routesList 中(已处理成多级嵌套路由)及缓存多级嵌套数组处理后的一维数组
|
||||||
await setFilterMenuAndCacheTagsViewRoutes();
|
setFilterMenuAndCacheTagsViewRoutes();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -121,7 +121,7 @@ export const useThemeConfig = defineStore('themeConfig', {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* 布局切换
|
* 布局切换
|
||||||
* 注意:为了演示,切换布局时,颜色会被还原成默认,代码位置:/@/layout/navBars/breadcrumb/setings.vue
|
* 注意:为了演示,切换布局时,颜色会被还原成默认,代码位置:/@/layout/navBars/topBar/setings.vue
|
||||||
* 中的 `initSetLayoutChange(设置布局切换,重置主题样式)` 方法
|
* 中的 `initSetLayoutChange(设置布局切换,重置主题样式)` 方法
|
||||||
*/
|
*/
|
||||||
// 布局切换:可选值"<defaults|classic|transverse|columns>",默认 defaults
|
// 布局切换:可选值"<defaults|classic|transverse|columns>",默认 defaults
|
||||||
|
@ -22,7 +22,7 @@ export const useUserInfo = defineStore('userInfo', {
|
|||||||
if (Session.get('userInfo')) {
|
if (Session.get('userInfo')) {
|
||||||
this.userInfos = Session.get('userInfo');
|
this.userInfos = Session.get('userInfo');
|
||||||
} else {
|
} else {
|
||||||
const userInfos: any = await this.getApiUserInfo();
|
const userInfos = <UserInfos>await this.getApiUserInfo();
|
||||||
this.userInfos = userInfos;
|
this.userInfos = userInfos;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -188,6 +188,18 @@
|
|||||||
color: var(--next-bg-topBarColor);
|
color: var(--next-bg-topBarColor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// 菜单收起时,图标不居中问题
|
||||||
|
.el-menu--collapse {
|
||||||
|
.el-menu-item .iconfont,
|
||||||
|
.el-sub-menu .iconfont,
|
||||||
|
.el-menu-item .fa,
|
||||||
|
.el-sub-menu .fa {
|
||||||
|
margin-right: 0 !important;
|
||||||
|
}
|
||||||
|
.el-sub-menu__title {
|
||||||
|
padding-right: 0 !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Tabs 标签页
|
/* Tabs 标签页
|
||||||
------------------------------- */
|
------------------------------- */
|
||||||
|
2
src/types/global.d.ts
vendored
2
src/types/global.d.ts
vendored
@ -27,6 +27,8 @@ declare module '*.vue' {
|
|||||||
/* eslint-disable */
|
/* eslint-disable */
|
||||||
declare interface Window {
|
declare interface Window {
|
||||||
nextLoading: boolean;
|
nextLoading: boolean;
|
||||||
|
BMAP_SATELLITE_MAP: any;
|
||||||
|
BMap: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 声明路由当前项类型
|
// 声明路由当前项类型
|
||||||
|
19
src/types/pinia.d.ts
vendored
19
src/types/pinia.d.ts
vendored
@ -3,15 +3,16 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
// 用户信息
|
// 用户信息
|
||||||
declare interface UserInfosState<T = any> {
|
declare interface UserInfos<T = any> {
|
||||||
userInfos: {
|
authBtnList: string[];
|
||||||
authBtnList: string[];
|
photo: string;
|
||||||
photo: string;
|
roles: string[];
|
||||||
roles: string[];
|
time: number;
|
||||||
time: number;
|
userName: string;
|
||||||
userName: string;
|
[key: string]: T;
|
||||||
[key: string]: T;
|
}
|
||||||
};
|
declare interface UserInfosState {
|
||||||
|
userInfos: UserInfos;
|
||||||
}
|
}
|
||||||
|
|
||||||
// 路由缓存列表
|
// 路由缓存列表
|
||||||
|
1
src/types/views.d.ts
vendored
1
src/types/views.d.ts
vendored
@ -325,5 +325,6 @@ declare type TableDemoState = {
|
|||||||
};
|
};
|
||||||
search: TableSearchType[];
|
search: TableSearchType[];
|
||||||
param: EmptyObjectType;
|
param: EmptyObjectType;
|
||||||
|
printName: string;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -27,6 +27,7 @@ import { onMounted, ref } from 'vue';
|
|||||||
import Cookies from 'js-cookie';
|
import Cookies from 'js-cookie';
|
||||||
import { storeToRefs } from 'pinia';
|
import { storeToRefs } from 'pinia';
|
||||||
import { useUserInfo } from '/@/stores/userInfo';
|
import { useUserInfo } from '/@/stores/userInfo';
|
||||||
|
import { Session } from '/@/utils/storage';
|
||||||
import { frontEndsResetRoute, setAddRoute, setFilterMenuAndCacheTagsViewRoutes } from '/@/router/frontEnd';
|
import { frontEndsResetRoute, setAddRoute, setFilterMenuAndCacheTagsViewRoutes } from '/@/router/frontEnd';
|
||||||
|
|
||||||
// 定义变量内容
|
// 定义变量内容
|
||||||
@ -40,6 +41,9 @@ const initUserAuth = () => {
|
|||||||
};
|
};
|
||||||
// 用户权限改变时
|
// 用户权限改变时
|
||||||
const onRadioChange = async () => {
|
const onRadioChange = async () => {
|
||||||
|
// 清空之前缓存的 userInfo,防止不请求接口。
|
||||||
|
// stores/userInfo.ts
|
||||||
|
Session.remove('userInfo');
|
||||||
// 模拟数据
|
// 模拟数据
|
||||||
frontEndsResetRoute();
|
frontEndsResetRoute();
|
||||||
Cookies.set('userName', userAuth.value);
|
Cookies.set('userName', userAuth.value);
|
||||||
|
@ -71,6 +71,8 @@ const state = reactive<TableDemoState>({
|
|||||||
pageNum: 1,
|
pageNum: 1,
|
||||||
pageSize: 10,
|
pageSize: 10,
|
||||||
},
|
},
|
||||||
|
// 打印标题
|
||||||
|
printName: 'vueNextAdmin 表格打印演示',
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -126,7 +126,7 @@ const initTime = () => {
|
|||||||
const convertData = (data: any) => {
|
const convertData = (data: any) => {
|
||||||
let res = [];
|
let res = [];
|
||||||
for (let i = 0; i < data.length; i++) {
|
for (let i = 0; i < data.length; i++) {
|
||||||
let geoCoord = state.echartsMapData[data[i].name];
|
let geoCoord = (<any>state.echartsMapData)[data[i].name];
|
||||||
if (geoCoord) {
|
if (geoCoord) {
|
||||||
res.push({
|
res.push({
|
||||||
name: data[i].name,
|
name: data[i].name,
|
||||||
@ -250,16 +250,16 @@ const initEchartsMap = () => {
|
|||||||
// BMAP_SATELLITE_MAP:卫星地图 (没有坐标, 绿绿的一片的卫星地图)
|
// BMAP_SATELLITE_MAP:卫星地图 (没有坐标, 绿绿的一片的卫星地图)
|
||||||
// BMAP_HYBRID_MAP:混合地图 (既有坐标,也是绿绿的一片的卫星地图)
|
// BMAP_HYBRID_MAP:混合地图 (既有坐标,也是绿绿的一片的卫星地图)
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
map.setMapType(BMAP_SATELLITE_MAP);
|
map.setMapType(window.BMAP_SATELLITE_MAP);
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
let bdary = new BMap.Boundary();
|
let bdary = new window.BMap.Boundary();
|
||||||
// 获取行政区域
|
// 获取行政区域
|
||||||
bdary.get('深圳', function (rs: any) {
|
bdary.get('深圳', function (rs: any) {
|
||||||
// 行政区域的点有多少个
|
// 行政区域的点有多少个
|
||||||
let count = rs.boundaries.length;
|
let count = rs.boundaries.length;
|
||||||
for (let i = 0; i < count; i++) {
|
for (let i = 0; i < count; i++) {
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
let ply = new BMap.Polygon(rs.boundaries[i], {
|
let ply = new window.BMap.Polygon(rs.boundaries[i], {
|
||||||
// 设置多边形边线线粗
|
// 设置多边形边线线粗
|
||||||
strokeWeight: 4,
|
strokeWeight: 4,
|
||||||
// 设置多边形边线透明度0-1
|
// 设置多边形边线透明度0-1
|
||||||
@ -280,7 +280,7 @@ const initEchartsMap = () => {
|
|||||||
// 初始化地图,设置中心点坐标和地图级别
|
// 初始化地图,设置中心点坐标和地图级别
|
||||||
// new BMap.Point('深圳市', 11)
|
// new BMap.Point('深圳市', 11)
|
||||||
// eslint-disable-next-line no-undef
|
// eslint-disable-next-line no-undef
|
||||||
map.centerAndZoom(new BMap.Point(114.064524, 22.549225), 11);
|
map.centerAndZoom(new window.BMap.Point(114.064524, 22.549225), 11);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
// 产业概况
|
// 产业概况
|
||||||
|
@ -67,8 +67,7 @@
|
|||||||
|
|
||||||
/* Advanced Options */
|
/* Advanced Options */
|
||||||
"skipLibCheck": true /* Skip type checking of declaration files. */,
|
"skipLibCheck": true /* Skip type checking of declaration files. */,
|
||||||
"forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */,
|
"forceConsistentCasingInFileNames": true /* Disallow inconsistently-cased references to the same file. */
|
||||||
"suppressImplicitAnyIndexErrors": true
|
|
||||||
},
|
},
|
||||||
"include": ["src/**/*.ts", "src/**/*.vue", "src/**/*.tsx", "src/**/*.d.ts"], // **Represents any directory, and * represents any file. Indicates that all files in the src directory will be compiled
|
"include": ["src/**/*.ts", "src/**/*.vue", "src/**/*.tsx", "src/**/*.d.ts"], // **Represents any directory, and * represents any file. Indicates that all files in the src directory will be compiled
|
||||||
"exclude": ["node_modules", "dist"] // Indicates the file directory that does not need to be compiled
|
"exclude": ["node_modules", "dist"] // Indicates the file directory that does not need to be compiled
|
||||||
|
@ -21,10 +21,7 @@ const viteConfig = defineConfig((mode: ConfigEnv) => {
|
|||||||
root: process.cwd(),
|
root: process.cwd(),
|
||||||
resolve: { alias },
|
resolve: { alias },
|
||||||
base: mode.command === 'serve' ? './' : env.VITE_PUBLIC_PATH,
|
base: mode.command === 'serve' ? './' : env.VITE_PUBLIC_PATH,
|
||||||
optimizeDeps: {
|
optimizeDeps: { exclude: ['vue-demi'] },
|
||||||
include: ['element-plus/lib/locale/lang/zh-cn', 'element-plus/lib/locale/lang/en', 'element-plus/lib/locale/lang/zh-tw'],
|
|
||||||
exclude: ['vue-demi'],
|
|
||||||
},
|
|
||||||
server: {
|
server: {
|
||||||
host: '0.0.0.0',
|
host: '0.0.0.0',
|
||||||
port: env.VITE_PORT as unknown as number,
|
port: env.VITE_PORT as unknown as number,
|
||||||
@ -49,11 +46,11 @@ const viteConfig = defineConfig((mode: ConfigEnv) => {
|
|||||||
assetFileNames: 'assets/[ext]/[name]-[hash].[ext]',
|
assetFileNames: 'assets/[ext]/[name]-[hash].[ext]',
|
||||||
manualChunks(id) {
|
manualChunks(id) {
|
||||||
if (id.includes('node_modules')) {
|
if (id.includes('node_modules')) {
|
||||||
return id.toString().match(/\/node_modules\/(?!.pnpm)(?<moduleName>[^\/]*)\//)?.groups.moduleName ?? 'vender';
|
return id.toString().match(/\/node_modules\/(?!.pnpm)(?<moduleName>[^\/]*)\//)?.groups!.moduleName ?? 'vender';
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
...(JSON.parse(env.VITE_OPEN_CDN) ? {external: buildConfig.external} : {}),
|
...(JSON.parse(env.VITE_OPEN_CDN) ? { external: buildConfig.external } : {}),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
css: { preprocessorOptions: { css: { charset: false } } },
|
css: { preprocessorOptions: { css: { charset: false } } },
|
||||||
|
Loading…
Reference in New Issue
Block a user