我不知道这是不是一个好习惯。但下面是我要做的事情:
我有两个延迟加载的模块:ManagementModule
和ConfigurationModule
,路由配置如下:
const routes: Routes = [
{path: '', redirectTo: 'management', pathMatch: 'full'}, {
path: 'configuration',
loadChildren: './configuration/configuration.module#ConfigurationModule',
canLoad: [UnconfiguredGuard]
},
{
path: 'management',
loadChildren: './management/management.module#ManagementModule',
canLoad: [ConfiguredGuard]
}
]
基本上,我们的想法是检查系统状态并重定向到不同的阶段,例如。如果尚未配置sys,请重定向到/configuration
,否则重定向到/management
。
在我将@ngrx/store
添加到项目之前,两个canLoad
后卫很简单:
// ConfiguredGuard:
canLoad(route) {
return this.configService.isConfigured()
.do((configured: boolean) => {
if (!configured) {
// redirect to /configuration if unconfigured
this.router.navigate(['/configuration']);
}
})
}
现在我想采用所有@ ngrx / {store,effects,router-store}库,然后将上面的防护改为:
canLoad(route) {
this.store.dispatch(new CheckInitStatus());
return this.store.select('initStatus')
.filter(s => s.checked) // check only when loaded
.map(s => s.status === 'initialized')
.catch(() => of(false));
}
当然,副作用是定义的:
@Effect()
check$: Observable<InitStatusAction> = this.actions$
.ofType<CheckInitStatus>(CHECK_INITSTATUS)
.mergeMap(() => this.fetchInitStatus())
.map(status => new InitStatusChecked({status, checked: true}))
.catch(e => of(new InitStatusCheckFailure(e)));
但导航没有发生,没有发出ROUTER_NAVIGATION
动作。我错过了什么吗?而且我想知道这是否是一个好习惯?我确实找到了canActivate
守卫用法的一些例子,但canLoad
守卫没有。请帮忙,谢谢!
答案 0 :(得分:4)
first()
之后将take(1)
或filter()
添加到运算符链中,我无法表达我现在的感觉......但我确实意识到我需要一种心态改变正确使用RxJS。
canLoad(route) {
this.store.dispatch(new CheckInitStatus());
return this.store.select('initStatus')
.filter(s => s.checked) // check only when loaded
.first() // or equivalently take(1)
.map(s => s.status === 'initialized')
.catch(() => of(false));
}
我应该更仔细地阅读canActivate
的示例,这个take(1)/first()
运算符就在那里,例如:https://toddmotto.com/preloading-ngrx-store-route-guards