Polish displaying-data.jade

This commit is contained in:
Yang Lin 2016-11-24 20:49:31 +08:00
parent 0e444d5d09
commit bc235b05ea
1 changed files with 57 additions and 56 deletions

View File

@ -12,11 +12,11 @@ block includes
You'll display the list of hero names and You'll display the list of hero names and
conditionally show a message below the list. conditionally show a message below the list.
本章中,我们将创建一个英雄列表组件。我们将显示英雄名字的列表,当选中一位英雄时,就在列表下方显示一条消息。 本章中,我们将创建一个英雄列表组件。我们将显示英雄名字的列表,在列表下方显示一条消息。
The final UI looks like this: The final UI looks like this:
最终的UI是这样的: 最终的用户界面是这样的:
figure.image-display figure.image-display
img(src="/resources/images/devguide/displaying-data/final.png" alt="Final UI") img(src="/resources/images/devguide/displaying-data/final.png" alt="Final UI")
@ -48,8 +48,8 @@ figure.image-display
is to bind the property name through interpolation. is to bind the property name through interpolation.
With interpolation, you put the property name in the view template, enclosed in double curly braces: `{{myHero}}`. With interpolation, you put the property name in the view template, enclosed in double curly braces: `{{myHero}}`.
要显示组件的属性,最简单的方式就是通过插值表达式(Interpolation)来绑定属性名。 要显示组件的属性,最简单的方式就是通过插值表达式interpolation来绑定属性名。
要使用插值表达式,就把属性名包裹在双花括号里放进视图模板,如`{{myHero}}`。 要使用插值表达式,就把属性名包裹在双花括号里放进视图模板,如`{{myHero}}`。
To build an illustrative example, start by creating a new project folder called <ngio-ex path="displaying-data"></ngio-ex> To build an illustrative example, start by creating a new project folder called <ngio-ex path="displaying-data"></ngio-ex>
and following the steps in [QuickStart](../quickstart.html). and following the steps in [QuickStart](../quickstart.html).
@ -58,7 +58,7 @@ figure.image-display
:marked :marked
Then modify the <ngio-ex path="app.component.ts"></ngio-ex> file by Then modify the <ngio-ex path="app.component.ts"></ngio-ex> file by
changing the template and the body of the component. changing the template and the body of the component.
然后,到`app.component.ts`文件中修改组件的模板和代码。 然后,到`app.component.ts`文件中修改组件的模板和代码。
@ -71,7 +71,7 @@ figure.image-display
:marked :marked
You added two properties to the formerly empty component: `title` and `myHero`. You added two properties to the formerly empty component: `title` and `myHero`.
再把两个属性`title`和`myHero`添加到之前空白的组件中。 再把两个属性`title`和`myHero`添加到之前空白的组件中。
The revised template displays the two component properties using double curly brace The revised template displays the two component properties using double curly brace
interpolation: interpolation:
@ -88,22 +88,22 @@ figure.image-display
quote (`'`)&mdash;allows you to compose a string over several lines, which makes the quote (`'`)&mdash;allows you to compose a string over several lines, which makes the
HTML more readable. HTML more readable.
模板是包在反引号(<code>\`</code>)中的一个多行字符串。 模板是包在ECMAScript 2015反引号<code>\`</code>中的一个多行字符串。
反引号(<code>\`</code>) —— 注意,不是单引号(') —— 有很多好用的特性。 反引号<code>\`</code> —— 注意,不是单引号(') —— 允许把一个字符串写在多行上,
我们在这里用到的是它把一个字符串写在多行上的能力这样我们的HTML模板就会更容易阅读。 使HTML模板更容易阅读。
:marked :marked
Angular automatically pulls the value of the `title` and `myHero` properties from the component and Angular automatically pulls the value of the `title` and `myHero` properties from the component and
inserts those values into the browser. Angular updates the display inserts those values into the browser. Angular updates the display
when these properties change. when these properties change.
Angular自动从组件中提取`title`和`myHero`属性的值,并且把这些值插入浏览器中。一旦这些属性发生变化Angular就会自动刷新显示。 Angular自动从组件中提取`title`和`myHero`属性的值,并且把这些值插入浏览器中。当这些属性发生变化时Angular就会自动刷新显示。
.l-sub-section .l-sub-section
:marked :marked
More precisely, the redisplay occurs after some kind of asynchronous event related to More precisely, the redisplay occurs after some kind of asynchronous event related to
the view, such as a keystroke, a timer completion, or a response to an HTTP request. the view, such as a keystroke, a timer completion, or a response to an HTTP request.
严格来说,“重新显示”是在某些与视图有关的异步事件之后发生的,比如:按键、定时器或收到异步`XHR`响应。 严格来说,“重新显示”是在某些与视图有关的异步事件之后发生的,比如:按键、定时器完成或对HTTP请求的响应。
:marked :marked
Notice that you don't call **new** to create an instance of the `AppComponent` class. Notice that you don't call **new** to create an instance of the `AppComponent` class.
@ -130,7 +130,7 @@ figure.image-display
Now run the app. It should display the title and hero name: Now run the app. It should display the title and hero name:
试一下本应用。它应该显示出标题和英雄名。 试一下本应用。它应该显示出标题和英雄名。
figure.image-display figure.image-display
img(src="/resources/images/devguide/displaying-data/title-and-hero.png" alt="Title and Hero") img(src="/resources/images/devguide/displaying-data/title-and-hero.png" alt="Title and Hero")
@ -139,12 +139,12 @@ figure.image-display
:marked :marked
The next few sections review some of the coding choices in the app. The next few sections review some of the coding choices in the app.
我们来回顾一下以前所做的决定,看看还有哪些其它选择。 我们来回顾一下以前所做的决定,看看还有哪些其它选择。
:marked :marked
## Template inline or template file? ## Template inline or template file?
## 内联(inline)模板还是模板文件? ## 内联inline模板还是模板文件?
You can store your component's template in one of two places. You can store your component's template in one of two places.
You can define it *inline* using the `template` property, or you can define You can define it *inline* using the `template` property, or you can define
@ -152,16 +152,16 @@ figure.image-display
the component metadata using the `@Component` !{_decorator}'s `templateUrl` property. the component metadata using the `@Component` !{_decorator}'s `templateUrl` property.
我们有两种地方可以用来存放组件模板。 我们有两种地方可以用来存放组件模板。
我们可以使用`template`属性把它定义为 *内联* 的,就像这里所做的一样。 我们可以使用`template`属性把它定义为 *内联* 的,或者把模板定义在一个独立的HTML文件中
或者可以把模板定义在一个独立的HTML文件中通过在`@Component`装饰器中的`templateUrl`属性把它链接到组件。 并通过在`@Component`装饰器中的`templateUrl`属性把它链接到组件。
The choice between inline and separate HTML is a matter of taste, The choice between inline and separate HTML is a matter of taste,
circumstances, and organization policy. circumstances, and organization policy.
Here the app uses inline HTML because the template is small and the demo Here the app uses inline HTML because the template is small and the demo
is simpler without the additional HTML file. is simpler without the additional HTML file.
到底选择内联HTML还是独立HTML取决于个人喜好、具体状况和组织级策略。 到底选择内联HTML还是独立HTML取决于个人喜好、具体状况和组织级策略。
这里我们选择内联HTML是因为模板很小而且这个演示很简单没有别的HTML文件 这里我们选择内联HTML是因为模板很小而且没有额外的HTML文件显得这个演示简单些
In either style, the template data bindings have the same access to the component's properties. In either style, the template data bindings have the same access to the component's properties.
@ -171,11 +171,11 @@ figure.image-display
:marked :marked
## Constructor or variable initialization? ## Constructor or variable initialization?
## 初始化:使用构造函数还是变量? ## 使用构造函数还是变量初始化
Although this example uses variable assignment to initialize the components, you can instead declare and initialize the properties using a constructor: Although this example uses variable assignment to initialize the components, you can instead declare and initialize the properties using a constructor:
虽然这个例子使用了变量赋值的方式初始化组件的属性。你可以使用构造函数来声明和初始化属性。 虽然这个例子使用了变量赋值的方式初始化组件,你还可以使用构造函数来声明和初始化属性。
+makeExcerpt('app/app-ctor.component.ts', 'class') +makeExcerpt('app/app-ctor.component.ts', 'class')
@ -192,7 +192,7 @@ figure.image-display
To display a list of heroes, begin by adding !{_an} !{_array} of hero names to the component and redefine `myHero` to be the first name in the !{_array}. To display a list of heroes, begin by adding !{_an} !{_array} of hero names to the component and redefine `myHero` to be the first name in the !{_array}.
我们准备显示一个英雄列表。先往组件中添加一个英雄名字数组,然后把`myHero`重定义为数组中的第一个名字。 要显示一个英雄列表,先向组件中添加一个英雄名字数组,然后把`myHero`重定义为数组中的第一个名字。
+makeExcerpt('app/app.component.2.ts', 'class') +makeExcerpt('app/app.component.2.ts', 'class')
@ -200,7 +200,7 @@ figure.image-display
Now use the Angular `ngFor` directive in the template to display Now use the Angular `ngFor` directive in the template to display
each item in the `heroes` list. each item in the `heroes` list.
现在我们在模板中使用Angular的`ngFor`“重复器”指令来显示`heroes`列表中的每一个条目 现在我们在模板中使用Angular的`ngFor`指令来显示`heroes`列表中的每一项
+makeExcerpt('app/app.component.2.ts', 'template') +makeExcerpt('app/app.component.2.ts', 'template')
@ -210,16 +210,16 @@ figure.image-display
It marks that `<li>` element (and its children) as the "repeater template": It marks that `<li>` element (and its children) as the "repeater template":
这个界面使用了由`<ul>`和`<li>`标签组成的无序列表。`<li>`元素里的`*ngFor`是Angular的“迭代”指令。 这个界面使用了由`<ul>`和`<li>`标签组成的无序列表。`<li>`元素里的`*ngFor`是Angular的“迭代”指令。
它将`<li>`元素和它的子级标记为“迭代模板”: 它将`<li>`元素和它的子级标记为“迭代模板”:
+makeExcerpt('app/app.component.2.ts ()', 'li', '') +makeExcerpt('app/app.component.2.ts ()', 'li', '')
.alert.is-important .alert.is-important
:marked :marked
Don't forget the leading asterisk (\*) in `*ngFor`. It is an essential part of the syntax. Don't forget the leading asterisk (\*) in `*ngFor`. It is an essential part of the syntax.
For more information, see the [Template Syntax](./template-syntax.html#ngFor) page. For more information, see the [Template Syntax](./template-syntax.html#ngFor) page.
不要忘记`*ngFor`中的前导星号(\*)。它是语法中不可或缺的一部分。 不要忘记`*ngFor`中的前导星号\*。它是语法中不可或缺的一部分。
要了解关于此语法和`ngFor`的更多知识,请参见[模板语法](./template-syntax.html#ngFor)一章。 要了解关于此语法和`ngFor`的更多知识,请参见[模板语法](./template-syntax.html#ngFor)一章。
:marked :marked
Notice the `hero` in the `ngFor` double-quoted instruction; Notice the `hero` in the `ngFor` double-quoted instruction;
@ -227,22 +227,23 @@ figure.image-display
more about template input variables in the [microsyntax](./template-syntax.html#ngForMicrosyntax) section of more about template input variables in the [microsyntax](./template-syntax.html#ngForMicrosyntax) section of
the [Template Syntax](./template-syntax.html) page. the [Template Syntax](./template-syntax.html) page.
注意看`ngFor`双引号表达式中的`hero`。 注意看`ngFor`双引号表达式中的`hero`,它是一个模板输入变量。
它是一个[模板输入变量](./template-syntax.html#ngForMicrosyntax)(译注即ngFor模板中从外界输入的变量)。 更多模板输入变量的信息,见[模板语法](./template-syntax.html)中的
[微语法microsyntax](./template-syntax.html#ngForMicrosyntax)。
Angular duplicates the `<li>` for each item in the list, setting the `hero` variable Angular duplicates the `<li>` for each item in the list, setting the `hero` variable
to the item (the hero) in the current iteration. Angular uses that variable as the to the item (the hero) in the current iteration. Angular uses that variable as the
context for the interpolation in the double curly braces. context for the interpolation in the double curly braces.
Angular为列表中的每一个条目复制一个`<li>`元素。在每个迭代中,都会把`hero`变量设置为当前条目(此英雄) Angular为列表中的每一英复制一个`<li>`元素,在每个迭代中,都会把`hero`变量设置为当前条目(英雄)
Angular把`hero`变量作为双花括号插值表达式的上下文。 Angular把`hero`变量作为双花括号插值表达式的上下文。
.l-sub-section .l-sub-section
:marked :marked
In this case, `ngFor` is displaying !{_an} !{_array}, but `ngFor` can In this case, `ngFor` is displaying !{_an} !{_array}, but `ngFor` can
repeat items for any [iterable](!{_iterableUrl}) object. repeat items for any [iterable](!{_iterableUrl}) object.
本例中,`ngFor`显示“数组”, 本例中,`ngFor`用于显示一个“数组”,
但`ngFor`可以为任何[可迭代Iterable](!{_iterableUrl})对象重复渲染条目。 但`ngFor`可以为任何[可迭代iterable](!{_iterableUrl})对象重复渲染条目。
:marked :marked
Now the heroes appear in an unordered list. Now the heroes appear in an unordered list.
@ -262,17 +263,17 @@ figure.image-display
In a simple demo, however, it's fine. In a simple demo, however, it's fine.
应用代码直接在组件内部直接定义了数据。 应用代码直接在组件内部直接定义了数据。
作为演示还可以,但它显然不是最佳实践。 作为演示还可以,但它显然不是最佳实践。
At the moment, the binding is to !{_an} !{_array} of strings. At the moment, the binding is to !{_an} !{_array} of strings.
In real applications, most bindings are to more specialized objects. In real applications, most bindings are to more specialized objects.
现在,我们绑定到了一个字符串数组。在真实的应用中偶尔这么做。但绝大多数时候,我们会绑定一些对象数组上。 现在,我们绑定到了一个字符串数组。在真实的应用中,大多时候我们会绑定到一个对象数组。
To convert this binding to use specialized objects, turn the !{_array} To convert this binding to use specialized objects, turn the !{_array}
of hero names into !{_an} !{_array} of `Hero` objects. For that you'll need a `Hero` class. of hero names into !{_an} !{_array} of `Hero` objects. For that you'll need a `Hero` class.
要将此绑定转换成使用特殊对象,把英雄名字放到`Hero`对象的数组。但首先得有一个`Hero`类。 要将此绑定转换成使用对象,需要把这个英雄名字数组变成`Hero`对象数组。但首先得有一个`Hero`类。
Create a new file in the `!{_appDir}` folder called <ngio-ex path="hero.ts"></ngio-ex> with the following code: Create a new file in the `!{_appDir}` folder called <ngio-ex path="hero.ts"></ngio-ex> with the following code:
@ -284,12 +285,12 @@ block hero-class
:marked :marked
You've defined a class with a constructor and two properties: `id` and `name`. You've defined a class with a constructor and two properties: `id` and `name`.
我们这样就定义了一个带有构造函数和两个属性:`id`和`name`的类。 我们定义了一个类,具有一个构造函数和两个属性:`id`和`name`。
It might not look like the class has properties, but it does. It might not look like the class has properties, but it does.
The declaration of the constructor parameters takes advantage of a TypeScript shortcut. The declaration of the constructor parameters takes advantage of a TypeScript shortcut.
它可能看上去不像是有属性的类但我们确实有。我们利用的是TypeScript提供的简写形式 —— 用构造函数的参数直接定义属性。 它可能看上去不像是有属性的类但我们确实有。我们利用的是TypeScript提供的简写形式 —— 用构造函数的参数直接定义属性。
Consider the first parameter: Consider the first parameter:
@ -304,11 +305,11 @@ block hero-class
* Declares a constructor parameter and its type * Declares a constructor parameter and its type
* 定义了一个构造函数参数及其类型 * 声明了一个构造函数参数及其类型
* Declares a public property of the same name * Declares a public property of the same name
* 定义了一个同名的公开属性 * 声明了一个同名的公共属性
* Initializes that property with the corresponding argument when we "new" an instance of the class * Initializes that property with the corresponding argument when we "new" an instance of the class
@ -323,7 +324,7 @@ block hero-class
The `heroes` property in the component can now use the `Hero` class to return !{_an} !{_array} The `heroes` property in the component can now use the `Hero` class to return !{_an} !{_array}
of `Hero` objects: of `Hero` objects:
我们让组件中的`heroes`属性返回这些`Hero`对象的数组。 我们让组件中的`heroes`属性返回一个`Hero`对象的数组。
+makeExcerpt('app/app.component.3.ts', 'heroes') +makeExcerpt('app/app.component.3.ts', 'heroes')
@ -351,17 +352,17 @@ block hero-class
Sometimes an app needs to display a view or a portion of a view only under specific circumstances. Sometimes an app needs to display a view or a portion of a view only under specific circumstances.
有时候,应用需要在特定情况下显示视图或视图的一部分。 有时候,应用需要在特定情况下显示视图或视图的一部分。
Let's change the example to display a message if there are more than three heroes. Let's change the example to display a message if there are more than three heroes.
让我们来修改这个例子,让它显示在多于3位英雄的情况下,显示一条消息。 让我们来修改这个例子,如果多于三位英雄,显示一条消息。
The Angular `ngIf` directive inserts or removes an element based on a !{_boolean} condition. The Angular `ngIf` directive inserts or removes an element based on a !{_boolean} condition.
To see it in action, add the following paragraph at the bottom of the template: To see it in action, add the following paragraph at the bottom of the template:
Angular的`NgIf`指令会基于条件的真假来显示或移除一个元素。 Angular的`ngIf`指令会根据一个布尔条件来显示或移除一个元素。
我们来亲自动手试一下,把下列语句加到模板的底部: 我们来亲自动手试一下,把下列语句加到模板的底部:
+makeExcerpt('app/app.component.ts', 'message') +makeExcerpt('app/app.component.ts', 'message')
@ -370,8 +371,8 @@ block hero-class
Don't forget the leading asterisk (\*) in `*ngIf`. It is an essential part of the syntax. Don't forget the leading asterisk (\*) in `*ngIf`. It is an essential part of the syntax.
Read more about `ngIf` and `*` in the [ngIf section](./template-syntax.html#ngIf) of the [Template Syntax](./template-syntax.html) page. Read more about `ngIf` and `*` in the [ngIf section](./template-syntax.html#ngIf) of the [Template Syntax](./template-syntax.html) page.
不要忘了`*ngIf`中的前导星号(\*)。它是本语法中不可或缺的一部分。 不要忘了`*ngIf`中的前导星号\*。它是本语法中不可或缺的一部分。
要学习关于此语法和`NgIf`的更多知识,请参见[模板语法](./template-syntax.html#ngIf)一章 更多`ngIf`和`* `的内容,见[模板语法](./template-syntax.html)中的[ngIf](./template-syntax.html#ngIf)
:marked :marked
The template expression inside the double quotes, The template expression inside the double quotes,
@ -382,23 +383,23 @@ block hero-class
see the [template expressions](./template-syntax.html#template-expressions) section of the see the [template expressions](./template-syntax.html#template-expressions) section of the
[Template Syntax](./template-syntax.html) page. [Template Syntax](./template-syntax.html) page.
双引号中的[模板表达式](./template-syntax.html#template-expressions)看起来很像JavaScript但它也_只是_“像”JavaScript 双引号中的模板表达式`*ngIf="heros.length > 3"`,外观和行为很象!{_Lang}
当组件中的英雄列表有三个以上的条目时Angular把这些语句添加到DOM中于是消息显示了出来。 当组件中的英雄列表有三个以上的条目时Angular把这个段落添加到DOM中于是消息显示了出来。
如果少于或等于三个条目Angular会移除这些语句于是没有消息显示 更多信息,见[模板语法](./template-syntax.html)中的[模板表达式](./template-syntax.html#template-expressions)
.alert.is-helpful .alert.is-helpful
:marked :marked
Angular isn't showing and hiding the message. It is adding and removing the paragraph element from the DOM. That improves performance, especially in larger projects when conditionally including or excluding Angular isn't showing and hiding the message. It is adding and removing the paragraph element from the DOM. That improves performance, especially in larger projects when conditionally including or excluding
big chunks of HTML with many data bindings. big chunks of HTML with many data bindings.
Angular并不是在显示和隐藏这条消息它是在从DOM中添加和移除这元素。 Angular并不是在显示和隐藏这条消息它是在从DOM中添加和移除这个段落元素。
在这个范例中,它们(在性能上)几乎等价。但是如果我们想要有条件的包含或排除“一大堆”带着很多数据绑定的HTML性能上的区别就会更加显著。 这会提高性能特别是在一些大的项目中有条件地包含或排除一大堆带着很多数据绑定的HTML时。
:marked :marked
Try it out. Because the !{_array} has four items, the message should appear. Try it out. Because the !{_array} has four items, the message should appear.
Go back into <ngio-ex path="app.component.ts"></ngio-ex> and delete or comment out one of the elements from the hero !{_array}. Go back into <ngio-ex path="app.component.ts"></ngio-ex> and delete or comment out one of the elements from the hero !{_array}.
The browser should refresh automatically and the message should disappear. The browser should refresh automatically and the message should disappear.
试一下。因为数组中有四个条目,所以消息应该显示出来。 试一下。因为这个数组中有四个条目,所以消息应该显示出来。
回到`app.component.ts`,并从英雄数组中删除或注释掉一个元素。 回到`app.component.ts`,并从英雄数组中删除或注释掉一个元素。
浏览器应该自动刷新,而消息应该会消失。 浏览器应该自动刷新,而消息应该会消失。
.l-main-section .l-main-section
@ -409,23 +410,23 @@ block hero-class
Now you know how to use: Now you know how to use:
现在我们知道了如何: 现在我们知道了如何使用以下内容
- **Interpolation** with double curly braces to display a component property - **Interpolation** with double curly braces to display a component property
- 用带有双花括号的**插值表达式Interpolation**来显示组件的一个属性 - 用带有双花括号的**插值表达式interpolation**来显示一个组件属性
- **ngFor** to display !{_an} !{_array} of items - **ngFor** to display !{_an} !{_array} of items
- 用**ngFor**来显示条目列表 - 用**ngFor**来显示数组
- A !{_Lang} class to shape the **model data** for our component and display properties of that model - A !{_Lang} class to shape the **model data** for our component and display properties of that model
- 用一个!{_Lang}类来为我们的组件描述**数据模型**并显示模型的各个属性。 - 用一个!{_Lang}类来为我们的组件描述**数据模型**并显示模型的属性。
- **ngIf** to conditionally display a chunk of HTML based on a boolean expression - **ngIf** to conditionally display a chunk of HTML based on a boolean expression
- **ngIf**用来根据一个布尔表达式有条件显示一段HTML - **ngIf**用来根据一个布尔表达式有条件显示一段HTML
Here's the final code: Here's the final code: