| 
									
										
										
										
											2016-05-04 05:18:38 -07:00
										 |  |  |  | block includes | 
					
						
							|  |  |  |  |   include ../_util-fns | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | //- The docs standard h4 style uppercases, making code terms unreadable. Override it. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | style. | 
					
						
							|  |  |  |  |   h4 {font-size: 17px !important; text-transform: none !important;} | 
					
						
							|  |  |  |  |   .syntax { font-family: Consolas, 'Lucida Sans', Courier, sans-serif; color: black; font-size: 85%; } | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   This guide looks at how Angular manipulates the DOM with **structural directives** and  | 
					
						
							|  |  |  |  |   how you can write your own structural directives to do the same thing. | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   在本章中,我们将看看Angular如何操纵DOM树,以及我们该如何在自己的指令中这么做。 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ### Table of contents | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   ### 目录 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [What are structural directives?](#definition) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [什么是结构型指令?](#definition) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [*NgIf* case study](#ngIf) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [*NgIf* 案例](#ngIf) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [The asterisk (*) prefix](#asterisk) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [星号(*)前缀](#asterisk) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [Inside *NgFor*](#ngFor) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     [*NgFor* 指令内幕](#ngFor) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |     * [microsyntax](#microsyntax) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |       [微语法](#microsyntax) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |     * [template input variables](#template-input-variable) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |       [模板输入变量](#template-input-variable) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |     * [one structural directive per element](#one-per-element) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |       [每个元素一个结构型指令](#one-per-element) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [Inside the *NgSwitch* directives](#ngSwitch) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     [*NgSwitch* 指令内幕](#ngSwitch) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [Prefer the (*) prefix](#prefer-asterisk) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [优先使用(*)前缀](#prefer-asterisk) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   * [The <ng-template> element](#template) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [<ng-template> 元素](#template) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [Group sibling elements with <ng-container>](#ng-container) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     [使用<ng-container>对兄弟元素进行分组](#ng-container) | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-31 15:25:27 -07:00
										 |  |  |  |   * [Write a structural directive](#unless) | 
					
						
							| 
									
										
										
										
											2016-05-20 16:18:58 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     [写自己的结构型指令](#unless) | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-20 16:18:58 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-03 17:11:17 -07:00
										 |  |  |  |   Try the <live-example></live-example>. | 
					
						
							| 
									
										
										
										
											2016-11-22 20:07:16 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-19 10:03:53 +01:00
										 |  |  |  |   试试<live-example>在线例子</live-example>。 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#definition | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							|  |  |  |  |   ## What are structural directives? | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   ## 什么是结构型指令? | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Structural directives are responsible for HTML layout. | 
					
						
							|  |  |  |  |   They shape or reshape the DOM's _structure_, typically by adding, removing, or manipulating | 
					
						
							|  |  |  |  |   elements. | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   结构型指令的职责是HTML布局。 | 
					
						
							|  |  |  |  |   它们塑造或重塑DOM的结构,比如添加、移除或维护这些元素。 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   As with other directives, you apply a structural directive to a _host element_.  | 
					
						
							|  |  |  |  |   The directive then does whatever it's supposed to do with that host element and its descendents. | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   像其它指令一样,你可以把结构型指令应用到一个*宿主元素*上。 | 
					
						
							|  |  |  |  |   然后它就可以对宿主元素及其子元素做点什么。 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   Structural directives are easy to recognize.  | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   An asterisk (*) precedes the directive attribute name as in this example. | 
					
						
							| 
									
										
										
										
											2017-02-26 21:44:00 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   结构型指令非常容易识别。 | 
					
						
							|  |  |  |  |   在这个例子中,星号(*)被放在指令的属性名之前。 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   No brackets. No parentheses. Just `*ngIf` set to a string. | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   没有方括号,没有圆括号,只是把`*ngIf`设置为一个字符串。 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   You'll learn in this guide that the [asterisk (*) is a convenience notation](#asterisk) | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   and the string isa [_microsyntax_](#microsyntax) rather than the usual | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   [template expression](template-syntax.html#template-expressions). | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   Angular desugars this notation into a marked-up `<ng-template>` that surrounds the | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   host element and its descendents.  | 
					
						
							|  |  |  |  |   Each structural directive does something different with that template. | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   在这个例子中,我们将学到[星号(*)这个简写方法](#asterisk),而这个字符串是一个[*微语法*](#microsyntax),而不是通常的[模板表达式](template-syntax.html#template-expressions)。 | 
					
						
							|  |  |  |  |   Angular会解开这个语法糖,变成一个`<ng-template>`标记,包裹着宿主元素及其子元素。 | 
					
						
							|  |  |  |  |   每个结构型指令都可以用这个模板做点不同的事情。 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Three of the common, built-in structural directives—[NgIf](template-syntax.html#ngIf),  | 
					
						
							|  |  |  |  |   [NgFor](template-syntax.html#ngFor), and [NgSwitch...](template-syntax.html#ngSwitch)—are  | 
					
						
							|  |  |  |  |   described in the [_Template Syntax_](template-syntax.html) guide and seen in samples throughout the Angular documentation.  | 
					
						
							|  |  |  |  |   Here's an example of them in a template: | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   三个常用的内置结构型指令 —— [NgIf](template-syntax.html#ngIf)、[NgFor](template-syntax.html#ngFor)和[NgSwitch...](template-syntax.html#ngSwitch)。 | 
					
						
							|  |  |  |  |   我们在[*模板语法*](template-syntax.html)一章中讲过它,并且在Angular文档的例子中到处都在用它。下面是模板中的例子: | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'built-in', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   This guide won't repeat how to _use_ them. But it does explain _how they work_ | 
					
						
							|  |  |  |  |   and how to [write your own](#unless) structural directive. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | .callout.is-helpful | 
					
						
							|  |  |  |  |   header Directive spelling | 
					
						
							|  |  |  |  |   :marked | 
					
						
							|  |  |  |  |      Throughout this guide, you'll see a directive spelled in both _UpperCamelCase_ and _lowerCamelCase_. | 
					
						
							|  |  |  |  |      Already you've seen `NgIf` and `ngIf`. | 
					
						
							|  |  |  |  |      There's a reason. `NgIf` refers to the directive _class_;  | 
					
						
							|  |  |  |  |      `ngIf` refers to the directive's _attribute name_.  | 
					
						
							|  |  |  |  |       | 
					
						
							|  |  |  |  |      A directive _class_ is spelled in _UpperCamelCase_ (`NgIf`). | 
					
						
							|  |  |  |  |      A directive's _attribute name_ is spelled in _lowerCamelCase_ (`ngIf`). | 
					
						
							|  |  |  |  |      The guide refers to the directive _class_ when talking about its properties and what the directive does. | 
					
						
							|  |  |  |  |      The guide refers to the _attribute name_ when describing how | 
					
						
							|  |  |  |  |      you apply the directive to an element in the HTML template. | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |     There are two other kinds of Angular directives, described extensively elsewhere: | 
					
						
							|  |  |  |  |     (1) components and (2) attribute directives. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |     A *component* manages a region of HTML in the manner of a native HTML element. | 
					
						
							|  |  |  |  |     Technically it's a directive with a template.  | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |     An [*attribute* directive](attribute-directives.html) changes the appearance or behavior | 
					
						
							|  |  |  |  |     of an element, component, or another directive. | 
					
						
							|  |  |  |  |     For example, the built-in [`NgStyle`](template-syntax.html#ngStyle) directive | 
					
						
							|  |  |  |  |     changes several element styles at the same time. | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |     You can apply many _attribute_ directives to one host element. | 
					
						
							|  |  |  |  |     You can [only apply one](#one-per-element) _structural_ directive to a host element. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#ngIf | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   ## NgIf case study | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   ## NgIf案例分析 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   `NgIf` is the simplest structural directive and the easiest to understand. | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   It takes a boolean expression and makes an entire chunk of the DOM appear or disappear. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-12 08:35:02 +08:00
										 |  |  |  |   我们重点看下`ngIf`。它是一个很好的结构型指令案例:它接受一个布尔值,并据此让一整块DOM树出现或消失。 | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif-true', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The `ngIf` directive doesn't hide elements with CSS. It adds and removes them physically from the DOM. | 
					
						
							|  |  |  |  |   Confirm that fact using browser developer tools to inspect the DOM. | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |   `ngIf`指令并不会隐藏元素。 | 
					
						
							|  |  |  |  |   使用浏览器的开发者工具就会看到:当`condition`为真的时候,只剩下了DOM顶部的段落,而底部无用的段落完全从DOM中消失了! | 
					
						
							| 
									
										
										
										
											2016-06-06 16:30:20 +01:00
										 |  |  |  |   在它的位置上是空白的`<script>`标签 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   img(src='/resources/images/devguide/structural-directives/element-not-in-dom.png' alt="ngIf=false element not in DOM") | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The top paragraph is in the DOM. The bottom, disused paragraph is not;  | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   in its place is a comment about "bindings" (more about that [later](#asterisk)). | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   When the condition is false, `NgIf` removes its host element from the DOM, | 
					
						
							|  |  |  |  |   detaches it from DOM events (the attachments that it made), | 
					
						
							|  |  |  |  |   detaches the component from Angular change detection, and destroys it. | 
					
						
							|  |  |  |  |   The component and DOM nodes can be garbage-collected and free up memory. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  |   ### Why *remove* rather than *hide*? | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   ### 为什么*移除*而不是*隐藏*? | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   A directive could hide the unwanted paragraph instead by setting its `display` style to `none`. | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   指令也可以通过把它的`display`风格设置为`none`而隐藏不需要的段落。 | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'display-none', '') | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							|  |  |  |  |   While invisible, the element remains in the DOM.  | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   当不可见时,这个元素仍然留在DOM中。 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/element-display-in-dom.png' alt="hidden element still in DOM") | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The difference between hiding and removing doesn't matter for a simple paragraph.  | 
					
						
							|  |  |  |  |   It does matter when the host element is attached to a resource intensive component. | 
					
						
							|  |  |  |  |   Such a component's behavior continues even when hidden. | 
					
						
							|  |  |  |  |   The component stays attached to its DOM element. It keeps listening to events. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  |   Angular keeps checking for changes that could affect data bindings. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Whatever the component was doing, it keeps doing. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   对于简单的段落,隐藏和移除之间的差异影响不大,但对于资源占用较多的组件是不一样的。当我们隐藏掉一个元素时,组件的行为还在继续 —— 它仍然附加在它所属的DOM元素上, | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   它也仍在监听事件。Angular会继续检查哪些能影响数据绑定的变更。 | 
					
						
							|  |  |  |  |   组件原本要做的那些事情仍在继续。 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Although invisible, the component—and all of its descendant components—tie up resources. | 
					
						
							|  |  |  |  |   The performance and memory burden can be substantial, responsiveness can degrade, and the user sees nothing. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   虽然不可见,组件及其各级子组件仍然占用着资源,而这些资源如果分配给别人可能会更有用。 | 
					
						
							|  |  |  |  |   在性能和内存方面的负担相当可观,而用户却可能无法从中受益。 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   On the positive side, showing the element again is quick. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  |   The component's previous state is preserved and ready to display. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The component doesn't re-initialize—an operation that could be expensive. | 
					
						
							|  |  |  |  |   So hiding and showing is sometimes the right thing to do. | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-05-28 10:44:47 +08:00
										 |  |  |  |   当然,从积极的一面看,重新显示这个元素会非常快。 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     组件以前的状态被保留着,并随时可以显示。 | 
					
						
							|  |  |  |  |     组件不用重新初始化 —— 该操作可能会比较昂贵。 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   But in the absence of a compelling reason to keep them around,  | 
					
						
							|  |  |  |  |   your preference should be to remove DOM elements that the user can't see | 
					
						
							|  |  |  |  |   and recover the unused resources with a structural directive like `NgIf` . | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  |   **These same considerations apply to every structural directive, whether built-in or custom.** | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Before applying a structural directive, you might want to pause for a moment  | 
					
						
							|  |  |  |  |   to consider the consequences of adding and removing elements and of creating and destroying components. | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2016-09-17 11:46:17 +08:00
										 |  |  |  |   **同样的考量也适用于每一个结构型指令,无论是内置的还是自定义的。** | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     我们应该提醒自己以及我们指令的使用者,来仔细考虑添加元素、移除元素以及创建和销毁组件的后果。 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#asterisk | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   ## The asterisk (*) prefix | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Surely you noticed the asterisk (*) prefix to the directive name | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   and wondered why it is necessary and what it does. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   Here is `*ngIf` displaying the hero's name if `hero` exists. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'asterisk', '') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The asterisk is "syntactic sugar" for something a bit more complicated. | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   Internally, Angular desugars it in two stages. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   First, it translates the `*ngIf="..."` into a template _attribute_, `template="ngIf ..."`,  like this. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif-template-attr', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   Then it translates the template _attribute_ into a `<ng-template>` _element_, wrapped around the host element, like this. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif-template', '') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   * The `*ngIf` directive moved to the `<ng-template>` element where it became a property binding,`[ngIf]`. | 
					
						
							|  |  |  |  |   * The rest of the `<div>`, including its class attribute, moved inside the `<ng-template>` element. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  |   None of these forms are actually rendered.  | 
					
						
							|  |  |  |  |   Only the finished product ends up in the DOM. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | figure.image-display | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   img(src='/resources/images/devguide/structural-directives/hero-div-in-dom.png' alt="hero div in DOM") | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   Angular consumed the `<ng-template>` content during its actual rendering and  | 
					
						
							|  |  |  |  |   replaced the `<ng-template>` with a diagnostic comment. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   The [`NgFor`](#ngFor) and [`NgSwitch...`](#ngSwitch) directives follow the same pattern. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | a#ngFor | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							|  |  |  |  |   ## Inside _*ngFor_ | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Angular transforms the `*ngFor` in similar fashion from asterisk (*) syntax through | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   template _attribute_ to `<ng-template>` _element_.  | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Here's a full-featured application of `NgFor`, written all three ways: | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'inside-ngfor', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   This is manifestly more complicated than `ngIf` and rightly so. | 
					
						
							|  |  |  |  |   The `NgFor` directive has more features, both required and optional, than the `NgIf` shown in this guide. | 
					
						
							|  |  |  |  |   At minimum `NgFor` needs a looping variable (`let hero`) and a list (`heroes`). | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   You enable these features in the string assigned to `ngFor`, which you write in Angular's [microsyntax](#microsyntax).  | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							|  |  |  |  | .alert.is-helpful | 
					
						
							|  |  |  |  |   :marked | 
					
						
							|  |  |  |  |     Everything _outside_ the `ngFor` string stays with the host element  | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |     (the `<div>`) as it moves inside the `<ng-template>`.  | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |     In this example, the `[ngClass]="odd"` stays on the `<div>`. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#microsyntax | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   ### Microsyntax | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The Angular microsyntax lets you configure a directive in a compact, friendly string. | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   The microsyntax parser translates that string into attributes on the `<ng-template>`: | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * The `let` keyword declares a [_template input variable_](#template-input-variable)  | 
					
						
							|  |  |  |  |   that you reference within the template. The input variables in this example are `hero`, `i`, and `odd`. | 
					
						
							|  |  |  |  |   The parser translates `let hero`, `let i`, and `let odd` into variables named,  | 
					
						
							|  |  |  |  |   `let-hero`, `let-i`, and `let-odd`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * The microsyntax parser takes `of` and `trackby`, title-cases them (`of` -> `Of`, `trackBy` -> `TrackBy`),  | 
					
						
							|  |  |  |  |   and prefixes them with the directive's attribute name (`ngFor`), yielding the names `ngForOf` and `ngForTrackBy`. | 
					
						
							|  |  |  |  |   Those are the names of two `NgFor` _input properties_ . | 
					
						
							|  |  |  |  |   That's how the directive learns that the list is `heroes` and the track-by function is `trackById`. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * As the `NgFor` directive loops through the list, it sets and resets properties of its own _context_ object. | 
					
						
							|  |  |  |  |   These properties include `index` and `odd` and a special property named `$implicit`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * The `let-i` and `let-odd` variables were defined as `let i=index` and `let odd=odd`. | 
					
						
							|  |  |  |  |   Angular sets them to the current value of the context's `index` and `odd` properties. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * The context property for `let-hero` wasn't specified.  | 
					
						
							|  |  |  |  |   It's intended source is implicit.  | 
					
						
							|  |  |  |  |   Angular sets `let-hero` to the value of the context's `$implicit` property | 
					
						
							|  |  |  |  |   which `NgFor` has initialized with the hero for the current iteration. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * The [API guide](../api/common/index/NgFor-directive.html "API: NgFor")  | 
					
						
							|  |  |  |  |   describes additional `NgFor` directive properties and context properties. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   These microsyntax mechanisms are available to you when you write your own structural directives. | 
					
						
							| 
									
										
										
										
											2017-04-12 08:01:02 +04:30
										 |  |  |  |   Studying the  | 
					
						
							|  |  |  |  |   [source code for `NgIf`](https://github.com/angular/angular/blob/master/packages/common/src/directives/ng_if.ts "Source: NgIf") | 
					
						
							|  |  |  |  |   and [`NgFor`](https://github.com/angular/angular/blob/master/packages/common/src/directives/ng_for_of.ts "Source: NgFor")  | 
					
						
							|  |  |  |  |   is a great way to learn more. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#template-input-variable | 
					
						
							|  |  |  |  | a#template-input-variables | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ### Template input variable | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   A _template input variable_ is a variable whose value you can reference _within_ a single instance of the template. | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   There are several such variables in this example: `hero`, `i`, and `odd`. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   All are preceded by the keyword `let`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   A _template input variable_ is **_not_** the same as a  | 
					
						
							|  |  |  |  |   [template _reference_ variable](template-syntax.html#ref-vars), | 
					
						
							|  |  |  |  |   neither _semantically_ nor _syntactically_. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   You declare a template _input_ variable using the `let` keyword (`let hero`).  | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The variable's scope is limited to a _single instance_ of the repeated template. | 
					
						
							|  |  |  |  |   You can use the same variable name again in the definition of other structural directives. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   You declare a template _reference_ variable by prefixing the variable name with `#` (`#var`). | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   A _reference_ variable refers to its attached element, component or directive. | 
					
						
							|  |  |  |  |   It can be accessed _anywhere_ in the _entire template_. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   Template _input_ and _reference_ variable names have their own namespaces. The `hero` in `let hero` is never the same | 
					
						
							|  |  |  |  |   variable as the `hero` declared as `#hero`. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | a#one-per-element | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ### One structural directive per host element | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-03 18:35:11 +01:00
										 |  |  |  |   Someday you'll want to repeat a block of HTML but only when a particular condition is true. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   You'll _try_ to put both an `*ngFor` and an `*ngIf` on the same host element. | 
					
						
							|  |  |  |  |   Angular won't let you. You may apply only one _structural_ directive to an element.  | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   The reason is simplicity. Structural directives can do complex things with the host element and its descendents. | 
					
						
							|  |  |  |  |   When two directives lay claim to the same host element, which one takes precedence?  | 
					
						
							|  |  |  |  |   Which should go first, the `NgIf` or the `NgFor`? Can the `NgIf` cancel the effect of the `NgFor`? | 
					
						
							|  |  |  |  |   If so (and it seems like it should be so), how should Angular generalize the ability to cancel for other structural directives? | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   There are no easy answers to these questions. Prohibiting multiple structural directives makes them moot. | 
					
						
							|  |  |  |  |   There's an easy solution for this use case: put the `*ngIf` on a container element that wraps the `*ngFor` element. | 
					
						
							|  |  |  |  |   One or both elements can be an [`ng-container`](#ngcontainer) so you don't have to introduce extra levels of HTML. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | a#ngSwitch | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   ## Inside _NgSwitch_ directives | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The Angular _NgSwitch_ is actually a set of cooperating directives: `NgSwitch`, `NgSwitchCase`, and `NgSwitchDefault`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Here's an example. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngswitch', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The switch value assigned to `NgSwitch` (`hero.emotion`) determines which | 
					
						
							|  |  |  |  |   (if any) of the switch cases are displayed. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   `NgSwitch` itself is not a structural directive.  | 
					
						
							|  |  |  |  |   It's an _attribute_ directive that controls the behavior of the other two switch directives. | 
					
						
							|  |  |  |  |   That's why you write `[ngSwitch]`, never `*ngSwitch`. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   `NgSwitchCase` and `NgSwitchDefault` _are_ structural directives.  | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   You attach them to elements using the asterisk (*) prefix notation.  | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   An `NgSwitchCase` displays its host element when its value matches the switch value. | 
					
						
							|  |  |  |  |   The `NgSwitchDefault` displays its host element when no sibling `NgSwitchCase` matches the switch value. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     *Design thought*: minimize initialization effort and consider caching state in a | 
					
						
							|  |  |  |  |     companion service. | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     *设计思路*:要最小化初始化的成本,并考虑把状态缓存在一个伴生的服务中。 | 
					
						
							|  |  |  |  |      | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   As with other structural directives, the `NgSwitchCase` and `NgSwitchDefault`  | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   can be desugared into the template _attribute_ form. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   **These same considerations apply to every structural directive, whether built-in or custom.** | 
					
						
							|  |  |  |  |   We should ask ourselves — and the users of our directives — to think carefully | 
					
						
							|  |  |  |  |   about the consequences of adding and removing elements and of creating and destroying components. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   **同样的考量也适用于每一个结构型指令,无论是内置的还是自定义的。** | 
					
						
							|  |  |  |  |     我们应该提醒自己以及我们指令的使用者,来仔细考虑添加元素、移除元素以及创建和销毁组件的后果。 | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   Let's see these dynamics at work. For fun, we'll stack the deck *against* | 
					
						
							|  |  |  |  |   our recommendation and consider a component called `heavy-loader` that | 
					
						
							|  |  |  |  |   ***pretends*** to load a ton of data when initialized. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   让我们在实践中看看这些变化。为了娱乐,我们设想在甲板上有个叫`heavy-loader`(重型起重机)的组件,它会***假装***在初始化时装载一吨数据。 | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   We'll display two instances of the component.  We toggle the visibility of the first one with CSS. | 
					
						
							|  |  |  |  |   We toggle the second into and out of the DOM with `ngIf`. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   我们将显示该组件的两个实例。我们使用CSS切换第一个实例的可见性,用`ngIf`把第二个实例添加到DOM和将其移除。 | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  | +makeTabs( | 
					
						
							|  |  |  |  |     `structural-directives/ts/src/app/structural-directives.component.html, | 
					
						
							|  |  |  |  |     structural-directives/ts/src/app/heavy-loader.component.ts`, | 
					
						
							|  |  |  |  |     'message-log,', | 
					
						
							|  |  |  |  |     'template (excerpt), heavy-loader.component.ts') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   That, in turn, can be desugared into the `<ng-template>` element form. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngswitch-template', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#prefer-asterisk | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   ## Prefer the asterisk (*) syntax. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   The asterisk (*) syntax is more clear than the other desugared forms. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Use [<ng-container>](#ng-container) when there's no single element | 
					
						
							|  |  |  |  |   to host the directive. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   While there's rarely a good reason to apply a structural directive in template _attribute_ or _element_ form, | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   it's still important to know that Angular creates a `<ng-template>` and to understand how it works. | 
					
						
							|  |  |  |  |   You'll refer to the `<ng-template>` when you [write your own structural directive](#unless). | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#template | 
					
						
							|  |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   ## The *<ng-template>* | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   The <ng-template> is an Angular element for rendering HTML. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   It is never displayed directly.  | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   In fact, before rendering the view, Angular _replaces_ the `<ng-template>` and its contents with a comment. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   If there is no structural directive and you merely wrap some elements in a `<ng-template>`, | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   those elements disappear. | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   That's the fate of the middle "Hip!" in the phrase "Hip! Hip! Hooray!". | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'template-tag', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Angular erases the middle "Hip!", leaving the cheer a bit less enthusiastic. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   借助内置的`ngOnInit`和`ngOnDestroy`[生命周期钩子](lifecycle-hooks.html),我们同时记录了组件的创建或销毁过程。 | 
					
						
							|  |  |  |  |   下面是它的操作演示: | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/template-rendering.png' width="350" alt="template tag rendering") | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   A structural directive puts a `<ng-template>` to work | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   as you'll see when you [write your own structural directive](#unless). | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | a#ngcontainer | 
					
						
							|  |  |  |  | a#ng-container | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   ## Group sibling elements with <ng-container> | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   There's often a _root_ element that can and should host the structural directive. | 
					
						
							|  |  |  |  |   The list element (`<li>`) is a typical host element of an `NgFor` repeater. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngfor-li', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   When there isn't a host element, you can usually wrap the content in a native HTML container element, | 
					
						
							|  |  |  |  |   such as a `<div>`, and attach the directive to that wrapper. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   Introducing another container element—typically a `<span>` or `<div>`—to  | 
					
						
							|  |  |  |  |   group the elements under a single _root_ is usually harmless.  | 
					
						
							|  |  |  |  |   _Usually_ ... but not _always_. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   The grouping element may break the template appearance because CSS styles  | 
					
						
							|  |  |  |  |   neither expect nor accommodate the new layout. | 
					
						
							|  |  |  |  |   For example, suppose you have the following paragraph layout. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif-span', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   You also have a CSS style rule that happens to apply to a `<span>` within a `<p>`aragraph. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.css', 'p-span', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The constructed paragraph renders strangely. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/bad-paragraph.png' alt="spanned paragraph with bad style") | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The `p span` style, intended for use elsewhere, was inadvertently applied here. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   Another problem: some HTML elements require all immediate children to be of a specific type. | 
					
						
							|  |  |  |  |   For example, the `<select>` element requires `<option>` children.  | 
					
						
							|  |  |  |  |   You can't wrap the _options_ in a conditional `<div>` or a `<span>`. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   When you try this, | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'select-span', '') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   the drop down is empty. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/bad-select.png' alt="spanned options don't work") | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The browser won't display an `<option>` within a `<span>`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   ### <ng-container> to the rescue | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   The Angular `<ng-container>` is a grouping element that doesn't interfere with styles or layout | 
					
						
							|  |  |  |  |   because Angular _doesn't put it in the DOM_.  | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   Here's the conditional paragraph again, this time using `<ng-container>`. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'ngif-ngcontainer', '') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   It renders properly. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/good-paragraph.png' alt="ngcontainer paragraph with proper style") | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   Now conditionally exclude a _select_ `<option>` with `<ng-container>`. | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | +makeExcerpt('src/app/app.component.html', 'select-ngcontainer', '') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   The drop down works properly. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/select-ngcontainer-anim.gif' alt="ngcontainer options work properly") | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  |   The `<ng-container>` is a syntax element recognized by the Angular parser. | 
					
						
							|  |  |  |  |   It's not a directive, component, class, or interface.  | 
					
						
							|  |  |  |  |   It's more like the curly braces in a JavaScript `if`-block: | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | code-example(language="javascript"). | 
					
						
							|  |  |  |  |   if (someCondition) { | 
					
						
							|  |  |  |  |     statement1; | 
					
						
							|  |  |  |  |     statement2; | 
					
						
							|  |  |  |  |     statement3; | 
					
						
							|  |  |  |  |   }  | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   Without those braces, JavaScript would only execute the first statement | 
					
						
							|  |  |  |  |   when you intend to conditionally execute all of them as a single block. | 
					
						
							|  |  |  |  |   The `<ng-container>` satisfies a similar need in Angular templates. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#unless | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | .l-main-section | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ## Write a structural directive | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   In this section, you write an `UnlessDirective` structural directive | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   that does the opposite of `NgIf`. | 
					
						
							|  |  |  |  |   `NgIf` displays the template content when the condition is `true`. | 
					
						
							|  |  |  |  |   `UnlessDirective` displays the content when the condition is ***false***. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'myUnless-1', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Creating a directive is similar to creating a component. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     创建指令很像创建组件。 | 
					
						
							|  |  |  |  |      | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   * Import the `Directive` decorator (instead of the `Component` decorator). | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |     导入`Directive`装饰器(而不再是`Component`)。 | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |   * Import the `Input`, `TemplateRef`, and `ViewContainerRef` symbols; you'll need them for _any_ structural directive . | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   * Apply the decorator to the directive class. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   * Set the CSS *attribute selector* that identifies the directive when applied to an element in a template. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Here's how you might begin: | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/unless.directive.ts (skeleton)') | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   The directive's _selector_ is typically the directive's **attribute name** in square brackets, `[myUnless]`. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The brackets define a CSS | 
					
						
							|  |  |  |  |   <a href="https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors" target="_blank" title="MDN: Attribute selectors">attribute selector</a>. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   The directive _attribute name_ should be spelled in _lowerCamelCase_ and begin with a prefix. | 
					
						
							|  |  |  |  |   Don't use `ng`. That prefix belongs to Angular. | 
					
						
							|  |  |  |  |   Pick something short that fits you or your company. | 
					
						
							|  |  |  |  |   In this example, the prefix is `my`. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The directive _class_ name ends in `Directive` per the [style guide](style-guide.html#02-03 "Angular Style Guide"). | 
					
						
							|  |  |  |  |   Angular's own directives do not. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ### _TemplateRef_ and _ViewContainerRef_ | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   A simple structural directive like this one creates an  | 
					
						
							|  |  |  |  |   [_embedded view_](../api/core/index/EmbeddedViewRef-class.html "API: EmbeddedViewRef") | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   from the Angular-generated `<ng-template>` and inserts that view in a  | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   [_view container_](../api/core/index/ViewContainerRef-class.html "API: ViewContainerRef") | 
					
						
							|  |  |  |  |   adjacent to the directive's original `<p>` host element. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   You'll acquire the `<ng-template>` contents with a | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   [`TemplateRef`](../api/core/index/TemplateRef-class.html "API: TemplateRef") | 
					
						
							|  |  |  |  |   and access the _view container_ through a | 
					
						
							|  |  |  |  |   [`ViewContainerRef`](../api/core/index/ViewContainerRef-class.html "API: ViewContainerRef"). | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |   You inject both in the directive constructor as private variables of the class. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/unless.directive.ts', 'ctor', '') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ### The _myUnless_ property | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   The directive consumer expects to bind a true/false condition to `[myUnless]`. | 
					
						
							|  |  |  |  |   That means the directive needs a `myUnless` property, decorated with `@Input`  | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | .l-sub-section | 
					
						
							|  |  |  |  |   :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |     Read about `@Input` in the [_Template Syntax_](template-syntax.html#inputs-outputs) guide. | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/unless.directive.ts', 'set', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Angular sets the  `myUnless` property whenever the value of the condition changes. | 
					
						
							|  |  |  |  |   Because the `myUnless` property does work, it needs a setter. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * If the condition is falsy and the view hasn't been created previously, | 
					
						
							|  |  |  |  |   tell the _view container_ to create the _embedded view_ from the template. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * If the condition is truthy and the view is currently displayed,  | 
					
						
							|  |  |  |  |   clear the container which also destroys the view. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Nobody reads the `myUnless` property so it doesn't need a getter. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   The completed directive code looks like this: | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/unless.directive.ts (excerpt)', 'no-docs') | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-03-31 01:13:42 +02:00
										 |  |  |  |   Add this directive to the `declarations` array of the AppModule. | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   Then create some HTML to try it. | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | +makeExcerpt('src/app/app.component.html', 'myUnless', '') | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-04-15 15:34:47 +08:00
										 |  |  |  |   When the `condition` is falsy, the top (A) paragraph appears and  the bottom (B) paragraph disappears. | 
					
						
							|  |  |  |  |   When the`condition` is truthy, the top (A)paragraph is removed  and the bottom (B)paragraph appears. | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   当`condition`为`false`时,顶部的段落就会显示出来,而底部的段落消失了。 | 
					
						
							|  |  |  |  |   当`condition`为`true`时,顶部的段落被移除了,而底部的段落显示了出来。 | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | figure.image-display | 
					
						
							|  |  |  |  |   img(src='/resources/images/devguide/structural-directives/unless-anim.gif' alt="UnlessDirective in action" ) | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | a#summary | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | .l-main-section | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | :marked | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   ## Summary | 
					
						
							| 
									
										
										
										
											2017-03-10 16:05:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   You can both try and download the source code for this guide in the <live-example></live-example>.   | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  |   Here is the source from the `src/app/` folder. | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-28 14:31:05 +08:00
										 |  |  |  |   本章相关的代码如下: | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-13 14:47:42 +00:00
										 |  |  |  | +makeTabs(` | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   structural-directives/ts/src/app/app.component.ts, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/app.component.html, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/app.component.css, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/app.module.ts, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/hero.ts, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/hero-switch.components.ts, | 
					
						
							|  |  |  |  |   structural-directives/ts/src/app/unless.directive.ts | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  |   `, | 
					
						
							|  |  |  |  |   null, | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   `app.component.ts, | 
					
						
							|  |  |  |  |    app.component.html, | 
					
						
							|  |  |  |  |    app.component.css, | 
					
						
							|  |  |  |  |    app.module.ts,  | 
					
						
							|  |  |  |  |    hero.ts, | 
					
						
							|  |  |  |  |    hero-switch.components.ts, | 
					
						
							|  |  |  |  |    unless.directive.ts | 
					
						
							|  |  |  |  |   `) | 
					
						
							| 
									
										
										
										
											2015-12-17 13:49:33 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  | :marked | 
					
						
							|  |  |  |  |   You learned | 
					
						
							| 
									
										
										
										
											2017-03-09 17:30:23 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * that structural directives manipulate HTML layout. | 
					
						
							|  |  |  |  |   * to use [`<ng-container>`](#ngcontainer) as a grouping element when there is no suitable host element. | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   * that the Angular desugars [asterisk (*) syntax](#asterisk) into a `<ng-template>`. | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * how that works for the `NgIf`, `NgFor` and `NgSwitch` built-in directives. | 
					
						
							| 
									
										
										
										
											2017-04-12 04:59:12 +02:00
										 |  |  |  |   * about the [_microsyntax_](#microsyntax) that expands into a [`<ng-template>`](#template). | 
					
						
							| 
									
										
										
										
											2017-02-06 19:06:13 -08:00
										 |  |  |  |   * to write a [custom structural directive](#unless), `UnlessDirective`. |