v4.3.1
17
CHANGELOG.md
@ -1,5 +1,22 @@
|
||||
# CHANGE LOG
|
||||
|
||||
## 4.3.1
|
||||
|
||||
根据反馈,尽可能的补充了一些代码注释。
|
||||
|
||||
### Feats
|
||||
|
||||
- 标签页右键菜单新增关闭当前页功能,优化了文案
|
||||
- `utils/basic` 包中的部分方法改为 effect 执行逻辑,避免使用 ref 注册的 dom 不能正确的被获取的问题
|
||||
- 新增 `scopeDispose`, `watchEffectWithTarget` 方法
|
||||
- `utils/cache` 新增 `hasStorage` 方法
|
||||
- 现在标签页会缓存,不再随着刷新后丢失
|
||||
- 新增 maximize 方法,并且基于该方法实现 LayoutContent 全屏效果
|
||||
|
||||
### Fixes
|
||||
|
||||
- 修复标签页右键菜单闪烁问题
|
||||
|
||||
## 4.3.0
|
||||
|
||||
提供了专用于一些模板的 `hooks`,可以通过这些方法调用模板的特定功能。并且该功能后续是模板维护的重点。
|
||||
|
@ -1,7 +1,7 @@
|
||||
{
|
||||
"name": "ray-template",
|
||||
"private": false,
|
||||
"version": "4.3.0",
|
||||
"version": "4.3.1",
|
||||
"type": "module",
|
||||
"engines": {
|
||||
"node": ">=16.0.0",
|
||||
|
@ -15,7 +15,7 @@ import { NInput, NForm, NFormItem, NButton, NSpace } from 'naive-ui'
|
||||
import AppAvatar from '@/app-components/app/AppAvatar/index'
|
||||
|
||||
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
|
||||
import { rules, useCondition } from '@/app-components/app/AppLockScreen/hook'
|
||||
import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared'
|
||||
import { useSettingGetters, useSettingActions } from '@/store'
|
||||
|
||||
import type { FormInst, InputInst } from 'naive-ui'
|
||||
|
@ -16,7 +16,7 @@ import AppAvatar from '@/app-components/app/AppAvatar/index'
|
||||
|
||||
import dayjs from 'dayjs'
|
||||
import { useSigningActions, useSettingActions } from '@/store'
|
||||
import { rules, useCondition } from '@/app-components/app/AppLockScreen/hook'
|
||||
import { rules, useCondition } from '@/app-components/app/AppLockScreen/shared'
|
||||
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
|
||||
import { useDevice } from '@/hooks/web/index'
|
||||
|
||||
@ -98,6 +98,9 @@ export default defineComponent({
|
||||
},
|
||||
render() {
|
||||
const { isTabletOrSmaller } = this
|
||||
const { HH_MM, AM_PM, YY_MM_DD, DDD } = this
|
||||
const hmSplit = HH_MM.split(':')
|
||||
const { unlockScreen, backToSigning } = this
|
||||
|
||||
return (
|
||||
<div class="app-lock-screen__unlock">
|
||||
@ -110,8 +113,8 @@ export default defineComponent({
|
||||
: '',
|
||||
]}
|
||||
>
|
||||
<div class="left">{this.HH_MM?.split(':')[0]}</div>
|
||||
<div class="right">{this.HH_MM?.split(':')[1]}</div>
|
||||
<div class="left">{hmSplit[0]}</div>
|
||||
<div class="right">{hmSplit[1]}</div>
|
||||
</div>
|
||||
<div class="app-lock-screen__unlock__content-avatar">
|
||||
<AppAvatar vertical align="center" avatarSize={52} />
|
||||
@ -129,24 +132,16 @@ export default defineComponent({
|
||||
maxlength={12}
|
||||
onKeydown={(e: KeyboardEvent) => {
|
||||
if (e.code === 'Enter') {
|
||||
this.unlockScreen()
|
||||
unlockScreen()
|
||||
}
|
||||
}}
|
||||
/>
|
||||
</NFormItem>
|
||||
<NSpace justify="space-between">
|
||||
<NButton
|
||||
type="primary"
|
||||
text
|
||||
onClick={this.backToSigning.bind(this)}
|
||||
>
|
||||
<NButton type="primary" text onClick={backToSigning.bind(this)}>
|
||||
返回登陆
|
||||
</NButton>
|
||||
<NButton
|
||||
type="primary"
|
||||
text
|
||||
onClick={this.unlockScreen.bind(this)}
|
||||
>
|
||||
<NButton type="primary" text onClick={unlockScreen.bind(this)}>
|
||||
进入系统
|
||||
</NButton>
|
||||
</NSpace>
|
||||
@ -154,10 +149,10 @@ export default defineComponent({
|
||||
</div>
|
||||
<div class="app-lock-screen__unlock__content-date">
|
||||
<div class="current-date">
|
||||
{this.HH_MM} <span>{this.AM_PM}</span>
|
||||
{HH_MM} <span>{AM_PM}</span>
|
||||
</div>
|
||||
<div class="current-year">
|
||||
{this.YY_MM_DD} <span>{this.DDD}</span>
|
||||
{YY_MM_DD} <span>{DDD}</span>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
@ -18,7 +18,7 @@ import { useSettingGetters } from '@/store'
|
||||
|
||||
import type { SettingState } from '@/store/modules/setting/type'
|
||||
|
||||
const AppStyleProvider = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'AppStyleProvider',
|
||||
setup(_, { expose }) {
|
||||
const { getAppTheme } = useSettingGetters()
|
||||
@ -33,22 +33,22 @@ const AppStyleProvider = defineComponent({
|
||||
const primaryColorOverride = getStorage<SettingState>(
|
||||
'piniaSettingStore',
|
||||
'localStorage',
|
||||
)
|
||||
) // 获取缓存 naive ui 配置项
|
||||
|
||||
if (primaryColorOverride) {
|
||||
const _p = get(
|
||||
const p = get(
|
||||
primaryColorOverride,
|
||||
'primaryColorOverride.common.primaryColor',
|
||||
primaryColor,
|
||||
)
|
||||
const _fp = colorToRgba(_p, 0.38)
|
||||
) // 获取主色调
|
||||
const fp = colorToRgba(p, 0.38) // 将主色调任意颜色转换为 rgba 格式
|
||||
|
||||
/** 设置全局主题色 css 变量 */
|
||||
body.style.setProperty('--ray-theme-primary-color', _p)
|
||||
body.style.setProperty('--ray-theme-primary-color', p) // 主色调
|
||||
body.style.setProperty(
|
||||
'--ray-theme-primary-fade-color',
|
||||
_fp || primaryFadeColor,
|
||||
)
|
||||
fp || primaryFadeColor,
|
||||
) // 降低透明度后的主色调
|
||||
}
|
||||
}
|
||||
|
||||
@ -73,8 +73,8 @@ const AppStyleProvider = defineComponent({
|
||||
* 根据 getAppTheme 进行初始化
|
||||
*/
|
||||
const body = document.body
|
||||
const darkClassName = 'ray-template--dark'
|
||||
const lightClassName = 'ray-template--light'
|
||||
const darkClassName = 'ray-template--dark' // 暗色类名
|
||||
const lightClassName = 'ray-template--light' // 明亮色类名
|
||||
|
||||
bool
|
||||
? removeClass(body, lightClassName)
|
||||
@ -86,6 +86,7 @@ const AppStyleProvider = defineComponent({
|
||||
syncPrimaryColorToBody()
|
||||
hiddenLoadingAnimation()
|
||||
|
||||
// 当切换主题时,更新 body 当前的注入 class
|
||||
watch(
|
||||
() => getAppTheme.value,
|
||||
(ndata) => {
|
||||
@ -102,5 +103,3 @@ const AppStyleProvider = defineComponent({
|
||||
return <div class="app-style-provider"></div>
|
||||
},
|
||||
})
|
||||
|
||||
export default AppStyleProvider
|
||||
|
@ -9,6 +9,14 @@
|
||||
* @remark 今天也是元气满满撸代码的一天
|
||||
*/
|
||||
|
||||
/**
|
||||
*
|
||||
* 全局水印注入
|
||||
*
|
||||
* 该组件启用时,会在全局(包括首页)展示
|
||||
* 如果你不希望在登录页显示,可以手动将该组件放置于 Layout 中
|
||||
*/
|
||||
|
||||
import { NWatermark } from 'naive-ui'
|
||||
|
||||
import { APP_WATERMARK_CONFIG } from '@/app-config/appConfig'
|
||||
|
@ -36,20 +36,17 @@ const { createAxiosInstance, beforeFetch, fetchError } = useAxiosInterceptor()
|
||||
// 请求拦截器
|
||||
server.interceptors.request.use(
|
||||
(request) => {
|
||||
// 生成 request instance
|
||||
createAxiosInstance(request, 'requestInstance')
|
||||
// 初始化拦截器所有已注入方法
|
||||
setupRequestInterceptor()
|
||||
// 执行拦截器所有已注入方法
|
||||
beforeFetch('requestInstance', 'implementRequestInterceptorArray', 'ok')
|
||||
createAxiosInstance(request, 'requestInstance') // 生成 request instance
|
||||
|
||||
setupRequestInterceptor() // 初始化拦截器所有已注入方法
|
||||
|
||||
beforeFetch('requestInstance', 'implementRequestInterceptorArray', 'ok') // 执行拦截器所有已注入方法
|
||||
|
||||
return request
|
||||
},
|
||||
(error) => {
|
||||
// 初始化拦截器所有已注入方法(错误状态)
|
||||
setupRequestErrorInterceptor()
|
||||
// 执行所有已注入方法
|
||||
fetchError('requestError', error, 'implementRequestInterceptorErrorArray')
|
||||
setupRequestErrorInterceptor() // 初始化拦截器所有已注入方法(错误状态)
|
||||
fetchError('requestError', error, 'implementRequestInterceptorErrorArray') // 执行所有已注入方法
|
||||
|
||||
return Promise.reject(error)
|
||||
},
|
||||
@ -58,17 +55,17 @@ server.interceptors.request.use(
|
||||
// 响应拦截器
|
||||
server.interceptors.response.use(
|
||||
(response) => {
|
||||
createAxiosInstance(response, 'responseInstance')
|
||||
setupResponseInterceptor()
|
||||
beforeFetch('responseInstance', 'implementResponseInterceptorArray', 'ok')
|
||||
createAxiosInstance(response, 'responseInstance') // 创建响应实例
|
||||
setupResponseInterceptor() // 注入响应成功待执行队列
|
||||
beforeFetch('responseInstance', 'implementResponseInterceptorArray', 'ok') // 执行响应成功拦截器
|
||||
|
||||
const { data } = response
|
||||
|
||||
return Promise.resolve(data)
|
||||
},
|
||||
(error) => {
|
||||
setupResponseErrorInterceptor()
|
||||
fetchError('responseError', error, 'implementResponseInterceptorErrorArray')
|
||||
setupResponseErrorInterceptor() // 注入响应失败待执行队列
|
||||
fetchError('responseError', error, 'implementResponseInterceptorErrorArray') // 执行响应失败后拦截器
|
||||
|
||||
return Promise.reject(error)
|
||||
},
|
||||
|
@ -19,7 +19,11 @@ import { AwesomeQR } from 'awesome-qr'
|
||||
import { isValueType, downloadAnyFile } from '@/utils/basic'
|
||||
import { call } from '@/utils/vue/index'
|
||||
|
||||
import type { QRCodeRenderResponse, GIFBuffer } from './type'
|
||||
import type {
|
||||
QRCodeRenderResponse,
|
||||
GIFBuffer,
|
||||
DownloadFilenameType,
|
||||
} from './type'
|
||||
import type { WatchStopHandle } from 'vue'
|
||||
|
||||
const readGIFAsArrayBuffer = (url: string): Promise<GIFBuffer> => {
|
||||
@ -114,7 +118,7 @@ export default defineComponent({
|
||||
}
|
||||
}
|
||||
|
||||
const downloadQRCode = (fileName?: string) => {
|
||||
const downloadQRCode = (fileName?: DownloadFilenameType) => {
|
||||
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
|
||||
downloadAnyFile(
|
||||
qrcodeURL.value,
|
||||
|
@ -22,7 +22,11 @@ export type QRCodeInst = {
|
||||
*
|
||||
* 如果未设置名称,则默认以 时间戳.png 命名
|
||||
*/
|
||||
downloadQRCode: (fileName?: string) => void
|
||||
downloadQRCode: (fileName?: DownloadFilenameType) => void
|
||||
}
|
||||
|
||||
export type GIFBuffer = string | ArrayBuffer | null
|
||||
|
||||
export type DefaultDownloadImageType = 'png' | 'jpg' | 'jpeg' | 'webp'
|
||||
|
||||
export type DownloadFilenameType = `${string}.${DefaultDownloadImageType}`
|
||||
|
@ -9,23 +9,46 @@
|
||||
* @remark 今天也是元气满满撸代码的一天
|
||||
*/
|
||||
|
||||
import { getVariableToRefs, setVariable } from '@/global-variable/index'
|
||||
import { setVariable } from '@/global-variable/index'
|
||||
import { LAYOUT_CONTENT_REF } from '@/app-config/routerConfig'
|
||||
import { useFullscreen } from 'vue-hooks-plus'
|
||||
import { useI18n } from '@/hooks/web/index'
|
||||
import { addStyle, removeStyle } from '@/utils/element'
|
||||
import { unrefElement } from '@/utils/vue/index'
|
||||
|
||||
import type { AppMenuOption } from '@/types/modules/app'
|
||||
import type { Ref } from 'vue'
|
||||
|
||||
export function useMainPage() {
|
||||
/**
|
||||
*
|
||||
* @param wait 延迟时长
|
||||
*
|
||||
* 刷新当前路由
|
||||
*/
|
||||
const reload = (wait = 800) => {
|
||||
setVariable('globalMainLayoutLoad', false)
|
||||
|
||||
setTimeout(() => setVariable('globalMainLayoutLoad', true), wait)
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param full 是否网页全屏内容区域
|
||||
*
|
||||
* 该方法仅针对于 LayoutContent 区域,并且依赖全局属性 layoutContentMaximize
|
||||
*/
|
||||
const maximize = (full: boolean) => {
|
||||
// setVariable('layoutContentMaximize', full)
|
||||
const contentEl = unrefElement(LAYOUT_CONTENT_REF as Ref<HTMLElement>)
|
||||
|
||||
if (contentEl) {
|
||||
const { left, top } = contentEl.getBoundingClientRect()
|
||||
|
||||
full
|
||||
? addStyle(contentEl, {
|
||||
transform: `translate(-${left}px, -${top}px)`,
|
||||
})
|
||||
: removeStyle(contentEl, ['transform'])
|
||||
}
|
||||
|
||||
setVariable('layoutContentMaximize', full)
|
||||
}
|
||||
|
||||
return {
|
||||
|
@ -11,7 +11,6 @@
|
||||
|
||||
import { useMenuGetters, useMenuActions } from '@/store'
|
||||
import { ROOT_ROUTE } from '@/app-config/appConfig'
|
||||
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
|
||||
|
||||
import type { MenuTagOptions, Key } from '@/types/modules/app'
|
||||
|
||||
@ -93,8 +92,13 @@ export function useMenuTag() {
|
||||
* 关闭所有标签并且导航至 root path
|
||||
*/
|
||||
const closeAll = () => {
|
||||
const option = getMenuTagOptions.value.find((curr) => curr.key === path)
|
||||
|
||||
if (option) {
|
||||
changeMenuModelValue(path, option)
|
||||
}
|
||||
|
||||
emptyMenuTagOptions()
|
||||
redirectRouterToDashboard(true)
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -22,6 +22,12 @@ import type { DayjsLocal } from '@/dayjs/type'
|
||||
* - locale: 切换 dayjs 语言配置
|
||||
*/
|
||||
export const useDayjs = () => {
|
||||
/**
|
||||
*
|
||||
* @param key 当前语言
|
||||
*
|
||||
* 手动配置 dayjs 语言(国际化)
|
||||
*/
|
||||
const locale = (key: DayjsLocal) => {
|
||||
const locale = DAYJS_LOCAL_MAP[key]
|
||||
|
||||
|
@ -15,14 +15,17 @@
|
||||
*/
|
||||
|
||||
import { useWindowSize } from '@vueuse/core'
|
||||
import { watchEffectWithTarget } from '@/utils/vue/index'
|
||||
|
||||
export function useDevice() {
|
||||
const { width, height } = useWindowSize()
|
||||
const isTabletOrSmaller = ref(false)
|
||||
|
||||
watchEffect(() => {
|
||||
const update = () => {
|
||||
isTabletOrSmaller.value = width.value <= 768
|
||||
})
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
|
||||
return {
|
||||
width,
|
||||
|
15
src/icons/close_left.svg
Normal file
@ -0,0 +1,15 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" width="1em" height="1em"
|
||||
viewBox="0 0 24 24">
|
||||
<g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
|
||||
stroke-width="2">
|
||||
<path stroke-dasharray="20" stroke-dashoffset="20" d="M3 3V21">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" dur="0.3s" values="20;0"></animate>
|
||||
</path>
|
||||
<path stroke-dasharray="15" stroke-dashoffset="15" d="M21 12H7.5">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" begin="0.4s" dur="0.2s" values="15;0"></animate>
|
||||
</path>
|
||||
<path stroke-dasharray="12" stroke-dashoffset="12" d="M7 12L14 19M7 12L14 5">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" begin="0.6s" dur="0.2s" values="12;0"></animate>
|
||||
</path>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 819 B |
19
src/icons/close_right.svg
Normal file
@ -0,0 +1,19 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" width="1em" height="1em"
|
||||
viewBox="0 0 24 24">
|
||||
<g transform="translate(24 0) scale(-1 1)">
|
||||
<g fill="none" stroke="currentColor" stroke-linecap="round" stroke-linejoin="round"
|
||||
stroke-width="2">
|
||||
<path stroke-dasharray="20" stroke-dashoffset="20" d="M3 3V21">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" dur="0.3s" values="20;0"></animate>
|
||||
</path>
|
||||
<path stroke-dasharray="15" stroke-dashoffset="15" d="M21 12H7.5">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" begin="0.4s" dur="0.2s"
|
||||
values="15;0"></animate>
|
||||
</path>
|
||||
<path stroke-dasharray="12" stroke-dashoffset="12" d="M7 12L14 19M7 12L14 5">
|
||||
<animate fill="freeze" attributeName="stroke-dashoffset" begin="0.6s" dur="0.2s"
|
||||
values="12;0"></animate>
|
||||
</path>
|
||||
</g>
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 916 B |
5
src/icons/fullscreen_fold.svg
Normal file
@ -0,0 +1,5 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" role="img" width="1em" height="1em"
|
||||
viewBox="0 0 16 16" class="iconify iconify--codicon">
|
||||
<path fill="currentColor"
|
||||
d="M3 12h10V4H3v8zm2-6h6v4H5V6zM2 6H1V2.5l.5-.5H5v1H2v3zm13-3.5V6h-1V3h-3V2h3.5l.5.5zM14 10h1v3.5l-.5.5H11v-1h3v-3zM2 13h3v1H1.5l-.5-.5V10h1v3z"></path>
|
||||
</svg>
|
After Width: | Height: | Size: 342 B |
@ -1,9 +1,11 @@
|
||||
<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
|
||||
fill="currentColor"
|
||||
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
|
||||
fill="currentColor"
|
||||
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>
|
Before Width: | Height: | Size: 2.3 KiB After Width: | Height: | Size: 2.4 KiB |
6
src/icons/out.svg
Normal file
@ -0,0 +1,6 @@
|
||||
<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink"
|
||||
aria-hidden="true" role="img" width="1em" height="1em" viewBox="0 0 24 24"
|
||||
class="iconify iconify--ri">
|
||||
<path fill="currentColor"
|
||||
d="M4 18h2v2h12V4H6v2H4V3a1 1 0 0 1 1-1h14a1 1 0 0 1 1 1v18a1 1 0 0 1-1 1H5a1 1 0 0 1-1-1v-3Zm2-7h7v2H6v3l-5-4l5-4v3Z"></path>
|
||||
</svg>
|
After Width: | Height: | Size: 356 B |
@ -1,3 +1,6 @@
|
||||
<svg t="1669082370283" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg" p-id="3070" width="200" height="200">
|
||||
<path d="M909.1 209.3l-56.4 44.1C775.8 155.1 656.2 92 521.9 92 290 92 102.3 279.5 102 511.5 101.7 743.7 289.8 932 521.9 932c181.3 0 335.8-115 394.6-276.1 1.5-4.2-0.7-8.9-4.9-10.3l-56.7-19.5c-4.1-1.4-8.6 0.7-10.1 4.8-1.8 5-3.8 10-5.9 14.9-17.3 41-42.1 77.8-73.7 109.4-31.6 31.6-68.4 56.4-109.3 73.8-42.3 17.9-87.4 27-133.8 27-46.5 0-91.5-9.1-133.8-27-40.9-17.3-77.7-42.1-109.3-73.8-31.6-31.6-56.4-68.4-73.7-109.4-17.9-42.4-27-87.4-27-133.9s9.1-91.5 27-133.9c17.3-41 42.1-77.8 73.7-109.4 31.6-31.6 68.4-56.4 109.3-73.8 42.3-17.9 87.4-27 133.8-27 46.5 0 91.5 9.1 133.8 27 40.9 17.3 77.7 42.1 109.3 73.8 9.9 9.9 19.2 20.4 27.8 31.4l-60.2 47c-5.3 4.1-3.5 12.5 3 14.1l175.6 43c5 1.2 9.9-2.6 9.9-7.7l0.8-180.9c-0.1-6.6-7.8-10.3-13-6.2z" p-id="3071" fill="currentColor"></path>
|
||||
<svg t="1669082370283" class="icon" viewBox="0 0 1024 1024" version="1.1"
|
||||
xmlns="http://www.w3.org/2000/svg" p-id="3070" width="200" height="200">
|
||||
<path
|
||||
d="M909.1 209.3l-56.4 44.1C775.8 155.1 656.2 92 521.9 92 290 92 102.3 279.5 102 511.5 101.7 743.7 289.8 932 521.9 932c181.3 0 335.8-115 394.6-276.1 1.5-4.2-0.7-8.9-4.9-10.3l-56.7-19.5c-4.1-1.4-8.6 0.7-10.1 4.8-1.8 5-3.8 10-5.9 14.9-17.3 41-42.1 77.8-73.7 109.4-31.6 31.6-68.4 56.4-109.3 73.8-42.3 17.9-87.4 27-133.8 27-46.5 0-91.5-9.1-133.8-27-40.9-17.3-77.7-42.1-109.3-73.8-31.6-31.6-56.4-68.4-73.7-109.4-17.9-42.4-27-87.4-27-133.9s9.1-91.5 27-133.9c17.3-41 42.1-77.8 73.7-109.4 31.6-31.6 68.4-56.4 109.3-73.8 42.3-17.9 87.4-27 133.8-27 46.5 0 91.5 9.1 133.8 27 40.9 17.3 77.7 42.1 109.3 73.8 9.9 9.9 19.2 20.4 27.8 31.4l-60.2 47c-5.3 4.1-3.5 12.5 3 14.1l175.6 43c5 1.2 9.9-2.6 9.9-7.7l0.8-180.9c-0.1-6.6-7.8-10.3-13-6.2z"
|
||||
p-id="3071" fill="currentColor"></path>
|
||||
</svg>
|
Before Width: | Height: | Size: 925 B After Width: | Height: | Size: 935 B |
@ -14,7 +14,7 @@ import './index.scss'
|
||||
import { NEllipsis, NPopover } from 'naive-ui'
|
||||
import RIcon from '@/components/RIcon/index'
|
||||
|
||||
const SiderBarLogo = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'SiderBarLogo',
|
||||
props: {
|
||||
collapsed: {
|
||||
@ -29,6 +29,13 @@ const SiderBarLogo = defineComponent({
|
||||
layout: { sideBarLogo },
|
||||
} = __APP_CFG__
|
||||
|
||||
/**
|
||||
*
|
||||
* 点击 logo 跳转页面
|
||||
* jumpType:
|
||||
* - station: 模板内跳转
|
||||
* - outsideStation: 新开页面跳转
|
||||
*/
|
||||
const handleSideBarLogoClick = () => {
|
||||
if (sideBarLogo && sideBarLogo.url) {
|
||||
sideBarLogo.jumpType === 'station'
|
||||
@ -80,5 +87,3 @@ const SiderBarLogo = defineComponent({
|
||||
) : null
|
||||
},
|
||||
})
|
||||
|
||||
export default SiderBarLogo
|
||||
|
@ -53,6 +53,10 @@ export default defineComponent({
|
||||
},
|
||||
})
|
||||
|
||||
/**
|
||||
*
|
||||
* 手动展开当前激活菜单项
|
||||
*/
|
||||
const showMenuOption = () => {
|
||||
const key = modelMenuKey.value as string
|
||||
|
||||
|
@ -25,8 +25,8 @@ $menuTagWrapperWidth: 76px;
|
||||
}
|
||||
|
||||
& .menu-tag__right-wrapper {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
// display: inline-flex;
|
||||
// align-items: center;
|
||||
|
||||
& .menu-tag__right-arrow {
|
||||
transform: rotate(270deg);
|
||||
@ -43,3 +43,10 @@ $menuTagWrapperWidth: 76px;
|
||||
cursor: pointer;
|
||||
}
|
||||
}
|
||||
|
||||
.menu-tag__dropdown {
|
||||
& .menu-tag__icon {
|
||||
width: 18px;
|
||||
height: 18px;
|
||||
}
|
||||
}
|
||||
|
@ -12,15 +12,21 @@
|
||||
/**
|
||||
*
|
||||
* 操作说明:
|
||||
* - 关闭全部: 关闭所有标签页, 并且重定向至根页面 rootRoute.path
|
||||
* - 关闭右侧: 关闭右侧所有标签, 如果选中标签页与当前激活页不一致并且激活页在右侧, 则会重定向至当前选中标签页
|
||||
* - 关闭左侧: 关闭左侧所有标签, 如果选中标签页与当前激活页不一致并且激活页在左侧, 则会重定向至当前选中标签页
|
||||
* - 关闭其他: 关闭其他所有标签, 如果选中标签页与当前激活页不一致并且激活页在其中, 则会重定向至当前选中标签页
|
||||
* - 关闭全部: 关闭所有标签页,并且重定向至根页面 rootRoute.path
|
||||
* - 关闭右侧: 关闭右侧所有标签,如果选中标签页与当前激活页不一致并且激活页在右侧,则会重定向至当前选中标签页
|
||||
* - 关闭左侧: 关闭左侧所有标签,如果选中标签页与当前激活页不一致并且激活页在左侧,则会重定向至当前选中标签页
|
||||
* - 关闭其他: 关闭其他所有标签,如果选中标签页与当前激活页不一致并且激活页在其中,则会重定向至当前选中标签页
|
||||
* - 关闭所有: 关闭所有标签页,并且重定向至 root page
|
||||
*
|
||||
* root path 标签不可被关闭, 所以不会显示关闭按钮
|
||||
* 页面刷新后, 仅会保留刷新前激活 key 的 tag 标签
|
||||
* root path 标签不可被关闭,所以不会显示关闭按钮
|
||||
* 页面刷新后,仅会保留刷新前激活 key 的 tag 标签
|
||||
*
|
||||
* 注入 MENU_TAG_DATA 属性, 用于动态更新 MenuTag 标签所在的位置
|
||||
* 注入 MENU_TAG_DATA 属性,用于动态更新 MenuTag 标签所在的位置
|
||||
*
|
||||
* 该模板中引入了 Root Path 概念,在 MenuTag 中除了关闭左右侧标签操作能主动移除 Root Tag 之外其余的操作都不允许
|
||||
*
|
||||
* outsideClick 方法优先级不如 contextmenu 事件高,所以可能会出现重复右键菜单时,闪烁问题
|
||||
* 虽然使用 throttle 方法进行优化,但是该问题本质并没有解决
|
||||
*/
|
||||
|
||||
import './index.scss'
|
||||
@ -29,15 +35,19 @@ import { NScrollbar, NTag, NSpace, NLayoutHeader, NDropdown } from 'naive-ui'
|
||||
import RIcon from '@/components/RIcon/index'
|
||||
import RMoreDropdown from '@/components/RMoreDropdown/index'
|
||||
|
||||
// import Reload from '@/icons/reload.svg?component'
|
||||
import CloseRight from '@/icons/close_right.svg?component'
|
||||
import CloseLeft from '@/icons/close_left.svg?component'
|
||||
|
||||
import { useMenuGetters, useMenuActions } from '@/store'
|
||||
import { uuid } from '@/utils/basic'
|
||||
import { hasClass } from '@/utils/element'
|
||||
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
|
||||
import { ROOT_ROUTE } from '@/app-config/appConfig'
|
||||
import { queryElements } from '@use-utils/element'
|
||||
import { renderNode } from '@/utils/vue/index'
|
||||
import { useMainPage } from '@/hooks/template/index'
|
||||
import { useMenuTag } from '@/hooks/template/index'
|
||||
import { throttle } from 'lodash-es'
|
||||
|
||||
import type { ScrollbarInst } from 'naive-ui'
|
||||
import type { MenuTagOptions, AppMenuOption } from '@/types/modules/app'
|
||||
@ -48,14 +58,9 @@ export default defineComponent({
|
||||
const scrollRef = ref<ScrollbarInst | null>(null)
|
||||
|
||||
const { getMenuKey, getMenuTagOptions } = useMenuGetters()
|
||||
const {
|
||||
changeMenuModelValue,
|
||||
spliceMenTagOptions,
|
||||
emptyMenuTagOptions,
|
||||
setMenuTagOptions,
|
||||
} = useMenuActions()
|
||||
const { changeMenuModelValue } = useMenuActions()
|
||||
const { path } = ROOT_ROUTE
|
||||
const { reload } = useMainPage()
|
||||
const { reload, maximize } = useMainPage()
|
||||
const {
|
||||
close,
|
||||
closeAll: $closeAll,
|
||||
@ -64,85 +69,63 @@ export default defineComponent({
|
||||
closeOther: $closeOther,
|
||||
} = useMenuTag()
|
||||
|
||||
const exclude = ['closeAll', 'closeRight', 'closeLeft', 'closeOther']
|
||||
const canDisabledOptions = [
|
||||
'closeAll',
|
||||
'closeRight',
|
||||
'closeLeft',
|
||||
'closeOther',
|
||||
'closeCurrentPage',
|
||||
] // 哪些下拉框允许禁用
|
||||
let currentContextmenuIndex = -1 // 当前右键标签页索引位置
|
||||
const iconConfig = {
|
||||
size: 16,
|
||||
}
|
||||
const moreOptions = ref([
|
||||
{
|
||||
label: '重新加载',
|
||||
label: '刷新页面',
|
||||
key: 'reloadCurrentPage',
|
||||
icon: () =>
|
||||
h(
|
||||
RIcon,
|
||||
{
|
||||
size: iconConfig.size,
|
||||
name: 'reload',
|
||||
},
|
||||
{},
|
||||
),
|
||||
icon: () => <RIcon name="reload" size={iconConfig.size} />,
|
||||
},
|
||||
{
|
||||
label: '关闭其他',
|
||||
key: 'closeOther',
|
||||
icon: () =>
|
||||
h(
|
||||
RIcon,
|
||||
{
|
||||
size: iconConfig.size,
|
||||
name: 'other',
|
||||
},
|
||||
{},
|
||||
),
|
||||
},
|
||||
{
|
||||
label: '关闭右侧',
|
||||
key: 'closeRight',
|
||||
icon: () =>
|
||||
h(
|
||||
RIcon,
|
||||
{
|
||||
size: iconConfig.size,
|
||||
name: 'right_arrow',
|
||||
},
|
||||
{},
|
||||
),
|
||||
},
|
||||
{
|
||||
label: '关闭左侧',
|
||||
key: 'closeLeft',
|
||||
icon: () =>
|
||||
h(
|
||||
RIcon,
|
||||
{
|
||||
size: iconConfig.size,
|
||||
name: 'left_arrow',
|
||||
},
|
||||
{},
|
||||
),
|
||||
label: '关闭当前页面',
|
||||
key: 'closeCurrentPage',
|
||||
icon: () => <RIcon name="close" size={iconConfig.size} />,
|
||||
},
|
||||
{
|
||||
type: 'divider',
|
||||
key: 'd1',
|
||||
},
|
||||
{
|
||||
label: '全部关闭',
|
||||
label: '关闭右侧标签页',
|
||||
key: 'closeRight',
|
||||
icon: () => <CloseRight class="menu-tag__icon" />,
|
||||
},
|
||||
{
|
||||
label: '关闭左侧标签页',
|
||||
key: 'closeLeft',
|
||||
icon: () => <CloseLeft class="menu-tag__icon" />,
|
||||
},
|
||||
{
|
||||
type: 'divider',
|
||||
key: 'd1',
|
||||
},
|
||||
{
|
||||
label: '关闭其他标签页',
|
||||
key: 'closeOther',
|
||||
icon: () => <RIcon name="other" size={iconConfig.size} />,
|
||||
},
|
||||
{
|
||||
label: '关闭所有标签页',
|
||||
key: 'closeAll',
|
||||
icon: () =>
|
||||
h(
|
||||
RIcon,
|
||||
{
|
||||
size: iconConfig.size,
|
||||
name: 'close',
|
||||
},
|
||||
{},
|
||||
),
|
||||
icon: () => <RIcon name="resize_h" size={iconConfig.size} />,
|
||||
disabled: false,
|
||||
},
|
||||
])
|
||||
const uuidScrollBar = uuid(16)
|
||||
]) // 下拉菜单
|
||||
const uuidScrollBar = uuid(16) // scroll bar uuid
|
||||
const actionMap = {
|
||||
closeCurrentPage: () => {
|
||||
getMenuKey.value !== path && close(currentContextmenuIndex)
|
||||
},
|
||||
reloadCurrentPage: () => {
|
||||
reload()
|
||||
},
|
||||
@ -165,7 +148,7 @@ export default defineComponent({
|
||||
y: 0,
|
||||
actionDropdownShow: false,
|
||||
})
|
||||
const MENU_TAG_DATA = 'menu_tag_data'
|
||||
const MENU_TAG_DATA = 'menu_tag_data' // 注入 tag 前缀
|
||||
|
||||
/**
|
||||
*
|
||||
@ -178,7 +161,7 @@ export default defineComponent({
|
||||
}
|
||||
|
||||
const setMoreOptionsDisabled = (
|
||||
key: string | number,
|
||||
key: (typeof moreOptions.value)[number]['key'],
|
||||
disabled: boolean,
|
||||
) => {
|
||||
moreOptions.value.forEach((curr) => {
|
||||
@ -192,12 +175,18 @@ export default defineComponent({
|
||||
|
||||
/**
|
||||
*
|
||||
* @param item 当前菜单值
|
||||
* @param option 当前菜单值
|
||||
*/
|
||||
const handleTagClick = (item: AppMenuOption) => {
|
||||
changeMenuModelValue(item.key as string, item)
|
||||
const handleTagClick = (option: AppMenuOption) => {
|
||||
actionState.actionDropdownShow = false
|
||||
|
||||
changeMenuModelValue(option.key as string, option)
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* 获取滚动条容器
|
||||
*/
|
||||
const getScrollElement = () => {
|
||||
const scroll = document.getElementById(uuidScrollBar) // 获取滚动条容器
|
||||
|
||||
@ -215,6 +204,12 @@ export default defineComponent({
|
||||
return
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param type 滚动方向
|
||||
*
|
||||
* 手动滚动容器
|
||||
*/
|
||||
const scrollX = (type: 'left' | 'right') => {
|
||||
const el = getScrollElement()
|
||||
|
||||
@ -254,22 +249,41 @@ export default defineComponent({
|
||||
actionState.actionDropdownShow = false
|
||||
currentContextmenuIndex = idx
|
||||
|
||||
nextTick().then(() => {
|
||||
nextTick(() => {
|
||||
actionState.actionDropdownShow = true
|
||||
actionState.x = e.clientX
|
||||
actionState.y = e.clientY
|
||||
})
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* 动态设置某些项禁用
|
||||
*/
|
||||
const setDisabledAccordionToIndex = () => {
|
||||
const length = getMenuTagOptions.value.length - 1
|
||||
|
||||
// 是否需要禁用关闭当前标签页
|
||||
if (getMenuKey.value === path) {
|
||||
setMoreOptionsDisabled('closeCurrentPage', true)
|
||||
} else {
|
||||
const isRoot = moreOptions.value[currentContextmenuIndex]
|
||||
|
||||
if (isRoot.key === 'closeCurrentPage') {
|
||||
setMoreOptionsDisabled('closeCurrentPage', true)
|
||||
} else {
|
||||
setMoreOptionsDisabled('closeCurrentPage', false)
|
||||
}
|
||||
}
|
||||
|
||||
// 是否需要禁用关闭右侧标签页
|
||||
if (currentContextmenuIndex === length) {
|
||||
setMoreOptionsDisabled('closeRight', true)
|
||||
} else if (currentContextmenuIndex < length) {
|
||||
setMoreOptionsDisabled('closeRight', false)
|
||||
}
|
||||
|
||||
// 是否需要禁用关闭左侧标签页
|
||||
if (currentContextmenuIndex === 0) {
|
||||
setMoreOptionsDisabled('closeLeft', true)
|
||||
} else if (currentContextmenuIndex > 0) {
|
||||
@ -336,7 +350,7 @@ export default defineComponent({
|
||||
const [menuTag] = tags
|
||||
|
||||
nextTick().then(() => {
|
||||
menuTag.scrollIntoView?.()
|
||||
menuTag.scrollIntoView?.(true)
|
||||
})
|
||||
}
|
||||
})
|
||||
@ -346,14 +360,16 @@ export default defineComponent({
|
||||
watch(
|
||||
() => getMenuTagOptions.value,
|
||||
(newData, oldData) => {
|
||||
// 当 menuTagOptions 长度为 1时,禁用所有 canDisabledOptions 匹配的项
|
||||
moreOptions.value.forEach((curr) => {
|
||||
if (exclude.includes(curr.key)) {
|
||||
if (canDisabledOptions.includes(curr.key)) {
|
||||
newData.length > 1
|
||||
? (curr.disabled = false)
|
||||
: (curr.disabled = true)
|
||||
}
|
||||
})
|
||||
|
||||
// 更新当前激活标签定位
|
||||
if (oldData?.length) {
|
||||
if (newData.length > oldData?.length) {
|
||||
updateScrollBarPosition()
|
||||
@ -366,12 +382,12 @@ export default defineComponent({
|
||||
immediate: true,
|
||||
},
|
||||
)
|
||||
|
||||
/** 动态设置关闭按钮是否可操作 */
|
||||
watch(
|
||||
() => actionState.actionDropdownShow,
|
||||
() => {
|
||||
setDisabledAccordionToIndex()
|
||||
// 使用节流函数,避免右键菜单闪烁问题
|
||||
throttle(setDisabledAccordionToIndex, 100)?.()
|
||||
},
|
||||
)
|
||||
|
||||
@ -388,35 +404,39 @@ export default defineComponent({
|
||||
scrollRef,
|
||||
uuidScrollBar,
|
||||
actionDropdownSelect,
|
||||
rootPath: path,
|
||||
actionState,
|
||||
handleContextMenu,
|
||||
setCurrentContextmenuIndex,
|
||||
menuTagMouseenter,
|
||||
menuTagMouseleave,
|
||||
MENU_TAG_DATA,
|
||||
iconConfig: {
|
||||
width: 20,
|
||||
height: 28,
|
||||
},
|
||||
maximize,
|
||||
}
|
||||
},
|
||||
render() {
|
||||
const iconConfig = {
|
||||
width: 20,
|
||||
height: 28,
|
||||
}
|
||||
const { iconConfig } = this
|
||||
const { maximize } = this
|
||||
|
||||
return (
|
||||
<NLayoutHeader>
|
||||
<div class="menu-tag">
|
||||
<NDropdown
|
||||
class="menu-tag__dropdown"
|
||||
options={this.moreOptions}
|
||||
x={this.actionState.x}
|
||||
y={this.actionState.y}
|
||||
keyboard={false}
|
||||
show={this.actionState.actionDropdownShow}
|
||||
trigger="manual"
|
||||
placement="bottom-start"
|
||||
onSelect={this.actionDropdownSelect.bind(this)}
|
||||
onClickoutside={() => {
|
||||
this.actionState.actionDropdownShow = false
|
||||
}}
|
||||
onSelect={this.actionDropdownSelect.bind(this)}
|
||||
/>
|
||||
<NSpace
|
||||
class="menu-tag-space"
|
||||
@ -441,6 +461,7 @@ export default defineComponent({
|
||||
}}
|
||||
>
|
||||
<NSpace
|
||||
ref="menuTagSpaceRef"
|
||||
class="menu-tag-wrapper"
|
||||
wrap={false}
|
||||
align="center"
|
||||
@ -468,7 +489,14 @@ export default defineComponent({
|
||||
))}
|
||||
</NSpace>
|
||||
</NScrollbar>
|
||||
<div class="menu-tag__right-wrapper">
|
||||
<NSpace
|
||||
class="menu-tag__right-wrapper"
|
||||
wrapItem={false}
|
||||
align="center"
|
||||
inline
|
||||
wrap={false}
|
||||
size={[6, 6]}
|
||||
>
|
||||
<RIcon
|
||||
name="expanded"
|
||||
width={iconConfig.width}
|
||||
@ -476,11 +504,22 @@ export default defineComponent({
|
||||
customClassName="menu-tag__right-arrow"
|
||||
onClick={this.scrollX.bind(this, 'right')}
|
||||
/>
|
||||
<RIcon
|
||||
name="fullscreen_fold"
|
||||
width={iconConfig.width}
|
||||
height={iconConfig.height}
|
||||
customClassName="menu-tag__right-setting"
|
||||
onClick={() => {
|
||||
maximize(true)
|
||||
}}
|
||||
/>
|
||||
<RMoreDropdown
|
||||
class="menu-tag__dropdown"
|
||||
options={this.moreOptions}
|
||||
trigger="click"
|
||||
onSelect={this.actionDropdownSelect.bind(this)}
|
||||
iconSize={20}
|
||||
keyboard={false}
|
||||
>
|
||||
<RIcon
|
||||
name="more"
|
||||
@ -490,7 +529,7 @@ export default defineComponent({
|
||||
onClick={this.setCurrentContextmenuIndex.bind(this)}
|
||||
/>
|
||||
</RMoreDropdown>
|
||||
</div>
|
||||
</NSpace>
|
||||
</NSpace>
|
||||
</div>
|
||||
</NLayoutHeader>
|
||||
|
@ -27,7 +27,7 @@ import type { DropdownOption } from 'naive-ui'
|
||||
import type { AppMenuOption } from '@/types/modules/app'
|
||||
|
||||
export default defineComponent({
|
||||
name: 'RBreadcrumb',
|
||||
name: 'SiderBarBreadcrumb',
|
||||
setup() {
|
||||
const { changeMenuModelValue } = useMenuActions()
|
||||
const { getBreadcrumbOptions } = useMenuGetters()
|
||||
@ -37,6 +37,12 @@ export default defineComponent({
|
||||
changeMenuModelValue(key, option as unknown as AppMenuOption)
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param option bread option
|
||||
*
|
||||
* 点击面包屑,如果包含子集,则允许点击更新页面
|
||||
*/
|
||||
const breadcrumbItemClick = (option: AppMenuOption) => {
|
||||
if (!option.children?.length) {
|
||||
const { meta = {} } = option
|
||||
|
@ -9,6 +9,15 @@
|
||||
* @remark 今天也是元气满满撸代码的一天
|
||||
*/
|
||||
|
||||
/**
|
||||
*
|
||||
* app search 搜索功能
|
||||
* 递归模糊查找所有在 getMenuOptions 中匹配的项
|
||||
* 只有在满足 validMenuItemShow 的时候,才会出现在搜索结果中
|
||||
*
|
||||
* 该功能不会在小尺寸屏幕中启用(isTabletOrSmaller = true)
|
||||
*/
|
||||
|
||||
import './index.scss'
|
||||
|
||||
import { NInput, NModal, NResult, NScrollbar, NSpace } from 'naive-ui'
|
||||
@ -98,7 +107,7 @@ export default defineComponent({
|
||||
}
|
||||
|
||||
/** 根据输入值模糊检索菜单 */
|
||||
const handleSearchMenuOptions = (value: string) => {
|
||||
const fuzzySearchMenuOptions = (value: string) => {
|
||||
const arr: AppMenuOption[] = []
|
||||
|
||||
const filterArr = (options: AppMenuOption[]) => {
|
||||
@ -108,11 +117,12 @@ export default defineComponent({
|
||||
}
|
||||
|
||||
/** 处理菜单名与输入值, 不区分大小写 */
|
||||
const _breadcrumbLabel = curr.breadcrumbLabel?.toLocaleLowerCase()
|
||||
const _value = String(value).toLocaleLowerCase()
|
||||
const $breadcrumbLabel = curr.breadcrumbLabel?.toLocaleLowerCase()
|
||||
const $value = String(value).toLocaleLowerCase()
|
||||
|
||||
// 是否模糊匹配字符、满足展示条件
|
||||
if (
|
||||
_breadcrumbLabel?.includes(_value) &&
|
||||
$breadcrumbLabel?.includes($value) &&
|
||||
validMenuItemShow(curr) &&
|
||||
!curr.children?.length
|
||||
) {
|
||||
@ -151,8 +161,9 @@ export default defineComponent({
|
||||
|
||||
/** 自动聚焦检索项 */
|
||||
const autoFocusingSearchItem = () => {
|
||||
const currentOption = state.searchOptions[searchElementIndex]
|
||||
const preOption = state.searchOptions[preSearchElementIndex]
|
||||
const currentOption = state.searchOptions[searchElementIndex] // 获取当前搜索项
|
||||
const preOption = state.searchOptions[preSearchElementIndex] // 获取上一搜索项
|
||||
const activeClass = 'content-item--active' // 激活样式 class name
|
||||
|
||||
if (currentOption) {
|
||||
nextTick().then(() => {
|
||||
@ -166,13 +177,13 @@ export default defineComponent({
|
||||
if (preSearchElementOptions?.length) {
|
||||
const [el] = preSearchElementOptions
|
||||
|
||||
removeClass(el, 'content-item--active')
|
||||
removeClass(el, activeClass)
|
||||
}
|
||||
|
||||
if (searchElementOptions?.length) {
|
||||
const [el] = searchElementOptions
|
||||
|
||||
addClass(el, 'content-item--active')
|
||||
addClass(el, activeClass)
|
||||
}
|
||||
})
|
||||
}
|
||||
@ -191,6 +202,19 @@ export default defineComponent({
|
||||
}
|
||||
}
|
||||
|
||||
/** 更新索引 */
|
||||
const updateIndex = (type: 'up' | 'down') => {
|
||||
if (type === 'up') {
|
||||
searchElementIndex =
|
||||
searchElementIndex - 1 < 0 ? 0 : searchElementIndex - 1
|
||||
} else if (type === 'down') {
|
||||
searchElementIndex =
|
||||
searchElementIndex + 1 >= state.searchOptions.length
|
||||
? state.searchOptions.length - 1
|
||||
: searchElementIndex + 1
|
||||
}
|
||||
}
|
||||
|
||||
/** 注册按键: 上、下、回车 */
|
||||
const registerChangeSearchElementIndex = (e: KeyboardEvent) => {
|
||||
const keyCode = e.key
|
||||
@ -200,21 +224,9 @@ export default defineComponent({
|
||||
e.stopPropagation()
|
||||
}
|
||||
|
||||
// 当初始化索引小于等于 0 时,缓存重置缓存索引
|
||||
preSearchElementIndex = searchElementIndex <= 0 ? 0 : searchElementIndex
|
||||
|
||||
/** 更新索引 */
|
||||
const updateIndex = (type: 'up' | 'down') => {
|
||||
if (type === 'up') {
|
||||
searchElementIndex =
|
||||
searchElementIndex - 1 < 0 ? 0 : searchElementIndex - 1
|
||||
} else if (type === 'down') {
|
||||
searchElementIndex =
|
||||
searchElementIndex + 1 >= state.searchOptions.length
|
||||
? state.searchOptions.length - 1
|
||||
: searchElementIndex + 1
|
||||
}
|
||||
}
|
||||
|
||||
switch (keyCode) {
|
||||
case 'ArrowUp':
|
||||
updateIndex('up')
|
||||
@ -257,6 +269,7 @@ export default defineComponent({
|
||||
)
|
||||
|
||||
watchEffect(() => {
|
||||
// 当处于小尺寸状态时,自动关闭搜索框
|
||||
if (isTabletOrSmaller.value) {
|
||||
modelShow.value = false
|
||||
}
|
||||
@ -279,7 +292,7 @@ export default defineComponent({
|
||||
...toRefs(state),
|
||||
modelShow,
|
||||
helperTipOptions,
|
||||
handleSearchMenuOptions: debounce(handleSearchMenuOptions, 300),
|
||||
fuzzySearchMenuOptions: debounce(fuzzySearchMenuOptions, 300),
|
||||
handleSearchItemClick,
|
||||
RenderPreIcon,
|
||||
isTabletOrSmaller,
|
||||
@ -306,7 +319,7 @@ export default defineComponent({
|
||||
size="large"
|
||||
v-model:value={this.searchValue}
|
||||
clearable
|
||||
onInput={this.handleSearchMenuOptions.bind(this)}
|
||||
onInput={this.fuzzySearchMenuOptions.bind(this)}
|
||||
>
|
||||
{{
|
||||
prefix: () => <RIcon name="search" size="24" />,
|
||||
|
@ -14,11 +14,12 @@ import RIcon from '@/components/RIcon'
|
||||
|
||||
import { useSettingGetters, useSettingActions } from '@/store'
|
||||
|
||||
const ThemeSwitch = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'ThemeSwitch',
|
||||
setup() {
|
||||
const { changeSwitcher } = useSettingActions()
|
||||
const { getAppTheme } = useSettingGetters()
|
||||
const modelAppThemeRef = ref(getAppTheme.value)
|
||||
|
||||
const handleRailStyle = ({ checked }: { checked: boolean }) => {
|
||||
return checked
|
||||
@ -34,6 +35,7 @@ const ThemeSwitch = defineComponent({
|
||||
changeSwitcher,
|
||||
getAppTheme,
|
||||
handleRailStyle,
|
||||
modelAppThemeRef,
|
||||
}
|
||||
},
|
||||
render() {
|
||||
@ -45,7 +47,7 @@ const ThemeSwitch = defineComponent({
|
||||
{{
|
||||
trigger: () => (
|
||||
<NSwitch
|
||||
v-model:value={this.getAppTheme}
|
||||
v-model:value={this.modelAppThemeRef}
|
||||
railStyle={this.handleRailStyle.bind(this)}
|
||||
onUpdateValue={(bool: boolean) =>
|
||||
this.changeSwitcher(bool, 'appTheme')
|
||||
@ -83,5 +85,3 @@ const ThemeSwitch = defineComponent({
|
||||
)
|
||||
},
|
||||
})
|
||||
|
||||
export default ThemeSwitch
|
||||
|
@ -1,3 +1,13 @@
|
||||
/**
|
||||
*
|
||||
* app setting 抽屉
|
||||
* 提供了一些基础的动态配置能力
|
||||
* 如果需要其他额外的配置项,可以按照当前的方式进行拓展
|
||||
*
|
||||
* 可能会疑问,为什么可配置项那么少?
|
||||
* 其实并不少,只是有一些东西,在我看来是没多大意义需要动态的去改动的,所以都是在 `app-config` 包中进行配置维护
|
||||
*/
|
||||
|
||||
import './index.scss'
|
||||
|
||||
import {
|
||||
@ -55,6 +65,7 @@ const SettingDrawer = defineComponent({
|
||||
emit('update:show', bool)
|
||||
},
|
||||
})
|
||||
// 过渡效果下拉
|
||||
const contentTransitionOptions = [
|
||||
{
|
||||
label: '无',
|
||||
|
@ -33,7 +33,7 @@ import {
|
||||
avatarDropdownClick,
|
||||
createLeftIconOptions,
|
||||
createRightIconOptions,
|
||||
} from './hook'
|
||||
} from './shared'
|
||||
import { useDevice } from '@/hooks/web/index'
|
||||
import { getVariableToRefs, setVariable } from '@/global-variable/index'
|
||||
import { useFullscreen } from 'vue-hooks-plus'
|
||||
@ -54,11 +54,11 @@ export default defineComponent({
|
||||
document.getElementsByTagName('html')[0],
|
||||
)
|
||||
const { getDrawerPlacement, getBreadcrumbSwitch } = useSettingGetters()
|
||||
const showSettings = ref(false)
|
||||
const showSettings = ref(false) // 是否显示设置抽屉
|
||||
const spaceItemStyle = {
|
||||
display: 'flex',
|
||||
}
|
||||
const globalSearchShown = ref(false)
|
||||
const globalSearchShown = ref(false) // 是否展示全局搜索
|
||||
const { isTabletOrSmaller } = useDevice()
|
||||
const globalDrawerValue = getVariableToRefs('globalDrawerValue')
|
||||
const globalMainLayoutLoad = getVariableToRefs('globalMainLayoutLoad')
|
||||
@ -114,7 +114,7 @@ export default defineComponent({
|
||||
},
|
||||
}
|
||||
|
||||
const toolIconClick = (key: IconEventMap) => {
|
||||
const toolIconClick = (key: keyof typeof iconEventMap) => {
|
||||
iconEventMap[key]?.()
|
||||
}
|
||||
|
||||
|
@ -1,12 +1,12 @@
|
||||
import { useI18n } from '@/hooks/web/index'
|
||||
import {
|
||||
useSigningActions,
|
||||
useSigningGetters,
|
||||
useSettingActions,
|
||||
} from '@/store'
|
||||
import { useSigningActions, useSettingActions } from '@/store'
|
||||
|
||||
import type { IconOptionsFC, IconOptions } from './type'
|
||||
|
||||
/**
|
||||
*
|
||||
* 创建头像点击下拉菜单
|
||||
*/
|
||||
export const createAvatarOptions = () => [
|
||||
{
|
||||
key: 'person',
|
||||
@ -26,7 +26,16 @@ export const createAvatarOptions = () => [
|
||||
},
|
||||
]
|
||||
|
||||
/**
|
||||
*
|
||||
* 头像 Dropdown 下拉点击事件 map
|
||||
*/
|
||||
const avatarDropdownActionMap = {
|
||||
/**
|
||||
*
|
||||
* 退出登陆
|
||||
* 清除所有 session 缓存并且重定向至登录页
|
||||
*/
|
||||
logout: () => {
|
||||
const { logout } = useSigningActions()
|
||||
|
||||
@ -40,6 +49,10 @@ const avatarDropdownActionMap = {
|
||||
},
|
||||
})
|
||||
},
|
||||
/**
|
||||
*
|
||||
* 锁定屏幕
|
||||
*/
|
||||
lockScreen: () => {
|
||||
const { changeSwitcher } = useSettingActions()
|
||||
|
||||
@ -53,6 +66,15 @@ export const avatarDropdownClick = (key: string | number) => {
|
||||
action ? action() : window.$message.info('这个人很懒, 没做这个功能~')
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param opts 当前操作栏依赖的 ref 集合
|
||||
*
|
||||
* 创建顶部操作栏左侧操作按钮
|
||||
* isTabletOrSmaller:
|
||||
* - true: 菜单按钮
|
||||
* - false: 刷新按钮
|
||||
*/
|
||||
export const createLeftIconOptions = (opts: IconOptionsFC) => {
|
||||
const { isTabletOrSmaller, globalMainLayoutLoad } = opts
|
||||
const { t } = useI18n()
|
||||
@ -79,6 +101,14 @@ export const createLeftIconOptions = (opts: IconOptionsFC) => {
|
||||
: notTableOrSmallerOptions
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param opts 当前操作栏依赖的 ref 集合
|
||||
*
|
||||
* 创建顶部操作栏右侧操作按钮
|
||||
* isTabletOrSmaller:
|
||||
* - false: 不展示搜索按钮
|
||||
*/
|
||||
export const createRightIconOptions = (opts: IconOptionsFC) => {
|
||||
const { isFullscreen, isTabletOrSmaller } = opts
|
||||
const { t } = useI18n()
|
@ -5,3 +5,48 @@
|
||||
display: none;
|
||||
}
|
||||
}
|
||||
|
||||
.r-layout-full__viewer-content--maximize .layout-content__maximize-out {
|
||||
position: fixed;
|
||||
width: 80px;
|
||||
height: 80px;
|
||||
border-radius: 50%;
|
||||
cursor: pointer;
|
||||
z-index: 99;
|
||||
right: -40px;
|
||||
top: -40px;
|
||||
@include flexCenter;
|
||||
transition: color 0.3s var(--r-bezier), background-color 0.3s var(--r-bezier);
|
||||
|
||||
& .ray-icon {
|
||||
transform: translate(-14px, 14px);
|
||||
}
|
||||
}
|
||||
|
||||
.r-layout-full__viewer-content--maximize--dark {
|
||||
@include useAppTheme('dark') {
|
||||
& .layout-content__maximize-out {
|
||||
color: #2c2a28;
|
||||
background: #757473;
|
||||
|
||||
&:hover {
|
||||
background-color: #d5d3d1;
|
||||
color: #44403c;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
.r-layout-full__viewer-content--maximize--light {
|
||||
@include useAppTheme('light') {
|
||||
& .layout-content__maximize-out {
|
||||
color: #eae9e8;
|
||||
background: #a19f9d;
|
||||
|
||||
&:hover {
|
||||
background-color: #44403c;
|
||||
color: #d5d3d1;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -20,13 +20,15 @@ import './index.scss'
|
||||
import { NSpin } from 'naive-ui'
|
||||
import RTransitionComponent from '@/components/RTransitionComponent/index.vue'
|
||||
import AppRequestCancelerProvider from '@/app-components/provider/AppRequestCancelerProvider/index'
|
||||
import RIcon from '@/components/RIcon/index'
|
||||
|
||||
import { getVariableToRefs } from '@/global-variable/index'
|
||||
import { useSettingGetters } from '@/store'
|
||||
import { useMainPage } from '@/hooks/template/index'
|
||||
|
||||
import type { GlobalThemeOverrides } from 'naive-ui'
|
||||
|
||||
const ContentWrapper = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'LayoutContentWrapper',
|
||||
setup() {
|
||||
const router = useRouter()
|
||||
@ -37,6 +39,8 @@ const ContentWrapper = defineComponent({
|
||||
opacitySpinning: '0',
|
||||
}
|
||||
const globalMainLayoutLoad = getVariableToRefs('globalMainLayoutLoad')
|
||||
const layoutContentMaximize = getVariableToRefs('layoutContentMaximize')
|
||||
const { maximize } = useMainPage()
|
||||
|
||||
const setupLayoutContentSpin = () => {
|
||||
router.beforeEach(() => {
|
||||
@ -55,10 +59,13 @@ const ContentWrapper = defineComponent({
|
||||
spinning,
|
||||
themeOverridesSpin,
|
||||
getContentTransition,
|
||||
layoutContentMaximize,
|
||||
maximize,
|
||||
}
|
||||
},
|
||||
render() {
|
||||
const { globalMainLayoutLoad } = this
|
||||
const { globalMainLayoutLoad, layoutContentMaximize } = this
|
||||
const { maximize } = this
|
||||
|
||||
return (
|
||||
<NSpin
|
||||
@ -67,6 +74,16 @@ const ContentWrapper = defineComponent({
|
||||
size="large"
|
||||
themeOverrides={this.themeOverridesSpin}
|
||||
>
|
||||
{layoutContentMaximize ? (
|
||||
<div
|
||||
class="layout-content__maximize-out"
|
||||
onClick={() => {
|
||||
maximize(false)
|
||||
}}
|
||||
>
|
||||
<RIcon name="out" size="16" cursor="pointer" />
|
||||
</div>
|
||||
) : null}
|
||||
<AppRequestCancelerProvider />
|
||||
{globalMainLayoutLoad ? (
|
||||
<RTransitionComponent
|
||||
@ -78,5 +95,3 @@ const ContentWrapper = defineComponent({
|
||||
)
|
||||
},
|
||||
})
|
||||
|
||||
export default ContentWrapper
|
||||
|
@ -11,7 +11,7 @@
|
||||
|
||||
import MenuTag from '@/layout/components/MenuTag/index'
|
||||
|
||||
const FeatureWrapper = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'LayoutFeatureWrapper',
|
||||
setup() {
|
||||
return {}
|
||||
@ -20,5 +20,3 @@ const FeatureWrapper = defineComponent({
|
||||
return <MenuTag />
|
||||
},
|
||||
})
|
||||
|
||||
export default FeatureWrapper
|
||||
|
@ -11,7 +11,7 @@
|
||||
|
||||
import './index.scss'
|
||||
|
||||
const FooterWrapper = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'LayoutFooterWrapper',
|
||||
setup() {
|
||||
const {
|
||||
@ -30,5 +30,3 @@ const FooterWrapper = defineComponent({
|
||||
)
|
||||
},
|
||||
})
|
||||
|
||||
export default FooterWrapper
|
||||
|
@ -12,7 +12,7 @@
|
||||
import { NSpace } from 'naive-ui'
|
||||
import SiderBar from '@/layout/components/SiderBar/index'
|
||||
|
||||
const HeaderWrapper = defineComponent({
|
||||
export default defineComponent({
|
||||
name: 'LayoutHeaderWrapper',
|
||||
setup() {
|
||||
return {}
|
||||
@ -25,5 +25,3 @@ const HeaderWrapper = defineComponent({
|
||||
)
|
||||
},
|
||||
})
|
||||
|
||||
export default HeaderWrapper
|
||||
|
@ -13,9 +13,7 @@
|
||||
|
||||
&.r-layout-full__viewer-content--maximize {
|
||||
position: fixed;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100vw;
|
||||
width: 100%;
|
||||
height: 100vh;
|
||||
transform-origin: center;
|
||||
z-index: 99;
|
||||
|
@ -27,9 +27,9 @@ import { useSettingGetters } from '@/store'
|
||||
export default defineComponent({
|
||||
name: 'RLayout',
|
||||
setup() {
|
||||
const layoutSiderBarRef = ref<HTMLElement>()
|
||||
const layoutMenuTagRef = ref<HTMLElement>()
|
||||
const layoutFooterRef = ref<HTMLElement>()
|
||||
const layoutSiderBarRef = ref<HTMLElement>() // 顶部操作栏 ref
|
||||
const layoutMenuTagRef = ref<HTMLElement>() // 标签页 ref
|
||||
const layoutFooterRef = ref<HTMLElement>() // 底部版权 ref
|
||||
|
||||
const { getMenuTagSwitch, getCopyrightSwitch } = useSettingGetters()
|
||||
const { getLockAppScreen } = useAppLockScreen()
|
||||
@ -70,6 +70,8 @@ export default defineComponent({
|
||||
ref={LAYOUT_CONTENT_REF}
|
||||
class={[
|
||||
'r-layout-full__viewer-content',
|
||||
'r-layout-full__viewer-content--maximize--light',
|
||||
'r-layout-full__viewer-content--maximize--dark',
|
||||
layoutContentMaximize
|
||||
? 'r-layout-full__viewer-content--maximize'
|
||||
: null,
|
||||
|
@ -67,7 +67,7 @@ icon: icon 图标, 用于 Menu 菜单(依赖 RIcon 组件实现)
|
||||
windowOpen: 超链接打开(新开窗口打开)
|
||||
role: 权限表
|
||||
hidden: 是否显示
|
||||
noLocalTitle: 不使用国际化渲染 Menu Titile
|
||||
noLocalTitle: 不使用国际化渲染 Menu Title
|
||||
ignoreAutoResetScroll: 该页面内容区域自动初始化滚动条位置
|
||||
keepAlive: 是否缓存该页面(需要配置 APP_KEEP_ALIVE setupKeepAlive 属性为 true 启用才有效)
|
||||
sameLevel: 是否标记该路由为平级模式,如果标记为平级模式,会使路由菜单项隐藏。如果在含有子节点处,设置了该属性,会导致子节点全部被隐藏。并且该模块,在后续的使用 url 地址导航跳转时,如果在非当前路由层级层面跳转的该路由,会在当前的面包屑后面追加该模块的信息,触发跳转时,不会修改面包屑、标签页(优先级最高)
|
||||
|
@ -20,8 +20,8 @@
|
||||
* 如果不设置 order 属性, 则会默认排在前面
|
||||
*/
|
||||
|
||||
import { combineRawRouteModules } from '@/router/helper/helper'
|
||||
import { orderRoutes } from '@/router/helper/helper'
|
||||
import { combineRawRouteModules } from '@/router/helper/setupHelper'
|
||||
import { orderRoutes } from '@/router/helper/setupHelper'
|
||||
|
||||
/** 获取所有被合并与排序的路由 */
|
||||
export const getAppRawRoutes = () => orderRoutes(combineRawRouteModules())
|
||||
|
@ -11,7 +11,7 @@
|
||||
|
||||
/**
|
||||
*
|
||||
* helper 包入口
|
||||
* setupHelper 包入口
|
||||
*
|
||||
* 该包一般是用于该模块一些处理的辅助方法
|
||||
* 通常不会用于其他地方
|
@ -1,5 +1,5 @@
|
||||
import { createRouter, createWebHashHistory } from 'vue-router'
|
||||
import { scrollViewToTop } from '@/router/helper/helper'
|
||||
import { scrollViewToTop } from '@/router/helper/setupHelper'
|
||||
import { vueRouterRegister } from '@/router/helper/routerCopilot'
|
||||
import { useVueRouter } from '@/hooks/web/index'
|
||||
|
||||
@ -10,6 +10,11 @@ import type { RouteRecordRaw, Router } from 'vue-router'
|
||||
|
||||
export let router: Router
|
||||
|
||||
/**
|
||||
*
|
||||
* 创建 vue router 实例
|
||||
* 注册 scrollBehavior 方法
|
||||
*/
|
||||
const createVueRouter = async () => {
|
||||
return createRouter({
|
||||
history: createWebHashHistory(),
|
||||
|
@ -4,11 +4,20 @@ import { ROOT_ROUTE } from '@/app-config/appConfig'
|
||||
import { expandRoutes } from '@/router/helper/expandRoutes'
|
||||
|
||||
export default async () => [
|
||||
/**
|
||||
*
|
||||
* 首页(一般为 Login 页面)
|
||||
* 整个模板默认导航地址
|
||||
*/
|
||||
{
|
||||
path: '/',
|
||||
name: 'login',
|
||||
component: () => import('@/views/login/index'),
|
||||
},
|
||||
/**
|
||||
*
|
||||
* App Layout 核心页面(一般为登陆后展示的页面)
|
||||
*/
|
||||
{
|
||||
path: '/',
|
||||
name: 'layout',
|
||||
|
@ -13,17 +13,61 @@
|
||||
- 缓存插件 key 应该按照 `piniaXXXStore` 格式命名(XXX 表示该包名称)
|
||||
|
||||
```ts
|
||||
export const useDemoStore = defineStore('demo', () => {}, {
|
||||
persist: {
|
||||
key: 'piniaDemoStore',
|
||||
paths: ['demoState'],
|
||||
storage: sessionStorage | localStorage,
|
||||
export const piniaDemoStore = defineStore(
|
||||
'demo',
|
||||
() => {
|
||||
const demoRef = ref('hello')
|
||||
|
||||
const updateDemoRef = (str: string) => (demoRef.value = str)
|
||||
|
||||
return {
|
||||
demoRef,
|
||||
updateDemoRef,
|
||||
}
|
||||
},
|
||||
})
|
||||
{
|
||||
persist: {
|
||||
key: 'piniaDemoStore',
|
||||
paths: ['demoRef'],
|
||||
storage: sessionStorage | localStorage,
|
||||
},
|
||||
},
|
||||
)
|
||||
```
|
||||
|
||||
- 最后在 index.ts 中暴露使用
|
||||
- 注册对应的 getters actions
|
||||
|
||||
```ts
|
||||
export { useDemo } from './modules/demo/index'
|
||||
// piniaDemoStore
|
||||
import piniaDemoStore from '../index'
|
||||
|
||||
export const useDemoGetters = () => {
|
||||
const variable = piniaDemoStore()
|
||||
|
||||
const getDemoRef = computed(() => variable.demoRef)
|
||||
|
||||
return {
|
||||
getDemoRef,
|
||||
}
|
||||
}
|
||||
|
||||
export const useMenuActions = () => {
|
||||
const { updateDemoRef } = piniaDemoStore()
|
||||
|
||||
return {
|
||||
updateDemoRef,
|
||||
}
|
||||
}
|
||||
```
|
||||
|
||||
- 最后在 index.ts 中暴露
|
||||
|
||||
```ts
|
||||
export { useDemoGetters, useMenuActions } from './modules/demo/index'
|
||||
```
|
||||
|
||||
- 使用
|
||||
|
||||
```ts
|
||||
import { useDemoGetters, useMenuActions } from '@/store'
|
||||
```
|
||||
|
@ -18,7 +18,7 @@
|
||||
*/
|
||||
import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'
|
||||
|
||||
// 导出仓库实例
|
||||
// 导出仓库实例,不建议直接使用 store
|
||||
export { piniaSettingStore } from './modules/setting/index' // import { piniaSettingStore } from '@/store' 即可使用
|
||||
export { piniaMenuStore } from './modules/menu/index'
|
||||
export { piniaSigningStore } from './modules/signing/index'
|
||||
|
@ -11,6 +11,18 @@
|
||||
|
||||
import type { StorageLike, RemoveStorageKey } from '@/types/modules/utils'
|
||||
|
||||
/**
|
||||
*
|
||||
* @param key cache key
|
||||
* @param storageType session or local
|
||||
*
|
||||
* 查找当前缓存中是否含有某个 key
|
||||
* 默认查找 sessionStorage
|
||||
*/
|
||||
function hasStorage(key: string, storageType: StorageLike = 'sessionStorage') {
|
||||
return getStorage(key, storageType) !== null
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param key 需要设置的key
|
||||
@ -52,8 +64,11 @@ function getStorage<T = unknown>(
|
||||
|
||||
/**
|
||||
*
|
||||
* @param key 需要获取目标缓存的key
|
||||
* @returns 获取缓存值
|
||||
* @param key cache
|
||||
* @param storageType session or local
|
||||
* @param defaultValue default value
|
||||
*
|
||||
* 获取缓存值
|
||||
*/
|
||||
function getStorage<T = unknown>(
|
||||
key: string,
|
||||
@ -121,4 +136,4 @@ function removeStorage(
|
||||
}
|
||||
}
|
||||
|
||||
export { setStorage, getStorage, removeStorage }
|
||||
export { setStorage, getStorage, removeStorage, hasStorage }
|
||||
|
@ -1,6 +1,7 @@
|
||||
import { isValueType } from '@/utils/basic'
|
||||
import { APP_REGEX } from '@/app-config/regexConfig'
|
||||
import { unrefElement } from '@/utils/vue/index'
|
||||
import { watchEffectWithTarget } from '@/utils/vue/index'
|
||||
|
||||
import type {
|
||||
EventListenerOrEventListenerObject,
|
||||
@ -25,11 +26,15 @@ export const on = (
|
||||
handler: EventListenerOrEventListenerObject,
|
||||
useCapture: boolean | AddEventListenerOptions = false,
|
||||
) => {
|
||||
const targetElement = unrefElement(target, window)
|
||||
const targetElement = computed(() => unrefElement(target, window))
|
||||
|
||||
if (targetElement && event && handler) {
|
||||
targetElement.addEventListener(event, handler, useCapture)
|
||||
const update = () => {
|
||||
if (targetElement.value && event && handler) {
|
||||
targetElement.value.addEventListener(event, handler, useCapture)
|
||||
}
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
@ -47,11 +52,15 @@ export const off = (
|
||||
handler: EventListenerOrEventListenerObject,
|
||||
useCapture: boolean | AddEventListenerOptions = false,
|
||||
) => {
|
||||
const targetElement = unrefElement(target, window)
|
||||
const targetElement = computed(() => unrefElement(target, window))
|
||||
|
||||
if (targetElement && event && handler) {
|
||||
targetElement.removeEventListener(event, handler, useCapture)
|
||||
const update = () => {
|
||||
if (targetElement.value && event && handler) {
|
||||
targetElement.value.removeEventListener(event, handler, useCapture)
|
||||
}
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
@ -65,17 +74,21 @@ export const addClass = (
|
||||
target: BasicTarget<Element | HTMLElement | SVGAElement>,
|
||||
className: string,
|
||||
) => {
|
||||
const targetElement = unrefElement(target)
|
||||
const targetElement = computed(() => unrefElement(target))
|
||||
|
||||
if (targetElement) {
|
||||
const classes = className.trim().split(' ')
|
||||
const update = () => {
|
||||
if (targetElement.value) {
|
||||
const classes = className.trim().split(' ')
|
||||
|
||||
classes.forEach((item) => {
|
||||
if (item) {
|
||||
targetElement.classList.add(item)
|
||||
}
|
||||
})
|
||||
classes.forEach((item) => {
|
||||
if (item) {
|
||||
targetElement.value!.classList.add(item)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
@ -90,23 +103,27 @@ export const removeClass = (
|
||||
target: BasicTarget<Element | HTMLElement | SVGAElement>,
|
||||
className: string | 'removeAllClass',
|
||||
) => {
|
||||
const targetElement = unrefElement(target)
|
||||
const targetElement = computed(() => unrefElement(target))
|
||||
|
||||
if (targetElement) {
|
||||
if (className === 'removeAllClass') {
|
||||
const classList = targetElement.classList
|
||||
const update = () => {
|
||||
if (targetElement.value) {
|
||||
if (className === 'removeAllClass') {
|
||||
const classList = targetElement.value.classList
|
||||
|
||||
classList.forEach((curr) => classList.remove(curr))
|
||||
} else {
|
||||
const classes = className.trim().split(' ')
|
||||
classList.forEach((curr) => classList.remove(curr))
|
||||
} else {
|
||||
const classes = className.trim().split(' ')
|
||||
|
||||
classes.forEach((item) => {
|
||||
if (item) {
|
||||
targetElement.classList.remove(item)
|
||||
}
|
||||
})
|
||||
classes.forEach((item) => {
|
||||
if (item) {
|
||||
targetElement.value!.classList.remove(item)
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
@ -162,35 +179,39 @@ export const addStyle = (
|
||||
target: BasicTarget<HTMLElement | SVGAElement>,
|
||||
styles: PartialCSSStyleDeclaration | string,
|
||||
) => {
|
||||
const targetElement = unrefElement(target)
|
||||
const targetElement = computed(() => unrefElement(target))
|
||||
|
||||
if (!targetElement) {
|
||||
if (!targetElement.value) {
|
||||
return
|
||||
}
|
||||
|
||||
let styleObj: PartialCSSStyleDeclaration
|
||||
|
||||
if (isValueType<string>(styles, 'String')) {
|
||||
styleObj = styles.split(';').reduce((pre, curr) => {
|
||||
const [key, value] = curr.split(':').map((s) => s.trim())
|
||||
const update = () => {
|
||||
if (isValueType<string>(styles, 'String')) {
|
||||
styleObj = styles.split(';').reduce((pre, curr) => {
|
||||
const [key, value] = curr.split(':').map((s) => s.trim())
|
||||
|
||||
if (key && value) {
|
||||
pre[key] = value
|
||||
if (key && value) {
|
||||
pre[key] = value
|
||||
}
|
||||
|
||||
return pre
|
||||
}, {} as PartialCSSStyleDeclaration)
|
||||
} else {
|
||||
styleObj = styles
|
||||
}
|
||||
|
||||
Object.keys(styleObj).forEach((key) => {
|
||||
const value = styleObj[key]
|
||||
|
||||
if (key in targetElement.value!.style) {
|
||||
targetElement.value!.style[key] = value
|
||||
}
|
||||
|
||||
return pre
|
||||
}, {} as PartialCSSStyleDeclaration)
|
||||
} else {
|
||||
styleObj = styles
|
||||
})
|
||||
}
|
||||
|
||||
Object.keys(styleObj).forEach((key) => {
|
||||
const value = styleObj[key]
|
||||
|
||||
if (key in targetElement.style) {
|
||||
targetElement.style[key] = value
|
||||
}
|
||||
})
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
@ -202,15 +223,19 @@ export const removeStyle = (
|
||||
target: BasicTarget<HTMLElement | SVGAElement>,
|
||||
styles: (keyof CSSStyleDeclaration & string)[],
|
||||
) => {
|
||||
const targetElement = unrefElement(target)
|
||||
const targetElement = computed(() => unrefElement(target))
|
||||
|
||||
if (!targetElement) {
|
||||
if (!targetElement.value) {
|
||||
return
|
||||
}
|
||||
|
||||
styles.forEach((curr) => {
|
||||
targetElement.style.removeProperty(curr)
|
||||
})
|
||||
const update = () => {
|
||||
styles.forEach((curr) => {
|
||||
targetElement.value!.style.removeProperty(curr)
|
||||
})
|
||||
}
|
||||
|
||||
watchEffectWithTarget(update)
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -1,3 +1,5 @@
|
||||
export { call } from './call'
|
||||
export { unrefElement } from './unrefElement'
|
||||
export { renderNode } from './renderNode'
|
||||
export { scopeDispose } from './scopeDispose'
|
||||
export { watchEffectWithTarget } from './watchEffectWithTarget'
|
||||
|
30
src/utils/vue/scopeDispose.ts
Normal file
@ -0,0 +1,30 @@
|
||||
/**
|
||||
*
|
||||
* @author Ray <https://github.com/XiaoDaiGua-Ray>
|
||||
*
|
||||
* @date 2023-11-10
|
||||
*
|
||||
* @workspace ray-template
|
||||
*
|
||||
* @remark 今天也是元气满满撸代码的一天
|
||||
*/
|
||||
|
||||
import { getCurrentScope, onScopeDispose } from 'vue'
|
||||
|
||||
import type { AnyFC } from '@/types/modules/utils'
|
||||
|
||||
/**
|
||||
*
|
||||
* @param fc effect 作用域卸载时需执行函数
|
||||
*
|
||||
* @remark 返回 true 表示获取到 effect 作用域并且卸载;false 表示未存在 effect 作用域
|
||||
*/
|
||||
export function scopeDispose(fc: AnyFC) {
|
||||
if (getCurrentScope()) {
|
||||
onScopeDispose(fc)
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
return false
|
||||
}
|
32
src/utils/vue/watchEffectWithTarget.ts
Normal file
@ -0,0 +1,32 @@
|
||||
/**
|
||||
*
|
||||
* @author Ray <https://github.com/XiaoDaiGua-Ray>
|
||||
*
|
||||
* @date 2023-11-10
|
||||
*
|
||||
* @workspace ray-template
|
||||
*
|
||||
* @remark 今天也是元气满满撸代码的一天
|
||||
*/
|
||||
|
||||
import { scopeDispose } from './scopeDispose'
|
||||
|
||||
import type { WatchOptionsBase } from 'vue'
|
||||
import type { AnyFC } from '@/types/modules/utils'
|
||||
|
||||
/**
|
||||
*
|
||||
* @param fc 副作用函数
|
||||
* @param watchOptions watchEffect 配置项
|
||||
*
|
||||
* 该方法使用 watchEffect 实现副作用函数的执行
|
||||
* 并且能够在 effect 作用域卸载时,自动停止监听
|
||||
*/
|
||||
export function watchEffectWithTarget(
|
||||
fc: AnyFC,
|
||||
watchOptions?: WatchOptionsBase,
|
||||
) {
|
||||
const stop = watchEffect(fc, watchOptions)
|
||||
|
||||
scopeDispose(stop)
|
||||
}
|
@ -35,6 +35,12 @@ export default defineComponent({
|
||||
|
||||
return (
|
||||
<NSpace wrapItem={false} vertical>
|
||||
<NCard title="接口说明">
|
||||
<h3>
|
||||
hooks/template 包存放模板专属 hook
|
||||
方法。这里不做过多的赘述,可以查看文档具体描述。
|
||||
</h3>
|
||||
</NCard>
|
||||
<NCard title="useAppMenu 导航方法">
|
||||
<h3>
|
||||
navigationTo
|
||||
|