mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
v3.1.4发布
This commit is contained in:
parent
0aab9340a0
commit
6a931c0cbb
@ -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,
|
||||
},
|
||||
],
|
||||
},
|
||||
}
|
||||
|
13
CHANGELOG.md
13
CHANGELOG.md
@ -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
|
||||
|
@ -27,6 +27,7 @@
|
||||
## 功能
|
||||
|
||||
- 主题切换
|
||||
- 带有拓展功能的表格
|
||||
- 封装 `axios` 自动取消重复请求
|
||||
- 动态菜单(多级菜单)
|
||||
- 主题色切换
|
||||
@ -36,7 +37,6 @@
|
||||
- 国际化(允许按模块管理语言包)
|
||||
- 权限路由
|
||||
- 动态切换主题、贴花的 `EChart` 图
|
||||
- 带有拓展功能的表格
|
||||
- 最佳构建体验
|
||||
- 体积分析
|
||||
- 还有一些不值一提的小东西...
|
||||
|
11
cfg.ts
11
cfg.ts
@ -9,6 +9,17 @@ import {
|
||||
import type { AppConfigExport } from './src/types/cfg'
|
||||
|
||||
const config: AppConfigExport = {
|
||||
/**
|
||||
*
|
||||
* 配置根页面
|
||||
* 该项目所有重定向至首页, 都依赖该配置项
|
||||
*
|
||||
* 如果修改了该项目的首页路由配置, 需要更改该配置项, 以免重定向首页操作出现错误
|
||||
*/
|
||||
rootRoute: {
|
||||
name: 'dashboard',
|
||||
path: '/dashboard',
|
||||
},
|
||||
/**
|
||||
*
|
||||
* icon: LOGO 图标, 依赖 `RayIcon` 实现
|
||||
|
@ -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
6
src/icons/close.svg
Normal 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
12
src/icons/more.svg
Normal 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
9
src/icons/other.svg
Normal 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 |
@ -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 {
|
||||
|
@ -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">
|
||||
{this.modelMenuTagOptions.map((curr, idx) => (
|
||||
<NTag
|
||||
closable={
|
||||
curr.key !== '/dashboard' &&
|
||||
this.modelMenuTagOptions.length > 1
|
||||
}
|
||||
onClose={() => this.handleCloseTag(idx)}
|
||||
type={curr.key === this.menuKey ? 'success' : 'info'}
|
||||
onClick={this.handleTagClick.bind(this, curr)}
|
||||
bordered={false}
|
||||
<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"
|
||||
>
|
||||
{typeof curr.label === 'function' ? curr.label() : curr.label}
|
||||
</NTag>
|
||||
))}
|
||||
{this.modelMenuTagOptions.map((curr, idx) => (
|
||||
<NTag
|
||||
closable={
|
||||
curr.key !== this.rootPath &&
|
||||
this.modelMenuTagOptions.length > 1
|
||||
}
|
||||
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}
|
||||
</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>
|
||||
</NScrollbar>
|
||||
</div>
|
||||
</NLayoutHeader>
|
||||
)
|
||||
},
|
||||
|
@ -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>
|
||||
|
@ -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
|
||||
|
@ -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
|
||||
}
|
||||
|
||||
|
37
src/router/configuration.ts
Normal file
37
src/router/configuration.ts
Normal 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']
|
@ -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) => {
|
||||
|
@ -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,
|
||||
},
|
||||
|
@ -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)
|
||||
|
@ -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,
|
||||
}
|
||||
},
|
||||
{
|
||||
|
@ -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
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -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
|
||||
|
2
src/types/store.d.ts
vendored
2
src/types/store.d.ts
vendored
@ -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
|
||||
}
|
||||
|
@ -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 {
|
||||
|
@ -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 {
|
||||
|
@ -35,6 +35,7 @@
|
||||
"src/**/*.vue",
|
||||
"src/*.ts",
|
||||
"src/*.vue",
|
||||
"src/*",
|
||||
"components.d.ts",
|
||||
"auto-imports.d.ts"
|
||||
],
|
||||
|
@ -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/
|
||||
|
Loading…
x
Reference in New Issue
Block a user