重写AppLockSreen组件

This commit is contained in:
ray_wuhao 2023-06-20 16:33:47 +08:00
parent 4e6f70280c
commit 2a64d07a42
12 changed files with 392 additions and 280 deletions

View File

@ -10,6 +10,7 @@
- 更新了 router permission 方法(路由守卫) - 更新了 router permission 方法(路由守卫)
- 补充了一些模块文档 - 补充了一些模块文档
- 搜索支持以菜单模块的 icon 进行渲染,如果为空则以 icon table 默认填充 - 搜索支持以菜单模块的 icon 进行渲染,如果为空则以 icon table 默认填充
- 重写锁屏功能,现在将锁屏逻辑与解锁逻辑拆分为两个组件
### Fixes ### Fixes

View File

@ -36,7 +36,8 @@
- 锁屏 - 锁屏
- 自动化路由 - 自动化路由
- 带有拓展功能的表格 - 带有拓展功能的表格
- 封装 `axios` 自动取消重复请求 - 封装 `axios` 自动取消重复请求,暴露拦截器注册器
- 全局菜单搜索
- 动态菜单(多级菜单) - 动态菜单(多级菜单)
- 主题色切换 - 主题色切换
- 错误页 - 错误页
@ -58,6 +59,7 @@
> 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。 > 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。
> 使用 `naive ui` 作为组件库。 > 使用 `naive ui` 作为组件库。
> 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。 > 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。
> 该模板不支持移动端设备。
## 提示 ## 提示

View File

@ -0,0 +1,36 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-20
*
* @workspace ray-template
*
* @remark
*/
/**
*
*
*
* ,
*/
const appLockScreen = useStorage('isAppLockScreen', false, sessionStorage, {
mergeDefaults: true,
})
const useAppLockScreen = () => {
const setLockAppScreen = (bool: boolean) => {
appLockScreen.value = bool
}
const getLockAppScreen = () => appLockScreen.value
return {
setLockAppScreen,
getLockAppScreen,
}
}
export default useAppLockScreen

View File

@ -0,0 +1,91 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-20
*
* @workspace ray-template
*
* @remark
*/
/** 锁屏界面 */
import { NInput, NForm, NFormItem, NButton, NSpace } from 'naive-ui'
import AppAvatar from '@/components/AppComponents/AppAvatar/index'
import { useSetting } from '@/store'
import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
import {
rules,
useCondition,
autoFouceInput,
} from '@/components/AppComponents/AppLockScreen/hook'
import type { FormInst, InputInst } from 'naive-ui'
const LockScreen = defineComponent({
name: 'LockScreen',
setup() {
const formInstRef = ref<FormInst | null>(null)
const inputInstRef = ref<InputInst | null>(null)
const { setLockAppScreen } = useAppLockScreen()
const { changeSwitcher } = useSetting()
const state = reactive({
lockCondition: useCondition(),
})
/** 锁屏 */
const lockScreen = () => {
formInstRef.value?.validate((error) => {
if (!error) {
setLockAppScreen(true)
changeSwitcher(true, 'lockScreenSwitch')
state.lockCondition = useCondition()
}
})
}
autoFouceInput(inputInstRef)
return {
...toRefs(state),
lockScreen,
formInstRef,
inputInstRef,
}
},
render() {
return (
<div class="app-lock-screen__input">
<AppAvatar vertical align="center" avatarSize={52} />
<NForm
ref="formInstRef"
model={this.lockCondition}
rules={rules}
labelPlacement="left"
>
<NFormItem path="lockPassword">
<NInput
ref="inputInstRef"
v-model:value={this.lockCondition.lockPassword}
type="password"
placeholder="请输入锁屏密码"
clearable
minlength={6}
maxlength={12}
/>
</NFormItem>
<NButton type="primary" onClick={this.lockScreen.bind(this)}>
</NButton>
</NForm>
</div>
)
},
})
export default LockScreen

View File

@ -0,0 +1,145 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-20
*
* @workspace ray-template
*
* @remark
*/
/** 解锁界面 */
import { NInput, NForm, NFormItem, NButton, NSpace } from 'naive-ui'
import AppAvatar from '@/components/AppComponents/AppAvatar/index'
import dayjs from 'dayjs'
import { useSetting, useSignin } from '@/store'
import {
rules,
useCondition,
autoFouceInput,
} from '@/components/AppComponents/AppLockScreen/hook'
import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
import type { FormInst, InputInst } from 'naive-ui'
const UnlockScreen = defineComponent({
name: 'UnlockScreen',
setup() {
const formRef = ref<FormInst | null>(null)
const inputInstRef = ref<InputInst | null>(null)
const { logout } = useSignin()
const { changeSwitcher } = useSetting()
const { setLockAppScreen } = useAppLockScreen()
const HH_MM_FORMAT = 'HH:mm'
const AM_PM_FORMAT = 'A'
const YY_MM_DD_FORMAT = 'YY年MM月DD日'
const DDD_FORMAT = 'ddd'
const state = reactive({
lockCondition: useCondition(),
HH_MM: dayjs().format(HH_MM_FORMAT),
AM_PM: dayjs().locale('en').format(AM_PM_FORMAT),
YY_MM_DD: dayjs().format(YY_MM_DD_FORMAT),
DDD: dayjs().format(DDD_FORMAT),
})
/** 退出登陆并且回到登陆页 */
const backToSignin = () => {
window.$dialog.warning({
title: '警告',
content: '是否返回到登陆页?',
positiveText: '确定',
negativeText: '取消',
onPositiveClick: () => {
logout()
setTimeout(() => {
changeSwitcher(false, 'lockScreenSwitch')
})
},
})
}
/** 解锁 */
const unlockScreen = () => {
formRef.value?.validate((error) => {
if (!error) {
setLockAppScreen(false)
changeSwitcher(false, 'lockScreenSwitch')
state.lockCondition = useCondition()
}
})
}
autoFouceInput(inputInstRef)
return {
...toRefs(state),
backToSignin,
unlockScreen,
formRef,
inputInstRef,
}
},
render() {
return (
<div class="app-lock-screen__unlock">
<div class="app-lock-screen__unlock__content">
<div class="app-lock-screen__unlock__content-bg">
<div class="left">{this.HH_MM?.split(':')[0]}</div>
<div class="right">{this.HH_MM?.split(':')[1]}</div>
</div>
<div class="app-lock-screen__unlock__content-avatar">
<AppAvatar vertical align="center" avatarSize={52} />
</div>
<div class="app-lock-screen__unlock__content-input">
<NForm ref="formRef" model={this.lockCondition} rules={rules}>
<NFormItem path="lockPassword">
<NInput
ref="inputInstRef"
v-model:value={this.lockCondition.lockPassword}
type="password"
placeholder="请输入解锁密码"
clearable
minlength={6}
maxlength={12}
/>
</NFormItem>
<NSpace justify="space-between">
<NButton
type="primary"
text
onClick={this.backToSignin.bind(this)}
>
</NButton>
<NButton
type="primary"
text
onClick={this.unlockScreen.bind(this)}
>
</NButton>
</NSpace>
</NForm>
</div>
<div class="app-lock-screen__unlock__content-date">
<div class="current-date">
{this.HH_MM}&nbsp;<span>{this.AM_PM}</span>
</div>
<div class="current-year">
{this.YY_MM_DD}&nbsp;<span>{this.DDD}</span>
</div>
</div>
</div>
</div>
)
},
})
export default UnlockScreen

View File

@ -0,0 +1,38 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-06-20
*
* @workspace ray-template
*
* @remark
*/
import type { InputInst } from 'naive-ui'
import type { Ref } from 'vue'
/** 统一的校验锁屏密码校验规则 */
export const rules = {
lockPassword: {
required: true,
message: '请输入正确格式密码',
min: 6,
max: 12,
trigger: ['input'],
},
}
/** 锁屏密码参数 */
export const useCondition = () => {
return {
lockPassword: null,
}
}
/** 自动获取焦点 */
export const autoFouceInput = (inputInstRef: Ref<InputInst | null>) => {
nextTick(() => {
inputInstRef.value?.focus()
})
}

