304 lines
6.6 KiB
Vue
304 lines
6.6 KiB
Vue
<template>
|
|
<div
|
|
:class="{ onLockLogin: showLogin }"
|
|
class="lockscreen"
|
|
@keyup="onLockLogin(true)"
|
|
@mousedown.stop
|
|
@contextmenu.prevent
|
|
>
|
|
<template v-if="!showLogin">
|
|
<div class="lock-box">
|
|
<div class="lock">
|
|
<span class="lock-icon" title="解锁屏幕" @click="onLockLogin(true)">
|
|
<n-icon>
|
|
<lock-outlined />
|
|
</n-icon>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<!--充电-->
|
|
<recharge
|
|
:battery="battery"
|
|
:battery-status="batteryStatus"
|
|
:calc-discharging-time="calcDischargingTime"
|
|
:calc-charging-time="calcChargingTime"
|
|
/>
|
|
|
|
<div class="local-time">
|
|
<div class="time">{{ hour }}:{{ minute }}</div>
|
|
<div class="date">{{ month }}月{{ day }}号,星期{{ week }}</div>
|
|
</div>
|
|
<div class="computer-status">
|
|
<span :class="{ offline: !online }" class="network">
|
|
<wifi-outlined class="network" />
|
|
</span>
|
|
<api-outlined />
|
|
</div>
|
|
</template>
|
|
|
|
<!--登录-->
|
|
<template v-if="showLogin">
|
|
<div class="login-box">
|
|
<n-avatar :size="128">
|
|
<n-icon>
|
|
<user-outlined />
|
|
</n-icon>
|
|
</n-avatar>
|
|
<div class="username">{{ loginParams.username }}</div>
|
|
<n-input
|
|
type="password"
|
|
autofocus
|
|
v-model:value="loginParams.password"
|
|
@keyup.enter="onLogin"
|
|
placeholder="请输入登录密码"
|
|
>
|
|
<template #suffix>
|
|
<n-icon @click="onLogin" style="cursor: pointer">
|
|
<LoadingOutlined v-if="loginLoading" />
|
|
<arrow-right-outlined v-else />
|
|
</n-icon>
|
|
</template>
|
|
</n-input>
|
|
|
|
<div class="flex w-full" v-if="isLoginError">
|
|
<span class="text-red-500">{{ errorMsg }}</span>
|
|
</div>
|
|
|
|
<div class="flex justify-around w-full mt-1">
|
|
<div><a @click="showLogin = false">返回</a></div>
|
|
<div><a @click="goLogin">重新登录</a></div>
|
|
<div><a @click="onLogin">进入系统</a></div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import { defineComponent, reactive, toRefs } from 'vue';
|
|
import { ResultEnum } from '@/enums/httpEnum';
|
|
import recharge from './Recharge.vue';
|
|
import {
|
|
LockOutlined,
|
|
LoadingOutlined,
|
|
UserOutlined,
|
|
ApiOutlined,
|
|
ArrowRightOutlined,
|
|
WifiOutlined,
|
|
} from '@vicons/antd';
|
|
|
|
import { useRouter, useRoute } from 'vue-router';
|
|
import { useOnline } from '@/hooks/useOnline';
|
|
import { useTime } from '@/hooks/useTime';
|
|
import { useBattery } from '@/hooks/useBattery';
|
|
import { useScreenLockStore } from '@/store/modules/screenLock';
|
|
import { UserInfoType, useUserStore } from '@/store/modules/user';
|
|
|
|
export default defineComponent({
|
|
name: 'ScreenLock',
|
|
components: {
|
|
LockOutlined,
|
|
LoadingOutlined,
|
|
UserOutlined,
|
|
ArrowRightOutlined,
|
|
ApiOutlined,
|
|
WifiOutlined,
|
|
recharge,
|
|
},
|
|
setup() {
|
|
const useScreenLock = useScreenLockStore();
|
|
const userStore = useUserStore();
|
|
|
|
// 获取时间
|
|
const { month, day, hour, minute, second, week } = useTime();
|
|
const { online } = useOnline();
|
|
|
|
const router = useRouter();
|
|
const route = useRoute();
|
|
|
|
const { battery, batteryStatus, calcDischargingTime, calcChargingTime } =
|
|
useBattery();
|
|
const userInfo: UserInfoType = userStore.getUserInfo || {};
|
|
const username = userInfo['username'] || '';
|
|
const state = reactive({
|
|
showLogin: false,
|
|
loginLoading: false, // 正在登录
|
|
isLoginError: false, //密码错误
|
|
errorMsg: '密码错误',
|
|
loginParams: {
|
|
username: username || '',
|
|
password: '',
|
|
},
|
|
});
|
|
|
|
// 解锁登录
|
|
const onLockLogin = (value: boolean) => (state.showLogin = value);
|
|
|
|
// 登录
|
|
const onLogin = async () => {
|
|
if (!state.loginParams.password.trim()) {
|
|
return;
|
|
}
|
|
const params = {
|
|
isLock: true,
|
|
...state.loginParams,
|
|
};
|
|
state.loginLoading = true;
|
|
const { status } = await userStore.login(params);
|
|
|
|
if (status === ResultEnum.STATUS) {
|
|
onLockLogin(false);
|
|
useScreenLock.setLock(false);
|
|
} else {
|
|
state.errorMsg = '密码错误';
|
|
state.isLoginError = true;
|
|
}
|
|
state.loginLoading = false;
|
|
};
|
|
|
|
//重新登录
|
|
const goLogin = () => {
|
|
onLockLogin(false);
|
|
useScreenLock.setLock(false);
|
|
router.replace({
|
|
path: '/login',
|
|
query: {
|
|
redirect: route.fullPath,
|
|
},
|
|
});
|
|
};
|
|
|
|
return {
|
|
...toRefs(state),
|
|
online,
|
|
month,
|
|
day,
|
|
hour,
|
|
minute,
|
|
second,
|
|
week,
|
|
battery,
|
|
batteryStatus,
|
|
calcDischargingTime,
|
|
calcChargingTime,
|
|
onLockLogin,
|
|
onLogin,
|
|
goLogin,
|
|
};
|
|
},
|
|
});
|
|
</script>
|
|
|
|
<style lang="scss" scoped>
|
|
.lockscreen {
|
|
position: fixed;
|
|
inset: 0;
|
|
z-index: 9999;
|
|
display: flex;
|
|
overflow: hidden;
|
|
color: white;
|
|
background: #000;
|
|
|
|
&.onLockLogin {
|
|
background-color: rgb(25 28 34 / 88%);
|
|
backdrop-filter: blur(7px);
|
|
}
|
|
|
|
.login-box {
|
|
position: absolute;
|
|
top: 45%;
|
|
left: 50%;
|
|
display: flex;
|
|
flex-direction: column;
|
|
align-items: center;
|
|
justify-content: center;
|
|
transform: translate(-50%, -50%);
|
|
|
|
> * {
|
|
margin-bottom: 14px;
|
|
}
|
|
|
|
.username {
|
|
font-size: 30px;
|
|
}
|
|
}
|
|
|
|
.lock-box {
|
|
position: absolute;
|
|
top: 20px;
|
|
left: 50%;
|
|
z-index: 100;
|
|
font-size: 34px;
|
|
transform: translateX(-50%);
|
|
|
|
.tips {
|
|
color: white;
|
|
cursor: text;
|
|
}
|
|
|
|
.lock {
|
|
display: flex;
|
|
justify-content: center;
|
|
|
|
.lock-icon {
|
|
cursor: pointer;
|
|
|
|
.anticon-unlock {
|
|
display: none;
|
|
}
|
|
|
|
&:hover .anticon-unlock {
|
|
display: initial;
|
|
}
|
|
|
|
&:hover .anticon-lock {
|
|
display: none;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
.local-time {
|
|
position: absolute;
|
|
bottom: 60px;
|
|
left: 60px;
|
|
font-family: helvetica;
|
|
|
|
.time {
|
|
font-size: 70px;
|
|
}
|
|
|
|
.date {
|
|
font-size: 40px;
|
|
}
|
|
}
|
|
|
|
.computer-status {
|
|
position: absolute;
|
|
right: 60px;
|
|
bottom: 60px;
|
|
font-size: 24px;
|
|
|
|
> * {
|
|
margin-left: 14px;
|
|
}
|
|
|
|
.network {
|
|
position: relative;
|
|
|
|
&.offline::before {
|
|
position: absolute;
|
|
top: 50%;
|
|
left: 50%;
|
|
z-index: 10;
|
|
width: 2px;
|
|
height: 28px;
|
|
content: '';
|
|
background-color: red;
|
|
transform: translate(-50%, -50%) rotate(45deg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|