v3.1.4发布

This commit is contained in:
ray_wuhao 2023-03-21 13:48:12 +08:00
parent 0aab9340a0
commit 6a931c0cbb
25 changed files with 622 additions and 55 deletions

View File

@ -100,7 +100,6 @@ module.exports = {
allowTaggedTemplates: true,
},
], // 禁止无用的表达式
'no-use-before-define': 2, // 禁止定义前使用
'no-useless-call': 2, // 禁止不必要的 `call` 和 `apply`
'no-var': 'error', // 禁用 `var`
'no-with': 2, // 禁用 `with`
@ -122,5 +121,14 @@ module.exports = {
],
'vue/require-v-for-key': ['error'],
'vue/require-valid-default-prop': ['error'],
'no-use-before-define': [
'error',
{
functions: true,
classes: true,
variables: false,
allowNamedExports: false,
},
],
},
}

View File

@ -1,5 +1,18 @@
# CHANGE LOG
## 3.1.4
### Fixes
- 修复主题色切换后,点击、鼠标滑入主题未被修改问题
- 修复 menu store 菜单切换可能会重复执行问题
### Feats
- 补充 MenuTag 标签页功能,现在支持丰富的关闭操作与右键菜单激活操作菜单功能
- 新增配置全局重定向地址配置(详情见:[cfg](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/cfg.ts)
- 补充了一些不值一提的小东西
## 3.1.3
### Fixes

View File

@ -27,6 +27,7 @@
## 功能
- 主题切换
- 带有拓展功能的表格
- 封装 `axios` 自动取消重复请求
- 动态菜单(多级菜单)
- 主题色切换
@ -36,7 +37,6 @@
- 国际化(允许按模块管理语言包)
- 权限路由
- 动态切换主题、贴花的 `EChart`
- 带有拓展功能的表格
- 最佳构建体验
- 体积分析
- 还有一些不值一提的小东西...

11
cfg.ts
View File

@ -9,6 +9,17 @@ import {
import type { AppConfigExport } from './src/types/cfg'
const config: AppConfigExport = {
/**
*
*
* ,
*
* , ,
*/
rootRoute: {
name: 'dashboard',
path: '/dashboard',
},
/**
*
* icon: LOGO , `RayIcon`

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": true,
"version": "3.1.3",
"version": "3.1.4",
"type": "module",
"scripts": {
"dev": "vite",

6
src/icons/close.svg Normal file
View File

@ -0,0 +1,6 @@
<svg t="1679235208726" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="7558" width="64" height="64">
<path
d="M212.898909 130.792727l270.522182 270.522182 270.522182-270.522182a58.181818 58.181818 0 1 1 82.292363 82.292364L565.713455 483.607273 836.235636 754.036364a58.181818 58.181818 0 1 1-82.292363 82.292363L483.421091 565.899636 212.898909 836.421818a58.181818 58.181818 0 0 1-82.292364-82.292363L401.221818 483.607273 130.606545 213.085091a58.181818 58.181818 0 0 1 82.292364-82.292364z"
fill="currentColor" p-id="7559"></path>
</svg>

After

Width:  |  Height:  |  Size: 597 B

12
src/icons/more.svg Normal file
View File

@ -0,0 +1,12 @@
<svg t="1679234409554" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="2793" width="64" height="64">
<path
d="M223.962372 607.897867c-52.980346 0-95.983874-43.003528-95.983874-95.983874s43.003528-95.983874 95.983874-95.983874 95.983874 43.003528 95.983874 95.983874S276.942718 607.897867 223.962372 607.897867z"
fill="currentColor" p-id="2794"></path>
<path
d="M511.913993 607.897867c-52.980346 0-95.983874-43.003528-95.983874-95.983874s43.003528-95.983874 95.983874-95.983874 95.983874 43.003528 95.983874 95.983874S564.894339 607.897867 511.913993 607.897867z"
fill="currentColor" p-id="2795"></path>
<path
d="M800.037628 607.897867c-52.980346 0-95.983874-43.003528-95.983874-95.983874s43.003528-95.983874 95.983874-95.983874 95.983874 43.003528 95.983874 95.983874S852.84596 607.897867 800.037628 607.897867z"
fill="currentColor" p-id="2796"></path>
</svg>

After

Width:  |  Height:  |  Size: 935 B

9
src/icons/other.svg Normal file
View File

@ -0,0 +1,9 @@
<svg t="1679316911025" class="icon" viewBox="0 0 1030 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="6862" width="64" height="64">
<path
d="M376.053929 561.350639H86.337861A86.440889 86.440889 0 0 0 0 647.6885v289.922125a86.492404 86.492404 0 0 0 86.337861 86.389375H376.053929a86.492404 86.492404 0 0 0 86.389375-86.389375v-289.922125A86.440889 86.440889 0 0 0 376.053929 561.350639z m8.396821 376.053929a8.551363 8.551363 0 0 1-8.396821 8.602877H86.337861a8.499849 8.499849 0 0 1-8.345306-8.39682v-289.922125a8.448335 8.448335 0 0 1 8.345306-8.345307H376.053929a8.499849 8.499849 0 0 1 8.396821 8.345307z"
p-id="6863"></path>
<path
d="M1018.694034 287.91307l-82.422779-142.488379a38.97052 38.97052 0 1 0-67.483651 38.996277l82.422779 142.488379a8.602878 8.602878 0 0 1-3.090854 11.487675l-251.08039 144.909548a8.087735 8.087735 0 0 1-6.336251 0.824228 8.242278 8.242278 0 0 1-5.151424-3.863568L540.899487 229.18684a8.499849 8.499849 0 0 1 3.090854-11.436161l251.028876-144.961062a38.996277 38.996277 0 0 0-38.944763-67.535165L504.839521 150.215515a85.668176 85.668176 0 0 0-40.284133 52.699064 84.637891 84.637891 0 0 0-1.906027 9.272563V127.90985A86.492404 86.492404 0 0 0 376.053929 41.520475H86.337861A86.440889 86.440889 0 0 0 0 127.90985v289.922125a86.440889 86.440889 0 0 0 86.337861 86.337861H376.053929a86.440889 86.440889 0 0 0 86.595432-86.337861V238.253345a85.822719 85.822719 0 0 0 10.302847 29.929772L618.170842 519.263507a85.513633 85.513633 0 0 0 61.817084 42.087132h-68.616963a86.492404 86.492404 0 0 0-86.389375 86.337861v289.922125a86.543918 86.543918 0 0 0 86.389375 86.389375H901.499145a86.492404 86.492404 0 0 0 86.337861-86.389375v-289.922125A86.440889 86.440889 0 0 0 901.499145 561.350639h-195.187444a85.925747 85.925747 0 0 0 29.723715-10.302847l251.08039-145.16712a86.440889 86.440889 0 0 0 31.578228-117.967602zM384.656807 417.831975A8.499849 8.499849 0 0 1 376.053929 426.177281H86.337861a8.448335 8.448335 0 0 1-8.345306-8.345306V127.90985a8.499849 8.499849 0 0 1 8.345306-8.396821H376.053929a8.551363 8.551363 0 0 1 8.396821 8.396821z m524.981587 229.856525v289.922125a8.499849 8.499849 0 0 1-8.345306 8.39682h-289.922125a8.551363 8.551363 0 0 1-8.396821-8.39682v-289.922125a8.499849 8.499849 0 0 1 8.396821-8.345307H901.499145a8.448335 8.448335 0 0 1 8.139249 8.345307z"
p-id="6864"></path>
</svg>

After

Width:  |  Height:  |  Size: 2.3 KiB

View File

@ -1,4 +1,5 @@
$space: calc($layoutRouterViewContainer / 2);
$menuTagWrapperWidth: 76px;
.menu-tag {
height: $layoutMenuHeight;
@ -7,6 +8,34 @@ $space: calc($layoutRouterViewContainer / 2);
& .menu-tag-sapce {
width: calc(100% - $space * 2);
padding: $space;
& .menu-tag-wrapper {
width: calc(100% - $space * 2 - $menuTagWrapperWidth);
}
& .ray-icon {
cursor: pointer;
}
& .menu-tag__left-arrow {
transform: rotate(90deg);
}
& .menu-tag__right-wrapper {
display: inline-flex;
align-items: center;
& .menu-tag__right-arrow {
transform: rotate(270deg);
}
& .menu-tag__right-setting {
width: 28px;
height: 20px;
// display: inline-flex;
// align-items: center;
}
}
}
& .n-tag {

View File

@ -10,19 +10,174 @@
*/
import './index.scss'
import { NScrollbar, NTag, NSpace, NLayoutHeader } from 'naive-ui'
import { useMenu } from '@/store'
import type { MenuOption } from 'naive-ui'
import { NScrollbar, NTag, NSpace, NLayoutHeader, NDropdown } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import { useMenu, useSetting } from '@/store'
import { uuid } from '@/utils/hook'
import { hasClass } from '@/utils/element'
import type { MenuOption, ScrollbarInst } from 'naive-ui'
const MenuTag = defineComponent({
name: 'MenuTag',
setup() {
const menuStore = useMenu()
const { menuKey } = storeToRefs(menuStore)
const { menuModelValueChange, spliceMenTagOptions } = menuStore
const scrollRef = ref<ScrollbarInst | null>(null)
const modelMenuTagOptions = computed(() => menuStore.menuTagOptions)
const menuStore = useMenu()
const settingStore = useSetting()
const router = useRouter()
const { menuKey, menuTagOptions } = storeToRefs(menuStore)
const {
menuModelValueChange,
spliceMenTagOptions,
emptyMenuTagOptions,
setMenuTagOptions,
} = menuStore
const { changeSwitcher } = settingStore
const {
rootRoute: { path },
} = __APP_CFG__
const exclude = ['closeAll', 'closeRight', 'closeLeft', 'closeOther']
let currentContentmenuIndex = -1 // 当前右键标签页索引位置
const modelMenuTagOptions = computed(() => menuTagOptions.value)
const moreOptions = ref([
{
label: '重新加载',
key: 'reloadCurrentPage',
icon: () =>
h(
RayIcon,
{
size: 16,
name: 'reload',
},
{},
),
},
{
label: '关闭其他',
key: 'closeOther',
icon: () =>
h(
RayIcon,
{
size: 16,
name: 'other',
},
{},
),
},
{
label: '关闭右侧',
key: 'closeRight',
icon: () =>
h(
RayIcon,
{
size: 16,
name: 'right_arrow',
},
{},
),
},
{
label: '关闭左侧',
key: 'closeLeft',
icon: () =>
h(
RayIcon,
{
size: 16,
name: 'left_arrow',
},
{},
),
},
{
type: 'divider',
key: 'd1',
},
{
label: '全部关闭',
key: 'closeAll',
icon: () =>
h(
RayIcon,
{
size: 16,
name: 'close',
},
{},
),
disabled: false,
},
])
const scrollBarUUID = uuid()
const actionMap = {
reloadCurrentPage: () => {
changeSwitcher(false, 'reloadRouteSwitch')
setTimeout(() => changeSwitcher(true, 'reloadRouteSwitch'))
},
closeAll: () => {
/**
*
* , (dashboard)
* ,
*/
if (moreOptions.value.length > 1) {
emptyMenuTagOptions()
router.replace({
path: path,
})
}
},
closeRight: () => {
/**
*
*
*
* menuKey ,
*/
const length = moreOptions.value.length
const routeItem = modelMenuTagOptions.value[currentContentmenuIndex]
spliceMenTagOptions(currentContentmenuIndex + 1, length - 1)
if (menuKey.value !== routeItem.key) {
menuModelValueChange(routeItem.key, routeItem)
}
},
closeLeft: () => {
spliceMenTagOptions(0, currentContentmenuIndex)
},
closeOther: () => {
/**
*
*
*
* menuKey ,
*/
const routeItem = modelMenuTagOptions.value[currentContentmenuIndex]
if (menuKey.value !== routeItem.key) {
emptyMenuTagOptions()
menuModelValueChange(routeItem.key, routeItem)
} else {
setMenuTagOptions(routeItem, false)
}
},
}
/** 右键菜单 */
const actionState = reactive({
x: 0,
y: 0,
actionDropdownShow: false,
})
/**
*
@ -30,10 +185,10 @@ const MenuTag = defineComponent({
*
* @remark `tag` ,
*/
const handleCloseTag = (idx: number) => {
const closeCurrentMenuTag = (idx: number) => {
spliceMenTagOptions(idx)
if (menuKey.value !== '/dashboard') {
if (menuKey.value !== path) {
const options = modelMenuTagOptions.value
const length = options.length
@ -43,6 +198,19 @@ const MenuTag = defineComponent({
}
}
const setMoreOptionsDisabled = (
key: string | number,
disabled: boolean,
) => {
moreOptions.value.forEach((curr) => {
if (curr.key === key) {
curr.disabled = disabled
return
}
})
}
/**
*
* @param item
@ -51,35 +219,214 @@ const MenuTag = defineComponent({
menuModelValueChange(item.key as string, item)
}
const handleScrollX = (type: 'left' | 'right') => {
const scroll = document.getElementById(scrollBarUUID) // 获取滚动条容器
if (scroll) {
/**
*
* (class: n-scrollbar-container)
* scrollLeft ,
*/
const scrollContentElement = Array.from(
scroll.childNodes,
) as HTMLElement[]
const findElement = scrollContentElement.find((el) =>
hasClass(el, 'n-scrollbar-container'),
)
const scrollX = findElement!.scrollLeft || 0
const rolling =
type === 'left' ? Math.max(0, scrollX - 200) : scrollX + 200
scrollRef.value?.scrollTo({
left: rolling,
behavior: 'smooth',
})
}
}
/** 更多操作操作栏 */
const actionDropdownSelect = (key: string | number) => {
actionState.actionDropdownShow = false
actionMap[key]?.()
}
/**
*
*
*
* ()
*/
const handleContextMenu = (idx: number, e: MouseEvent) => {
e.preventDefault()
actionState.actionDropdownShow = false
currentContentmenuIndex = idx
nextTick().then(() => {
actionState.actionDropdownShow = true
actionState.x = e.clientX
actionState.y = e.clientY
})
}
const setDisabledAccordionToIndex = () => {
const length = modelMenuTagOptions.value.length - 1
if (currentContentmenuIndex === length) {
setMoreOptionsDisabled('closeRight', true)
} else if (currentContentmenuIndex < length) {
setMoreOptionsDisabled('closeRight', false)
}
if (currentContentmenuIndex === 0) {
setMoreOptionsDisabled('closeLeft', true)
} else if (currentContentmenuIndex > 0) {
setMoreOptionsDisabled('closeLeft', false)
}
}
/**
*
* , currentContentmenuIndex
*
*
*/
const setCurrentContentmenuIndex = () => {
const index = modelMenuTagOptions.value.findIndex(
(curr) => curr.key === menuKey.value,
)
currentContentmenuIndex = index
setDisabledAccordionToIndex()
}
/** 如果有且只有一个标签页时, 禁止全部关闭操作 */
watch(
() => modelMenuTagOptions.value,
(newData) => {
moreOptions.value.forEach((curr) => {
if (exclude.includes(curr.key)) {
newData.length > 1
? (curr.disabled = false)
: (curr.disabled = true)
}
})
},
{
immediate: true,
deep: true,
},
)
/** 动态设置关闭按钮是否可操作 */
watch(
() => actionState.actionDropdownShow,
() => {
setDisabledAccordionToIndex()
},
)
return {
modelMenuTagOptions,
menuModelValueChange,
handleCloseTag,
closeCurrentMenuTag,
menuKey,
handleTagClick,
moreOptions,
handleScrollX,
scrollRef,
scrollBarUUID,
actionDropdownSelect,
rootPath: path,
actionState,
handleContextMenu,
setCurrentContentmenuIndex,
}
},
render() {
return (
<NLayoutHeader>
<NScrollbar class="menu-tag" xScrollable>
<NSpace class="menu-tag-sapce" wrap={false} align="center">
<div class="menu-tag">
<NDropdown
options={this.moreOptions}
x={this.actionState.x}
y={this.actionState.y}
show={this.actionState.actionDropdownShow}
trigger="manual"
placement="bottom-start"
onClickoutside={() => {
this.actionState.actionDropdownShow = false
}}
onSelect={this.actionDropdownSelect.bind(this)}
/>
<NSpace
class="menu-tag-sapce"
wrap={false}
align="center"
justify="space-between"
inline
wrapItem={false}
>
<RayIcon
name="expanded"
width="20"
height="28"
customClassName="menu-tag__left-arrow"
onClick={this.handleScrollX.bind(this, 'left')}
/>
<NScrollbar xScrollable ref="scrollRef" id={this.scrollBarUUID}>
<NSpace
class="menu-tag-wrapper"
wrap={false}
align="center"
justify="start"
>
{this.modelMenuTagOptions.map((curr, idx) => (
<NTag
closable={
curr.key !== '/dashboard' &&
curr.key !== this.rootPath &&
this.modelMenuTagOptions.length > 1
}
onClose={() => this.handleCloseTag(idx)}
onClose={() => this.closeCurrentMenuTag(idx)}
type={curr.key === this.menuKey ? 'success' : 'info'}
onClick={this.handleTagClick.bind(this, curr)}
bordered={false}
onContextmenu={this.handleContextMenu.bind(this, idx)}
>
{typeof curr.label === 'function' ? curr.label() : curr.label}
{typeof curr.label === 'function'
? curr.label()
: curr.label}
</NTag>
))}
</NSpace>
</NScrollbar>
<div class="menu-tag__right-wrapper">
<RayIcon
name="expanded"
width="20"
height="28"
customClassName="menu-tag__right-arrow"
onClick={this.handleScrollX.bind(this, 'right')}
/>
<NDropdown
options={this.moreOptions}
trigger="click"
onSelect={this.actionDropdownSelect.bind(this)}
>
<RayIcon
name="more"
width="20"
height="28"
customClassName="menu-tag__right-setting"
onClick={this.setCurrentContentmenuIndex.bind(this)}
/>
</NDropdown>
</div>
</NSpace>
</div>
</NLayoutHeader>
)
},

View File

@ -127,7 +127,7 @@ const SettingDrawer = defineComponent({
</NDivider>
<NColorPicker
swatches={useSwatchesColorOptions()}
v-model:value={this.primaryColorOverride.common.primaryColor}
v-model:value={this.primaryColorOverride.common!.primaryColor}
onUpdateValue={this.changePrimaryColor.bind(this)}
/>
<NDivider titlePlacement="center"></NDivider>

View File

@ -86,7 +86,7 @@ const SiderBar = defineComponent({
reload: () => {
changeSwitcher(false, 'reloadRouteSwitch')
setTimeout(() => changeSwitcher(true, 'reloadRouteSwitch'), 1.5 * 1000)
setTimeout(() => changeSwitcher(true, 'reloadRouteSwitch'))
},
setting: () => {
showSettings.value = true

View File

@ -23,9 +23,7 @@
*/
import { useSignin } from '@/store'
const BASIC_ROUTER = ['login', 'error-page', 'doc']
const BASE_ROLES = ['admin']
import { whiteRoutes, superAdmin } from './configuration'
export const validRole = (options: IMenuOptions) => {
const { role } = storeToRefs(useSignin())
@ -35,11 +33,11 @@ export const validRole = (options: IMenuOptions) => {
meta?.hidden === undefined || meta?.hidden === false ? false : meta?.hidden
// 如果是超级管理员(预设为 admin), 则根据其菜单栏(hidden)字段判断是否显示
if (BASE_ROLES.includes(role.value)) {
if (superAdmin.length && superAdmin.includes(role.value)) {
return true && !hidden
} else {
// 如果为基础路由, 不进行鉴权则根据其菜单栏(hidden)字段判断是否显示
if (BASIC_ROUTER.includes(name)) {
if (whiteRoutes.includes(name)) {
return true && !hidden
}

View File

@ -0,0 +1,37 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-03-19
*
* @workspace ray-template
*
* @remark
*/
/**
*
*
*
* , meta hidden
* mete hidden false
*/
/**
*
*
*
* , name , ()
* ,
*
* , ( basic.ts 使)
*/
export const whiteRoutes = ['login', 'error-page', 'doc']
/**
*
*
*
* ,
*/
export const superAdmin = ['admin']

View File

@ -30,10 +30,14 @@ import type { Router, NavigationGuardNext } from 'vue-router'
export const permissionRouter = (router: Router) => {
const { beforeEach } = router
const redirectToDashboard = (next: NavigationGuardNext) => {
next('/dashboard')
const {
rootRoute: { path },
} = __APP_CFG__
setCache('menuKey', '/dashboard')
const redirectToDashboard = (next: NavigationGuardNext) => {
next(path)
setCache('menuKey', path)
}
beforeEach((to, from, next) => {

View File

@ -1,6 +1,10 @@
import Layout from '@/layout/index'
import childrenRoutes from './modules/index'
const {
rootRoute: { path },
} = __APP_CFG__
export const constantRoutes = [
{
path: '/',
@ -10,7 +14,7 @@ export const constantRoutes = [
{
path: '/',
name: 'layout',
redirect: '/dashboard',
redirect: path,
component: Layout,
children: childrenRoutes,
},

View File

@ -92,13 +92,15 @@ export const parse = (
/**
*
* @param item menu options
* @param key current menu key
* @param menuTagOptions menu tag options
*
* @remark
*/
export const matchMenuOption = (
item: IMenuOptions,
key: MenuKey,
menuTagOptions: TagMenuOptions[],
menuTagOptions: MenuTagOptions[],
) => {
if (item.path !== key) {
const tag = menuTagOptions.find((curr) => curr.path === item.path)

View File

@ -35,23 +35,44 @@ export const useMenu = defineStore(
const route = useRoute()
const { t } = useI18n()
const {
rootRoute: { path },
} = __APP_CFG__
const cacheMenuKey =
getCache('menuKey') === 'no' ? '/dashboard' : getCache('menuKey')
getCache('menuKey') === 'no' ? path : getCache('menuKey')
const menuState = reactive({
menuKey: cacheMenuKey as MenuKey, // 当前菜单 `key`
options: [] as IMenuOptions[], // 菜单列表
collapsed: false, // 是否折叠菜单
menuTagOptions: [] as TagMenuOptions[], // tag 标签菜单
menuTagOptions: [] as MenuTagOptions[], // tag 标签菜单
breadcrumbOptions: [] as IMenuOptions[], // 面包屑菜单
})
/**
*
* @param options menu options
* @param key target key
*
* @remark
*/
const getCompleteRoutePath = (
options: IMenuOptions[],
key: string | number,
) => {
const ops = parse(options, 'key', key)
return ops
}
/**
*
* @param key `key`
* @param item `item`
*
* `menu key`
* @remark `menu key`
* @remark , key (router push )
*/
const menuModelValueChange = (key: string | number, item: MenuOption) => {
const meta = item.meta as RouteMeta
@ -62,7 +83,7 @@ export const useMenu = defineStore(
// 防止重复点击做重复操作处理
if (menuState.menuKey !== key) {
matchMenuOption(
item as unknown as TagMenuOptions,
item as unknown as MenuTagOptions,
menuState.menuKey,
menuState.menuTagOptions,
)
@ -70,16 +91,17 @@ export const useMenu = defineStore(
menuState.breadcrumbOptions = parse(menuState.options, 'key', key) // 获取面包屑
if (key[0] !== '/') {
const p = menuState.breadcrumbOptions
const path = getCompleteRoutePath(menuState.options, key)
.map((curr) => curr.key)
.join('/')
router.push(p)
router.push(path)
} else {
router.push(item.path as string)
}
menuState.menuKey = key
setCache('menuKey', key)
}
}
@ -109,6 +131,28 @@ export const useMenu = defineStore(
matchMenuItem(menuState.options as MenuOption[])
}
/**
*
* @param optins menu tag option(s)
* @param isAppend true: (push), false:
*/
const setMenuTagOptions = (
optins: MenuTagOptions | MenuTagOptions[],
isAppend = true,
) => {
const isArray = Array.isArray(optins)
if (isAppend) {
isArray
? menuState.menuTagOptions.push(...optins)
: menuState.menuTagOptions.push(optins)
} else {
isArray
? (menuState.menuTagOptions = optins)
: (menuState.menuTagOptions = [optins])
}
}
/**
*
* @remark ,
@ -159,7 +203,7 @@ export const useMenu = defineStore(
: route
if (curr.path === cacheMenuKey) {
menuState.menuTagOptions.push(attr)
setMenuTagOptions(attr)
}
attr.show = validRole(curr)
@ -187,8 +231,23 @@ export const useMenu = defineStore(
const collapsedMenu = (collapsed: boolean) =>
(menuState.collapsed = collapsed)
const spliceMenTagOptions = (idx: number) =>
menuState.menuTagOptions.splice(idx, 1)
/**
*
* @param idx
* @param length
*
* @returns
*/
const spliceMenTagOptions = (idx: number, length = 1) =>
menuState.menuTagOptions.splice(idx, length)
/**
*
* @remark menuTagOptions
*/
const emptyMenuTagOptions = () => {
menuState.menuTagOptions = []
}
watch(
() => route.fullPath,
@ -206,6 +265,8 @@ export const useMenu = defineStore(
setupAppRoutes,
collapsedMenu,
spliceMenTagOptions,
emptyMenuTagOptions,
setMenuTagOptions,
}
},
{

View File

@ -23,6 +23,7 @@ export const useSetting = defineStore(
primaryColorOverride: {
common: {
primaryColor: '#2d8cf0', // 主题色
primaryColorHover: '#2d8cf0',
},
},
themeValue: false, // `true` 为黑夜主题, `false` 为白色主题
@ -44,6 +45,7 @@ export const useSetting = defineStore(
const changePrimaryColor = (value: string) => {
settingState.primaryColorOverride.common!.primaryColor = value
settingState.primaryColorOverride.common!.primaryColorHover = value
}
/**

View File

@ -15,6 +15,11 @@ export interface LayoutSideBarLogo {
export type LayoutCopyright = string | number | VNodeChild
export interface RootRoute {
name: string
path: string
}
export interface HTMLTitle {
name: string
transformIndexHtml: (title: string) => string
@ -28,6 +33,7 @@ export interface Config {
copyright?: LayoutCopyright
sideBarLogo?: LayoutSideBarLogo
mixinCSS?: string
rootRoute?: RootRoute
}
export type Recordable<T = unknown> = Record<string, T>
@ -43,6 +49,7 @@ export interface AppConfig {
copyright?: LayoutCopyright
sideBarLogo?: LayoutSideBarLogo
}
rootRoute: RootRoute
}
export type AppConfigExport = Config & UserConfigExport

View File

@ -17,7 +17,7 @@ declare global {
noLocalTitle?: string | number
}
declare interface TagMenuOptions extends IMenuOptions {}
declare interface MenuTagOptions extends IMenuOptions {}
declare type MenuKey = null | string | number
}

View File

@ -6,8 +6,12 @@ const ErrorPage = defineComponent({
setup() {
const router = useRouter()
const {
rootRoute: { path },
} = __APP_CFG__
const handleBack = () => {
router.push('/dashboard')
router.push(path)
}
return {

View File

@ -12,6 +12,9 @@ const Signin = defineComponent({
const signinStore = useSignin()
const { signin } = signinStore
const {
rootRoute: { path },
} = __APP_CFG__
const useSigninForm = () => ({
name: 'ray',
@ -48,7 +51,7 @@ const Signin = defineComponent({
setCache('token', 'tokenValue')
setCache('person', signinForm.value)
router.push('/dashboard')
router.push(path)
}, 2 * 1000)
}
} else {

View File

@ -35,6 +35,7 @@
"src/**/*.vue",
"src/*.ts",
"src/*.vue",
"src/*",
"components.d.ts",
"auto-imports.d.ts"
],

View File

@ -23,8 +23,16 @@ import config from './cfg'
import pkg from './package.json'
const { dependencies, devDependencies, name, version } = pkg
const { server, buildOptions, alias, title, copyright, sideBarLogo, mixinCSS } =
config
const {
server,
buildOptions,
alias,
title,
copyright,
sideBarLogo,
mixinCSS,
rootRoute,
} = config
/**
*
@ -42,6 +50,7 @@ const __APP_CFG__ = {
copyright,
sideBarLogo,
},
rootRoute,
}
// https://vitejs.dev/config/