7

如何在 Angular 2 中重新加载相同的组件?

下面是我的代码:

import { Component, OnInit, ElementRef, Renderer } from '@angular/core';
import { Router, ActivatedRoute, Params } from '@angular/router';
import { productModel } from '../_models/index';
import { categoryListService } from '../_services/index';

@Component({
  selector: 'app-product',
  templateUrl: 'product.component.html',
  styleUrls: ['product.component.css']
})
export class productComponent implements OnInit {
  uidproduct: productModel;
  param: number;
  constructor(
    private elementRef: ElementRef,
    private route: ActivatedRoute,
    private router: Router,
    private categoryListService: categoryListService) { }

  ngOnInit() {
    this.route.params.subscribe(product => {
      console.log('logging sub product obj', product);
    });
    this.uidproduct = JSON.parse(sessionStorage.getItem('product'));
    var s = document.createElement("script");
    s.type = "text/javascript";
    s.src = "http://this/external/script/needs/to/be/loaded/each/time.js";
    this.elementRef.nativeElement.appendChild(s);
  }
  nextproduct(){ 
    let i = this.uidproduct.order;
    this.categoryListService.findNextproduct(this.uidproduct);
    this.param = ++i;
    this.router.navigate([`/product/${this.param}`]);
  }
}

nextproduct()绑定到模板中的点击事件。

uidproduct是一个具有许多属性的 JSON 对象,我正在更新 DOM{{uidproduct.classname}}

我在这样的模板中使用它:

<div id="selected-product" class="{{uidproduct.classname}}">

当我单击<button (click)="nextproduct()">它时,它将更改 DOM 中的类属性,但我需要重新加载组件以使外部脚本生效。

4

2 回答 2

22

您可以使用*ngIf重新呈现模板的内容:

@Component({
  selector: '...',
  template: `
<ng-container *ngIf="!rerender">
 template content here
</ng-container>`
})
export class MyComponent {
  rerender = false;
  constructor(private cdRef:ChangeDetectorRef){}
  doRerender() {
    this.rerender = true;
    this.cdRef.detectChanges();
    this.rerender = false;
  }
}
于 2017-02-21T13:51:52.713 回答
2

我不明白你为什么需要重新加载组件。如果您绑定到 的各个字段uidproduct,那么重新加载应该会刷新组件中显示的值。所以重新加载组件只会增加开销。

如果这里没有提到一个很好的原因为什么你认为你仍然需要这样做,那么这就是你要做的:

  1. 导航到另一个(可能为空白)组件。
  2. 直接返回。

问题是您需要等待第一个导航完成才能进行第二个导航。

在您的组件中,导入 NavigationEnd:

import { Router, NavigationEnd } from '@angular/router';

然后在您的构造函数中订阅它:

constructor(private thingService: ThisThingService, private router: Router) { 
   router.events.subscribe(event => {
    if (event instanceof NavigationEnd) {
      if (event.url === '/blank') {
        this.router.navigate(['product']); 
      }
    }
  });

请注意,我等待NavigationEnd发生,然后检查我是否正在路由到我的空白组件。如果它是空白组件的路径,我会导航回产品。如果您确实需要传递该 ID,只需将其存储在您的对象中并在此处添加即可。

nextproduct()导航到 ,而不是路由到您的产品页面blank

this.router.navigate(['blank']);

这应该很好地重新加载您的组件。

为了简单起见,我故意留下的问题是subscribe构造函数中的调用将在每次重新加载时执行。因此,作为对读者的练习,将其从构造函数中取出并为其创建一个不错的服务,或者将其移至应用程序组件的构造函数,或者移至路由模块或任何对您有意义的地方。

于 2017-02-21T13:46:46.433 回答