mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
v3.3.5
This commit is contained in:
parent
14b96fccac
commit
c17f941169
13
CHANGELOG.md
13
CHANGELOG.md
@ -1,5 +1,18 @@
|
|||||||
# CHANGE LOG
|
# CHANGE LOG
|
||||||
|
|
||||||
|
## 3.3.5
|
||||||
|
|
||||||
|
### Feats
|
||||||
|
|
||||||
|
- Router Meta 属性支持自定义图标,不再局限于 RayIcon,支持自定义图标
|
||||||
|
- 更改部分组件默认值,默认值统一为 `null`
|
||||||
|
- 调整 validRole 方法逻辑,将该方法以前逻辑拆分为 validRole 与 validMenuItemShow 两个方法
|
||||||
|
- 新增使用手册
|
||||||
|
|
||||||
|
### 补充
|
||||||
|
|
||||||
|
> 由于文档已经拖更很久,所以补充一个使用手册。最近太忙了,一直忙着更新完善模板本身,文档的事情暂时没有时间去维护更新,所以与模板断层太久。。。后续有时间肯定会补上!!!
|
||||||
|
|
||||||
## 3.3.4
|
## 3.3.4
|
||||||
|
|
||||||
### Feats
|
### Feats
|
||||||
|
19
MANUAL.md
Normal file
19
MANUAL.md
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
## Ray Template 使用手册
|
||||||
|
|
||||||
|
## 前言
|
||||||
|
|
||||||
|
> `Ray Template` 默认使用 `yarn` 作为包管理器,并且默认启用严格模式的 `eslint`。
|
||||||
|
|
||||||
|
### 使用
|
||||||
|
|
||||||
|
#### 依赖安装
|
||||||
|
|
||||||
|
```sh
|
||||||
|
# yarn
|
||||||
|
yarn
|
||||||
|
|
||||||
|
# npm
|
||||||
|
npm i
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 未完待续。。。后续慢慢更新该文件
|
@ -51,4 +51,4 @@ export const WHITE_ROUTES = ['login', 'error-page', 'doc']
|
|||||||
* 超级管理员
|
* 超级管理员
|
||||||
* 配置默认超级管理员, 默认拥有全部最高权限
|
* 配置默认超级管理员, 默认拥有全部最高权限
|
||||||
*/
|
*/
|
||||||
export const SUPER_ADMIN = ['admin']
|
export const SUPER_ADMIN: (string | number)[] = ['admin']
|
||||||
|
@ -1,3 +1,22 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-Ray>
|
||||||
|
*
|
||||||
|
* @date 2023-06-14
|
||||||
|
*
|
||||||
|
* @workspace ray-template
|
||||||
|
*
|
||||||
|
* @remark 今天也是元气满满撸代码的一天
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* 全局注入 naive ui 提示性组件
|
||||||
|
* 使用该组件注册后, 可以直接通过 window.$message、window.$notification、window.$dialog、window.$loadingBar 访问
|
||||||
|
* 但是, 使用该组件注册后, 使用 window.$notification 组件时不能更改 placement 位置(只能默认右上角弹出)
|
||||||
|
* 如果需要更改弹出位置, 需要在需要地方重新定义组件注册
|
||||||
|
*/
|
||||||
|
|
||||||
import {
|
import {
|
||||||
NDialogProvider,
|
NDialogProvider,
|
||||||
NLoadingBarProvider,
|
NLoadingBarProvider,
|
||||||
|
@ -41,7 +41,7 @@ const RayIcon = defineComponent({
|
|||||||
customClassName: {
|
customClassName: {
|
||||||
/** 自定义 class name */
|
/** 自定义 class name */
|
||||||
type: String,
|
type: String,
|
||||||
default: '',
|
default: null,
|
||||||
},
|
},
|
||||||
depth: {
|
depth: {
|
||||||
/** 图标深度 */
|
/** 图标深度 */
|
||||||
|
@ -29,7 +29,7 @@ const RayIframe = defineComponent({
|
|||||||
iframeWrapperClass: {
|
iframeWrapperClass: {
|
||||||
/** 自定义类名 */
|
/** 自定义类名 */
|
||||||
type: String,
|
type: String,
|
||||||
default: '',
|
default: null,
|
||||||
},
|
},
|
||||||
frameborder: {
|
frameborder: {
|
||||||
/** 边框尺寸, 0 则不显示 */
|
/** 边框尺寸, 0 则不显示 */
|
||||||
@ -94,7 +94,7 @@ const RayIframe = defineComponent({
|
|||||||
default: () => ({}),
|
default: () => ({}),
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
setup(props) {
|
setup(props, { expose }) {
|
||||||
const cssVars = computed(() => {
|
const cssVars = computed(() => {
|
||||||
const cssVar = {
|
const cssVar = {
|
||||||
'--ray-iframe-frameborder': completeSize(props.frameborder),
|
'--ray-iframe-frameborder': completeSize(props.frameborder),
|
||||||
@ -125,6 +125,8 @@ const RayIframe = defineComponent({
|
|||||||
return iframeEl
|
return iframeEl
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expose()
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
on(getIframeRef(), 'load', iframeLoadSuccess.bind(this))
|
on(getIframeRef(), 'load', iframeLoadSuccess.bind(this))
|
||||||
on(getIframeRef(), 'error', iframeLoadError)
|
on(getIframeRef(), 'error', iframeLoadError)
|
||||||
|
@ -17,7 +17,7 @@ import RayIcon from '@/components/RayIcon/index'
|
|||||||
import { on, off } from '@/utils/element'
|
import { on, off } from '@/utils/element'
|
||||||
import { debounce } from 'lodash-es'
|
import { debounce } from 'lodash-es'
|
||||||
import { useMenu } from '@/store'
|
import { useMenu } from '@/store'
|
||||||
import { validRole } from '@/router/helper/routerCopilot'
|
import { validMenuItemShow } from '@/router/helper/routerCopilot'
|
||||||
|
|
||||||
import type { MenuOption } from 'naive-ui'
|
import type { MenuOption } from 'naive-ui'
|
||||||
import type { AppRouteMeta } from '@/router/type'
|
import type { AppRouteMeta } from '@/router/type'
|
||||||
@ -90,7 +90,7 @@ const GlobalSeach = defineComponent({
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
_breadcrumbLabel?.includes(_value) &&
|
_breadcrumbLabel?.includes(_value) &&
|
||||||
validRole(curr) &&
|
validMenuItemShow(curr) &&
|
||||||
!curr.children?.length
|
!curr.children?.length
|
||||||
) {
|
) {
|
||||||
arr.push(curr)
|
arr.push(curr)
|
||||||
|
@ -26,18 +26,44 @@ import type { Router } from 'vue-router'
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @remark 校验当前路由
|
* 校验当前菜单项是否与权限匹配
|
||||||
|
* 仅做对于 Meta Role 配置是否匹配做校验, 不关心 Meta Hidden 属性
|
||||||
|
*
|
||||||
|
* 如果为超级管理员, 则会默认获取所有权限
|
||||||
*/
|
*/
|
||||||
export const validRole = (option: IMenuOptions) => {
|
export const validRole = (option: IMenuOptions) => {
|
||||||
const { signinCallback } = storeToRefs(useSignin())
|
const { signinCallback } = storeToRefs(useSignin())
|
||||||
const role = computed(() => signinCallback.value.role)
|
const role = computed(() => signinCallback.value.role)
|
||||||
|
|
||||||
|
const { meta } = option
|
||||||
|
|
||||||
|
if (SUPER_ADMIN?.length && SUPER_ADMIN.includes(role.value)) {
|
||||||
|
return true
|
||||||
|
} else {
|
||||||
|
if (meta?.role) {
|
||||||
|
return meta.role.includes(role.value)
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @remark 校验当前路由
|
||||||
|
*
|
||||||
|
* 该方法进行校验时, 会将 hidden 与 role 一起进行校验
|
||||||
|
* 如果有一条不满足校验, 则视为校验失败
|
||||||
|
*
|
||||||
|
* 如果你仅仅是希望校验是否满足权限, 应该使用另一个方法 validRole
|
||||||
|
*/
|
||||||
|
export const validMenuItemShow = (option: IMenuOptions) => {
|
||||||
const { meta, name } = option
|
const { meta, name } = option
|
||||||
const hidden =
|
const hidden =
|
||||||
meta?.hidden === undefined || meta?.hidden === false ? false : meta?.hidden
|
meta?.hidden === undefined || meta?.hidden === false ? false : meta?.hidden
|
||||||
|
|
||||||
// 如果是超级管理员(预设为 admin), 则根据其菜单栏(hidden)字段判断是否显示
|
// 如果是超级管理员(预设为 admin), 则根据其菜单栏(hidden)字段判断是否显示
|
||||||
if (SUPER_ADMIN.length && SUPER_ADMIN.includes(role.value)) {
|
if (validRole(option)) {
|
||||||
return true && !hidden
|
return true && !hidden
|
||||||
} else {
|
} else {
|
||||||
// 如果为基础路由, 不进行鉴权则根据其菜单栏(hidden)字段判断是否显示
|
// 如果为基础路由, 不进行鉴权则根据其菜单栏(hidden)字段判断是否显示
|
||||||
@ -47,7 +73,7 @@ export const validRole = (option: IMenuOptions) => {
|
|||||||
|
|
||||||
// 判断权限是否匹配和菜单栏(hidden)字段判断是否显示
|
// 判断权限是否匹配和菜单栏(hidden)字段判断是否显示
|
||||||
if (meta?.role) {
|
if (meta?.role) {
|
||||||
return meta.role.includes(role.value) && !hidden
|
return validRole(option) && !hidden
|
||||||
}
|
}
|
||||||
|
|
||||||
return true && !hidden
|
return true && !hidden
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
import type { RouteRecordRaw } from 'vue-router'
|
import type { RouteRecordRaw } from 'vue-router'
|
||||||
import type { Recordable } from '@/types/type-utils'
|
import type { Recordable } from '@/types/type-utils'
|
||||||
import type { DefineComponent } from 'vue'
|
import type { DefineComponent, VNode } from 'vue'
|
||||||
|
|
||||||
export type Component<T = any> =
|
export type Component<T = any> =
|
||||||
| DefineComponent<{}, {}, any>
|
| DefineComponent<{}, {}, any>
|
||||||
@ -11,7 +11,7 @@ export type Component<T = any> =
|
|||||||
|
|
||||||
export interface AppRouteMeta {
|
export interface AppRouteMeta {
|
||||||
i18nKey?: string
|
i18nKey?: string
|
||||||
icon?: string
|
icon?: string | VNode
|
||||||
windowOpen?: string
|
windowOpen?: string
|
||||||
role?: string[]
|
role?: string[]
|
||||||
hidden?: boolean
|
hidden?: boolean
|
||||||
|
@ -11,6 +11,12 @@
|
|||||||
|
|
||||||
/** 本方法感谢 <https://yunkuangao.me/> 的支持 */
|
/** 本方法感谢 <https://yunkuangao.me/> 的支持 */
|
||||||
|
|
||||||
|
import { MENU_COLLAPSED_CONFIG, ROOT_ROUTE } from '@/appConfig/appConfig'
|
||||||
|
import RayIcon from '@/components/RayIcon/index'
|
||||||
|
import { validteValueType } from '@/utils/hook'
|
||||||
|
|
||||||
|
import type { VNode } from 'vue'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @param node 当前节点
|
* @param node 当前节点
|
||||||
@ -127,3 +133,26 @@ export const updateDocumentTitle = (option: IMenuOptions) => {
|
|||||||
|
|
||||||
document.title = breadcrumbLabel + ' - ' + spliceTitle
|
document.title = breadcrumbLabel + ' - ' + spliceTitle
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const hasMenuIcon = (option: IMenuOptions) => {
|
||||||
|
const { meta } = option
|
||||||
|
|
||||||
|
if (!meta.icon) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (validteValueType(meta.icon, 'Object')) {
|
||||||
|
return () => meta.icon
|
||||||
|
}
|
||||||
|
|
||||||
|
const icon = h(
|
||||||
|
RayIcon,
|
||||||
|
{
|
||||||
|
name: meta!.icon as string,
|
||||||
|
size: MENU_COLLAPSED_CONFIG.MENU_COLLAPSED_ICON_SIZE,
|
||||||
|
},
|
||||||
|
{},
|
||||||
|
)
|
||||||
|
|
||||||
|
return () => icon
|
||||||
|
}
|
||||||
|
@ -26,8 +26,13 @@ import { NEllipsis } from 'naive-ui'
|
|||||||
import RayIcon from '@/components/RayIcon/index'
|
import RayIcon from '@/components/RayIcon/index'
|
||||||
|
|
||||||
import { getCache, setCache } from '@/utils/cache'
|
import { getCache, setCache } from '@/utils/cache'
|
||||||
import { validRole } from '@/router/helper/routerCopilot'
|
import { validMenuItemShow } from '@/router/helper/routerCopilot'
|
||||||
import { parse, matchMenuOption, updateDocumentTitle } from './helper'
|
import {
|
||||||
|
parse,
|
||||||
|
matchMenuOption,
|
||||||
|
updateDocumentTitle,
|
||||||
|
hasMenuIcon,
|
||||||
|
} from './helper'
|
||||||
import { useI18n } from '@/locales/useI18n'
|
import { useI18n } from '@/locales/useI18n'
|
||||||
import { MENU_COLLAPSED_CONFIG, ROOT_ROUTE } from '@/appConfig/appConfig'
|
import { MENU_COLLAPSED_CONFIG, ROOT_ROUTE } from '@/appConfig/appConfig'
|
||||||
import routeModules from '@/router/routeModules'
|
import routeModules from '@/router/routeModules'
|
||||||
@ -186,21 +191,10 @@ export const useMenu = defineStore(
|
|||||||
}),
|
}),
|
||||||
breadcrumbLabel: label.value,
|
breadcrumbLabel: label.value,
|
||||||
} as IMenuOptions
|
} as IMenuOptions
|
||||||
/** 是否有 icon */
|
/** 合并 icon */
|
||||||
const expandIcon = {
|
const attr: IMenuOptions = Object.assign({}, route, {
|
||||||
icon: () =>
|
icon: hasMenuIcon(option),
|
||||||
h(
|
})
|
||||||
RayIcon,
|
|
||||||
{
|
|
||||||
name: meta!.icon as string,
|
|
||||||
size: MENU_COLLAPSED_CONFIG.MENU_COLLAPSED_ICON_SIZE,
|
|
||||||
},
|
|
||||||
{},
|
|
||||||
),
|
|
||||||
}
|
|
||||||
const attr: IMenuOptions = meta?.icon
|
|
||||||
? Object.assign({}, route, expandIcon)
|
|
||||||
: route
|
|
||||||
|
|
||||||
if (option.path === cacheMenuKey) {
|
if (option.path === cacheMenuKey) {
|
||||||
/** 设置菜单标签 */
|
/** 设置菜单标签 */
|
||||||
@ -209,7 +203,8 @@ export const useMenu = defineStore(
|
|||||||
updateDocumentTitle(attr)
|
updateDocumentTitle(attr)
|
||||||
}
|
}
|
||||||
|
|
||||||
attr.show = validRole(option)
|
/** 检查该菜单项是否展示 */
|
||||||
|
attr.show = validMenuItemShow(option)
|
||||||
|
|
||||||
return attr
|
return attr
|
||||||
}
|
}
|
||||||
@ -218,9 +213,10 @@ export const useMenu = defineStore(
|
|||||||
const catchArr: IMenuOptions[] = []
|
const catchArr: IMenuOptions[] = []
|
||||||
|
|
||||||
for (const curr of routes) {
|
for (const curr of routes) {
|
||||||
if (curr.children?.length && validRole(curr)) {
|
if (curr.children?.length && validMenuItemShow(curr)) {
|
||||||
curr.children = resolveRoutes(curr.children, index++)
|
curr.children = resolveRoutes(curr.children, index++)
|
||||||
} else if (!validRole(curr)) {
|
} else if (!validMenuItemShow(curr)) {
|
||||||
|
/** 如果校验失败, 则不会添加进 menu options */
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user