Angular路由守卫
引言在企业应用中权限、复杂页多路由数据处理、进入与离开路由数据处理这些是非常常见的需求。 当希望用户离开一个正常编辑页时,要中断并提醒用户是否真的要离开时,如果在Angular中应该怎么做呢? 其实Angular路由守卫属性可以帮我们做更多有意义的事,而且非常简单。 什么是路由守卫?Angular 的 canActivate 控制是否允许进入路由。 canActivateChild 等同 canDeactivate 控制是否允许离开路由。 canLoad 控制是否允许延迟加载整个模块。 例如: { path: 'logics',loadChildren: './logics/logics.module#LogicsModule',canLoad: [ AuthGuard ] } 这四个属性非常好理解,而且作用各自不同。然后当进入与离开能够有效控制权时,对于前面我提到的若干问题,就可以非常好的处理。 如何创建?四个属性虽然名称不同,但其基本的使用方式非常相近。四种不同守卫方式有者四个不同的接口与之相对应。
@Injectable() export class CanAdminProvide implements CanActivate { constructor(private userSrv: UserService,private msg: NzMessageService) {} canActivate( route: ActivatedRouteSnapshot,state: RouterStateSnapshot): boolean | Observable<boolean> | Promise<boolean> { return new Observable((observer) => { // 拥有 `admin` 角色 if (this.userSrv.hasRole('admin')) { observer.next(true); observer.complete(); return; } this.msg.error('授权不足'); observer.next(false); observer.complete(); }); } } 每种接口要都需要相应的实现某个方法,就上而论,继承 四种类型守卫接口都返回一个布尔类型值,其实从这四种参数的名称 最后,把它运用到相应的路由上即可,例如: { path: 'admin',component: GuardAdminComponent,canActivate: [ CanAdminProvide ] } 当然,别忘记注册 一些实践离开时提醒四种守卫只有一种离开类型 @Injectable() export class CanLeaveProvide implements CanDeactivate<GuardComponent> { constructor (private confirmSrv: NzModalService) {} canDeactivate( component: GuardComponent,currentRoute: ActivatedRouteSnapshot,currentState: RouterStateSnapshot,nextState?: RouterStateSnapshot): boolean | Observable<boolean> | Promise<boolean> { return new Observable((observer) => { this.confirmSrv.confirm({ title: '确认要离开吗?',content: '你已经填写了部分表单离开会放弃已经填写的内容。',okText: '离开',cancelText: '取消',onOk: () => { observer.next(true); observer.complete(); },onCancel: () => { observer.next(false); observer.complete(); } }); }); } } 这里返回的是一个 // 允许 observer.next(true); // 或拒绝 // observer.next(false); observer.complete(); 来处理 上面使用的 ng-zorro-antd 的确认对话框来提醒用户是否需要离开,若选择【离开】则跳转至目标路由,反之保留当前路由状态。
角色受限这是再正常不过的功能,若用户进入一个未授权的路由时,甚至是某个迟延加载模块下所有路由;若用户无权限时,如何提醒用户。 此时 @Injectable() export class CanAuthProvide implements CanActivate,CanLoad { constructor(private userSrv: UserService,private msg: NzMessageService) {} check(): Observable<boolean> { return new Observable((observer) => { if (this.userSrv.isLogin) { observer.next(true); observer.complete(); return; } this.msg.error('权限不足'); observer.next(false); observer.complete(); }); } canActivate( route: ActivatedRouteSnapshot,state: RouterStateSnapshot): boolean | Observable<boolean> | Promise<boolean> { return this.check(); } canLoad(route: Route): boolean | Observable<boolean> | Promise<boolean> { return this.check(); } } 因此,一个类中具有两种不同守卫的能力,更对于代码组织也更优雅。同样,需要运用到相应的路由当中。 { path: 'auth',component: GuardAuthComponent,canActivate: [ CanAuthProvide ] },{ path: 'admin',loadChildren: './admin/admin.module#AdminModule',canLoad: [ CanAuthProvide ] } 此后,若一个普通员工账号要想进入(哪怕浏览器地址栏录入)未授权的路由
总结路由守卫对于权限控制非常便利,当然其粒度当然只能在页面层级。倘若需要对按钮粒度也只能利用指令的方式,而二者的结合可以极大的改善权限控制埋点的代码量。 Happy coding! (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |
- angular js点击tab中的li标签加载相应div区域
- sass – 如何创建_custom.scss以覆盖Bootstrap 4
- 仿照Bootstrap的input 修改 upload上传图片的样式
- Webservice学习笔记六,SOAP, REST and XML-RPC报
- angularjs – 如何使用DOM操作正确单元测试指令?
- scala – Play 2.0表单 – 字段“验证”方法不是
- 从shell命令管道输出到python脚本
- bash – 从文件夹外部执行Node.js文件中断文件路
- ‘=?’的含义是什么?在angularJS指令隔离范围声
- 是否有可能在Angular中同时拥有css和scss