在main.js中,通过引入路由文件后,通过该对象使用
相当于路由的生命周期
const router =VueRouter.createRouter({ ... })
1、全局前置守卫
(1)router.beforeEach
注册一个全局前置守卫,在进入到其他路由之前调用
全局前置守卫按照创建顺序调用
let remove=router.beforeEach((to,from,next)=>{
to是去到的路由对象
from是正要离开的路由
方式一:
return false; 和next(false)效果一致
return ...; 和router.push中参数一致
return Error; 和next(error)效果一致
return undefined或true; 调用下一个导航守卫
方式二:next
next来执行去到下一个路由的操作,调用next后才能跳转到下一个路由
next(undefiend或true): 进行管道中的下一个钩子,如果全部钩子执行完了则导航的状态就是confirmed(确认的)。
next(false): 中断当前的导航,如果浏览器的URL改变了(可能是用户手动或者浏览器后退按钮),那么URL地址会重置到from路由对应的地址。
next('/') 或者 next({ path: '/' }): 跳转到一个不同的地址。且允许设置诸如 replace: true、name: 'home' 之类的选项以及任何用在 router-link 的 to prop 或 router.push 中的选项。
next(error):如果传入 next 的参数是一个 Error 实例,则导航会被终止且该错误会被传递给router.onError()注册过的回调。
})
可根据在路由元信息meta中设置数据,在路由守卫里进行对应的逻辑操作
to.matched[0].meta.xx
一个路由匹配到的所有路由记录会暴露在路由对象matched数组里
或to.meta 是一个非递归合并所有meta字段(从父字段到子字段)
remove(); 清除守卫
2、全局解析守卫
router.beforeResolve
注册一个全局守卫这和router.beforeEach 类似,区别是在导航被确认之前,同时在所有组件内守卫和异步路由组件被解析之后,解析守卫就被调用。
是获取数据或执行任何其他操作(如果用户无法进入页面时你希望避免执行的操作)的理想位置
let remove=router.beforeResolve(async to => {
if (to.meta.requiresCamera) {
return ...
}else{
return false;
}
})
remove(); 清除守卫
3、全局后置钩子
不会接受next函数也不会改变导航本身
对于分析、更改页面标题、声明页面等辅助功能以及许多其他事情都很有用
let remove=router.afterEach((to, from,failure) => {
failure 为更具体的导航错误信息
如果导航被阻止,导致用户停留在同一个页面上,返回的Promise的解析值将是 Navigation Failure
通过导入的NavigationFailureType提供的类型和isNavigationFailure判断导航错误类型,如果第二个参数不传,则只会检查这个failure是不是一个Navigation Failure
if(isNavigationFailure(failure, NavigationFailureType.aborted)){
NavigationFailureType.aborted:在导航守卫中返回false 中断了本次导航。
NavigationFailureType.cancelled: 在当前导航还没有完成之前又有了一个新的导航。比如,在等待导航守卫的过程中又调用了router.push
NavigationFailureType.duplicated:导航被阻止,因为我们已经在目标位置了
NavigationFailureType.redirected:导航被重定向了
}
...
})
remove(); 清除注册的守卫
4、路由独享守卫
放在router文件里,和component同级
从一个不同的路由进入路由时触发,不会在params、query或hash改变时触发
routes: [
{
path: '/foo',
component: Foo,
beforeEnter: (to, from, next) => {
return ...
或
next(...)
...
}
beforeEnter:[fn1,fn2] 可以传递一个数组函数
}
]
5、组件内的守卫
在组件内部和生命周期函数格式一样使用
beforeRouteEnter (to, from, next) {
(1)在渲染该组件的对应路由被 confirm 前调用
(2)不!能!获取组件实例 `this`
(3)因为当守卫执行前,组件实例还没被创建
(4)虽然没有this,但相比于其他组件内守卫,可以通过next的回调函数获取到组件实例
return ...
或
next((vm)=>...)
},
beforeRouteUpdate (to, from) {
(1)在当前路由改变,但是该组件被复用时调用
(2)举例来说,对于一个带有动态参数的路径 /foo/:id,在 /foo/1 和 /foo/2 之间跳转的时候,
(3)由于会渲染同样的 Foo 组件,因此组件实例会被复用。而这个钩子就会在这个情况下被调用。
(4)可以访问组件实例 `this`
},
beforeRouteLeave (to, from) {
(1)导航离开该组件的对应路由时调用
(2)可以访问组件实例 `this`
return false 阻止离开
}
6、完整的导航解析流程
导航被触发。
在失活的组件里调用离开守卫。
调用全局的 beforeEach 守卫。
在重用的组件里调用 beforeRouteUpdate 守卫 (2.2+)。
在路由配置里调用 beforeEnter。
解析异步路由组件。
在被激活的组件里调用 beforeRouteEnter。
调用全局的 beforeResolve 守卫 (2.5+)。
导航被确认。
调用全局的 afterEach 钩子。
触发 DOM 更新。
用创建好的实例调用 beforeRouteEnter 守卫中传给 next 的回调函数。
7、数据获取
导航完成之后获取:先完成导航,然后在接下来的组件生命周期钩子中获取数据。
导航完成之前获取:导航完成前,在路由进入的守卫(beforeRouteEnter、beforeRouteUpdate)中获取数据,在数据获取成功后执行导航。
在为后面的视图获取数据时,用户会停留在当前的界面,因此建议在数据获取期间,显示一些进度条或者别的指示。如果数据获取失败,同样有必要展示一些全局的错误提醒
代码示例:
路由守卫判断是否登录:
// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import App from './App'
import router from './router'
import http from './http/http'
import { Toast } from 'vant';
Vue.prototype.$http=http;
Vue.prototype.$toast=Toast;
Vue.config.productionTip = false
/* eslint-disable no-new */
new Vue({
el: '#app',
router,
components: { App },
template: '<App/>'
})
router.beforeEach((to, from, next) => {
if(localStorage.getItem('token'))
{
next();
}else{
next('/login')
}
})