xiaoxian521
1 year ago
23 changed files with 778 additions and 957 deletions
-
3build/optimize.ts
-
32package.json
-
1149pnpm-lock.yaml
-
2public/serverConfig.json
-
4src/directives/auth/index.ts
-
33src/directives/copy/index.ts
-
27src/directives/elResizeDetector/index.ts
-
3src/directives/index.ts
-
55src/directives/optimize/index.ts
-
9src/layout/components/panel/index.vue
-
19src/layout/components/sidebar/horizontal.vue
-
11src/layout/components/sidebar/mixNav.vue
-
43src/layout/components/sidebar/vertical.vue
-
76src/layout/components/tag/index.vue
-
33src/layout/hooks/useNav.ts
-
12src/layout/index.vue
-
2src/layout/types.ts
-
5src/router/index.ts
-
2src/router/utils.ts
-
1src/store/modules/types.ts
-
12src/utils/mitt.ts
-
93types/global.d.ts
-
105types/router.d.ts
1149
pnpm-lock.yaml
File diff suppressed because it is too large
View File
File diff suppressed because it is too large
View File
@ -0,0 +1,33 @@ |
|||
import { message } from "@/utils/message"; |
|||
import { useEventListener } from "@vueuse/core"; |
|||
import { copyTextToClipboard } from "@pureadmin/utils"; |
|||
import { Directive, type DirectiveBinding } from "vue"; |
|||
|
|||
interface CopyEl extends HTMLElement { |
|||
copyValue: string; |
|||
} |
|||
|
|||
/** 文本复制指令(默认双击复制) */ |
|||
export const copy: Directive = { |
|||
mounted(el: CopyEl, binding: DirectiveBinding) { |
|||
const { value } = binding; |
|||
if (value) { |
|||
el.copyValue = value; |
|||
const arg = binding.arg ?? "dblclick"; |
|||
// Register using addEventListener on mounted, and removeEventListener automatically on unmounted
|
|||
useEventListener(el, arg, () => { |
|||
const success = copyTextToClipboard(el.copyValue); |
|||
success |
|||
? message("复制成功", { type: "success" }) |
|||
: message("复制失败", { type: "error" }); |
|||
}); |
|||
} else { |
|||
throw new Error( |
|||
'[Directive: copy]: need value! Like v-copy="modelValue"' |
|||
); |
|||
} |
|||
}, |
|||
updated(el: CopyEl, binding: DirectiveBinding) { |
|||
el.copyValue = binding.value; |
|||
} |
|||
}; |
@ -1,27 +0,0 @@ |
|||
import { Directive, type DirectiveBinding, type VNode } from "vue"; |
|||
import elementResizeDetectorMaker from "element-resize-detector"; |
|||
import type { Erd } from "element-resize-detector"; |
|||
import { emitter } from "@/utils/mitt"; |
|||
|
|||
const erd: Erd = elementResizeDetectorMaker({ |
|||
strategy: "scroll" |
|||
}); |
|||
|
|||
export const resize: Directive = { |
|||
mounted(el: HTMLElement, binding?: DirectiveBinding, vnode?: VNode) { |
|||
erd.listenTo(el, elem => { |
|||
const width = elem.offsetWidth; |
|||
const height = elem.offsetHeight; |
|||
if (binding?.instance) { |
|||
emitter.emit("resize", { detail: { width, height } }); |
|||
} else { |
|||
vnode.el.dispatchEvent( |
|||
new CustomEvent("resize", { detail: { width, height } }) |
|||
); |
|||
} |
|||
}); |
|||
}, |
|||
unmounted(el: HTMLElement) { |
|||
erd.uninstall(el); |
|||
} |
|||
}; |
@ -1,2 +1,3 @@ |
|||
export * from "./auth"; |
|||
export * from "./elResizeDetector"; |
|||
export * from "./copy"; |
|||
export * from "./optimize"; |
@ -0,0 +1,55 @@ |
|||
import { |
|||
isFunction, |
|||
isObject, |
|||
isArray, |
|||
debounce, |
|||
throttle |
|||
} from "@pureadmin/utils"; |
|||
import { useEventListener } from "@vueuse/core"; |
|||
import { Directive, type DirectiveBinding } from "vue"; |
|||
|
|||
/** 防抖(v-optimize或v-optimize:debounce)、节流(v-optimize:throttle)指令 */ |
|||
export const optimize: Directive = { |
|||
mounted(el: HTMLElement, binding: DirectiveBinding) { |
|||
const { value } = binding; |
|||
const optimizeType = binding.arg ?? "debounce"; |
|||
const type = ["debounce", "throttle"].find(t => t === optimizeType); |
|||
if (type) { |
|||
if (value && value.event && isFunction(value.fn)) { |
|||
let params = value?.params; |
|||
if (params) { |
|||
if (isArray(params) || isObject(params)) { |
|||
params = isObject(params) ? Array.of(params) : params; |
|||
} else { |
|||
throw new Error( |
|||
"[Directive: optimize]: `params` must be an array or object" |
|||
); |
|||
} |
|||
} |
|||
// Register using addEventListener on mounted, and removeEventListener automatically on unmounted
|
|||
useEventListener( |
|||
el, |
|||
value.event, |
|||
type === "debounce" |
|||
? debounce( |
|||
params ? () => value.fn(...params) : value.fn, |
|||
value?.timeout ?? 200, |
|||
value?.immediate ?? false |
|||
) |
|||
: throttle( |
|||
params ? () => value.fn(...params) : value.fn, |
|||
value?.timeout ?? 1000 |
|||
) |
|||
); |
|||
} else { |
|||
throw new Error( |
|||
"[Directive: optimize]: `event` and `fn` are required, and `fn` must be a function" |
|||
); |
|||
} |
|||
} else { |
|||
throw new Error( |
|||
"[Directive: optimize]: only `debounce` and `throttle` are supported" |
|||
); |
|||
} |
|||
} |
|||
}; |
@ -1,21 +1,13 @@ |
|||
import type { Emitter } from "mitt"; |
|||
import mitt from "mitt"; |
|||
|
|||
/** 全局公共事件需要在此处添加类型 */ |
|||
type Events = { |
|||
resize: { |
|||
detail: { |
|||
width: number; |
|||
height: number; |
|||
}; |
|||
}; |
|||
openPanel: string; |
|||
tagViewsChange: string; |
|||
tagViewsShowModel: string; |
|||
logoChange: boolean; |
|||
changLayoutRoute: { |
|||
indexPath: string; |
|||
parentPath: string; |
|||
}; |
|||
changLayoutRoute: string; |
|||
}; |
|||
|
|||
export const emitter: Emitter<Events> = mitt<Events>(); |
@ -0,0 +1,105 @@ |
|||
// 全局路由类型声明
|
|||
|
|||
import { type RouteComponent, type RouteLocationNormalized } from "vue-router"; |
|||
|
|||
declare global { |
|||
interface ToRouteType extends RouteLocationNormalized { |
|||
meta: CustomizeRouteMeta; |
|||
} |
|||
|
|||
/** |
|||
* @description 完整子路由的`meta`配置表 |
|||
*/ |
|||
interface CustomizeRouteMeta { |
|||
/** 菜单名称(兼容国际化、非国际化,如何用国际化的写法就必须在根目录的`locales`文件夹下对应添加) `必填` */ |
|||
title: string; |
|||
/** 菜单图标 `可选` */ |
|||
icon?: string | FunctionalComponent | IconifyIcon; |
|||
/** 菜单名称右侧的额外图标 */ |
|||
extraIcon?: string | FunctionalComponent | IconifyIcon; |
|||
/** 是否在菜单中显示(默认`true`)`可选` */ |
|||
showLink?: boolean; |
|||
/** 是否显示父级菜单 `可选` */ |
|||
showParent?: boolean; |
|||
/** 页面级别权限设置 `可选` */ |
|||
roles?: Array<string>; |
|||
/** 按钮级别权限设置 `可选` */ |
|||
auths?: Array<string>; |
|||
/** 路由组件缓存(开启 `true`、关闭 `false`)`可选` */ |
|||
keepAlive?: boolean; |
|||
/** 内嵌的`iframe`链接 `可选` */ |
|||
frameSrc?: string; |
|||
/** `iframe`页是否开启首次加载动画(默认`true`)`可选` */ |
|||
frameLoading?: boolean; |
|||
/** 页面加载动画(有两种形式,一种直接采用vue内置的`transitions`动画,另一种是使用`animate.css`写进、离场动画)`可选` */ |
|||
transition?: { |
|||
/** |
|||
* @description 当前路由动画效果 |
|||
* @see {@link https://next.router.vuejs.org/guide/advanced/transitions.html#transitions}
|
|||
* @see animate.css {@link https://animate.style}
|
|||
*/ |
|||
name?: string; |
|||
/** 进场动画 */ |
|||
enterTransition?: string; |
|||
/** 离场动画 */ |
|||
leaveTransition?: string; |
|||
}; |
|||
// 是否不添加信息到标签页,(默认`false`)
|
|||
hiddenTag?: boolean; |
|||
/** 动态路由可打开的最大数量 `可选` */ |
|||
dynamicLevel?: number; |
|||
/** 将某个菜单激活 |
|||
* (主要用于通过`query`或`params`传参的路由,当它们通过配置`showLink: false`后不在菜单中显示,就不会有任何菜单高亮, |
|||
* 而通过设置`activePath`指定激活菜单即可获得高亮,`activePath`为指定激活菜单的`path`) |
|||
*/ |
|||
activePath?: string; |
|||
} |
|||
|
|||
/** |
|||
* @description 完整子路由配置表 |
|||
*/ |
|||
interface RouteChildrenConfigsTable { |
|||
/** 子路由地址 `必填` */ |
|||
path: string; |
|||
/** 路由名字(对应不要重复,和当前组件的`name`保持一致)`必填` */ |
|||
name?: string; |
|||
/** 路由重定向 `可选` */ |
|||
redirect?: string; |
|||
/** 按需加载组件 `可选` */ |
|||
component?: RouteComponent; |
|||
meta?: CustomizeRouteMeta; |
|||
/** 子路由配置项 */ |
|||
children?: Array<RouteChildrenConfigsTable>; |
|||
} |
|||
|
|||
/** |
|||
* @description 整体路由配置表(包括完整子路由) |
|||
*/ |
|||
interface RouteConfigsTable { |
|||
/** 路由地址 `必填` */ |
|||
path: string; |
|||
/** 路由名字(保持唯一)`可选` */ |
|||
name?: string; |
|||
/** `Layout`组件 `可选` */ |
|||
component?: RouteComponent; |
|||
/** 路由重定向 `可选` */ |
|||
redirect?: string; |
|||
meta?: { |
|||
/** 菜单名称(兼容国际化、非国际化,如何用国际化的写法就必须在根目录的`locales`文件夹下对应添加)`必填` */ |
|||
title: string; |
|||
/** 菜单图标 `可选` */ |
|||
icon?: string | FunctionalComponent | IconifyIcon; |
|||
/** 是否在菜单中显示(默认`true`)`可选` */ |
|||
showLink?: boolean; |
|||
/** 菜单升序排序,值越高排的越后(只针对顶级路由)`可选` */ |
|||
rank?: number; |
|||
}; |
|||
/** 子路由配置项 */ |
|||
children?: Array<RouteChildrenConfigsTable>; |
|||
} |
|||
} |
|||
|
|||
// https://router.vuejs.org/zh/guide/advanced/meta.html#typescript
|
|||
declare module "vue-router" { |
|||
interface RouteMeta extends CustomizeRouteMeta {} |
|||
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue