vue實(shí)現(xiàn)提示保存后退出的方法
假設(shè)有這樣一個(gè)需求,用戶(hù)在一個(gè)頁(yè)面內(nèi)編輯文字,但是并未點(diǎn)擊保存并且跳轉(zhuǎn)到了下一個(gè)路由。比較好的做法應(yīng)該是給出一個(gè)提示—“您編輯的內(nèi)容還未保存,是否確認(rèn)退出?”用戶(hù)如果點(diǎn)擊“確定”,那么不保存當(dāng)前內(nèi)容直接退出,用戶(hù)如果點(diǎn)擊“取消”,則取消本次路由跳轉(zhuǎn),繼續(xù)留在原來(lái)的頁(yè)面。
嘗試的錯(cuò)誤做法
一開(kāi)始的時(shí)候我是想著使用vuex結(jié)合vue router的beforeEach導(dǎo)航守衛(wèi)來(lái)實(shí)現(xiàn)。代碼如下:
首先在vuex中新增一個(gè)狀態(tài)值—introduceState
const store = new Vuex.Store({
strict: true, // process.env.NODE_ENV !== 'production', 直接修改state 拋出異常
state: {
....
introduceState: false,
....
},
getters: {
introduceState: state => state.currentMenus
},
mutations: {
// 更新introduceState的值
changeIntroduceState (state, value) {
state.introduceState = value
}
}
})
用戶(hù)在點(diǎn)擊跳轉(zhuǎn)到另一個(gè)頁(yè)面的時(shí)候會(huì)觸發(fā)生命周期函數(shù)beforeDestroy,在這個(gè)函數(shù)中我們可以檢測(cè)用戶(hù)的編輯內(nèi)容是否保存,如果尚未保存。
如果內(nèi)容尚未保存,我們就彈出一個(gè)提示框,當(dāng)用戶(hù)選擇取消的時(shí)候,就將vuex中的introduceState值更新為true。
</script>
import { mapGetters, mapActions, mapMutations } from "vuex"
export default {
data() {
return {
contentHasSave: false // 記錄用戶(hù)是否已經(jīng)保存內(nèi)容
}
},
methods: {
...mapMutations({
changeIntroduceState: changeIntroduceState
})
},
beforeDestory: function(){
if(!contentHasSave){
// 使用element的提示框
this.$confirm('您還未保存簡(jiǎn)介,確定需要提出嗎?', '提示', {
confirmButtonText: '確定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
// 選擇確定,正常跳轉(zhuǎn)
})
.catch(() => {
// 選擇取消
this.changeIntroduceState(true)
})
}
}
}
</script>
最后在router的beforeEach的導(dǎo)航守衛(wèi)里監(jiān)測(cè)from為當(dāng)前頁(yè)面的所有路由跳轉(zhuǎn)。當(dāng)state的introduceState為true的時(shí)候使用next(false)來(lái)取消本次路由跳轉(zhuǎn)
import Vue from "vue";
import VueRouter from "vue-router";
import routeConfig from "./routes";
import {sync} from "vuex-router-sync";
import store from "../store";
//加載路由中間件
Vue.use(VueRouter)
//定義路由
const router = new VueRouter({
routes: routeConfig,
//mode: 'history'
})
sync(store, router)
router.beforeEach((to, from, next) => {
// 簡(jiǎn)介也未提交,取消跳轉(zhuǎn)
if(from.fullPath === '/adwords/introduce' && store.state.introduceState === 'not-save'){
next(false)
}
})
export default router
這種做法其實(shí)是行不通的,因?yàn)閎eforeEach方法的執(zhí)行其實(shí)是在組件beforeDestory的方法之前執(zhí)行的,也就是說(shuō)beforeEach執(zhí)行的時(shí)候introduceState的值根本沒(méi)有被更新為true。
正確的做法
后來(lái)自己去翻vue router的官方文檔,找到了一個(gè)絕妙的方法,那就是組件內(nèi)的導(dǎo)航守衛(wèi)。
const Foo = {
template: `...`,
beforeRouteEnter (to, from, next) {
// 在渲染該組件的對(duì)應(yīng)路由被 confirm 前調(diào)用
// 不!能!獲取組件實(shí)例 `this`
// 因?yàn)楫?dāng)守衛(wèi)執(zhí)行前,組件實(shí)例還沒(méi)被創(chuàng)建
},
beforeRouteUpdate (to, from, next) {
// 在當(dāng)前路由改變,但是該組件被復(fù)用時(shí)調(diào)用
// 舉例來(lái)說(shuō),對(duì)于一個(gè)帶有動(dòng)態(tài)參數(shù)的路徑 /foo/:id,在 /foo/1 和 /foo/2 之間跳轉(zhuǎn)的時(shí)候,
// 由于會(huì)渲染同樣的 Foo 組件,因此組件實(shí)例會(huì)被復(fù)用。而這個(gè)鉤子就會(huì)在這個(gè)情況下被調(diào)用。
// 可以訪(fǎng)問(wèn)組件實(shí)例 `this`
},
beforeRouteLeave (to, from, next) {
// 導(dǎo)航離開(kāi)該組件的對(duì)應(yīng)路由時(shí)調(diào)用
// 可以訪(fǎng)問(wèn)組件實(shí)例 `this`
}
}
上面的描述很清楚,于是我就在組件的js代碼里加了一個(gè)beforeRouteLeave方法,然后彈出提示框,實(shí)現(xiàn)提示保存后退出的功能。
</script>
export default {
data() {
return {
contentHasSave: false // 記錄用戶(hù)是否已經(jīng)保存內(nèi)容
}
},
// 組件內(nèi)導(dǎo)航鉤子,處理未保存退出的情況
beforeRouteLeave: function(to, from , next){
if(this.buttonText === '提交'){
next(false)
this.$confirm('您還未保存簡(jiǎn)介,確定需要提出嗎?', '提示', {
confirmButtonText: '確定',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
// 選擇確定
next()
})
}
}
}
</script>
實(shí)現(xiàn)效果如下:


