如何使ngrx-store与canLoad guard

时间:2017-10-30 09:28:56

标签: angular ngrx ngrx-effects ngrx-store-4.0

我不知道这是不是一个好习惯。但下面是我要做的事情:

我有两个延迟加载的模块:ManagementModuleConfigurationModule,路由配置如下:

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守卫没有。请帮忙,谢谢!

1 个答案:

答案 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