View File

@ -1,19 +1,23 @@
.lock-screen { .app-lock-screen__content {
position: fixed; & .app-lock-screen__input {
left: 0; & button[class*="n-button"] {
right: 0; width: 100%;
top: 0; }
bottom: 0;
background: black;
& .lock-screen__content { & form[class*="n-form"] {
margin: 24px 0px;
}
}
& .app-lock-screen__unlock {
.app-lock-screen__unlock__content {
position: relative; position: relative;
width: 100%; width: 100%;
height: 100%; height: 100%;
@include flexCenter; @include flexCenter;
flex-direction: column; flex-direction: column;
& .lock-screen__content-bg { & .app-lock-screen__unlock__content-bg {
position: absolute; position: absolute;
width: 100%; width: 100%;
height: 100%; height: 100%;
@ -33,19 +37,19 @@
} }
} }
& .lock-screen__content-avatar { & .app-lock-screen__unlock__content-avatar {
margin-top: 5px; margin-top: 5px;
color: #bababa; color: #bababa;
font-weight: 500; font-weight: 500;
z-index: 1; z-index: 1;
} }
& .lock-screen__content-input { & .app-lock-screen__unlock__content-input {
width: 260px; width: 260px;
z-index: 1; z-index: 1;
} }
& .lock-screen__content-date { & .app-lock-screen__unlock__content-date {
position: fixed; position: fixed;
width: 100%; width: 100%;
font-size: 3rem; font-size: 3rem;
@ -58,10 +62,7 @@
& .current-date span { & .current-date span {
font-size: 1.5rem; font-size: 1.5rem;
} }
}
// & .current-year span {
// font-size: 0.75rem;
// }
} }
} }
} }

View File