以上這篇vue實(shí)現(xiàn)提示保存后退出的方法就是小編分享給大家的全部?jī)?nèi)容了,希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
淺談Vue3.0新版API之composition-api入坑指南
這篇文章主要介紹了Vue3.0新版API之composition-api入坑指南,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-04-04
vue在.js文件中如何進(jìn)行路由跳轉(zhuǎn)
這篇文章主要介紹了vue在.js文件中如何進(jìn)行路由跳轉(zhuǎn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12
vue.js中methods watch和computed的區(qū)別示例詳解
methods,watch和computed都是以函數(shù)為基礎(chǔ)的,但各自卻都不同,這篇文章主要給大家介紹了關(guān)于vue.js中methods watch和computed區(qū)別的相關(guān)資料,需要的朋友可以參考下2021-08-08
vue.js todolist實(shí)現(xiàn)代碼
這篇文章主要介紹了vue.js todolist實(shí)現(xiàn)代碼,需要的朋友可以參考下2017-10-10
ant design中upload組件上傳大文件,顯示進(jìn)度條進(jìn)度的實(shí)例
這篇文章主要介紹了ant design中upload組件上傳大文件,顯示進(jìn)度條進(jìn)度的實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-10-10
element-ui 實(shí)現(xiàn)響應(yīng)式導(dǎo)航欄的示例代碼
這篇文章主要介紹了element-ui 實(shí)現(xiàn)響應(yīng)式導(dǎo)航欄的示例代碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05
解決vue keep-alive 數(shù)據(jù)更新的問(wèn)題
今天小編就為大家分享一篇解決vue keep-alive 數(shù)據(jù)更新的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-09-09
關(guān)于vue-resource報(bào)錯(cuò)450的解決方案
本篇文章主要介紹關(guān)于vue-resource報(bào)錯(cuò)450的解決方案,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-07-07
vue3(vite)設(shè)置代理封裝axios api解耦功能
這篇文章主要介紹了vue3(vite)設(shè)置代理封裝axios api解耦,本文結(jié)合示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12

