Angular 2 Resolve只能运行一次

7
我正在尝试创建一个简单的解析器,以返回用户信息。但是,在我的应用程序中,我需要使用自己的类来包装http组件,以便返回可观察的Subject。第一个用户链接时,我的解析器按预期工作,但在UserProfileComponent中点击任何用户链接后,解析器不会第二次完成。
我可以看到浏览器URL随着新ID而改变,并且UserResolve中的console.log被打印出来,但什么也没有发生。
我的路由器:
import { Routes, RouterModule } from '@angular/router';
import { DashboardComponent } from './dashboard/dashboard.component';
import { UserProfileComponent } from './user-profile/user-profile.component';
import { LoginComponent } from './login/login.component';
import { AuthGuard } from './_resolve/guard.auth';
import { UserResolve } from './_resolve/user.resolve';

const appRoutes: Routes = [
  { path: '', component: DashboardComponent, canActivate: [AuthGuard] },
  { path: 'user/:id', component: UserProfileComponent, canActivate: [AuthGuard], resolve: { user: UserResolve } },
  { path: 'login', component: LoginComponent },

  // otherwise redirect to home
  { path: '**', redirectTo: '' }
];

export const Routing = RouterModule.forRoot(appRoutes);

UserResolve:

import { Injectable } from '@angular/core';
import { Resolve, ActivatedRouteSnapshot } from '@angular/router';
import { ApiService } from '../_services/api.service';
import { Observable } from 'rxjs';

import { User } from '../_models/user';

@Injectable()
export class UserResolve implements Resolve<User> {

  constructor(private apiService:ApiService) {}

  resolve(route: ActivatedRouteSnapshot):Observable<User> {
    console.log('getting user ' + route.params['id'] + ' with resolver');
    return this.apiService.getUser(route.params['id']).first();
  }
}

ApiService:

import { Injectable } from '@angular/core';
import { Http, Headers, Response, RequestOptions } from '@angular/http';
import { Observable } from 'rxjs/Observable';
import 'rxjs/Rx';

import { RefreshHttpClient } from '../_injectables/refresh-http-client';
import { User } from '../_models/user';

@Injectable()
export class ApiService {

  constructor(private rfhttp:RefreshHttpClient) { }

  getUser(userId):Observable<User> {
    return this.rfhttp.get('/api/user?userId=' + userId);
  }

}

并刷新HttpClient的“get”方法:

...
get(url):Observable<any> {
  var sub = new Subject<any>();
  var obs = this.http.get(url, {
    headers: this.createAuthorizationHeaders()
  })
  .map(res => res.json())
  .subscribe(
    data => {
      return sub.next(data);
    },
    error => {
      if (error.status == 401 || error.status == 403) {
        this.refreshAndRetry(() => {
          this.http.get(url, {
            headers: this.createAuthorizationHeaders()
          })
          .map(res => res.json())
          .subscribe(
            data => {
              return sub.next(data);
            },
            error => {
              console.error(error);
              return null;
            });
        });
      } else {
        console.error(error);
        return null;
      }
    }
  );

  return sub.asObservable();
}
...

编辑:添加UserProfileComponent:

import { Component, OnInit } from '@angular/core';
import { ActivatedRoute } from '@angular/router';

import { User } from '../_models/user';

@Component({
  selector: 'app-user-profile',
  templateUrl: './user-profile.component.html',
  styleUrls: ['./user-profile.component.scss']
})
export class UserProfileComponent implements OnInit {

  abstract:Boolean;
  user:User;

  constructor(
    private route:ActivatedRoute
  ) { }

  ngOnInit() {
    this.user = this.route.snapshot.data['user']['data'];
    console.log(this.user);

...

请注意,我的API服务器返回一个带有"data"属性的JSON结构,因此使用['data']
除了第一次导航之外,在这个组件中ngOnInit不会被调用,这让我认为Resolver Observable从未完成。

也许你可以在“RefreshHttpClient get”方法堆栈中添加额外的跟踪信息? - Sebas
另外一件事,你确定你的方法是正确的吗?在我的看法中,路由似乎不是模拟端点参数的正确方式。 - Sebas
我的方法可能不正确,我是 Angular 2 的新移民。我在 RefreshHttpClient 的 get 方法中添加了日志记录,调用已经完成(它达到了 sub.next(data))。似乎我的 Resolve 没有收到 Observable 完成的信号。 - alreit
请问您能否同时粘贴UserProfileComponent代码,包括访问已解析数据(“user”)的部分? - Vilmantas Baranauskas
也许不需要使用.next()。请参阅https://angular.io/docs/ts/latest/api/router/index/Resolve-interface.html。 - Sebas
没有 next()(或 take(1)),第一个导航也永远不会解析。 - alreit
1个回答

6
也许问题在于您访问已解决的数据的方式。
当URL更改时,您的组件不会重新初始化,因此如果您想根据URL参数在组件本身中加载数据,则应订阅路由参数更改。
因此,当您访问已解决的数据时,您应该考虑到它可能会在组件重新初始化之前发生更改。
编辑:从UserProfileComponent代码中可以看出,我的假设很可能是正确的。尝试使用订阅来读取已解决的数据。
this.route.data.subscribe(d => this.user = d['user']['data']);

这是有道理的,尽管我不明白为什么如果路由器不尝试加载一个新的UserProfileComponent实例,Resolve还会被触发。但听起来我需要采用不同的方法! - alreit
Resolve被触发是因为Angular很聪明 :-) 顺便提一下,在ngOnDestroy()中取消订阅以避免泄漏。 - Vilmantas Baranauskas
谢谢!我一直在努力让我的解析器渲染数据,你的答案帮助我找到了解决方法!谢谢!!@Vilmantas Baranauskas - Kimmiekim

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接