Merge remote-tracking branch 'origin/master'
# Conflicts: # public/docs/ts/latest/cookbook/_data.json # public/docs/ts/latest/guide/_data.json # public/docs/ts/latest/guide/security.jade # public/docs/ts/latest/tutorial/toh-pt6.jade
This commit is contained in:
commit
d7a71ccf59
|
@ -18,10 +18,6 @@ before_install:
|
||||||
before_script:
|
before_script:
|
||||||
- sh -e /etc/init.d/xvfb start
|
- sh -e /etc/init.d/xvfb start
|
||||||
install:
|
install:
|
||||||
- npm install --no-optional
|
- ./script/install.sh
|
||||||
- npm install --prefix public/docs/_examples
|
|
||||||
- npm install --prefix public/docs/_examples/_protractor
|
|
||||||
- npm run webdriver:update --prefix public/docs/_examples/_protractor
|
|
||||||
- gulp add-example-boilerplate
|
|
||||||
script:
|
script:
|
||||||
- gulp $SCRIPT
|
- gulp $SCRIPT
|
||||||
|
|
|
@ -97,14 +97,15 @@ mixin makeExcerpt(_filePath, _region, _title, stylePatterns)
|
||||||
- var filePath = adjustments.filePath;
|
- var filePath = adjustments.filePath;
|
||||||
- var title = adjustments.title;
|
- var title = adjustments.title;
|
||||||
- var region = _region || parenText;
|
- var region = _region || parenText;
|
||||||
- var excerpt = !region || parenText === '' ? 'excerpt' : region;
|
- var excerpt = !region || parenText === '' ? 'excerpt' : parenText || region;
|
||||||
- if (title) title = title + ' (' + excerpt + ')';
|
- if (title) title = title + ' (' + excerpt + ')';
|
||||||
+makeExample(filePath, region, title, stylePatterns)(format='.')
|
+makeExample(filePath, region, title, stylePatterns)(format='.')
|
||||||
|
|
||||||
//- Extract the doc example name from `current`.
|
//- Get the doc example name either from `_example` if set, or
|
||||||
|
//- extract the example name from `current`.
|
||||||
- var getExampleName = function() {
|
- var getExampleName = function() {
|
||||||
- var dir = current.path[current.path.length - 1];
|
- var dir = current.path[current.path.length - 1];
|
||||||
- return dir == 'latest' ? current.source : dir;
|
- return _example ? _example : dir == 'latest' ? current.source : dir;
|
||||||
- };
|
- };
|
||||||
|
|
||||||
mixin makeTabs(filePaths, regions, tabNames, stylePatterns)
|
mixin makeTabs(filePaths, regions, tabNames, stylePatterns)
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
/* #docplaster */
|
/* #docregion */
|
||||||
/* #docregion css */
|
|
||||||
h1 {
|
h1 {
|
||||||
font-size: 1.2em;
|
font-size: 1.2em;
|
||||||
color: #999;
|
color: #999;
|
||||||
|
@ -28,4 +27,3 @@ nav a:hover {
|
||||||
nav a.router-link-active {
|
nav a.router-link-active {
|
||||||
color: #039be5;
|
color: #039be5;
|
||||||
}
|
}
|
||||||
/* #enddocregion css */
|
|
||||||
|
|
|
@ -9,5 +9,7 @@
|
||||||
<input [(ngModel)]="hero.name" placeholder="name" />
|
<input [(ngModel)]="hero.name" placeholder="name" />
|
||||||
</div>
|
</div>
|
||||||
<button (click)="goBack()">Back</button>
|
<button (click)="goBack()">Back</button>
|
||||||
|
<!-- #docregion save -->
|
||||||
<button (click)="save()">Save</button>
|
<button (click)="save()">Save</button>
|
||||||
</div>
|
<!-- #enddocregion save -->
|
||||||
|
</div>
|
||||||
|
|
|
@ -50,11 +50,11 @@ export class HeroDetailComponent implements OnInit {
|
||||||
.catch(error => this.error = error); // TODO: Display error message
|
.catch(error => this.error = error); // TODO: Display error message
|
||||||
}
|
}
|
||||||
// #enddocregion save
|
// #enddocregion save
|
||||||
// #docregion goback
|
// #docregion goBack
|
||||||
goBack(savedHero: Hero = null) {
|
goBack(savedHero: Hero = null) {
|
||||||
this.close.emit(savedHero);
|
this.close.emit(savedHero);
|
||||||
if (this.navigated) { window.history.back(); }
|
if (this.navigated) { window.history.back(); }
|
||||||
}
|
}
|
||||||
// #enddocregion goback
|
// #enddocregion goBack
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -12,11 +12,11 @@ import { Hero } from './hero';
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export class HeroService {
|
export class HeroService {
|
||||||
|
|
||||||
|
// #docregion getHeroes
|
||||||
private heroesUrl = 'app/heroes'; // URL to web api
|
private heroesUrl = 'app/heroes'; // URL to web api
|
||||||
|
|
||||||
constructor(private http: Http) { }
|
constructor(private http: Http) { }
|
||||||
|
|
||||||
// #docregion get-heroes
|
|
||||||
getHeroes(): Promise<Hero[]> {
|
getHeroes(): Promise<Hero[]> {
|
||||||
return this.http.get(this.heroesUrl)
|
return this.http.get(this.heroesUrl)
|
||||||
// #docregion to-promise
|
// #docregion to-promise
|
||||||
|
@ -29,7 +29,7 @@ export class HeroService {
|
||||||
.catch(this.handleError);
|
.catch(this.handleError);
|
||||||
// #enddocregion catch
|
// #enddocregion catch
|
||||||
}
|
}
|
||||||
// #enddocregion get-heroes
|
// #enddocregion getHeroes
|
||||||
|
|
||||||
getHero(id: number) {
|
getHero(id: number) {
|
||||||
return this.getHeroes()
|
return this.getHeroes()
|
||||||
|
@ -45,7 +45,7 @@ export class HeroService {
|
||||||
}
|
}
|
||||||
// #enddocregion save
|
// #enddocregion save
|
||||||
|
|
||||||
// #docregion delete-hero
|
// #docregion delete
|
||||||
delete(hero: Hero) {
|
delete(hero: Hero) {
|
||||||
let headers = new Headers();
|
let headers = new Headers();
|
||||||
headers.append('Content-Type', 'application/json');
|
headers.append('Content-Type', 'application/json');
|
||||||
|
@ -57,9 +57,9 @@ export class HeroService {
|
||||||
.toPromise()
|
.toPromise()
|
||||||
.catch(this.handleError);
|
.catch(this.handleError);
|
||||||
}
|
}
|
||||||
// #enddocregion delete-hero
|
// #enddocregion delete
|
||||||
|
|
||||||
// #docregion post-hero
|
// #docregion post
|
||||||
// Add new Hero
|
// Add new Hero
|
||||||
private post(hero: Hero): Promise<Hero> {
|
private post(hero: Hero): Promise<Hero> {
|
||||||
let headers = new Headers({
|
let headers = new Headers({
|
||||||
|
@ -71,9 +71,9 @@ export class HeroService {
|
||||||
.then(res => res.json().data)
|
.then(res => res.json().data)
|
||||||
.catch(this.handleError);
|
.catch(this.handleError);
|
||||||
}
|
}
|
||||||
// #enddocregion post-hero
|
// #enddocregion post
|
||||||
|
|
||||||
// #docregion put-hero
|
// #docregion put
|
||||||
// Update existing Hero
|
// Update existing Hero
|
||||||
private put(hero: Hero) {
|
private put(hero: Hero) {
|
||||||
let headers = new Headers();
|
let headers = new Headers();
|
||||||
|
@ -87,13 +87,13 @@ export class HeroService {
|
||||||
.then(() => hero)
|
.then(() => hero)
|
||||||
.catch(this.handleError);
|
.catch(this.handleError);
|
||||||
}
|
}
|
||||||
// #enddocregion put-hero
|
// #enddocregion put
|
||||||
|
|
||||||
// #docregion error-handler
|
// #docregion handleError
|
||||||
private handleError(error: any) {
|
private handleError(error: any) {
|
||||||
console.error('An error occurred', error);
|
console.error('An error occurred', error);
|
||||||
return Promise.reject(error.message || error);
|
return Promise.reject(error.message || error);
|
||||||
}
|
}
|
||||||
// #enddocregion error-handler
|
// #enddocregion handleError
|
||||||
}
|
}
|
||||||
// #enddocregion
|
|
||||||
|
|
|
@ -5,18 +5,19 @@
|
||||||
<span class="hero-element">
|
<span class="hero-element">
|
||||||
<span class="badge">{{hero.id}}</span> {{hero.name}}
|
<span class="badge">{{hero.id}}</span> {{hero.name}}
|
||||||
</span>
|
</span>
|
||||||
<!-- #docregion delete-hero -->
|
<!-- #docregion delete -->
|
||||||
<button class="delete-button" (click)="delete(hero, $event)">Delete</button>
|
<button class="delete-button" (click)="deleteHero(hero, $event)">Delete</button>
|
||||||
<!-- #enddocregion delete-hero -->
|
<!-- #enddocregion delete -->
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
|
||||||
<!-- #docregion add-hero -->
|
<!-- #docregion add-and-error -->
|
||||||
|
<div class="error" *ngIf="error">{{error}}</div>
|
||||||
<button (click)="addHero()">Add New Hero</button>
|
<button (click)="addHero()">Add New Hero</button>
|
||||||
<div *ngIf="addingHero">
|
<div *ngIf="addingHero">
|
||||||
<my-hero-detail (close)="close($event)"></my-hero-detail>
|
<my-hero-detail (close)="close($event)"></my-hero-detail>
|
||||||
</div>
|
</div>
|
||||||
<!-- #enddocregion add-hero -->
|
<!-- #enddocregion add-and-error -->
|
||||||
|
|
||||||
<div *ngIf="selectedHero">
|
<div *ngIf="selectedHero">
|
||||||
<h2>
|
<h2>
|
||||||
|
|
|
@ -18,7 +18,9 @@ export class HeroesComponent implements OnInit {
|
||||||
heroes: Hero[];
|
heroes: Hero[];
|
||||||
selectedHero: Hero;
|
selectedHero: Hero;
|
||||||
addingHero = false;
|
addingHero = false;
|
||||||
|
// #docregion error
|
||||||
error: any;
|
error: any;
|
||||||
|
// #enddocregion error
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
private router: Router,
|
private router: Router,
|
||||||
|
@ -28,10 +30,10 @@ export class HeroesComponent implements OnInit {
|
||||||
this.heroService
|
this.heroService
|
||||||
.getHeroes()
|
.getHeroes()
|
||||||
.then(heroes => this.heroes = heroes)
|
.then(heroes => this.heroes = heroes)
|
||||||
.catch(error => this.error = error); // TODO: Display error message
|
.catch(error => this.error = error);
|
||||||
}
|
}
|
||||||
|
|
||||||
// #docregion add
|
// #docregion addHero
|
||||||
addHero() {
|
addHero() {
|
||||||
this.addingHero = true;
|
this.addingHero = true;
|
||||||
this.selectedHero = null;
|
this.selectedHero = null;
|
||||||
|
@ -41,10 +43,10 @@ export class HeroesComponent implements OnInit {
|
||||||
this.addingHero = false;
|
this.addingHero = false;
|
||||||
if (savedHero) { this.getHeroes(); }
|
if (savedHero) { this.getHeroes(); }
|
||||||
}
|
}
|
||||||
// #enddocregion add
|
// #enddocregion addHero
|
||||||
|
|
||||||
// #docregion delete
|
// #docregion deleteHero
|
||||||
delete(hero: Hero, event: any) {
|
deleteHero(hero: Hero, event: any) {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
this.heroService
|
this.heroService
|
||||||
.delete(hero)
|
.delete(hero)
|
||||||
|
@ -52,9 +54,9 @@ export class HeroesComponent implements OnInit {
|
||||||
this.heroes = this.heroes.filter(h => h !== hero);
|
this.heroes = this.heroes.filter(h => h !== hero);
|
||||||
if (this.selectedHero === hero) { this.selectedHero = null; }
|
if (this.selectedHero === hero) { this.selectedHero = null; }
|
||||||
})
|
})
|
||||||
.catch(error => this.error = error); // TODO: Display error message
|
.catch(error => this.error = error);
|
||||||
}
|
}
|
||||||
// #enddocregion delete
|
// #enddocregion deleteHero
|
||||||
|
|
||||||
ngOnInit() {
|
ngOnInit() {
|
||||||
this.getHeroes();
|
this.getHeroes();
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
// #docregion
|
// #docregion , init
|
||||||
export class InMemoryDataService {
|
export class InMemoryDataService {
|
||||||
createDb() {
|
createDb() {
|
||||||
let heroes = [
|
let heroes = [
|
||||||
|
|
|
@ -1,8 +1,7 @@
|
||||||
|
/* #docregion */
|
||||||
|
.error {color:red;}
|
||||||
button.delete-button{
|
button.delete-button{
|
||||||
float:right;
|
float:right;
|
||||||
background-color: gray !important;
|
background-color: gray !important;
|
||||||
color:white;
|
color:white;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -73,11 +73,6 @@
|
||||||
"intro": "Learn how to apply CSS styles to components."
|
"intro": "Learn how to apply CSS styles to components."
|
||||||
},
|
},
|
||||||
|
|
||||||
"security": {
|
|
||||||
"title": "Security",
|
|
||||||
"intro": "Prevent security vulnerabilities"
|
|
||||||
},
|
|
||||||
|
|
||||||
"hierarchical-dependency-injection": {
|
"hierarchical-dependency-injection": {
|
||||||
"title": "Hierarchical Dependency Injectors",
|
"title": "Hierarchical Dependency Injectors",
|
||||||
"navTitle": "Hierarchical Injectors",
|
"navTitle": "Hierarchical Injectors",
|
||||||
|
@ -115,6 +110,11 @@
|
||||||
"intro": "Discover the basics of screen navigation with the Angular 2 router."
|
"intro": "Discover the basics of screen navigation with the Angular 2 router."
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"security": {
|
||||||
|
"title": "Security",
|
||||||
|
"intro": "Prevent security vulnerabilities"
|
||||||
|
},
|
||||||
|
|
||||||
"structural-directives": {
|
"structural-directives": {
|
||||||
"title": "Structural Directives",
|
"title": "Structural Directives",
|
||||||
"intro": "Angular has a powerful template engine that lets us easily manipulate the DOM structure of our elements."
|
"intro": "Angular has a powerful template engine that lets us easily manipulate the DOM structure of our elements."
|
||||||
|
@ -141,11 +141,5 @@
|
||||||
"title": "Webpack: an introduction",
|
"title": "Webpack: an introduction",
|
||||||
"intro": "Create your Angular 2 applications with a Webpack based tooling",
|
"intro": "Create your Angular 2 applications with a Webpack based tooling",
|
||||||
"hide": true
|
"hide": true
|
||||||
},
|
|
||||||
|
|
||||||
"glossary": {
|
|
||||||
"title": "Glossary",
|
|
||||||
"intro": "Brief definitions of the most important words in the Angular 2 vocabulary",
|
|
||||||
"basics": true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,12 @@
|
||||||
"hide": true
|
"hide": true
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"dynamic-form": {
|
||||||
|
"title": "Dynamic Forms",
|
||||||
|
"intro": "Render dynamic forms with NgFormModel",
|
||||||
|
"basics": true
|
||||||
|
},
|
||||||
|
|
||||||
"set-document-title": {
|
"set-document-title": {
|
||||||
"title": "设置文档标题",
|
"title": "设置文档标题",
|
||||||
"intro": "使用Title服务来设置文档标题或窗口标题"
|
"intro": "使用Title服务来设置文档标题或窗口标题"
|
||||||
|
|
|
@ -117,7 +117,6 @@
|
||||||
"intro": "管道可以在模板中转换显示的内容。"
|
"intro": "管道可以在模板中转换显示的内容。"
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
||||||
"router-deprecated": {
|
"router-deprecated": {
|
||||||
"title": "Router (Deprecated Beta)",
|
"title": "Router (Deprecated Beta)",
|
||||||
"intro": "The deprecated Beta Router.",
|
"intro": "The deprecated Beta Router.",
|
||||||
|
@ -129,6 +128,11 @@
|
||||||
"intro": "揭示如何通过Angular 2路由进行基本的屏幕导航。"
|
"intro": "揭示如何通过Angular 2路由进行基本的屏幕导航。"
|
||||||
},
|
},
|
||||||
|
|
||||||
|
"security": {
|
||||||
|
"title": "Security",
|
||||||
|
"intro": "Developing for content security in Angular applications"
|
||||||
|
},
|
||||||
|
|
||||||
"structural-directives": {
|
"structural-directives": {
|
||||||
"title": "结构型指令",
|
"title": "结构型指令",
|
||||||
"intro": "Angular有一个强力的模板引擎,它能让你轻松维护元素的DOM树结构。"
|
"intro": "Angular有一个强力的模板引擎,它能让你轻松维护元素的DOM树结构。"
|
||||||
|
|
|
@ -121,6 +121,13 @@ h2#xss 防止跨站脚本(XSS)
|
||||||
当值从模版通过属性(Property)、DOM元素属性(Attribte)、CSS类绑定或者插值表达式等方式被插入到COM的时候,
|
当值从模版通过属性(Property)、DOM元素属性(Attribte)、CSS类绑定或者插值表达式等方式被插入到COM的时候,
|
||||||
Angular将对值进行无害化,去掉不可信的值。
|
Angular将对值进行无害化,去掉不可信的值。
|
||||||
|
|
||||||
|
**Angular templates are the same as executable code**: HTML, attributes, and binding expressions
|
||||||
|
(but not the values bound!) in templates are trusted to be safe. That means applications must
|
||||||
|
prevent potentially attacker controlled values from ever making it into the source code of a
|
||||||
|
template. Never generate template source code by concatenating user input and templates! Using
|
||||||
|
the [offline template compiler](#offline-template-compiler) is an effective way to prevent these
|
||||||
|
vulnerabilities, also known as template injection.
|
||||||
|
|
||||||
### Sanitization and security contexts
|
### Sanitization and security contexts
|
||||||
### 无害化和安全环境
|
### 无害化和安全环境
|
||||||
|
|
||||||
|
@ -208,15 +215,14 @@ figure.image-display
|
||||||
|
|
||||||
### 内容安全策略
|
### 内容安全策略
|
||||||
|
|
||||||
A [Content Security Policy (CSP)](https://developer.mozilla.org/en-
|
A [Content Security Policy (CSP)]
|
||||||
US/docs/Web/Security/CSP/Introducing_Content_Security_Policy) is a defense-in-depth technique to
|
(http://www.html5rocks.com/en/tutorials/security/content-security-policy/) is a defense-in-depth
|
||||||
prevent XSS. To enable CSP, configure your web server to return an appropriate
|
technique to prevent XSS. To enable CSP, configure your web server to return an appropriate
|
||||||
`Content-Security-Policy` HTTP header. Learn more at
|
`Content-Security-Policy` HTTP header.
|
||||||
[OWASP](https://www.owasp.org/index.php/Content_Security_Policy).
|
|
||||||
|
|
||||||
[内容安全策略(CSP)](https://developer.mozilla.org/en-
|
[内容安全策略(CSP)](https://developer.mozilla.org/en-
|
||||||
US/docs/Web/Security/CSP/Introducing_Content_Security_Policy)是用来防止XSS的深度防御技术。
|
US/docs/Web/Security/CSP/Introducing_Content_Security_Policy)是用来防止XSS的深度防御技术。
|
||||||
为了打开CSP,配置你的Web服务器,让它返回合适的`Content_Security_Policy` HTTP头字段。参见[OWASP](https://www.owasp.org/index.php/Content_Security_Policy)。
|
为了打开CSP,配置你的Web服务器,让它返回合适的`Content_Security_Policy` HTTP头字段。
|
||||||
|
|
||||||
<a id="offline-template-compiler"></a>
|
<a id="offline-template-compiler"></a>
|
||||||
### Use the Offline Template Compiler
|
### Use the Offline Template Compiler
|
||||||
|
@ -238,11 +244,12 @@ figure.image-display
|
||||||
|
|
||||||
### 服务器端XSS保护
|
### 服务器端XSS保护
|
||||||
|
|
||||||
HTML constructed on the server is vulnerable to injection attacks. When generating server side
|
HTML constructed on the server is vulnerable to injection attacks. Injecting template code into an
|
||||||
HTML, e.g. for the initial page load of the Angular application, make sure to use a templating
|
Angular application is the same as injecting executable code (e.g. JavaScript) into the
|
||||||
language that automatically escapes values to prevent XSS vulnerabilities on the server. Do not
|
application; it gives the attacker full control over the application. To prevent this, make sure
|
||||||
generate Angular templates on the server side using a templating language, this carries a high
|
to use a templating language that automatically escapes values to prevent XSS vulnerabilities on
|
||||||
risk of introducing template injection vulnerabilities.
|
the server. Do not generate Angular templates on the server side using a templating language, this
|
||||||
|
carries a high risk of introducing template injection vulnerabilities.
|
||||||
|
|
||||||
服务器端构造的HTML很容易被注入攻击。当在服务器端生成HTML,比如Angular应用程序的初始化页面加载时,
|
服务器端构造的HTML很容易被注入攻击。当在服务器端生成HTML,比如Angular应用程序的初始化页面加载时,
|
||||||
确保在服务器端使用一个能够自动无害化值以防止XSS漏洞的模版语言。不要在服务器端使用一个模版语言生成Angular模版,
|
确保在服务器端使用一个能够自动无害化值以防止XSS漏洞的模版语言。不要在服务器端使用一个模版语言生成Angular模版,
|
||||||
|
@ -295,10 +302,10 @@ figure.image-display
|
||||||
:marked
|
:marked
|
||||||
If we need to convert user input into a trusted value, it can be convenient to do so in a
|
If we need to convert user input into a trusted value, it can be convenient to do so in a
|
||||||
controller method. The template below allows users to enter a YouTube video ID, and load the
|
controller method. The template below allows users to enter a YouTube video ID, and load the
|
||||||
corresponding video in an `<iframe>`. `<iframe src>` is a resource URL, because an untrusted
|
corresponding video in an `<iframe>`. The `<iframe src>` attribute is a resource URL security
|
||||||
source can e.g. smuggle in file downloads that unsuspecting users would execute. So we call a
|
context, because an untrusted source can e.g. smuggle in file downloads that unsuspecting users
|
||||||
method on the controller to construct a new, trusted video URL, which is then bound to the
|
would execute. So we call a method on the controller to construct a trusted video URL, which
|
||||||
`<iframe src>`.
|
Angular then allows binding into `<iframe src>`.
|
||||||
|
|
||||||
如果需要转换用户输入到一个信任的值,我们可以很方便的在控制器方法里面处理。下面的模版允许用户输入一个YouTube视频ID,
|
如果需要转换用户输入到一个信任的值,我们可以很方便的在控制器方法里面处理。下面的模版允许用户输入一个YouTube视频ID,
|
||||||
然后加载相应的视频到一个`<iframe>`。`<iframe src>`是一个资源URL,因为一个不可信的资源可以截换文件下载,被天真的用户执行。
|
然后加载相应的视频到一个`<iframe>`。`<iframe src>`是一个资源URL,因为一个不可信的资源可以截换文件下载,被天真的用户执行。
|
||||||
|
@ -363,7 +370,9 @@ h3#xsrf 跨站请求伪造(XSRF)
|
||||||
|
|
||||||
Angular applications can customize cookie and header names by binding their own
|
Angular applications can customize cookie and header names by binding their own
|
||||||
`CookieXSRFStrategy` value, or implement an entirely custom `XSRFStrategy` by providing a custom
|
`CookieXSRFStrategy` value, or implement an entirely custom `XSRFStrategy` by providing a custom
|
||||||
binding for that type.
|
binding for that type, by adding
|
||||||
|
`provide(XSRFStrategy, {useValue: new CookieXSRFStrategy('myCookieName', 'My-Header-Name')})` or
|
||||||
|
`provide(XSRFStrategy, {useClass: MyXSRFStrategy})` to your providers list.
|
||||||
|
|
||||||
Angular应用程序可以自定义cookie和头字段名字,可以通过绑定它们自己的`CookieXSRFStrategy`值,
|
Angular应用程序可以自定义cookie和头字段名字,可以通过绑定它们自己的`CookieXSRFStrategy`值,
|
||||||
也可以通过提供一个自定义类型绑定实现一个完全制定`XSRFStrategy`。
|
也可以通过提供一个自定义类型绑定实现一个完全制定`XSRFStrategy`。
|
||||||
|
|
|
@ -2041,7 +2041,7 @@ figure.image-display
|
||||||
## Template reference variables
|
## Template reference variables
|
||||||
## 模板引用变量
|
## 模板引用变量
|
||||||
|
|
||||||
A **template reference variable** is a reference to an DOM element or directive within a template.
|
A **template reference variable** is a reference to a DOM element or directive within a template.
|
||||||
|
|
||||||
**模板引用变量**是模板中对DOM元素或指令的引用。
|
**模板引用变量**是模板中对DOM元素或指令的引用。
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,13 @@
|
||||||
include ../_util-fns
|
- var _example = 'toh-6';
|
||||||
|
|
||||||
|
block includes
|
||||||
|
include ../_util-fns
|
||||||
|
- var _Http = 'Http'; // Angular `Http` library name.
|
||||||
|
- var _Angular_Http = 'Angular <code>Http</code>'
|
||||||
|
- var _Angular_http_library = 'Angular HTTP library'
|
||||||
|
- var _HTTP_PROVIDERS = 'HTTP_PROVIDERS'
|
||||||
|
- var _JSON_stringify = 'JSON.stringify'
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
# Getting and Saving Data with HTTP
|
# Getting and Saving Data with HTTP
|
||||||
|
|
||||||
|
@ -12,9 +20,9 @@ include ../_util-fns
|
||||||
客户很欣赏我们的进展!
|
客户很欣赏我们的进展!
|
||||||
现在,他们想要从服务器获取英雄数据,然后让用户添加、编程和删除英雄,并且把这些修改结果保存回服务器。
|
现在,他们想要从服务器获取英雄数据,然后让用户添加、编程和删除英雄,并且把这些修改结果保存回服务器。
|
||||||
|
|
||||||
In this chapter we teach our application to make the corresponding http calls to a remote server's web api.
|
In this chapter we teach our application to make the corresponding HTTP calls to a remote server's web API.
|
||||||
|
|
||||||
在这一章中,我们要让应用程序学会通过http调用来访问远程服务器上相应的Web API。
|
在这一章中,我们要让应用程序学会通过HTTP调用来访问远程服务器上相应的Web API。
|
||||||
|
|
||||||
p Run the #[+liveExampleLink2('', 'toh-6')] for this part.
|
p Run the #[+liveExampleLink2('', 'toh-6')] for this part.
|
||||||
|
|
||||||
|
@ -31,70 +39,77 @@ p 运行这部分的#[+liveExampleLink2('在线例子', 'toh-6')]。
|
||||||
|
|
||||||
在[前一章](toh-pt5.html)中,我们学会了在仪表盘和固定的英雄列表之间导航,并编辑选定的英雄。这也就是本章的起点。
|
在[前一章](toh-pt5.html)中,我们学会了在仪表盘和固定的英雄列表之间导航,并编辑选定的英雄。这也就是本章的起点。
|
||||||
|
|
||||||
### Keep the app transpiling and running
|
block start-server-and-watch
|
||||||
|
:marked
|
||||||
### 保持应用的转译与运行
|
### Keep the app transpiling and running
|
||||||
|
|
||||||
Open a terminal/console window and enter the following command to
|
### 保持应用的转译与运行
|
||||||
start the TypeScript compiler, start the server, and watch for changes:
|
|
||||||
|
Open a terminal/console window and enter the following command to
|
||||||
打开terminal/console窗口,输入下列命令来启动TypeScript编译器,它会启动开发服务器,并监视文件变更:
|
start the TypeScript compiler, start the server, and watch for changes:
|
||||||
|
|
||||||
|
打开terminal/console窗口,输入下列命令来启动TypeScript编译器,它会启动开发服务器,并监视文件变更:
|
||||||
|
|
||||||
code-example(language="bash").
|
code-example(language="bash").
|
||||||
npm start
|
npm start
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
The application runs and updates automatically as we continue to build the Tour of Heroes.
|
The application runs and updates automatically as we continue to build the Tour of Heroes.
|
||||||
|
|
||||||
当我们继续构建《英雄指南》时,应用会运行并自动更新。
|
当我们继续构建《英雄指南》时,应用会运行并自动更新。
|
||||||
|
|
||||||
.l-main-section
|
.l-main-section#http-providers
|
||||||
:marked
|
h1 Providing HTTP Services
|
||||||
## Prepare for Http
|
|
||||||
|
|
||||||
## 准备Http模块
|
|
||||||
|
|
||||||
`Http` is ***not*** a core Angular module.
|
h1 准备HTTP服务
|
||||||
It's Angular's optional approach to web access and it exists as a separate add-on module called `@angular/http`,
|
|
||||||
shipped in a separate script file as part of the Angular npm package.
|
|
||||||
|
|
||||||
`Http`***并不是***Angular的核心模块。
|
|
||||||
它是Angular用来进行Web访问的一种可选方式,并通过Angular包中一个名叫`@angular/http`的独立附属模块发布了出来。
|
|
||||||
|
|
||||||
Fortunately we're ready to import from `@angular/http` because `systemjs.config` configured *SystemJS* to load that library when we need it.
|
block http-library
|
||||||
|
:marked
|
||||||
幸运的是,`systemjs.config`中已经配置好了*SystemJS*,并在必要时加载它,因此我们已经为从`@angular/http`中导入它做好了准备。
|
`Http` is ***not*** a core Angular module.
|
||||||
|
It's Angular's optional approach to web access and it exists as a separate add-on module called `@angular/http`,
|
||||||
|
shipped in a separate script file as part of the Angular npm package.
|
||||||
|
|
||||||
|
`Http`***并不是***Angular的核心模块。
|
||||||
|
它是Angular用来进行Web访问的一种可选方式,并通过Angular包中一个名叫`@angular/http`的独立附属模块发布了出来。
|
||||||
|
|
||||||
|
|
||||||
|
Fortunately we're ready to import from `@angular/http` because `systemjs.config` configured *SystemJS* to load that library when we need it.
|
||||||
|
|
||||||
|
幸运的是,`systemjs.config`中已经配置好了*SystemJS*,并在必要时加载它,因此我们已经为从`@angular/http`中导入它做好了准备。
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
### Register (provide) *http* services
|
### Register (provide) *HTTP* services
|
||||||
|
|
||||||
### 注册(供应)*http*服务
|
### 注册(供应)*http*服务
|
||||||
|
|
||||||
Our app will depend upon the Angular `http` service which itself depends upon other supporting services.
|
|
||||||
The `HTTP_PROVIDERS` array from `@angular/http` library holds providers for the complete set of http services.
|
|
||||||
|
|
||||||
我们的应用将会依赖于Angular的`http`服务,它本身又依赖于其它支持类服务。
|
|
||||||
来自`@angular/http`库中的`HTTP_PROVIDERS`数组保存着这些http相关服务供应商的全集。
|
|
||||||
|
|
||||||
We should be able to access these services from anywhere in the application.
|
block http-providers
|
||||||
So we register them in the `bootstrap` method of `main.ts` where we
|
:marked
|
||||||
|
Our app will depend upon the Angular `http` service which itself depends upon other supporting services.
|
||||||
|
The `HTTP_PROVIDERS` array from `@angular/http` library holds providers for the complete set of http services.
|
||||||
|
|
||||||
|
我们的应用将会依赖于Angular的`http`服务,它本身又依赖于其它支持类服务。
|
||||||
|
来自`@angular/http`库中的`HTTP_PROVIDERS`数组保存着这些http相关服务供应商的全集。
|
||||||
|
|
||||||
|
:marked
|
||||||
|
We should be able to access `!{_Http}` services from anywhere in the application.
|
||||||
|
So we register them in the `bootstrap` call of `main.!{_docsFor}` where we
|
||||||
launch the application and its root `AppComponent`.
|
launch the application and its root `AppComponent`.
|
||||||
|
|
||||||
我们要能从本应用的任何地方访问这些服务,所以,我们就要在`main.ts`中的`bootstrap`方法中注册它们。这里同时也是我们启动应用及其根组件`AppComponent`的地方。
|
我们要能从本应用的任何地方访问这些服务,所以,我们就要在`main.ts`中的`bootstrap`方法中注册它们。这里同时也是我们启动应用及其根组件`AppComponent`的地方。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/main.ts','v1','app/main.ts (v1)')(format='.')
|
+makeExcerpt('app/main.ts','v1')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Notice that we supply the `HTTP_PROVIDERS` in an array as the second parameter to the `bootstrap` method.
|
Notice that we supply `!{_HTTP_PROVIDERS}` in !{_an} !{_array} as the second parameter to the `bootstrap` method.
|
||||||
This has the same effect the `providers` array in `@Component` metadata.
|
This has the same effect as the `providers` !{_array} in `@Component` !{_decorator}.
|
||||||
|
|
||||||
注意,我们把`HTTP_PROVIDERS`传入了`bootstrap`方法第二个参数的数组中。这与`@Component`元数据中的`providers`数组有同样的效果。
|
注意,我们把`HTTP_PROVIDERS`传入了`bootstrap`方法第二个参数的数组中。这与`@Component`元数据中的`providers`数组有同样的效果。
|
||||||
|
|
||||||
.l-main-section
|
.l-main-section
|
||||||
:marked
|
:marked
|
||||||
## Simulating the web api
|
## Simulating the web API
|
||||||
|
|
||||||
## 模拟web api
|
## 模拟web API
|
||||||
|
|
||||||
We generally recommend registering application-wide services in the root `AppComponent` *providers*.
|
We generally recommend registering application-wide services in the root `AppComponent` *providers*.
|
||||||
Here we're registering in `main` for a special reason.
|
Here we're registering in `main` for a special reason.
|
||||||
|
@ -108,54 +123,55 @@ code-example(language="bash").
|
||||||
我们的应用正处于开发的早期阶段,并且离进入产品阶段还很远。
|
我们的应用正处于开发的早期阶段,并且离进入产品阶段还很远。
|
||||||
我们甚至都还没有一个用来处理英雄相关请求的Web服务器,在此之前,*我们将不得不伪造一个*。
|
我们甚至都还没有一个用来处理英雄相关请求的Web服务器,在此之前,*我们将不得不伪造一个*。
|
||||||
|
|
||||||
We're going to *trick* the http client into fetching and saving data from
|
We're going to *trick* the HTTP client into fetching and saving data from
|
||||||
a demo/development service, the *in-memory web api*.
|
a mock service, the *in-memory web API*.
|
||||||
|
|
||||||
我们要*耍点小花招*,让http客户端从一个Demo/开发服务(*内存(in-memory)Web API*)中获取和保存数据。
|
我们要*耍点小花招*,让http客户端从一个Mock服务(*内存(in-memory)Web API*)中获取和保存数据。
|
||||||
|
|
||||||
The application itself doesn't need to know and shouldn't know about this.
|
The application itself doesn't need to know and shouldn't know about this.
|
||||||
So we'll slip the *in-memory web api* into the configuration *above* the `AppComponent`.
|
So we'll slip the in-memory web API into the configuration *above* the `AppComponent`.
|
||||||
|
|
||||||
应用本身不需要也不应该知道这些细节。
|
应用本身不需要也不应该知道这些细节。
|
||||||
所以,我们要把*内存Web API*引入`AppComponent`的上一级配置中。
|
所以,我们要把内存Web API引入`AppComponent`的上一级配置中。
|
||||||
|
|
||||||
Here is a version of `main` that performs this trick
|
Here is a version of `main` that performs this trick
|
||||||
|
|
||||||
这个版本的`main`文件就是用来耍这个小花招的:
|
这个版本的`main`文件就是用来耍这个小花招的:
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/main.ts', 'final', 'app/main.ts (final)')(format=".")
|
+makeExcerpt('app/main.ts', 'final')
|
||||||
|
|
||||||
:marked
|
block backend
|
||||||
We're replacing the default `XHRBackend`, the service that talks to the remote server,
|
|
||||||
with the *in-memory web api* service after priming it with the following `in-memory-data.service.ts` file:
|
|
||||||
|
|
||||||
借助下面这样的`in-memory-data.service.ts`文件,我们把默认的`XHRBackend`服务(该服务负责与远端服务器对话)替换成了*内存Web API*服务:
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/in-memory-data.service.ts', null, 'app/in-memory-data.service.ts')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
This file replaces the `mock-heroes.ts` which is now safe to delete.
|
|
||||||
|
|
||||||
该文件代替了`mock-heroes.ts`文件,它现在可以安全的删除了。
|
|
||||||
|
|
||||||
.alert.is-helpful
|
|
||||||
:marked
|
:marked
|
||||||
This chapter is an introduction to the Angular http client.
|
We're replacing the default `XHRBackend`, the service that talks to the remote server,
|
||||||
Please don't be distracted by the details of this backend substitution. Just follow along with the example.
|
with the in-memory web API service after priming it as follows:
|
||||||
|
|
||||||
本章是对Angular中http客户端的介绍。不要因为这种“替换后端”的细节而分心。先不要管为什么,只管照着这个例子做就可以了。
|
我们把默认的`XHRBackend`服务(该服务负责与远端服务器对话)替换成了*内存Web API*服务:
|
||||||
|
|
||||||
Learn more later about the *in-memory web api* in the [Http chapter](../guide/server-communication.html#!#in-mem-web-api).
|
+makeExample('app/in-memory-data.service.ts', 'init')
|
||||||
Remember, the *in-memory web api* is only useful in the early stages of development and for demonstrations such as this Tour of Heroes.
|
|
||||||
Skip it when you have a real web api server.
|
|
||||||
|
|
||||||
要学习关于*内存Web API*的更多知识,请参阅[Http](../guide/server-communication.html#!#in-mem-web-api)一章。
|
p This file replaces the #[code #[+adjExPath('mock-heroes.ts')]] which is now safe to delete.
|
||||||
记住,*内存Web API*只在开发的早期阶段有用,比如这个《英雄指南》。
|
|
||||||
如果你已经有了一个真实的Web API服务器,请尽管跳过它。
|
p 该文件代替了#[code #[+adjExPath('mock-heroes.ts')]]文件,它现在可以安全的删除了。
|
||||||
|
|
||||||
|
block dont-be-distracted-by-backend-subst
|
||||||
|
.alert.is-helpful
|
||||||
|
:marked
|
||||||
|
This chapter is an introduction to the !{_Angular_http_library}.
|
||||||
|
Please don't be distracted by the details of this backend substitution. Just follow along with the example.
|
||||||
|
|
||||||
|
本章是对Angular中http客户端的介绍。不要因为这种“替换后端”的细节而分心。先不要管为什么,只管照着这个例子做就可以了。
|
||||||
|
|
||||||
|
Learn more later about the in-memory web API in the [HTTP client chapter](../guide/server-communication.html#!#in-mem-web-api).
|
||||||
|
Remember, the in-memory web API is only useful in the early stages of development and for demonstrations such as this Tour of Heroes.
|
||||||
|
Skip it when you have a real web API server.
|
||||||
|
|
||||||
|
要学习关于*内存Web API*的更多知识,请参阅[HTTP客户端](../guide/server-communication.html#!#in-mem-web-api)一章。
|
||||||
|
记住,*内存Web API*只在开发的早期阶段有用,比如这个《英雄指南》。
|
||||||
|
如果你已经有了一个真实的Web API服务器,请尽管跳过它。
|
||||||
|
|
||||||
.l-main-section
|
.l-main-section
|
||||||
:marked
|
:marked
|
||||||
## Heroes and Http
|
## Heroes and HTTP
|
||||||
|
|
||||||
## 英雄与Http
|
## 英雄与Http
|
||||||
|
|
||||||
|
@ -163,78 +179,78 @@ code-example(language="bash").
|
||||||
|
|
||||||
来看看我们目前的`HeroService`的实现
|
来看看我们目前的`HeroService`的实现
|
||||||
|
|
||||||
+makeExample('toh-4/ts/app/hero.service.ts', 'get-heroes', 'app/hero.service.ts (getHeroes - old)')(format=".")
|
+makeExample('toh-4/ts/app/hero.service.ts', 'get-heroes', 'app/hero.service.ts (old getHeroes)')(format=".")
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
We returned a promise resolved with mock heroes.
|
We returned a !{_Promise} resolved with mock heroes.
|
||||||
It may have seemed like overkill at the time, but we were anticipating the
|
It may have seemed like overkill at the time, but we were anticipating the
|
||||||
day when we fetched heroes with an http client and we knew that would have to be an asynchronous operation.
|
day when we fetched heroes with an HTTP client and we knew that would have to be an asynchronous operation.
|
||||||
|
|
||||||
我们返回一个promise,它用mock版的英雄列表进行解析。
|
我们返回一个promise,它用mock版的英雄列表进行解析。
|
||||||
在当时,它可能看起来做得有点过分了,不过那时候我们就预料到有这么这一天会通过一个http客户端来获取英雄数据,而且我们知道,那必然是一个异步操作。
|
在当时,它可能看起来做得有点过分了,不过那时候我们就预料到有这么这一天会通过一个http客户端来获取英雄数据,而且我们知道,那必然是一个异步操作。
|
||||||
|
|
||||||
That day has arrived! Let's convert `getHeroes()` to use Angular's `Http` client:
|
That day has arrived! Let's convert `getHeroes()` to use HTTP:
|
||||||
|
|
||||||
那一天到来了!我们把`getHeroes()`换成用Angular的`Http`客户端的。
|
那一天到来了!我们把`getHeroes()`换成用HTTP。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'get-heroes', 'app/hero.service.ts (getHeroes using Http)')(format=".")
|
+makeExcerpt('app/hero.service.ts (new constructor and revised getHeroes)', 'getHeroes')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
### Http Promise
|
### HTTP !{_Promise}
|
||||||
|
|
||||||
### Http承诺(Promise)
|
### Http承诺(Promise)
|
||||||
|
|
||||||
We're still returning a promise but we're creating it differently.
|
We're still returning a !{_Promise} but we're creating it differently.
|
||||||
|
|
||||||
我们仍然返回一个承诺,但这次会用不同的方式创建它。
|
block get-heroes-details
|
||||||
|
:marked
|
||||||
|
The Angular `http.get` returns an RxJS `Observable`.
|
||||||
|
*Observables* are a powerful way to manage asynchronous data flows.
|
||||||
|
We'll learn about `Observables` *later*.
|
||||||
|
|
||||||
|
Angular的`http.get`返回一个RxJS的`Observable`对象。
|
||||||
|
*Observable(可观察对象)*是一个管理异步数据流的强力方式。
|
||||||
|
后面我们还会进一步学习`Observable`。
|
||||||
|
|
||||||
The Angular `http.get` returns an RxJS `Observable`.
|
For *now* we get back on familiar ground by immediately converting that `Observable` to a `Promise` using the `toPromise` operator.
|
||||||
*Observables* are a powerful way to manage asynchronous data flows.
|
|
||||||
We'll learn about `Observables` *later*.
|
*现在*,我们先利用`toPromise`操作符把`Observable`直接转换成`Promise`对象,回到已经熟悉的地盘。
|
||||||
|
|
||||||
|
+makeExample('toh-6/ts/app/hero.service.ts', 'to-promise')(format=".")
|
||||||
|
|
||||||
|
:marked
|
||||||
|
Unfortunately, the Angular `Observable` doesn't have a `toPromise` operator ... not out of the box.
|
||||||
|
The Angular `Observable` is a bare-bones implementation.
|
||||||
|
|
||||||
|
不幸的是,Angular的`Observable`并没有一个`toPromise`操作符... 没有打包在一起发布。
|
||||||
|
Angular的`Observable`只是一个骨架实现。
|
||||||
|
|
||||||
Angular的`http.get`返回一个RxJS的`Observable`对象。
|
There are scores of operators like `toPromise` that extend `Observable` with useful capabilities.
|
||||||
*Observable(可观察对象)*是一个管理异步数据流的强力方式。
|
If we want those capabilities, we have to add the operators ourselves.
|
||||||
后面我们还会进一步学习`Observable`。
|
That's as easy as importing them from the RxJS library like this:
|
||||||
|
|
||||||
|
有一大票像`toPromise`这样的操作符,会扩展`Observable`,为其添加有用的能力。
|
||||||
|
如果我们希望得到那些能力,就得自己添加那些操作符。
|
||||||
|
那很容易,只要从RxJS库中导入它们就可以了,就像这样:
|
||||||
|
|
||||||
|
+makeExample('toh-6/ts/app/hero.service.ts', 'rxjs')(format=".")
|
||||||
|
|
||||||
For *now* we get back on familiar ground by immediately converting that `Observable` to a `Promise` using the `toPromise` operator.
|
:marked
|
||||||
|
### Extracting the data in the *then* callback
|
||||||
|
|
||||||
|
### 在*then*回调中提取出数据
|
||||||
|
|
||||||
|
In the *promise*'s `then` callback we call the `json` method of the http `Response` to extract the
|
||||||
|
data within the response.
|
||||||
|
|
||||||
|
在*promise*的`then`回调中,我们调用http的`Reponse`对象的`json`方法,以提取出其中的数据。
|
||||||
|
|
||||||
*现在*,我们先利用`toPromise`操作符把`Observable`直接转换成`Promise`对象,回到已经熟悉的地盘。
|
+makeExample('toh-6/ts/app/hero.service.ts', 'to-data')(format=".")
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'to-promise')(format=".")
|
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Unfortunately, the Angular `Observable` doesn't have a `toPromise` operator ... not out of the box.
|
That response JSON has a single `data` property.
|
||||||
The Angular `Observable` is a bare-bones implementation.
|
The `data` property holds the !{_array} of *heroes* that the caller really wants.
|
||||||
|
So we grab that !{_array} and return it as the resolved !{_Promise} value.
|
||||||
不幸的是,Angular的`Observable`并没有一个`toPromise`操作符... 没有打包在一起发布。
|
|
||||||
Angular的`Observable`只是一个骨架实现。
|
|
||||||
|
|
||||||
There are scores of operators like `toPromise` that extend `Observable` with useful capabilities.
|
|
||||||
If we want those capabilities, we have to add the operators ourselves.
|
|
||||||
That's as easy as importing them from the RxJS library like this:
|
|
||||||
|
|
||||||
有一大票像`toPromise`这样的操作符,会扩展`Observable`,为其添加有用的能力。
|
|
||||||
如果我们希望得到那些能力,就得自己添加那些操作符。
|
|
||||||
那很容易,只要从RxJS库中导入它们就可以了,就像这样:
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'rxjs')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
### Extracting the data in the *then* callback
|
|
||||||
|
|
||||||
### 在*then*回调中提取出数据
|
|
||||||
|
|
||||||
In the *promise*'s `then` callback we call the `json` method of the http `Response` to extract the
|
|
||||||
data within the response.
|
|
||||||
|
|
||||||
在*promise*的`then`回调中,我们调用http的`Reponse`对象的`json`方法,以提取出其中的数据。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'to-data')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
That object returned by `json` has a single `data` property.
|
|
||||||
The `data` property holds the array of *heroes* that the caller really wants.
|
|
||||||
So we grab that array and return it as the resolved promise value.
|
|
||||||
|
|
||||||
这个由`json`方法返回的对象只有一个`data`属性。
|
这个由`json`方法返回的对象只有一个`data`属性。
|
||||||
这个`data`属性保存了*英雄*数组,这个数组才是调用者真正想要的。
|
这个`data`属性保存了*英雄*数组,这个数组才是调用者真正想要的。
|
||||||
|
@ -243,21 +259,21 @@ code-example(language="bash").
|
||||||
.alert.is-important
|
.alert.is-important
|
||||||
:marked
|
:marked
|
||||||
Pay close attention to the shape of the data returned by the server.
|
Pay close attention to the shape of the data returned by the server.
|
||||||
This particular *in-memory web api* example happens to return an object with a `data` property.
|
This particular *in-memory web API* example happens to return an object with a `data` property.
|
||||||
Your api might return something else.
|
Your API might return something else.
|
||||||
|
|
||||||
仔细看看这个由服务器返回的数据的形态。
|
仔细看看这个由服务器返回的数据的形态。
|
||||||
这个*内存Web API*的范例中所做的是返回一个带有`data`属性的对象。
|
这个*内存Web API*的范例中所做的是返回一个带有`data`属性的对象。
|
||||||
你的API也可以返回其它东西。
|
你的API也可以返回其它东西。
|
||||||
|
|
||||||
Adjust the code to match *your web api*.
|
Adjust the code to match *your web API*.
|
||||||
|
|
||||||
请随意调整这些代码,以适应*你自己的Web API*。
|
请随意调整这些代码,以适应*你自己的Web API*。
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
The caller is unaware of these machinations. It receives a promise of *heroes* just as it did before.
|
The caller is unaware of these machinations. It receives a !{_Promise} of *heroes* just as it did before.
|
||||||
It has no idea that we fetched the heroes from the server.
|
It has no idea that we fetched the heroes from the (mock) server.
|
||||||
It knows nothing of the twists and turns required to turn the http response into heroes.
|
It knows nothing of the twists and turns required to convert the HTTP response into heroes.
|
||||||
Such is the beauty and purpose of delegating data access to a service like this `HeroService`.
|
Such is the beauty and purpose of delegating data access to a service like this `HeroService`.
|
||||||
|
|
||||||
调用者不关心这些实现机制,它仍然像以前那样取得一个包含*英雄数据*的承诺。
|
调用者不关心这些实现机制,它仍然像以前那样取得一个包含*英雄数据*的承诺。
|
||||||
|
@ -269,37 +285,37 @@ code-example(language="bash").
|
||||||
|
|
||||||
### 错误处理
|
### 错误处理
|
||||||
|
|
||||||
At the end of `getHeroes` we `catch` server failures and pass them to an error handler:
|
At the end of `getHeroes()` we `catch` server failures and pass them to an error handler:
|
||||||
|
|
||||||
在`getHeroes`的最后,我们`catch`了服务器的失败信息,并把它们传给了错误处理器:
|
在`getHeroes()`的最后,我们`catch`了服务器的失败信息,并把它们传给了错误处理器:
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'catch')(format=".")
|
+makeExcerpt('app/hero.service.ts', 'catch')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
This is a critical step!
|
This is a critical step!
|
||||||
We must anticipate http failures as they happen frequently for reasons beyond our control.
|
We must anticipate HTTP failures as they happen frequently for reasons beyond our control.
|
||||||
|
|
||||||
这是一个关键的步骤!
|
这是一个关键的步骤!
|
||||||
我们必须预料到http请求会失败,因为有太多我们无法控制的原因可能导致它们频繁出现各种错误。
|
我们必须预料到http请求会失败,因为有太多我们无法控制的原因可能导致它们频繁出现各种错误。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'error-handler', 'app/hero.service.ts (Error handler)')(format=".")
|
+makeExcerpt('app/hero.service.ts', 'handleError')
|
||||||
|
- var rejected_promise = _docsFor == 'dart' ? 'propagated exception' : 'rejected promise';
|
||||||
:marked
|
:marked
|
||||||
In this demo service we log the error to the console; we should do better in real life.
|
In this demo service we log the error to the console; we should do better in real life.
|
||||||
|
|
||||||
在这个范例服务中,我们把错误记录到控制台中;在真实世界中,我们应该做得更好。
|
在这个范例服务中,我们把错误记录到控制台中;在真实世界中,我们应该做得更好。
|
||||||
|
|
||||||
We've also decided to return a user friendly form of the error to
|
We've also decided to return a user friendly form of the error to
|
||||||
the caller in a rejected promise so that the caller can display a proper error message to the user.
|
the caller in a !{rejected_promise} so that the caller can display a proper error message to the user.
|
||||||
|
|
||||||
我们还要通过一个被拒绝(rejected)的承诺(promise)来把该错误用一个用户友好的格式返回给调用者,以便调用者能把一个合适的错误信息显示给用户。
|
我们还要通过一个被拒绝(rejected)的承诺(promise)来把该错误用一个用户友好的格式返回给调用者,以便调用者能把一个合适的错误信息显示给用户。
|
||||||
|
|
||||||
### Promises are Promises
|
### !{_Promise}s are !{_Promise}s
|
||||||
|
|
||||||
### 承诺仍然是承诺
|
### 承诺仍然是承诺
|
||||||
|
|
||||||
Although we made significant *internal* changes to `getHeroes()`, the public signature did not change.
|
Although we made significant *internal* changes to `getHeroes()`, the public signature did not change.
|
||||||
We still return a promise. We won't have to update any of the components that call `getHeroes()`.
|
We still return a !{_Promise}. We won't have to update any of the components that call `getHeroes()`.
|
||||||
|
|
||||||
虽然我们对`getHeroes()`做了一些重要的*内部*修改,该方法公开的函数签名却没有任何变化。
|
虽然我们对`getHeroes()`做了一些重要的*内部*修改,该方法公开的函数签名却没有任何变化。
|
||||||
我们返回的仍然是一个承诺,不用被迫更新任何一个调用了`getHeroes()`的组件。
|
我们返回的仍然是一个承诺,不用被迫更新任何一个调用了`getHeroes()`的组件。
|
||||||
|
@ -310,11 +326,11 @@ code-example(language="bash").
|
||||||
|
|
||||||
## 添加、编辑、删除
|
## 添加、编辑、删除
|
||||||
|
|
||||||
Our stakeholders are incredibly pleased with the added flexibility from the api integration, but it doesn't stop there. Next we want to add the capability to add, edit and delete heroes.
|
Our stakeholders are incredibly pleased with the added flexibility from the API integration, but it doesn't stop there. Next we want to add the capability to add, edit and delete heroes.
|
||||||
|
|
||||||
我们的客户对这种富有弹性的API集成方式感到非常高兴,但事情可不会到此为止。下一步我们还要增加对英雄的添加、编辑和删除操作。
|
我们的客户对这种富有弹性的API集成方式感到非常高兴,但事情可不会到此为止。下一步我们还要增加对英雄的添加、编辑和删除操作。
|
||||||
|
|
||||||
We'll complete `HeroService` by creating `post`, `put` and `delete` http calls to meet our new requirements.
|
We'll complete `HeroService` by creating `post`, `put` and `delete` methods to meet our new requirements.
|
||||||
|
|
||||||
我们将为`HeroService`添加`post`、`put`和`delete`的http调用来彻底满足我们的需求。
|
我们将为`HeroService`添加`post`、`put`和`delete`的http调用来彻底满足我们的需求。
|
||||||
|
|
||||||
|
@ -324,13 +340,13 @@ code-example(language="bash").
|
||||||
### Post
|
### Post
|
||||||
|
|
||||||
We are using `post` to add new heroes. Post requests require a little bit more setup than Get requests, but the format is as follows:
|
We are using `post` to add new heroes. Post requests require a little bit more setup than Get requests, but the format is as follows:
|
||||||
|
|
||||||
我们使用`post`来添加新的英雄。post请求比get请求稍多一点儿设置工作,大致写法如下:
|
我们使用`post`来添加新的英雄。post请求比get请求稍多一点儿设置工作,大致写法如下:
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'post-hero', 'app/hero.service.ts (post hero)')(format=".")
|
+makeExcerpt('app/hero.service.ts', 'post')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Now we create a header and set the content type to `application/json`. We'll call `JSON.stringify` before we post to convert the hero object to a string.
|
For Post requests we create a header and set the content type to `application/json`. We'll call `!{_JSON_stringify}` before we post to convert the hero object to a string.
|
||||||
|
|
||||||
现在,我们创建了一个请求头,并把内容类型(content-type)设置为`application/json`。在我们post之前,要先调用`JSON.stringify`来把这个`hero`对象转换成一个字符串。
|
现在,我们创建了一个请求头,并把内容类型(content-type)设置为`application/json`。在我们post之前,要先调用`JSON.stringify`来把这个`hero`对象转换成一个字符串。
|
||||||
|
|
||||||
|
@ -338,25 +354,24 @@ code-example(language="bash").
|
||||||
|
|
||||||
### Put
|
### Put
|
||||||
|
|
||||||
`put` is used to edit a specific hero, but the structure is very similar to a `post` request. The only difference is that we have to change the url slightly by appending the id of the hero we want to edit.
|
Put will be used to update an individual hero. Its structure is very similar to Post requests. The only difference is that we have to change the url slightly by appending the id of the hero we want to update.
|
||||||
|
Put用来编辑一个指定的英雄,但是代码结构和POST请求很相似。唯一的不同是,我们要修改url,为它附加上我们想要编辑的那位英雄的id。
|
||||||
|
|
||||||
`put`用来编辑一个指定的英雄,但是代码结构和`post`请求很类似。唯一的不同是,我们要修改url,为它附加上我们想要编辑的那位英雄的id。
|
+makeExcerpt('app/hero.service.ts', 'put')
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'put-hero', 'app/hero.service.ts (put hero)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
### Delete
|
### Delete
|
||||||
|
|
||||||
### Delete
|
### Delete
|
||||||
|
|
||||||
`delete` is used to delete heroes and the format is identical to `put` except for the function name.
|
Delete will be used to delete heroes and its format is like `put` except for the function name.
|
||||||
|
|
||||||
|
Delete用来删除英雄,它的书写方式几乎和Put完全相同 —— 除了函数名。
|
||||||
|
|
||||||
`delete`用来删除英雄,它的书写方式几乎和`put`完全相同 —— 除了函数名。
|
+makeExcerpt('app/hero.service.ts', 'delete')
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'delete-hero', 'app/hero.service.ts (delete hero)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
We add a `catch` to handle our errors for all three cases.
|
We add a `catch` to handle errors for all three methods.
|
||||||
|
|
||||||
我们添加了`catch`,用来处理这三种情况下的错误。
|
我们添加了`catch`,用来处理这三种情况下的错误。
|
||||||
|
|
||||||
|
@ -365,19 +380,19 @@ code-example(language="bash").
|
||||||
|
|
||||||
### Save
|
### Save
|
||||||
|
|
||||||
We combine the call to the private `post` and `put` methods in a single `save` method. This simplifies the public api and makes the integration with `HeroDetailComponent` easier. `HeroService` determines which method to call based on the state of the `hero` object. If the hero already has an id we know it's an edit. Otherwise we know it's an add.
|
We combine the call to the private `post` and `put` methods in a single `save` method. This simplifies the public API and makes the integration with `HeroDetailComponent` easier. `HeroService` determines which method to call based on the state of the `hero` object. If the hero already has an id we know it's an edit. Otherwise we know it's an add.
|
||||||
|
|
||||||
我们把对私有方法`post`和`put`的调用组合进一个单独的`save`方法。这会简化公开的API,以简化与`HeroDetailComponent`的集成。
|
我们把对私有方法`post`和`put`的调用组合进一个单独的`save`方法。这会简化公开的API,以简化与`HeroDetailComponent`的集成。
|
||||||
`HeroService`会根据`hero`对象的状态来决定该调用哪个方法。如果这个英雄已经有了一个id,我们就知道它应该是编辑操作,否则就是添加。
|
`HeroService`会根据`hero`对象的状态来决定该调用哪个方法。如果这个英雄已经有了一个id,我们就知道它应该是编辑操作,否则就是添加。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', 'save', 'app/hero.service.ts (save hero)')(format=".")
|
+makeExcerpt('app/hero.service.ts', 'save')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
After these additions our `HeroService` looks like this:
|
After these additions our `HeroService` looks like this:
|
||||||
|
|
||||||
填加完所有这些之后,我们的`HeroService`看起来像这样:
|
填加完所有这些之后,我们的`HeroService`看起来像这样:
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero.service.ts', null, 'app/hero.service.ts')(format=".")
|
+makeExample('app/hero.service.ts')
|
||||||
|
|
||||||
.l-main-section
|
.l-main-section
|
||||||
:marked
|
:marked
|
||||||
|
@ -387,211 +402,234 @@ code-example(language="bash").
|
||||||
|
|
||||||
Loading heroes using `Http` required no changes outside of `HeroService`, but we added a few new features as well.
|
Loading heroes using `Http` required no changes outside of `HeroService`, but we added a few new features as well.
|
||||||
In the following section we will update our components to use our new methods to add, edit and delete heroes.
|
In the following section we will update our components to use our new methods to add, edit and delete heroes.
|
||||||
|
|
||||||
使用`Http`加载英雄数据并不需要在`HeroService`外部做任何修改,但我们却顺利的添加了一些新特性。
|
使用`Http`加载英雄数据并不需要在`HeroService`外部做任何修改,但我们却顺利的添加了一些新特性。
|
||||||
在后面的小节中,我们将更新我们的组件来调用这些新方法,以添加、编辑和删除英雄。
|
在后面的小节中,我们将更新我们的组件来调用这些新方法,以添加、编辑和删除英雄。
|
||||||
|
|
||||||
Before we can add those methods, we need to initialize some variables with their respective imports.
|
block hero-detail-comp-extra-imports-and-vars
|
||||||
|
|
||||||
在这么做之前,我们先得用它们各自的import语句来初始化一些变量。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero-detail.component.ts', 'variables-imports', 'app/hero-detail.component.ts')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
### Add/Edit in the *HeroDetailComponent*
|
|
||||||
|
|
||||||
### 在*HeroDetailComponent`中添加/编辑
|
|
||||||
|
|
||||||
We already have `HeroDetailComponent` for viewing details about a specific hero.
|
|
||||||
Add and Edit are natural extensions of the detail view, so we are able to reuse `HeroDetailComponent` with a few tweaks.
|
|
||||||
The original component was created to render existing data, but to add new data we have to initialize the `hero` property to an empty `Hero` object.
|
|
||||||
|
|
||||||
我们已经有了`HeroDetailComponent`,用以查看指定英雄的详情。
|
|
||||||
添加和编辑功能是对详情页的自然扩展,所以我们可以复用`HeroDetailComponent`,只要做少量修改就可以了。
|
|
||||||
该组件原本是用来渲染现存数据的,要想添加新的数据,我们就得把`hero`属性初始化在一个空的`Hero`对象。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero-detail.component.ts', 'ngOnInit', 'app/hero-detail.component.ts (ngOnInit)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
In order to differentiate between add and edit we are adding a check to see if an id is passed in the url. If the id is absent we bind `HeroDetailComponent` to an empty `Hero` object. In either case, any edits made through the UI will be bound back to the same `hero` property.
|
|
||||||
|
|
||||||
为了区分添加和编辑操作,我们增加了一步检查,看看url中是否传入了id。如果没有id,我们就把`HeroDetailComponent`绑定到一个空的`Hero`对象。
|
|
||||||
无论是哪种情况,通过UI进行的任何编辑操作都会被绑定回同一个`hero`属性。
|
|
||||||
|
|
||||||
The next step is to add a save method to `HeroDetailComponent` and call the corresponding save method in `HeroesService`.
|
|
||||||
|
|
||||||
下一步是为`HeroDetailComponent`添加一个`save`方法,并且调用`HeroesService`中相应的`save`方法。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero-detail.component.ts', 'save', 'app/hero-detail.component.ts (save)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
The same save method is used for both add and edit since `HeroService` will know when to call `post` vs `put` based on the state of the `Hero` object.
|
|
||||||
|
|
||||||
无论添加还是删除,所用的都是同一个`save`方法,这是因为`HeroService`会根据`Hero`对象的状态判断出什么时候该调用`post`什么时候该调用`put`。
|
|
||||||
|
|
||||||
After we save a hero, we redirect the browser back to the previous page using the `goBack()` method.
|
|
||||||
|
|
||||||
在我们保存了英雄之后,我们使用`goBack()`方法让浏览器回到前一个页面。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/hero-detail.component.ts', 'goback', 'app/hero-detail.component.ts (goBack)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
Here we call `emit` to notify that we just added or modified a hero. `HeroesComponent` is listening for this notification and will automatically refresh the list of heroes to include our recent updates.
|
|
||||||
|
|
||||||
这里我们调用了`emit`来通知别人:我们刚刚添加或修改了一个英雄。`HeroesComponent`正在监听这个通知,并自动刷新英雄列表,以体现我们最近所做的修改。
|
|
||||||
|
|
||||||
.l-sub-section
|
|
||||||
:marked
|
:marked
|
||||||
The `emit` "handshake" between `HeroDetailComponent` and `HeroesComponent` is an example of component to component communication. This is a topic for another day, but we have detailed information in our <a href="/docs/ts/latest/cookbook/component-communication.html#!#child-to-parent">Component Interaction Cookbook</a>
|
Before we can add those methods, we need to initialize some variables with their respective imports.
|
||||||
|
|
||||||
|
在这么做之前,我们先得用它们各自的import语句来初始化一些变量。
|
||||||
|
|
||||||
`emit`在`HeroDetailComponent`和`HeroesComponent`之间的这种握手是组件间通讯的例子之一。
|
+makeExcerpt('app/hero-detail.component.ts ()', 'variables-imports')
|
||||||
这不是今天的话题,我们在<a href="/docs/ts/latest/cookbook/component-communication.html#!#child-to-parent">组件间交互</a>一章中讲了一些详细信息。
|
|
||||||
|
block hero-detail-comp-updates
|
||||||
|
:marked
|
||||||
|
### Add/Edit in the *HeroDetailComponent*
|
||||||
|
|
||||||
|
We already have `HeroDetailComponent` for viewing details about a specific hero.
|
||||||
|
Add and Edit are natural extensions of the detail view, so we are able to reuse `HeroDetailComponent` with a few tweaks.
|
||||||
|
|
||||||
|
我们已经有了`HeroDetailComponent`,用以查看指定英雄的详情。
|
||||||
|
添加和编辑功能是对详情页的自然扩展,所以我们可以复用`HeroDetailComponent`,只要做少量修改就可以了。
|
||||||
|
|
||||||
|
The original component was created to render existing data, but to add new data we have to initialize the `hero` property to an empty `Hero` object.
|
||||||
|
|
||||||
|
该组件原本是用来渲染现存数据的,要想添加新的数据,我们就得把`hero`属性初始化在一个空的`Hero`对象。
|
||||||
|
|
||||||
|
+makeExcerpt('app/hero-detail.component.ts', 'ngOnInit')
|
||||||
|
|
||||||
|
:marked
|
||||||
|
In order to differentiate between add and edit we are adding a check to see if an id is passed in the url. If the id is absent we bind `HeroDetailComponent` to an empty `Hero` object. In either case, any edits made through the UI will be bound back to the same `hero` property.
|
||||||
|
|
||||||
|
为了区分添加和编辑操作,我们增加了一步检查,看看url中是否传入了id。如果没有id,我们就把`HeroDetailComponent`绑定到一个空的`Hero`对象。
|
||||||
|
无论是哪种情况,通过UI进行的任何编辑操作都会被绑定回同一个`hero`属性。
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Here is `HeroDetailComponent` with its new save button.
|
Add a save method to `HeroDetailComponent` and call the corresponding save method in `HeroesService`.
|
||||||
|
|
||||||
|
+makeExcerpt('app/hero-detail.component.ts', 'save')
|
||||||
|
|
||||||
|
block hero-detail-comp-save-and-goback
|
||||||
|
:marked
|
||||||
|
The same save method is used for both add and edit since `HeroService` will know when to call `post` vs `put` based on the state of the `Hero` object.
|
||||||
|
|
||||||
|
无论添加还是删除,所用的都是同一个`save`方法,这是因为`HeroService`会根据`Hero`对象的状态判断出什么时候该调用`post`什么时候该调用`put`。
|
||||||
|
|
||||||
|
After we save a hero, we redirect the browser back to the previous page using the `goBack()` method.
|
||||||
|
|
||||||
|
在我们保存了英雄之后,我们使用`goBack()`方法让浏览器回到前一个页面。
|
||||||
|
|
||||||
|
+makeExcerpt('app/hero-detail.component.ts', 'goBack')
|
||||||
|
|
||||||
|
:marked
|
||||||
|
Here we call `emit` to notify that we just added or modified a hero. `HeroesComponent` is listening for this notification and will automatically refresh the list of heroes to include our recent updates.
|
||||||
|
|
||||||
|
这里我们调用了`emit`来通知别人:我们刚刚添加或修改了一个英雄。`HeroesComponent`正在监听这个通知,并自动刷新英雄列表,以体现我们最近所做的修改。
|
||||||
|
|
||||||
|
.l-sub-section
|
||||||
|
:marked
|
||||||
|
The `emit` "handshake" between `HeroDetailComponent` and `HeroesComponent` is an example of component to component communication. This is a topic for another day, but we have detailed information in our <a href="/docs/ts/latest/cookbook/component-communication.html#!#child-to-parent">Component Interaction Cookbook</a>
|
||||||
|
|
||||||
|
`emit`在`HeroDetailComponent`和`HeroesComponent`之间的这种握手是组件间通讯的例子之一。
|
||||||
|
这不是今天的话题,我们在<a href="/docs/ts/latest/cookbook/component-communication.html#!#child-to-parent">组件间交互</a>一章中讲了一些详细信息。
|
||||||
|
|
||||||
|
:marked
|
||||||
|
Here is `HeroDetailComponent` with its new save button and the corresponding HTML.
|
||||||
|
|
||||||
这里是`HeroDetailComponent`及其新的“保存”按钮。
|
这里是`HeroDetailComponent`及其新的“保存”按钮。
|
||||||
|
|
||||||
figure.image-display
|
figure.image-display
|
||||||
img(src='/resources/images/devguide/toh/hero-details-save-button.png' alt="Hero Details With Save Button")
|
img(src='/resources/images/devguide/toh/hero-details-save-button.png' alt="Hero Details With Save Button")
|
||||||
|
|
||||||
|
+makeExcerpt('app/hero-detail.component.html', 'save')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
### Add/Delete in the *HeroesComponent*
|
### Add/Delete in the *HeroesComponent*
|
||||||
|
|
||||||
### 在*HeroesComponent`中添加/删除
|
### 在*HeroesComponent`中添加/删除
|
||||||
|
|
||||||
|
We'll be reporting propagated HTTP errors, let's start by adding the following
|
||||||
|
field to the `HeroesComponent` class:
|
||||||
|
|
||||||
|
我们将报告所出现的HTTP错误,从往`HeroesComponent`类中添加下列字段开始:
|
||||||
|
|
||||||
|
+makeExcerpt('app/heroes.component.ts', 'error', '')
|
||||||
|
|
||||||
|
:marked
|
||||||
The user can *add* a new hero by clicking a button and entering a name.
|
The user can *add* a new hero by clicking a button and entering a name.
|
||||||
|
|
||||||
用户可以通过点击一个按钮并输入英雄的名字来*添加*一个新的英雄。
|
用户可以通过点击一个按钮并输入英雄的名字来*添加*一个新的英雄。
|
||||||
|
|
||||||
When the user clicks the *Add New Hero* button, we display the `HeroDetailComponent`.
|
block add-new-hero-via-detail-comp
|
||||||
We aren't navigating to the component so it won't receive a hero `id`;
|
:marked
|
||||||
As we noted above, that is the component's cue to create and present an empty hero.
|
When the user clicks the *Add New Hero* button, we display the `HeroDetailComponent`.
|
||||||
|
We aren't navigating to the component so it won't receive a hero `id`;
|
||||||
当用户点击*Add New Hero*按钮时,我们显示`HeroDetailComponent`。我们不会导航到那个组件,所以它也不会接收到英雄的`id`;
|
as we noted above, that is the component's cue to create and present an empty hero.
|
||||||
正如我们以前提过的,组件会据此创建并展示一个空白的英雄。
|
|
||||||
|
当用户点击*Add New Hero*按钮时,我们显示`HeroDetailComponent`。我们不会导航到那个组件,所以它也不会接收到英雄的`id`;
|
||||||
Add the following HTML to the `heroes.component.html`, just below the hero list (the `*ngFor`).
|
正如我们以前提过的,组件会据此创建并展示一个空白的英雄。
|
||||||
|
|
||||||
|
- var _below = _docsFor == 'dart' ? 'before' : 'below';
|
||||||
|
:marked
|
||||||
|
Add the following to the heroes component HTML, just !{_below} the hero list (`<ul class="heroes">...</ul>`).
|
||||||
|
|
||||||
往`heroes.component.html`中,英雄列表(`*ngFor`)的紧下方,添加下列HTML。
|
往`heroes.component.html`中,英雄列表(`*ngFor`)的紧下方,添加下列HTML。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/heroes.component.html', 'add-hero', 'app/heroes.component.html (add)')(format=".")
|
+makeExcerpt('app/heroes.component.html', 'add-and-error')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
|
The first line will display an error message if there is any. The remaining HTML is for adding heroes.
|
||||||
|
|
||||||
|
如果有错误信息,它将显示在第一行。剩下的HTML是用来添加英雄的。
|
||||||
|
|
||||||
The user can *delete* an existing hero by clicking a delete button next to the hero's name.
|
The user can *delete* an existing hero by clicking a delete button next to the hero's name.
|
||||||
|
Add the following to the heroes component HTML right after the hero name in the repeated `<li>` tag:
|
||||||
|
|
||||||
用户可以通过点击英雄名后面的删除按钮来*删除*一个现存的英雄。
|
用户可以通过点击英雄名后面的删除按钮来*删除*一个现存的英雄。
|
||||||
|
|
||||||
Add the following HTML to the `heroes.component.html` right after the name in the repeated `<li>` tag:
|
|
||||||
|
|
||||||
往`heroes.component.html`中往`<li>`标签中名字的紧后面添加下列HTML代码:
|
往`heroes.component.html`中往`<li>`标签中名字的紧后面添加下列HTML代码:
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/heroes.component.html', 'delete-hero', 'app/heroes.component.html (delete)')(format=".")
|
+makeExample('app/heroes.component.html', 'delete')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Now let's fix-up the `HeroesComponent` to support the *add* and *delete* actions in the template.
|
Now let's fix-up the `HeroesComponent` to support the *add* and *delete* actions used in the template.
|
||||||
Let's start with *add*.
|
Let's start with *add*.
|
||||||
|
|
||||||
现在,让我们修改`HeroesComponent`的模板来支持*添加*和*删除*操作。
|
block heroes-comp-directives
|
||||||
让我们从*添加*开始。
|
|
||||||
|
|
||||||
We're using the `HeroDetailComponent` to capture the new hero information.
|
|
||||||
We have to tell Angular about that by importing the `HeroDetailComponent` and referencing it in the component metadata `directives` array.
|
|
||||||
|
|
||||||
我们正在用`HeroDetailComponent`捕获新英雄的信息。
|
|
||||||
为了告诉Angular这一点,我们还得导入`HeroDetailComponent`并把它加入组件元数据的`directives`数组中。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/heroes.component.ts', 'hero-detail-component', 'app/heroes.component.ts (HeroDetailComponent)')(format=".")
|
|
||||||
|
|
||||||
.l-sub-section
|
|
||||||
:marked
|
:marked
|
||||||
These are the same lines that we removed in the previous [Routing](toh-pt5.html) chapter.
|
We're using the `HeroDetailComponent` to capture the new hero information.
|
||||||
We didn't know at the time that we'd need the *HeroDetailComponent* again. So we tidied up.
|
We have to tell Angular about that by importing the `HeroDetailComponent` and referencing it in the component metadata `directives` array.
|
||||||
|
|
||||||
|
我们正在用`HeroDetailComponent`捕获新英雄的信息。
|
||||||
|
为了告诉Angular这一点,我们还得导入`HeroDetailComponent`并把它加入组件元数据的`directives`数组中。
|
||||||
|
|
||||||
|
+makeExcerpt('app/heroes.component.ts (HeroDetailComponent)', 'hero-detail-component')
|
||||||
|
|
||||||
|
.l-sub-section
|
||||||
|
:marked
|
||||||
|
These are the same lines that we removed in the previous [Routing](toh-pt5.html) chapter.
|
||||||
|
We didn't know at the time that we'd need the *HeroDetailComponent* again. So we tidied up.
|
||||||
|
|
||||||
|
同样的这些代码,我们曾在前面的[路由](toh-pt5.html)一章中移除过。
|
||||||
|
我们不知道现在会再次需要*`HeroDetailComponent`*类,所以把它们清理掉了。
|
||||||
|
|
||||||
同样的这些代码,我们曾在前面的[路由](toh-pt5.html)一章中移除过。
|
Now we *must* put these lines back. If we don't, Angular will ignore the `<my-hero-detail>`
|
||||||
我们不知道现在会再次需要*`HeroDetailComponent`*类,所以把它们清理掉了。
|
tag and pushing the *Add New Hero* button will have no visible effect.
|
||||||
|
|
||||||
|
现在,我们*必须*把这些行加回来。如果不这样,Angular将会忽略`<my-hero-detail>`标签,并且点击*Add New Hero*按钮时将不会有任何可见的效果。
|
||||||
|
|
||||||
|
:marked
|
||||||
|
Implement the click handler for the *Add New Hero* button.
|
||||||
|
|
||||||
|
然后我们实现*Add New Hero*按钮的点击事件处理器。
|
||||||
|
|
||||||
Now we *must* put these lines back. If we don't, Angular will ignore the `<my-hero-detail>`
|
+makeExcerpt('app/heroes.component.ts', 'addHero')
|
||||||
tag and pushing the *Add New Hero* button will have no visible effect.
|
|
||||||
|
|
||||||
现在,我们*必须*把这些行加回来。如果不这样,Angular将会忽略`<my-hero-detail>`标签,并且点击*Add New Hero*按钮时将不会有任何可见的效果。
|
block heroes-comp-add
|
||||||
|
:marked
|
||||||
|
The `HeroDetailComponent` does most of the work. All we do is toggle an `*ngIf` flag that
|
||||||
|
swaps it into the DOM when we add a hero and removes it from the DOM when the user is done.
|
||||||
|
|
||||||
|
`HeroDetailComponent`做了大部分工作。我们所要做的就是切换`*ngIf`的标志量:当用户添加英雄时,把它放进DOM中;当用户添加完毕时,把它从DOM中移除。
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Next we implement the click handler for the *Add New Hero* button.
|
|
||||||
|
|
||||||
接下来,我们实现*Add New Hero*按钮的点击事件处理器。
|
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/heroes.component.ts', 'add', 'app/heroes.component.ts (add)')(format=".")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
The `HeroDetailComponent` does most of the work. All we do is toggle an `*ngIf` flag that
|
|
||||||
swaps it into the DOM when we add a hero and removes it from the DOM when the user is done.
|
|
||||||
|
|
||||||
`HeroDetailComponent`做了大部分工作。我们所要做的就是切换`*ngIf`的标志量:当用户添加英雄时,把它放进DOM中;当用户添加完毕时,把它从DOM中移除。
|
|
||||||
|
|
||||||
The *delete* logic is a bit trickier.
|
The *delete* logic is a bit trickier.
|
||||||
|
|
||||||
*删除*逻辑略有点棘手。
|
*删除*逻辑略有点棘手。
|
||||||
|
|
||||||
+makeExample('toh-6/ts/app/heroes.component.ts', 'delete', 'app/heroes.component.ts (delete)')(format=".")
|
+makeExcerpt('app/heroes.component.ts', 'deleteHero')
|
||||||
|
|
||||||
:marked
|
:marked
|
||||||
Of course we delegate the persistence of hero deletion to the `HeroService`.
|
Of course we delegate the persistence of hero deletion to the `HeroService`.
|
||||||
But the component is still responsible for updating the display.
|
But the component is still responsible for updating the display.
|
||||||
So the *delete* method removes the deleted hero from the list.
|
So the *delete* method removes the deleted hero from the list.
|
||||||
|
|
||||||
固然,我们已经把删除英雄的操作委托给了`HeroService`去执行,但是该组件仍然有责任更新显示,所以这个*`delete`*方法还从列表中移除了这个被删掉的英雄。
|
block review
|
||||||
|
:marked
|
||||||
|
### Let's see it
|
||||||
|
|
||||||
|
### 我们来看看
|
||||||
|
|
||||||
|
Here are the fruits of labor in action:
|
||||||
|
|
||||||
|
下面是这些劳动成果的操作演示:
|
||||||
|
|
||||||
|
figure.image-display
|
||||||
|
img(src='/resources/images/devguide/toh/toh-http.anim.gif' alt="Heroes List Editting w/ HTTP")
|
||||||
|
|
||||||
|
.l-main-section
|
||||||
:marked
|
:marked
|
||||||
### Let's see it
|
## Application structure and code
|
||||||
|
|
||||||
### 我们来看看
|
|
||||||
|
|
||||||
Here are the fruits of labor in action:
|
|
||||||
|
|
||||||
下面是这些劳动成果的操作演示:
|
|
||||||
|
|
||||||
figure.image-display
|
|
||||||
img(src='/resources/images/devguide/toh/toh-http.anim.gif' alt="Heroes List Editting w/ HTTP")
|
|
||||||
|
|
||||||
:marked
|
|
||||||
### Review the App Structure
|
|
||||||
|
|
||||||
### 回顾应用结构
|
### 回顾应用结构
|
||||||
|
|
||||||
Let’s verify that we have the following structure after all of our good refactoring in this chapter:
|
|
||||||
|
|
||||||
经过本章的所有重构之后,我们来验证下是否已经有了下列结构:
|
|
||||||
|
|
||||||
.filetree
|
p.
|
||||||
.file angular2-tour-of-heroes
|
Review the sample source code in the #[+liveExampleLink2('', 'toh-6')] for this chapter.
|
||||||
.children
|
Verify that we have the following structure:
|
||||||
.file app
|
|
||||||
.children
|
p.
|
||||||
.file app.component.ts
|
在#[+liveExampleLink2('', 'toh-6')]中回顾本章的范例代码,验证是否得到了如下结构:
|
||||||
.file app.component.css
|
|
||||||
.file dashboard.component.css
|
block filetree
|
||||||
.file dashboard.component.html
|
.filetree
|
||||||
.file dashboard.component.ts
|
.file angular2-tour-of-heroes
|
||||||
.file hero.ts
|
.children
|
||||||
.file hero-detail.component.css
|
.file app
|
||||||
.file hero-detail.component.html
|
.children
|
||||||
.file hero-detail.component.ts
|
.file app.component.ts
|
||||||
.file hero.service.ts
|
.file app.component.css
|
||||||
.file heroes.component.css
|
.file dashboard.component.css
|
||||||
.file heroes.component.html
|
.file dashboard.component.html
|
||||||
.file heroes.component.ts
|
.file dashboard.component.ts
|
||||||
.file main.ts
|
.file hero.ts
|
||||||
.file hero-data.service.ts
|
.file hero-detail.component.css
|
||||||
.file node_modules ...
|
.file hero-detail.component.html
|
||||||
.file typings ...
|
.file hero-detail.component.ts
|
||||||
.file index.html
|
.file hero.service.ts
|
||||||
.file package.json
|
.file heroes.component.css
|
||||||
.file styles.css
|
.file heroes.component.html
|
||||||
.file sample.css
|
.file heroes.component.ts
|
||||||
.file systemjs.config.json
|
.file main.ts
|
||||||
.file tsconfig.json
|
.file in-memory-data.service.ts (new)
|
||||||
.file typings.json
|
.file node_modules ...
|
||||||
|
.file typings ...
|
||||||
|
.file index.html
|
||||||
|
.file package.json
|
||||||
|
.file sample.css (new)
|
||||||
|
.file styles.css
|
||||||
|
.file systemjs.config.json
|
||||||
|
.file tsconfig.json
|
||||||
|
.file typings.json
|
||||||
|
|
||||||
.l-main-section
|
.l-main-section
|
||||||
:marked
|
:marked
|
||||||
|
@ -607,7 +645,7 @@ figure.image-display
|
||||||
|
|
||||||
- 我们添加了在应用程序中使用Http的必备依赖。
|
- 我们添加了在应用程序中使用Http的必备依赖。
|
||||||
|
|
||||||
- We refactored HeroService to load heroes from an api.
|
- We refactored HeroService to load heroes from an API.
|
||||||
|
|
||||||
- 我们重构了HeroService,以通过api来加载英雄数据。
|
- 我们重构了HeroService,以通过api来加载英雄数据。
|
||||||
|
|
||||||
|
@ -619,26 +657,31 @@ figure.image-display
|
||||||
|
|
||||||
- 我们更新了组件,以允许用户添加、编辑和删除英雄。
|
- 我们更新了组件,以允许用户添加、编辑和删除英雄。
|
||||||
|
|
||||||
- We configured an in-memory web api.
|
- We configured an in-memory web API.
|
||||||
|
|
||||||
- 我们配置了一个内存Web API。
|
- 我们配置了一个内存Web API。
|
||||||
|
|
||||||
Below is a summary of the files we changed.
|
Below is a summary of the files we changed and added.
|
||||||
|
|
||||||
下面是我们修改之后的文件汇总。
|
下面是我们添加之后的文件汇总。
|
||||||
|
|
||||||
+makeTabs(
|
block file-summary
|
||||||
`toh-6/ts/app/app.component.ts,
|
+makeTabs(
|
||||||
toh-6/ts/app/heroes.component.ts,
|
`toh-6/ts/app/app.component.ts,
|
||||||
toh-6/ts/app/heroes.component.html,
|
toh-6/ts/app/heroes.component.ts,
|
||||||
toh-6/ts/app/hero-detail.component.ts,
|
toh-6/ts/app/heroes.component.html,
|
||||||
toh-6/ts/app/hero-detail.component.html,
|
toh-6/ts/app/hero-detail.component.ts,
|
||||||
toh-6/ts/app/hero.service.ts`,
|
toh-6/ts/app/hero-detail.component.html,
|
||||||
null,
|
toh-6/ts/app/hero.service.ts,
|
||||||
`app.comp...ts,
|
toh-6/ts/app/in-memory-data.service.ts,
|
||||||
heroes.comp...ts,
|
toh-6/ts/sample.css`,
|
||||||
heroes.comp...html,
|
null,
|
||||||
hero-detail.comp...ts,
|
`app.comp...ts,
|
||||||
hero-detail.comp...html,
|
heroes.comp...ts,
|
||||||
hero.service.ts`
|
heroes.comp...html,
|
||||||
)
|
hero-detail.comp...ts,
|
||||||
|
hero-detail.comp...html,
|
||||||
|
hero.service.ts,
|
||||||
|
in-memory-data.service.ts,
|
||||||
|
sample.css`
|
||||||
|
)
|
||||||
|
|
|
@ -0,0 +1,9 @@
|
||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
set -ex -o pipefail
|
||||||
|
|
||||||
|
npm install --no-optional
|
||||||
|
(cd public/docs/_examples && npm install)
|
||||||
|
(cd public/docs/_examples/_protractor && npm install)
|
||||||
|
npm run webdriver:update --prefix public/docs/_examples/_protractor
|
||||||
|
gulp add-example-boilerplate
|
Loading…
Reference in New Issue