Vue實現(xiàn)鎖屏功能的示例代碼
這兩天剛做了個新需求,要做個系統(tǒng)鎖屏(當然鎖的是當前的系統(tǒng)),就類似于電腦鎖屏似的。
共兩種情況下鎖屏,一種是無操作一定時間后自動鎖屏;第二種是可以按下組合鍵(快捷鍵)主動進行鎖屏。下面具體來說一下該需求的想法、思路以及實現(xiàn)
然后再說下想法思路吧。首先實現(xiàn)鎖屏一般也就兩種方式:
一種是去寫個頁面,當達到條件時要跳轉(zhuǎn)到該鎖屏頁面,但是有一些要注意的地方, 比如:如果跳轉(zhuǎn)的時候使用的$router.push,那么我是可以通過點擊瀏覽器的回退按鈕回到上一個頁面的,這個要做一下限制。還有一種可能是我如果直接在瀏覽器上輸入某個頁面的路徑,那么是不是也會跳轉(zhuǎn)?所以我們要有個全局的變量來記錄當前是不是鎖屏狀態(tài),如果是鎖屏狀態(tài),我們要對其進行限制。當然這只是舉例的兩種情況,真實做起來或許還有其他情況,反正要考慮周全,不然后面會出現(xiàn)你意想不到的bug
第二種就是直接去寫遮罩層了,也是我當前實現(xiàn)鎖屏的方式。這種方式相對上面來講,會比較簡單一些,也還算方便,我們只需要在達到條件后將該遮罩層顯示出來就可以了,當然遮罩層的優(yōu)先級肯定要設(shè)置最高的,要不然怎么覆蓋掉任意當前的頁面呢,也正是因為這個優(yōu)先級最高,導致在這個遮罩層組件中有些$message這樣的消息提示或彈窗顯示不出來。這也算是一個缺陷。不過一般鎖屏頁面也沒什么太多操作,主要是頁面有個背景圖,然后輸入解鎖密碼進行解鎖就可以了,所以影響并不大。還有一種需要注意,就是我現(xiàn)在已經(jīng)鎖屏頁面了,我這時候打開控制臺,然后是不是可以去修改這個遮罩層的樣式呢?比如我給遮罩層加個display: none; 遮罩層是不是沒了?答案是的,所以我們要做限制! 打開控制臺的方式無非也就兩種方式,一種是F12(Fn + F12),另一種是鼠標右鍵,然后點擊檢查。這樣,我們鎖屏出來的時候就監(jiān)聽鍵盤按下事件,阻止F12的按鍵,同時也阻止鼠標右鍵行為,這樣就可以了,看上去簡單粗暴哈,但也是為了實現(xiàn)這么個需求嘛。
下面再說一下實現(xiàn)的方式。
首先我把遮罩層單獨寫了個組件,里面也都是遮罩層的樣式、邏輯。注意這是個組件,而不是個頁面,哪里用到哪里引入就可以了。
然后就是遮罩層什么時候顯示的問題了,既然是鎖屏,那么肯定是跟登錄沒有關(guān)系的,也就是我們要剛一進系統(tǒng)就開始計時了,比如我們的需求是一分鐘無操作后就要鎖屏,那么我們只需要這個計時無操作的時間就可以了,什么算是操作了?什么算是無操作,鼠標點擊就是操作了,鼠標雙擊是操作了,鼠標右鍵是操作了,鼠標移動是操作了,鼠標滾輪是操作了,鍵盤按下是操作了,基本就這些事件,也就是我們在剛進入系統(tǒng)時就要監(jiān)聽這些事件,同時也開始計時(計時就是定義變量初始值為0,然后setInterval每隔1s加1),如果有其中某個事件觸發(fā)了,我們就要清除掉定時器,并將變量初始化為初始值也就是0,并重新計時,基本跟防抖差不多。然后我們無操作滿足時間后會顯示遮罩層,也就是進入鎖屏狀態(tài),這個是,我們就要把那些事件監(jiān)聽移除掉,并清除掉定時器,但是注意不要將變量初始化為0,因為我們要通過這個變量和我們設(shè)置時間做對比來決定是否顯示遮罩層,進入遮罩層之后就是遮罩層的操作了,解鎖后呢,我們將變量的值初始化為0,隱藏遮罩層,并重新監(jiān)聽那些事件,重新計時就可以了。
哎喲,說太多了說太多了~ 下面上代碼了
我們首先在.env(development / production)的文件中定義個變量,這個變量就是無操作多長時間進入鎖屏,單位 s。也就是可配置的
# 鎖屏時間(s)
VUE_APP_LOCK_TIME = 300
這個也就是說無操作5分鐘后進入鎖屏狀態(tài)
然后我們需要在Vuex中保存一個計時時間lockTime,也就是當這個時間大于等于我們設(shè)置的那個時間就進入鎖屏,即 lockTime >= parseInt(process.env.VUE_APP_LOCK_TIME)
我這里分了個模塊 common (src/store/modules/common.js)
export default {
namespaced: true,
state: {
// 無操作計時時間
lockTime: 0
},
mutations: {
updatelockTime(state, lockTime) {
state.lockTime = lockTime
}
}
}然后是遮罩層(鎖屏)什么時候進行顯示的邏輯:
我這里是封裝在了mixins中,這是為了可以把鎖屏的邏輯單獨抽離出來,邏輯更清晰一些,后面如果說我不想在整個系統(tǒng)鎖屏了,我只要在某些頁面中,也就是只在系統(tǒng)的A頁面、B頁面中才會有鎖屏,其他都不需要鎖屏,那我們直接將邏輯混入到組件就可以了。
// 引入鎖屏遮罩層組件
import Lock from '@/components/lock'
export default {
data() {
return {
timer: null
}
},
components: { Lock },
computed: {
isLock() {
return this.lockTime >= parseInt(process.env.VUE_APP_LOCK_TIME)
},
lockTime: {
get() {
return this.$store.state.common.lockTime
},
set(lockTime) {
this.$store.commit('common/updatelockTime', lockTime)
}
}
},
watch: {
isLock() {
if (this.isLock) {
this.removeHandle()
} else {
this.init()
}
}
},
mounted() {
this.init()
},
destroyed() {
this.removeHandle()
},
methods: {
init() {
this.reckonByTime()
document.addEventListener('click', this.clickHandle)
document.addEventListener('dblclick', this.dblclickHandle)
document.addEventListener('contextmenu', this.contextmenuHandle)
document.addEventListener('mousemove', this.mousemoveHandle)
document.addEventListener('mousewheel', this.mousewheelHandle)
document.addEventListener('keydown', this.keydownHandle)
},
// 移除事件監(jiān)聽、定時器等
removeHandle() {
if (this.timer) {
clearInterval(this.timer)
this.timer = null
}
document.removeEventListener('click', this.clickHandle)
document.removeEventListener('dblclick', this.dblclickHandle)
document.removeEventListener('contextmenu', this.contextmenuHandle)
document.removeEventListener('mousemove', this.mousemoveHandle)
document.removeEventListener('mousewheel', this.mousewheelHandle)
document.removeEventListener('keydown', this.keydownHandle)
},
// 無操作計時
reckonByTime() {
if (this.timer) {
this.lockTime = 0
clearInterval(this.timer)
this.timer = null
}
this.timer = setInterval(() => {
this.lockTime += 1
}, 1000)
},
// 鼠標點擊
clickHandle() {
this.reckonByTime()
},
// 鼠標雙擊
dblclickHandle() {
this.reckonByTime()
},
// 鼠標右鍵
contextmenuHandle() {
this.reckonByTime()
},
// 鼠標移動
mousemoveHandle() {
this.reckonByTime()
},
// 鼠標滾輪
mousewheelHandle() {
this.reckonByTime()
},
// 鍵盤按下
keydownHandle(event) {
const { altKey, ctrlKey, keyCode } = event
if (altKey && ctrlKey && keyCode == 76) {
// Ctrl + Alt + L 快捷鍵直接鎖屏
this.lockTime = parseInt(process.env.VUE_APP_LOCK_TIME)
} else {
this.reckonByTime()
}
}
}
}然后是鎖屏組件(遮罩層組件)
@/components/lock/index.vue
<template>
<div class="lock">
<div>
<img src="../../assets/img/lock/lock-icon.png" alt="">
<el-input class="lockPasswdInp" v-show="!passwdError" ref="lockPasswdInp" size="small" show-password placeholder="密碼" v-model="passwd">
<template slot="append">
<el-button @click="unLock()" style="background: transparent;">
<i class="el-icon-right"></i>
</el-button>
</template>
</el-input>
<div class="passwdError-container" v-show="passwdError">
<div class="passwdError">密碼不正確。請再試一次。</div>
<div class="confirm-btn" @click="reset()">確認</div>
</div>
</div>
<!-- <img @click="logout" class="logout" src="../../assets/img/system/logout.png"> -->
</div>
</template>
<script>
/**
* 注意:由于遮罩層優(yōu)先級太高 類似于$message之類的消息提示等會顯示不出來
*/
export default {
data() {
return {
passwd: '',
passwdError: false
}
},
computed: {
lockPassword: () => process.env.VUE_APP_LOCK_PASSWORD || '123456'
},
mounted() {
this.reset()
document.addEventListener('keydown', this.keyDownHandle)
document.addEventListener('contextmenu', this.contextmenuHandle)
},
destroyed() {
document.removeEventListener('keydown', this.keyDownHandle)
document.removeEventListener('contextmenu', this.contextmenuHandle)
},
methods:{
// 重置初始化
reset() {
this.passwdError = false
this.passwd = ''
this.$nextTick(() => {
this.$refs.lockPasswdInp.focus()
})
},
// 解鎖
unLock() {
if(this.passwd != this.lockPassword) {
return this.passwdError = true
}
this.$store.commit('common/updatelockTime', 0)
},
// 監(jiān)聽鼠標按下事件,阻止 F12 事件
keyDownHandle(event) {
if(event.keyCode == 13) {
if(this.passwdError) {
this.reset()
} else {
this.unLock()
}
}
return (event.keyCode !== 123 ) || (event.returnValue = false)
},
// 阻止鼠標右鍵事件
contextmenuHandle(event) {
return event.returnValue = false;
},
// // 退出登錄
// logout() {
// this.$store.commit('common/updatelockTime', 0)
// this.$router.replace('/login')
// /**
// * 走接口 清楚本地緩存等
// * ...
// */
// }
}
}
</script>
<style scoped>
.lock {
width: 100%;
height: 100vh;
background: #ccc;
position: fixed;
top: 0;
left: 0;
z-index: 999999;
display: flex;
justify-content: center;
align-items: center;
background-image: url('../../assets/img/lock/lock-bg.jpg');
background-repeat: no-repeat;
background-size: 100% 100%;
}
.lock > div {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
}
.lockPasswdInp {
margin-top: 8px;
}
/deep/ .el-input__inner {
background-color: transparent !important;
border: 1px solid #0076c8 !important;
color: #fff;
}
/deep/ .el-input-group__append {
background-color: rgba(6, 14, 22, .5);
border: 1px solid #0076c8;
border-left-color: transparent;
}
/deep/ .el-input-group__append:hover {
background-color: rgba(6, 14, 22, .6);
cursor: pointer;
}
.passwdError-container {
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
}
.passwdError {
width: 260px;
text-align: center;
color: #fff;
font-size: 13px;
margin: 10px 0;
}
.confirm-btn {
width: 70px;
height: 28px;
text-align: center;
line-height: 28px;
color: #fff;
font-size: 13px;
background-color: rgba(6, 14, 22, .5);
border: 1px solid #0076c8;
border-radius: 3px;
cursor: pointer;
}
.confirm-btn:hover {
background-color: rgba(6, 14, 22, .8);
}
/* .logout {
position: fixed;
bottom: 20px;
right: 20px;
height: 40px;
cursor: pointer;
} */
</style>最后App.vue中混入上面的邏輯lock.js,然后引入lock.vue組件并使用
因為我的需求是整個系統(tǒng)都要有鎖屏的功能,所以我將其寫在App.vue中
<template>
<div id="app">
<router-view />
<Lock v-if="isLock" />
</div>
</template>
<script>
import lockMixin from "@/mixins/lock";
export default {
mixins: [lockMixin]
}
</script>
<style lang="scss">
* {
margin: 0;
padding: 0;
}
#app {
width: 100vw;
height: 100vh;
overflow-y: hidden;
}
</style>最好再給大家上個鎖屏遮罩層的樣式效果吧,我這個當然是在無操作5分鐘才顯示出來的哈

以上就是Vue實現(xiàn)鎖屏功能的示例代碼的詳細內(nèi)容,更多關(guān)于Vue鎖屏的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
在 Vue-CLI 中引入 simple-mock實現(xiàn)簡易的 API Mock 接口數(shù)據(jù)模擬
本文以 Vue-CLI 為例介紹引入 simple-mock 實現(xiàn)前端開發(fā)數(shù)據(jù)模擬的步驟。感興趣的朋友跟隨小編一起看看吧2018-11-11
關(guān)于antd中select搜索框改變搜索值的問題
這篇文章主要介紹了關(guān)于antd中select搜索框改變搜索值的問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-04-04
基于vue中解決v-for使用報紅并出現(xiàn)警告的問題
下面小編就為大家分享一篇基于vue中解決v-for使用報紅并出現(xiàn)警告的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-03-03
Vue中.vue文件比main.js先執(zhí)行的問題及解決
這篇文章主要介紹了Vue中.vue文件比main.js先執(zhí)行的問題及解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-12-12
vuejs router history 配置到iis的方法
今天小編就為大家分享一篇vuejs router history 配置到iis的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09

