This commit is contained in:
ray_wuhao 2023-06-15 13:40:18 +08:00
parent 14b96fccac
commit c17f941169
11 changed files with 135 additions and 31 deletions

View File

@ -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
View File

@ -0,0 +1,19 @@
## Ray Template 使用手册
## 前言
> `Ray Template` 默认使用 `yarn` 作为包管理器,并且默认启用严格模式的 `eslint`
### 使用
#### 依赖安装
```sh
# yarn
yarn
# npm
npm i
```
#### 未完待续。。。后续慢慢更新该文件

View File

@ -51,4 +51,4 @@ export const WHITE_ROUTES = ['login', 'error-page', 'doc']
* *
* , * ,
*/ */
export const SUPER_ADMIN = ['admin'] export const SUPER_ADMIN: (string | number)[] = ['admin']

View File

@ -1,3 +1,22 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-14
*
* @workspace ray-template
*
* @remark
*/
/**
*
* naive ui
* 使, window.$messagewindow.$notificationwindow.$dialogwindow.$loadingBar 访
* , 使, 使 window.$notification placement ()
* ,
*/
import { import {
NDialogProvider, NDialogProvider,
NLoadingBarProvider, NLoadingBarProvider,

View File

@ -41,7 +41,7 @@ const RayIcon = defineComponent({
customClassName: { customClassName: {
/** 自定义 class name */ /** 自定义 class name */
type: String, type: String,
default: '', default: null,
}, },
depth: { depth: {
/** 图标深度 */ /** 图标深度 */

View File

@ -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)

View File

@ -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)

View File

@ -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

View File

@ -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

View File

@ -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
}

View File

@ -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
} }