javascript - 如何使用 Angular 2 正确检测计算样式属性的变化?

标签 javascript css angular dom getcomputedstyle

我创建了一个 Angular 2 应用程序,其中包含一个组件,我试图相对于另一个组件的 HTML 元素定位其 HTML 元素。为了计算准确的坐标,我需要知道浏览器呈现的我自己的元素的宽度和高度。

我正在使用 window.getComputedStyle(this.elementRef.nativeElement) 来获取这些属性。我注意到,随着页面的呈现,属性不断变化,直到页面呈现完成。为了了解任何更改并调整位置,我检查了组件的 ngAfterViewChecked 方法中的值。

但是,当渲染导致新的计算样式属性时,ngAfterViewChecked 似乎没有被调用,因为我发现我的钩子(Hook)不是不再调用,即使计算的样式属性仍在变化。我假设 Angular 框架不是为了检测这种变化而设计的。

我的第一次尝试是实现 ngDoCheck Hook ,但似乎在一段时间后和计算出的样式属性具有最终值之前都没有调用这个 Hook 。我假设我还没有完全理解这个钩子(Hook)究竟应该在什么时候被调用。

我终于发现,如果我在 ngAfterViewChecked 中实现一个 setTimeout 函数,这会导致稍后再次调用同一个钩子(Hook),即使我只传递一个虚拟对象功能如:

setTimeout(() => {}, 500);

但坦率地说,我不明白为什么会这样。 有人可以向我解释一下 setTimeoutngAfterViewChecked Hook 之间的联系吗?

虽然这似乎是一个有点肮脏的解决方法:检测和处理计算样式属性更改的正确“Angular ”方法是什么?


代码摘录:

export class MyComponent implements OnInit, AfterViewChecked
{
  private cssWidth: number;
  private cssHeight: number;

  constructor(private elementRef: ElementRef, private renderer: Renderer2)
  {
  }

  ngAfterViewChecked()
  {
    console.log("ngAfterViewChecked");
    this.updateView();
  }

  public updateView()
  {
    const sizeX = parseFloat(window.getComputedStyle(this.elementRef.nativeElement).width) || 0;
    const sizeY = parseFloat(window.getComputedStyle(this.elementRef.nativeElement.height) || 0;

    if (sizeX === this.cssWidth && sizeY === this.cssHeight) {
      // no change
      return;
    }

    console.log("Change detected!");

    // TODO Does not work without this dummy timeout function (else no more changes detected) - why so??
    setTimeout(() => {}, 500);

    [.. doing the positioning here ..]
 }

最佳答案

见上图here查看生命周期图。如果您在 doCheck 中的 Hook 不起作用,则 AfterViewChecked 也不会起作用,因为 doCheck 会启动 AfterViewChecked。

每次异步进程完成时,都会发生另一轮 doCheck。那会调用 AfterViewChecked,然后调用您的 updateView 函数。 SetTimout() 是一个异步过程。我怀疑上面的代码会原样无限循环。

我相信您想要的是让您在此处显示的内容的父组件使用 ngAfterViewChecked() 并调用 updateView()。探讨了该主题 here .

让两个组件协同工作的 Angular 方法是让父组件可以在子组件之间传达更改,因为父组件是可以查看子组件中发生的事件的上下文。

我个人会使用的解决方案是 EventEmitter,在未显示的组件中,在 ngOnInit() 函数中。它会通知父级它已初始化。然后,在父组件中,有一个函数调用发生在发出的事件上,告诉子组件你在这里更新 View 。

关于javascript - 如何使用 Angular 2 正确检测计算样式属性的变化?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/43659016/

相关文章:

javascript - Quill Link 工具提示默认不可见

html - 为什么元素被背景图像推出 View ?

angular - 通过 spy /karma 模拟服务类时无法读取未定义的属性 'doc'

javascript - 窗口位置 href 更改不会重新加载

javascript - Redux 的状态是 promise

css - 定位 iframe

angular - 使用 Observable 闪烁 *ngIf 的内容

javascript - AngularJS2 : How to focus input like document. getElementById(..).focus 在 javascript 中?

javascript - 为什么我们在 jQuery 中使用 "({ })"?

html - HTML 中长图像的背景代码未产生预期结果