基于Vue3+TypeScript實(shí)現(xiàn)防止支付密碼被瀏覽器保存功能
背景介紹
最近做 PC 端支付功能,輸入支付密碼的時(shí)候,Chrome 很“體貼”地彈了個(gè):
"是否保存密碼?"
說(shuō)實(shí)話,這功能擱登錄場(chǎng)景,確實(shí)挺方便。 但支付密碼?說(shuō)難聽(tīng)點(diǎn),萬(wàn)一用戶真保存了,自己都忘了,或者電腦不是自己的,密碼直接暴露。 于是,產(chǎn)品一句話:
“瀏覽器這個(gè)彈窗,給我去掉。”
好家伙,去掉瀏覽器行為,這不是在為難我初級(jí)開(kāi)發(fā)。

方案思路
- 項(xiàng)目基于 Vue3 + TypeScript + Element Plus,常規(guī)做法是用
<el-input type="password" />來(lái)實(shí)現(xiàn)密碼輸入。 但只要type="password",瀏覽器一定會(huì)觸發(fā)保存密碼提示。
所以我的思路是:
把 type 改成 text,規(guī)避瀏覽器的默認(rèn)密碼保存行為
輸入框里不直接展示真實(shí)密碼,輸入什么都用 ● 替換
真實(shí)密碼值自己內(nèi)部單獨(dú)存一份;
實(shí)現(xiàn)步驟
基礎(chǔ)版本
<template>
<el-input
v-model="pwdSymbol"
placeholder="請(qǐng)輸入"
type="text"
@input="handleInput"
@blur="handleBlur"
/>
</template>
<script setup lang="ts">
const pwdSymbol = ref("");
const pwd = ref("");
const handleInput = (value: string) => {
const newValue = value.replace(/●/g, "")
const lastLength = pwdSymbol.value.length;
if (newValue) {
pwd.value += newValue;
} else {
if (!lastLength) {
pwd.value = "";
} else {
const arr = pwd.value.split("");
arr.splice(-1, 1);
pwd.value = arr.join("");
}
}
let data = "";
for (let i = 0; i < value.length; i++) {
data += "●";
}
pwdSymbol.value = data;
};
const handleBlur = () => {
console.log(pwd.value);
};
</script>
pwdSymbol用來(lái)展示●,所以我們還需要聲明一個(gè)變量,用來(lái)存儲(chǔ)用戶真正輸入的值pwdhandleBlur主要是為了查看我們輸入的密碼是否正確
遇到的問(wèn)題
- 只能支持「從最后一位輸入」
- 中間位置輸入、刪除,真實(shí)密碼全亂套
- 撤回(Ctrl+Z)直接把密碼還原,真實(shí)值跟展示完全對(duì)不上
進(jìn)階方案:精準(zhǔn)控制光標(biāo) & 禁止撤回
- 通過(guò)
ref獲取原生 input DOM,拿到光標(biāo)位置,精準(zhǔn)處理字符串 - 撤回操作干脆禁了,省事
// 設(shè)置input的ref
const elInputRef = ref();
const handleInput = (value: string) => {
const el = elInputRef.value?.input;
const cursorPos = el.selectionStart;
const newValue = value.replace(/●/g, "");
const lastLength = pwdSymbol.value.length;
if (newValue) {
if (cursorPos != lastLength) {
// 說(shuō)明不是往后添加,而是在中間插入
const arr = pwd.value.split("");
arr.splice(cursorPos - 1, 0, newValue);
pwd.value = arr.join("");
} else {
if (value.length === 1) {
// 防止選擇內(nèi)容直接替換
pwd.value = newValue;
} else {
pwd.value += newValue;
}
}
} else {
if (!lastLength) {
pwd.value = "";
} else {
// 需要辨別是刪除還是撤回
const arr = pwd.value.split("");
arr.splice(cursorPos, 1);
pwd.value = arr.join("");
}
}
let data = "";
for (let i = 0; i < value.length; i++) {
data += "●";
}
pwdSymbol.value = data;
};
const handleKeydown = (e: KeyboardEvent) => {
// 判斷是否按下撤回快捷鍵(Windows: Ctrl+Z,Mac: Meta+Z)
if ((e.ctrlKey || e.metaKey) && e.key.toLowerCase() === "z") {
e.preventDefault(); // 阻止默認(rèn)撤回行為
console.log("用戶進(jìn)行了撤回操作");
}
};
最終封裝
嫌麻煩可以直接封裝成組件,業(yè)務(wù)場(chǎng)景復(fù)用更方便,這里就不貼重復(fù)代碼了,思路一樣。
最后
這需求可能不算技術(shù)含量特別高,但實(shí)際業(yè)務(wù)里,安全和用戶體驗(yàn)都挺重要。
如果你們項(xiàng)目也有類似場(chǎng)景,希望這篇能省你點(diǎn)時(shí)間。
到此這篇關(guān)于基于Vue3+TypeScript實(shí)現(xiàn)防止支付密碼被瀏覽器保存功能的文章就介紹到這了,更多相關(guān)Vue3防止支付密碼被保存內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vue打印的對(duì)象在瀏覽器中顯示的問(wèn)題分析
因?yàn)樵趘ue中對(duì)象都是用了代理重寫了get,由于get重寫也就導(dǎo)致了瀏覽器不能直接獲取到具體的值,因此才會(huì)在打印的時(shí)候?yàn)?..,手動(dòng)點(diǎn)擊展開(kāi)才顯示具體的值,這篇文章主要介紹了為什么vue打印的對(duì)象在瀏覽器中顯示,需要的朋友可以參考下2024-04-04
Vue3發(fā)送post請(qǐng)求出現(xiàn)400?Bad?Request報(bào)錯(cuò)的解決辦法
這篇文章主要給大家介紹了關(guān)于Vue3發(fā)送post請(qǐng)求出現(xiàn)400?Bad?Request報(bào)錯(cuò)的解決辦法,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2023-02-02
vue將data恢復(fù)到初始狀態(tài) && 重新渲染組件實(shí)例
這篇文章主要介紹了vue將data恢復(fù)到初始狀態(tài) && 重新渲染組件實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09
vue分類篩選filter方法簡(jiǎn)單實(shí)例
這篇文章主要介紹了vue分類篩選filter方法的簡(jiǎn)單實(shí)例,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03
Vue vxe-table使用問(wèn)題收錄小結(jié)
這篇文章主要為大家介紹了Vue vxe-table使用問(wèn)題收錄小結(jié),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-09-09
vue3.0中使用element UI表單遍歷校驗(yàn)問(wèn)題解決
本文主要介紹了vue3.0中使用element UI表單遍歷校驗(yàn)問(wèn)題解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04