@ -13,139 +13,28 @@
* *
* , * ,
* *
*
* @deprecated
* ,
* , 使
*/ */
import './index.scss' import './index.scss'
import { NModal, NInput, NForm, NFormItem, NButton, NSpace } from 'naive-ui' import { NModal } from 'naive-ui'
import AppAvatar from '@/components/AppComponents/AppAvatar/index' import LockScreen from './components/LockScreen'
import UnlockScreen from './components/UnlockScreen'
import { useSetting, useSignin } from '@/store' import { useSetting } from '@/store'
import { getCache, setCache } from '@/utils/cache' import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
import dayjs from 'dayjs'
import { APP_CATCH_KEY } from '@/appConfig/appConfig'
import type { FormInst, InputInst } from 'naive-ui' const AppLockScreen = defineComponent({
name: 'AppLockScreen',
const LockScreen = defineComponent({
name: 'LockScreen',
setup() { setup() {
const formRef = ref<FormInst>()
const inputInstRef = ref<InputInst>()
const settingStore = useSetting() const settingStore = useSetting()
const signinStore = useSignin() const { lockScreenSwitch } = storeToRefs(settingStore)
/** lockScreenSwitch 检测是否激活锁屏弹窗 */
const { lockScreenSwitch, lockScreenInputSwitch } =
storeToRefs(settingStore)
const { changeSwitcher } = settingStore
const { logout } = signinStore
const HH_MM_FORMAT = 'HH:mm' const { getLockAppScreen } = useAppLockScreen()
const AM_PM_FORMAT = 'A'
const YY_MM_DD_FORMAT = 'YY年MM月DD日'
const DDD_FORMAT = 'ddd'
const state = reactive({
lockCondition: {
pwd: null,
},
HH_MM: dayjs().format(HH_MM_FORMAT),
AM_PM: dayjs().locale('en').format(AM_PM_FORMAT),
YY_MM_DD: dayjs().format(YY_MM_DD_FORMAT),
DDD: dayjs().format(DDD_FORMAT),
})
const rules = {
pwd: {
required: true,
message: '请输入正确格式密码',
min: 6,
max: 12,
trigger: ['input', 'blur'],
},
}
/** 检测是否处于锁屏状态 */
const isLock = useStorage('isLockScreen', false, sessionStorage, {
mergeDefaults: true,
})
const signin = getCache(APP_CATCH_KEY.signin)
const handleLockScreen = () => {
formRef.value?.validate((error) => {
if (!error) {
isLock.value = true
state.lockCondition.pwd = null
setCache('lockScreenPassword', state.lockCondition.pwd)
changeSwitcher(true, 'lockScreenSwitch')
}
})
}
const dayInterval = setInterval(() => {
state.HH_MM = dayjs().format(HH_MM_FORMAT)
state.AM_PM = dayjs().format(AM_PM_FORMAT)
}, 60_000)
const yearInterval = setInterval(() => {
state.YY_MM_DD = dayjs().format(YY_MM_DD_FORMAT)
state.DDD = dayjs().format(DDD_FORMAT)
}, 86_400_000)
const handleBackToSignin = () => {
window.$dialog.warning({
title: '警告',
content: '是否返回到登陆页?',
positiveText: '确定',
negativeText: '取消',
onPositiveClick: () => {
logout()
setTimeout(() => {
changeSwitcher(false, 'lockScreenSwitch')
})
},
})
}
const handleUnlockScreen = () => {
formRef.value?.validate((error) => {
if (!error) {
isLock.value = false
state.lockCondition.pwd = null
changeSwitcher(false, 'lockScreenSwitch')
}
})
}
/** 当弹窗出现后, 自动聚焦密码输入框 */
const handleModalUpdateShow = () => {
nextTick(() => {
inputInstRef.value?.focus()
})
}
onBeforeUnmount(() => {
clearInterval(dayInterval)
clearInterval(yearInterval)
})
return { return {
lockScreenSwitch, lockScreenSwitch,
lockScreenInputSwitch, getLockAppScreen,
rules,
...toRefs(state),
isLock,
handleLockScreen,
formRef,
signin,
handleBackToSignin,
handleUnlockScreen,
inputInstRef,
handleModalUpdateShow,
} }
}, },
render() { render() {
@ -156,105 +45,15 @@ const LockScreen = defineComponent({
show show
maskClosable={false} maskClosable={false}
closeOnEsc={false} closeOnEsc={false}
preset={!this.isLock ? 'dialog' : undefined} preset={!this.getLockAppScreen() ? 'dialog' : undefined}
title="锁定屏幕" title="锁定屏幕"
onAfterEnter={this.handleModalUpdateShow.bind(this)}
> >
{!this.isLock ? ( <div class="app-lock-screen__content">
/** 输入界面 */ {!this.getLockAppScreen() ? <LockScreen /> : <UnlockScreen />}
<div class="lock-screen__input">
<AppAvatar vertical align="center" avatarSize={52} />
<NForm
ref="formRef"
model={this.lockCondition}
rules={this.rules}
labelPlacement="left"
style={{
margin: '24px 0',
}}
>
<NFormItem path="pwd">
<NInput
v-model:value={this.lockCondition.pwd}
type="password"
placeholder="请输入锁屏密码"
clearable
minlength={6}
maxlength={12}
ref="inputInstRef"
/>
</NFormItem>
<NButton
type="primary"
onClick={this.handleLockScreen.bind(this)}
style={{
width: '100%',
}}
>
</NButton>
</NForm>
</div> </div>
) : (
/** 锁屏界面 */
<div class="lock-screen">
<div class="lock-screen__content">
<div class="lock-screen__content-bg">
<div class="left">{this.HH_MM?.split(':')[0]}</div>
<div class="right">{this.HH_MM?.split(':')[1]}</div>
</div>
<div class="lock-screen__content-avatar">
<AppAvatar vertical align="center" avatarSize={52} />
</div>
<div class="lock-screen__content-input">
<NForm
ref="formRef"
model={this.lockCondition}
rules={this.rules}
>
<NFormItem path="pwd">
<NInput
v-model:value={this.lockCondition.pwd}
type="password"
placeholder="请输入解锁密码"
clearable
minlength={6}
maxlength={12}
ref="inputInstRef"
/>
</NFormItem>
<NSpace justify="space-between">
<NButton
type="primary"
text
onClick={this.handleBackToSignin.bind(this)}
>
</NButton>
<NButton
type="primary"
text
onClick={this.handleUnlockScreen.bind(this)}
>
</NButton>
</NSpace>
</NForm>
</div>
<div class="lock-screen__content-date">
<div class="current-date">
{this.HH_MM}&nbsp;<span>{this.AM_PM}</span>
</div>
<div class="current-year">
{this.YY_MM_DD}&nbsp;<span>{this.DDD}</span>
</div>
</div>
</div>
</div>
)}
</NModal> </NModal>
) )
}, },
}) })
export default LockScreen export default AppLockScreen

