2016-05-06 06:17:34 -07:00
block includes
include ../_util-fns
2015-11-21 11:23:40 -08:00
2016-01-23 18:21:09 +00:00
- var top="vertical-align:top"
2015-11-21 11:23:40 -08:00
:marked
# Component Lifecycle
2016-05-23 22:27:46 +01:00
# 组件生命周期
2015-11-21 11:23:40 -08:00
A Component has a lifecycle managed by Angular itself. Angular creates it, renders it, creates and renders its children,
checks it when its data-bound properties change, and destroys it before removing it from the DOM.
2016-05-23 22:27:46 +01:00
一个组件有一个被Angular本身管理的生命周期。Angular创建它, 渲染它, 创建和渲染它的子级, 在它的数据绑定属性变化时检查它, 并在它被从DOM移除前销毁它。
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
Angular offers **component lifecycle hooks**
2015-11-21 11:23:40 -08:00
that give us visibility into these key moments and the ability to act when they occur.
2016-05-23 22:27:46 +01:00
Angular提供**组件生命周期钩子**,提高了这些关键时刻的能见度,并给了我们在它们发生的时候采取行动的能力。
2016-01-13 09:41:16 +00:00
2015-11-21 11:23:40 -08:00
We cover these hooks in this chapter and demonstrate how they work in code.
2016-01-23 18:21:09 +00:00
2016-05-23 22:27:46 +01:00
我们在本章节覆盖这些钩子,演示它们在代码中是怎么工作的。
2016-01-23 18:21:09 +00:00
* [The lifecycle hooks](#hooks-overview)
2016-05-23 22:27:46 +01:00
* [生命周期钩子](#hooks-overview)
2016-01-23 18:21:09 +00:00
* [The hook-call sequence](#hook-sequence)
2016-05-23 22:27:46 +01:00
* [钩子调用顺序](#hook-sequence)
2016-01-23 18:21:09 +00:00
* [Other Angular lifecycle hooks](#other-lifecycles)
2016-05-23 22:27:46 +01:00
* [其他Angular生命周期钩子](#other-lifecycles)
2016-01-23 18:21:09 +00:00
* [The lifecycle sample](#the-sample)
2016-05-23 22:27:46 +01:00
* [生命周期例子](#the-sample)
2016-01-23 18:21:09 +00:00
* [All](#peek-a-boo)
2016-05-23 22:27:46 +01:00
* [全部](#peek-a-boo)
* [Spying OnInit and OnDestroy](#spy)
* [窥测OnInit和OnDestroy](#spy)
2016-01-23 18:21:09 +00:00
* [OnChanges](#onchanges)
2016-05-23 22:27:46 +01:00
* [OnChanges](#onchanges)
* [DoCheck](#docheck)
2016-01-23 18:21:09 +00:00
* [DoCheck](#docheck)
2016-05-23 22:27:46 +01:00
2016-01-23 18:21:09 +00:00
* [AfterViewInit and AfterViewChecked](#afterview)
2016-05-23 22:27:46 +01:00
* [AfterContentInit和AfterContentChecked](#aftercontent)
2016-01-13 09:41:16 +00:00
2016-05-20 16:18:58 -07:00
p Try the #[+liveExampleLink2()].
2015-11-21 11:23:40 -08:00
2016-05-23 22:27:46 +01:00
p 试一试#[+liveExampleLink2()]。
2016-05-06 06:17:34 -07:00
a#hooks-overview
2015-11-21 11:23:40 -08:00
.l-main-section
:marked
2016-01-23 18:21:09 +00:00
## Component lifecycle Hooks
2016-05-23 22:27:46 +01:00
##组件生命周期钩子
2015-11-21 11:23:40 -08:00
Directive and component instances have a lifecycle
as Angular creates, updates, and destroys them.
2016-05-23 22:27:46 +01:00
2016-05-24 22:29:33 +01:00
在Angular新建、更新和销毁指令和组件的实例时, 它们有一个生命周期。
2016-01-13 09:41:16 +00:00
2015-11-21 11:23:40 -08:00
Developers can tap into key moments in that lifecycle by implementing
2016-05-06 06:17:34 -07:00
one or more of the *Lifecycle Hook* interfaces in the Angular `core` library.
2016-01-23 18:21:09 +00:00
2016-05-24 22:29:33 +01:00
开发者可以介入这些生命周期的关键时刻, 通过实现一个或多个Angular `Core`库里的*生命周期钩子*接口。
2016-01-23 18:21:09 +00:00
Each interface has a single hook method whose name is the interface name prefixed with `ng`.
For example, the `OnInit` interface has a hook method named `ngOnInit`.
We might implement it in a component class like this:
2016-05-24 22:29:33 +01:00
每个接口都有单独一个钩子方法,它们的名字是对应接口名字加上前缀`ng`。比如,`OnInit`接口有一个钩子方法叫做`ngOnInit`。
我们可能在一个组件类像这样实现它:
2016-01-23 18:21:09 +00:00
+makeExample('lifecycle-hooks/ts/app/peek-a-boo.component.ts', 'ngOnInit', 'peek-a-boo.component.ts (excerpt)')(format='.')
:marked
No directive or component will implement all of them and some of the hooks only make sense for components.
Angular only calls a directive/component hook method *if it is defined*.
2016-05-24 22:29:33 +01:00
没有指令或者组件实现所有的接口, 有些钩子是针对组件的。Angular只有在指令/组件的钩子方法已经被定义时调用它们。
2016-05-06 06:17:34 -07:00
block optional-interfaces
.l-sub-section
:marked
### Interface optional?
2016-05-24 22:29:33 +01:00
### 接口是可选的?
2016-05-06 06:17:34 -07:00
The interfaces are optional for JavaScript and Typescript developers from a purely technical perspective.
The JavaScript language doesn't have interfaces.
Angular can't see TypeScript interfaces at runtime because they disappear from the transpiled JavaScript.
2016-05-24 22:29:33 +01:00
从纯粹的技术性方面来看, 接口对JavaScript和TypeScript开发者是可选的。JavaScript语言本身没有接口。
Angular在运行是看不到TypeScript接口, 因为它们在编译为JavaScript的时候消失了。
2015-11-21 11:23:40 -08:00
2016-05-06 06:17:34 -07:00
Fortunately, they aren't necessary.
We don't have to add the lifecycle hook interfaces to our directives and components to benefit from the hooks themselves.
2016-05-24 22:29:33 +01:00
幸运的事,它们不是必须的。我们不需要在指令和组件上面添加生命周期钩子接口来获得钩子带来的好处。
2015-11-21 11:23:40 -08:00
2016-05-06 06:17:34 -07:00
Angular instead inspects our directive and component classes and calls the hook methods *if they are defined*.
Angular will find and call methods like `ngOnInit()`, with or without the interfaces.
2016-05-24 22:29:33 +01:00
取而代之, Angular检测指令和组件类, 并在钩子方法已经被定义时调用它们。
Agnular会找到并调用像`ngOnInit()`的钩子方法,有没有接口并不影响。
2016-01-13 09:41:16 +00:00
2016-05-06 06:17:34 -07:00
Nonetheless, we strongly recommend adding interfaces to TypeScript directive classes
2016-05-24 22:29:33 +01:00
in order to benefit from strong typing and editor tooling.
虽然如此, 我们强烈推荐在TypeScript指令类里添加接口, 来获得强类型和编辑工具辅助等好处。
2016-01-23 18:21:09 +00:00
2016-04-06 11:45:12 +03:00
:marked
2016-01-23 18:21:09 +00:00
Here are the component lifecycle hook methods:
2016-05-24 22:29:33 +01:00
这里是组件生命周期钩子方法列表:
2016-01-23 18:21:09 +00:00
### Directives and Components
2016-05-24 22:29:33 +01:00
### 指令和组件
2016-01-23 18:21:09 +00:00
table(width="100%")
col(width="20%")
col(width="80%")
tr
th Hook
th Purpose
tr(style=top)
td ngOnInit
td
:marked
Initialize the directive/component after Angular initializes the data-bound input properties.
2016-05-24 22:29:33 +01:00
在Angular初始化数据绑定输入属性后, 用来初始化指令或组件。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngOnChanges
td
:marked
Respond after Angular sets a data-bound input property.
The method receives a `changes` object of current and previous values.
2016-05-24 22:29:33 +01:00
在Angular设置一个数据绑定输入属性后作出反应。这个方法接受一个包含了当前和之前的值的`changes`对象。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngDoCheck
td
:marked
Detect and act upon changes that Angular can or won't
detect on its own. Called every change detection run.
2016-05-24 22:29:33 +01:00
用来监测Angular自己可以或者不会检测的变化, 并作出相应行动。贼每个变化监测运行时被调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngOnDestroy
td
:marked
Cleanup just before Angular destroys the directive/component.
Unsubscribe observables and detach event handlers to avoid memory leaks.
2016-05-24 22:29:33 +01:00
在Angular销毁指令或组件之前做一些清洁工作, 不如退订监测和分离时间处理器, 以防内存泄露。
2016-01-23 18:21:09 +00:00
:marked
### Components only
2016-05-24 22:29:33 +01:00
### 只适用于组件
2016-01-23 18:21:09 +00:00
table(width="100%")
col(width="20%")
col(width="80%")
tr
th Hook
th Purpose
tr(style=top)
td ngAfterContentInit
td
:marked
After Angular projects external content into its view.
2016-05-24 22:29:33 +01:00
在Angular投射外来内容到自己的视图后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterContentChecked
td
:marked
2016-05-24 22:29:33 +01:00
After Angular checks the bindings of the external content that it projected into its view.#
在Angular检查它投射到自己视图的外部内容的绑定后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterViewInit
td
:marked
After Angular creates the component's view(s).
2016-05-24 22:29:33 +01:00
在Angular新建组件的视图后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterViewChecked
td
:marked
After Angular checks the bindings of the component's view(s).
2016-05-24 22:29:33 +01:00
在Angular检查组件视图的绑定后调用。
2016-01-23 18:21:09 +00:00
:marked
Angular does not call the hook methods in this order.
2016-05-24 22:29:33 +01:00
Angular不会按照上面的顺序调用这些钩子方法。
2016-01-23 18:21:09 +00:00
a(id="hook-sequence")
.l-main-section
:marked
## Lifecycle sequence
2016-05-24 22:29:33 +01:00
## 生命周期顺序
2016-01-23 18:21:09 +00:00
*After* Angular creates a component/directive by `new`-ing its constructor,
2016-05-24 22:29:33 +01:00
it calls the lifecycle hook methods in the following sequence at specific moments:
Angular使用构造函数新建一个组件或指令后, 它在特定的时刻按照下面的顺序调用生命周期钩子方法@
2016-01-23 18:21:09 +00:00
table(width="100%")
col(width="20%")
col(width="80%")
tr
th Hook
th Timing
tr(style=top)
td ngOnChanges
td
:marked
before `ngOnInit` and when a data-bound input property value changes.
2016-05-24 22:29:33 +01:00
当数据绑定输入属性的值变化时,在`ngOnInit`之前调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngOnInit
td
:marked
after the first `ngOnChanges`.
2016-05-24 22:29:33 +01:00
在第一个`ngOnChanges`时候调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngDoCheck
td
:marked
during every Angular change detection cycle.
2016-05-24 22:29:33 +01:00
在每个Angular变化监测周期中调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterContentInit
td
:marked
after projecting content into the component.
2016-05-24 22:29:33 +01:00
投射内容到组件后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterContentChecked
td
:marked
after every check of projected component content.
2016-05-24 22:29:33 +01:00
每次检查被投射的组件内容之后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterViewInit
td
:marked
after initializing the component's views and child views.
2016-05-24 22:29:33 +01:00
初始化组件视图和子视图之后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngAfterViewChecked
td
:marked
after every check of the component's views and child views.
2016-05-24 22:29:33 +01:00
每次检查组件视图和子视图之后调用。
2016-01-23 18:21:09 +00:00
tr(style=top)
td ngOnDestroy
td
:marked
just before Angular destroys the directive/component.
2016-05-24 22:29:33 +01:00
在Angular销毁指令和组件之前调用。
2016-01-23 18:21:09 +00:00
a(id="other-lifecycles")
.l-main-section
:marked
## Other lifecycle hooks
2016-05-24 22:29:33 +01:00
## 其他生命周期钩子
2016-05-06 06:17:34 -07:00
Other Angular sub-systems may have their own lifecycle hooks apart from the component hooks we've listed.
2016-01-23 18:21:09 +00:00
The router, for instance, also has it's own [router lifecycle hooks](router.html#router-lifecycle-hooks)
that allow us to tap into specific moments in route navigation.
2016-05-24 22:29:33 +01:00
其他Angular子系统除了有我们已经列出来的组件钩子外, 可能有它们自己的生命周期钩子。比如路由器, 它有自己的[路由器生命周期钩子](router.html#router-lifecycle-hooks),
为我们提供了插入到特定的路由浏览时刻的能力。
2016-01-23 18:21:09 +00:00
A parallel can be drawn between `ngOnInit` and `routerOnActivate`.
Both are prefixed so as to avoid collision, and both run right when a component is 'booting' up.
2016-05-24 22:29:33 +01:00
一对平衡线可以在`ngOnInit`和`routerOnActivate`之间被划出来。两种都有前缀用来防止冲突,两者都是在一个组件刚刚“启动”时被调用。
2016-01-23 18:21:09 +00:00
3rd party libraries might implement their hooks as well in order to give us, the developers, more
control over how these libraries are used.
2016-05-24 22:29:33 +01:00
第三方库也可能实现它们自己的钩子,用来给我们开发者提供更多如何使用它们的控制。
2016-01-23 18:21:09 +00:00
2016-05-06 06:17:34 -07:00
a#the-sample
2016-01-23 18:21:09 +00:00
.l-main-section
2016-05-06 06:17:34 -07:00
h2 Lifecycle exercises
2016-05-24 22:29:33 +01:00
h2 生命周期练习
2016-05-06 06:17:34 -07:00
p.
2016-05-20 16:18:58 -07:00
The #[+liveExampleLink()]
2016-01-23 18:21:09 +00:00
demonstrates the lifecycle hooks in action through a series of exercises
presented as components under the control of the root `AppComponent`.
2016-05-24 22:29:33 +01:00
#[+liveExampleLink()] 通过一些在根组件`AppComponent`控制下的组件的练习,演示了生命周期钩子的运作。
2016-05-06 06:17:34 -07:00
:marked
2016-01-23 18:21:09 +00:00
They follow a common pattern: a *parent* component serves as a test rig for
a *child* component that illustrates one or more of the lifecycle hook methods.
2016-05-24 22:29:33 +01:00
它们遵循了一个常用的模式:一个*子级*组件演示一个或多个生命周期钩子方法,一个*父级*组件被当作该*子级*组件的测试台。
2016-01-23 18:21:09 +00:00
Here's a brief description of each exercise:
2016-05-24 22:29:33 +01:00
下面是每个练习简短的描述:
2016-01-23 18:21:09 +00:00
table(width="100%")
col(width="20%")
col(width="80%")
tr
2016-05-24 22:29:33 +01:00
th Component 组件
th Description 描述
2016-01-23 18:21:09 +00:00
tr(style=top)
td <a href="#peek-a-boo">Peek-a-boo</a>
td
:marked
Demonstrates every lifecycle hook.
Each hook method writes to the on-screen log.
2016-05-24 22:29:33 +01:00
展示每个生命周期钩子,每个钩子方法在屏幕上显示日志。
2016-01-23 18:21:09 +00:00
tr(style=top)
td <a href="#spy">Spy</a>
td
:marked
Directives have lifecycle hooks too.
We create a `SpyDirective` that logs when the element it spies upon is
created or destroyed using the `ngOnInit` and `ngOnDestroy` hooks.
2016-05-24 22:29:33 +01:00
指令也有生命周期钩子。我们新建一个`SpyDirective`,利用`ngOnInit`和`ngOnDestroy`钩子,在每个被监视的元素被新建或销毁时输出日志。
2016-01-23 18:21:09 +00:00
We apply the `SpyDirective` to a `<div>` in an `ngFor` *hero* repeater
managed by the parent `SpyComponent`.
2016-05-24 22:29:33 +01:00
我们把`SpyDirective`应用到一个在父级组件里的`ngFor`*英雄*重复器的`<div>`里面。
2016-01-23 18:21:09 +00:00
tr(style=top)
td <a href="#onchanges">OnChanges</a>
td
:marked
See how Angular calls the `ngOnChanges` hook with a `changes` object
every time one of the component input properties changes.
Shows how to interpret the `changes` object.
2016-05-24 22:29:33 +01:00
2016-01-23 18:21:09 +00:00
tr(style=top)
td <a href="#docheck">DoCheck</a>
td
:marked
2016-05-06 06:17:34 -07:00
Implements an `ngDoCheck` method with custom change detection.
2016-01-13 15:00:43 -07:00
See how often Angular calls this hook and watch it post changes to a log.
2016-01-23 18:21:09 +00:00
tr(style=top)
td <a href="#afterview">AfterView</a>
td
:marked
Shows what Angular means by a *view*.
Demonstrates the `ngAfterViewInit` and `ngAfterViewChecked` hooks.
tr(style=top)
td <a href="#aftercontent">AfterContent</a>
td
:marked
Shows how to project external content into a component and
how to distinguish projected content from a component's view children.
Demonstrates the `ngAfterContentInit` and `ngAfterContentChecked` hooks.
tr(style=top)
td Counter
td
:marked
Demonstrates a combination of a component and a directive
each with its own hooks.
In this example, a `CounterComponent` logs a change (via `ngOnChanges`)
every time the parent component increments its input counter property.
Meanwhile, we apply the `SpyDirective` from the previous example
to the `CounterComponent` log and watch log entries be created and destroyed.
:marked
We discuss the exercises in further detail over this chapter as we learn more about the lifecycle hooks.
a(id="peek-a-boo")
.l-main-section
:marked
## Peek-a-boo: all hooks
The `PeekABooComponent` demonstrates all of the hooks in one component.
In real life, we'd rarely if ever implement all of the interfaces like this.
We do so in peek-a-boo in order to watch Angular call the hooks in the expected order.
In this snapshot, we clicked the *Create...* button and then the *Destroy...* button.
figure.image-display
img(src="/resources/images/devguide/lifecycle-hooks/peek-a-boo.png" alt="Peek-a-boo")
:marked
The sequence of log messages follows the prescribed hook calling order:
`OnChanges`, `OnInit`, `DoCheck` (3x), `AfterContentInit`, `AfterContentChecked` (3x),
`AfterViewInit`, `AfterViewChecked` (3x), and `OnDestroy`.
.l-sub-section
:marked
The constructor isn't an Angular hook *per se*.
We log in it to confirm that input properties (the `name` property in this case) have no assigned values at construction.
:marked
Had we clicked the *Update Hero* button, we'd have seen another `OnChanges` and two more triplets of
2016-05-06 06:17:34 -07:00
`DoCheck`, `AfterContentChecked` and `AfterViewChecked`.
2016-01-23 18:21:09 +00:00
Clearly these three hooks fire a *lot* and we must keep the logic we put in these hooks
as lean as possible!
Our next examples focus on hook details.
.a(id="spy")
.l-main-section
:marked
## Spying *OnInit* and *OnDestroy*
We're going undercover for these two hooks. We want to know when an element is initialized or destroyed,
but we don't want *it* to know we're watching.
This is the perfect infiltration job for a directive.
Our heroes will never know it's there.
.l-sub-section
:marked
Kidding aside, we're emphasizing two key points:
1. Angular calls hook methods for *directives* as well as components.
2. A spy directive can gives us insight into a DOM object that we cannot change directly.
Obviously we can't change the implementation of a native `div`.
We can't modify a third party component either.
But we can watch both with a directive.
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
:marked
Our sneaky spy directive is simple, consisting almost entirely of `ngOnInit` and `ngOnDestroy` hooks
that log messages to the parent via an injected `LoggerService`.
+makeExample('lifecycle-hooks/ts/app/spy.directive.ts', 'spy-directive')(format=".")
:marked
We can apply the spy to any native or component element and it'll be initialized and destroyed
at the same time as that element.
Here we attach it to the repeated hero `<div>`
2016-05-06 06:17:34 -07:00
+makeExample('lifecycle-hooks/ts/app/spy.component.html', 'template')(format=".")
2016-01-23 18:21:09 +00:00
:marked
Each spy's birth and death marks the birth and death of the attached hero `<div>`
with an entry in the *Hook Log* as we see here:
figure.image-display
img(src='/resources/images/devguide/lifecycle-hooks/spy-directive.gif' alt="Spy Directive")
:marked
2016-04-13 18:42:43 -04:00
Adding a hero results in a new hero `<div>`. The spy's `ngOnInit` logs that event.
2016-01-23 18:21:09 +00:00
We see a new entry for each hero.
The *Reset* button clears the `heroes` list.
Angular removes all hero divs from the DOM and destroys their spy directives at the same time.
The spy's `ngOnDestroy` method reports its last moments.
The `ngOnInit` and `ngOnDestroy` methods have more vital roles to play in real applications.
Let's see why we need them.
### OnInit
We turn to `ngOnInit` for two main reasons:
1. To perform complex initializations shortly after construction
1. To set up the component after Angular sets the input properties
An `ngOnInit` often fetches data for the component as shown in the
[Tutorial](../tutorial/toh-pt4.html#oninit) and [HTTP](server-communication.html#oninit) chapters.
We don't fetch data in a component constructor. Why?
Because experienced developers agree that components should be cheap and safe to construct.
We shouldn't worry that a new component will try to contact a remote server when
created under test or before we decide to display it.
Constructors should do no more than set the initial local variables to simple values.
When a component must start working _soon_ after creation,
2016-04-13 18:42:43 -04:00
we can count on Angular to call the `ngOnInit` method to jumpstart it.
2016-01-23 18:21:09 +00:00
That's where the heavy initialization logic belongs.
Remember also that a directive's data-bound input properties are not set until _after construction_.
That's a problem if we need to initialize the directive based on those properties.
They'll have been set when our `ngOninit` runs.
.l-sub-section
:marked
Our first opportunity to access those properties is the `ngOnChanges` method which
Angular calls before `ngOnit`. But Angular calls `ngOnChanges` many times after that.
It only calls `ngOnit` once.
:marked
### OnDestroy
Put cleanup logic in `ngOnDestroy`, the logic that *must* run before Angular destroys the directive.
This is the time to notify another part of the application that this component is going away.
This is the place to free resources that won't be garbage collected automatically.
Unsubscribe from observables and DOM events. Stop interval timers.
Unregister all callbacks that this directive registered with global or application services.
We risk memory leaks if we neglect to do so.
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
.l-main-section
:marked
## OnChanges
We monitor the `OnChanges` hook in this example.
Angular calls its `ngOnChanges` method whenever it detects changes to ***input properties*** of the component (or directive).
Here is our implementation of the hook.
+makeExample('lifecycle-hooks/ts/app/on-changes.component.ts', 'ng-on-changes', 'OnChangesComponent (ngOnChanges)')(format=".")
:marked
The `ngOnChanges` method takes an object that maps each changed property name to a
[SimpleChange](../api/core/SimpleChange-class.html) object with the current and previous property values.
We iterate over the changed properties and log them.
The input properties for our example `OnChangesComponent` are `hero` and `power`.
+makeExample('lifecycle-hooks/ts/app/on-changes.component.ts', 'inputs')(format=".")
:marked
The parent binds to them like this:
+makeExample('lifecycle-hooks/ts/app/on-changes-parent.component.html', 'on-changes')
:marked
Here's the sample in action as we make changes.
figure.image-display
img(src='/resources/images/devguide/lifecycle-hooks/on-changes-anim.gif' alt="OnChanges")
:marked
We see log entries as the string value of the *power* property changes. But the `ngOnChanges` did not catch changes to `hero.name`
That's surprising at first.
Angular only calls the hook when the value of the input property changes.
The value of the `hero` property is the *reference to the hero object*.
Angular doesn't care that the hero's own `name` property changed.
The hero object *reference* didn't change so, from Angular's perspective, there is no change to report!
.l-main-section
:marked
## DoCheck
2016-01-13 15:00:43 -07:00
We can use the `DoCheck` hook to detect and act upon changes that Angular doesn't catch on its own.
.l-sub-section
:marked
With this method we can detect a change that Angular overlooked.
What we do with that information to refresh the display is a separate matter.
:marked
2016-01-23 18:21:09 +00:00
The *DoCheck* sample extends the *OnChanges* sample with this implementation of `DoCheck`:
+makeExample('lifecycle-hooks/ts/app/do-check.component.ts', 'ng-do-check', 'DoCheckComponent (ngDoCheck)')(format=".")
:marked
We manually check everything that we care about, capturing and comparing against previous values.
We write a special message to the log when there are no substantive changes
to the hero or the power so we can keep an eye on the method's performance characteristics.
The results are illuminating:
2015-11-21 11:23:40 -08:00
2016-01-23 18:21:09 +00:00
figure.image-display
img(src='/resources/images/devguide/lifecycle-hooks/do-check-anim.gif' alt="DoCheck")
2015-11-21 11:23:40 -08:00
:marked
2016-01-23 18:21:09 +00:00
We now are able to detect when the hero's `name` has changed. But we must be careful.
The `ngDoCheck` hook is called with enormous frequency —
after _every_ change detection cycle no matter where the change occurred.
2016-05-06 06:17:34 -07:00
It's called over twenty times in this example before the user can do anything.
2016-01-23 18:21:09 +00:00
Most of these initial checks are triggered by Angular's first rendering of *unrelated data elsewhere on the page*.
Mere mousing into another input box triggers a call.
Relatively few calls reveal actual changes to pertinent data.
Clearly our implementation must be very lightweight or the user experience may suffer.
2015-11-21 11:23:40 -08:00
.l-sub-section
:marked
2016-05-06 06:17:34 -07:00
We also see that the `ngOnChanges` method is called in contradiction of the
2016-04-19 17:44:38 -07:00
[incorrect API documentation](../api/core/DoCheck-interface.html).
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
.l-main-section
:marked
## AfterView
The *AfterView* sample explores the `AfterViewInit` and `AfterViewChecked` hooks that Angular calls
2016-05-06 06:17:34 -07:00
*after* it creates a component's child views.
2016-01-23 18:21:09 +00:00
Here's a child view that displays a hero's name in an input box:
+makeExample('lifecycle-hooks/ts/app/after-view.component.ts', 'child-view', 'ChildComponent')(format=".")
2015-11-21 11:23:40 -08:00
:marked
2016-01-23 18:21:09 +00:00
The `AfterViewComponent` displays this child view *within its template*:
+makeExample('lifecycle-hooks/ts/app/after-view.component.ts', 'template', 'AfterViewComponent (template)')(format=".")
:marked
The following hooks take action based on changing values *within the child view*
which we can only reach by querying for the child view via the property decorated with
[@ViewChild](../api/core/ViewChild-var.html).
2015-11-21 11:23:40 -08:00
2016-01-23 18:21:09 +00:00
+makeExample('lifecycle-hooks/ts/app/after-view.component.ts', 'hooks', 'AfterViewComponent (class excerpts)')(format=".")
.a(id="wait-a-tick")
:marked
### Abide by the unidirectional data flow rule
2016-05-03 14:06:32 +02:00
The `doSomething` method updates the screen when the hero name exceeds 10 characters.
2016-01-23 18:21:09 +00:00
2016-05-03 14:06:32 +02:00
+makeExample('lifecycle-hooks/ts/app/after-view.component.ts', 'do-something', 'AfterViewComponent (doSomething)')(format=".")
2016-01-23 18:21:09 +00:00
:marked
2016-05-06 06:17:34 -07:00
Why does the `doSomething` method wait a tick before updating `comment`?
2016-01-23 18:21:09 +00:00
2016-05-06 06:17:34 -07:00
Because we must adhere to Angular's unidirectional data flow rule which says that
2016-01-23 18:21:09 +00:00
we may not update the view *after* it has been composed.
Both hooks fire after the component's view has been composed.
Angular throws an error if we update component's data-bound `comment` property immediately (try it!).
2016-05-06 06:17:34 -07:00
block tick-methods
:marked
The `LoggerService.tick` methods, which are implemented by a call to `setTimeout`, postpone the update one turn of the of the browser's JavaScript cycle ... and that's long enough.
2016-01-13 09:41:16 +00:00
2016-05-06 06:17:34 -07:00
:marked
2016-01-23 18:21:09 +00:00
Here's *AfterView* in action
figure.image-display
img(src='/resources/images/devguide/lifecycle-hooks/after-view-anim.gif' alt="AfterView")
:marked
Notice that Angular frequently calls `AfterViewChecked`, often when there are no changes of interest.
Write lean hook methods to avoid performance problems.
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
.l-main-section
:marked
## AfterContent
The *AfterContent* sample explores the `AfterContentInit` and `AfterContentChecked` hooks that Angular calls
*after* Angular projects external content into the component.
### Content projection
*Content projection* is a way to import HTML content from outside the component and insert that content
into the component's template in a designated spot.
.l-sub-section
:marked
Angular 1 developers know this technique as *transclusion*.
:marked
We'll illustrate with a variation on the [previous](#afterview) example
whose behavior and output is almost the same.
This time, instead of including the child view within the template, we'll import it from
the `AfterContentComponent`'s parent. Here's the parent's template.
+makeExample('lifecycle-hooks/ts/app/after-content.component.ts', 'parent-template', 'AfterContentParentComponent (template excerpt)')(format=".")
:marked
2016-05-06 06:17:34 -07:00
Notice that the `<my-child>` tag is tucked between the `<after-content>` tags.
2016-01-23 18:21:09 +00:00
We never put content between a component's element tags *unless we intend to project that content
into the component*.
Now look at the component's template:
+makeExample('lifecycle-hooks/ts/app/after-content.component.ts', 'template', 'AfterContentComponent (template)')(format=".")
:marked
2016-05-06 06:17:34 -07:00
The `<ng-content>` tag is a *placeholder* for the external content.
2016-01-23 18:21:09 +00:00
They tell Angular where to insert that content.
2016-05-06 06:17:34 -07:00
In this case, the projected content is the `<my-child>` from the parent.
2016-01-23 18:21:09 +00:00
figure.image-display
img(src='/resources/images/devguide/lifecycle-hooks/projected-child-view.png' width="230" alt="Projected Content")
2016-03-23 20:30:09 -07:00
:marked
2016-01-23 18:21:09 +00:00
.l-sub-section
:marked
The tell-tale signs of *content projection* are (a) HTML between component element tags
2016-05-06 06:17:34 -07:00
and (b) the presence of `<ng-content>` tags in the component's template.
2016-01-23 18:21:09 +00:00
:marked
### AfterContent hooks
*AfterContent* hooks are similar to the *AfterView* hooks. The key difference is the kind of child component
that we're looking for.
* The *AfterView* hooks concern `ViewChildren`, the child components whose element tags
appear *within* the component's template.
* The *AfterContent* hooks concern `ContentChildren`, the child components that Angular
projected into the component.
The following *AfterContent* hooks take action based on changing values in a *content child*
which we can only reach by querying for it via the property decorated with
[@ContentChild](../api/core/ContentChild-var.html).
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
+makeExample('lifecycle-hooks/ts/app/after-content.component.ts', 'hooks', 'AfterContentComponent (class excerpts)')(format=".")
2016-01-13 09:41:16 +00:00
2016-01-23 18:21:09 +00:00
:marked
### No unidirectional flow worries
2016-05-03 14:06:32 +02:00
This component's `doSomething` method update's the component's data-bound `comment` property immediately.
2016-01-23 18:21:09 +00:00
There's no [need to wait](#wait-a-tick).
Recall that Angular calls both *AfterContent* hooks before calling either of the *AfterView* hooks.
Angular completes composition of the projected content *before* finishing the composition of this component's view.
We still have a window of opportunity to modify that view.