2

我想我会创建一个非常简单的登录表单,其中包含组件绑定的用户名和密码属性,这些属性将通过以下步骤运行:

  1. 使用 fetch() 调用提交凭据
  2. THEN获取Response结果对象的JSON内容
  3. 然后检查该内容以获取服务器端检查的结果

如果凭据错误,它将更改一个组件属性,该属性会告诉 Angular 将一个类应用于用户名和密码输入以使它们暂时变为红色(使用 setTimeout 将其改回)

我遇到的问题是 Angular 无法正确应用该类,我不知道为什么。我决定创建一个简化的测试项目来缩小问题范围,最终导致包含 system-polyfills/when.js。

此代码依次经过 1、2、3,并将其设置在组件属性中并将其输出到调试控制台。Angular 会正确渲染组件属性,除非包含 system-polyfill,在这种情况下,它将停止在 1 并且永远不会将其更改为 2 或 3,即使控制台显示该属性实际上已更改:

export class App {
    private stateIndicator:string = "0";

    public showState(state:string) {
        console.log('State: ' + state);
        this.stateIndicator = state;

    }

    public showFetchProblem() {
        this.showState('1')
        fetch('http://www.google.com/robots.txt',{mode:'no-cors'}).then((response:any) => {
            this.showState('2')
            response.text().then((value) => {
                this.showState('3')
            });
        });
    }
}

我创建了以下 Plunker 来演示这种行为: http ://plnkr.co/edit/GR9U9fTctmkSGsPTySAI?p=preview

是的,显而易见的解决方案是:

  1. 不要手动包含系统填充物,或
  2. 如果需要,在 SystemJS 之前手动包含不同的 Promise polyfill

但我仍然很好奇为什么会发生这种情况,希望有人能对此有所了解(并可能有助于解决基本问题)。

编辑:这个的原标题是Why is Angular 2's template rendering misbehaving when using system-polyfills (when.js)。感谢 Thierry 和 Günter 的贡献并指出 Angular 2 对区域的使用在这里发挥了作用。对于将来遇到这种情况的任何人,这里有两篇优秀的文章,它们更详细地解释了区域,并且在您遇到这种情况时将增强您对这种情况的理解:

  1. http://blog.thoughtram.io/angular/2016/01/22/understanding-zones.html
  2. http://blog.thoughtram.io/angular/2016/02/01/zones-in-angular-2.html
4

2 回答 2

2

更新

不知道解释了,但解决方法有效。但我认为还有其他一些潜在的问题。

另请参阅https://github.com/angular/angular/issues/7792#issuecomment-211217775

原来的

我认为问题是由于fetch没有被 Angulars zone 修补造成的

To work around make the code execute inside Angulars zone manually

export class App {
    constructor(private zone:NgZone) {}

    private stateIndicator:string = "0";

    public showState(state:string) {
        console.log('State: ' + state);
        this.stateIndicator = state;

    }

    public showFetchProblem() {
        this.showState('1')
        fetch('http://www.google.com/robots.txt',{mode:'no-cors'}).then((response:any) => {
            this.showState('2')
            response.text().then((value) => {
                this.zone.run(() => this.showState('3'));
            });
        });
    }
}

Plunker 示例

于 2016-04-18T10:17:58.887 回答
2

Promise polyfill 提供了一个自定义的 Promise 实现,它似乎在 Angular2 的范围之外执行(即不在一个区域中)。

如果您在由 Angular2(使用NgZone类)管理的区域内显式执行代码,则在system-polyfill.js包含文件时它可以工作。

这是一个示例:

constructor(private zone:NgZone) {}

public showFetchProblem() {
  this.showState('1')
  this.zone.run(() => {
    fetch('http://www.google.com/robots.txt',{mode:'no-cors'}).then((response:any) => {
      this.showState('2')
      response.text().then((value) => {
        this.showState('3')
      });
    });
  });
}

请参阅相应的 plunkr:http ://plnkr.co/edit/EJgZKWVx6FURrelMEzN0?p=preview 。

于 2016-04-18T10:22:51.590 回答