View File

@ -38,7 +38,7 @@ import type { IconEventMapOptions, IconEventMap } from './type'
const SiderBar = defineComponent({ const SiderBar = defineComponent({
name: 'SiderBar', name: 'SiderBar',
setup(_, { expose }) { setup() {
const settingStore = useSetting() const settingStore = useSetting()
const { t } = useI18n() const { t } = useI18n()

View File

@ -24,6 +24,7 @@ import {
LAYOUT_CONTENT_REF, LAYOUT_CONTENT_REF,
} from '@/appConfig/routerConfig' } from '@/appConfig/routerConfig'
import { layoutHeaderCssVars } from '@/layout/layoutResize' import { layoutHeaderCssVars } from '@/layout/layoutResize'
import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
const Layout = defineComponent({ const Layout = defineComponent({
name: 'RLayout', name: 'RLayout',
@ -37,9 +38,7 @@ const Layout = defineComponent({
const { height: windowHeight } = useWindowSize() const { height: windowHeight } = useWindowSize()
const { menuTagSwitch: modelMenuTagSwitch } = storeToRefs(settingStore) const { menuTagSwitch: modelMenuTagSwitch } = storeToRefs(settingStore)
const { setupAppRoutes } = menuStore const { setupAppRoutes } = menuStore
const isLock = useStorage('isLockScreen', false, sessionStorage, { const { getLockAppScreen } = useAppLockScreen()
mergeDefaults: true,
})
const cssVarsRef = layoutHeaderCssVars([ const cssVarsRef = layoutHeaderCssVars([
layoutSiderBarRef, layoutSiderBarRef,
layoutMenuTagRef, layoutMenuTagRef,
@ -53,7 +52,7 @@ const Layout = defineComponent({
windowHeight, windowHeight,
modelMenuTagSwitch, modelMenuTagSwitch,
cssVarsRef, cssVarsRef,
isLock, getLockAppScreen,
LAYOUT_CONTENT_REF, LAYOUT_CONTENT_REF,
layoutSiderBarRef, layoutSiderBarRef,
layoutMenuTagRef, layoutMenuTagRef,
@ -65,7 +64,7 @@ const Layout = defineComponent({
class={['layout']} class={['layout']}
style={[`height: ${this.windowHeight}px`, this.cssVarsRef]} style={[`height: ${this.windowHeight}px`, this.cssVarsRef]}
> >
{!this.isLock ? ( {!this.getLockAppScreen() ? (
<NLayout class="layout-full" hasSider> <NLayout class="layout-full" hasSider>
<Menu /> <Menu />
<NLayout class="layout__view-container__layout"> <NLayout class="layout__view-container__layout">

View File

@ -21,7 +21,7 @@ const setupTemplate = async () => {
await setupI18n(app) await setupI18n(app)
setupStore(app) await setupStore(app)
setupRouter(app) setupRouter(app)
@ -43,7 +43,7 @@ const setupWujieTemplate = async () => {
await setupI18n(instance) await setupI18n(instance)
setupStore(instance) await setupStore(instance)
setupRouter(instance) setupRouter(instance)

View File

@ -26,7 +26,7 @@ export { useKeepAlive } from './modules/keep-alive/index'
import type { App } from 'vue' import type { App } from 'vue'
/** 设置并且注册 pinia */ /** 设置并且注册 pinia */
export const setupStore = (app: App<Element>) => { export const setupStore = async (app: App<Element>) => {
const store = createPinia() const store = createPinia()
app.use(store) app.use(store)