Merge pull request #143 from angular/merge-26-11-16

Merge official doc updates -  26/11/2016
This commit is contained in:
Rex 2016-11-27 20:07:54 +00:00 committed by GitHub
commit 12bb596cd6
120 changed files with 3238 additions and 3128 deletions

1
.gitignore vendored
View File

@ -32,3 +32,4 @@ eplnkr.html
protractor-results.txt protractor-results.txt
link-checker-results.txt link-checker-results.txt
*a2docs.css *a2docs.css
/dist

View File

@ -10,7 +10,7 @@ env:
- DBUS_SESSION_BUS_ADDRESS=/dev/null - DBUS_SESSION_BUS_ADDRESS=/dev/null
- DISPLAY=:99.0 - DISPLAY=:99.0
- CHROME_BIN=chromium-browser - CHROME_BIN=chromium-browser
- LATEST_RELEASE=2.2.0 - LATEST_RELEASE=2.2.3
- TASK_FLAGS="--dgeni-log=warn" - TASK_FLAGS="--dgeni-log=warn"
matrix: matrix:
- TASK=lint - TASK=lint

View File

@ -823,7 +823,8 @@ gulp.task('_harp-compile', function() {
gulp.task('_shred-devguide-examples', ['_shred-clean-devguide', '_copy-example-boilerplate'], function() { gulp.task('_shred-devguide-examples', ['_shred-clean-devguide', '_copy-example-boilerplate'], function() {
// Split big shredding task into partials 2016-06-14 // Split big shredding task into partials 2016-06-14
var examplePaths = globby.sync(EXAMPLES_PATH+'/*/', {ignore: ['**/node_modules', '**/_boilerplate']}); const exPath = path.join(EXAMPLES_PATH, (argv.filter || '') + '*');
var examplePaths = globby.sync(exPath, {ignore: ['**/node_modules', '**/_boilerplate']});
var promise = Promise.resolve(true); var promise = Promise.resolve(true);
examplePaths.forEach(function (examplePath) { examplePaths.forEach(function (examplePath) {
promise = promise.then(() => docShredder.shredSingleExampleDir(_devguideShredOptions, examplePath)); promise = promise.then(() => docShredder.shredSingleExampleDir(_devguideShredOptions, examplePath));
@ -847,7 +848,7 @@ gulp.task('_shred-clean-devguide-shared-jade', function(cb) {
}); });
gulp.task('_shred-clean-devguide', function(cb) { gulp.task('_shred-clean-devguide', function(cb) {
var cleanPath = path.join(_devguideShredOptions.fragmentsDir, '**/*.*') var cleanPath = path.join(_devguideShredOptions.fragmentsDir, (argv.filter || '*') + '*/*.*')
return del([ cleanPath, '!**/*.ovr.*', '!**/_api/**']); return del([ cleanPath, '!**/*.ovr.*', '!**/_api/**']);
}); });

View File

@ -15,12 +15,7 @@
"type": "git", "type": "git",
"url": "https://github.com/angular/angular.io.git" "url": "https://github.com/angular/angular.io.git"
}, },
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"bugs": { "bugs": {
"url": "" "url": ""
}, },

View File

@ -23,7 +23,7 @@ div(class="main-footer" data-swiftype-index="false")
li <a href="/about/">About</a> li <a href="/about/">About</a>
li <a href="/about/">关于</a> li <a href="/about/">关于</a>
li <a href="/resources/#Education">Books & Training</a> li <a href="/resources/#Education">Books & Training</a>
li <a href="/resources/">书籍与培训</a> li <a href="/resources/#Education">书籍与培训</a>
li <a href="/resources/">Tools & Libraries</a> li <a href="/resources/">Tools & Libraries</a>
li <a href="/resources/">工具与库</a> li <a href="/resources/">工具与库</a>
li <a href="/resources/">Community</a> li <a href="/resources/">Community</a>

View File

@ -9,9 +9,4 @@ header(class="background-sky l-relative")
.announcement-bar-slide.clearfix .announcement-bar-slide.clearfix
img(src="/translate/cn/gdd.svg" width="64") img(src="/translate/cn/gdd.svg" width="64")
p <strong>2016 Google 开发者大会来啦!(北京 & 上海)</strong> p <strong>2016 Google 开发者大会来啦!(北京 & 上海)</strong>
a(href="http://www.google.cn/intl/zh-CN/events/developerday2016/" target="_blank" class="button md-button") 立即报名 a(href="http://www.google.cn/intl/zh-CN/events/developerday2016/" target="_blank" class="button md-button") 立即报名
.announcement-bar-slide.clearfix
img(src="/resources/images/logos/anglebrackets/devintersection.png" width="64")
p <strong>参加11月14-16号在阿姆斯特丹召开的DEVintersection</strong>
a(href="https://www.devintersectioneurope.com/#!/" target="_blank" class="button md-button") 立即报名

View File

@ -1,5 +1,7 @@
- var currentPage = false - var currentPage = false
- var nextPage = false - var nextPage = false
- var hideNextPage = false;
- var data = public.docs[current.path[1]][current.path[2]][current.path[3]]._data - var data = public.docs[current.path[1]][current.path[2]][current.path[3]]._data
for page, slug in data for page, slug in data
@ -7,13 +9,16 @@ for page, slug in data
// CHECK IF CURRENT PAGE IS SET, THEN SET NEXT PAGE // CHECK IF CURRENT PAGE IS SET, THEN SET NEXT PAGE
if currentPage if currentPage
if !nextPage && page.nextable && !page.hide if !nextPage && page.nextable && !page.hide
.l-sub-section if !hideNextPage
h3 下一步 .l-sub-section
a.translated-cn(href="/docs/#{current.path[1]}/#{current.path[2]}/#{current.path[3]}/#{slug}.html") #{page.title} h3 下一步
a.translated-cn(href="/docs/#{current.path[1]}/#{current.path[2]}/#{current.path[3]}/#{slug}.html") #{page.title}
//NEXT PAGE HAS NOW BEEN SET //NEXT PAGE HAS NOW BEEN SET
- var nextPage = true - var nextPage = true
- hideNextPage = page.hideNextPage
// SET CURRENT PAGE FLAG WHEN YOU PASS IT // SET CURRENT PAGE FLAG WHEN YOU PASS IT
if current.path[4] == slug if current.path[4] == slug
- var currentPage = true - var currentPage = true

View File

@ -40,7 +40,10 @@
- var _liveLink = 'live link'; - var _liveLink = 'live link';
- var _ngRepoURL = 'https://github.com/angular/angular'; - var _ngRepoURL = 'https://github.com/angular/angular';
- var _ngDocRepoURL = 'https://github.com/angular/angular.io'; - var _ngDocRepoURL = 'https://github.com/angular/angular.io';
- var _qsRepo = 'https://github.com/angular/quickstart/blob/master/README.md' - var _qsRepo = 'https://github.com/angular/quickstart';
- var _qsRepoZip = _qsRepo + '/archive/master.zip';
- var _npm = 'npm';
//- NgModule related //- NgModule related
- var _AppModuleVsAppComp = 'AppModule' - var _AppModuleVsAppComp = 'AppModule'

View File

@ -1,6 +0,0 @@
# Why the _fragments dir is checked in
Within this repo files generated as a result of shredding the `_examples` dir ( the contents of the `_fragments` dir) are checked in so that we can avoid running the
shredder over the entire `_examples` dir each time someone refreshes the repo ( the `shred-full` gulp task).
The gulp `serve-and-watch` shredder is only a partial shredder. It only shreds files in directories changed during
the current session.

View File

@ -1,6 +1,7 @@
{ {
"name": "angular-examples", "name": "angular-examples",
"version": "1.0.0", "version": "1.0.0",
"private": true,
"description": "Example package.json, only contains needed scripts for examples. See _examples/package.json for master package.json.", "description": "Example package.json, only contains needed scripts for examples. See _examples/package.json for master package.json.",
"scripts": { "scripts": {
"start": "tsc && concurrently \"tsc -w\" \"lite-server\" ", "start": "tsc && concurrently \"tsc -w\" \"lite-server\" ",
@ -24,12 +25,7 @@
}, },
"keywords": [], "keywords": [],
"author": "", "author": "",
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": {}, "dependencies": {},
"devDependencies": {}, "devDependencies": {},
"repository": {} "repository": {}

View File

@ -83,40 +83,10 @@ nav a.active {
background-color: #DDD; background-color: #DDD;
left: .1em; left: .1em;
} }
.items li.selected:hover {
background-color: #BBD8DC;
color: white;
}
.items .text {
position: relative;
top: -3px;
}
.items {
margin: 0 0 2em 0;
list-style-type: none;
padding: 0;
width: 24em;
}
.items li {
cursor: pointer;
position: relative;
left: 0;
background-color: #EEE;
margin: .5em;
padding: .3em 0;
height: 1.6em;
border-radius: 4px;
}
.items li:hover {
color: #607D8B;
background-color: #DDD;
left: .1em;
}
.items li.selected { .items li.selected {
background-color: #CFD8DC; background-color: #CFD8DC;
color: white; color: white;
} }
.items li.selected:hover { .items li.selected:hover {
background-color: #BBD8DC; background-color: #BBD8DC;
} }

View File

@ -6,7 +6,7 @@
"sourceMap": true, "sourceMap": true,
"emitDecoratorMetadata": true, "emitDecoratorMetadata": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"removeComments": false, "lib": ["es2015", "dom"],
"noImplicitAny": true, "noImplicitAny": true,
"suppressImplicitAnyIndexErrors": true, "suppressImplicitAnyIndexErrors": true,
"typeRoots": [ "typeRoots": [

View File

@ -1,4 +1,4 @@
'use strict'; // necessary for es6 output in node 'use strict'; // necessary for es6 output in node
import { browser, element, by } from 'protractor'; import { browser, element, by } from 'protractor';
@ -11,7 +11,7 @@ describe('AOT Compilation', function () {
it('should load page and click button', function (done) { it('should load page and click button', function (done) {
let headingSelector = element.all(by.css('h1')).get(0); let headingSelector = element.all(by.css('h1')).get(0);
expect(headingSelector.getText()).toEqual('My First Angular App'); expect(headingSelector.getText()).toEqual('Hello Angular');
expect(element.all(by.xpath('//div[text()="Magneta"]')).get(0).isPresent()).toBe(true); expect(element.all(by.xpath('//div[text()="Magneta"]')).get(0).isPresent()).toBe(true);
expect(element.all(by.xpath('//div[text()="Bombasto"]')).get(0).isPresent()).toBe(true); expect(element.all(by.xpath('//div[text()="Bombasto"]')).get(0).isPresent()).toBe(true);

View File

@ -1,6 +1,6 @@
<!-- #docregion --> <!-- #docregion -->
<button (click)="toggleHeading()">Toggle Heading</button> <button (click)="toggleHeading()">Toggle Heading</button>
<h1 *ngIf="showHeading">My First Angular App</h1> <h1 *ngIf="showHeading">Hello Angular</h1>
<h3>List of Heroes</h3> <h3>List of Heroes</h3>
<div *ngFor="let hero of heroes">{{hero}}</div> <div *ngFor="let hero of heroes">{{hero}}</div>

View File

@ -6,7 +6,7 @@
"sourceMap": true, "sourceMap": true,
"emitDecoratorMetadata": true, "emitDecoratorMetadata": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"removeComments": false, "lib": ["es2015", "dom"],
"noImplicitAny": true, "noImplicitAny": true,
"suppressImplicitAnyIndexErrors": true "suppressImplicitAnyIndexErrors": true
}, },

View File

@ -32,6 +32,7 @@ import {
AppComponent, AppComponent,
CarComponent, CarComponent,
HeroesComponent, HeroesComponent,
// #enddocregion ngmodule
HeroListComponent, HeroListComponent,
InjectorComponent, InjectorComponent,
TestComponent, TestComponent,
@ -46,6 +47,7 @@ import {
Provider8Component, Provider8Component,
Provider9Component, Provider9Component,
Provider10Component, Provider10Component,
// #docregion ngmodule
], ],
// #docregion ngmodule-providers // #docregion ngmodule-providers
providers: [ providers: [
@ -56,4 +58,3 @@ import {
bootstrap: [ AppComponent ] bootstrap: [ AppComponent ]
}) })
export class AppModule { } export class AppModule { }
// #enddocregion ngmodule

View File

@ -5,7 +5,7 @@
<div [hidden]="submitted"> <div [hidden]="submitted">
<h1>Hero Form</h1> <h1>Hero Form</h1>
<!-- #docregion ngSubmit --> <!-- #docregion ngSubmit -->
<form *ngIf="active" (ngSubmit)="onSubmit()" #heroForm="ngForm"> <form (ngSubmit)="onSubmit()" #heroForm="ngForm">
<!-- #enddocregion ngSubmit --> <!-- #enddocregion ngSubmit -->
<!-- #enddocregion edit-div --> <!-- #enddocregion edit-div -->
<div class="form-group"> <div class="form-group">
@ -16,7 +16,7 @@
[(ngModel)]="model.name" name="name" [(ngModel)]="model.name" name="name"
#name="ngModel" > #name="ngModel" >
<!-- #docregion hidden-error-msg --> <!-- #docregion hidden-error-msg -->
<div [hidden]="name.valid || name.pristine" <div [hidden]="name.valid || name.pristine"
class="alert alert-danger"> class="alert alert-danger">
<!-- #enddocregion hidden-error-msg --> <!-- #enddocregion hidden-error-msg -->
Name is required Name is required
@ -34,9 +34,9 @@
<label for="power">Hero Power</label> <label for="power">Hero Power</label>
<select class="form-control" id="power" <select class="form-control" id="power"
required required
[(ngModel)]="model.power" name="power" [(ngModel)]="model.power" name="power"
#power="ngModel" > #power="ngModel" >
<option *ngFor="let p of powers" [value]="p">{{p}}</option> <option *ngFor="let pow of powers" [value]="pow">{{pow}}</option>
</select> </select>
<div [hidden]="power.valid || power.pristine" class="alert alert-danger"> <div [hidden]="power.valid || power.pristine" class="alert alert-danger">
Power is required Power is required
@ -46,12 +46,18 @@
<!-- #docregion submit-button --> <!-- #docregion submit-button -->
<button type="submit" class="btn btn-default" [disabled]="!heroForm.form.valid">Submit</button> <button type="submit" class="btn btn-default" [disabled]="!heroForm.form.valid">Submit</button>
<!-- #enddocregion submit-button --> <!-- #enddocregion submit-button -->
<!-- #docregion new-hero-button-form-reset -->
<!-- #docregion new-hero-button --> <button type="button" class="btn btn-default" (click)="newHero(); heroForm.reset()">New Hero</button>
<button type="button" class="btn btn-default" (click)="newHero()">New Hero</button> <!-- #enddocregion new-hero-button-form-reset -->
<!-- #enddocregion new-hero-button -->
<!-- #enddocregion final --> <!-- #enddocregion final -->
<i>with</i> reset
&nbsp;&nbsp;
<!-- #docregion new-hero-button-no-reset -->
<button type="button" class="btn btn-default" (click)="newHero()">New Hero</button>
<!-- #enddocregion new-hero-button-no-reset -->
<i>without</i> reset
<!-- NOT SHOWN IN DOCS --> <!-- NOT SHOWN IN DOCS -->
<div> <div>
<hr> <hr>
@ -126,7 +132,7 @@
<div class="form-group"> <div class="form-group">
<label for="power">Hero Power</label> <label for="power">Hero Power</label>
<select class="form-control" id="power" required> <select class="form-control" id="power" required>
<option *ngFor="let p of powers" [value]="p">{{p}}</option> <option *ngFor="let pow of powers" [value]="pow">{{pow}}</option>
</select> </select>
</div> </div>
@ -165,7 +171,7 @@
<select class="form-control" id="power" <select class="form-control" id="power"
required required
[(ngModel)]="model.power" name="power"> [(ngModel)]="model.power" name="power">
<option *ngFor="let p of powers" [value]="p">{{p}}</option> <option *ngFor="let pow of powers" [value]="pow">{{pow}}</option>
</select> </select>
</div> </div>
@ -178,12 +184,12 @@
<!-- EXTRA MATERIAL FOR DOCUMENTATION --> <!-- EXTRA MATERIAL FOR DOCUMENTATION -->
<hr> <hr>
<!-- #docregion ngModel-1--> <!-- #docregion ngModelName-1 -->
<input type="text" class="form-control" id="name" <input type="text" class="form-control" id="name"
required required
[(ngModel)]="model.name" name="name"> [(ngModel)]="model.name" name="name">
TODO: remove this: {{model.name}} TODO: remove this: {{model.name}}
<!-- #enddocregion ngModel-1--> <!-- #enddocregion ngModelName-1 -->
<hr> <hr>
<!-- #docregion ngModel-3--> <!-- #docregion ngModel-3-->
<input type="text" class="form-control" id="name" <input type="text" class="form-control" id="name"
@ -193,16 +199,6 @@
TODO: remove this: {{model.name}} TODO: remove this: {{model.name}}
<!-- #enddocregion ngModel-3--> <!-- #enddocregion ngModel-3-->
<hr> <hr>
<!-- #docregion form-active -->
<form *ngIf="active">
<!-- #enddocregion form-active -->
<!-- #docregion ngModelName-1 -->
<input type="text" class="form-control" id="name"
required
[(ngModel)]="model.name" name="name" >
<!-- #enddocregion ngModelName-1 -->
<hr>
<!-- #docregion ngModelName-2 --> <!-- #docregion ngModelName-2 -->
<input type="text" class="form-control" id="name" <input type="text" class="form-control" id="name"
required required
@ -210,6 +206,5 @@
#spy > #spy >
<br>TODO: remove this: {{spy.className}} <br>TODO: remove this: {{spy.className}}
<!-- #enddocregion ngModelName-2 --> <!-- #enddocregion ngModelName-2 -->
</form>
</div> </div>

View File

@ -29,22 +29,10 @@ export class HeroFormComponent {
// #enddocregion first // #enddocregion first
// #docregion final // #docregion final
// Reset the form with a new hero AND restore 'pristine' class state
// by toggling 'active' flag which causes the form
// to be removed/re-added in a tick via NgIf
// TODO: Workaround until NgForm has a reset method (#6822)
// #docregion new-hero // #docregion new-hero
active = true;
// #docregion new-hero-v1
newHero() { newHero() {
this.model = new Hero(42, '', ''); this.model = new Hero(42, '', '');
// #enddocregion new-hero-v1
this.active = false;
setTimeout(() => this.active = true, 0);
// #docregion new-hero-v1
} }
// #enddocregion new-hero-v1
// #enddocregion new-hero // #enddocregion new-hero
// #enddocregion final // #enddocregion final
//////// NOT SHOWN IN DOCS //////// //////// NOT SHOWN IN DOCS ////////

View File

@ -2,7 +2,4 @@
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app.module'; import { AppModule } from './app.module';
// Compiles the module (asynchronously) with the runtime compiler
// which generates a compiled module factory in memory.
// Then bootstraps with that factory, targeting the browser.
platformBrowserDynamic().bootstrapModule(AppModule); platformBrowserDynamic().bootstrapModule(AppModule);

View File

@ -1,6 +1,7 @@
{ {
"name": "angular-examples-master", "name": "angular-examples-master",
"version": "1.0.0", "version": "1.0.0",
"private": true,
"description": "Master package.json, the superset of all dependencies for all of the _example package.json files. See _boilerplate/package.json for example npm scripts.", "description": "Master package.json, the superset of all dependencies for all of the _example package.json files. See _boilerplate/package.json for example npm scripts.",
"scripts": { "scripts": {
"protractor": "protractor", "protractor": "protractor",
@ -8,12 +9,7 @@
}, },
"keywords": [], "keywords": [],
"author": "", "author": "",
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": { "dependencies": {
"@angular/common": "~2.2.0", "@angular/common": "~2.2.0",
"@angular/compiler": "~2.2.0", "@angular/compiler": "~2.2.0",
@ -27,7 +23,7 @@
"@angular/router": "~3.2.0", "@angular/router": "~3.2.0",
"@angular/upgrade": "~2.2.0", "@angular/upgrade": "~2.2.0",
"angular-in-memory-web-api": "~0.1.15", "angular-in-memory-web-api": "~0.1.16",
"core-js": "^2.4.1", "core-js": "^2.4.1",
"reflect-metadata": "^0.1.8", "reflect-metadata": "^0.1.8",
@ -46,7 +42,6 @@
"@types/angular-resource": "^1.5.6", "@types/angular-resource": "^1.5.6",
"@types/angular-route": "^1.3.2", "@types/angular-route": "^1.3.2",
"@types/angular-sanitize": "^1.3.3", "@types/angular-sanitize": "^1.3.3",
"@types/core-js": "^0.9.34",
"@types/jasmine": "~2.5.36", "@types/jasmine": "~2.5.36",
"@types/node": "^6.0.45", "@types/node": "^6.0.45",
"@types/selenium-webdriver": "^2.53.32", "@types/selenium-webdriver": "^2.53.32",

View File

@ -1,11 +1,9 @@
// #docregion , import // #docregion
import 'package:angular2/core.dart'; import 'package:angular2/core.dart';
// #enddocregion import
// #docregion metadata
@Component( @Component(
selector: 'my-app', selector: 'my-app',
template: '<h1>Hello Angular!</h1>') template: '<h1>Hello {{name}}</h1>')
// #enddocregion metadata class AppComponent {
// #docregion class var name = 'Angular';
class AppComponent {} }

View File

@ -2,7 +2,9 @@
<!DOCTYPE html> <!DOCTYPE html>
<html> <html>
<head> <head>
<title>Getting Started</title> <title>Hello Angular</title>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="styles.css"> <link rel="stylesheet" href="styles.css">
<!-- #docregion loaddart --> <!-- #docregion loaddart -->
@ -11,6 +13,8 @@
<!-- #enddocregion loaddart --> <!-- #enddocregion loaddart -->
</head> </head>
<body> <body>
<my-app>Loading...</my-app> <!-- #docregion my-app-->
<my-app>Loading AppComponent content here ...</my-app>
<!-- #enddocregion my-app-->
</body> </body>
</html> </html>

View File

@ -4,7 +4,7 @@ import { browser, element, by } from 'protractor';
describe('QuickStart E2E Tests', function () { describe('QuickStart E2E Tests', function () {
let expectedMsg = 'Hello Angular!'; let expectedMsg = 'Hello Angular';
beforeEach(function () { beforeEach(function () {
browser.get(''); browser.get('');

View File

@ -10,7 +10,7 @@
ng.core.Component({ ng.core.Component({
// #enddocregion ng-namespace-funcs // #enddocregion ng-namespace-funcs
selector: 'my-app', selector: 'my-app',
template: '<h1>Hello Angular!</h1>' template: '<h1>Hello Angular</h1>'
// #docregion ng-namespace-funcs // #docregion ng-namespace-funcs
}) })
// #enddocregion component // #enddocregion component

View File

@ -5,12 +5,7 @@
"start": "npm run lite", "start": "npm run lite",
"lite": "lite-server" "lite": "lite-server"
}, },
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": { "dependencies": {
"@angular/common": "~2.2.0", "@angular/common": "~2.2.0",
"@angular/compiler": "~2.2.0", "@angular/compiler": "~2.2.0",

View File

@ -1 +0,0 @@
!systemjs.config.1.js

View File

@ -1,14 +1,8 @@
// #docregion // #docregion
// #docregion import
import { Component } from '@angular/core'; import { Component } from '@angular/core';
// #enddocregion import
// #docregion metadata
@Component({ @Component({
selector: 'my-app', selector: 'my-app',
template: '<h1>Hello Angular!</h1>' template: `<h1>Hello {{name}}</h1>`
}) })
// #enddocregion metadata export class AppComponent { name = 'Angular'; }
// #docregion class
export class AppComponent { }
// #enddocregion class

View File

@ -1,13 +1,11 @@
// #docregion // #docregion
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser'; import { BrowserModule } from '@angular/platform-browser';
import { AppComponent } from './app.component';
import { AppComponent } from './app.component';
@NgModule({ @NgModule({
imports: [ BrowserModule ], imports: [ BrowserModule ],
declarations: [ AppComponent ], declarations: [ AppComponent ],
bootstrap: [ AppComponent ] bootstrap: [ AppComponent ]
}) })
export class AppModule { } export class AppModule { }

View File

@ -1,9 +1,5 @@
// #docregion
// #docregion import
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic'; import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app.module'; import { AppModule } from './app.module';
// #enddocregion import
const platform = platformBrowserDynamic(); platformBrowserDynamic().bootstrapModule(AppModule);
platform.bootstrapModule(AppModule);

View File

@ -1,37 +1,36 @@
<!DOCTYPE html> <!DOCTYPE html>
<!-- #docregion -->
<html> <html>
<head> <head>
<title>Angular QuickStart</title> <title>Hello Angular</title>
<meta charset="UTF-8"> <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1"> <meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="styles.css"> <style>
body {color:#369;font-family: Arial,Helvetica,sans-serif;}
</style>
<!-- 1. Load libraries --> <!-- Polyfills for older browsers -->
<!-- #docregion libraries -->
<!-- #docregion polyfills --> <!-- #docregion polyfills -->
<!-- Polyfill for older browsers -->
<script src="node_modules/core-js/client/shim.min.js"></script> <script src="node_modules/core-js/client/shim.min.js"></script>
<!-- #enddocregion polyfills --> <!-- #enddocregion polyfills -->
<script src="node_modules/zone.js/dist/zone.js"></script> <script src="node_modules/zone.js/dist/zone.js"></script>
<script src="node_modules/reflect-metadata/Reflect.js"></script> <script src="node_modules/reflect-metadata/Reflect.js"></script>
<script src="node_modules/systemjs/dist/system.src.js"></script> <script src="node_modules/systemjs/dist/system.src.js"></script>
<!-- #enddocregion libraries -->
<!-- 2. Configure SystemJS --> <!-- #docregion autobootstrap-->
<!-- #docregion systemjs --> <script> window.autoBootstrap = true; </script>
<!-- #enddocregion autobootstrap-->
<script src="systemjs.config.js"></script> <script src="systemjs.config.js"></script>
<script> <script>
System.import('app').catch(function(err){ console.error(err); }); System.import('app').catch(function(err){ console.error(err); });
</script> </script>
<!-- #enddocregion systemjs -->
</head> </head>
<!-- 3. Display the application -->
<!-- #docregion my-app -->
<body> <body>
<my-app>Loading...</my-app> <!-- #docregion my-app-->
<my-app>Loading AppComponent content here ...</my-app>
<!-- #enddocregion my-app-->
</body> </body>
<!-- #enddocregion my-app -->
</html> </html>

View File

@ -7,12 +7,7 @@
"tsc": "tsc", "tsc": "tsc",
"tsc:w": "tsc -w" "tsc:w": "tsc -w"
}, },
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": { "dependencies": {
"@angular/common": "~2.2.0", "@angular/common": "~2.2.0",
"@angular/compiler": "~2.2.0", "@angular/compiler": "~2.2.0",

View File

@ -1,9 +1,9 @@
{ {
"description": "QuickStart", "description": "QuickStart",
"files": [ "files": [
"!**/*.d.ts", "app/app.component.ts",
"!**/*.js", "index.html"
"!**/*.[1].*"
], ],
"open": "app/app.component.ts",
"tags": ["quickstart"] "tags": ["quickstart"]
} }

View File

@ -6,7 +6,7 @@
"sourceMap": true, "sourceMap": true,
"emitDecoratorMetadata": true, "emitDecoratorMetadata": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"removeComments": false, "lib": ["es2015", "dom"],
"noImplicitAny": false "noImplicitAny": false
} }
} }

View File

@ -0,0 +1,17 @@
'use strict'; // necessary for es6 output in node
import { browser, element, by } from 'protractor';
describe('QuickStart E2E Tests', function () {
let expectedMsg = 'Hello Angular';
beforeEach(function () {
browser.get('');
});
it(`should display: ${expectedMsg}`, function () {
expect(element(by.css('h1')).getText()).toEqual(expectedMsg);
});
});

View File

@ -0,0 +1,35 @@
/* tslint:disable:no-unused-variable */
import { AppComponent } from './app.component';
import { async, ComponentFixture, TestBed } from '@angular/core/testing';
import { By } from '@angular/platform-browser';
import { DebugElement } from '@angular/core';
//////// SPECS /////////////
describe('AppComponent', function () {
let de: DebugElement;
let comp: AppComponent;
let fixture: ComponentFixture<AppComponent>;
beforeEach(async(() => {
TestBed.configureTestingModule({
declarations: [ AppComponent ]
})
.compileComponents();
}));
beforeEach(() => {
fixture = TestBed.createComponent(AppComponent);
comp = fixture.componentInstance;
de = fixture.debugElement.query(By.css('h1'));
});
it('should create component', () => expect(comp).toBeDefined() );
it('should have expected <h1> text', () => {
fixture.detectChanges();
const h1 = de.nativeElement;
expect(h1.innerText).toMatch(/angular/i,
'<h1> should say something about "Angular"');
});
});

View File

@ -0,0 +1,8 @@
// #docregion
import { Component } from '@angular/core';
@Component({
selector: 'my-app',
template: `<h1>Hello {{name}}</h1>`
})
export class AppComponent { name = 'Angular'; }

View File

@ -1,8 +1,11 @@
// #docregion // #docregion
import { NgModule } from '@angular/core'; import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser'; import { BrowserModule } from '@angular/platform-browser';
import { AppComponent } from './app.component';
@NgModule({ @NgModule({
imports: [ BrowserModule ] imports: [ BrowserModule ],
declarations: [ AppComponent ],
bootstrap: [ AppComponent ]
}) })
export class AppModule { } export class AppModule { }

View File

@ -0,0 +1,5 @@
// #docregion
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app.module';
platformBrowserDynamic().bootstrapModule(AppModule);

View File

@ -0,0 +1,31 @@
<!DOCTYPE html>
<!-- #docregion -->
<html>
<head>
<title>Hello Angular</title>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="stylesheet" href="styles.css">
<style>
body {color:#369;font-family: Arial,Helvetica,sans-serif;}
</style>
<!-- Polyfills for older browsers -->
<script src="node_modules/core-js/client/shim.min.js"></script>
<script src="node_modules/zone.js/dist/zone.js"></script>
<script src="node_modules/reflect-metadata/Reflect.js"></script>
<script src="node_modules/systemjs/dist/system.src.js"></script>
<script src="systemjs.config.js"></script>
<script>
System.import('app').catch(function(err){ console.error(err); });
</script>
</head>
<body>
<!-- #docregion my-app-->
<my-app><!-- content managed by Angular --></my-app>
<!-- #enddocregion my-app-->
</body>
</html>

View File

@ -0,0 +1,11 @@
{
"description": "QuickStart Setup",
"files": [
"app/app.component.ts",
"app/app.module.ts",
"app/main.ts",
"index.html"
],
"open": "app/app.component.ts",
"tags": ["quickstart setup"]
}

View File

@ -7,12 +7,7 @@
"lite": "lite-server", "lite": "lite-server",
"start": "concurrently \"npm run tsc:w\" \"npm run lite\" " "start": "concurrently \"npm run tsc:w\" \"npm run lite\" "
}, },
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": { "dependencies": {
"angular2": "2.0.0-beta.0", "angular2": "2.0.0-beta.0",
"systemjs": "0.19.6", "systemjs": "0.19.6",

View File

@ -2,7 +2,7 @@
<h3>Top Heroes</h3> <h3>Top Heroes</h3>
<div class="grid grid-pad"> <div class="grid grid-pad">
<!-- #docregion click --> <!-- #docregion click -->
<a *ngFor="let hero of heroes" [routerLink]="['./HeroDetail', {id: hero.id}]" class="col-1-4"> <a *ngFor="let hero of heroes" [routerLink]="['HeroDetail', {id: hero.id.toString()}]" class="col-1-4">
<!-- #enddocregion click --> <!-- #enddocregion click -->
<div class="module hero"> <div class="module hero">
<h4>{{hero.name}}</h4> <h4>{{hero.name}}</h4>

View File

@ -37,15 +37,13 @@ class HeroDetailComponent implements OnInit {
// #docregion ngOnInit // #docregion ngOnInit
Future<Null> ngOnInit() async { Future<Null> ngOnInit() async {
var idString = _routeParams.get('id'); var _id = _routeParams.get('id');
var id = int.parse(idString ?? '', onError: (_) => null); var id = int.parse(_id ?? '', onError: (_) => null);
if (id != null) hero = await (_heroService.getHero(id)); if (id != null) hero = await (_heroService.getHero(id));
} }
// #enddocregion ngOnInit // #enddocregion ngOnInit
// #docregion goBack // #docregion goBack
void goBack() { void goBack() => _location.back();
_location.back();
}
// #enddocregion goBack // #enddocregion goBack
} }

View File

@ -13,7 +13,7 @@ import 'hero_search_component.dart';
selector: 'my-dashboard', selector: 'my-dashboard',
templateUrl: 'dashboard_component.html', templateUrl: 'dashboard_component.html',
styleUrls: const ['dashboard_component.css'], styleUrls: const ['dashboard_component.css'],
directives: const [HeroSearchComponent]) directives: const [HeroSearchComponent, ROUTER_DIRECTIVES])
// #enddocregion search // #enddocregion search
class DashboardComponent implements OnInit { class DashboardComponent implements OnInit {
List<Hero> heroes; List<Hero> heroes;

View File

@ -1,7 +1,7 @@
<!-- #docregion --> <!-- #docregion -->
<h3>Top Heroes</h3> <h3>Top Heroes</h3>
<div class="grid grid-pad"> <div class="grid grid-pad">
<a *ngFor="let hero of heroes" [routerLink]="['/detail', hero.id]" class="col-1-4"> <a *ngFor="let hero of heroes" [routerLink]="['HeroDetail', {id: hero.id.toString()}]" class="col-1-4">
<div class="module hero"> <div class="module hero">
<h4>{{hero.name}}</h4> <h4>{{hero.name}}</h4>
</div> </div>

View File

@ -1,10 +1,10 @@
// #docplaster // #docplaster
// #docregion , v2 // #docregion , v2
import 'dart:async'; import 'dart:async';
import 'dart:html';
import 'package:angular2/core.dart'; import 'package:angular2/core.dart';
import 'package:angular2/router.dart'; import 'package:angular2/router.dart';
import 'package:angular2/platform/common.dart';
import 'hero.dart'; import 'hero.dart';
import 'hero_service.dart'; import 'hero_service.dart';
@ -18,12 +18,13 @@ class HeroDetailComponent implements OnInit {
Hero hero; Hero hero;
final HeroService _heroService; final HeroService _heroService;
final RouteParams _routeParams; final RouteParams _routeParams;
final Location _location;
HeroDetailComponent(this._heroService, this._routeParams); HeroDetailComponent(this._heroService, this._routeParams, this._location);
Future<Null> ngOnInit() async { Future<Null> ngOnInit() async {
var idString = _routeParams.get('id'); var _id = _routeParams.get('id');
var id = int.parse(idString, onError: (_) => null); var id = int.parse(_id ?? '', onError: (_) => null);
if (id != null) hero = await (_heroService.getHero(id)); if (id != null) hero = await (_heroService.getHero(id));
} }
@ -34,7 +35,5 @@ class HeroDetailComponent implements OnInit {
} }
// #enddocregion save // #enddocregion save
void goBack() { void goBack() => _location.back();
window.history.back();
}
} }

View File

@ -6,7 +6,7 @@
"sourceMap": true, "sourceMap": true,
"emitDecoratorMetadata": true, "emitDecoratorMetadata": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"removeComments": false, "lib": ["es2015", "dom"],
"noImplicitAny": true, "noImplicitAny": true,
"suppressImplicitAnyIndexErrors": true, "suppressImplicitAnyIndexErrors": true,
"typeRoots": [ "typeRoots": [

View File

@ -8,7 +8,7 @@
"sourceMap": true, "sourceMap": true,
"emitDecoratorMetadata": true, "emitDecoratorMetadata": true,
"experimentalDecorators": true, "experimentalDecorators": true,
"removeComments": false, "lib": ["es2015", "dom"],
"noImplicitAny": true, "noImplicitAny": true,
"suppressImplicitAnyIndexErrors": true, "suppressImplicitAnyIndexErrors": true,
"typeRoots": [ "typeRoots": [

View File

@ -4,12 +4,7 @@
"version": "0.0.0", "version": "0.0.0",
"description": "A tutorial application for AngularJS", "description": "A tutorial application for AngularJS",
"repository": "https://github.com/angular/angular-phonecat", "repository": "https://github.com/angular/angular-phonecat",
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"devDependencies": { "devDependencies": {
"bower": "^1.7.7", "bower": "^1.7.7",
"http-server": "^0.9.0", "http-server": "^0.9.0",

View File

@ -20,15 +20,14 @@ export class KeyUpComponent_v1 {
// #enddocregion key-up-component-1-class, key-up-component-1-class-no-type // #enddocregion key-up-component-1-class, key-up-component-1-class-no-type
/* /*
// #docregion key-up-component-1-class-no-type // #docregion key-up-component-1-class-no-type
// without strong typing onKey(event:any) { // without type info
onKey(event:any) {
this.values += event.target.value + ' | '; this.values += event.target.value + ' | ';
} }
// #enddocregion key-up-component-1-class-no-type // #enddocregion key-up-component-1-class-no-type
*/ */
// #docregion key-up-component-1-class // #docregion key-up-component-1-class
// with strong typing
onKey(event: KeyboardEvent) { onKey(event: KeyboardEvent) { // with type info
this.values += (<HTMLInputElement>event.target).value + ' | '; this.values += (<HTMLInputElement>event.target).value + ' | ';
} }
// #docregion key-up-component-1-class-no-type // #docregion key-up-component-1-class-no-type
@ -53,23 +52,22 @@ export class KeyUpComponent_v2 {
} }
// #enddocregion key-up-component-2 // #enddocregion key-up-component-2
////////////////////////////////////////// //////////////////////////////////////////
// #docregion key-up-component-3 // #docregion key-up-component-3
@Component({ @Component({
selector: 'key-up3', selector: 'key-up3',
template: ` template: `
<input #box (keyup.enter)="values=box.value"> <input #box (keyup.enter)="onEnter(box.value)">
<p>{{values}}</p> <p>{{value}}</p>
` `
}) })
export class KeyUpComponent_v3 { export class KeyUpComponent_v3 {
values = ''; value = '';
onEnter(value: string) { this.value = value; }
} }
// #enddocregion key-up-component-3 // #enddocregion key-up-component-3
////////////////////////////////////////// //////////////////////////////////////////
// #docregion key-up-component-4 // #docregion key-up-component-4
@ -77,13 +75,14 @@ export class KeyUpComponent_v3 {
selector: 'key-up4', selector: 'key-up4',
template: ` template: `
<input #box <input #box
(keyup.enter)="values=box.value" (keyup.enter)="update(box.value)"
(blur)="values=box.value"> (blur)="update(box.value)">
<p>{{values}}</p> <p>{{value}}</p>
` `
}) })
export class KeyUpComponent_v4 { export class KeyUpComponent_v4 {
values = ''; value = '';
update(value: string) { this.value = value; }
} }
// #enddocregion key-up-component-4 // #enddocregion key-up-component-4

View File

@ -7,12 +7,7 @@
"test": "karma start", "test": "karma start",
"build": "rimraf dist && webpack --config config/webpack.prod.js --progress --profile --bail" "build": "rimraf dist && webpack --config config/webpack.prod.js --progress --profile --bail"
}, },
"licenses": [ "license": "MIT",
{
"type": "MIT",
"url": "https://github.com/angular/angular.io/blob/master/LICENSE"
}
],
"dependencies": { "dependencies": {
"@angular/common": "~2.2.0", "@angular/common": "~2.2.0",
"@angular/compiler": "~2.2.0", "@angular/compiler": "~2.2.0",
@ -27,7 +22,6 @@
"zone.js": "^0.6.25" "zone.js": "^0.6.25"
}, },
"devDependencies": { "devDependencies": {
"@types/core-js": "^0.9.34",
"@types/node": "^6.0.45", "@types/node": "^6.0.45",
"@types/jasmine": "^2.5.35", "@types/jasmine": "^2.5.35",
"angular2-template-loader": "^0.4.0", "angular2-template-loader": "^0.4.0",

View File

@ -7,12 +7,19 @@
"banner": "AngularDart is <b>2.0</b>. View the <a href='https://github.com/dart-lang/angular2/blob/master/CHANGELOG.md' target='_blank'>change log</a> to see enhancements, fixes, and breaking changes." "banner": "AngularDart is <b>2.0</b>. View the <a href='https://github.com/dart-lang/angular2/blob/master/CHANGELOG.md' target='_blank'>change log</a> to see enhancements, fixes, and breaking changes."
}, },
"cli-quickstart": {
"icon": "query-builder",
"title": "CLI Quickstart",
"subtitle": "TypeScript",
"description": "Use the CLI tool to quickly build Angular applications",
"hide": true
},
"quickstart": { "quickstart": {
"icon": "query-builder", "icon": "query-builder",
"title": "Quickstart", "title": "Quickstart",
"subtitle": "Dart", "subtitle": "Dart",
"description": "Get up and running with Angular", "banner": "A quick look at Angular basics"
"banner": "This QuickStart guide demonstrates how to build and run a simple Angular application."
}, },
"tutorial": { "tutorial": {

View File

@ -22,7 +22,10 @@ include ../../../_includes/_util-fns
- var _ngRepoURL = 'https://github.com/dart-lang/angular2'; - var _ngRepoURL = 'https://github.com/dart-lang/angular2';
//- Don't override this value quite yet: //- Don't override this value quite yet:
//- var _ngDocRepoURL = 'https://github.com/dart-lang/site-webdev'; //- var _ngDocRepoURL = 'https://github.com/dart-lang/site-webdev';
- var _qsRepo = 'https://github.com/angular-examples/quickstart/archive/master.zip' - var _qsRepo = 'https://github.com/angular-examples/quickstart'
- var _qsRepoZip = _qsRepo + '/archive/master.zip';
- var _npm = 'pub';
//- NgModule related //- NgModule related
- var _AppModuleVsAppComp = 'AppComponent' - var _AppModuleVsAppComp = 'AppComponent'

View File

@ -19,11 +19,13 @@ block annotation-defn
block bootstrap-defn-top block bootstrap-defn-top
:marked :marked
We launch an Angular application by "bootstrapping" it with the You launch an Angular application by "bootstrapping" it with the
[bootstrap][bootstrap] method. The `bootstrap` method identifies an [bootstrap][bootstrap] method. Bootstraping identifies an
application's top level "root" [Component](#component) and optionally application's top level "root" [component](#component), which is
the first component that is loaded for the application, and optionally
registers service [providers](#provider) with the [dependency injection registers service [providers](#provider) with the [dependency injection
system](#dependency-injection). system](#dependency-injection).
For more information, see the [Setup](!{docsLatest}/guide/setup.html) page.
[bootstrap]: !{docsLatest}/api/angular2.platform.browser/bootstrap.html [bootstrap]: !{docsLatest}/api/angular2.platform.browser/bootstrap.html
@ -42,17 +44,6 @@ block module-defn
the chapter on "Libraries and Scripts" in the the chapter on "Libraries and Scripts" in the
[Dart Language Specification](https://www.dartlang.org/docs/spec/). [Dart Language Specification](https://www.dartlang.org/docs/spec/).
block routing-component-defn
:marked
A [Component](#component) with an attached router.
In most cases, the component became attached to a [router](#router) by means
of a `@RouterConfig` #{decorator} that defined routes to views controlled by this component.
The component's template has a `RouterOutlet` element where it can display views produced by the router.
It likely has anchor tags or buttons with `RouterLink` directives that users can click to navigate.
block append snake-case-defn block append snake-case-defn
:marked :marked
Library and file names are often spelled in snake_case. Examples include: Library and file names are often spelled in snake_case. Examples include:
@ -63,6 +54,6 @@ block zone-defn
Zones are a mechanism for encapsulating and intercepting Zones are a mechanism for encapsulating and intercepting
a Dart application's asynchronous activity. a Dart application's asynchronous activity.
To learn more, consult the [zones article][zones]. Learn more about zones in this [article][zones].
[zones]: https://www.dartlang.org/articles/libraries/zones [zones]: https://www.dartlang.org/articles/libraries/zones

View File

@ -2,11 +2,29 @@
"index": { "index": {
"title": "Documentation Overview", "title": "Documentation Overview",
"navTitle": "Overview", "navTitle": "Overview",
"description": "How to read and use this documentation", "intro": "How to read and use this documentation",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
"setup": {
"title": "Setup for local development",
"navTitle": "Setup",
"intro": "Install the Angular QuickStart seed for faster, more efficient development on your machine",
"nextable": true,
"hideNextPage": true,
"basics": true
},
"learning-angular": {
"title": "Learning Angular",
"navTitle": "Learning Angular",
"intro": "A suggested path through the documentation for Angular newcomers",
"nextable": true,
"hideNextPage": true,
"basics": true
},
"architecture": { "architecture": {
"title": "Architecture Overview", "title": "Architecture Overview",
"navTitle": "Architecture", "navTitle": "Architecture",
@ -17,7 +35,7 @@
"displaying-data": { "displaying-data": {
"title": "Displaying Data", "title": "Displaying Data",
"intro": "Interpolation and other forms of property binding help us show app data in the UI.", "intro": "Property binding helps show app data in the UI.",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
@ -63,16 +81,29 @@
"basics": true "basics": true
}, },
"glossary": {
"title": "Glossary",
"intro": "Brief definitions of the most important words in the Angular vocabulary",
"basics": true
},
"change-log": {
"hide": true,
"title": "Change Log",
"intro": "An annotated history of recent documentation improvements.",
"basics": true
},
"ngmodule": { "ngmodule": {
"hide": true,
"title": "Angular Modules (NgModule)", "title": "Angular Modules (NgModule)",
"intro": "Define application modules with @NgModule", "intro": "Define application modules with @NgModule"
"hide": true
}, },
"animations": { "animations": {
"hide": true,
"title": "Animations", "title": "Animations",
"intro": "A guide to Angular's animation system.", "intro": "A guide to Angular's animation system."
"hide": true
}, },
"attribute-directives": { "attribute-directives": {
@ -90,12 +121,6 @@
"intro": "Learn how to apply CSS styles to components." "intro": "Learn how to apply CSS styles to components."
}, },
"glossary": {
"title": "Glossary",
"intro": "Angular中最重要的词汇的简要定义",
"basics": true
},
"hierarchical-dependency-injection": { "hierarchical-dependency-injection": {
"title": "Hierarchical Dependency Injectors", "title": "Hierarchical Dependency Injectors",
"navTitle": "Hierarchical Injectors", "navTitle": "Hierarchical Injectors",
@ -104,7 +129,7 @@
"server-communication": { "server-communication": {
"title": "HTTP Client", "title": "HTTP Client",
"intro": "Talk to a remote server with an HTTP Client." "intro": "Use an HTTP Client to talk to a remote server."
}, },
"lifecycle-hooks": { "lifecycle-hooks": {
@ -113,9 +138,9 @@
}, },
"npm-packages": { "npm-packages": {
"hide": true,
"title": "Npm Packages", "title": "Npm Packages",
"intro": "Details of the recommended npm packages and the different kinds of package dependencies", "intro": "Recommended npm packages, and how to specify package dependencies"
"hide": true
}, },
"pipes": { "pipes": {
@ -125,7 +150,7 @@
"router": { "router": {
"title": "Routing & Navigation", "title": "Routing & Navigation",
"intro": "Discover the basics of screen navigation with the Angular Component Router." "intro": "Discover the basics of screen navigation with the Angular Router."
}, },
"security": { "security": {
@ -133,32 +158,38 @@
"intro": "Developing for content security in Angular applications" "intro": "Developing for content security in Angular applications"
}, },
"setup-systemjs-anatomy": {
"hide": true,
"title": "Setup Anatomy",
"intro": "Inside the local development environment for SystemJS"
},
"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."
}, },
"testing": { "testing": {
"hide": true,
"title": "Testing", "title": "Testing",
"intro": "Techniques and practices for testing an Angular app", "intro": "Techniques and practices for testing an Angular app"
"hide": true
}, },
"typescript-configuration": { "typescript-configuration": {
"hide": true,
"title": "TypeScript Configuration", "title": "TypeScript Configuration",
"intro": "TypeScript configuration for Angular developers", "intro": "TypeScript configuration for Angular developers"
"hide": true
}, },
"upgrade": { "upgrade": {
"hide": true,
"title": "Upgrading from 1.x", "title": "Upgrading from 1.x",
"intro": "Angular 1 applications can be incrementally upgraded to Angular 2.", "intro": "Incrementally upgrade an Angular 1 application to Angular 2."
"hide": true
}, },
"webpack": { "webpack": {
"hide": true,
"title": "Webpack: an introduction", "title": "Webpack: an introduction",
"intro": "Create your Angular applications with a Webpack based tooling", "intro": "Create Angular applications with a Webpack based tooling"
"hide": true
} }
} }

View File

@ -10,6 +10,34 @@ block ctor-syntax
We also leveraged Dart's constructor syntax for declaring parameters and We also leveraged Dart's constructor syntax for declaring parameters and
initializing properties simultaneously. initializing properties simultaneously.
block register-provider-ngmodule
:marked
Before we do, let's see an example of provider registration during bootstrapping:
+makeExcerpt('app/main.1.ts (discouraged)', 'bootstrap-discouraged', '')
:marked
The injector now knows about our `HeroService`.
An instance of our `HeroService` will be available for injection across our entire application.
Of course we can't help wondering about that comment telling us not to do it this way.
It *will* work. It's just not a best practice.
The bootstrap provider option is intended for configuring and overriding Angular's own
preregistered services, such as its routing support.
The preferred approach is to register application providers in application components.
Because the `HeroService` is used within the *Heroes* feature area &mdash;
and nowhere else &mdash; the ideal place to register it is in the top-level `HeroesComponent`.
block ngmodule-vs-component
:marked
Look at the `providers` part of the `@Component` annotation.
An instance of the `HeroService` is now available for injection in this `HeroesComponent`
and all of its child components.
The `HeroesComponent` itself doesn't happen to need the `HeroService`.
But its child `HeroListComponent` does, so we head there next.
block injectable-not-always-needed-in-ts block injectable-not-always-needed-in-ts
//- The [Angular Dart Transformer](https://github.com/angular/angular/wiki/Angular-2-Dart-Transformer) //- The [Angular Dart Transformer](https://github.com/angular/angular/wiki/Angular-2-Dart-Transformer)
//- generates static code to replace the use of dart:mirrors. It requires that types be //- generates static code to replace the use of dart:mirrors. It requires that types be

View File

@ -2,3 +2,4 @@ extends ../../../ts/_cache/guide/index.jade
block includes block includes
include ../_util-fns include ../_util-fns
- var _angular_io = 'website';

View File

@ -0,0 +1,4 @@
extends ../../../ts/_cache/guide/learning-angular.jade
block includes
include ../_util-fns

View File

@ -0,0 +1,58 @@
extends ../../../ts/_cache/guide/setup.jade
block includes
include ../_util-fns
- var _prereq = 'the Dart SDK';
- var _playground = 'repository';
- var _Install = 'Get';
//- npm/pub commands
- var _install = 'get';
- var _start = 'serve';
block qs-seed
:marked
The <live-example name="quickstart">QuickStart project</live-example> can
conveniently be used to seed new projects. It contains the following core files:
block core-files
+makeTabs(`
quickstart/ts/app/app.component.ts,
quickstart/ts/app/main.ts,
quickstart/ts/index.html,
quickstart/dart/pubspec.yaml,
quickstart/ts/styles.css`,
',,,,quickstart',
`app/app.component.ts,
app/main.ts,
index.html,
pubspec.yaml,
styles.css (excerpt)`)
:marked
These files are organized as follows:
.filetree
.file angular_quickstart
.children
.file lib
.children
.file app_component.dart
.file pubspec.yaml
.file web
.children
.file index.html
.file main.dart
.file styles.css
block install-tooling
:marked
Install the **[Dart SDK](https://www.dartlang.org/downloads/)**,
if not already on your machine, and any tools you like to use with Dart.
The Dart SDK includes tools such as **[pub][pub]**, the Dart package manager.
If you don't have a favorite Dart editor already, try
[WebStorm][WS], which comes with a Dart plugin.
You can also download [Dart plugins for other IDEs and editors][DT].
[WS]: https://confluence.jetbrains.com/display/WI/Getting+started+with+Dart
[DT]: https://www.dartlang.org/tools/
[pub]: https://www.dartlang.org/tools/pub/

View File

@ -2,209 +2,4 @@ extends ../../ts/_cache/quickstart.jade
block includes block includes
include _util-fns include _util-fns
- var _Install = 'Get' - var _on_Plunkr = '';
- var _prereq = 'the Dart SDK'
- var _angular_browser_uri = 'angular2/platform/browser.dart'
- var _angular_core_uri = 'angular2/core.dart'
- var _stepInit = 3
block setup-tooling
:marked
Install the **[Dart SDK](https://www.dartlang.org/downloads/)**,
if not already on your machine, and any tools you like to use with Dart.
The Dart SDK includes tools such as **[pub][pub]**, the Dart package manager.
If you don't have a favorite Dart editor already, try
[WebStorm][WS], which comes with a Dart plugin.
You can also download [Dart plugins for other IDEs and editors][DT].
[WS]: https://confluence.jetbrains.com/display/WI/Getting+started+with+Dart
[DT]: https://www.dartlang.org/tools/
[pub]: https://www.dartlang.org/tools/pub/
block package-and-config-files
:marked
In the project folder just created, create a file named
**[pubspec.yaml][pubspec]** with the code below.
This pubspec must specify the **angular2** and **browser**
packages as dependencies, as well as the `angular2` transformer.
It can also specify other packages and transformers for the app to use,
such as [dart_to_js_script_rewriter](https://pub.dartlang.org/packages/dart_to_js_script_rewriter).
[pubspec]: https://www.dartlang.org/tools/pub/pubspec.html
+makeExample('quickstart/dart/pubspec.yaml', null, 'pubspec.yaml')
block install-packages
:marked
From the project folder, run `pub get` to install the angular2 and browser
packages (along with the packages they depend on).
code-example(language="sh" class="code-shell").
pub get
block create-your-app
:marked
Let's create a folder to hold our application and add a super-simple Angular component.
block annotation-fields
:marked
The call to the `@Component` constructor has two
named parameters, `selector` and `template`.
block create-main
p.
Now we need something to tell Angular to load the root component.
Create:
ul
li a #[b folder named #[code web]]
li a <b>file named #[code #[+adjExPath('app/main.ts')]]</b> with the following content:
block commentary-on-index-html
:marked
Note the `<my-app>` tag in the `<body>`, this is *where your app lives!*
block run-app
p.
We have a few options for running our app.
One is to launch a local HTTP server
and then view the app in
<a href="https://www.dartlang.org/tools/dartium/">Dartium</a>.
We can use any web server, such as WebStorm's server
or Python's SimpleHTTPServer.
p.
Another option is to build and serve the app using <code>pub serve</code>,
and then run it by visiting <b><code>http://localhost:8080</code></b> in any modern browser.
Pub serve generates JavaScript on the fly,
which can take a while when first visiting the page.
Pub serve also runs in <b><i>watch mode</i></b>, and will recompile and subsequently serve
any changed assets.
p.
Once the app is running, the browser window should show the following:
block build-app
//- Remove details of building from QS for now. (It is too early for these details.)
if false
.alert.is-important
:marked
If you don't see **Hello Angular!**, make sure you've entered all the code correctly,
in the [proper folders](#wrap-up),
and run `pub get`.
.l-verbose-section#section-angular-run-app
:marked
### Building the app (generating JavaScript)
Before deploying the app, we need to generate JavaScript files.
The `pub build` command makes that easy.
code-example(language="sh" class="code-shell").
&gt; <span class="blk">pub build</span>
Loading source assets...
:marked
The generated JavaScript appears, along with supporting files,
under a directory named `build`.
#angular_transformer
h4 Using the Angular transformer
p.
When generating JavaScript for an Angular app,
be sure to use the Angular transformer.
It analyzes the Dart code,
converting reflection-using code to static code
that Dart's build tools can compile to faster, smaller JavaScript.
The highlighted lines in <code>pubspec.yaml</code>
configure the Angular transformer:
- var stylePattern = { otl: /(transformers:)|(- angular2:)|(entry_points.*$)/gm };
+makeExample('quickstart/dart/pubspec.yaml', null, 'pubspec.yaml', stylePattern)
p.
The <code>entry_points</code> item
identifies the Dart file in our app
that has a <code>main()</code> function.
For more information, see the
<a href="https://github.com/angular/angular/wiki/Angular-2-Dart-Transformer">Angular
transformer wiki page</a>.
.l-sub-section#performance
h3 Performance, the transformer, and Angular libraries
p.
When an app imports <code>bootstrap.dart</code>,
it also gets <code>dart:mirrors</code>,
a reflection library that
causes performance problems when compiled to JavaScript.
Don't worry,
the Angular transformer converts the app's entry points
(<code>entry_points</code> in <code>pubspec.yaml</code>)
so that they don't use mirrors.
#dart_to_js_script_rewriter
h4 Using dart_to_js_script_rewriter
:marked
To improve the app's performance, convert the
HTML file to directly include the generated JavaScript;
one way to do that is with `dart_to_js_script_rewriter`.
To use the rewriter, specify `dart_to_js_script_rewriter` in both
the `dependencies` and `transformers` sections of the pubspec.
- var stylePattern = { otl: /(dart_to_js_script_rewriter.*$)|(- dart_to_js_script_rewriter.*$)|(dependencies:)|(transformers:)/gm };
+makeExample('quickstart/dart/pubspec.yaml', null, 'pubspec.yaml', stylePattern)
.alert.is-important
:marked
The `dart_to_js_script_rewriter` transformer must be
**after** the `angular2` transformer in `pubspec.yaml`.
:marked
For more information, see the docs for
[dart_to_js_script_rewriter](https://pub.dartlang.org/packages/dart_to_js_script_rewriter).
block server-watching
:marked
To see the new version, just reload the page.
.alert.is-important
:marked
Be sure to terminate your local server once you stop working on this app.
block project-file-structure
.filetree
.file angular_quickstart
.children
.file lib
.children
.file app_component.dart
.file pubspec.yaml
.file web
.children
.file index.html
.file main.dart
.file styles.css
.l-verbose-section
:marked
This figure doesn't show generated files and directories.
For example, a `pubspec.lock` file
specifies versions and other identifying information for
the packages that our app depends on.
The `pub build` command creates a `build` directory
containing the JavaScript version of our app.
Pub, IDEs, and other tools often create
other directories and dotfiles.
block project-files
+makeTabs(`
quickstart/ts/app/app.component.ts,
quickstart/ts/app/main.ts,
quickstart/ts/index.html,
quickstart/dart/pubspec.yaml,
quickstart/ts/styles.css`,
',,,,quickstart',
`app/app.component.ts,
app/main.ts,
index.html,
pubspec.yaml,
styles.css (excerpt)`)

View File

@ -42,7 +42,7 @@ block angular-router
:marked :marked
The Angular router is a combination of multiple services The Angular router is a combination of multiple services
(`ROUTER_PROVIDERS`), multiple directives (`ROUTER_DIRECTIVES`), and a (`ROUTER_PROVIDERS`), multiple directives (`ROUTER_DIRECTIVES`), and a
configuration annotation (`RouteConfig`). We'll get them all by importing configuration annotation (`RouteConfig`). You get them all by importing
the router library: the router library:
+makeExcerpt('app/app.component.ts (router imports)', 'import-router') +makeExcerpt('app/app.component.ts (router imports)', 'import-router')
@ -53,22 +53,22 @@ block angular-router
Not all apps need routing, which is why the Angular *Component Router* is Not all apps need routing, which is why the Angular *Component Router* is
in a separate, optional library module. in a separate, optional library module.
Like for any service, we make router services available to the application Like for any service, you make router services available to the application
by adding them to the `providers` list. Let's update the `directives` and by adding them to the `providers` list. Update the `directives` and
`providers` lists to include the router assets: `providers` lists to include the router assets:
+makeExcerpt('app/app.component.ts (excerpt)', 'directives-and-providers') +makeExcerpt('app/app.component.ts (excerpt)', 'directives-and-providers')
:marked :marked
Notice that we also removed the `HeroesComponent` from the `directives` list. `AppComponent` no longer shows heroes, that will be the router's job,
`AppComponent` no longer shows heroes; that will be the router's job. so you can remove the `HeroesComponent` from the `directives` list.
We'll soon remove `<my-heroes>` from the template too. You'll soon remove `<my-heroes>` from the template too.
block router-config-intro block router-config-intro
:marked :marked
### Configure routes and add the router ### Configure routes and add the router
The `AppComponent` doesn't have a router yet. We'll use the `@RouteConfig` The `AppComponent` doesn't have a router yet. You'll use the `@RouteConfig`
annotation to simultaneously: annotation to simultaneously:
- Assign a router to the component - Assign a router to the component
@ -77,12 +77,12 @@ block router-config-intro
block routerLink block routerLink
:marked :marked
Notice the `[routerLink]` binding in the anchor tag. Notice the `[routerLink]` binding in the anchor tag.
We bind the `RouterLink` directive (another of the `ROUTER_DIRECTIVES`) to a list You bind the `RouterLink` directive (another of the `ROUTER_DIRECTIVES`) to a list
that tells the router where to navigate when the user clicks the link. that tells the router where to navigate when the user clicks the link.
We define a *routing instruction* with a *link parameters list*. You define a *routing instruction* with a *link parameters list*.
The list only has one element in our little sample, the quoted ***name* of the route** to follow. The list only has one element in our little sample, the quoted ***name* of the route** to follow.
Looking back at the route configuration, we confirm that `'Heroes'` is the name of the route to the `HeroesComponent`. Looking back at the route configuration, confirm that `'Heroes'` is the name of the route to the `HeroesComponent`.
.l-sub-section .l-sub-section
:marked :marked
Learn about the *link parameters list* Learn about the *link parameters list*
@ -90,8 +90,8 @@ block routerLink
block redirect-vs-use-as-default block redirect-vs-use-as-default
:marked :marked
We don't need a route definition for that. Instead, You don't need a route definition for that. Instead,
we add `useAsDefault: true` to the dashboard *route definition* and the add `useAsDefault: true` to the dashboard *route definition* and the
router will display the dashboard when the browser URL doesn't match an existing route. router will display the dashboard when the browser URL doesn't match an existing route.
block templateUrl-path-resolution block templateUrl-path-resolution
@ -105,7 +105,7 @@ block templateUrl-path-resolution
block route-params block route-params
:marked :marked
We will no longer receive the hero in a parent component property binding. You will no longer receive the hero in a parent component property binding.
The new `HeroDetailComponent` should take the `id` parameter from the router's The new `HeroDetailComponent` should take the `id` parameter from the router's
`RouteParams` service and use the `HeroService` to fetch the hero with that `id`. `RouteParams` service and use the `HeroService` to fetch the hero with that `id`.
@ -116,12 +116,12 @@ block ngOnInit
block extract-id block extract-id
:marked :marked
Notice how we extract the `id` by calling the `RouteParams.get` method. Notice how you can extract the `id` by calling the `RouteParams.get` method.
block heroes-component-cleanup block heroes-component-cleanup
:marked :marked
Because the template for `HeroesComponent` no longer uses `HeroDetailComponent` Because the template for `HeroesComponent` no longer uses `HeroDetailComponent`
directly &mdash; instead using the router to _navigate_ to it &mdash; we can directly &mdash; instead using the router to _navigate_ to it &mdash; you can
drop the `directives` argument from `@Component` and remove the unused hero detail drop the `directives` argument from `@Component` and remove the unused hero detail
import. The revised `@Component` looks like this: import. The revised `@Component` looks like this:
@ -138,7 +138,7 @@ block router-link-active
**The *router-link-active* class** **The *router-link-active* class**
The Angular Router adds the `router-link-active` class to the HTML navigation element The Angular Router adds the `router-link-active` class to the HTML navigation element
whose route matches the active route. All we have to do is define the style for it. Sweet! whose route matches the active route. All you have to do is define the style for it. Sweet!
block file-tree-end block file-tree-end
.filetree .filetree

View File

@ -10,4 +10,4 @@
h3 Dart h3 Dart
</a> p: <a href="/docs/dart/latest">Angular Dart - 最新版本 p: <a href="/docs/dart/latest">Angular Dart - 最新版本</a>

View File

@ -208,7 +208,7 @@ code-example(format="").
:marked :marked
The `template` property holds the component's companion template. The `template` property holds the component's companion template.
A template is a form of HTML that tells Angular how to render a view. A template is a form of HTML that tells Angular how to render a view.
Our template is a single line of HTML announcing "Hello Angular!". Our template is a single line of HTML announcing "Hello Angular".
Now we need something to tell Angular to load this component. Now we need something to tell Angular to load this component.

View File

@ -13,7 +13,7 @@ block includes
Most Angular terms are everyday English words Most Angular terms are everyday English words
with a specific meaning within the Angular system. with a specific meaning within the Angular system.
We have gathered here the most prominent terms This glossary lists the most prominent terms
and a few less familiar ones that have unusual or and a few less familiar ones that have unusual or
unexpected definitions. unexpected definitions.
@ -23,30 +23,28 @@ block includes
.l-main-section#A .l-main-section#A
a#aot
:marked
## Ahead-of-Time (AoT) compilation
.l-sub-section
:marked
You can compile Angular applications at build-time.
By compiling your application<span if-docs="ts"> using the compiler-cli, `ngc`</span>, you can bootstrap directly
to a<span if-docs="ts"> module</span> factory, meaning you don't need to include the Angular compiler in your JavaScript bundle.
Ahead-of-time compiled applications also benefit from decreased load time and increased performance.
+ifDocsFor('ts') +ifDocsFor('ts')
a#aot
:marked :marked
## Ahead-of-Time (AoT) Compilation ## Angular module
.l-sub-section .l-sub-section
:marked :marked
Angular applications can be compiled by developers at build-time. Helps you organize an application into cohesive blocks of functionality.
By compiling your application using the compiler-cli, `ngc`, you can bootstrap directly An Angular module identifies the components, directives, and pipes that the application uses along with the list of external Angular modules that the application needs, such as `FormsModule`.
to a Module Factory, meaning you don't need to include the Angular compiler in your javascript bundle.
Ahead-of-time compiled applications also benefit from decreased load time and increased
performance.
:marked Every Angular application has an application root module class. By convention, the class is
## Angular Module called `AppModule` and resides in a file named `app.module.ts`.
.l-sub-section
:marked
Helps us organize an application into cohesive blocks of functionality.
An Angular module identifies the components, directives, and pipes that are used by the application
along with the list of external Angular modules that the application needs, such as `FormsModule`.
Every Angular application has an application root module class. By convention the class is For details and examples, see the [Angular Module](!{docsLatest}/guide/ngmodule.html) page.
called `AppModule` and resides in a file named `app.component.ts`.
See the [Angular Module](!{docsLatest}/guide/ngmodule.html) chapter for details and examples.
+ifDocsFor('ts|dart') +ifDocsFor('ts|dart')
:marked :marked
@ -57,15 +55,14 @@ block includes
In practice, a synonym for [Decoration](#decorator). In practice, a synonym for [Decoration](#decorator).
:marked :marked
## Attribute Directive ## Attribute directive
.l-sub-section .l-sub-section
:marked :marked
A category of [Directive](#directive) that can listen to and modify the behavior of A category of [directive](#directive) that can listen to and modify the behavior of
other HTML elements, attributes, properties, and components. They are usually represented other HTML elements, attributes, properties, and components. They are usually represented
as HTML attributes, hence the name. as HTML attributes, hence the name.
The `ngClass` directive for adding and removing CSS class names is a good example of A good example of an attribute directive is the `ngClass` directive for adding and removing CSS class names.
an Attribute Directive.
.l-main-section#B .l-main-section#B
@ -74,7 +71,7 @@ block includes
## Barrel ## Barrel
.l-sub-section .l-sub-section
:marked :marked
A barrel is a way to *rollup exports* from several ES2015 modules into a single convenience ES2015 module. A barrel is a way to *rollup exports* from several ES2015 modules into a single convenient ES2015 module.
The barrel itself is an ES2015 module file that re-exports *selected* exports of other ES2015 modules. The barrel itself is an ES2015 module file that re-exports *selected* exports of other ES2015 modules.
Imagine three ES2015 modules in a `heroes` folder: Imagine three ES2015 modules in a `heroes` folder:
@ -94,7 +91,7 @@ block includes
import { Hero } from '../heroes/hero.model.ts'; import { Hero } from '../heroes/hero.model.ts';
import { HeroService } from '../heroes/hero.service.ts'; import { HeroService } from '../heroes/hero.service.ts';
:marked :marked
We can add a barrel to the `heroes` folder (called `index` by convention) that exports all of these items: You can add a barrel to the `heroes` folder (called `index`, by convention) that exports all of these items:
code-example. code-example.
export * from './hero.model.ts'; // re-export all of its exports export * from './hero.model.ts'; // re-export all of its exports
export * from './hero.service.ts'; // re-export all of its exports export * from './hero.service.ts'; // re-export all of its exports
@ -106,13 +103,9 @@ block includes
:marked :marked
The Angular [scoped packages](#scoped-package) each have a barrel named `index`. The Angular [scoped packages](#scoped-package) each have a barrel named `index`.
That's why we can write this:
+makeExcerpt('quickstart/ts/app/app.component.ts', 'import', '')
.alert.is-important .alert.is-important
:marked :marked
Note that you can often achieve this same goal using [Angular modules](#angular-module) instead. Note that you can often achieve this using [Angular modules](#angular-module) instead.
:marked :marked
## Binding ## Binding
@ -121,8 +114,8 @@ block includes
Almost always refers to [Data Binding](#data-binding) and the act of Almost always refers to [Data Binding](#data-binding) and the act of
binding an HTML object property to a data object property. binding an HTML object property to a data object property.
May refer to a [Dependency Injection](#dependency-injection) binding May refer to a [dependency injection](#dependency-injection) binding
between a "token" or "key" and a dependency [provider](#provider). between a "token", also referred to as a "key", and a dependency [provider](#provider).
This more rare usage should be clear in context. This more rare usage should be clear in context.
:marked :marked
@ -130,11 +123,10 @@ block includes
.l-sub-section .l-sub-section
block bootstrap-defn-top block bootstrap-defn-top
:marked :marked
We launch an Angular application by "bootstrapping" it using the application root Angular module (`AppModule`). You launch an Angular application by "bootstrapping" it using the application root Angular module (`AppModule`). Bootstrapping identifies an application's top level "root" [component](#component), which is the first component that is loaded for the application.
The bootstraping identifies an application's top level "root" [Component](#component), which is the first For more information, see the [Setup](!{docsLatest}/guide/setup.html) page.
component that is loaded for the application. For more information see the [QuickStart](!{docsLatest}/quickstart.html).
:marked :marked
One can bootstrap multiple apps in the same `index.html`, each with its own top level root. You can bootstrap multiple apps in the same `index.html`, each with its own top level root.
.l-main-section#C .l-main-section#C
:marked :marked
@ -142,31 +134,29 @@ block includes
.l-sub-section .l-sub-section
:marked :marked
The practice of writing compound words or phrases such that each word or abbreviation begins with a capital letter The practice of writing compound words or phrases such that each word or abbreviation begins with a capital letter
_except the first letter which is a lowercase letter_. _except the first letter, which is lowercase_.
Function, property, and method names are typically spelled in camelCase. Examples include: `square`, `firstName` and `getHeroes`. Function, property, and method names are typically spelled in camelCase. Examples include: `square`, `firstName` and `getHeroes`. Notice that `square` is an example of how you write a single word in camelCase.
This form is also known as **lower camel case**, to distinguish it from **upper camel case** which we call [PascalCase](#pascalcase). This form is also known as **lower camel case**, to distinguish it from **upper camel case**, which is [PascalCase](#pascalcase).
When we write "camelCase" in this documentation we always mean *lower camel case*. When you see "camelCase" in this documentation it always means *lower camel case*.
:marked :marked
## Component ## Component
.l-sub-section .l-sub-section
:marked :marked
An Angular class responsible for exposing data An Angular class responsible for exposing data to a [view](#view) and handling most of the views display and user-interaction logic.
to a [View](#view) and handling most of the views display
and user-interaction logic.
The Component is one of the most important building blocks in the Angular system. The *component* is one of the most important building blocks in the Angular system.
It is, in fact, an Angular [Directive](#directive) with a companion [Template](#template). It is, in fact, an Angular [directive](#directive) with a companion [template](#template).
The developer applies the `!{_at}Component` !{_decoratorLink} to You apply the `!{_at}Component` !{_decoratorLink} to
the component class, thereby attaching to the class the essential component metadata the component class, thereby attaching to the class the essential component metadata
that Angular needs to create a component instance and render it with its template that Angular needs to create a component instance and render it with its template
as a view. as a view.
Those familiar with "MVC" and "MVVM" patterns will recognize Those familiar with "MVC" and "MVVM" patterns will recognize
the Component in the role of "Controller" or "View Model". the component in the role of "controller" or "view model".
.l-main-section#D .l-main-section#D
:marked :marked
@ -181,56 +171,48 @@ block includes
spelled in dash-case. spelled in dash-case.
:marked :marked
## Data Binding ## Data binding
.l-sub-section .l-sub-section
:marked :marked
Applications display data values to a user and respond to user Applications display data values to a user and respond to user
actions (clicks, touches, keystrokes). actions (clicks, touches, keystrokes).
We could push application data values into HTML, attach Instead of manually pushing application data values into HTML, attaching
event listeners, pull changed values from the screen, and event listeners, pulling changed values from the screen, and
update application data values ... all by hand. updating application data values, you can use data binding by declaring the relationship between an HTML widget and data source and let the
Or we could declare the relationship between an HTML widget
and an application data source ... and let a data binding
framework handle the details. framework handle the details.
Data Binding is that second approach. Angular has a rich Angular has a rich data binding framework with a variety of data binding
data binding framework with a variety of data binding
operations and supporting declaration syntax. operations and supporting declaration syntax.
The many forms of binding include: Read about the forms of binding in the [Template Syntax](!{docsLatest}/guide/template-syntax.html#data-binding) page:
* [Interpolation](!{docsLatest}/guide/template-syntax.html#interpolation) * [Interpolation](!{docsLatest}/guide/template-syntax.html#interpolation).
* [Property Binding](!{docsLatest}/guide/template-syntax.html#property-binding) * [Property binding](!{docsLatest}/guide/template-syntax.html#property-binding).
* [Event Binding](!{docsLatest}/guide/template-syntax.html#event-binding) * [Event binding](!{docsLatest}/guide/template-syntax.html#event-binding).
* [Attribute Binding](!{docsLatest}/guide/template-syntax.html#attribute-binding) * [Attribute binding](!{docsLatest}/guide/template-syntax.html#attribute-binding).
* [Class Binding](!{docsLatest}/guide/template-syntax.html#class-binding) * [Class binding](!{docsLatest}/guide/template-syntax.html#class-binding).
* [Style Binding](!{docsLatest}/guide/template-syntax.html#style-binding) * [Style binding](!{docsLatest}/guide/template-syntax.html#style-binding).
* [Two-way data binding with ngModel](!{docsLatest}/guide/template-syntax.html#ng-model) * [Two-way data binding with ngModel](!{docsLatest}/guide/template-syntax.html#ngModel).
Learn more about data binding in the
[Template Syntax](!{docsLatest}/guide/template-syntax.html#data-binding) chapter.
+ifDocsFor('ts|dart') +ifDocsFor('ts|dart')
a#decorator a#decorator
a#decoration a#decoration
:marked :marked
## Decorator | Decoration ## Decorator | decoration
.l-sub-section .l-sub-section
block decorator-defn block decorator-defn
:marked :marked
A Decorator is a **function** that adds metadata to a class, its members (properties, methods) and function arguments. A decorator is a **function** that adds metadata to a class, its members (properties, methods) and function arguments.
Decorators are a JavaScript language [feature](https://github.com/wycats/javascript-decorators), implemented in TypeScript and proposed for ES2016 (AKA ES7). Decorators are a JavaScript language [feature](https://github.com/wycats/javascript-decorators), implemented in TypeScript and proposed for ES2016 (AKA ES7).
We apply a decorator by positioning it To apply a decorator, position it immediately above or to the left of the thing it decorates.
immediately above or to the left of the thing it decorates.
Angular has its own set of decorators to help it interoperate with our application parts. Angular has its own set of decorators to help it interoperate with your application parts.
Here is an example of a `@Component` decorator that identifies a Here is an example of a `@Component` decorator that identifies a
class as an Angular [Component](#component) and an `@Input` decorator applied to a property class as an Angular [component](#component) and an `@Input` decorator applied to the `name` property
of that component. of that component. The elided object argument to the `@Component` decorator would contain the pertinent component metadata.
The elided object argument to the `@Component` decorator would contain the pertinent component metadata.
``` ```
@Component({...}) @Component({...})
export class AppComponent { export class AppComponent {
@ -244,77 +226,75 @@ block includes
.alert.is-important .alert.is-important
:marked :marked
Always include the parentheses `()` when applying a decorator. Always include parentheses `()` when applying a decorator.
A decorator is a **function** that must be called when applied.
:marked :marked
## Dependency Injection ## Dependency injection
.l-sub-section .l-sub-section
:marked :marked
Dependency Injection is both a design pattern and a mechanism Dependency injection is both a design pattern and a mechanism
for creating and delivering parts of an application to other for creating and delivering parts of an application to other
parts of an application that request them. parts of an application that request them.
Angular developers prefer to build applications by defining many simple parts Angular developers prefer to build applications by defining many simple parts
that each do one thing well and then wire them together at runtime. that each do one thing well and then wiring them together at runtime.
These parts often rely on other parts. An Angular [Component](#component) These parts often rely on other parts. An Angular [component](#component)
part might rely on a service part to get data or perform a calculation. When a part might rely on a service part to get data or perform a calculation. When
part "A" relies on another part "B", we say that "A" depends on "B" and part "A" relies on another part "B", you say that "A" depends on "B" and
that "B" is a dependency of "A". that "B" is a dependency of "A".
We can ask a "Dependency Injection System" to create "A" You can ask a "dependency injection system" to create "A"
for us and handle all the dependencies. for us and handle all the dependencies.
If "A" needs "B" and "B" needs "C", the system resolves that chain of dependencies If "A" needs "B" and "B" needs "C", the system resolves that chain of dependencies
and returns a fully prepared instance of "A". and returns a fully prepared instance of "A".
Angular provides and relies upon its own sophisticated Angular provides and relies upon its own sophisticated
[Dependency Injection](dependency-injection.html) system [dependency injection](dependency-injection.html) system
to assemble and run applications by "injecting" application parts to assemble and run applications by "injecting" application parts
into other application parts where and when needed. into other application parts where and when needed.
At the core there is an [`Injector`](#injector) that returns dependency values on request. At the core there is an [`injector`](#injector) that returns dependency values on request.
The expression `injector.get(token)` returns the value associated with the given token. The expression `injector.get(token)` returns the value associated with the given token.
A token is an Angular type (`OpaqueToken`). We rarely deal with tokens directly; most A token is an Angular type (`OpaqueToken`). You rarely deal with tokens directly; most
methods accept a class name (`Foo`) or a string ("foo") and Angular converts it methods accept a class name (`Foo`) or a string ("foo") and Angular converts it
to a token. When we write `injector.get(Foo)`, the injector returns to a token. When you write `injector.get(Foo)`, the injector returns
the value associated with the token for the `Foo` class, typically an instance of `Foo` itself. the value associated with the token for the `Foo` class, typically an instance of `Foo` itself.
Angular makes similar requests internally during many of its operations During many of its operations, Angular makes similar requests internally, such as when it creates a [`component`](#component) for display.
as when it creates a [`Component`](#component) for display.
The `Injector` maintains an internal map of tokens to dependency values. The `Injector` maintains an internal map of tokens to dependency values.
If the `Injector` can't find a value for a given token, it creates If the `Injector` can't find a value for a given token, it creates
a new value using a `Provider` for that token. a new value using a `Provider` for that token.
A [Provider](#provider) is a recipe for A [provider](#provider) is a recipe for
creating new instances of a dependency value associated with a particular token. creating new instances of a dependency value associated with a particular token.
An injector can only create a value for a given token if it has An injector can only create a value for a given token if it has
a `Provider` for that token in its internal provider registry. a `provider` for that token in its internal provider registry.
Registering providers is a critical preparatory step. Registering providers is a critical preparatory step.
Angular registers some of its own providers with every injector. Angular registers some of its own providers with every injector.
We can register our own providers. We can register our own providers.
Learn more in the [Dependency Injection](!{docsLatest}/guide/dependency-injection.html) chapter. Read more in the [Dependency Injection](!{docsLatest}/guide/dependency-injection.html) page.
:marked :marked
## Directive ## Directive
.l-sub-section .l-sub-section
:marked :marked
An Angular class responsible for creating, re-shaping, and interacting with HTML elements An Angular class responsible for creating, reshaping, and interacting with HTML elements
in the browser DOM. Directives are Angular's most fundamental feature. in the browser DOM. Directives are Angular's most fundamental feature.
A Directive is almost always associated with an HTML element or attribute. A Directive is almost always associated with an HTML element or attribute.
We often refer to such an element or attribute as the directive itself. We often refer to such an element or attribute as the directive itself.
When Angular finds a directive in an HTML template, When Angular finds a directive in an HTML template,
it creates the matching directive class instance it creates the matching directive class instance
and gives that instance control over that portion of the browser DOM. and gives the instance control over that portion of the browser DOM.
Developers can invent custom HTML markup (e.g., `<my-directive>`) to You can invent custom HTML markup (for example, `<my-directive>`) to
associate with their custom directives. They add this custom markup to HTML templates associate with your custom directives. You add this custom markup to HTML templates
as if they were writing native HTML. In this way, directives become extensions of as if you were writing native HTML. In this way, directives become extensions of
HTML itself. HTML itself.
Directives fall into one of three categories: Directives fall into one of three categories:
@ -324,12 +304,12 @@ block includes
They are the building blocks of an Angular application and the They are the building blocks of an Angular application and the
developer can expect to write a lot of them. developer can expect to write a lot of them.
1. [Attribute Directives](#attribute-directive) that can listen to and modify the behavior of 1. [Attribute directives](#attribute-directive) that can listen to and modify the behavior of
other HTML elements, attributes, properties, and components. They are usually represented other HTML elements, attributes, properties, and components. They are usually represented
as HTML attributes, hence the name. as HTML attributes, hence the name.
1. [Structural Directives](#structural-directive), a directive responsible for 1. [Structural directives](#structural-directive), a directive responsible for
shaping or re-shaping HTML layout, typically by adding, removing, or manipulating shaping or reshaping HTML layout, typically by adding, removing, or manipulating
elements and their children. elements and their children.
.l-main-section#E .l-main-section#E
@ -342,9 +322,9 @@ block includes
The latest approved version of JavaScript is The latest approved version of JavaScript is
[ECMAScript 2016](http://www.ecma-international.org/ecma-262/7.0/) [ECMAScript 2016](http://www.ecma-international.org/ecma-262/7.0/)
(AKA "ES2016" or "ES7") and many Angular developers will write their applications (AKA "ES2016" or "ES7") and many Angular developers write their applications
either in this version of the language or a dialect that strives to be either in this version of the language or a dialect that strives to be
compatible with it such as [TypeScript](#typesScript). compatible with it, such as [TypeScript](#typescript).
Most modern browsers today only support the much older "ECMAScript 5" (AKA ES5) standard. Most modern browsers today only support the much older "ECMAScript 5" (AKA ES5) standard.
Applications written in ES2016, ES2015 or one of their dialects must be "[transpiled](#transpile)" Applications written in ES2016, ES2015 or one of their dialects must be "[transpiled](#transpile)"
@ -356,17 +336,17 @@ block includes
## ES2015 ## ES2015
.l-sub-section .l-sub-section
:marked :marked
Short hand for "[ECMAScript 2015](#ecmascript=2015)". Short hand for [ECMAScript](#ecmascript) 2015.
:marked :marked
## ES6 ## ES6
.l-sub-section .l-sub-section
:marked :marked
Short hand for "[ECMAScript 2015](#ecmascript=2015)". Short hand for [ECMAScript](#ecmascript) 2015.
:marked :marked
## ES5 ## ES5
.l-sub-section .l-sub-section
:marked :marked
Short hand for "ECMAScript 5", the version of JavaScript run by most modern browsers. Short hand for [ECMAScript](#ecmascript) 5, the version of JavaScript run by most modern browsers.
See [ECMAScript](#ecmascript). See [ECMAScript](#ecmascript).
a#F a#F
@ -386,41 +366,39 @@ a#H
.l-sub-section .l-sub-section
:marked :marked
A directive property that can be the ***target*** of a A directive property that can be the ***target*** of a
[Property Binding](!{docsLatest}/guide/template-syntax.html#property-binding). [property binding](!{docsLatest}/guide/template-syntax.html#property-binding) (explained in detail in the [Template Syntax](!{docsLatest}/guide/template-syntax.html) page).
Data values flow *into* this property from the data source identified Data values flow *into* this property from the data source identified
in the template expression to the right of the equal sign. in the template expression to the right of the equal sign.
See the [Template Syntax](!{docsLatest}/guide/template-syntax.html#inputs-outputs) chapter. See the [Input and output properties](!{docsLatest}/guide/template-syntax.html#inputs-outputs) section of the [Template Syntax](!{docsLatest}/guide/template-syntax.html) page.
:marked :marked
## Interpolation ## Interpolation
.l-sub-section .l-sub-section
:marked :marked
A form of [Property Data Binding](#data-binding) in which a A form of [property data binding](#data-binding) in which a
[template expression](#template-expression) between double-curly braces [template expression](#template-expression) between double-curly braces
renders as text. That text may be concatenated with neighboring text renders as text. That text may be concatenated with neighboring text
before it is assigned to an element property before it is assigned to an element property
or displayed between element tags as in this example. or displayed between element tags, as in this example.
code-example(language="html" escape="html"). code-example(language="html" escape="html").
<label>My current hero is {{hero.name}}</label> <label>My current hero is {{hero.name}}</label>
:marked :marked
Learn more about interpolation in the Read more about [interpolation](!{docsLatest}/guide/template-syntax.html#interpolation) in the
[Template Syntax](!{docsLatest}/guide/template-syntax.html#interpolation) chapter. [Template Syntax](!{docsLatest}/guide/template-syntax.html) page.
.l-main-section#J .l-main-section#J
+ifDocsFor('ts') a#jit
a#jit :marked
## Just-in-Time (JiT) compilation
.l-sub-section
:marked :marked
## Just-in-Time (JiT) Compilation With Angular _just-in-time_ bootstrapping you compile your components<span if-docs="ts"> and modules</span> in the browser
.l-sub-section and launch the application dynamically. This is a good choice during development.
:marked Consider using the [ahead-of-time](#aot) mode for production apps.
With Angular _Just-in-time_ bootstrapping you compile your components and modules in the
browser
and launch the application dynamically. This is a good choice during development.
Consider the [Ahead-of-time](#aot) mode for production apps.
.l-main-section#K .l-main-section#K
:marked :marked
@ -431,29 +409,29 @@ a#H
.l-main-section#L .l-main-section#L
:marked :marked
## Lifecycle Hooks ## Lifecycle hooks
.l-sub-section .l-sub-section
:marked :marked
[Directives](#directive) and [Components](#component) have a lifecycle [Directives](#directive) and [components](#component) have a lifecycle
managed by Angular as it creates, updates and destroys them. managed by Angular as it creates, updates, and destroys them.
Developers can tap into key moments in that lifecycle by implementing You can tap into key moments in that lifecycle by implementing
one or more of the "Lifecycle Hook" interfaces. one or more of the lifecycle hook interfaces.
Each interface has a single hook method whose name is the interface name prefixed with `ng`. Each interface has a single hook method whose name is the interface name prefixed with `ng`.
For example, the `OnInit` interface has a hook method names `ngOnInit`. For example, the `OnInit` interface has a hook method named `ngOnInit`.
Angular calls these hook methods in the following order: Angular calls these hook methods in the following order:
* `ngOnChanges` - called when an [input](#input)/[output](#output) binding values change * `ngOnChanges` - when an [input](#input)/[output](#output) binding value changes.
* `ngOnInit` - after the first `ngOnChanges` * `ngOnInit` - after the first `ngOnChanges`.
* `ngDoCheck` - developer's custom change detection * `ngDoCheck` - developer's custom change detection.
* `ngAfterContentInit` - after component content initialized * `ngAfterContentInit` - after component content initialized.
* `ngAfterContentChecked` - after every check of component content * `ngAfterContentChecked` - after every check of component content.
* `ngAfterViewInit` - after component's view(s) are initialized * `ngAfterViewInit` - after component's view(s) are initialized.
* `ngAfterViewChecked` - after every check of a component's view(s) * `ngAfterViewChecked` - after every check of a component's view(s).
* `ngOnDestroy` - just before the directive is destroyed. * `ngOnDestroy` - just before the directive is destroyed.
Learn more in the [Lifecycle Hooks](!{docsLatest}/guide/lifecycle-hooks.html) chapter. Read more in the [Lifecycle Hooks](!{docsLatest}/guide/lifecycle-hooks.html) page.
.l-main-section#M .l-main-section#M
@ -465,14 +443,13 @@ a#H
:marked :marked
In Angular, there are two types of modules: In Angular, there are two types of modules:
- [Angular modules](#angular-module). - [Angular modules](#angular-module).
See the [Angular Module](!{docsLatest}/guide/ngmodule.html) chapter for details and examples. For details and examples, see the [Angular Modules](!{docsLatest}/guide/ngmodule.html) page.
- ES2015 modules as described in this section. - ES2015 modules, as described in this section.
:marked :marked
Angular apps are modular. Angular apps are modular.
In general, we assemble our application from many modules, both the ones we write ourselves In general, you assemble your application from many modules, both the ones you write and the ones you acquire from others.
and the ones we acquire from others.
A typical module is a cohesive block of code dedicated to a single purpose. A typical module is a cohesive block of code dedicated to a single purpose.
@ -480,21 +457,19 @@ a#H
A module that needs that thing, **imports** it. A module that needs that thing, **imports** it.
The structure of Angular modules and the import/export syntax The structure of Angular modules and the import/export syntax
is based on the [ES2015](#es2015) module standard is based on the [ES2015 module standard](http://www.2ality.com/2014/09/es6-modules-final.html).
described [here](http://www.2ality.com/2014/09/es6-modules-final.html).
An application that adheres to this standard requires a module loader to An application that adheres to this standard requires a module loader to
load modules on request and resolve inter-module dependencies. load modules on request, and resolve inter-module dependencies.
Angular does not ship with a module loader and does not have a preference Angular does not ship with a module loader and does not have a preference
for any particular 3rd party library (although most samples use SystemJS). for any particular 3rd party library (although most examples use SystemJS).
Application developers may pick any module library that conforms to the standard You may pick any module library that conforms to the standard.
Modules are typically named after the file in which the exported thing is defined. Modules are typically named after the file in which the exported thing is defined.
The Angular [DatePipe](https://github.com/angular/angular/blob/master/modules/@angular/common/src/pipes/date_pipe.ts) The Angular [DatePipe](https://github.com/angular/angular/blob/master/modules/@angular/common/src/pipes/date_pipe.ts)
class belongs to a feature module named `date_pipe` in the file `date_pipe.ts`. class belongs to a feature module named `date_pipe` in the file `date_pipe.ts`.
Developers rarely access Angular feature modules directly. You rarely access Angular feature modules directly. You usually import them from one of the Angular [scoped packages](#scoped-package) such as `@angular/core`.
We usually import them from one of the Angular [scoped packages](#scoped-package) such as `@angular/core`.
a#N a#N
.l-main-section#O .l-main-section#O
@ -504,8 +479,8 @@ a#N
## Observable ## Observable
.l-sub-section .l-sub-section
:marked :marked
We can think of an observable as an array whose items arrive asynchronously over time. You can think of an observable as an array whose items arrive asynchronously over time.
Observables help us manage asynchronous data, such as data coming from a backend service. Observables help you manage asynchronous data, such as data coming from a backend service.
Observables are used within Angular itself, including Angular's event system and its http client service. Observables are used within Angular itself, including Angular's event system and its http client service.
To use observables, Angular uses a third-party library called Reactive Extensions (RxJS). To use observables, Angular uses a third-party library called Reactive Extensions (RxJS).
@ -515,12 +490,12 @@ a#N
## Output ## Output
.l-sub-section .l-sub-section
:marked :marked
A directive property that can be the ***target*** of an A directive property that can be the ***target*** of
[Event Binding](!{docsLatest}/guide/template-syntax.html#property-binding). [event binding](!{docsLatest}/guide/template-syntax.html#event-binding).
Events stream *out* of this property to the receiver identified Events stream *out* of this property to the receiver identified
in the template expression to the right of the equal sign. in the template expression to the right of the equal sign.
See the [Template Syntax](!{docsLatest}/guide/template-syntax.html#inputs-outputs) chapter. See the [Input and output properties](!{docsLatest}/guide/template-syntax.html#inputs-outputs) section of the [Template Syntax](!{docsLatest}/guide/template-syntax.html) page.
.l-main-section#P .l-main-section#P
@ -528,20 +503,18 @@ a#N
## PascalCase ## PascalCase
.l-sub-section .l-sub-section
:marked :marked
The practice of writing compound words or phrases such that each word or abbreviation begins with a capital letter. The practice of writing individual words, compound words, or phrases such that each word or abbreviation begins with a capital letter. Class names are typically spelled in PascalCase. Examples include: `Person` and `HeroDetailComponent`.
Class names are typically spelled in PascalCase. Examples include: `Person` and `HeroDetailComponent`.
This form is also known as **upper camel case**, to distinguish it from **lower camel case** which we simply call [camelCase](#camelcase). This form is also known as **upper camel case** to distinguish it from **lower camel case**, which is simply called [camelCase](#camelcase). In this documentation, "PascalCase" means *upper camel case* and "camelCase" means *lower camel case*.
In this documentation, "PascalCase" means *upper camel case* and "camelCase" means *lower camel case*.
:marked :marked
## Pipe ## Pipe
.l-sub-section .l-sub-section
:marked :marked
An Angular pipe is a function that transforms input values to output values for An Angular pipe is a function that transforms input values to output values for
display in a [view](#view). We use the `!{_at}Pipe` !{_decoratorLink} display in a [view](#view). Use the `!{_at}Pipe` !{_decoratorLink}
to associate the pipe function with a name. We can then use that to associate the pipe function with a name. You then use that
name in our HTML to declaratively transform values on screen. name in your HTML to declaratively transform values on screen.
Here's an example that uses the built-in `currency` pipe to display Here's an example that uses the built-in `currency` pipe to display
a numeric value in the local currency. a numeric value in the local currency.
@ -549,24 +522,22 @@ a#N
code-example(language="html" escape="html"). code-example(language="html" escape="html").
<label>Price: </label>{{product.price | currency}} <label>Price: </label>{{product.price | currency}}
:marked :marked
Learn more in the chapter on [pipes](!{docsLatest}/guide/pipes.html) . Read more in the page on [pipes](!{docsLatest}/guide/pipes.html).
- var _ProviderUrl = docsLatest+'/api/'+(lang == 'dart' ? 'angular2.core' : 'core/index')+'/Provider-class.html'
:marked :marked
## Provider ## Provider
.l-sub-section .l-sub-section
:marked :marked
A [Provider](!{_ProviderUrl}) creates a new instance of a dependency for the A _provider_ creates a new instance of a dependency for the
[Dependency Injection](#dependency-injection) system. [dependency injection](#dependency-injection) system.
It relates a lookup token to code &mdash; sometimes called a "recipe" &mdash; It relates a lookup token to code&mdash;sometimes called a "recipe"&mdash;that can create a dependency value.
that can create a dependency value.
a#Q a#Q
.l-main-section#R .l-main-section#R
+ifDocsFor('ts|js') +ifDocsFor('ts|js')
:marked :marked
## Reactive Forms ## Reactive forms
.l-sub-section .l-sub-section
:marked :marked
A technique for building Angular forms through code in a component. A technique for building Angular forms through code in a component.
@ -578,24 +549,22 @@ a#Q
- The template input elements do *not* use `ngModel`. - The template input elements do *not* use `ngModel`.
- The associated Angular directives are all prefixed with `Form` such as `FormGroup`, `FormControl`, and `FormControlName`. - The associated Angular directives are all prefixed with `Form` such as `FormGroup`, `FormControl`, and `FormControlName`.
Reactive forms are powerful, flexible, and great for more complex data entry form scenarios, such as dynamic generation Reactive forms are powerful, flexible, and great for more complex data entry form scenarios such as dynamic generation of form controls.
of form controls.
:marked :marked
## Router ## Router
.l-sub-section .l-sub-section
:marked :marked
Most applications consist of many screens or [views](#view). Most applications consist of many screens or [views](#view).
The user navigates among them by clicking links and buttons The user navigates among them by clicking links and buttons,
and taking other similar actions that cause the application to and performing other similar actions that cause the application to
replace one view with another. replace one view with another.
The Angular [Component Router](!{docsLatest}/guide/router.html) is a richly featured mechanism for configuring The Angular [component router](!{docsLatest}/guide/router.html) is a richly featured mechanism for configuring and managing the entire view navigation process including the creation and destruction
and managing the entire view navigation process including the creation and destruction
of views. of views.
+ifDocsFor('ts|js') +ifDocsFor('ts|js')
:marked :marked
In most cases, components becomes attached to a [router](#router) by means In most cases, components become attached to a [router](#router) by means
of a `RouterConfig` that defines routes to views. of a `RouterConfig` that defines routes to views.
A [routing component's](#routing-component) template has a `RouterOutlet` element A [routing component's](#routing-component) template has a `RouterOutlet` element
@ -604,31 +573,30 @@ a#Q
Other views in the application likely have anchor tags or buttons with `RouterLink` Other views in the application likely have anchor tags or buttons with `RouterLink`
directives that users can click to navigate. directives that users can click to navigate.
See the [Component Router](!{docsLatest}/guide/router.html) chapter to learn more. For more information, see the [Routing & Navigation](!{docsLatest}/guide/router.html) page.
+ifDocsFor('ts|js') +ifDocsFor('ts|js')
:marked :marked
## RouterModule ## Router module
.l-sub-section .l-sub-section
:marked :marked
A separate [Angular module](#angular-module) that provides the necessary service providers and directives for navigating through application views. A separate [Angular module](#angular-module) that provides the necessary service providers and directives for navigating through application views.
See the [Component Router](!{docsLatest}/guide/router.html) chapter to learn more. For more information, see the [Routing & Navigation](!{docsLatest}/guide/router.html) page.
:marked :marked
## Routing Component ## Routing component
.l-sub-section .l-sub-section
block routing-component-defn :marked
:marked An Angular [component](#component) with a `RouterOutlet` that displays views based on router navigations.
An Angular [Component](#component) with a RouterOutlet that displays views based on router navigations.
See the [Component Router](!{docsLatest}/guide/router.html) chapter to learn more. For more information, see the [Routing & Navigation](!{docsLatest}/guide/router.html) page.
.l-main-section#S .l-main-section#S
+ifDocsFor('ts|js') +ifDocsFor('ts|js')
:marked :marked
## Scoped Package ## Scoped package
.l-sub-section .l-sub-section
:marked :marked
Angular modules are delivered within *scoped packages* such as `@angular/core`, `@angular/common`, `@angular/platform-browser-dynamic`, Angular modules are delivered within *scoped packages* such as `@angular/core`, `@angular/common`, `@angular/platform-browser-dynamic`,
@ -636,9 +604,9 @@ a#Q
A [*scoped package*](https://docs.npmjs.com/misc/scope) is a way to group related *npm* packages. A [*scoped package*](https://docs.npmjs.com/misc/scope) is a way to group related *npm* packages.
We import a scoped package the same way we'd import a *normal* package. You import a scoped package the same way that you'd import a *normal* package.
The only difference, from a consumer perspective, The only difference, from a consumer perspective,
is that the package name begins with the Angular *scope name*, `@angular`. is that the *scoped package* name begins with the Angular *scope name*, `@angular`.
+makeExcerpt('architecture/ts/app/app.component.ts', 'import', '') +makeExcerpt('architecture/ts/app/app.component.ts', 'import', '')
@ -649,38 +617,34 @@ a#snake-case
.l-sub-section .l-sub-section
block snake-case-defn block snake-case-defn
:marked :marked
The practice of writing compound words or phrases such that each word is separated by an The practice of writing compound words or phrases such that an
underscore (`_`). This form is also known as **underscore case**. underscore (`_`) separates one word from the next. This form is also known as **underscore case**.
:marked :marked
## Service ## Service
.l-sub-section .l-sub-section
:marked :marked
Components are great and all, but what do we do with data or logic that are not associated For data or logic that is not associated
with a specific view or that we want to share across components? We build services! with a specific view or that you want to share across components, build services.
Applications often require services such as a hero data service or a logging service. Applications often require services such as a hero data service or a logging service.
Our components depend on these services to do the heavy lifting.
A service is a class with a focused purpose. A service is a class with a focused purpose.
We often create a service to implement features that are We often create a service to implement features that are
independent from any specific view, independent from any specific view,
provide share data or logic across components, or encapsulate external interactions. provide shared data or logic across components, or encapsulate external interactions.
See the [Services](!{docsLatest}/tutorial/toh-pt4.html) chapter of the tutorial to learn more. For more information, see the [Services](!{docsLatest}/tutorial/toh-pt4.html) page of the [Tour of Heroes](!{docsLatest}/tutorial/) tutorial.
:marked :marked
## Structural Directive ## Structural directive
.l-sub-section .l-sub-section
:marked :marked
A category of [Directive](#directive) that can A category of [directive](#directive) that can
shape or re-shape HTML layout, typically by adding, removing, or manipulating shape or reshape HTML layout, typically by adding, removing, or manipulating
elements and their children. elements and their children; for example, the `ngIf` "conditional element" directive and the `ngFor` "repeater" directive.
The `ngIf` "conditional element" directive and the `ngFor` "repeater" directive are Read more in the [Structural Directives](!{docsLatest}/guide/structural-directives.html) page.
good examples in this category.
See the [Structural Directives](!{docsLatest}/guide/structural-directives.html) chapter to learn more.
.l-main-section#T .l-main-section#T
:marked :marked
@ -688,14 +652,13 @@ a#snake-case
.l-sub-section .l-sub-section
:marked :marked
A template is a chunk of HTML that Angular uses to render a [view](#view) with A template is a chunk of HTML that Angular uses to render a [view](#view) with
the support and continuing guidance of an Angular [Directive](#directive), the support and continuing guidance of an Angular [directive](#directive),
most notably a [Component](#component). most notably a [component](#component).
We write templates in a special [Template Syntax](!{docsLatest}/guide/template-syntax.html).
+ifDocsFor('ts|js') +ifDocsFor('ts|js')
:marked :marked
## Template-Driven Forms ## Template-driven forms
.l-sub-section .l-sub-section
:marked :marked
A technique for building Angular forms using HTML forms and input elements in the view. A technique for building Angular forms using HTML forms and input elements in the view.
@ -704,47 +667,47 @@ a#snake-case
When building template-driven forms: When building template-driven forms:
- The "source of truth" is the template. The validation is defined using attributes on the individual input elements. - The "source of truth" is the template. The validation is defined using attributes on the individual input elements.
- [Two-way binding](#data-binding) with `ngModel` keeps the component model in synchronization with the user's entry into the input elements. - [Two-way binding](#data-binding) with `ngModel` keeps the component model in synchronization with the user's entry into the input elements.
- Behind the scenes, Angular creates a new control for each input element that has a `name` attribute and - Behind the scenes, Angular creates a new control for each input element, provided you have set up a `name` attribute and two-way binding for each input.
two-way binding set up.
- The associated Angular directives are all prefixed with `ng` such as `ngForm`, `ngModel`, and `ngModelGroup`. - The associated Angular directives are all prefixed with `ng` such as `ngForm`, `ngModel`, and `ngModelGroup`.
Template-driven forms are convenient, quick, and simple and are a good choice for many basic data entry form scenarios. Template-driven forms are convenient, quick, and simple. They are a good choice for many basic data entry form scenarios.
Learn how to build template-driven forms Read about how to build template-driven forms
in the [Forms](!{docsLatest}/guide/forms.html) chapter. in the [Forms](!{docsLatest}/guide/forms.html) page.
:marked :marked
## Template Expression ## Template expression
.l-sub-section .l-sub-section
:marked :marked
An expression is a !{_Lang}-like syntax that Angular evaluates within An expression is a !{_Lang}-like syntax that Angular evaluates within
a [data binding](#data-binding). Learn how to write template expressions a [data binding](#data-binding).
in the [Template Syntax](!{docsLatest}/guide/template-syntax.html#template-expressions) chapter.
Read about how to write template expressions
in the [Template Syntax](!{docsLatest}/guide/template-syntax.html#template-expressions) page.
:marked :marked
## Transpile ## Transpile
.l-sub-section .l-sub-section
:marked :marked
The process of transforming code written in one form of JavaScript The process of transforming code written in one form of JavaScript
(e.g., TypeScript) into another form of JavaScript (e.g., [ES5](#es5)). (for example, TypeScript) into another form of JavaScript (for example, [ES5](#es5)).
:marked :marked
## TypeScript ## TypeScript
.l-sub-section .l-sub-section
:marked :marked
A version of JavaScript that supports most [ECMAScript 2015](#ecmascript=2015) A version of JavaScript that supports most [ECMAScript 2015](#es2015)
language features and many features that may arrive in future versions language features such as [decorators](#decorator).
of JavaScript such as [Decorators](#decorator).
TypeScript is also noteable for its optional typing system which gives TypeScript is also noteable for its optional typing system, which gives
us compile-time type-checking and strong tooling support (e.g. "intellisense", us compile-time type checking and strong tooling support (for example, "intellisense",
code completion, refactoring, and intelligent search). Many code editors code completion, refactoring, and intelligent search). Many code editors
and IDEs support TypeScript either natively or with plugins. and IDEs support TypeScript either natively or with plugins.
TypeScript is the preferred language for Angular development although TypeScript is the preferred language for Angular development although
we are welcome to write in other JavaScript dialects such as [ES5](#es5). you can use other JavaScript dialects such as [ES5](#es5).
Learn more about TypeScript on its [website](http://www.typescriptlang.org/). Read more about TypeScript at [typescript.org](http://www.typescriptlang.org/).
a#U a#U
.l-main-section#V .l-main-section#V
@ -756,10 +719,10 @@ a#U
A view is a portion of the screen that displays information and responds A view is a portion of the screen that displays information and responds
to user actions such as clicks, mouse moves, and keystrokes. to user actions such as clicks, mouse moves, and keystrokes.
Angular renders a view under the control of one or more [Directives](#directive), Angular renders a view under the control of one or more [directives](#directive),
especially [Component](#component) directives and their companion [Templates](#template). especially [component](#component) directives and their companion [templates](#template).
The Component plays such a prominent role that we often The component plays such a prominent role that it's often
find it convenient to refer to a component as a view. convenient to refer to a component as a view.
Views often contain other views and any view might be loaded and unloaded Views often contain other views and any view might be loaded and unloaded
dynamically as the user navigates through the application, typically dynamically as the user navigates through the application, typically
@ -779,16 +742,16 @@ a#Y
a JavaScript application's asynchronous activity. a JavaScript application's asynchronous activity.
The browser DOM and JavaScript have a limited number The browser DOM and JavaScript have a limited number
of asynchronous activities, activities such as DOM events (e.g., clicks), of asynchronous activities, activities such as DOM events (for example, clicks),
[promises](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise), and [promises](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise), and
[XHR](https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest) [XHR](https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest)
calls to remote servers. calls to remote servers.
Zones intercept all of these activities and give a "zone client" the opportunity Zones intercept all of these activities and give a "zone client" the opportunity
to take action before and after the async activity completes. to take action before and after the async activity finishes.
Angular runs our application in a zone where it can respond to Angular runs your application in a zone where it can respond to
asynchronous events by checking for data changes and updating asynchronous events by checking for data changes, and updating
the information it displays via [data bindings](#data-binding). the information it displays via [data bindings](#data-binding).
Learn more about zones in this Learn more about zones in this

View File

@ -247,43 +247,49 @@ block ctor-syntax
We don't have to create an Angular injector. We don't have to create an Angular injector.
Angular creates an application-wide injector for us during the bootstrap process. Angular creates an application-wide injector for us during the bootstrap process.
+makeExample('dependency-injection/ts/app/main.ts', 'bootstrap', 'app/main.ts (excerpt)')(format='.') +makeExcerpt('app/main.ts', 'bootstrap')
:marked :marked
We do have to configure the injector by registering the **providers** We do have to configure the injector by registering the **providers**
that create the services our application requires. that create the services our application requires.
We'll explain what [providers](#providers) are later in this chapter. We'll explain what [providers](#providers) are later in this chapter.
Before we do, let's see an example of provider registration during bootstrapping:
+makeExample('dependency-injection/ts/app/main.1.ts', 'bootstrap-discouraged')(format='.') block register-provider-ngmodule
:marked
We can either register a provider within an [NgModule](ngmodule.html) or in application components
:marked ### Registering providers in an NgModule
The injector now knows about our `HeroService`. Here's our AppModule where we register a `Logger`, a `UserService`, and an `APP_CONFIG` provider.
An instance of our `HeroService` will be available for injection across our entire application.
Of course we can't help wondering about that comment telling us not to do it this way. - var app_module_ts = 'app/app.module.ts';
It *will* work. It's just not a best practice. +makeExcerpt(app_module_ts + ' (excerpt)', 'ngmodule', app_module_ts, { otl: /(providers:)/ })
The bootstrap provider option is intended for configuring and overriding Angular's own //- The preferred approach is to register application providers in application components.
preregistered services, such as its routing support. //- Because the `HeroService` is used within the *Heroes* feature area &mdash;
//- and nowhere else &mdash; the ideal place to register it is in the top-level `HeroesComponent`.
The preferred approach is to register application providers in application components.
Because the `HeroService` is used within the *Heroes* feature area &mdash;
and nowhere else &mdash; the ideal place to register it is in the top-level `HeroesComponent`.
:marked :marked
### Registering providers in a component ### Registering providers in a component
Here's a revised `HeroesComponent` that registers the `HeroService`. Here's a revised `HeroesComponent` that registers the `HeroService`.
- var stylePattern = { otl: /(providers:.*),/ }; - var stylePattern = { otl: /(providers:[^,]+),/ };
+makeExample('dependency-injection/ts/app/heroes/heroes.component.1.ts', 'full','app/heroes/heroes.component.ts', stylePattern)(format='.') +makeExample('app/heroes/heroes.component.1.ts', 'full', 'app/heroes/heroes.component.ts', stylePattern)(format='.')
:marked block ngmodule-vs-component
Look closely at the `providers` part of the `@Component` metadata. :marked
An instance of the `HeroService` is now available for injection in this `HeroesComponent` ### When to use the NgModule and when an application component?
and all of its child components.
The `HeroesComponent` itself doesn't happen to need the `HeroService`. On the one hand, a provider in an NgModule is registered in the root injector. That means that every provider
But its child `HeroListComponent` does, so we head there next. registered within an NgModule will be accessible in the _entire application_.
On the other hand, a provider registered in an application component is available only on that component and all its children.
We want the `APP_CONFIG` service to be available all across the application, but a `HeroService` is only used within the *Heroes*
feature area and nowhere else.
.l-sub-section
:marked
Also see *"Should I add app-wide providers to the root `AppModule` or the root `AppComponent`?"* in the [NgModule FAQ](../cookbook/ngmodule-faq.html#root-component-or-module).
:marked :marked
### Preparing the HeroListComponent for injection ### Preparing the HeroListComponent for injection
@ -387,7 +393,6 @@ block ctor-syntax
The constructor now asks for an injected instance of a `Logger` and stores it in a private property called `#{_priv}logger`. The constructor now asks for an injected instance of a `Logger` and stores it in a private property called `#{_priv}logger`.
We call that property within our `getHeroes` method when anyone asks for heroes. We call that property within our `getHeroes` method when anyone asks for heroes.
//- FIXME refer to Dart API when that page becomes available.
- var injUrl = '../api/core/index/Injectable-decorator.html'; - var injUrl = '../api/core/index/Injectable-decorator.html';
h3#injectable Why @Injectable()? h3#injectable Why @Injectable()?
:marked :marked
@ -462,7 +467,7 @@ block injectable-not-always-needed-in-ts
Our logger service is quite simple: Our logger service is quite simple:
+makeExample('dependency-injection/ts/app/logger.service.ts', null, 'app/logger.service.ts') +makeExample('app/logger.service.ts')
block real-logger block real-logger
//- N/A //- N/A
@ -470,9 +475,9 @@ block real-logger
:marked :marked
We're likely to need the same logger service everywhere in our application, We're likely to need the same logger service everywhere in our application,
so we put it in the project's `#{_appDir}` folder, and so we put it in the project's `#{_appDir}` folder, and
we register it in the `providers` #{_array} of the metadata for our application root component, `AppComponent`. we register it in the `providers` #{_array} of our application !{_moduleVsComp}, `!{_AppModuleVsAppComp}`.
+makeExcerpt('app/providers.component.ts','providers-logger','app/app.component.ts (excerpt)') +makeExcerpt('app/providers.component.ts (excerpt)', 'providers-logger','app/app.module.ts')
:marked :marked
If we forget to register the logger, Angular throws an exception when it first looks for the logger: If we forget to register the logger, Angular throws an exception when it first looks for the logger:
@ -497,7 +502,7 @@ code-example(format="nocode").
We must register a service *provider* with the injector, or it won't know how to create the service. We must register a service *provider* with the injector, or it won't know how to create the service.
Earlier we registered the `Logger` service in the `providers` #{_array} of the metadata for the `AppComponent` like this: Earlier we registered the `Logger` service in the `providers` #{_array} of the metadata for the `AppModule` like this:
+makeExample('dependency-injection/ts/app/providers.component.ts','providers-logger') +makeExample('dependency-injection/ts/app/providers.component.ts','providers-logger')
@ -766,7 +771,7 @@ block what-should-we-use-as-token
The TypeScript interface disappears from the generated JavaScript. The TypeScript interface disappears from the generated JavaScript.
There is no interface type information left for Angular to find at runtime. There is no interface type information left for Angular to find at runtime.
//- FIXME simplify once APIs are defined for Dart. //- FIXME update once https://github.com/dart-lang/angular2/issues/16 is addressed.
- var opaquetoken = _docsFor == 'dart' ? '<b>OpaqueToken</b>' : '<a href="../api/core/index/OpaqueToken-class.html"><b>OpaqueToken</b></a>' - var opaquetoken = _docsFor == 'dart' ? '<b>OpaqueToken</b>' : '<a href="../api/core/index/OpaqueToken-class.html"><b>OpaqueToken</b></a>'
:marked :marked
### OpaqueToken ### OpaqueToken
@ -796,9 +801,9 @@ block what-should-we-use-as-token
block dart-map-alternative block dart-map-alternative
:marked :marked
Or we can provide and inject the configuration object in our top-level `AppComponent`. Or we can provide and inject the configuration object in an ngModule like `AppModule`.
+makeExcerpt('app/app.component.ts','providers') +makeExcerpt('app/app.module.ts','ngmodule-providers')
#optional #optional
:marked :marked

View File

@ -1,19 +1,6 @@
block includes block includes
include ../_util-fns include ../_util-fns
- var _angular_io = 'angular.io';
figure
img(src="/resources/images/devguide/intro/people.png" alt="Us" align="left" style="width:200px; margin-left:-40px;margin-right:10px")
:marked
This is a practical guide to Angular for experienced programmers who
are building client applications in HTML and #{_Lang}. <br class="l-clear-left">
## Organization
The documentation is divided into major thematic sections, each
a collection of pages devoted to that theme.
block js-alert
- var __lang = _docsFor || current.path[1] || 'ts'; - var __lang = _docsFor || current.path[1] || 'ts';
- var guideData = public.docs[__lang].latest.guide._data; - var guideData = public.docs[__lang].latest.guide._data;
@ -22,6 +9,19 @@ block js-alert
- if (!guideData[page].basics && !guideData[page].hide) { advancedLandingPage = page; break; } - if (!guideData[page].basics && !guideData[page].hide) { advancedLandingPage = page; break; }
- } - }
- var advancedUrl = './' + advancedLandingPage + '.html' - var advancedUrl = './' + advancedLandingPage + '.html'
:marked
This page describes the Angular documentation at a high level.
If you're new to Angular, you may want to visit "[Learning Angular](learning-angular.html)" first.
## Themes
The documentation is divided into major thematic sections, each
a collection of pages devoted to that theme.
block js-alert
- var top="vertical-align:top" - var top="vertical-align:top"
table(width="100%") table(width="100%")
col(width="15%") col(width="15%")
@ -30,19 +30,22 @@ table(width="100%")
td <b><a href="../quickstart.html">QuickStart</a></b> td <b><a href="../quickstart.html">QuickStart</a></b>
td td
:marked :marked
The foundation for every page and sample in this documentation. A first taste of Angular<span if-docs="ts"> with zero installation.
Run "Hello World" in an online code editor and start playing with live code</span>.
tr(style=top) tr(style=top)
td <b><a href="./">Guide</a></b> td <b>Guide</b>
td td
:marked :marked
The essential ingredients of Angular development. Learn the Angular basics (you're already here!) like the setup for local development,
displaying data and accepting user input, injecting application services into components,
and building simple forms.
tr(style=top) tr(style=top)
td <b><a href="../api">API Reference</a></b> td <b><a href="../api/">API Reference</a></b>
td td
:marked :marked
Authoritative details about each member of the Angular libraries. Authoritative details about each of the Angular libraries.
tr(style=top) tr(style=top)
td <b><a href="../tutorial">Tutorial</a></b> td <b><a href="../tutorial/">Tutorial</a></b>
td td
:marked :marked
A step-by-step, immersive approach to learning Angular that A step-by-step, immersive approach to learning Angular that
@ -53,62 +56,37 @@ table(width="100%")
:marked :marked
In-depth analysis of Angular features and development practices. In-depth analysis of Angular features and development practices.
tr(style=top) tr(style=top)
td <b><a href="../cookbook">Cookbook</a></b> td <b><a href="../cookbook/">Cookbook</a></b>
td td
:marked :marked
Recipes for specific application challenges, mostly code snippets with a minimum of exposition. Recipes for specific application challenges, mostly code snippets with a minimum of exposition.
:marked :marked
## Learning path A few early pages are written as tutorials and are clearly marked as such.
The rest of the pages highlight key points in code rather than explain each step necessary to build the sample.
You don't have to read the guide straight through. Most pages stand on their own. You can always get the full source through the #{_liveLink}s.
For those new to Angular, the recommended learning path runs through the *Guide* section:
1. For the big picture, read the [Architecture](architecture.html) overview.
1. Try [QuickStart](../quickstart.html). QuickStart is the "Hello, World" of Angular.
It shows you how to set up the libraries and tools you'll need to write *any* Angular app.
1. Take the *Tour of Heroes* [tutorial](../tutorial), which picks up where QuickStart leaves off,
and builds a simple data-driven app. The app demonstrates the essential characteristics of a professional application:
a sensible project structure, data binding, master/detail, services, dependency injection, navigation, and remote data access.
1. [Displaying Data](displaying-data.html) explains how to display information on the screen.
1. [User Input](user-input.html) covers how Angular responds to user behavior.
1. [Forms](forms.html) handles user data entry and validation within the UI.
1. [Dependency Injection](dependency-injection.html) is the way to build large, maintainable applications
from small, single-purpose parts.
1. [Template Syntax](template-syntax.html) is a comprehensive study of Angular template HTML.
After reading the above sections, you can skip to any other pages on this site.
## Code samples ## Code samples
Each page includes code snippets that you can reuse in your applications. Each page includes code snippets from a sample application that accompanies the page.
These snippets are excerpts from a sample application that accompanies the page. You can reuse these snippets in your applications.
Look for a link to a running version of that sample near the top of each page, Look for a link to a running version of that sample, often near the top of the page,
such as this <live-example name="architecture"></live-example> from the [Architecture](architecture.html) page. such as this <live-example name="architecture"></live-example> from the [Architecture](architecture.html) page.
<p if-docs="ts"> <span if-docs="ts">
The link launches a browser-based code editor where you can inspect, modify, save, and download the code. The link launches a browser-based, code editor where you can inspect, modify, save, and download the code.
</p> </span>
A few early pages are written as tutorials and are clearly marked as such.
The rest of the pages highlight key points in code rather than explain each step necessary to build the sample.
You can always get the full source through the #{_liveLink}.
## Reference pages ## Reference pages
- The [Cheat Sheet](cheatsheet.html) lists Angular syntax for common scenarios. * The [Cheat Sheet](cheatsheet.html) lists Angular syntax for common scenarios.
- The [Glossary](glossary.html) defines terms that Angular developers should know. * The [Glossary](glossary.html) defines terms that Angular developers should know.
- The [API Reference](../api/) is the authority on every public-facing member of the Angular libraries. <li if-docs="ts">The [Change Log](change-log.html) announces what's new and changed in the documentation.</li>
* The [API Reference](../api/) is the authority on every public-facing member of the Angular libraries.
## We welcome feedback! ## Feedback
- Use the [website GitHub repo](!{_ngDocRepoURL}) for **documentation** issues and pull requests. We welcome feedback!
- Use the [Angular GitHub repo](!{_ngRepoURL}) to report issues with **Angular** itself.
* Use the <a href="!{_ngDocRepoURL}" target="_blank" title="angular docs on github">!{_angular_io} Github repository</a> for **documentation** issues and pull requests.
* Use the <a href="!{_ngRepoURL}" target="_blank" title="angular source on github">Angular Github repository</a> to report issues with **Angular** itself.

View File

@ -0,0 +1,44 @@
block includes
include ../_util-fns
figure
img(src="/resources/images/devguide/intro/people.png" width="200px" height="152px" alt="Us" align="left" style="margin-left:-40px;margin-right:10px" )
:marked
Everyone learns differently.
You don't have to read the documentation straight through. Most pages stand on their own.
Those new to Angular may wish to follow this popular learning path.
<br class="l-clear-left">
:marked
1. [Setup](setup.html "Setup locally withe Quickstart seed") for local Angular development, if you haven't already done so.
1. Take the [*Tour of Heroes* tutorial](../tutorial "Tour of Heroes").
The *Tour of Heroes* takes you step-by-step from [setup](setup.html)
to a full-featured example that demonstrates the essential characteristics of a professional application:
a sensible project structure, data binding, master/detail, services, dependency injection, navigation, and remote data access.
1. <a id="architecture"></a>Read the [Architecture](architecture.html) overview for the big picture.
<li if-docs="ts"><p>
[The Root Module](appmodule.html) introduces the `NgModule` class that tells Angular how to compile and run your application.
</p></li>
1. [Displaying Data](displaying-data.html) shows how data binding puts component property values on screen.
1. [User Input](user-input.html) explains how to respond to user-initiated DOM events.
1. [Forms](forms.html) covers data entry and validation within the UI.
1. [Dependency Injection](dependency-injection.html) is the way to build large, maintainable applications
from small, single-purpose parts.
1. [Template Syntax](template-syntax.html) is a comprehensive study of Angular template HTML.
After reading the above sections, feel free to skip around among the other pages on this site.
.l-sub-section
:marked
### Next Step
Try the [tutorial](../tutorial "Tour of Heroes") if you're ready to start coding or
visit the [Architecture](architecture.html "Basic Concepts") guide if you prefer to learn the basic concepts first.

View File

@ -648,8 +648,9 @@ a#in-mem-web-api
:marked :marked
## Appendix: Tour of Heroes in-memory server ## Appendix: Tour of Heroes in-memory server
If we only cared to retrieve data, we could tell Angular to get the heroes from a `heroes.json` file like this one: If the app only needed to retrieve data, you could get the heroes from a `heroes.json` file:
+makeJson('server-communication/ts/app/heroes.json', null, 'app/heroes.json')(format=".") - var _heroesJsonPath = (_docsFor == 'dart' ? 'web' : 'app') + '/heroes.json';
+makeJson('server-communication/' + _docsFor + '/' + _heroesJsonPath, null, _heroesJsonPath)(format=".")
.l-sub-section .l-sub-section
:marked :marked
We wrap the heroes array in an object with a `data` property for the same reason that a data server does: We wrap the heroes array in an object with a `data` property for the same reason that a data server does:

View File

@ -0,0 +1,180 @@
block includes
include ../_util-fns
- var _prereq = 'node and npm';
- var _playground = 'playground';
- var _Install = 'Install';
//- npm commands
- var _install = 'install';
- var _start = 'start';
a#develop-locally
:marked
## Setup a local development environment
<span if-docs="ts">
The <live-example name=quickstart>QuickStart live-coding</live-example> example is an Angular _playground_.
It's not where you'd develop a real application.
You [should develop locally](#why-locally "Why develop locally") on your own machine ... and that's also how we think you should learn Angular.
</span>
Setting up a new project on your machine is quick and easy with the **QuickStart seed**,
maintained [on github](!{_qsRepo} "Install the github QuickStart repo").
Make sure you have [!{_prereq} installed](#install-prerequisites "What if you don't have !{_prereq}?").
Then ...
1. Create a project folder (you can call it `quickstart` and rename it later).
1. [Clone](#clone "Clone it from github") or [download](#download "download it from github") the **QuickStart seed** into your project folder.
1. !{_Install} [!{_npm}](#install-prerequisites "What if you don't have !{_prereq}?") packages.
1. Run `!{_npm} !{_start}` to launch the sample application.
a#clone
:marked
### Clone
Perform the _clone-to-launch_ steps with these terminal commands.
code-example(language="sh" class="code-shell").
git clone !{_qsRepo}.git quickstart
cd quickstart
!{_npm} !{_install}
!{_npm} !{_start}
a#download
:marked
### Download
<a href="!{_qsRepoZip}" title="Download the QuickStart seed repository">Download the QuickStart seed</a>
and unzip it into your project folder. Then perform the remaining steps with these terminal commands.
code-example(language="sh" class="code-shell").
cd quickstart
!{_npm} !{_install}
!{_npm} !{_start}
.l-main-section#seed
:marked
## What's in the QuickStart seed?
block qs-seed
:marked
The **QuickStart seed** contains the same application as the QuickStart playground
and even has <live-example>its own _playground_</live-example>
that accomodates development of richer examples in a live coding environment.
But it's true purpose is to provide a solid foundation for _local_ development.
Consequently, there are _many more files_ in the project folder on your machine,
most of which you can [learn about later](setup-systemjs-anatomy.html "Setup Anatomy").
block core-files
:marked
Focus on the following three TypeScript (`.ts`) files in the **`/app`** folder.
.filetree
.file app
.children
.file app.component.ts
.file app.module.ts
.file main.ts
+makeTabs(`
setup/ts/app/app.component.ts,
setup/ts/app/app.module.ts,
setup/ts/app/main.ts
`, '', `
app/app.component.ts,
app/app.module.ts,
app/main.ts
`)(format='.')
:marked
All guides and cookbooks have _at least these core files_. Each file has a distinct purpose and evolves independently as the application grows.
style td, th {vertical-align: top}
table(width="100%")
col(width="20%")
col(width="80%")
tr
th File
th Purpose
tr
td <ngio-ex>app.component.ts</ngio-ex>
td
:marked
Defines the same `AppComponent` as the one in the QuickStart !{_playground}.
It is the **root** component of what will become a tree of nested components
as the application evolves.
tr(if-docs="ts")
td <code>app.module.ts</code>
td
:marked
Defines `AppModule`, the [root module](appmodule.html "AppModule: the root module") that tells Angular how to assemble the application.
Right now it declares only the `AppComponent`.
Soon there will be more components to declare.
tr
td <ngio-ex>main.ts</ngio-ex>
td
:marked
Compiles the application with the [JiT compiler](../glossary.html#jit)
and [bootstraps](appmodule.html#main "bootstrap the application") the application to run in the browser.
That's a reasonable choice for the development of most projects and
it's the only viable choice for a sample running in a _live-coding_ environment like Plunker.
You'll learn about alternative compiling and deployment options later in the documentation.
.l-sub-section
:marked
### Next Step
If you're new to Angular, we recommend staying on the [learning path](learning-angular.html).
br
br
a#install-prerequisites
.l-main-section
:marked
## Appendix: !{_prereq}
block install-tooling
:marked
Node.js and npm are essential to modern web development with Angular and other platforms.
Node powers client development and build tools.
The _npm_ package manager, itself a _node_ application, installs JavaScript libraries.
<a href="https://docs.npmjs.com/getting-started/installing-node" target="_blank" title="Installing Node.js and updating npm">
Get them now</a> if they're not already installed on your machine.
**Verify that you are running node `v4.x.x` or higher and npm `3.x.x` or higher**
by running the commands `node -v` and `npm -v` in a terminal/console window.
Older versions produce errors.
We recommend [nvm](https://github.com/creationix/nvm) for managing multiple versions of node and npm.
You may need [nvm](https://github.com/creationix/nvm) if you already have projects running on your machine that
use other versions of node and npm.
+ifDocsFor('ts')
a#why-locally
.l-main-section
:marked
## Appendix: Why develop locally
<live-example>Live coding</live-example> in the browser is a great way to explore Angular.
Links on almost every documentation page open completed samples in the browser.
You can play with the sample code, share your changes with friends, and download and run the code on your own machine.
The [QuickStart](../quickstart.html "Angular QuickStart Playground") shows just the `AppComponent` file.
It creates the equivalent of `app.module.ts` and `main.ts` internally _for the playground only_.
so the reader can discover Angular without distraction.
The other samples are based on the QuickStart seed.
As much fun as this is ...
* you can't ship your app in plunker
* you aren't always online when writing code
* transpiling TypeScript in the browser is slow
* the type support, refactoring, and code completion only work in your local IDE
Use the <live-example><i>live coding</i></live-example> environment as a _playground_,
a place to try the documentation samples and experiment on your own.
It's the perfect place to reproduce a bug when you want to
<a href="https://github.com/angular/angular.io/issues/new" target="_blank" title="File a documentation issue">file a documentation issue</a> or
<a href="https://github.com/angular/angular/issues/new" target="_blank" title="File an Angular issue">file an issue with Angular itself</a>.
For real development, we strongly recommend [developing locally](#develop-locally).

View File

@ -1,382 +1,45 @@
block includes block includes
include _util-fns include _util-fns
- var _Install = 'Install' - var _on_Plunkr = 'on Plunkr';
- var _prereq = 'Node.js and npm'
- var _angular_browser_uri = '@angular/platform-browser-dynamic'
- var _angular_core_uri = '@angular/core'
- var _stepInit = 4 // Step # after NgModule step
- var _quickstartSrcURL='https://github.com/angular/quickstart/blob/master/README.md'
//- TS/Dart shared step counter :marked
- var step = _stepInit Angular applications are made of _components_.
A _component_ is the combination of an HTML template and a component class that controls a portion of the screen. Here is an example of a component that displays a simple string:
+makeExample('app/app.component.ts')(format='.')
:marked
You can try this out without installing anything. Open the <live-example>QuickStart example !{_on_Plunkr}</live-example> in another tab
and follow along.
Every component begins with an `@Component` [!{_decorator}](glossary.html#!{_decorator} '"!{_decorator}" explained')
<span if-docs="ts">function</span> that
<span if-docs="ts">takes a _metadata_ object. The metadata object</span> describes how the HTML template and component class work together.
The `selector` property tells Angular to display the component inside a custom `<my-app>` tag in the `index.html`.
+makeExample('index.html','my-app','index.html (inside <body>)')(format='.')
:marked
The `template` property defines a message inside an `<h1>` header.
The message starts with "Hello" and ends with `{{name}}`
which is an Angular [interpolation binding](guide/displaying-data.html) expression.
At runtime, Angular replaces `{{name}}` with the value of the component's `name` property.
In the example, change the component class's `name` property from `'Angular'` to `'World'` and see what happens.
Interpolation binding is one of many Angular features you'll discover in this documentation.
+ifDocsFor('ts') +ifDocsFor('ts')
aside.is-right
:marked
The live example link opens the finished application in
<a href="http://plnkr.co/" title="Plunker" target="_blank">Plunker</a> so that you can interact
with the code. You'll find live examples at the start of most sections.
:marked
The QuickStart application has the structure of a real-world Angular application and
displays the simple message:
figure.image-display
img(src='/resources/images/devguide/quickstart/hello-angular.png' alt="Output of QuickStart app")
:marked
**Try it out**. Here's a link to a <live-example></live-example>.
+ifDocsFor('ts')
:marked
You can also <a href="!{_quickstartSrcURL}" target="_blank">
clone the entire QuickStart application</a> from GitHub.
h1 Build this application!
:marked
- [Prerequisite](#prereq): Install #{_prereq}.
- [Step 1](#create-and-configure): Create and configure the project.
- [Step 2](#ngmodule): Create your application.
<li if-docs="ts">[Step 3](#root-component): Create a component and add it to your application.</li>
- [Step !{step++}](#main): Start up your application.
- [Step !{step++}](#index): Define the web page that hosts the application.
- [Step !{step++}](#build-and-run): Build and run the application.
- [Step !{step++}](#make-changes): Make some live changes.
- [Wrap up and Next Steps](#wrap-up)
- var step = _stepInit // reinitialize step counter for headers to come
.l-main-section#prereq
h2 Prerequisite: Install #{_prereq}
block setup-tooling
:marked
If Node.js and npm aren't already on your machine, <a href="http://blog.npmjs.org/post/85484771375/how-to-install-npm"
target="_blank">install them</a>. Our examples require node **v4.x.x** or higher and
npm **3.x.x** or higher. To check which version you are using, run `node -v` and `npm -v`
in a terminal window.
.l-main-section
h2#create-and-configure Step 1: Create and configure the project
- var _package_and_config_files = _docsFor == 'dart' ? 'pubspec.yaml' : 'configuration files'
:marked
In this step you will:
* [Create the project folder](#create-the-project-folder)
* [Create #{_package_and_config_files}](#add-config-files)
* [#{_Install} packages](#install-packages)
h3 Create the project folder
:marked
Using a terminal window, create a directory for the project, and change into this
directory.
- var _ = _docsFor == 'dart' ? '_' : '-';
code-example(language="sh" class="code-shell").
mkdir angular!{_}quickstart
cd angular!{_}quickstart
h3#add-config-files Create #{_package_and_config_files}
block package-and-config-files
- var _tsconfigUri = 'guide/typescript-configuration.html#tsconfig'
p Our typical Angular project needs several configuration files:
ul
li.
#[b package.json] identifies npm package dependencies for the project.
li.
#[b tsconfig.json] defines how the TypeScript compiler generates JavaScript from the project's
files.
li.
#[b systemjs.config.js] provides information to a module loader about where to find
application modules, and registers all the necessary packages. It also
contains other packages that will be needed by later documentation examples.
p.
Create each of these files in your project directory. Populate them by pasting in text from
the tabs in the example box below.
a#config-files
+makeTabs(`
quickstart/ts/package.1.json,
quickstart/ts/tsconfig.1.json,
quickstart/ts/systemjs.config.1.js
`, '', `
package.json,
tsconfig.json,
systemjs.config.js
`)
p.
Learn more about these configuration files in the
#[a(href="guide/npm-packages.html") Npm Package Configuration] guide and the
#[a(href="#{_tsconfigUri}") TypeScript Configuration] guide.
A detailed discussion of module loading is beyond the scope of this guide.
.callout.is-helpful .callout.is-helpful
header SystemJS or Webpack? header A word about TypeScript
p. p.
Although we use SystemJS for illustrative purposes here, it's only one option for loading This example is written in <a href="http://www.typescriptlang.org/" target="_blank" title="TypeScript">TypeScript</a>, a superset of JavaScript. Angular
modules. Use the module loader that you prefer. For Webpack and Angular, see <a uses TypeScript because its types make it easy to support developer productivity with tooling. You can also write Angular code in JavaScript; <a href="cookbook/ts-to-js.html">this guide</a> explains how.
href="guide/webpack.html" >
Webpack: an Introduction</a>. Or, learn more about SystemJS configuration in general <a href="https://github.com/systemjs/systemjs/blob/master/docs/config-api.md" target="_blank">here</a>.
.l-sub-section
h3#install-packages #{_Install} packages
block install-packages
:marked
Using npm from the command line, install the packages listed in `package.json` with the command:
code-example(language="sh" class="code-shell").
npm install
:marked
Error messages&mdash;in red&mdash;might appear during the install, and you might see `npm WARN` messages. As long as there are no `npm ERR!` messages at the end, you can assume success.
:marked
You should now have the following structure:
.filetree
.file angular-quickstart
.children
.file node_modules ...
.file package.json
.file systemjs.config.js
.file tsconfig.json
:marked
You're now ready to write some code!
.l-main-section
h2#ngmodule Step 2: Create your application
block create-your-app
:marked
You compose Angular applications into closely related blocks of functionality with
[NgModules](guide/ngmodule.html). Angular itself is split into separate Angular Modules. This
makes it possible for you to keep payload size small by only importing the parts of Angular
that your application needs.
Every Angular application has at least one module: the _root module_, named `AppModule` here.
:marked
**Create #{_an} #{_appDir} subfolder** off the project root directory:
code-example.code-shell.
mkdir #{_appDir}
+ifDocsFor('ts')
:marked
Create the file `app/app.module.ts` with the following content:
+makeExample('app/app.module.1.ts')(format='.')
:marked
This is the entry point to your application.
Since the QuickStart application is a web application that runs in a browser, your root module
needs to import the
[`BrowserModule`](../latest/api/platform-browser/index/BrowserModule-class.html)
from `@angular/platform-browser` to the `imports` array.
This is the smallest amount of Angular that is needed for a minimal application to run in the
browser.
The QuickStart application doesn't do anything else, so you don't need any other modules. In a real
application, you'd likely import [`FormsModule`](../latest/api/forms/index/FormsModule-class.html)
as well as [`RouterModule`](../latest/api/router/index/RouterModule-class.html) and
[`HttpModule`](../latest/api/http/index/HttpModule-class.html). These are introduced in the
[Tour of Heroes Tutorial](./tutorial/).
.l-main-section
h2#root-component Step 3: Create a component and add it to your application
:marked
Every Angular application has at least one component: the _root component_, named `AppComponent`
here.
Components are the basic building blocks of Angular applications. A component controls a portion
of the screen&mdash;a *view*&mdash;through its associated template.
#app-component
:marked
**Create the component file** <span ngio-ex>app/app.component.ts</span> with the following content:
+makeExample('app/app.component.ts')
- var _decorator_function = _docsFor == 'dart' ? 'annotation' : 'decorator function';
:marked
The QuickStart application has the same essential structure as any other Angular component:
* **An import statement**. Importing gives your component access to
Angular's core [`@Component` !{_decorator_function}](./api/core/index/Component-decorator.html).
* **A @Component #{_decorator}** that associates *metadata* with the
`AppComponent` component class:
- a *selector* that specifies a simple CSS selector for an HTML element that represents
the component.
- a *template* that tells Angular how to render the component's view.
* **A component class** that controls the appearance and behavior of a view
through its template. Here, you only have the root component, `AppComponent`. Since you don't
need any application logic in the simple QuickStart example, it's empty.
+ifDocsFor('ts')
:marked
You *export* the `AppComponent` class so that you can *import* it into the application that you
just created.
Edit the file `app/app.module.ts` to import your new `AppComponent` and add it in the
declarations and bootstrap fields in the `NgModule` decorator:
+makeExample('app/app.module.ts')
.l-main-section
h2#main Step !{step++}: Start up your application
block create-main
:marked
Now you need to tell Angular to start up your application.
Create the file `app/main.ts` with the following content:
+makeExample('app/main.ts')
- var _pBD_bootstrapModule = _docsFor == 'dart' ? _bootstrapModule : 'platformBrowserDynamic().bootstrapModule'
:marked
This code initializes the platform that your application runs in, then uses the platform to
bootstrap your `!{_AppModuleVsAppComp}`.
### Why create separate *<span ngio-ex>main.ts</span>*<span if-docs="ts">, app module</span> and app component files?
App bootstrapping is a separate concern from<span if-docs="ts"> creating a module or</span>
presenting a view. Testing the component is much easier if it doesn't also try to run the entire application.
.callout.is-helpful
header Bootstrapping is platform-specific
:marked
Because the QuickStart application runs directly in the browser, `main.ts` imports the
`!{_platformBrowserDynamicVsBootStrap}` function from `#{_angular_browser_uri}`, not
`#{_angular_core_uri}`. On a mobile device, you might load a !{_moduleVsComp} with
[Apache Cordova](https://cordova.apache.org/) or
[NativeScript](https://www.nativescript.org/), using a bootstrap function that's specific
to that platform.
.l-main-section
h2#index Step !{step++}: Define the web page that hosts the application
:marked
In the *#{_indexHtmlDir}* folder,
create an `index.html` file and paste the following lines into it:
+makeExample('index.html')
block commentary-on-index-html
:marked
The noteworthy sections here are:
* JavaScript libraries: `core-js` polyfills for older browsers, the `zone.js` and
`reflect-metadata` libraries needed by Angular, and the `SystemJS` library for module loading.
* Configuration file for `SystemJS`, and a script
where you import and run the `app` module which refers to the `main` file that you just
wrote.
* The `<my-app>` tag in the `<body>` which is *where your app lives!*
:marked
### Add some style
Styles aren't essential, but they're nice, and `index.html` assumes that you have
a stylesheet called `styles.css`.
Create a `styles.css` file in the *#{_indexHtmlDir}* folder, and start styling,
perhaps with the minimal styles shown below.
+makeExcerpt('styles.css (excerpt)', 'quickstart')
.callout.is-helpful
:marked
For the full set of master styles used by the documentation samples,
see [styles.css](https://github.com/angular/angular.io/blob/master/public/docs/_examples/_boilerplate/styles.css).
.l-main-section#build-and-run
h2 Step !{step++}: Build and run the application
block run-app
:marked
Open a terminal window and enter this command:
code-example.code-shell.
npm start
aside.is-right
:marked :marked
[Read more](https://github.com/angular/quickstart/blob/master/README.md#npm-scripts) about ### Next step
other useful npm scripts included in this example's `package.json`.
:marked To learn how to write a real application, your next step is to set up a local development
That command runs the following two parallel node processes: environment and begin exploring with code. The [**Developer Guide**](guide/index.html)
* The TypeScript compiler in watch mode. shows you how.
* A static file server called _lite-server_ that loads `index.html` in a browser
and refreshes the browser when application files change.
In a few moments, a browser tab should open and display the following:
figure.image-display
img(src='/resources/images/devguide/quickstart/hello-angular.png' alt="Output of QuickStart app")
block build-app
//- Nothing for ts.
.l-main-section#make-changes
h2 Step !{step++}: Make some live changes
:marked
Try changing the message in <span ngio-ex>app/app.component.ts</span> to "Hello Again Angular!".
block server-watching
:marked
The TypeScript compiler and `lite-server` will detect your change, recompile the TypeScript into JavaScript,
refresh the browser, and display your revised message.
Close the terminal window when you're done to terminate both the compiler and the server.
.l-main-section
h2#wrap-up Wrap up and next steps
:marked
The final project folder structure looks like this:
block project-file-structure
.filetree
.file angular-quickstart
.children
.file app
.children
.file app.component.ts
.file app.module.ts
.file main.ts
.file node_modules ...
.file index.html
.file package.json
.file styles.css
.file systemjs.config.js
.file tsconfig.json
:marked
To see the file contents, open the <live-example></live-example>.
.l-main-section
:marked
## What next?
This first application doesn't do much. It's basically "Hello, World" for Angular.
You wrote a little Angular component, created a simple `index.html`, and launched with a
static file server.
+ifDocsFor('ts')
:marked
You also created the basic application setup that you'll re-use for other
sections in this guide. From here, the changes you'll make in the
`package.json` or `index.html` files are only minor updates to add libraries or some css
stylesheets. You also won't need to revisit module loading again.
:marked
To take the next step and build a small application that demonstrates real features that you can
build with Angular, carry on to the [Tour of Heroes tutorial](./tutorial)!

View File

@ -79,5 +79,3 @@ figure.image-display
met in countless applications. Everything has a reason. met in countless applications. Everything has a reason.
And well meet many of the core fundamentals of Angular along the way. And well meet many of the core fundamentals of Angular along the way.
[Let's get started!](./toh-pt1.html)

View File

@ -3,7 +3,6 @@
block includes block includes
include ../_util-fns include ../_util-fns
- var _appRoutingTsVsAppComp = 'app.module.ts' - var _appRoutingTsVsAppComp = 'app.module.ts'
- var _declsVsDirectives = 'declarations'
- var _RoutesVsAtRouteConfig = 'Routes' - var _RoutesVsAtRouteConfig = 'Routes'
- var _RouterModuleVsRouterDirectives = 'RouterModule' - var _RouterModuleVsRouterDirectives = 'RouterModule'
- var _redirectTo = 'redirectTo' - var _redirectTo = 'redirectTo'
@ -511,10 +510,16 @@ block route-params
+makeExcerpt('app/hero-detail.component.ts (constructor)', 'ctor') +makeExcerpt('app/hero-detail.component.ts (constructor)', 'ctor')
+ifDocsFor('ts')
:marked
Also import the `switchMap` operator to use later with the route parameters `Observable`.
+makeExcerpt('app/hero-detail.component.ts (switchMap import)', 'rxjs-import')
:marked :marked
We tell the class that we want to implement the `OnInit` interface. We tell the class that we want to implement the `OnInit` interface.
+makeExcerpt('app/hero-detail.component.ts', 'implement', '')(format=".") +makeExcerpt('app/hero-detail.component.ts', 'implement', '')
block ngOnInit block ngOnInit
:marked :marked
@ -526,14 +531,27 @@ block ngOnInit
block extract-id block extract-id
:marked :marked
Notice how we extract the `id` by calling the `forEach` method Note how the `switchMap` operator maps the id in the observable route parameters
which will deliver our !{_array} of route parameters. to a new `Observable`, the result of the `HeroService.getHero` method.
If the user re-navigates to this component while a getHero request is still inflight,
switchMap cancels that old request before calling `HeroService.getHero` again.
- var _str2int = _docsFor == 'dart' ? '<code>int.parse</code> static method' : 'JavaScript (+) operator' - var _str2int = _docsFor == 'dart' ? '<code>int.parse</code> static method' : 'JavaScript (+) operator'
:marked :marked
The hero `id` is a number. Route parameters are *always strings*. The hero `id` is a number. Route parameters are *always strings*.
So we convert the route parameter value to a number with the !{_str2int}. So we convert the route parameter value to a number with the !{_str2int}.
+ifDocsFor('ts')
.l-sub-section
:marked
### Do I need to unsubscribe?
The `Router` manages the [observables](../guide/router.html#activated-route) it provides and localizes
the subscriptions. The subscriptions are cleaned up when the component is destroyed, protecting against
memory leaks, so we don't need to _unsubscribe_ from the route params `Observable`.
:marked
### Add *HeroService.getHero* ### Add *HeroService.getHero*
The problem with this bit of code is that `HeroService` doesn't have a `getHero` method! The problem with this bit of code is that `HeroService` doesn't have a `getHero` method!
@ -654,16 +672,16 @@ block extract-id
+makeExample('app/app-routing.module.ts') +makeExample('app/app-routing.module.ts')
:marked :marked
Noteworthy points, typical of _Routing Modules_: Noteworthy points, typical of _Routing Modules_:
* Pull the routes into a variable. You might export it in future and it clarifies the _Routing Module_ pattern. * Pulls the routes into a variable. You might export it in future and it clarifies the _Routing Module_ pattern.
* Add `RouterModule.forRoot(routes)` to `imports`. * Adds `RouterModule.forRoot(routes)` to `imports`.
* Add `RouterModule` to `exports` so that the components in the companion module have access to Router declarables * Adds `RouterModule` to `exports` so that the components in the companion module have access to Router declarables
such as `RouterLink` and `RouterOutlet`. such as `RouterLink` and `RouterOutlet`.
* No `declarations`! Declarations are the responsibility of the companion module. * No `declarations`! Declarations are the responsibility of the companion module.
* Add module `providers` for guard services if you have them; there are none in this example. * Adds module `providers` for guard services if you have them; there are none in this example.
### Update _AppModule_ ### Update _AppModule_
@ -676,7 +694,7 @@ block extract-id
null, null,
`app/app.module.ts (after), app/app.module.ts (before)`) `app/app.module.ts (after), app/app.module.ts (before)`)
:marked :marked
It's simpler and focused on indentifying the key pieces of the application. It's simpler and focused on identifying the key pieces of the application.
.l-main-section .l-main-section
:marked :marked
@ -870,9 +888,7 @@ block css-files
We can also create styles at the *application level* outside of any component. We can also create styles at the *application level* outside of any component.
Our designers provided some basic styles to apply to elements across the entire app. Our designers provided some basic styles to apply to elements across the entire app.
These correspond to the full set of master styles that we These correspond to the full set of master styles that we installed earlier during [setup](../guide/setup.html).
introduced earlier (see
[QuickStart, "Add some style"](../quickstart.html#add-some-style)).
Here is an excerpt: Here is an excerpt:
+makeExcerpt('styles.css (excerpt)', 'toh') +makeExcerpt('styles.css (excerpt)', 'toh')

View File

@ -618,3 +618,10 @@ block file-summary
hero-search.component.css, hero-search.component.css,
rxjs-extensions.ts` rxjs-extensions.ts`
) )
.l-sub-section
:marked
### Next Step
Return to the [learning path](../guide/learning-angular.html#architecture) where
you can read about the concepts and practices you discovered in this tutorial.

View File

@ -19,8 +19,7 @@
"icon": "query-builder", "icon": "query-builder",
"title": "快速起步", "title": "快速起步",
"subtitle": "TypeScript", "subtitle": "TypeScript",
"description": "Angular快速起步", "banner": "快速体验 Angular"
"banner": "本“快速起步”指南将演示如何用TypeScript构建并运行简单的Angular应用。"
}, },
"tutorial": { "tutorial": {

View File

@ -1,33 +1,34 @@
include _util-fns include _util-fns
:marked :marked
Good tools make application development quicker and easier to maintain than Good tools make application development quicker and easier to maintain than
if we did everything by hand. if we did everything by hand.
The [**Angular-CLI**](https://cli.angular.io/) is a **_command line interface_** tool The [**Angular-CLI**](https://cli.angular.io/) is a **_command line interface_** tool
that can create a project, add files, and perform a variety of on-going development tasks such that can create a project, add files, and perform a variety of on-going development tasks such
as testing, bundling, and deployment. as testing, bundling, and deployment.
Our goal in this CLI QuickStart chapter is to build and run a super-simple Angular Our goal in this CLI QuickStart chapter is to build and run a super-simple Angular
application in TypeScript, using Angular-CLI application in TypeScript, using Angular-CLI
while adhering to the [Style Guide](./guide/style-guide.html) recommendations that while adhering to the [Style Guide](./guide/style-guide.html) recommendations that
benefit _every_ Angular project. benefit _every_ Angular project.
By the end of the chapter, we'll have a basic understanding of development with the CLI By the end of the chapter, we'll have a basic understanding of development with the CLI
and a foundation for both these documentation samples and our real world applications. and a foundation for both these documentation samples and our real world applications.
We'll pursue these ends in the following high-level steps: We'll pursue these ends in the following high-level steps:
1. [Set up](#devenv) the development environment 1. [Set up](#devenv) the development environment
2. [Create](#create-proj) a new project and skeleton application 2. [Create](#create-proj) a new project and skeleton application
3. [Serve](#serve) the application 3. [Serve](#serve) the application
4. [Edit](#first-component) the application 4. [Edit](#first-component) the application
.l-main-section .l-main-section
h2#devenv Step 1. Set up the Development Environment h2#devenv Step 1. Set up the Development Environment
:marked :marked
We need to set up our development environment before we can do anything. We need to set up our development environment before we can do anything.
Install **[Node.js® and npm](https://nodejs.org/en/download/)** Install **[Node.js® and npm](https://nodejs.org/en/download/)**
if they are not already on your machine. if they are not already on your machine.
.l-sub-section .l-sub-section
@ -37,15 +38,15 @@ h2#devenv Step 1. Set up the Development Environment
Older versions produce errors. Older versions produce errors.
:marked :marked
Then **install the [Angular-CLI](https://github.com/angular/angular-cli)** globally. Then **install the [Angular-CLI](https://github.com/angular/angular-cli)** globally.
code-example(format=""). code-example(format="").
npm install -g angular-cli npm install -g angular-cli
.l-main-section .l-main-section
h2#create-project Step 2. Create a new project h2#create-project Step 2. Create a new project
:marked :marked
Open a terminal window. Open a terminal window.
.alert.is-important .alert.is-important
:marked :marked
_Windows Developers_: open a console window _as an **administrator**_. _Windows Developers_: open a console window _as an **administrator**_.
@ -75,9 +76,9 @@ code-example(format="").
:marked :marked
The `ng serve` command launches the server, watches our files, The `ng serve` command launches the server, watches our files,
and rebuilds the app as we make changes to the files. and rebuilds the app as we make changes to the files.
Open a browser on `http://localhost:4200/`; the app greets us with a message: Open a browser on `http://localhost:4200/`; the app greets us with a message:
figure.image-display figure.image-display
img(src='/resources/images/devguide/cli-quickstart/app-works.png' alt="Our app works!") img(src='/resources/images/devguide/cli-quickstart/app-works.png' alt="Our app works!")
@ -96,19 +97,19 @@ h2#first-component Step 4: Edit our first Angular component
:marked :marked
Open the component file and change the `title` property from _cli-quickstart works!_ to _My First Angular App_: Open the component file and change the `title` property from _cli-quickstart works!_ to _My First Angular App_:
+makeExample('cli-quickstart/ts/src/app/cli-quickstart.component.ts', 'title', 'src/app/cli-quickstart.component.ts')(format=".") +makeExample('cli-quickstart/ts/src/app/cli-quickstart.component.ts', 'title', 'src/app/cli-quickstart.component.ts')(format=".")
:marked :marked
The browser reloads automatically and we see the revised title. That's nice, but we can make it look better. The browser reloads automatically and we see the revised title. That's nice, but we can make it look better.
Open `src/app/cli-quickstart.component.css` and give our component some style Open `src/app/cli-quickstart.component.css` and give our component some style
+makeExample('cli-quickstart/ts/src/app/cli-quickstart.component.css', null, 'src/app/cli-quickstart.component.css')(format=".") +makeExample('cli-quickstart/ts/src/app/cli-quickstart.component.css', null, 'src/app/cli-quickstart.component.css')(format=".")
figure.image-display figure.image-display
img(src='/resources/images/devguide/cli-quickstart/hello-angular.png' alt="Output of QuickStart app") img(src='/resources/images/devguide/cli-quickstart/hello-angular.png' alt="Output of QuickStart app")
:marked :marked
Looking good! Looking good!
@ -177,7 +178,7 @@ h3#component-decorator @Component decorator
:marked :marked
This particular metadata object has four fields, a `moduleId`, a `selector` a `templateUrl` and an array of `styleUrls`. This particular metadata object has four fields, a `moduleId`, a `selector` a `templateUrl` and an array of `styleUrls`.
The **moduleId** specifies the location of _this_ component definition file The **moduleId** specifies the location of _this_ component definition file
so that Angular can find the corresponding _template_ and _style_ files with URLs that so that Angular can find the corresponding _template_ and _style_ files with URLs that
are relative to this component file. are relative to this component file.
@ -193,14 +194,14 @@ h3#component-decorator @Component decorator
The **templateUrl** locates the component's companion template HTML file. The **templateUrl** locates the component's companion template HTML file.
The template tells Angular how to render this component's view. The template tells Angular how to render this component's view.
Our template is a single `<h1>` element surrounding some peculiar Angular data binding syntax. Our template is a single `<h1>` element surrounding some peculiar Angular data binding syntax.
+makeExample('src/app/cli-quickstart.component.html', null, 'src/app/cli-quickstart.component.html')(format='.') +makeExample('src/app/cli-quickstart.component.html', null, 'src/app/cli-quickstart.component.html')(format='.')
:marked :marked
The `{{title}}` is an _interpolation_ binding that causes Angular to display the component's The `{{title}}` is an _interpolation_ binding that causes Angular to display the component's
`title` property. After out edit, Angular displays "Hello Angular!". `title` property. After out edit, Angular displays "Hello Angular".
We'll learn more about data binding as we read through the documentation. We'll learn more about data binding as we read through the documentation.
The **styleUrls** array specifies the location(s) of the component's private CSS style file(s). The **styleUrls** array specifies the location(s) of the component's private CSS style file(s).
The CLI generated an empty file for us and we added styles to it. The CLI generated an empty file for us and we added styles to it.
@ -216,10 +217,10 @@ h3#component-decorator @Component decorator
:marked :marked
### The *main.ts* file ### The *main.ts* file
How does Angular know what to do with our component? We have to tell it. How does Angular know what to do with our component? We have to tell it.
We _bootstrap_ our application in the file `main.ts`. We _bootstrap_ our application in the file `main.ts`.
+makeExcerpt('src/main.ts', 'important') +makeExcerpt('src/main.ts', 'important')
:marked :marked
@ -257,17 +258,17 @@ h3#component-decorator @Component decorator
We might launch the `CliQuickstartAppComponent` in multiple environments with different bootstrappers. We might launch the `CliQuickstartAppComponent` in multiple environments with different bootstrappers.
Testing the component is much easier if it doesn't also try to run the entire application. Testing the component is much easier if it doesn't also try to run the entire application.
Let's make the small extra effort to do it *the right way*. Let's make the small extra effort to do it *the right way*.
### Loading the application with SystemJS ### Loading the application with SystemJS
The CLI uses `System.js` to load the application. We just need to call `System.import` and pass it our `main.ts` The CLI uses `System.js` to load the application. We just need to call `System.import` and pass it our `main.ts`
file to boot our application. file to boot our application.
+makeExcerpt('src/index.html', 'import') +makeExcerpt('src/index.html', 'import')
:marked :marked
### Displaying the root component ### Displaying the root component
When Angular calls the `bootstrap` function in `main.ts`, it reads the `CliQuickstartAppComponent` When Angular calls the `bootstrap` function in `main.ts`, it reads the `CliQuickstartAppComponent`
metadata, finds the `cli-quickstart-app` selector, locates an element tag named `cli-quickstart-app` metadata, finds the `cli-quickstart-app` selector, locates an element tag named `cli-quickstart-app`
in the `<body>` tag of the `index.html`, and renders our application's view between those tags. in the `<body>` tag of the `index.html`, and renders our application's view between those tags.

View File

@ -2,7 +2,7 @@
"index": { "index": {
"title": "烹饪宝典", "title": "烹饪宝典",
"navTitle": "概览", "navTitle": "概览",
"description": "一组常见Angular应用场景的“烹饪宝典”" "intro": "一组常见Angular应用场景的“烹饪宝典”"
}, },
"aot-compiler": { "aot-compiler": {
@ -57,8 +57,8 @@
}, },
"ts-to-js": { "ts-to-js": {
"title": "从TypeScript到JavaScript", "title": "从 TypeScript JavaScript",
"intro": "把Angular的TypeScript范例转换为ES6和ES5 JavaScript" "intro": "把 Angular TypeScript 范例转换为 ES6 ES5 JavaScript"
}, },
"visual-studio-2015": { "visual-studio-2015": {

View File

@ -233,9 +233,9 @@ table(width="100%")
Angular 2没有引导指令。 Angular 2没有引导指令。
我们总是通过显式调用一个`bootstrap`函数,并传入应用模块的名字(`AppComponent`)来启动应用。 我们总是通过显式调用一个`bootstrap`函数,并传入应用模块的名字(`AppComponent`)来启动应用。
For more information see [Quick Start](../quickstart.html). For more information see the [Setup](../guide/setup.html) page.
要了解更多,参见[“快速起步”](../quickstart.html)。 要了解更多,参见[搭建本地开发环境](../guide/setup.html)。
tr(style=top) tr(style=top)
td td
:marked :marked

View File

@ -56,7 +56,7 @@ a#overview
这是本文档中展示过的标准开发方式。 这是本文档中展示过的标准开发方式。
它很不错,但是有自己的缺点。 它很不错,但是有自己的缺点。
JiT compilation incurs a runtime performance penalty. JiT compilation incurs a runtime performance penalty.
Views take longer to render because of the in-browser compilation step. Views take longer to render because of the in-browser compilation step.
The application is bigger because it includes the Angular compiler The application is bigger because it includes the Angular compiler
and a lot of library code that the application won't actually need. and a lot of library code that the application won't actually need.
@ -98,7 +98,7 @@ a#aot-jit
**渲染得更快** **渲染得更快**
With AoT, the browser downloads a pre-compiled version of the application. With AoT, the browser downloads a pre-compiled version of the application.
The browser loads executable code so it can render the application immediately, without waiting to compile the app first. The browser loads executable code so it can render the application immediately, without waiting to compile the app first.
使用AoT浏览器下载预编译版本的应用程序。 使用AoT浏览器下载预编译版本的应用程序。
@ -118,7 +118,7 @@ a#aot-jit
**需要下载的Angular框架体积更小** **需要下载的Angular框架体积更小**
There's no need to download the Angular compiler if the app is already compiled. There's no need to download the Angular compiler if the app is already compiled.
The compiler is roughly half of Angular itself, so omitting it dramatically reduces the application payload. The compiler is roughly half of Angular itself, so omitting it dramatically reduces the application payload.
如果应用已经编译过了自然不需要再下载Angular编译器了。 如果应用已经编译过了自然不需要再下载Angular编译器了。
@ -155,10 +155,10 @@ a#compile
### 为离线编译做准备 ### 为离线编译做准备
Take the <a href='/docs/ts/latest/quickstart.html'>QuickStart</a> as a starting point. Take the <a href='../guide/setup.html'>Setup</a> as a starting point.
A few minor changes to the lone `app.component` lead to these two class and html files: A few minor changes to the lone `app.component` lead to these two class and html files:
本烹饪书以<a href='/docs/ts/latest/quickstart.html'>“快速起步”</a>作为起始点。 本烹饪书以<a href='../guide/setup.html'>搭建本地开发环境</a>作为起始点。
只要单独对`app.component`文件的类文件和html文件做少量修改就可以了。 只要单独对`app.component`文件的类文件和html文件做少量修改就可以了。
+makeTabs( +makeTabs(
@ -182,7 +182,7 @@ code-example(format='.').
你要用`@angular/compiler-cli`包中提供的`ngc`编译器来代替TypeScript编译器`tsc`)。 你要用`@angular/compiler-cli`包中提供的`ngc`编译器来代替TypeScript编译器`tsc`)。
`ngc` is a drop-in replacement for `tsc` and is configured much the same way. `ngc` is a drop-in replacement for `tsc` and is configured much the same way.
`ngc`是一个`tsc`的高仿替代品,它们的配置方式几乎完全一样。 `ngc`是一个`tsc`的高仿替代品,它们的配置方式几乎完全一样。
@ -202,14 +202,14 @@ code-example(format='.').
`compilerOptions`部分只修改了一个属性:**把`module`设置为`es2015`。 `compilerOptions`部分只修改了一个属性:**把`module`设置为`es2015`。
这一点非常重要,我们会在后面的[摇树优化](#tree-shaking)部分解释为什么。 这一点非常重要,我们会在后面的[摇树优化](#tree-shaking)部分解释为什么。
What's really new is the `ngc` section at the bottom called `angularCompilerOptions`. What's really new is the `ngc` section at the bottom called `angularCompilerOptions`.
Its `"genDir"` property tells the compiler Its `"genDir"` property tells the compiler
to store the compiled output files in a new `aot` folder. to store the compiled output files in a new `aot` folder.
`ngc`区真正新增的内容是底部的`angularCompilerOptions`。 `ngc`区真正新增的内容是底部的`angularCompilerOptions`。
它的`"genDir"`属性告诉编译器把编译结果保存在新的`aot`目录下。 它的`"genDir"`属性告诉编译器把编译结果保存在新的`aot`目录下。
The `"skipMetadataEmit" : true` property prevents the compiler from generating metadata files with the compiled application. The `"skipMetadataEmit" : true` property prevents the compiler from generating metadata files with the compiled application.
Metadata files are not necessary when targeting TypeScript files, so there is no reason to include them. Metadata files are not necessary when targeting TypeScript files, so there is no reason to include them.
`"skipMetadataEmit" : true`属性阻止编译器为编译后的应用生成元数据文件。 `"skipMetadataEmit" : true`属性阻止编译器为编译后的应用生成元数据文件。
@ -237,11 +237,11 @@ code-example(format='.').
`ngc`希望`-p`选项指向一个`tsconfig.json`文件,或者一个包含`tsconfig.json`文件的目录。 `ngc`希望`-p`选项指向一个`tsconfig.json`文件,或者一个包含`tsconfig.json`文件的目录。
After `ngc` completes, look for a collection of _NgFactory_ files in the `aot` folder (the folder specified as `genDir` in `tsconfig-aot.json`). After `ngc` completes, look for a collection of _NgFactory_ files in the `aot` folder (the folder specified as `genDir` in `tsconfig-aot.json`).
在`ngc`完成时,会在`aot`目录下看到一组*NgFactory*文件(该目录是在`tsconfig-aot.json`的`genDir`属性中指定的)。 在`ngc`完成时,会在`aot`目录下看到一组*NgFactory*文件(该目录是在`tsconfig-aot.json`的`genDir`属性中指定的)。
These factory files are essential to the compiled application. These factory files are essential to the compiled application.
Each component factory creates an instance of the component at runtime by combining the original class file Each component factory creates an instance of the component at runtime by combining the original class file
and a JavaScript representation of the component's template. and a JavaScript representation of the component's template.
Note that the original component class is still referenced internally by the generated factory. Note that the original component class is still referenced internally by the generated factory.
@ -284,7 +284,7 @@ a#bootstrap
引导的方式从引导`AppModule`改成了引导生成的模块工厂:`AppModuleNgFactory`。 引导的方式从引导`AppModule`改成了引导生成的模块工厂:`AppModuleNgFactory`。
Switch from the `platformBrowserDynamic.bootstrap` used in JiT compilation to Switch from the `platformBrowserDynamic.bootstrap` used in JiT compilation to
`platformBrowser().bootstrapModuleFactory` and pass in the `AppModuleNgFactory`. `platformBrowser().bootstrapModuleFactory` and pass in the `AppModuleNgFactory`.
从使用JiT编译时的`platformBrowserDynamic.bootstrap`换成了`platformBrowser().bootstrapModuleFactory`,并把`AppModuleNgFactory`传进去。 从使用JiT编译时的`platformBrowserDynamic.bootstrap`换成了`platformBrowser().bootstrapModuleFactory`,并把`AppModuleNgFactory`传进去。
@ -318,7 +318,7 @@ a#tree-shaking
AoT编译为接下来通过一个叫做*摇树优化*的过程做好了准备。 AoT编译为接下来通过一个叫做*摇树优化*的过程做好了准备。
摇树优化器从上到下遍历依赖图谱,并且*摇掉*用不到的代码,这些代码就像是圣诞树中那些死掉的松针一样。 摇树优化器从上到下遍历依赖图谱,并且*摇掉*用不到的代码,这些代码就像是圣诞树中那些死掉的松针一样。
Tree Shaking can greatly reduce the downloaded size of the application Tree Shaking can greatly reduce the downloaded size of the application
by removing unused portions of both source and library code. by removing unused portions of both source and library code.
In fact, most of the reduction in small apps comes from removing unreferenced Angular features. In fact, most of the reduction in small apps comes from removing unreferenced Angular features.
@ -330,7 +330,7 @@ a#tree-shaking
例如,这个演示程序中没有用到`@angular/forms`库中的任何东西那么也就没有理由去下载这些与表单有关的Angular代码了。摇树优化可以帮你确保这一点。 例如,这个演示程序中没有用到`@angular/forms`库中的任何东西那么也就没有理由去下载这些与表单有关的Angular代码了。摇树优化可以帮你确保这一点。
Tree Shaking and AoT compilation are separate steps. Tree Shaking and AoT compilation are separate steps.
Tree Shaking can only target JavaScript code. Tree Shaking can only target JavaScript code.
AoT compilation converts more of the application to JavaScript, AoT compilation converts more of the application to JavaScript,
which in turn makes more of the application "Tree Shakable". which in turn makes more of the application "Tree Shakable".
@ -414,14 +414,14 @@ code-example(format='.').
可观察对象库*RxJS*是Angular所依赖的基础之一它就是发布成了ES5 JavaScript的*CommonJS*模块。 可观察对象库*RxJS*是Angular所依赖的基础之一它就是发布成了ES5 JavaScript的*CommonJS*模块。
Luckily there is a Rollup plugin that modifies _RxJs_ Luckily there is a Rollup plugin that modifies _RxJs_
to use the ES `import` and `export` statements that Rollup requires. to use the ES `import` and `export` statements that Rollup requires.
Rollup then preserves in the final bundle the parts of `RxJS` referenced by the application. Rollup then preserves in the final bundle the parts of `RxJS` referenced by the application.
幸运的是有一个Rollup插件它会修改*RxJS*以使用Rollup所需的ES`import`和`export`语句。 幸运的是有一个Rollup插件它会修改*RxJS*以使用Rollup所需的ES`import`和`export`语句。
然后Rollup就可以把该应用中用到的那部分`RxJS`代码留在“捆”文件中了。 然后Rollup就可以把该应用中用到的那部分`RxJS`代码留在“捆”文件中了。
+makeExample('cb-aot-compiler/ts/rollup-config.js','commonjs','rollup-config.js (CommonJs to ES2015 Plugin)')(format='.') +makeExample('cb-aot-compiler/ts/rollup-config.js','commonjs','rollup-config.js (CommonJs to ES2015 Plugin)')(format='.')
:marked :marked
*Minification* *Minification*
@ -475,7 +475,7 @@ a#load
## 加载捆文件 ## 加载捆文件
Loading the generated application bundle does not require a module loader like SystemJS. Loading the generated application bundle does not require a module loader like SystemJS.
Remove the scripts that concern SystemJS. Remove the scripts that concern SystemJS.
Instead, load the bundle file using a single `script` tag: Instead, load the bundle file using a single `script` tag:
@ -483,7 +483,7 @@ a#load
移除与SystemJS有关的那些脚本吧。 移除与SystemJS有关的那些脚本吧。
改用`script`标签来加载这些捆文件: 改用`script`标签来加载这些捆文件:
+makeExample('cb-aot-compiler/ts/index.html','bundle','index.html (load bundle)')(format='.') +makeExample('cb-aot-compiler/ts/index.html','bundle','index.html (load bundle)')(format='.')
a#serve a#serve
.l-main-section .l-main-section
@ -510,7 +510,7 @@ a#source-code
:marked :marked
## AoT QuickStart Source Code ## AoT QuickStart Source Code
## AoT快速开始源代码 ## AoT快速起步源代码
Here's the pertinent source code: Here's the pertinent source code:
@ -538,11 +538,11 @@ a#toh
## 英雄指南 ## 英雄指南
The sample above is a trivial variation of the QuickStart app. The sample above is a trivial variation of the QuickStart app.
In this section you apply what you've learned about AoT compilation and Tree Shaking In this section you apply what you've learned about AoT compilation and Tree Shaking
to an app with more substance, the tutorial [_Tour of Heroes_](../tutorial/toh-pt6.html). to an app with more substance, the tutorial [_Tour of Heroes_](../tutorial/toh-pt6.html).
上面的例子是快速开始应用的一个简单的变体。 上面的例子是《快速起步》应用的一个简单的变体。
在本节中,你将在一个更多内容的应用 - [英雄指南](../tutorial/toh-pt6.html)上使用从AoT编译和摇树优化学到的知识。 在本节中,你将在一个更多内容的应用 - [英雄指南](../tutorial/toh-pt6.html)上使用从AoT编译和摇树优化学到的知识。
### JiT in development, AoT in production ### JiT in development, AoT in production
@ -591,19 +591,13 @@ a#toh
The AoT version loads the entire application in a single script, `aot/dist/build.js`. The AoT version loads the entire application in a single script, `aot/dist/build.js`.
It does not need `SystemJS` or the `reflect-metadata` shim; those scripts are absent from its `index.html` It does not need `SystemJS` or the `reflect-metadata` shim; those scripts are absent from its `index.html`
AoT版本用一个单独的脚本来加载整个应用 - `aot/dist/build.js`。它不需要`SystemJS`和`reflect-metadata`垫片,所以它们不会出现在`index.html`中。
***main.ts***
***main.ts*** ***main.ts***
JiT and AoT applications boot in much the same way but require different Angular libraries to do so. JiT and AoT applications boot in much the same way but require different Angular libraries to do so.
The key differences, covered in the [Bootstrap](#bootstrap) section above, The key differences, covered in the [Bootstrap](#bootstrap) section above,
are evident in these `main` files which can and should reside in the same folder: are evident in these `main` files which can and should reside in the same folder:
JiT和AoT应用启动非常相似但需要加载不同的Angular库来实现。 AoT版本用一个单独的脚本来加载整个应用 - `aot/dist/build.js`。它不需要`SystemJS`和`reflect-metadata`垫片,所以它们不会出现在`index.html`中。
主要的差异,见上面的[启动](#bootstrap)一节,
可以在那些`main`文件中明显看出区别,它们可以也应该位于同一目录。
+makeTabs( +makeTabs(
`toh-6/ts/app/main-aot.ts, `toh-6/ts/app/main-aot.ts,
@ -623,7 +617,7 @@ a#toh
For example, a `'hero.component.html'` URL means that the template file is a sibling of its companion `hero.component.ts` file. For example, a `'hero.component.html'` URL means that the template file is a sibling of its companion `hero.component.ts` file.
AoT编译器要求`@Component`外部模板和CSS文件的路径是相对组件的。 AoT编译器要求`@Component`外部模板和CSS文件的路径是相对组件的。
意思是,`@Component.templateUrl`的值是一个相对组件类文件`foo.component.html`的路径,不管`foo.component.ts`在项目的哪个目录 意思是,`@Component.templateUrl`的值是*相对*组件类文件的路径,比如`'hero.component.html'`路径的意思是模板文件在`hero.component.ts`文件的隔壁
While JiT app URLs are more flexible, stick with _component-relative_ URLs for compatibility with AoT compilation. While JiT app URLs are more flexible, stick with _component-relative_ URLs for compatibility with AoT compilation.
@ -769,7 +763,7 @@ code-example(format='.').
### 检查包裹 ### 检查包裹
It's fascinating to see what the generated JavaScript bundle looks like after Rollup. It's fascinating to see what the generated JavaScript bundle looks like after Rollup.
The code is minified, so you won't learn much from inspecting the bundle directly. The code is minified, so you won't learn much from inspecting the bundle directly.
But the <a href="https://github.com/danvk/source-map-explorer/blob/master/README.md" target="_blank">source-map-explorer</a> But the <a href="https://github.com/danvk/source-map-explorer/blob/master/README.md" target="_blank">source-map-explorer</a>
tool can be quite revealing. tool can be quite revealing.

View File

@ -12,7 +12,7 @@ include ../_util-fns
For an in-depth look at each fundamental concepts in component communication, we can find detailed description and For an in-depth look at each fundamental concepts in component communication, we can find detailed description and
samples in the [Component Communication]() document. samples in the [Component Communication]() document.
要深入了解组件通讯的各个基本概念,在[组件通讯Component Communication]()文档中可以找到详细的描述和例子。 要深入了解组件通讯的各个基本概念,在[组件通讯]()文档中可以找到详细的描述和例子。
<a id="toc"></a> <a id="toc"></a>
:marked :marked

View File

@ -2,80 +2,80 @@ include ../_util-fns
:marked :marked
We can't always justify the cost and time to build handcrafted forms, especially if we'll need a great number of them, they're similar to each other, and they change frequently to meet rapidly changing business and regulatory requirements. We can't always justify the cost and time to build handcrafted forms, especially if we'll need a great number of them, they're similar to each other, and they change frequently to meet rapidly changing business and regulatory requirements.
有时候手动编写和维护表单所需工作量和时间会过大。特别是在需要编写大量表单时。表单都很相似,而且随着业务和监管需求的迅速变化,表单也要随之变化,这样维护的成本过高。 有时候手动编写和维护表单所需工作量和时间会过大。特别是在需要编写大量表单时。表单都很相似,而且随着业务和监管需求的迅速变化,表单也要随之变化,这样维护的成本过高。
It may be more economical to create the forms dynamically, based on metadata that describe the business object model. It may be more economical to create the forms dynamically, based on metadata that describe the business object model.
基于业务对象模型的元数据,动态创建表单可能会更划算。 基于业务对象模型的元数据,动态创建表单可能会更划算。
In this cookbook we show how to use `formGroup` to dynamically render a simple form with different control types and validation. In this cookbook we show how to use `formGroup` to dynamically render a simple form with different control types and validation.
It's a primitive start. It's a primitive start.
It might evolve to support a much richer variety of questions, more graceful rendering, and superior user experience. It might evolve to support a much richer variety of questions, more graceful rendering, and superior user experience.
All such greatness has humble beginnings. All such greatness has humble beginnings.
在此烹饪宝典中,我们会展示如何利用`formGroup`来动态渲染一个简单的表单,包括各种控件类型和验证规则。 在此烹饪宝典中,我们会展示如何利用`formGroup`来动态渲染一个简单的表单,包括各种控件类型和验证规则。
这个起点很简陋,但可以在这个基础上添加丰富多彩的问卷问题、更优美的渲染以及更卓越的用户体验。 这个起点很简陋,但可以在这个基础上添加丰富多彩的问卷问题、更优美的渲染以及更卓越的用户体验。
In our example we use a dynamic form to build an online application experience for heroes seeking employment. In our example we use a dynamic form to build an online application experience for heroes seeking employment.
The agency is constantly tinkering with the application process. The agency is constantly tinkering with the application process.
We can create the forms on the fly *without changing our application code*. We can create the forms on the fly *without changing our application code*.
在本例中,我们使用动态表单,为正在找工作的英雄们创建一个在线申请表。英雄管理局会不断修改申请流程,我们要在*不修改应用代码*的情况下,动态创建这些表单。 在本例中,我们使用动态表单,为正在找工作的英雄们创建一个在线申请表。英雄管理局会不断修改申请流程,我们要在*不修改应用代码*的情况下,动态创建这些表单。
<a id="toc"></a> <a id="toc"></a>
:marked :marked
## Table of contents ## Table of contents
## 目录 ## 目录
[Bootstrap](#bootstrap) [Bootstrap](#bootstrap)
[程序启动](#bootstrap) [程序启动](#bootstrap)
[Question Model](#object-model) [Question Model](#object-model)
[问卷问题模型](#object-model) [问卷问题模型](#object-model)
[Form Component](#form-component) [Form Component](#form-component)
[表单组件](#form-component) [表单组件](#form-component)
[Questionnaire Metadata](#questionnaire-metadata) [Questionnaire Metadata](#questionnaire-metadata)
[问卷元数据](#questionnaire-metadata) [问卷元数据](#questionnaire-metadata)
[Dynamic Template](#dynamic-template) [Dynamic Template](#dynamic-template)
[动态模板](#dynamic-template) [动态模板](#dynamic-template)
:marked :marked
**See the <live-example name="cb-dynamic-form"></live-example>**. **See the <live-example name="cb-dynamic-form"></live-example>**.
**参见<live-example name="cb-dynamic-form">在线例子</live-example>**。 **参见<live-example name="cb-dynamic-form">在线例子</live-example>**。
.l-main-section .l-main-section
<a id="bootstrap"></a> <a id="bootstrap"></a>
:marked :marked
## Bootstrap ## Bootstrap
## 程序启动 ## 程序启动
We start by creating an `NgModule` called `AppModule`. We start by creating an `NgModule` called `AppModule`.
让我们从创建一个名叫`AppModule`的`NgModule`开始。 让我们从创建一个名叫`AppModule`的`NgModule`开始。
In our example we will be using Reactive Forms. In our example we will be using Reactive Forms.
在本例子中我们将使用响应式表单Reactive Forms 在本例子中我们将使用响应式表单Reactive Forms
Reactive Forms belongs to a different `NgModule` called `ReactiveFormsModule`, so in order to access any Reactive Forms directives, we have to import `ReactiveFormsModule` from `AppModule`. Reactive Forms belongs to a different `NgModule` called `ReactiveFormsModule`, so in order to access any Reactive Forms directives, we have to import `ReactiveFormsModule` from the `@angular/forms` library.
响应式表单属于另外一个叫做`ReactiveFormsModule`的`NgModule`,所以,为了使用响应式表单类的指令,我们得往`AppModule`中引入`ReactiveFormsModule`模块。 响应式表单属于另外一个叫做`ReactiveFormsModule`的`NgModule`,所以,为了使用响应式表单类的指令,我们得往`@angular/forms`库中引入`ReactiveFormsModule`模块。
We bootstrap our `AppModule` in main.ts. We bootstrap our `AppModule` in main.ts.
我们在main.ts中启动`AppModule`。 我们在main.ts中启动`AppModule`。
+makeTabs( +makeTabs(
@ -90,17 +90,17 @@ include ../_util-fns
<a id="object-model"></a> <a id="object-model"></a>
:marked :marked
## Question Model ## Question Model
## 问卷问题模型 ## 问卷问题模型
The next step is to define an object model that can describe all scenarios needed by the form functionality. The next step is to define an object model that can describe all scenarios needed by the form functionality.
The hero application process involves a form with a lot of questions. The hero application process involves a form with a lot of questions.
The "question" is the most fundamental object in the model. The "question" is the most fundamental object in the model.
第一步是定义一个对象模型,用来描述所有表单功能需要的场景。英雄的申请流程涉及到一个包含很多问卷问题的表单。问卷问题是最基础的对象模型。 第一步是定义一个对象模型,用来描述所有表单功能需要的场景。英雄的申请流程涉及到一个包含很多问卷问题的表单。问卷问题是最基础的对象模型。
We have created `QuestionBase` as the most fundamental question class. We have created `QuestionBase` as the most fundamental question class.
下面是我们建立的最基础的问卷问题基类,名叫`QuestionBase`。 下面是我们建立的最基础的问卷问题基类,名叫`QuestionBase`。
+makeExample('cb-dynamic-form/ts/app/question-base.ts','','app/question-base.ts') +makeExample('cb-dynamic-form/ts/app/question-base.ts','','app/question-base.ts')
@ -110,9 +110,9 @@ include ../_util-fns
The idea is that the form will be bound to specific question types and render the appropriate controls dynamically. The idea is that the form will be bound to specific question types and render the appropriate controls dynamically.
在这个基础上,我们派生出两个新类`TextboxQuestion` 和 `DropdownQuestion`,分别代表文本框和下拉框。这么做的初衷是,表单能动态绑定到特定的问卷问题类型,并动态渲染出合适的控件。 在这个基础上,我们派生出两个新类`TextboxQuestion` 和 `DropdownQuestion`,分别代表文本框和下拉框。这么做的初衷是,表单能动态绑定到特定的问卷问题类型,并动态渲染出合适的控件。
`TextboxQuestion` supports multiple html5 types like text, email, url etc via the `type` property. `TextboxQuestion` supports multiple html5 types like text, email, url etc via the `type` property.
`TextboxQuestion`可以通过`type`属性来支持多种HTML5元素类型比如文本、邮件、网址等。 `TextboxQuestion`可以通过`type`属性来支持多种HTML5元素类型比如文本、邮件、网址等。
+makeExample('cb-dynamic-form/ts/app/question-textbox.ts',null,'app/question-textbox.ts')(format='.') +makeExample('cb-dynamic-form/ts/app/question-textbox.ts',null,'app/question-textbox.ts')(format='.')
@ -121,34 +121,34 @@ include ../_util-fns
`DropdownQuestion` presents a list of choices in a select box. `DropdownQuestion` presents a list of choices in a select box.
`DropdownQuestion`表示一个带可选项列表的选择框。 `DropdownQuestion`表示一个带可选项列表的选择框。
+makeExample('cb-dynamic-form/ts/app/question-dropdown.ts',null,'app/question-dropdown.ts')(format='.') +makeExample('cb-dynamic-form/ts/app/question-dropdown.ts',null,'app/question-dropdown.ts')(format='.')
:marked :marked
Next we have defined `QuestionControlService`, a simple service for transforming our questions to a `FormGroup`. Next we have defined `QuestionControlService`, a simple service for transforming our questions to a `FormGroup`.
In a nutshell, the form group consumes the metadata from the question model and allows us to specify default values and validation rules. In a nutshell, the form group consumes the metadata from the question model and allows us to specify default values and validation rules.
接下来,我们定义了`QuestionControlService`,一个可以把问卷问题转换为`FormGroup`的服务。 接下来,我们定义了`QuestionControlService`,一个可以把问卷问题转换为`FormGroup`的服务。
简而言之,这个`FormGroup`使用问卷模型的元数据,并允许我们设置默认值和验证规则。 简而言之,这个`FormGroup`使用问卷模型的元数据,并允许我们设置默认值和验证规则。
+makeExample('cb-dynamic-form/ts/app/question-control.service.ts',null,'app/question-control.service.ts')(format='.') +makeExample('cb-dynamic-form/ts/app/question-control.service.ts',null,'app/question-control.service.ts')(format='.')
<a id="form-component"></a> <a id="form-component"></a>
:marked :marked
## Question form components ## Question form components
## 问卷表单组件 ## 问卷表单组件
Now that we have defined the complete model we are ready to create components to represent the dynamic form. Now that we have defined the complete model we are ready to create components to represent the dynamic form.
现在我们已经有一个定义好的完整模型了,接着就可以开始创建一个展现动态表单的组件。 现在我们已经有一个定义好的完整模型了,接着就可以开始创建一个展现动态表单的组件。
:marked :marked
`DynamicFormComponent` is the entry point and the main container for the form. `DynamicFormComponent` is the entry point and the main container for the form.
`DynamicFormComponent`是表单的主要容器和入口点。 `DynamicFormComponent`是表单的主要容器和入口点。
+makeTabs( +makeTabs(
`cb-dynamic-form/ts/app/dynamic-form.component.html, `cb-dynamic-form/ts/app/dynamic-form.component.html,
cb-dynamic-form/ts/app/dynamic-form.component.ts`, cb-dynamic-form/ts/app/dynamic-form.component.ts`,
@ -160,10 +160,10 @@ include ../_util-fns
It presents a list of questions, each question bound to a `<df-question>` component element. It presents a list of questions, each question bound to a `<df-question>` component element.
The `<df-question>` tag matches the `DynamicFormQuestionComponent`, The `<df-question>` tag matches the `DynamicFormQuestionComponent`,
the component responsible for rendering the details of each _individual_ question based on values in the data-bound question object. the component responsible for rendering the details of each _individual_ question based on values in the data-bound question object.
它代表了问卷问题列表,每个问题都被绑定到一个`<df-question>`组件元素。 它代表了问卷问题列表,每个问题都被绑定到一个`<df-question>`组件元素。
`<df-question>`标签匹配到的是组件`DynamicFormQuestionComponent`,该组件的职责是根据各个问卷问题对象的值来动态渲染表单控件。 `<df-question>`标签匹配到的是组件`DynamicFormQuestionComponent`,该组件的职责是根据各个问卷问题对象的值来动态渲染表单控件。
+makeTabs( +makeTabs(
`cb-dynamic-form/ts/app/dynamic-form-question.component.html, `cb-dynamic-form/ts/app/dynamic-form-question.component.html,
cb-dynamic-form/ts/app/dynamic-form-question.component.ts`, cb-dynamic-form/ts/app/dynamic-form-question.component.ts`,
@ -177,63 +177,63 @@ include ../_util-fns
The `ngSwitch` determines which type of question to display. The `ngSwitch` determines which type of question to display.
请注意,这个组件能代表模型里的任何问题类型。目前,还只有两种问题类型,但可以添加更多类型。可以用`ngSwitch`决定显示哪种类型的问题。 请注意,这个组件能代表模型里的任何问题类型。目前,还只有两种问题类型,但可以添加更多类型。可以用`ngSwitch`决定显示哪种类型的问题。
In both components we're relying on Angular's **formGroup** to connect the template HTML to the In both components we're relying on Angular's **formGroup** to connect the template HTML to the
underlying control objects, populated from the question model with display and validation rules. underlying control objects, populated from the question model with display and validation rules.
在这两个组件中我们依赖Angular的**formGroup**来把模板HTML和底层控件对象连接起来该对象从问卷问题模型里获取渲染和验证规则。 在这两个组件中我们依赖Angular的**formGroup**来把模板HTML和底层控件对象连接起来该对象从问卷问题模型里获取渲染和验证规则。
`formControlName` and `formGroup` are directives defined in `ReactiveFormsModule`. Our templates can can access these directives directly since we imported `ReactiveFormsModule` from `AppModule`. `formControlName` and `formGroup` are directives defined in `ReactiveFormsModule`. Our templates can can access these directives directly since we imported `ReactiveFormsModule` from `AppModule`.
`formControlName`和`formGroup`是在`ReactiveFormsModule`中定义的指令。我们之所以能在模板中使用它们,是因为我们往`AppModule`中导入了`ReactiveFormsModule`。 `formControlName`和`formGroup`是在`ReactiveFormsModule`中定义的指令。我们之所以能在模板中使用它们,是因为我们往`AppModule`中导入了`ReactiveFormsModule`。
<a id="questionnaire-metadata"></a> <a id="questionnaire-metadata"></a>
:marked :marked
## Questionnaire data ## Questionnaire data
## 问卷数据 ## 问卷数据
:marked :marked
`DynamicFormComponent` expects the list of questions in the form of an array bound to `@Input() questions`. `DynamicFormComponent` expects the list of questions in the form of an array bound to `@Input() questions`.
`DynamicForm`期望得到一个问题列表,该列表被绑定到`@Input() questions`属性。 `DynamicForm`期望得到一个问题列表,该列表被绑定到`@Input() questions`属性。
The set of questions we have defined for the job application is returned from the `QuestionService`. The set of questions we have defined for the job application is returned from the `QuestionService`.
In a real app we'd retrieve these questions from storage. In a real app we'd retrieve these questions from storage.
`QuestionService`会返回为工作申请表定义的那组问题列表。在真实的应用程序环境中,我们会从数据库里获得这些问题列表。 `QuestionService`会返回为工作申请表定义的那组问题列表。在真实的应用程序环境中,我们会从数据库里获得这些问题列表。
The key point is that we control the hero job application questions entirely through the objects returned from `QuestionService`. The key point is that we control the hero job application questions entirely through the objects returned from `QuestionService`.
Questionnaire maintenance is a simple matter of adding, updating, and removing objects from the `questions` array. Questionnaire maintenance is a simple matter of adding, updating, and removing objects from the `questions` array.
关键是,我们完全根据`QuestionService`返回的对象来控制英雄的工作申请表。 关键是,我们完全根据`QuestionService`返回的对象来控制英雄的工作申请表。
要维护这份问卷,只要非常简单的添加、更新和删除`questions`数组中的对象就可以了。 要维护这份问卷,只要非常简单的添加、更新和删除`questions`数组中的对象就可以了。
+makeExample('cb-dynamic-form/ts/app/question.service.ts','','app/question.service.ts') +makeExample('cb-dynamic-form/ts/app/question.service.ts','','app/question.service.ts')
:marked :marked
Finally, we display an instance of the form in the `AppComponent` shell. Finally, we display an instance of the form in the `AppComponent` shell.
最后,在`AppComponent`里显示出表单。 最后,在`AppComponent`里显示出表单。
+makeExample('cb-dynamic-form/ts/app/app.component.ts','','app.component.ts') +makeExample('cb-dynamic-form/ts/app/app.component.ts','','app.component.ts')
<a id="dynamic-template"></a> <a id="dynamic-template"></a>
:marked :marked
## Dynamic Template ## Dynamic Template
## 动态模板 ## 动态模板
Although in this example we're modelling a job application for heroes, there are no references to any specific hero question Although in this example we're modelling a job application for heroes, there are no references to any specific hero question
outside the objects returned by `QuestionService`. outside the objects returned by `QuestionService`.
在这个例子中,虽然我们是在为英雄的工作申请表建模,但是除了`QuestionService`返回的那些对象外,没有其它任何地方是与英雄有关的。 在这个例子中,虽然我们是在为英雄的工作申请表建模,但是除了`QuestionService`返回的那些对象外,没有其它任何地方是与英雄有关的。
This is very important since it allows us to repurpose the components for any type of survey This is very important since it allows us to repurpose the components for any type of survey
as long as it's compatible with our *question* object model. as long as it's compatible with our *question* object model.
The key is the dynamic data binding of metadata used to render the form The key is the dynamic data binding of metadata used to render the form
without making any hardcoded assumptions about specific questions. without making any hardcoded assumptions about specific questions.
In addition to control metadata, we are also adding validation dynamically. In addition to control metadata, we are also adding validation dynamically.
这点非常重要,因为只要与*问卷*对象模型兼容,就可以在任何类型的调查问卷中复用这些组件。 这点非常重要,因为只要与*问卷*对象模型兼容,就可以在任何类型的调查问卷中复用这些组件。
这里的关键是用到元数据的动态数据绑定来渲染表单,对问卷问题没有任何硬性的假设。除控件的元数据外,还可以动态添加验证规则。 这里的关键是用到元数据的动态数据绑定来渲染表单,对问卷问题没有任何硬性的假设。除控件的元数据外,还可以动态添加验证规则。
@ -241,21 +241,21 @@ include ../_util-fns
When the form is valid, we can click *Save* and the app renders the current form values as JSON. When the form is valid, we can click *Save* and the app renders the current form values as JSON.
This proves that any user input is bound back to the data model. This proves that any user input is bound back to the data model.
Saving and retrieving the data is an exercise for another time. Saving and retrieving the data is an exercise for another time.
表单验证通过之前,*保存*按钮是禁用的。验证通过后,就可以点击*保存*按钮程序会把当前值渲染成JSON显示出来。 表单验证通过之前,*保存*按钮是禁用的。验证通过后,就可以点击*保存*按钮程序会把当前值渲染成JSON显示出来。
这表明任何用户输入都被传到了数据模型里。至于如何储存和提取数据则是另一话题了。 这表明任何用户输入都被传到了数据模型里。至于如何储存和提取数据则是另一话题了。
:marked :marked
The final form looks like this: The final form looks like this:
完整的表单看起来是这样的: 完整的表单看起来是这样的:
figure.image-display figure.image-display
img(src="/resources/images/cookbooks/dynamic-form/dynamic-form.png" alt="Dynamic-Form") img(src="/resources/images/cookbooks/dynamic-form/dynamic-form.png" alt="Dynamic-Form")
:marked :marked
[Back to top](#top) [Back to top](#top)
[回到顶部](#top) [回到顶部](#top)

View File

@ -27,7 +27,7 @@ include ../_util-fns
This cookbook explains how to set up the QuickStart files with an **ASP.NET 4.x project** in This cookbook explains how to set up the QuickStart files with an **ASP.NET 4.x project** in
Visual Studio 2015. Visual Studio 2015.
本烹饪书解释了如何使用Visual Studio 2015在**ASP.NET 4.x项目**中设置**快速开始**文件。 本烹饪书解释了如何使用Visual Studio 2015在**ASP.NET 4.x项目**中设置**《快速起步》**文件。
.l-sub-section .l-sub-section
:marked :marked
If you prefer a `File | New Project` experience and are using **ASP.NET Core**, If you prefer a `File | New Project` experience and are using **ASP.NET Core**,
@ -333,7 +333,7 @@ h2#build-and-run 第五步:构建和运行应用
:marked :marked
The default browser opens and displays the QuickStart sample application. The default browser opens and displays the QuickStart sample application.
默认浏览器打开并显示快速开始例子应用。 默认浏览器打开并显示《快速起步》例子应用。
Try editing any of the project files. *Save* and refresh the browser to Try editing any of the project files. *Save* and refresh the browser to
see the changes. see the changes.

View File

@ -28,20 +28,22 @@ block includes
.l-main-section#A .l-main-section#A
+ifDocsFor('ts') a#aot
a#aot :marked
## Ahead-of-Time (AoT) compilation
## 预ahead-of-time, AoT编译
.l-sub-section
:marked :marked
## Ahead-of-time (AoT) compilation You can compile Angular applications at build-time.
## 预ahead-of-time, AoT编译 By compiling your application<span if-docs="ts"> using the compiler-cli, `ngc`</span>, you can bootstrap directly
.l-sub-section to a<span if-docs="ts"> module</span> factory, meaning you don't need to include the Angular compiler in your JavaScript bundle.
:marked Ahead-of-time compiled applications also benefit from decreased load time and increased performance.
You can compile Angular applications at build-time.
By compiling your application using the compiler-cli, `ngc`, you can bootstrap directly
to a module factory, meaning you don't need to include the Angular compiler in your JavaScript bundle.
Ahead-of-time compiled applications also benefit from decreased load time and increased performance.
开发者可以在构造时(build-time)编译Angular应用程序。通过`Compiler-cli` - `ngc`编译应用程序,应用可以从一个模块工厂(Module Factory)直接启动意思是不再需要把Angular编译器添加到JavaScript包中。预编译的应用程序将加载迅速并具有更高的性能。 开发者可以在构造时(build-time)编译Angular应用程序。通过`Compiler-cli` - `ngc`编译应用程序,应用可以从一个模块工厂(Module Factory)直接启动意思是不再需要把Angular编译器添加到JavaScript包中。预编译的应用程序将加载迅速并具有更高的性能。
+ifDocsFor('ts')
:marked :marked
## Angular module ## Angular module
@ -145,12 +147,6 @@ block includes
Angular的每个[范围化包(Scoped Package)](#scoped-package)都有一个叫做`index`的封装桶。 Angular的每个[范围化包(Scoped Package)](#scoped-package)都有一个叫做`index`的封装桶。
That's why we can write this:
这就是为什么可以这样写:
+makeExcerpt('quickstart/ts/app/app.component.ts', 'import', '')
.alert.is-important .alert.is-important
:marked :marked
Note that you can often achieve this using [Angular modules](#angular-module) instead. Note that you can often achieve this using [Angular modules](#angular-module) instead.
@ -180,9 +176,11 @@ block includes
.l-sub-section .l-sub-section
block bootstrap-defn-top block bootstrap-defn-top
:marked :marked
You launch an Angular application by "bootstrapping" it using the application root Angular module (`AppModule`). Bootstrapping identifies an application's top level "root" [component](#component), which is the first component that is loaded for the application. For more information, see [QuickStart](!{docsLatest}/quickstart.html). You launch an Angular application by "bootstrapping" it using the application root Angular module (`AppModule`). Bootstrapping identifies an application's top level "root" [component](#component), which is the first component that is loaded for the application.
For more information, see the [Setup](!{docsLatest}/guide/setup.html) page.
通过一个名叫`bootstrap`的方法来引导Angular应用程序。这个`bootstrap`方法会识别应用程序的顶级“根”[组件(Component)](#component),并可能通过[依赖注入体系(Dependency Injection System)](#dependency-injection)注册服务的[提供商(Provider)](#provider)。要了解详情,参见[快速起步](!{docsLatest}/quickstart.html)。 通过一个名叫`bootstrap`的方法来引导Angular应用程序。这个`bootstrap`方法会识别应用程序的顶级“根”[组件(Component)](#component),也就是应用加载的第一个组件。
要了解详情,参见[搭建本地开发环境](!{docsLatest}/setup.html)页。
:marked :marked
You can bootstrap multiple apps in the same `index.html`, each with its own top level root. You can bootstrap multiple apps in the same `index.html`, each with its own top level root.
@ -588,14 +586,15 @@ a#H
.l-main-section#J .l-main-section#J
+ifDocsFor('ts') a#jit
a#jit :marked
:marked ## Just-in-Time (JiT) compilation
## Just-in-time (JiT) compilation
## 即时just-in-time, JiT编译 ## 即时just-in-time, JiT编译
.l-sub-section
.l-sub-section
:marked :marked
With Angular _just-in-time_ bootstrapping you compile your components and modules in the browser With Angular _just-in-time_ bootstrapping you compile your components<span if-docs="ts"> and modules</span> in the browser
and launch the application dynamically. This is a good choice during development. and launch the application dynamically. This is a good choice during development.
Consider using the [ahead-of-time](#aot) mode for production apps. Consider using the [ahead-of-time](#aot) mode for production apps.
@ -908,15 +907,12 @@ a#Q
## Routing component ## Routing component
## 路由组件Routing Component ## 路由组件Routing Component
.l-sub-section .l-sub-section
block routing-component-defn :marked
:marked An Angular [component](#component) with a `RouterOutlet` that displays views based on router navigations.
An Angular [component](#component) with a RouterOutlet that displays views based on router navigations.
带有RouterOutlet的Angular[组件](#component)基于路由器导航来显示视图。 带有RouterOutlet的Angular[组件](#component)基于路由器导航来显示视图。For more information, see the [Routing & Navigation](!{docsLatest}/guide/router.html) page.
For more information, see the [Routing & Navigation](!{docsLatest}/guide/router.html) page. 要了解更多,请参见[路由与导航](!{docsLatest}/guide/router.html)页。
要了解更多,请参见[路由与导航](!{docsLatest}/guide/router.html)页。
.l-main-section#S .l-main-section#S
@ -1082,7 +1078,7 @@ a#snake-case
## TypeScript ## TypeScript
.l-sub-section .l-sub-section
:marked :marked
A version of JavaScript that supports most [ECMAScript 2015](#ecmascript=2015) A version of JavaScript that supports most [ECMAScript 2015](#es2015)
language features such as [decorators](#decorator). language features such as [decorators](#decorator).
一种支持了几乎所有[ECMAScript 2015](#ecmascript=2015)语言特性和一些未来版本可能有的特性(比如[装饰器(Decorator)](#decorator))的JavaScript语言。 一种支持了几乎所有[ECMAScript 2015](#ecmascript=2015)语言特性和一些未来版本可能有的特性(比如[装饰器(Decorator)](#decorator))的JavaScript语言。

View File

@ -2,11 +2,29 @@
"index": { "index": {
"title": "文档概览", "title": "文档概览",
"navTitle": "概览", "navTitle": "概览",
"description": "如何阅读本文档", "intro": "如何阅读本文档",
"nextable": true, "nextable": true,
"basics": true "basics": true
}, },
"setup": {
"title": "搭建本地开发环境",
"navTitle": "开发环境",
"intro": "安装 Angular 《快速起步》种子,更快更有效地在本地开发应用",
"nextable": true,
"hideNextPage": true,
"basics": true
},
"learning-angular": {
"title": "学习 Angular",
"navTitle": "学习 Angular",
"intro": "Angular 初学者的推荐学习路径",
"nextable": true,
"hideNextPage": true,
"basics": true
},
"architecture": { "architecture": {
"title": "架构概览", "title": "架构概览",
"navTitle": "架构", "navTitle": "架构",
@ -15,6 +33,15 @@
"basics": true "basics": true
}, },
"appmodule": {
"title": "AppModule: 根模块",
"navTitle": "根模块",
"intro": "如何在根\"AppModule\"中构建和启动应用。",
"nextable": true,
"basics": true
},
"displaying-data": { "displaying-data": {
"title": "显示数据", "title": "显示数据",
"intro": "属性绑定机制把数据显示到UI上。", "intro": "属性绑定机制把数据显示到UI上。",
@ -136,6 +163,11 @@
"intro": "开发“内容安全”的Angular应用。" "intro": "开发“内容安全”的Angular应用。"
}, },
"setup-systemjs-anatomy": {
"title": "搭建剖析",
"intro": "解析 SystemJS 本地开发环境"
},
"structural-directives": { "structural-directives": {
"title": "结构型指令", "title": "结构型指令",
"intro": "Angular有一个强力的模板引擎它能让你轻松维护元素的DOM树结构。" "intro": "Angular有一个强力的模板引擎它能让你轻松维护元素的DOM树结构。"

View File

@ -0,0 +1,241 @@
include ../_util-fns
:marked
An Angular module class describes how the application parts fit together.
Every application has at least one Angular module, the _root_ module
that you [bootstrap](#main) to launch the application.
You can call it anything you want. The conventional name is `AppModule`.
Angular 模块类描述应用的部件是如何组合在一起的。
每个应用都至少有一个 Angular 模块,也就是*根*模块,用来[引导](#main)并运行应用。
你可以为它取任何名字。常规名字是`AppModule`。
The [setup](setup.html) instructions produce a new project with the following minimal `AppModule`.
You'll evolve this module as your application grows.
[搭建本地开发环境](setup.html)讲解了如何使用下面这个最小的`AppModule`来创建一个新项目。
这个模块随着应用的成长而演变。
+makeExample('setup/ts/app/app.module.ts','', 'app/app.module.ts')(format='.')
:marked
After the `import` statements, you come to a class adorned with the
**`@NgModule`** [_decorator_](glossary.html#decorator '"Decorator" explained').
`import`声明之后,有个类被 **`@NgModule`**[装饰器](glossary.html#decorator '"Decorator" explained')所装饰。
The `@NgModule` decorator identifies `AppModule` as an Angular module class (also called an `NgModule` class).
`@NgModule` takes a _metadata_ object that tells Angular how to compile and launch the application.
该`@NgModule`装饰器将`AppModule`标记为 Angular 模块类(也叫`NgModule`类)。
`@NgModule`接受一个*元数据*对象,为 Angular 描述如何编译和启动应用。
* **_imports_** &mdash; the `BrowserModule` that this and every application needs to run in a browser.
* ***imports*** —— `BrowserModule`,这个和每个在浏览器中运行应用都需要它。
* **_declarations_** &mdash; the application's lone component, which is also ...
* ***declarations*** —— 应用的唯一组件,它同时也是...
* **_bootstrap_** &mdash; the _root_ component that Angular creates and inserts into the `index.html` host web page.
* ***bootstrap*** —— *根*组件Angular 创建它并将其插入到`index.html`宿主页面。
The [Angular Modules (NgModules)](ngmodule.html) guide dives deeply into the details of Angular modules.
All you need to know at the moment is a few basics about these three properties.
[Angular 模块 (NgModules)](ngmodule.html)指南深入讲解了 Angular 模块。
现在先初步了解这三个属性。
a#imports
:marked
### The _imports_ array
### *imports* 数组
Angular modules are a way to consolidate features that belong together into discrete units.
Many features of Angular itself are organized as Angular modules.
HTTP services are in the `HttpModule`. The router is in the `RouterModule`.
Eventually you may create a feature module.
Angular 模块是用来合并那些属于离散单元的特性的。
Angular 自身的许多特性也是通过 Angular 模块组织的。
HTTP 服务在`HttpModule`里。路由器在`RouterModule`中。
最后,你可能也会创建特性模块。
Add a module to the `imports` array when the application requires its features.
当应用需要模块的特性时,将其添加到`imports`数组中。
_This_ application, like most applications, executes in a browser.
Every application that executes in a browser needs the `BrowserModule` from `@angular/platform-browser`.
So every such application includes the `BrowserModule` in its _root_ `AppModule`'s `imports` array.
Other guide and cookbook pages will tell you when you need to add additional modules to this array.
*本*应用和大多数其他应用一样,在浏览器中运行。
每个浏览器中运行的应用都需要`@angular/platform-browser`里的`BrowserModule`。
所以每个这样的应用都在其*根*`AppModule`的`imports`数组中包含了`BrowserModule`。
在需要添加额外模块到此数组时,其他指南和烹饪书页面会告诉你。
.alert.is-important
:marked
**Only `NgModule` classes** go in the `imports` array. Don't put any other kind of class in `imports`.
`imports`数组中应该**只有`NgModule`类**。不要放置其他类型的类。
.l-sub-section
:marked
Don't confuse the `import` statements at the top of the file with the Angular module's `imports` array.
They have different jobs.
不要将 Angular 模块的`imports`数组与文件顶部的`import`声明弄混淆了。它们的功能不同。
The _JavaScript_ `import` statements give you access to symbols _exported_ by other files
so you can reference them within _this_ file.
They have nothing to do with Angular and Angular knows nothing about them.
*JavaScript* 的`import`声明允许你访问在其他文件中*导出*的符号,这样你可以在*当前*文件引用它们。
它们与 Angular 毫无关系,而且 Angular 对它们一无所知。
The _module's_ `imports` array tells Angular about specific Angular modules &mdash; classes decorated with `@NgModule` &mdash;
that the application needs to function properly.
*模块*的`imports`数组为 Angular 提供特定 Angular 模块——被`@NgModule`装饰的类——应用需要它们来正常工作。
a#declarations
:marked
### The _declarations_ array
### *declarations* 数组
You must declare _every_ component in one (and _only one_) `NgModule` class.
You tell Angular which components belong to the `AppModule` by listing it in the module's `declarations` array.
As you create more components, you'll add them to `declarations`.
*每个*组件必须在一个,而且仅仅一个`NgModule`类中声明。
通过将其列到`AppModule`模块的`declarations`数组中,你告诉 Angular 哪个组件属于`AppModule`。
在创建更多组件的过程中,逐步将它们添加到`declarations`中。
You'll learn to create two other kinds of classes &mdash;
[directives](attribute-directives.html) and [pipes](pipes.html) &mdash;
that you must also add to the `declarations` array.
你会学到怎样创建其他两种类——[指令](attribute-directives.html)和[管道](pipes.html)——它们也必须被添加到`declarations`数组。
.alert.is-important
:marked
**Only _declarables_** &mdash; _components_, _directives_ and _pipes_ &mdash; belong in the `declarations` array. &nbsp;
Don't put any other kind of class in `declarations`; _not_ `NgModule` classes, _not_ service classes, _not_ model classes.
**只有*可以声明的***——*组件*、*指令*和*管道*——属于`declarations`数组。不要将其他类型的类添加到`declarations`中,非`NgModule`类, 非服务类,亦非模型类。
a#bootstrap-array
:marked
### The _bootstrap_ array
### *bootstrap* 数组
You launch the application by [_bootstrapping_](#main) the root `AppModule`.
Among other things, the _bootstrapping_ process creates the component(s) listed in the `bootstrap` array
and inserts each one into the browser DOM.
通过[_引导_](#main)根`AppModule`来启动应用。在启动过程中,其中一步是创建列在`bootstrap`数组的组件并将它们每一个都插入到浏览器的DOM中。
Each bootstrapped component is the base of its own tree of components.
Inserting a bootstrapped component usually triggers a cascade of component creations that fill out that tree.
每个引导的组件都是它自己的组件树的根。
插入一个被引导的组件通常触发一系列组件的创建并形成组件树。
While you can put more than one component tree on a host web page, that's not typical.
Most applications have only one component tree and they bootstrap a single _root_ component.
虽然你可以将多个组件树插入到宿主页面,但是这并不典型。
大多数应用只有一个组件树,它们引导单一*根*组件。
You can call the one _root_ component anything you want but most developers call it `AppComponent`.
你可以为这个*根*组件取任何名字,但是大多数程序员将其取名为`AppComponent`。
Which brings us to the _bootstrapping_ process itself.
下面让我们来看看*引导*过程本身。
a#main
l-main-section
:marked
## Bootstrap in _main.ts_
## 在*main.ts*中引导
There are many ways to bootstrap an application.
The variations depend upon how you want to compile the application and where you want to run it.
引导应用的方法很多。
它们取决于你想如何编译应用以及应用将在哪儿运行。
In the beginning, you will compile the application dynamically with the _Just-in-Time (JiT)_ compiler
and you'll run it in a browser. You can learn about other options later.
在开始时你将使用_即时 (JiT) _编译器动态编译应用。然后在浏览器中运行它。
稍后,你可以了解其他选项。
The recommended place to bootstrap a JiT-compiled browser application is in a separate file
in the `app` folder named `app/main.ts`
引导即时编译的浏览器应用的推荐地点是在`app`目录中一个名为`app/main.ts`的单独文件中。
+makeExample('setup/ts/app/main.ts','','app/main.ts')(format='.')
:marked
This code creates a browser platform for dynamic (JiT) compilation and
bootstrap's the `AppModule` described above.
该代码为动态 (JiT) 编译创建浏览器平台,并引导上面提到的`AppModule`。
The _bootstrapping_ process sets up the execution environment,
digs the _root_ `AppComponent` out of the module's `bootstrap` array, &nbsp;
creates an instance of the component and inserts it within the element tag identified by the component's `selector`.
引导过程搭建运行环境,从模块的`bootstrap`数组中提出*根*`AppComponent` —— 创建该组件的实例并将其插入到组件`selector`标识的元素标签中。
The `AppComponent` selector &mdash; here and in most documentation samples &mdash; is `my-app`
so Angular looks for a `<my-app>` tag in the `index.html` like this one ...
`AppComponent`选择器——在这里以及文档大部分例子中——都是`my-app`,所以 Angular 在`index.html`中查找像这样的`<my-app>`标签...
+makeExample('setup/ts/index.html','my-app')(format='.')
:marked
... and displays the `AppComponent` there.
...然后在那儿显示`AppComponent`。
This file is very stable. Once you've set it up, you may never change it again.
该文件非常稳定。一旦被配置,它可能永远不会再被修改。
a#quickstart-appmodule
l-main-section
:marked
## QuickStart's _AppModule_
## 《快速起步》的`AppModule`
Every Angular application must have a root `NgModule`, even the [QuickStart](../quickstart.html).
You didn't see it but it was there.
每个 Angular 应用必须有一个跟`NgModule`,包括[快速起步](../quickstart.html)在内。
虽然你看不到它,但是它在那儿。
A script in the `index.html` generated a hidden `AppModule` and bootstrapped it for you
so you could focus on the `AppComponent` and discover the _essential Angular_ more quickly.
`index.html`中的一个脚本生成了隐藏的`AppModule`并为你引导它。这样你可以专注于`AppComponent`,以更快的了解* Angular 的基础*。
If you're feeling adventurous, add your own `AppModule` and `main.ts` to the
live code in the _QuickStart_ plunker.
你可以添加自己的`AppModule`和`main.ts`到plunker上*快速起步*的在线代码中。
Remove the following `<script>` tag from the `index.html` and see _your_ work in action.
从`index.html`中删除下面的`<script>`标签,看看*你*的劳动成果。
+makeExample('quickstart/ts/index.html','autobootstrap','Remove this script tag from "index.html"')(format='.')

View File

@ -5,7 +5,7 @@ block includes
:marked :marked
Angular is a framework for building client applications in HTML and Angular is a framework for building client applications in HTML and
either JavaScript or a language (like Dart or TypeScript) that compiles to JavaScript. either JavaScript or a language like TypeScript that compiles to JavaScript.
Angular是一个用HTML和JavaScript或者一个可以编译成JavaScript的语言例如Dart或者TypeScript来构建客户端应用的框架。 Angular是一个用HTML和JavaScript或者一个可以编译成JavaScript的语言例如Dart或者TypeScript来构建客户端应用的框架。
@ -98,7 +98,7 @@ figure
Angular应用是模块化的并且Angular有自己的模块系统它被称为_Angular模块_或_NgModules_。 Angular应用是模块化的并且Angular有自己的模块系统它被称为_Angular模块_或_NgModules_。
_Angular modules_ are a big deal. _Angular modules_ are a big deal.
This page introduces modules; the [Angular modules](ngmodule.html) page covers them in depth. This page introduces modules; the [Angular modules](ngmodule.html) page covers them in depth.
_Angular模块_很重要。 _Angular模块_很重要。
@ -106,11 +106,12 @@ figure
<br class="l-clear-both"><br> <br class="l-clear-both"><br>
:marked :marked
Every Angular app has at least one module, the _root module_, conventionally named `AppModule`. Every Angular app has at least one Angular module class, [the _root module_](appmodule.html "AppModule: the root module"),
conventionally named `AppModule`.
每个Angular应用至少有一个模块_根模块_习惯上命名为`AppModule`。 每个Angular应用至少有一个模块_根模块_习惯上命名为`AppModule`。
While the _root module_ may be the only module in a small application, most apps have many more While the _root module_ may be the only module in a small application, most apps have many more
_feature modules_, each a cohesive block of code dedicated to an application domain, _feature modules_, each a cohesive block of code dedicated to an application domain,
a workflow, or a closely related set of capabilities. a workflow, or a closely related set of capabilities.
@ -157,7 +158,7 @@ figure
* `providers` - [服务](#services)的创建者,并加入到全局服务表中,可用于应用任何部分。 * `providers` - [服务](#services)的创建者,并加入到全局服务表中,可用于应用任何部分。
* `bootstrap` - the main application view, called the _root component_, * `bootstrap` - the main application view, called the _root component_,
that hosts all other app views. Only the _root module_ should set this `bootstrap` property. that hosts all other app views. Only the _root module_ should set this `bootstrap` property.
* `bootstrap` - 指定应用的主视图称为_根组件_它是所有其它视图的宿主。只有_根模块_才能设置`bootstrap`属性。 * `bootstrap` - 指定应用的主视图称为_根组件_它是所有其它视图的宿主。只有_根模块_才能设置`bootstrap`属性。
@ -277,7 +278,7 @@ figure
要了解更多,参见[Angular模块](ngmodule.html)页。 要了解更多,参见[Angular模块](ngmodule.html)页。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked
@ -333,7 +334,7 @@ figure
当用户在这个应用中漫游时Angular会创建、更新和销毁组件。 当用户在这个应用中漫游时Angular会创建、更新和销毁组件。
应用可以通过[生命周期钩子](lifecycle-hooks.html)在组件生命周期的各个时间点上插入自己的操作,例如上面声明的`ngOnInit()`。 应用可以通过[生命周期钩子](lifecycle-hooks.html)在组件生命周期的各个时间点上插入自己的操作,例如上面声明的`ngOnInit()`。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked
@ -488,7 +489,7 @@ figure
这种架构处理方式是你向代码中添加元数据以便Angular知道该怎么做。 这种架构处理方式是你向代码中添加元数据以便Angular知道该怎么做。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked
@ -668,7 +669,7 @@ block dart-bool
当然,我们也能编写自己的指令。像`HeroListComponent`这样的组件就是一种自定义指令。 当然,我们也能编写自己的指令。像`HeroListComponent`这样的组件就是一种自定义指令。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked
@ -769,7 +770,7 @@ figure
Angular帮助我们*追随*这些原则 —— 它让我们能轻易地把应用逻辑拆分到服务,并通过*依赖注入*来在组件中使用这些服务。 Angular帮助我们*追随*这些原则 —— 它让我们能轻易地把应用逻辑拆分到服务,并通过*依赖注入*来在组件中使用这些服务。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked
@ -889,7 +890,7 @@ figure
* 把*提供商*注册到注入器。 * 把*提供商*注册到注入器。
.l-hr .l-hr
.l-main-section .l-main-section
:marked :marked

View File

@ -3,19 +3,19 @@ block includes
:marked :marked
An **Attribute** directive changes the appearance or behavior of a DOM element. An **Attribute** directive changes the appearance or behavior of a DOM element.
**属性**型指令用于改变一个DOM元素的外观或行为。 **属性**型指令用于改变一个DOM元素的外观或行为。
:marked :marked
# Contents # Contents
# 目录 # 目录
* [Directives overview](#directive-overview) * [Directives overview](#directive-overview)
* [指令概览](#directive-overview) * [指令概览](#directive-overview)
* [Build a simple attribute directive](#write-directive) * [Build a simple attribute directive](#write-directive)
* [创建简单的属性型指令](#write-directive) * [创建简单的属性型指令](#write-directive)
* [Apply the attribute directive to an element in a template](#apply-directive) * [Apply the attribute directive to an element in a template](#apply-directive)
* [把这个属性型指令应用到模板中的元素](#apply-directive) * [把这个属性型指令应用到模板中的元素](#apply-directive)
* [Respond to user-initiated events](#respond-to-user) * [Respond to user-initiated events](#respond-to-user)
@ -24,7 +24,7 @@ block includes
* [使用数据绑定把值传到指令中](#bindings) * [使用数据绑定把值传到指令中](#bindings)
* [Bind to a second property](#second-property) * [Bind to a second property](#second-property)
* [绑定第二个属性](#second-property) * [绑定第二个属性](#second-property)
试试<live-example>在线例子</live-example>。 试试<live-example>在线例子</live-example>。
@ -34,26 +34,27 @@ a#directive-overview
## Directives overview ## Directives overview
## 指令概览 ## 指令概览
There are three kinds of directives in Angular: There are three kinds of directives in Angular:
在Angular中有三种类型的指令 在Angular中有三种类型的指令
1. Components&mdash;directives with a template. 1. Components&mdash;directives with a template.
1. 组件 - 拥有模板的指令 1. 组件 - 拥有模板的指令
1. Structural directives&mdash;change the DOM layout by adding and removing DOM elements. 1. Structural directives&mdash;change the DOM layout by adding and removing DOM elements.
1. 结构型指令 - 通过添加和移除DOM元素改变DOM格局的指令 1. 结构型指令 - 通过添加和移除DOM元素改变DOM格局的指令
1. Attribute directives&mdash;change the appearance or behavior of an element. 1. Attribute directives&mdash;change the appearance or behavior of an element.
1. 属性型指令 - 改变元素显示和行为的指令。 1. 属性型指令 - 改变元素显示和行为的指令。
*Components* are the most common of the three directives. Read more about creating them *Components* are the most common of the three directives.
in step three of [QuickStart](../quickstart.html#root-component). You saw a component for the first time in the [QuickStart](../quickstart.html) example.
*组件*是这三种指令中最常用的,我们在构建应用程序时会写大量组件。参见[快速开始](../quickstart.html#root-component)第三步,了解更多创建组件的信息。 *组件*是这三种指令中最常用的。
你在[快速起步](../quickstart.html#root-component)例子中第一次见到组件。
*Structural Directives* change the structure of the view. Two examples are [NgFor](template-syntax.html#ngFor) and [NgIf](template-syntax.html#ngIf) *Structural Directives* change the structure of the view. Two examples are [NgFor](template-syntax.html#ngFor) and [NgIf](template-syntax.html#ngIf)
in the [Template Syntax](template-syntax.html) page. in the [Template Syntax](template-syntax.html) page.
*结构型*指令会通过添加/删除DOM元素来更改DOM树布局。[NgFor](template-syntax.html#ngFor)和[NgIf](template-syntax.html#ngIf)就是两个最熟悉的例子。 *结构型*指令会通过添加/删除DOM元素来更改DOM树布局。[NgFor](template-syntax.html#ngFor)和[NgIf](template-syntax.html#ngIf)就是两个最熟悉的例子。
*Attribute directives* are used as attributes of elements. The built-in [NgStyle](template-syntax.html#ngStyle) directive in the [Template Syntax](template-syntax.html) page, for example, *Attribute directives* are used as attributes of elements. The built-in [NgStyle](template-syntax.html#ngStyle) directive in the [Template Syntax](template-syntax.html) page, for example,
@ -65,33 +66,33 @@ a#directive-overview
a#write-directive a#write-directive
:marked :marked
## Build a simple attribute directive ## Build a simple attribute directive
## 创建一个简单的属性型指令 ## 创建一个简单的属性型指令
An attribute directive minimally requires building a controller class annotated with An attribute directive minimally requires building a controller class annotated with
`@Directive`, which specifies the selector that identifies `@Directive`, which specifies the selector that identifies
the attribute. the attribute.
The controller class implements the desired directive behavior. The controller class implements the desired directive behavior.
属性型指令至少需要一个带有`@Directive`装饰器的控制器类。该装饰器指定了一个选择器,用于指出与此指令相关联的属性名字。 属性型指令至少需要一个带有`@Directive`装饰器的控制器类。该装饰器指定了一个选择器,用于指出与此指令相关联的属性名字。
控制器类实现了指令需要具备的行为。 控制器类实现了指令需要具备的行为。
This page demonstrates building a simple attribute This page demonstrates building a simple attribute
directive to set an element's background color directive to set an element's background color
when the user hovers over that element. when the user hovers over that element.
本章展示了如何创建简单的属性型指令,在用户鼠标悬浮在一个元素上时,改变它的背景色 本章展示了如何创建简单的属性型指令,在用户鼠标悬浮在一个元素上时,改变它的背景色
.l-sub-section .l-sub-section
:marked :marked
Technically, a directive isn't necessary to simply set the background color. Style binding can set styles as follows: Technically, a directive isn't necessary to simply set the background color. Style binding can set styles as follows:
实际上,指令并不一定只是简单的设置背景颜色。样式绑定可以像下面这样设置样式: 实际上,指令并不一定只是简单的设置背景颜色。样式绑定可以像下面这样设置样式:
+makeExample('attribute-directives/ts/app/app.component.1.html','p-style-background') +makeExample('attribute-directives/ts/app/app.component.1.html','p-style-background')
:marked :marked
Read more about [style binding](template-syntax.html#style-binding) on the [Template Syntax](template-syntax.html) page. Read more about [style binding](template-syntax.html#style-binding) on the [Template Syntax](template-syntax.html) page.
参见[模板语法](template-syntax.html)章的[样式绑定](template-syntax.html#style-binding)。 参见[模板语法](template-syntax.html)章的[样式绑定](template-syntax.html#style-binding)。
For a simple example, though, this will demonstrate how attribute directives work. For a simple example, though, this will demonstrate how attribute directives work.
@ -100,44 +101,45 @@ a#write-directive
:marked :marked
### Write the directive code ### Write the directive code
### 编写指令代码
Create a new project folder (`attribute-directives`) and follow the steps in [QuickStart](../quickstart.html).
创建一个项目文件夹(`attribute-directives`)并按照[快速起步](../quickstart.html)中的步骤进行初始化。
include ../_quickstart_repo ### 编写指令代码
Follow the [setup](setup.html) instructions for creating a new project
named <span ngio-ex>attribute-directives</span>.
按照[搭建本地开发环境](setup.html)的说明,创建一个项目文件夹<span ngio-ex>attribute-directives</span>。
:marked :marked
Create the following source file in the indicated folder with the following code: Create the following source file in the indicated folder with the following code:
在指定的文件夹下创建下列源码文件: 在指定的文件夹下创建下列源码文件:
+makeExample('app/highlight.directive.1.ts') +makeExample('app/highlight.directive.1.ts')
block highlight-directive-1 block highlight-directive-1
:marked :marked
The `import` statement specifies symbols from the Angular `core`: The `import` statement specifies symbols from the Angular `core`:
`import`语句指定了从Angular的`core`库导入的一些符号。 `import`语句指定了从Angular的`core`库导入的一些符号。
1. `Directive` provides the functionality of the `@Directive` decorator. 1. `Directive` provides the functionality of the `@Directive` decorator.
1. `Directive`提供`@Directive`装饰器功能。 1. `Directive`提供`@Directive`装饰器功能。
1. `ElementRef` [injects](dependency-injection.html) into the directive's constructor 1. `ElementRef` [injects](dependency-injection.html) into the directive's constructor
1. `ElementRef` [注入](dependency-injection.html)到指令构造函数中。 1. `ElementRef` [注入](dependency-injection.html)到指令构造函数中。
so the code can access the DOM element. so the code can access the DOM element.
这样代码可以访问DOM元素。 这样代码可以访问DOM元素。
1. `Input` allows data to flow from the binding expression into the directive. 1. `Input` allows data to flow from the binding expression into the directive.
1. `Input`将数据从绑定表达式传达到指令中。 1. `Input`将数据从绑定表达式传达到指令中。
1. `Renderer` allows the code to change the DOM element's style. 1. `Renderer` allows the code to change the DOM element's style.
1. `Renderer` 让代码可以改变DOM元素的样式。 1. `Renderer` 让代码可以改变DOM元素的样式。
Next, the `@Directive` decorator function contains the directive metadata in a configuration object Next, the `@Directive` decorator function contains the directive metadata in a configuration object
@ -147,38 +149,38 @@ block highlight-directive-1
:marked :marked
`@Directive` requires a CSS selector to identify `@Directive` requires a CSS selector to identify
the HTML in the template that is associated with the directive. the HTML in the template that is associated with the directive.
属性型指令的`@Directive`装饰器需要一个css选择器以便从模板中识别出关联到这个指令的HTML。 属性型指令的`@Directive`装饰器需要一个css选择器以便从模板中识别出关联到这个指令的HTML。
The [CSS selector for an attribute](https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors) The [CSS selector for an attribute](https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors)
is the attribute name in square brackets. is the attribute name in square brackets.
Here, the directive's selector is `[myHighlight]`. Here, the directive's selector is `[myHighlight]`.
Angular will locate all elements in the template that have an attribute named `myHighlight`. Angular will locate all elements in the template that have an attribute named `myHighlight`.
[css中的attribute选择器](https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors)就是属性名称加方括号。 [css中的attribute选择器](https://developer.mozilla.org/en-US/docs/Web/CSS/Attribute_selectors)就是属性名称加方括号。
这里,指令的选择器是`[myHighlight]`Angular将会在模板中找到带有`myHighlight`这个属性的元素。 这里,指令的选择器是`[myHighlight]`Angular将会在模板中找到带有`myHighlight`这个属性的元素。
.l-sub-section .l-sub-section
:marked :marked
### Why not call it "highlight"? ### Why not call it "highlight"?
### 为什么不直接叫做"highlight" ### 为什么不直接叫做"highlight"
Though *highlight* is a more concise name than *myHighlight* and would work, Though *highlight* is a more concise name than *myHighlight* and would work,
a best practice is to prefix selector names to ensure a best practice is to prefix selector names to ensure
they don't conflict with standard HTML attributes. they don't conflict with standard HTML attributes.
This also reduces the risk colliding with third-party directive names. This also reduces the risk colliding with third-party directive names.
理论上,*highlight*是一个比*myHighlight*更好的名字,而且在这里它确实能工作。 理论上,*highlight*是一个比*myHighlight*更好的名字,而且在这里它确实能工作。
但是最佳实践是在选择器名字前面添加前缀以确保它们不会与标准HTML属性冲突。 但是最佳实践是在选择器名字前面添加前缀以确保它们不会与标准HTML属性冲突。
它同时减少了与第三方指令名字发生冲突的危险。 它同时减少了与第三方指令名字发生冲突的危险。
Make sure you do **not** prefix the `highlight` directive name with **`ng`** because Make sure you do **not** prefix the `highlight` directive name with **`ng`** because
that prefix is reserved for Angular and using it could cause bugs that are difficult to diagnose. For a simple demo, the short prefix, `my`, helps distinguish your custom directive. that prefix is reserved for Angular and using it could cause bugs that are difficult to diagnose. For a simple demo, the short prefix, `my`, helps distinguish your custom directive.
确认你**不会**给自己的`highlight`指令添加**`ng`**前缀。 确认你**不会**给自己的`highlight`指令添加**`ng`**前缀。
那个前缀属于Angular使用它可能导致无法检测的问题。比如这个简短的前缀`my`可以帮助你区分自定义指令。 那个前缀属于Angular使用它可能导致无法检测的问题。比如这个简短的前缀`my`可以帮助你区分自定义指令。
p p
| After the #[code @Directive] metadata comes the directive's controller class, called #[code HighlightDirective], which contains the logic for the directive. | After the #[code @Directive] metadata comes the directive's controller class, called #[code HighlightDirective], which contains the logic for the directive.
p p
@ -186,7 +188,7 @@ p
+ifDocsFor('ts') +ifDocsFor('ts')
| Exporting #[code HighlightDirective] makes it accessible to other components. | Exporting #[code HighlightDirective] makes it accessible to other components.
+ifDocsFor('ts') +ifDocsFor('ts')
| 导出#[code HighlightDirective]以便让它可以被其它组件访问。 | 导出#[code HighlightDirective]以便让它可以被其它组件访问。
:marked :marked
@ -198,18 +200,18 @@ p
Angular会为每个被指令匹配上的元素创建一个该指令控制器类的实例并把Angular的`ElementRef`和`Renderer`注入进它的构造函数。 Angular会为每个被指令匹配上的元素创建一个该指令控制器类的实例并把Angular的`ElementRef`和`Renderer`注入进它的构造函数。
`ElementRef`是一个服务它赋予我们直接访问DOM元素的能力。通过它的`nativeElement`属性和`Renderer`服务,我们可以设置元素的样式。 `ElementRef`是一个服务它赋予我们直接访问DOM元素的能力。通过它的`nativeElement`属性和`Renderer`服务,我们可以设置元素的样式。
.l-main-section .l-main-section
a#apply-directive a#apply-directive
:marked :marked
## Apply the attribute directive ## Apply the attribute directive
## 使用属性型指令 ## 使用属性型指令
To use the new `HighlightDirective`, create a template that To use the new `HighlightDirective`, create a template that
applies the directive as an attribute to a paragraph (`p`) element. applies the directive as an attribute to a paragraph (`p`) element.
In Angular terms, the `<p>` element will be the attribute **host**. In Angular terms, the `<p>` element will be the attribute **host**.
要使用这个新的`HighlightDirective`,创建一个模板,把这个指令作为属性应用到一个段落(`p`)元素上。 要使用这个新的`HighlightDirective`,创建一个模板,把这个指令作为属性应用到一个段落(`p`)元素上。
用Angular的话说`<p>`元素就是这个属性型指令的**宿主**。 用Angular的话说`<p>`元素就是这个属性型指令的**宿主**。
p p
@ -230,7 +232,7 @@ p
Next, add an `import` statement to fetch the `Highlight` directive and Next, add an `import` statement to fetch the `Highlight` directive and
add that class to the `declarations` NgModule metadata. This way Angular add that class to the `declarations` NgModule metadata. This way Angular
recognizes the directive when it encounters `myHighlight` in the template. recognizes the directive when it encounters `myHighlight` in the template.
接下来,添加了一个`import`语句来获得'Highlight'指令类,并把这个类添加到`AppComponent`组件的`declarations`数组中。 接下来,添加了一个`import`语句来获得'Highlight'指令类,并把这个类添加到`AppComponent`组件的`declarations`数组中。
这样当Angular在模板中遇到`myHighlight`时,就能认出这是指令了。 这样当Angular在模板中遇到`myHighlight`时,就能认出这是指令了。
@ -240,22 +242,22 @@ p
Now when the app runs, the `myHighlight` directive highlights the paragraph text. Now when the app runs, the `myHighlight` directive highlights the paragraph text.
运行应用,就会看到我们的指令确实高亮了段落中的文本。 运行应用,就会看到我们的指令确实高亮了段落中的文本。
figure.image-display figure.image-display
img(src="/resources/images/devguide/attribute-directives/first-highlight.png" alt="First Highlight") img(src="/resources/images/devguide/attribute-directives/first-highlight.png" alt="First Highlight")
.l-sub-section .l-sub-section
:marked :marked
### Your directive isn't working? ### Your directive isn't working?
### 你的指令没生效? ### 你的指令没生效?
Did you remember to add the directive to the the `declarations` attribute of `@NgModule`? It is easy to forget! Did you remember to add the directive to the the `declarations` attribute of `@NgModule`? It is easy to forget!
你记着设置`@NgModule`的`declarations`数组了吗?它很容易被忘掉。 你记着设置`@NgModule`的`declarations`数组了吗?它很容易被忘掉。
Open the console in the browser tools and look for an error like this: Open the console in the browser tools and look for an error like this:
打开浏览器调试工具的控制台,会看到像这样的错误信息: 打开浏览器调试工具的控制台,会看到像这样的错误信息:
code-example(format="nocode"). code-example(format="nocode").
EXCEPTION: Template parse errors: EXCEPTION: Template parse errors:
@ -269,13 +271,13 @@ figure.image-display
Angular检测到你正在尝试绑定到*某些东西*,但它不认识。所以它在`declarations`元数据数组中查找。 Angular检测到你正在尝试绑定到*某些东西*,但它不认识。所以它在`declarations`元数据数组中查找。
把`HighlightDirective`列在元数据的这个数组中Angular就会检查对应的导入语句从而找到`highlight.directive.ts`,并了解`myHightlight`的功能。 把`HighlightDirective`列在元数据的这个数组中Angular就会检查对应的导入语句从而找到`highlight.directive.ts`,并了解`myHightlight`的功能。
:marked :marked
To summarize, Angular found the `myHighlight` attribute on the `<p>` element. It created To summarize, Angular found the `myHighlight` attribute on the `<p>` element. It created
an instance of the `HighlightDirective` class, an instance of the `HighlightDirective` class,
injecting a reference to the element into the constructor injecting a reference to the element into the constructor
where the `<p>` element's background style is set to yellow. where the `<p>` element's background style is set to yellow.
总结Angular在`<p>`元素上发现了一个`myHighlight`属性。 总结Angular在`<p>`元素上发现了一个`myHighlight`属性。
然后它创建了一个`HighlightDirective`类的实例,并把所在元素的引用注入到了指令的构造函数中。 然后它创建了一个`HighlightDirective`类的实例,并把所在元素的引用注入到了指令的构造函数中。
在构造函数中,我们把`<p>`元素的背景设置为了黄色。 在构造函数中,我们把`<p>`元素的背景设置为了黄色。
@ -284,22 +286,22 @@ figure.image-display
a#respond-to-user a#respond-to-user
:marked :marked
## Respond to user-initiated events ## Respond to user-initiated events
## 响应用户引发的事件 ## 响应用户引发的事件
Currently, `myHighlight` simply sets an element color. Currently, `myHighlight` simply sets an element color.
The directive should set the color when the user hovers over an element. The directive should set the color when the user hovers over an element.
当前,`myHighlight`只是简单的设置元素的颜色。 当前,`myHighlight`只是简单的设置元素的颜色。
这个指令应该在用户鼠标悬浮一个元素时,设置它的颜色。 这个指令应该在用户鼠标悬浮一个元素时,设置它的颜色。
This requires two things: This requires two things:
我们需要: 我们需要:
1. detecting when the user hovers into and out of the element. 1. detecting when the user hovers into and out of the element.
1. 检测用户的鼠标啥时候进入和离开这个元素。 1. 检测用户的鼠标啥时候进入和离开这个元素。
2. responding to those actions by setting and clearing the highlight color. 2. responding to those actions by setting and clearing the highlight color.
2. 通过设置和清除高亮色来响应这些操作。 2. 通过设置和清除高亮色来响应这些操作。
@ -307,7 +309,7 @@ a#respond-to-user
To do this, you can apply the `@HostListener` !{_decorator} to methods which are called when an event is raised. To do this, you can apply the `@HostListener` !{_decorator} to methods which are called when an event is raised.
把`host`属性加入指令的元数据中,并给它一个配置对象,用来指定两个鼠标事件,并在它们被触发时,调用指令中的方法: 把`host`属性加入指令的元数据中,并给它一个配置对象,用来指定两个鼠标事件,并在它们被触发时,调用指令中的方法:
+makeExample('attribute-directives/ts/app/highlight.directive.2.ts','host')(format=".") +makeExample('attribute-directives/ts/app/highlight.directive.2.ts','host')(format=".")
.l-sub-section .l-sub-section
@ -315,32 +317,32 @@ a#respond-to-user
The `@HostListener` !{_decorator} refers to the DOM element that hosts an attribute directive, the `<p>` in this case. The `@HostListener` !{_decorator} refers to the DOM element that hosts an attribute directive, the `<p>` in this case.
`@HostListener`装饰器引用的是我们这个属性型指令的宿主元素,在这个例子中就是`<p>`。 `@HostListener`装饰器引用的是我们这个属性型指令的宿主元素,在这个例子中就是`<p>`。
It is possible to attach event listeners by manipulating the host DOM element directly, but It is possible to attach event listeners by manipulating the host DOM element directly, but
可以通过直接操纵DOM元素的方式给宿主DOM元素挂上一个事件监听器但是 可以通过直接操纵DOM元素的方式给宿主DOM元素挂上一个事件监听器但是
there are at least three problems with such an approach: there are at least three problems with such an approach:
但这种方法至少有三个问题: 但这种方法至少有三个问题:
1. You have to write the listeners correctly. 1. You have to write the listeners correctly.
1. 必须正确的书写事件监听器。 1. 必须正确的书写事件监听器。
1. The code must *detach* the listener when the directive is destroyed to avoid memory leaks. 1. The code must *detach* the listener when the directive is destroyed to avoid memory leaks.
1. 当指令被销毁的时候,必须*摘掉*事件监听器,否则就会导致内存泄露。 1. 当指令被销毁的时候,必须*摘掉*事件监听器,否则就会导致内存泄露。
1. Talking to DOM API directly isn't a best practice. 1. Talking to DOM API directly isn't a best practice.
1. 必须直接和DOM API打交道但正如我们学过的那样应该避免这样做。 1. 必须直接和DOM API打交道但正如我们学过的那样应该避免这样做。
:marked :marked
Now implement the two mouse event handlers: Now implement the two mouse event handlers:
现在,我们实现那两个鼠标事件处理器: 现在,我们实现那两个鼠标事件处理器:
+makeExample('attribute-directives/ts/app/highlight.directive.2.ts','mouse-methods')(format=".") +makeExample('attribute-directives/ts/app/highlight.directive.2.ts','mouse-methods')(format=".")
:marked :marked
@ -353,9 +355,9 @@ a#respond-to-user
+makeExample('attribute-directives/ts/app/highlight.directive.2.ts','ctor')(format=".") +makeExample('attribute-directives/ts/app/highlight.directive.2.ts','ctor')(format=".")
:marked :marked
Here's the updated directive: Here's the updated directive:
这里是更新过的指令: 这里是更新过的指令:
+makeExample('app/highlight.directive.2.ts') +makeExample('app/highlight.directive.2.ts')
:marked :marked
@ -363,7 +365,7 @@ a#respond-to-user
disappears as it moves out. disappears as it moves out.
We run the app and confirm that the background color appears as we move the mouse over the `p` and We run the app and confirm that the background color appears as we move the mouse over the `p` and
disappears as we move out. disappears as we move out.
运行本应用,就可以确认:当把鼠标移到`p`上的时候,背景色就出现了,而移开的时候,它消失了。 运行本应用,就可以确认:当把鼠标移到`p`上的时候,背景色就出现了,而移开的时候,它消失了。
figure.image-display figure.image-display
img(src="/resources/images/devguide/attribute-directives/highlight-directive-anim.gif" alt="Second Highlight") img(src="/resources/images/devguide/attribute-directives/highlight-directive-anim.gif" alt="Second Highlight")
@ -371,7 +373,7 @@ figure.image-display
a#bindings a#bindings
:marked :marked
## Pass values into the directive using data binding ## Pass values into the directive using data binding
## 通过绑定来传递值到指令中 ## 通过绑定来传递值到指令中
Currently the highlight color is hard-coded within the directive. That's inflexible. Currently the highlight color is hard-coded within the directive. That's inflexible.
@ -382,13 +384,13 @@ a#bindings
+makeExample('attribute-directives/ts/app/app.component.html','pHost') +makeExample('attribute-directives/ts/app/app.component.html','pHost')
:marked :marked
You can extend the directive class with a bindable **input** `highlightColor` property and use it to highlight text. You can extend the directive class with a bindable **input** `highlightColor` property and use it to highlight text.
我们将给指令类增加一个可绑定**输入**属性`highlightColor`,当需要高亮文本的时候,就用它。 我们将给指令类增加一个可绑定**输入**属性`highlightColor`,当需要高亮文本的时候,就用它。
Here is the final version of the class: Here is the final version of the class:
这里是该类的最终版: 这里是该类的最终版:
+makeExcerpt('app/highlight.directive.ts', 'class') +makeExcerpt('app/highlight.directive.ts', 'class')
a#input a#input
@ -398,7 +400,7 @@ a#input
新的`highlightColor`属性被称为“输入”属性,这是因为数据流是从绑定表达式到这个指令的。 新的`highlightColor`属性被称为“输入”属性,这是因为数据流是从绑定表达式到这个指令的。
注意,在定义这个属性的时候,我们调用了`@Input()`#{_decoratorCn}。 注意,在定义这个属性的时候,我们调用了`@Input()`#{_decoratorCn}。
+makeExcerpt('app/highlight.directive.ts', 'color') +makeExcerpt('app/highlight.directive.ts', 'color')
:marked :marked
@ -406,30 +408,30 @@ a#input
property binding under the `myHighlight` alias. property binding under the `myHighlight` alias.
Without this input metadata Angular rejects the binding. Without this input metadata Angular rejects the binding.
See the [appendix](#why-input) below for more information. See the [appendix](#why-input) below for more information.
`@Input`把元数据添加到了类上,这让`highlightColor`能被以`myHighlight`为别名进行绑定。 `@Input`把元数据添加到了类上,这让`highlightColor`能被以`myHighlight`为别名进行绑定。
必须添加这个input元数据否则Angular会拒绝绑定。 必须添加这个input元数据否则Angular会拒绝绑定。
参见下面的[附录](#why-input)来了解为何如此。 参见下面的[附录](#why-input)来了解为何如此。
.l-sub-section .l-sub-section
:marked :marked
### @Input(_alias_) ### @Input(_alias_)
### @Input(_别名_) ### @Input(_别名_)
Currently, the code **aliases** the `highlightColor` property with the attribute name by Currently, the code **aliases** the `highlightColor` property with the attribute name by
passing `myHighlight` into the `@Input` #{_decorator}: passing `myHighlight` into the `@Input` #{_decorator}:
当前,代码通过将`myHighlight`传递到`@Input`装饰器,把`myHighlight`属性**别名**到属性名字上。 当前,代码通过将`myHighlight`传递到`@Input`装饰器,把`myHighlight`属性**别名**到属性名字上。
+makeExcerpt('app/highlight.directive.ts', 'color', '') +makeExcerpt('app/highlight.directive.ts', 'color', '')
:marked :marked
The code binds to the attribute name, `myHighlight`, but the The code binds to the attribute name, `myHighlight`, but the
the directive property name is `highlightColor`. That's a disconnect. the directive property name is `highlightColor`. That's a disconnect.
代码绑定到`myHighlight`属性名,但是指令属性名为`highlightColor`。这是一个断点。 代码绑定到`myHighlight`属性名,但是指令属性名为`highlightColor`。这是一个断点。
You can resolve the discrepancy by renaming the property to `myHighlight` and define it as follows: You can resolve the discrepancy by renaming the property to `myHighlight` and define it as follows:
你可以通过重命名属性名到`myHighlight`来移除这个区别,像这样: 你可以通过重命名属性名到`myHighlight`来移除这个区别,像这样:
+makeExcerpt('app/highlight.directive.ts', 'highlight', '') +makeExcerpt('app/highlight.directive.ts', 'highlight', '')
@ -440,7 +442,7 @@ a#input
现在,通过输入型属性得到了高亮的颜色,然后修改`onMouseEnter()`来使用它代替硬编码的那个颜色名。 现在,通过输入型属性得到了高亮的颜色,然后修改`onMouseEnter()`来使用它代替硬编码的那个颜色名。
我们还把红色定义为默认颜色,以便在用户忘了绑定颜色时作为备用。 我们还把红色定义为默认颜色,以便在用户忘了绑定颜色时作为备用。
+makeExcerpt('attribute-directives/ts/app/highlight.directive.ts', 'mouse-enter', '') +makeExcerpt('attribute-directives/ts/app/highlight.directive.ts', 'mouse-enter', '')
:marked :marked
To let users pick the highlight color and bind their choice to the directive, To let users pick the highlight color and bind their choice to the directive,
@ -453,21 +455,21 @@ a#input
.l-sub-section .l-sub-section
:marked :marked
### Where is the templated *color* property? ### Where is the templated *color* property?
### 模板的*color*属性在哪里? ### 模板的*color*属性在哪里?
You may notice that the radio button click handlers in the template set a `color` property You may notice that the radio button click handlers in the template set a `color` property
and the code is binding that `color` to the directive. and the code is binding that `color` to the directive.
However, you never defined a color property for the host `AppComponent`. However, you never defined a color property for the host `AppComponent`.
Yet this code works. Where is the template `color` value going? Yet this code works. Where is the template `color` value going?
你可能注意到,模板中的单选按钮的点击事件处理器设置了一个`color`属性,而且把`color`绑定到指令上。 你可能注意到,模板中的单选按钮的点击事件处理器设置了一个`color`属性,而且把`color`绑定到指令上。
但是,你从未在这个宿主`AppComponent`中定义`color`属性,代码仍然工作正常。模板的`color`值去哪儿了? 但是,你从未在这个宿主`AppComponent`中定义`color`属性,代码仍然工作正常。模板的`color`值去哪儿了?
Browser debugging reveals that Angular dynamically added a `color` property Browser debugging reveals that Angular dynamically added a `color` property
to the runtime instance of the `AppComponent`. to the runtime instance of the `AppComponent`.
在浏览器中调试就会发现Angular在`AppComponent`的运行期实例上添加了一个`color`属性。 在浏览器中调试就会发现Angular在`AppComponent`的运行期实例上添加了一个`color`属性。
This is *convenient* behavior but it is also *implicit* behavior that could be confusing. This is *convenient* behavior but it is also *implicit* behavior that could be confusing.
@ -478,7 +480,7 @@ a#input
:marked :marked
Here is the second version of the directive in action. Here is the second version of the directive in action.
下面是指令操作演示的第二版。 下面是指令操作演示的第二版。
figure.image-display figure.image-display
img(src="/resources/images/devguide/attribute-directives/highlight-directive-v2-anim.gif" alt="Highlight v.2") img(src="/resources/images/devguide/attribute-directives/highlight-directive-v2-anim.gif" alt="Highlight v.2")
@ -487,10 +489,10 @@ figure.image-display
a#second-property a#second-property
:marked :marked
## Bind to a second property ## Bind to a second property
## 绑定到第二个属性 ## 绑定到第二个属性
This example directive only has a single customizable property. A real app often needs more. This example directive only has a single customizable property. A real app often needs more.
本例的指令只有一个可定制属性,真实的引用通常需要更多。 本例的指令只有一个可定制属性,真实的引用通常需要更多。
Let's allow the template developer to set the default color&mdash;the color that prevails until the user picks a highlight color. Let's allow the template developer to set the default color&mdash;the color that prevails until the user picks a highlight color.
@ -502,17 +504,17 @@ a#second-property
:marked :marked
The `defaultColor` property has a setter that overrides the hard-coded default color, "red". The `defaultColor` property has a setter that overrides the hard-coded default color, "red".
You don't need a getter. You don't need a getter.
`defaultColor`属性是一个setter函数它代替了硬编码的默认颜色“red”。不需要getter函数。 `defaultColor`属性是一个setter函数它代替了硬编码的默认颜色“red”。不需要getter函数。
How do you bind to it? The app is already using `myHighlight` attribute name as a binding target. How do you bind to it? The app is already using `myHighlight` attribute name as a binding target.
该如何绑定到它?别忘了已经把`myHighlight`属性名用作绑定目标了。 该如何绑定到它?别忘了已经把`myHighlight`属性名用作绑定目标了。
Remember that a *component is a directive, too*. Remember that a *component is a directive, too*.
You can add as many component property bindings as you need by stringing them along in the template You can add as many component property bindings as you need by stringing them along in the template
as in this example that sets the `a`, `b`, `c` properties to the string literals 'a', 'b', and 'c'. as in this example that sets the `a`, `b`, `c` properties to the string literals 'a', 'b', and 'c'.
记住,*组件也是指令*。 记住,*组件也是指令*。
只要需要,就可以通过把它们依次串在模板中来为组件添加多个属性绑定。 只要需要,就可以通过把它们依次串在模板中来为组件添加多个属性绑定。
下面这个例子中就把`a`、`b`、`c`属性设置为了字符串字面量'a', 'b', 'c'。 下面这个例子中就把`a`、`b`、`c`属性设置为了字符串字面量'a', 'b', 'c'。
@ -520,7 +522,7 @@ code-example(format="." ).
&lt;my-component [a]="'a'" [b]="'b'" [c]="'c'">&lt;my-component> &lt;my-component [a]="'a'" [b]="'b'" [c]="'c'">&lt;my-component>
:marked :marked
The same holds true for an attribute directive. The same holds true for an attribute directive.
在属性型指令中也可以这样做。 在属性型指令中也可以这样做。
+makeExample('attribute-directives/ts/app/app.component.html', 'defaultColor')(format=".") +makeExample('attribute-directives/ts/app/app.component.html', 'defaultColor')(format=".")
:marked :marked
@ -531,7 +533,7 @@ code-example(format="." ).
我们*还*把字符串字面量'violet'绑定到了`defaultColor`上。 我们*还*把字符串字面量'violet'绑定到了`defaultColor`上。
Here is the final version of the directive in action. Here is the final version of the directive in action.
下面就是该指令最终版的操作演示。 下面就是该指令最终版的操作演示。
figure.image-display figure.image-display
img(src="/resources/images/devguide/attribute-directives/highlight-directive-final-anim.gif" alt="Final Highlight") img(src="/resources/images/devguide/attribute-directives/highlight-directive-final-anim.gif" alt="Final Highlight")
@ -539,9 +541,9 @@ figure.image-display
.l-main-section .l-main-section
:marked :marked
## Summary ## Summary
## 总结 ## 总结
This page covered how to: This page covered how to:
本章介绍了如何: 本章介绍了如何:
@ -552,7 +554,7 @@ figure.image-display
- [Respond to **events** to change behavior based on an event](#respond-to-user). - [Respond to **events** to change behavior based on an event](#respond-to-user).
- [响应**事件**,以便基于事件改变行为](#respond-to-user) - [响应**事件**,以便基于事件改变行为](#respond-to-user)
- [Use **binding** to pass values to the attribute directive](#bindings). - [Use **binding** to pass values to the attribute directive](#bindings).
- 以及[使用**绑定**来把值传给属性型指令](#bindings)。 - 以及[使用**绑定**来把值传给属性型指令](#bindings)。
The final source: The final source:
@ -579,32 +581,32 @@ a#why-input
.l-main-section .l-main-section
:marked :marked
### Appendix: Input properties ### Appendix: Input properties
### 附录Input属性 ### 附录Input属性
In this demo, the `highlightColor` property is an ***input*** property of In this demo, the `highlightColor` property is an ***input*** property of
`HighlightDirective`. `HighlightDirective`.
本例中, `highlightColor`属性是`HighlightDirective`指令的一个***input***属性。 本例中, `highlightColor`属性是`HighlightDirective`指令的一个***input***属性。
You've seen properties in bindings before but never had to declare them as anything. Why now? You've seen properties in bindings before but never had to declare them as anything. Why now?
以前也见过属性绑定,但我们从没有定义过它们。为什么现在就不行了? 以前也见过属性绑定,但我们从没有定义过它们。为什么现在就不行了?
Angular makes a subtle but important distinction between binding **sources** and **targets**. Angular makes a subtle but important distinction between binding **sources** and **targets**.
Angular在绑定的**源**和**目标**之间有一个巧妙但重要的区别。 Angular在绑定的**源**和**目标**之间有一个巧妙但重要的区别。
In all previous bindings, the directive or component property was a binding ***source***. In all previous bindings, the directive or component property was a binding ***source***.
A property is a *source* if it appears in the template expression to the ***right*** of the equals (=). A property is a *source* if it appears in the template expression to the ***right*** of the equals (=).
在以前的所有绑定中,指令或组件的属性都是绑定***源***。 在以前的所有绑定中,指令或组件的属性都是绑定***源***。
如果属性出现在了模板表达式等号(=)的***右侧***,它就是一个*源*。 如果属性出现在了模板表达式等号(=)的***右侧***,它就是一个*源*。
A property is a *target* when it appears in **square brackets** ([ ]) to the **left** of the equals (=) A property is a *target* when it appears in **square brackets** ([ ]) to the **left** of the equals (=)
as it is does when binding to the `myHighlight` property of the `HighlightDirective`. as it is does when binding to the `myHighlight` property of the `HighlightDirective`.
如果它出现在了**方括号**([ ])中,并且出现在等号(=)的**左侧**,它就是一个*目标*…… 如果它出现在了**方括号**([ ])中,并且出现在等号(=)的**左侧**,它就是一个*目标*……
就像在绑定到`HighlightDirective`的`myHighlight`属性时所做的那样。 就像在绑定到`HighlightDirective`的`myHighlight`属性时所做的那样。
+makeExample('attribute-directives/ts/app/app.component.html','pHost')(format=".") +makeExample('attribute-directives/ts/app/app.component.html','pHost')(format=".")
@ -612,26 +614,26 @@ a#why-input
The 'color' in `[myHighlight]="color"` is a binding ***source***. The 'color' in `[myHighlight]="color"` is a binding ***source***.
A source property doesn't require a declaration. A source property doesn't require a declaration.
`[myHighlight]="color"`中的'color'就是绑定***源***。 `[myHighlight]="color"`中的'color'就是绑定***源***。
源属性不需要特别声明。 源属性不需要特别声明。
The 'myHighlight' in `[myHighlight]="color"` *is* a binding ***target***. The 'myHighlight' in `[myHighlight]="color"` *is* a binding ***target***.
You must declare it as an *input* property or You must declare it as an *input* property or
Angular rejects the binding with a clear error. Angular rejects the binding with a clear error.
`[myHighlight]="color"`中的'myHighlight'就是绑定***目标***。 `[myHighlight]="color"`中的'myHighlight'就是绑定***目标***。
必须把它定义为一个*Input*属性否则Angular就会拒绝这次绑定并给出一个明确的错误。 必须把它定义为一个*Input*属性否则Angular就会拒绝这次绑定并给出一个明确的错误。
Angular treats a *target* property differently for a good reason. Angular treats a *target* property differently for a good reason.
A component or directive in target position needs protection. A component or directive in target position needs protection.
Angular这样区别对待*目标*属性有充分的理由。 Angular这样区别对待*目标*属性有充分的理由。
作为目标的组件或指令需要保护。 作为目标的组件或指令需要保护。
Imagine that `HighlightDirective` did truly wonderous things in a Imagine that `HighlightDirective` did truly wonderous things in a
popular open source project. popular open source project.
假想一下,`HighlightDirective`真是一个好东西。 假想一下,`HighlightDirective`真是一个好东西。
我们优雅的把它当作礼物送给全世界。 我们优雅的把它当作礼物送给全世界。
@ -646,5 +648,5 @@ a#why-input
The ***input*** declaration ensures that consumers of your directive can only bind to The ***input*** declaration ensures that consumers of your directive can only bind to
the properties of the public API but nothing else. the properties of the public API but nothing else.
于是,这种*输入*声明可以确保指令的消费者只能绑定到公开API中的属性其它的都不行。 于是,这种*输入*声明可以确保指令的消费者只能绑定到公开API中的属性其它的都不行。

View File

@ -7,6 +7,21 @@ block includes
我们将持续不断的更新和改进Angular文档。本日志记录了近期最重要的变更。 我们将持续不断的更新和改进Angular文档。本日志记录了近期最重要的变更。
## QuickStart Rewrite (2016-11-18)
## 全新《快速起步》 (2016-11-18)
The QuickStart is completely rewritten so that it actually is quick.
It references a minimal "Hello Angular" app running in Plunker.
The new [Setup](setup.html) page tells you how to install a local development environment
by downloading (or cloning) the QuickStart github repository.
You are no longer asked to copy-and-paste code into setup files that were not explained anyway.
《快速起步》被重新编写,变得更加快速。
它使用了在 Plunker 中运行的最小化的 “Hello Angular” 应用。
新添加的[搭建本地开发环境](setup.html)页面解释了如何通过下载或者克隆 QuickStart github 库来安装本地开发环境。
你将不再需要拷贝粘贴代码到一些并没有对其解释的配置文件中。
## Sync with Angular v.2.2.0 (2016-11-14) ## Sync with Angular v.2.2.0 (2016-11-14)
## 与Angular v.2.2.0同步(2016-11-14) ## 与Angular v.2.2.0同步(2016-11-14)
@ -108,7 +123,7 @@ block includes
## 与Angular v.2.1.0同步(2016-10-12) ## 与Angular v.2.1.0同步(2016-10-12)
Docs and code samples updated and tested with Angular v.2.1.0 Docs and code samples updated and tested with Angular v.2.1.0
使用Angular v.2.1.0更新和测试所有文档和代码例子。 使用Angular v.2.1.0更新和测试所有文档和代码例子。
@ -122,7 +137,7 @@ block includes
followed by the more advanced considerations of compiling and bundling the Tour of Heroes. followed by the more advanced considerations of compiling and bundling the Tour of Heroes.
全新[预编译(AoT)](../cookbook/aot-compiler.html)烹饪书介绍了什么是AoT编译和为何你需要它。 全新[预编译(AoT)](../cookbook/aot-compiler.html)烹饪书介绍了什么是AoT编译和为何你需要它。
它以**快速开始**应用程序开始讲解,接着介绍了编译和构建**英雄指南**的更高级的注意事项。 它以**快速起步**应用程序开始讲解,接着介绍了编译和构建**英雄指南**的更高级的注意事项。
## Sync with Angular v.2.0.2 (2016-10-6) ## Sync with Angular v.2.0.2 (2016-10-6)

View File

@ -408,7 +408,7 @@ block ctor-syntax
不需要创建 Angular 注入器。 不需要创建 Angular 注入器。
Angular 在启动过程中自动为我们创建一个应用级注入器。 Angular 在启动过程中自动为我们创建一个应用级注入器。
+makeExample('dependency-injection/ts/app/main.ts', 'bootstrap', 'app/main.ts (excerpt)')(format='.') +makeExcerpt('app/main.ts', 'bootstrap')
:marked :marked
We do have to configure the injector by registering the **providers** We do have to configure the injector by registering the **providers**
@ -418,24 +418,24 @@ block ctor-syntax
我们必须通过注册**提供商 (provider)** 来配置注入器,这些提供商为应用创建所需服务。 我们必须通过注册**提供商 (provider)** 来配置注入器,这些提供商为应用创建所需服务。
在本章的稍后部分会解释什么是[提供商](#providers)。 在本章的稍后部分会解释什么是[提供商](#providers)。
We can either register a provider within an [NgModule](ngmodule.html) or in application components block register-provider-ngmodule
:marked
We can either register a provider within an [NgModule](ngmodule.html) or in application components
或者在 [NgModule](ngmodule.html) 中注册提供商,或者在应用组件中。 或者在 [NgModule](ngmodule.html) 中注册提供商,或者在应用组件中。
### Registering providers in an NgModule ### Registering providers in an NgModule
### 在 NgModule 中注册提供商
### 在 NgModule 中注册提供商 Here's our AppModule where we register a `Logger`, a `UserService`, and an `APP_CONFIG` provider.
Here's our AppModule where we register a `Logger`, a `UserService`, and an `APP_CONFIG` provider. 下面的例子是在 AppModule 中注册`Logger`、`UserService`和`APP_CONFIG`提供商。
下面的例子是在 AppModule 中注册`Logger`、`UserService`和`APP_CONFIG`提供商。 - var app_module_ts = 'app/app.module.ts';
+makeExcerpt(app_module_ts + ' (excerpt)', 'ngmodule',app_module_ts, { otl: /(providers:)/ })
- var stylePattern = { otl: /(providers)/ }; //-The preferred approach is to register application providers in application components.
//-Because the `HeroService` is used within the *Heroes* feature area &mdash;
+makeExample('dependency-injection/ts/app/app.module.ts', 'ngmodule','app/app.module.ts', stylePattern)(format='.') //-and nowhere else &mdash; the ideal place to register it is in the top-level `HeroesComponent`.
<!--The preferred approach is to register application providers in application components.
Because the `HeroService` is used within the *Heroes* feature area &mdash;
and nowhere else &mdash; the ideal place to register it is in the top-level `HeroesComponent`.-->
:marked :marked
### Registering providers in a component ### Registering providers in a component
@ -445,33 +445,34 @@ block ctor-syntax
下面是更新的`HerosComponent`,它注册了`HeroService`。 下面是更新的`HerosComponent`,它注册了`HeroService`。
+makeExample('dependency-injection/ts/app/heroes/heroes.component.1.ts', 'full','app/heroes/heroes.component.ts', stylePattern)(format='.') - var stylePattern = { otl: /(providers:[^,]+),/ };
+makeExample('app/heroes/heroes.component.1.ts', 'full', 'app/heroes/heroes.component.ts', stylePattern)(format='.')
:marked block ngmodule-vs-component
### When to use the NgModule and when an application component?
### 该用 NgModule 还是应用组件?
On the one hand, a provider in an NgModule is registered in the root injector. That means that every provider
registered within an NgModule will be accessible in the entire application.
一方面NgModule 中的提供商是被注册到根注入器。这意味着在 NgModule 中注册的提供商可以被整个应用访问。
On the other hand, a provider registered in an application component is available only on that component and all its children.
另一方面,在应用组件中注册的提供商只在该组件及其子组件中可用。
We want the `APP_CONFIG` service to be available all across the application, but a `HeroService` is only used within the *Heroes*
feature area &mdash; and nowhere else. &mdash;
我们希望`APP_CONFIG`服务在整个应用中可用,而`HeroService`只需在*英雄*特性区可用,在其它地方都不可用。
.l-sub-section
:marked :marked
Read also **Should I add app-wide providers to the root `AppModule` or the root `AppComponent`?** in the [NgModule FAQ](../cookbook/ngmodule-faq.html#root-component-or-module) chapter. ### When to use the NgModule and when an application component?
参见 [NgModule FAQ](../cookbook/ngmodule-faq.html#root-component-or-module) 一章的 ### 该用 NgModule 还是应用组件?
**我该把“全应用级”提供商加到根模块`AppModule`还是根组件`AppComponent`**
On the one hand, a provider in an NgModule is registered in the root injector. That means that every provider
registered within an NgModule will be accessible in the _entire application_.
一方面NgModule 中的提供商是被注册到根注入器。这意味着在 NgModule 中注册的提供商可以被整个应用访问。
On the other hand, a provider registered in an application component is available only on that component and all its children.
另一方面,在应用组件中注册的提供商只在该组件及其子组件中可用。
We want the `APP_CONFIG` service to be available all across the application, but a `HeroService` is only used within the *Heroes*
feature area and nowhere else.
我们希望`APP_CONFIG`服务在整个应用中可用,而`HeroService`只需在*英雄*特性区可用,在其它地方都不可用。
.l-sub-section
:marked
Also see *"Should I add app-wide providers to the root `AppModule` or the root `AppComponent`?"* in the [NgModule FAQ](../cookbook/ngmodule-faq.html#root-component-or-module) .
参见 [NgModule FAQ](../cookbook/ngmodule-faq.html#root-component-or-module) 一章的
**我该把“全应用级”提供商加到根模块`AppModule`还是根组件`AppComponent`**
:marked :marked
### Preparing the HeroListComponent for injection ### Preparing the HeroListComponent for injection
@ -623,7 +624,6 @@ block ctor-syntax
现在,这个构造函数要求注入一个`Logger`类的实例,并把它存到名为`logger`的私有属性中。 现在,这个构造函数要求注入一个`Logger`类的实例,并把它存到名为`logger`的私有属性中。
当别人请求英雄数据时,在`getHeroes`方法中调用这个属性的方法。 当别人请求英雄数据时,在`getHeroes`方法中调用这个属性的方法。
//- FIXME refer to Dart API when that page becomes available.
- var injUrl = '../api/core/index/Injectable-decorator.html'; - var injUrl = '../api/core/index/Injectable-decorator.html';
h3#injectable Why @Injectable()? h3#injectable Why @Injectable()?
h3#injectable 为什么要用 @Injectable()? h3#injectable 为什么要用 @Injectable()?
@ -662,7 +662,7 @@ block injectable-not-always-needed-in-ts
li li
p <b>Future proofing:</b> No need to remember <code>@Injectable()</code> when we add a dependency later. p <b>Future proofing:</b> No need to remember <code>@Injectable()</code> when we add a dependency later.
p <b>面向未来:</b> 没有必要记得在后来添加依赖的时候添加 <code>@Injectable()</code>。 p <b>面向未来:</b> 没有必要记得在后来添加依赖的时候添加 <code>@Injectable()</code>。
li li
p <b>Consistency:</b> All services follow the same rules, and we don't have to wonder why #{_a} #{_decorator} is missing. p <b>Consistency:</b> All services follow the same rules, and we don't have to wonder why #{_a} #{_decorator} is missing.
p <b>一致性:</b>所有的服务都遵循同样的规则,不需要考虑为什么某个地方少了一个。 p <b>一致性:</b>所有的服务都遵循同样的规则,不需要考虑为什么某个地方少了一个。
@ -747,7 +747,7 @@ block injectable-not-always-needed-in-ts
这个日志服务很简单: 这个日志服务很简单:
+makeExample('dependency-injection/ts/app/logger.service.ts', null, 'app/logger.service.ts') +makeExample('app/logger.service.ts')
block real-logger block real-logger
//- N/A //- N/A
@ -755,12 +755,12 @@ block real-logger
:marked :marked
We're likely to need the same logger service everywhere in our application, We're likely to need the same logger service everywhere in our application,
so we put it in the project's `#{_appDir}` folder, and so we put it in the project's `#{_appDir}` folder, and
we register it in the `providers` #{_array} of the metadata for our application module, `AppModule`. we register it in the `providers` #{_array} of our application !{_moduleVsComp}, `!{_AppModuleVsAppComp}`.
应用的每个角落都可能需要日志服务,所以把它放到项目的`#{_appDir}`目录, 应用的每个角落都可能需要日志服务,所以把它放到项目的`#{_appDir}`目录,
并在应用模块`AppModule`的元数据`providers`数组里注册它。 并在应用模块`AppModule`的元数据`providers`数组里注册它。
+makeExcerpt('app/providers.component.ts','providers-logger','app/app.module.ts (excerpt)') +makeExcerpt('app/providers.component.ts (excerpt)', 'providers-logger','app/app.module.ts')
:marked :marked
If we forget to register the logger, Angular throws an exception when it first looks for the logger: If we forget to register the logger, Angular throws an exception when it first looks for the logger:
@ -873,7 +873,7 @@ block provider-ctor-args
第一个是[令牌 (token)](#token),它作为键值 (key) 使用,用于定位依赖值和注册提供商。 第一个是[令牌 (token)](#token),它作为键值 (key) 使用,用于定位依赖值和注册提供商。
The second is a !{_secondParam}, The second is a !{_secondParam},
which we can think of as a *recipe* for creating the dependency value. which we can think of as a *recipe* for creating the dependency value.
There are many ways to create dependency values ... and many ways to write a recipe. There are many ways to create dependency values ... and many ways to write a recipe.
@ -1226,9 +1226,8 @@ block what-should-we-use-as-token
这不是 Angular 的错。接口只是 TypeScript 设计时 (design-time) 的概念。JavaScript 没有接口。 这不是 Angular 的错。接口只是 TypeScript 设计时 (design-time) 的概念。JavaScript 没有接口。
TypeScript 接口不会出现在生成的 JavaScript 代码中。 TypeScript 接口不会出现在生成的 JavaScript 代码中。
在运行期,没有接口类型信息可供 Angular 查找。 在运行期,没有接口类型信息可供 Angular 查找。
// end Typescript only
//- FIXME simplify once APIs are defined for Dart. //- FIXME update once https://github.com/dart-lang/angular2/issues/16 is addressed.
- var opaquetoken = _docsFor == 'dart' ? '<b>OpaqueToken</b>' : '<a href="../api/core/index/OpaqueToken-class.html"><b>OpaqueToken</b></a>' - var opaquetoken = _docsFor == 'dart' ? '<b>OpaqueToken</b>' : '<a href="../api/core/index/OpaqueToken-class.html"><b>OpaqueToken</b></a>'
:marked :marked
### OpaqueToken ### OpaqueToken

View File

@ -51,14 +51,13 @@ figure.image-display
要显示组件的属性最简单的方式就是通过插值表达式interpolation来绑定属性名。 要显示组件的属性最简单的方式就是通过插值表达式interpolation来绑定属性名。
要使用插值表达式,就把属性名包裹在双花括号里放进视图模板,如`{{myHero}}`。 要使用插值表达式,就把属性名包裹在双花括号里放进视图模板,如`{{myHero}}`。
To build an illustrative example, start by creating a new project folder called <ngio-ex path="displaying-data"></ngio-ex> Follow the [setup](setup.html) instructions for creating a new project
and following the steps in [QuickStart](../quickstart.html). named <ngio-ex path="displaying-data"></ngio-ex>.
我们来一起做个简明的小例子。创建一个新的项目文件夹(<ngio-ex path="displaying-data"></ngio-ex>),并且完成[“快速起步”](../quickstart.html)中的步骤 按照[搭建本地开发环境](setup.html)的说明,创建一个新项目,名为<ngio-ex path="displaying-data"></ngio-ex>。
: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`文件中修改组件的模板和代码。
@ -101,7 +100,7 @@ figure.image-display
.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.
严格来说“重新显示”是在某些与视图有关的异步事件之后发生的比如按键、定时器完成或对HTTP请求的响应。 严格来说“重新显示”是在某些与视图有关的异步事件之后发生的比如按键、定时器完成或对HTTP请求的响应。
@ -111,12 +110,11 @@ figure.image-display
注意,我们从没调用过**new**来创建`AppComponent`类的实例是Angular替我们创建了它。那么它是如何创建的呢 注意,我们从没调用过**new**来创建`AppComponent`类的实例是Angular替我们创建了它。那么它是如何创建的呢
The CSS `selector` in the `@Component` !{_decorator} specifies an element named `my-app`. The CSS `selector` in the `@Component` !{_decorator} specifies an element named `<my-app>`.
Remember back in [QuickStart](../quickstart.html) that you added the `<my-app>` That element is a placeholder in the body of your `index.html` file:
element to the body of your `index.html` file:
注意`@Component`装饰器中指定的CSS选择器`selector`,它指定了一个叫`my-app`的元素。 注意`@Component`装饰器中指定的CSS选择器`selector`,它指定了一个叫`my-app`的元素。
回忆下,在[“快速起步”](../quickstart.html)一章中,我们曾把一个`<my-app>`元素添加到`index.html`的`body`里 该元素是`index.html`的`body`里的占位符
+makeExcerpt('index.html', 'body') +makeExcerpt('index.html', 'body')
@ -217,7 +215,7 @@ figure.image-display
.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)一章。

View File

@ -15,13 +15,13 @@ include ../_util-fns
但是,要想做出具有贴心的数据输入体验的表单, 但是,要想做出具有贴心的数据输入体验的表单,
引导用户明晰、高效地完成表单背后的工作流程,挑战就大多了。 引导用户明晰、高效地完成表单背后的工作流程,挑战就大多了。
*That* takes design skills that are, to be frank, well out of scope for this chapter. *That* takes design skills that are, to be frank, well out of scope for this guide.
坦白地讲,*这当中*所需要的设计技能超出了本章的范围。 坦白地讲,*这当中*所需要的设计技能超出了本章的范围。
It also takes framework support for It also takes framework support for
**two-way data binding, change tracking, validation, and error handling** **two-way data binding, change tracking, validation, and error handling**
... which we shall cover in this chapter on Angular forms. ... which we shall cover in this guide on Angular forms.
**双向数据绑定、变更跟踪、有效性验证和错误处理**等功能离不开框架的支持。 **双向数据绑定、变更跟踪、有效性验证和错误处理**等功能离不开框架的支持。
本章将介绍Angular表单相关的内容。 本章将介绍Angular表单相关的内容。
@ -65,16 +65,14 @@ include ../_util-fns
## 模板驱动的表单 ## 模板驱动的表单
Many of us will build forms by writing templates in the Angular [template syntax](./template-syntax.html) with Many of us will build forms by writing templates in the Angular [template syntax](./template-syntax.html) with
the form-specific directives and techniques described in this chapter. the form-specific directives and techniques described in this guide.
通常使用Angular[模板语法](./template-syntax.html)编写模板,结合本章所描述的表单专用指令和技术来构建表单。 通常使用Angular[模板语法](./template-syntax.html)编写模板,结合本章所描述的表单专用指令和技术来构建表单。
.l-sub-section .l-sub-section
:marked :marked
That's not the only way to create a form but it's the way we'll cover in this chapter. That's not the only way to create a form but it's the way we'll cover in this guide.
这不是创建表单的唯一方式,本章中只介绍模板驱动的表单。(译注Angular支持的另一种方式叫做模型驱动表单Model-Driven Forms) 这不是创建表单的唯一方式,本章中只介绍模板驱动的表单。(译注Angular支持的另一种方式叫做模型驱动表单Model-Driven Forms)
:marked :marked
We can build almost any form we need with an Angular template &mdash; login forms, contact forms ... pretty much any business forms. We can build almost any form we need with an Angular template &mdash; login forms, contact forms ... pretty much any business forms.
We can lay out the controls creatively, bind them to data, specify validation rules and display validation errors, We can lay out the controls creatively, bind them to data, specify validation rules and display validation errors,
@ -170,12 +168,11 @@ figure.image-display
## 搭建 ## 搭建
Create a new project folder (`angular-forms`) and follow the steps in the [QuickStart](../quickstart.html). Follow the [setup](setup.html) instructions for creating a new project
named <span ngio-ex>angular-forms</span>.
创建新的项目文件夹 (`angular-forms`),并且完成[“快速起步”](../quickstart.html)中的步骤 按照[搭建本地开发环境](setup.html)的说明,创建一个名为<span ngio-ex>angular-forms</span>的新项目
include ../_quickstart_repo
:marked
## Create the Hero Model Class ## Create the Hero Model Class
## 创建Hero模型类 ## 创建Hero模型类
@ -229,13 +226,14 @@ code-example(format="").
## 创建表单组件 ## 创建表单组件
An Angular form has two parts: an HTML-based template and a code-based Component to handle data and user interactions. An Angular form has two parts: an HTML-based _template_ and a component _class_
to handle data and user interactions programmatically.
Angular 表单分为两部分:基于 HTML 的模板和基于代码的组件。组件用来处理数据和用户交互。 Angular 表单分为两部分:基于 HTML 的*模板*和组件*类*,用来程序处理数据和用户交互。
We begin with the Component because it states, in brief, what the Hero editor can do. We begin with the class because it states, in brief, what the hero editor can do.
先从组件开始,是因为它可以简要说明英雄编辑器能做什么。 先从组件开始,是因为它可以简要说明英雄编辑器能做什么。
Create a new file called `hero-form.component.ts` and give it the following definition: Create a new file called `hero-form.component.ts` and give it the following definition:
@ -248,7 +246,7 @@ code-example(format="").
这个组件没有什么特别的地方,没有表单相关的东西,与之前写过的组件没什么不同。 这个组件没有什么特别的地方,没有表单相关的东西,与之前写过的组件没什么不同。
Understanding this component requires only the Angular concepts weve learned in previous chapters Understanding this component requires only the Angular concepts weve learned in previous guides
只需要前面章节中学过的概念,就可以完全理解这个组件: 只需要前面章节中学过的概念,就可以完全理解这个组件:
@ -288,10 +286,13 @@ code-example(format="").
在最后增加`diagnostic`属性它返回这个模型的JSON形式。 在最后增加`diagnostic`属性它返回这个模型的JSON形式。
在开发过程中,它用于调试,最后清理时会丢弃它。 在开发过程中,它用于调试,最后清理时会丢弃它。
Why don't we write the template inline in the component file as we often do ### Why the separate template file?
elsewhere in the Developer Guide?
这次为什么不像在开发指南其它地方那样,以内联的方式把模板写在组件文件中呢? ### 为何分离模板文件?
Why don't we write the template inline in the component file as we often do elsewhere?
为什么不与我们在其他地方常常做的那样,以内联的方式把模板写在组件文件中呢?
There is no “right” answer for all occasions. We like inline templates when they are short. There is no “right” answer for all occasions. We like inline templates when they are short.
Most form templates won't be short. TypeScript and JavaScript files generally aren't the best place to Most form templates won't be short. TypeScript and JavaScript files generally aren't the best place to
@ -303,12 +304,13 @@ code-example(format="").
而且没有几个编辑器能对混写的 HTML 和代码提供足够的帮助。 而且没有几个编辑器能对混写的 HTML 和代码提供足够的帮助。
我们还是喜欢内容清晰、目标明确的短文件,像这个一样。 我们还是喜欢内容清晰、目标明确的短文件,像这个一样。
We made a good choice to put the HTML template elsewhere. Form templates tend to be quite large even when displaying a small number of fields
We'll write that template in a moment. Before we do, we'll take a step back so it's usually best to put the HTML template in a separate file.
and revise the `app.module.ts` and `app.component.ts` to make use of our new `HeroFormComponent`. We'll write that template file in a moment. Before we do, we'll take a step back
and revise the `app.module.ts` and `app.component.ts` to make use of the new `HeroFormComponent`.
把 HTML 模板放在其它地方是个不错的选择。等会儿就去写那个模板 就算是在仅仅显示少数表单项目时表单模板一般都比较庞大。所以通常最好的方式是将HTML模板放到单独的文件中
在这之前,先回来修改`app.module.ts`和`app.component.ts`文件,用上新写的`HeroFormComponent`组件 一会儿将编写这个模板文件。在这之前,先退一步,再看看`app.module.ts`和`app.component.ts`,让它们使用新的`HeroFormComponent`
.l-main-section .l-main-section
:marked :marked
@ -354,7 +356,7 @@ code-example(format="").
.alert.is-important .alert.is-important
:marked :marked
If a component, directive, or pipe belongs to a module in the `imports` array, _DON'T_ declare it in the `declarations` array. If a component, directive, or pipe belongs to a module in the `imports` array, _DON'T_ re-declare it in the `declarations` array.
If you wrote it and it should belong to this module, _DO_ declare it in the `declarations` array. If you wrote it and it should belong to this module, _DO_ declare it in the `declarations` array.
如果组件、指令或管道出现在模块的`imports`数组中_不要_把它声明在`declarations`数组中。 如果组件、指令或管道出现在模块的`imports`数组中_不要_把它声明在`declarations`数组中。
@ -378,9 +380,9 @@ code-example(format="").
:marked :marked
.l-sub-section .l-sub-section
:marked :marked
There is only one change: There is only one change.
仅有的一处修改是: 仅有的一处修改
1. The `template` is simply the new element tag identified by the component's `selector` property. 1. The `template` is simply the new element tag identified by the component's `selector` property.
This will display the hero form when the application component is loaded. This will display the hero form when the application component is loaded.
@ -420,8 +422,7 @@ code-example(format="").
The `container`, `form-group`, `form-control`, and `btn` classes The `container`, `form-group`, `form-control`, and `btn` classes
come from [Twitter Bootstrap](http://getbootstrap.com/css/). Purely cosmetic. come from [Twitter Bootstrap](http://getbootstrap.com/css/). Purely cosmetic.
We're using Bootstrap to gussy up our form. We're using Bootstrap to give the form a little style!
Hey, what's a form without a little style!
`container`、`form-group`、`form-control`和`btn`类来自 [Twitter Bootstrap](http://getbootstrap.com/css/)。纯粹是装饰。 `container`、`form-group`、`form-control`和`btn`类来自 [Twitter Bootstrap](http://getbootstrap.com/css/)。纯粹是装饰。
我们使用 Bootstrap 来美化表单。嘿,一点样式都没有的表单算个啥! 我们使用 Bootstrap 来美化表单。嘿,一点样式都没有的表单算个啥!
@ -474,7 +475,7 @@ ol
We'll add a `select` to our We'll add a `select` to our
form and bind the options to the `powers` list using `ngFor`, form and bind the options to the `powers` list using `ngFor`,
a technique we might have seen before in the [Displaying Data](./displaying-data.html) chapter. a technique seen previously in the [Displaying Data](./displaying-data.html) guide.
在表单中添加`select`,用`ngFor`把`powers`列表绑定到列表选项。 在表单中添加`select`,用`ngFor`把`powers`列表绑定到列表选项。
我们在之前的[显示数据](./displaying-data.html)一章中见过`ngFor`。 我们在之前的[显示数据](./displaying-data.html)一章中见过`ngFor`。
@ -486,7 +487,7 @@ ol
:marked :marked
We are repeating the `<options>` tag for each power in the list of Powers. We are repeating the `<options>` tag for each power in the list of Powers.
The `p` template input variable is a different power in each iteration; The `pow` template input variable is a different power in each iteration;
we display its name using the interpolation syntax with the double-curly-braces. we display its name using the interpolation syntax with the double-curly-braces.
列表中的每一项超能力都会渲染成`<option>`标签。 列表中的每一项超能力都会渲染成`<option>`标签。
@ -495,10 +496,9 @@ ol
<a id="ngModel"></a> <a id="ngModel"></a>
.l-main-section .l-main-section
:marked :marked
## Two-way data binding with **ngModel** ## Two-way data binding with **_ngModel_**
## 使用 **ngModel** 进行双向数据绑定
## 使用 ***ngModel*** 进行双向数据绑定
Running the app right now would be disappointing. Running the app right now would be disappointing.
如果立即运行此应用,你将会失望。 如果立即运行此应用,你将会失望。
@ -507,7 +507,7 @@ figure.image-display
img(src="/resources/images/devguide/forms/hero-form-3.png" width="400px" alt="没有数据绑定的早期表单") img(src="/resources/images/devguide/forms/hero-form-3.png" width="400px" alt="没有数据绑定的早期表单")
:marked :marked
We don't see hero data because we are not binding to the `Hero` yet. We don't see hero data because we are not binding to the `Hero` yet.
We know how to do that from earlier chapters. We know how to do that from earlier guides.
[Displaying Data](./displaying-data.html) taught us Property Binding. [Displaying Data](./displaying-data.html) taught us Property Binding.
[User Input](./user-input.html) showed us how to listen for DOM events with an [User Input](./user-input.html) showed us how to listen for DOM events with an
Event Binding and how to update a component property with the displayed value. Event Binding and how to update a component property with the displayed value.
@ -532,7 +532,7 @@ figure.image-display
找到“Name”对应的`<input>`标签,并且像这样修改它: 找到“Name”对应的`<input>`标签,并且像这样修改它:
+makeExample('forms/ts/app/hero-form.component.html', 'ngModel-1','app/hero-form.component.html (节选)')(format=".") +makeExample('forms/ts/app/hero-form.component.html', 'ngModelName-1','app/hero-form.component.html (节选)')(format=".")
.l-sub-section .l-sub-section
:marked :marked
@ -575,7 +575,7 @@ figure.image-display
:marked :marked
Internally Angular creates `FormControls` and registers them with an `NgForm` directive that Angular Internally Angular creates `FormControls` and registers them with an `NgForm` directive that Angular
attached to the `<form>` tag. Each `FormControl` is registered under the name we assigned to the `name` attribute. attached to the `<form>` tag. Each `FormControl` is registered under the name we assigned to the `name` attribute.
We'll talk about `NgForm` [later in this chapter](#ngForm). We'll talk about `NgForm` [later in this guide](#ngForm).
在内部Angular 创建了一些`FormControl`,并把它们注册到`NgForm`指令,再将该指令附加到`<form>`标签。 在内部Angular 创建了一些`FormControl`,并把它们注册到`NgForm`指令,再将该指令附加到`<form>`标签。
注册每个`FormControl`时,使用`name`属性值作为键值。[本章后面](#ngForm)会讨论`NgForm`。 注册每个`FormControl`时,使用`name`属性值作为键值。[本章后面](#ngForm)会讨论`NgForm`。
@ -584,7 +584,7 @@ figure.image-display
Let's add similar `[(ngModel)]` bindings and `name` attributes to *Alter Ego* and *Hero Power*. Let's add similar `[(ngModel)]` bindings and `name` attributes to *Alter Ego* and *Hero Power*.
We'll ditch the input box binding message We'll ditch the input box binding message
and add a new binding at the top to the component's `diagnostic` property. and add a new binding at the top to the component's `diagnostic` property.
Then we can confirm that two-way data binding works *for the entire Hero model*. Then we can confirm that two-way data binding works *for the entire hero model*.
为*第二人格*和*超能力*属性添加类似的`[(ngModel)]`绑定和`name`属性。 为*第二人格*和*超能力*属性添加类似的`[(ngModel)]`绑定和`name`属性。
抛弃输入框的绑定消息,在组件顶部添加到`diagnostic`属性的新绑定。 抛弃输入框的绑定消息,在组件顶部添加到`diagnostic`属性的新绑定。
@ -609,7 +609,7 @@ figure.image-display
每个 input 元素都有`name`属性Angular 表单用它注册控件。 每个 input 元素都有`name`属性Angular 表单用它注册控件。
:marked :marked
If we ran the app right now and changed every Hero model property, the form might display like this: If we ran the app right now and changed every hero model property, the form might display like this:
如果现在运行本应用,修改 Hero 模型的每个属性,表单看起来像这样: 如果现在运行本应用,修改 Hero 模型的每个属性,表单看起来像这样:
figure.image-display figure.image-display
@ -626,43 +626,43 @@ figure.image-display
.l-sub-section .l-sub-section
:marked :marked
### Inside [(ngModel)] ### Inside _[(ngModel)]_
### [(ngModel)]内幕 ### [(ngModel)]内幕
*This section is an optional deep dive into [(ngModel)]. Not interested? Skip ahead!* *This section is an optional deep dive into [(ngModel)]. Not interested? Skip ahead!*
*本节是对[(ngModel)]的深入剖析,它是可选的。不感兴趣?跳过它!* *本节是对[(ngModel)]的深入剖析,它是可选的。不感兴趣?跳过它!*
The punctuation in the binding syntax, <span style="font-family:courier"><b>[()]</b></span>, is a good clue to what's going on. The punctuation in the binding syntax, <span style="font-family:courier"><b>[()]</b></span>, is a good clue to what's going on.
绑定语法中的<span style="font-family:courier"><b>[()]</b></span>是很好的线索。 绑定语法中的<span style="font-family:courier"><b>[()]</b></span>是很好的线索。
In a Property Binding, a value flows from the model to a target property on screen. In a Property Binding, a value flows from the model to a target property on screen.
We identify that target property by surrounding its name in brackets, <span style="font-family:courier"><b>[]</b></span>. We identify that target property by surrounding its name in brackets, <span style="font-family:courier"><b>[]</b></span>.
This is a one-way data binding **from the model to the view**. This is a one-way data binding **from the model to the view**.
在属性绑定中,值从模型中流动到屏幕上的目标属性 (property)。 在属性绑定中,值从模型中流动到屏幕上的目标属性 (property)。
通过把属性名括在方括号中来标记出目标属性,<span style="font-family:courier"><b>[]</b></span>。 通过把属性名括在方括号中来标记出目标属性,<span style="font-family:courier"><b>[]</b></span>。
这是**从模型到视图**的单向数据绑定。 这是**从模型到视图**的单向数据绑定。
In an Event Binding, we flow the value from the target property on screen to the model. In an Event Binding, we flow the value from the target property on screen to the model.
We identify that target property by surrounding its name in parentheses, <span style="font-family:courier"><b>()</b></span>. We identify that target property by surrounding its name in parentheses, <span style="font-family:courier"><b>()</b></span>.
This is a one-way data binding in the opposite direction **from the view to the model**. This is a one-way data binding in the opposite direction **from the view to the model**.
在事件绑定中,值从屏幕上的目标属性流动到模型。 在事件绑定中,值从屏幕上的目标属性流动到模型。
通过把属性名括在圆括号中来标记出目标属性,<span style="font-family:courier"><b>()</b></span>。 通过把属性名括在圆括号中来标记出目标属性,<span style="font-family:courier"><b>()</b></span>。
这是**从视图到模型**的反向单向数据绑定。 这是**从视图到模型**的反向单向数据绑定。
No wonder Angular chose to combine the punctuation as <span style="font-family:courier"><b>[()]</b></span> No wonder Angular chose to combine the punctuation as <span style="font-family:courier"><b>[()]</b></span>
to signify a two-way data binding and a **flow of data in both directions**. to signify a two-way data binding and a **flow of data in both directions**.
不出所料Angular选择了组合标点 <span style="font-family:courier"><b>[()]</b></span> 来标记出双向数据绑定和**双向数据流**。 不出所料Angular选择了组合标点 <span style="font-family:courier"><b>[()]</b></span> 来标记出双向数据绑定和**双向数据流**。
In fact, we can break the `NgModel` binding into its two separate modes In fact, we can break the `NgModel` binding into its two separate modes
as we do in this re-write of the "Name" `<input>` binding: as we do in this re-write of the "Name" `<input>` binding:
事实上,可以把`NgModel`绑定拆成两个独立的绑定就像下面重写的“Name”`<input>`绑定一样: 事实上,可以把`NgModel`绑定拆成两个独立的绑定就像下面重写的“Name”`<input>`绑定一样:
+makeExample('forms/ts/app/hero-form.component.html', 'ngModel-3','app/hero-form.component.html (excerpt)')(format=".") +makeExample('forms/ts/app/hero-form.component.html', 'ngModel-3','app/hero-form.component.html (excerpt)')(format=".")
:marked :marked
@ -699,13 +699,13 @@ figure.image-display
只有当需要在事件处理函数中做一些特别的事情(比如合并或限制按键频率)时,才会拆分出独立的事件处理函数。 只有当需要在事件处理函数中做一些特别的事情(比如合并或限制按键频率)时,才会拆分出独立的事件处理函数。
Learn more about `NgModel` and other template syntax in the Learn more about `NgModel` and other template syntax in the
[Template Syntax](./template-syntax.html) chapter. [Template Syntax](./template-syntax.html) guide.
要学习`ngModel`和其它模板语法的更多知识,见[模板语法](./template-syntax.html)。 要学习`ngModel`和其它模板语法的更多知识,见[模板语法](./template-syntax.html)。
.l-main-section .l-main-section
:marked :marked
## Track change-state and validity with **ngModel** ## Track change-state and validity with **_ngModel_**
## 通过 **ngModel** 跟踪修改状态与有效性验证 ## 通过 **ngModel** 跟踪修改状态与有效性验证
@ -917,7 +917,7 @@ figure.image-display
现在,把`div`元素的`hidden`属性绑定到`name`控件的属性,这样就可以控制“姓名”字段错误信息的可见性了。 现在,把`div`元素的`hidden`属性绑定到`name`控件的属性,这样就可以控制“姓名”字段错误信息的可见性了。
+makeExample('forms/ts/app/hero-form.component.html', +makeExample('forms/ts/app/hero-form.component.html',
'hidden-error-msg', 'hidden-error-msg',
'app/hero-form.component.html (节选)') 'app/hero-form.component.html (节选)')(format='.')
:marked :marked
In this example, we hide the message when the control is valid or pristine; In this example, we hide the message when the control is valid or pristine;
pristine means the user hasn't changed the value since it was displayed in this form. pristine means the user hasn't changed the value since it was displayed in this form.
@ -942,7 +942,7 @@ figure.image-display
如果当控件是“全新”状态时也隐藏消息,就能达到这个目的。 如果当控件是“全新”状态时也隐藏消息,就能达到这个目的。
在往表单中[添加新英雄](#new-hero)时,将看到这种选择的重要性。 在往表单中[添加新英雄](#new-hero)时,将看到这种选择的重要性。
The Hero *Alter Ego* is optional so we can leave that be. The hero *Alter Ego* is optional so we can leave that be.
英雄的*第二人格*是可选项,所以不用改它。 英雄的*第二人格*是可选项,所以不用改它。
@ -964,17 +964,18 @@ figure.image-display
## 添加英雄及重置表单 ## 添加英雄及重置表单
We'd like to add a new hero in this form. We'd like to add a new hero in this form.
We place a "New Hero" button at the bottom of the form and bind its click event to a component method. We place a "New Hero" button at the bottom of the form and bind its click event to a `newHero` component method.
我们希望在这个表单中添加新的英雄。 我们希望在这个表单中添加新的英雄。
在表单的底部放置“New Hero新增英雄”按钮并把它的点击事件绑定到组件方法。 在表单的底部放置“New Hero新增英雄”按钮并把它的点击事件绑定到`newHero`组件。
+makeExample('forms/ts/app/hero-form.component.html', +makeExample('forms/ts/app/hero-form.component.html',
'new-hero-button', 'new-hero-button-no-reset',
'app/hero-form.component.html (新增英雄按钮)') 'app/hero-form.component.html (新增英雄按钮)')
:marked :marked
+makeExample('forms/ts/app/hero-form.component.ts', +makeExample('forms/ts/app/hero-form.component.ts',
'new-hero-v1', 'new-hero',
'app/hero-form.component.ts (新增英雄按钮 - v1)')(format=".") 'app/hero-form.component.ts (新英雄方法)')(format=".")
:marked :marked
Run the application again, click the *New Hero* button, and the form clears. Run the application again, click the *New Hero* button, and the form clears.
The *required* bars to the left of the input box are red, indicating invalid `name` and `power` properties. The *required* bars to the left of the input box are red, indicating invalid `name` and `power` properties.
@ -987,59 +988,39 @@ figure.image-display
错误信息是隐藏的,因为表单还是全新的,还没有修改任何东西。 错误信息是隐藏的,因为表单还是全新的,还没有修改任何东西。
Enter a name and click *New Hero* again. Enter a name and click *New Hero* again.
This time we see an error message! Why? We don't want that when we display a new (empty) hero. The app displays a **_Name is required_** error message!
We don't want error messages when we create a new (empty) hero.
Why are we getting one now?
输入名字,再次点击 *New Hero* 按钮。 输入名字,再次点击 *New Hero* 按钮。
这次,出现了错误信息!为什么?我们不希望显示新(空)的英雄时,出现错误信息。 这次,出现了错误信息!为什么?我们不希望显示新(空)的英雄时,出现错误信息。
Inspecting the element in the browser tools reveals that the *name* input box is no longer pristine. Inspecting the element in the browser tools reveals that the *name* input box is _no longer pristine_.
Replacing the hero *did not restore the pristine state* of the control. The form remembers that we entered a name before clicking *New Hero*.
Replacing the hero *did not restore the pristine state* of the form controls.
使用浏览器工具审查这个元素就会发现,这个*name*输入框并不是全新的。 使用浏览器工具审查这个元素就会发现,这个*name*输入框并不是全新的。
表单记得我们在点击*New Hero*前输入的名字。
更换了英雄*并不会重置控件的“全新”状态*。 更换了英雄*并不会重置控件的“全新”状态*。
.l-sub-section
:marked
Upon reflection, we realize that Angular cannot distinguish between
replacing the entire hero and clearing the `name` property programmatically.
Angular makes no assumptions and leaves the control in its current, dirty state.
沉思一番你会发现Angular 没办法区分是替换了整个英雄数据还是用程序清除了`name`属性。 We have to clear all of the flags imperatively which we can do
Angular 不能作出假设,只好让控件保留当前状态 —— 脏状态。 by calling the form's `reset()` method after calling the `newHero()` method.
:marked
We'll have to reset the form controls manually with a small trick.
We add an `active` flag to the component, initialized to `true`. When we add a new hero,
we toggle `active` false and then immediately back to true with a quick `setTimeout`.
可以用个小花招来重置表单控件。 我们必须清除所有标记,在调用`newHero()`方法后调用表单的`reset()`方法即可。
给组件添加`active`标记,初始化为`true`。当添加新的英雄时,把`active`标记设置为`false`
再通过快速的`setTimeout`函数迅速把它设置回`true`。
+makeExample('forms/ts/app/hero-form.component.ts',
'new-hero',
'app/hero-form.component.ts (新增英雄 - 最终版)')(format=".")
:marked
Then we bind the form element to this `active` flag.
然后,把 form 元素绑定到这个`active`标志。
+makeExample('forms/ts/app/hero-form.component.html', +makeExample('forms/ts/app/hero-form.component.html',
'form-active', 'new-hero-button-form-reset',
'app/hero-form.component.html (Form标签)') 'app/hero-form.component.html (Reset the form)')
:marked :marked
With `NgIf` bound to the `active` flag, Now clicking "New Hero" both resets the form and its control flags.
clicking "New Hero" removes the form from the DOM and recreates it in a blink of an eye.
The re-created form is in a pristine state. The error message is hidden.
因为`NgIf`绑定到`active`标志,点击 "New Hero" 将从DOM中移除这个表单并在一眨眼的功夫重建它。 现在点击“New Hero”重设表单和它的控制标记。
重新创建的表单处于“全新”状态。错误信息被隐藏了。
.l-sub-section
:marked
This is a temporary workaround while we await a proper form reset feature.
这只是临时的变通方案,将来会用更好的方式来重置表单。
:marked :marked
.l-main-section .l-main-section
:marked :marked
## Submit the form with **ngSubmit** ## Submit the form with **_ngSubmit_**
## 使用 **ngSubmit** 提交表单 ## 使用 **ngSubmit** 提交表单
@ -1072,10 +1053,9 @@ figure.image-display
<a id="ngForm"></a> <a id="ngForm"></a>
.l-sub-section .l-sub-section
:marked :marked
### The NgForm directive ### The _NgForm_ directive
### NgForm指令 ### NgForm指令
What `NgForm` directive? We didn't add an [NgForm](../api/forms/index/NgForm-directive.html) directive! What `NgForm` directive? We didn't add an [NgForm](../api/forms/index/NgForm-directive.html) directive!
什么`NgForm`指令?之前没有添加过[NgForm](../api/common/index/NgForm-directive.html)指令啊! 什么`NgForm`指令?之前没有添加过[NgForm](../api/common/index/NgForm-directive.html)指令啊!
@ -1144,7 +1124,7 @@ figure.image-display
jazzing it up won't teach us anything new about forms. jazzing it up won't teach us anything new about forms.
But this is an opportunity to exercise some of our newly won But this is an opportunity to exercise some of our newly won
binding skills. binding skills.
If you're not interested, you can skip to the chapter's conclusion If you're not interested, you can skip to the guide's conclusion
and not miss a thing. and not miss a thing.
对演示来说,这个收场很平淡的。老实说,即使让它更出彩,也无法教给我们任何关于表单的新知识。 对演示来说,这个收场很平淡的。老实说,即使让它更出彩,也无法教给我们任何关于表单的新知识。
@ -1214,7 +1194,7 @@ figure.image-display
## 结论 ## 结论
The Angular form techniques discussed in this chapter take The Angular form techniques discussed in this guide take
advantage of the following framework features to provide support for data modification, validation and more: advantage of the following framework features to provide support for data modification, validation and more:
本章讨论的 Angular 表单技术利用了下列框架特性来支持数据修改、验证和更多操作: 本章讨论的 Angular 表单技术利用了下列框架特性来支持数据修改、验证和更多操作:

View File

@ -1,18 +1,26 @@
block includes block includes
include ../_util-fns include ../_util-fns
- var _angular_io = 'angular.cn';
- var __lang = _docsFor || current.path[1] || 'ts';
- var guideData = public.docs[__lang].latest.guide._data;
- var advancedLandingPage = '';
- for(var page in guideData) {
- if (!guideData[page].basics && !guideData[page].hide) { advancedLandingPage = page; break; }
- }
- var advancedUrl = './' + advancedLandingPage + '.html'
figure
img(src="/resources/images/devguide/intro/people.png" alt="Us" align="left" style="width:200px; margin-left:-40px;margin-right:10px")
:marked :marked
This is a practical guide to Angular for experienced programmers who This page describes the Angular documentation at a high level.
are building client applications in HTML and #{_Lang}. <br class="l-clear-left"> If you're new to Angular, you may want to visit "[Learning Angular](learning-angular.html)" first.
这是一份Angular实战指南。面向的是正在用HTML和#{langName}构建客户端应用的、有经验的程序员。 本页是Angular文档的概述。
如果你刚接触 Angular请先访问"[学习 Angular](learning-angular.html)"。
## Organization ## Themes
## 组织结构 ## 文档结构
The documentation is divided into major thematic sections, each The documentation is divided into major thematic sections, each
a collection of pages devoted to that theme. a collection of pages devoted to that theme.
@ -38,31 +46,36 @@ table(width="100%")
p <b><a href="../quickstart.html">快速起步</a></b> p <b><a href="../quickstart.html">快速起步</a></b>
td td
:marked :marked
The foundation for every page and sample in this documentation. A first taste of Angular<span if-docs="ts"> with zero installation.
Run "Hello World" in an online code editor and start playing with live code</span>.
<span if-docs="ts">零配置第一次尝试 Angular </span><span if-docs="ts">在在线代码编辑器中运行“Hello World”并利用在线代码开始体验</span>。
本文档中每一个页面和范例的基础工作。
tr(style=top) tr(style=top)
td td
p <b><a href="./">Guide</a></b> p <b><a href="./">Guide</a></b>
p <b><a href="./">指南</a></b> p <b><a href="./">指南</a></b>
td td
:marked :marked
The essential ingredients of Angular development. Learn the Angular basics (you're already here!) like the setup for local development,
displaying data and accepting user input, injecting application services into components,
and building simple forms.
Angular开发中必不可少的要素。 学习 Angular 基础知识(你已经在这儿了!),比如搭建本地开发环境、显示数据和接受用户输入、注入应用程序服务到组件中,
以及构建简单表单。
tr(style=top) tr(style=top)
td td
p <b><a href="../api">API Reference</a></b> p <b><a href="../api/">API Reference</a></b>
p <b><a href="../api">API参考手册</a></b> p <b><a href="../api/">API参考手册</a></b>
td td
:marked :marked
Authoritative details about each member of the Angular libraries. Authoritative details about each of the Angular libraries.
关于Angular库中每一个成员的详尽、权威的资料。 关于Angular库中每一个成员的详尽、权威的资料。
tr(style=top) tr(style=top)
td td
p <b><a href="../tutorial">Tutorial</a></b> p <b><a href="../tutorial/">Tutorial</a></b>
p <b><a href="../tutorial">教程</a></b> p <b><a href="../tutorial/">教程</a></b>
td td
:marked :marked
A step-by-step, immersive approach to learning Angular that A step-by-step, immersive approach to learning Angular that
@ -81,8 +94,8 @@ table(width="100%")
深入分析Angular的特性和开发实践。 深入分析Angular的特性和开发实践。
tr(style=top) tr(style=top)
td td
p <b><a href="../cookbook">Cookbook</a></b> p <b><a href="../cookbook/">Cookbook</a></b>
p <b><a href="../cookbook">烹饪宝典</a></b> p <b><a href="../cookbook/">烹饪宝典</a></b>
td td
:marked :marked
Recipes for specific application challenges, mostly code snippets with a minimum of exposition. Recipes for specific application challenges, mostly code snippets with a minimum of exposition.
@ -90,112 +103,67 @@ table(width="100%")
一组解决实际应用中某些特定挑战的“菜谱”,大部分是代码片段随带少量的详细阐述。 一组解决实际应用中某些特定挑战的“菜谱”,大部分是代码片段随带少量的详细阐述。
:marked :marked
## Learning path
## 学习路径
You don't have to read the guide straight through. Most pages stand on their own.
我们不需要从头到尾依次阅读本指南。大部分页面都是独立的。
For those new to Angular, the recommended learning path runs through the *Guide* section:
对于Angular新手建议的学习路径是走完“指南”区
1. For the big picture, read the [Architecture](architecture.html) overview.
1. 要了解全景图,请阅读[架构](architecture.html)概览。
1. Try [QuickStart](../quickstart.html). QuickStart is the "Hello, World" of Angular.
It shows you how to set up the libraries and tools you'll need to write *any* Angular app.
1. 试用[“快速起步”](../quickstart.html)。“快速起步”是Angular的“Hello, World”。
它会告诉我们如何设置*任何*Angular应用程序都会用到的库和工具。
1. Take the *Tour of Heroes* [tutorial](../tutorial), which picks up where QuickStart leaves off,
and builds a simple data-driven app. The app demonstrates the essential characteristics of a professional application:
a sensible project structure, data binding, master/detail, services, dependency injection, navigation, and remote data access.
1. 学习*英雄指南*[教程](../tutorial) ,它将从“快速起步”出发,最终构建出一个简单的数据驱动的应用。
它虽简单,但也具有我们写一个专业应用时所需的一切基本特性:
实用的项目结构、数据绑定、主从视图、服务、依赖注入、导航,以及远程数据访问。
1. [Displaying Data](displaying-data.html) explains how to display information on the screen.
1. [显示数据](displaying-data.html)解释了如何把信息显示到屏幕上。
1. [User Input](user-input.html) covers how Angular responds to user behavior.
1. [用户输入](user-input.html)解释了Angular如何响应用户行为。
1. [Forms](forms.html) handles user data entry and validation within the UI.
1. [表单](forms.html)用来在UI中处理用户输入的数据并进行有效性验证。
1. [Dependency Injection](dependency-injection.html) is the way to build large, maintainable applications
from small, single-purpose parts.
1. [依赖注入](dependency-injection.html)是我们把小型、单一用途的部件组装成大型、可维护的应用的方法。
1. [Template Syntax](template-syntax.html) is a comprehensive study of Angular template HTML.
1. [模板语法](template-syntax.html)是对Angular模板HTML的全面讲解。
After reading the above sections, you can skip to any other pages on this site.
读完这些,你就可以跳到本网站的任意页面去阅读了。
## Code samples
## 代码范例
Each page includes code snippets that you can reuse in your applications.
These snippets are excerpts from a sample application that accompanies the page.
每个页面都包含一些能在我们自己的应用中复用的代码片段。
这些片段来自于相应页面中附带的范例应用。
Look for a link to a running version of that sample near the top of each page,
such as this <live-example name="architecture"></live-example> from the [Architecture](architecture.html) page.
在每页靠近顶部的地方都可以看到一个链接,指向这个范例的可执行版本,比如[架构](architecture.html)一章中的<live-example name="architecture"></live-example>。
<p if-docs="ts">
The link launches a browser-based code editor where you can inspect, modify, save, and download the code.
</p>
<p if-docs="ts">
这个链接启动一个基于浏览器的代码编辑器,在这里,我们可以查看、修改、保存和下载这些代码。
</p>
A few early pages are written as tutorials and are clearly marked as such. A few early pages are written as tutorials and are clearly marked as such.
The rest of the pages highlight key points in code rather than explain each step necessary to build the sample. The rest of the pages highlight key points in code rather than explain each step necessary to build the sample.
You can always get the full source through the #{_liveLink}. You can always get the full source through the #{_liveLink}s.
少量早期页面是作为教程来写的,并被清晰的标注出来。 少量早期页面是作为教程来写的,并被清晰的标注出来。
其它页面的目的是展示代码中的关键点,而不是解释构建这个范例所需的每一个步骤。 其它页面的目的是展示代码中的关键点,而不是解释构建这个范例所需的每一个步骤。
我们可以通过在线例子的链接找到完整的源代码。 我们可以通过在线例子的链接找到完整的源代码。
## Code samples
## 代码例子
Each page includes code snippets from a sample application that accompanies the page.
You can reuse these snippets in your applications.
每章都包含了该章附带例子应用的代码片段。你可以在你的应用中再利用这些代码片段。
Look for a link to a running version of that sample, often near the top of the page,
such as this <live-example name="architecture"></live-example> from the [Architecture](architecture.html) page.
<span if-docs="ts">
The link launches a browser-based, code editor where you can inspect, modify, save, and download the code.
</span>
查找这些例子在线版本的链接,通常在页面的开头部分附近。
例如[架构](architecture.html)章的<live-example name="architecture">在线例子</live-example>。
<span if-docs="ts">
该链接启动浏览器代码编辑器,你可以查看、修改、保存和下载代码。
</span>
## Reference pages ## Reference pages
## 参考资料 ## 参考资料
- The [Cheat Sheet](cheatsheet.html) lists Angular syntax for common scenarios. * The [Cheat Sheet](cheatsheet.html) lists Angular syntax for common scenarios.
- [速查表](cheatsheet.html)列出了Angular在常见场景下的语法。 * [速查表](cheatsheet.html)列出了Angular在常见场景下的语法。
- The [Glossary](glossary.html) defines terms that Angular developers should know. * The [Glossary](glossary.html) defines terms that Angular developers should know.
- [词汇表](glossary.html)定义了Angular开发者需要知道的术语。 * [词汇表](glossary.html)定义了Angular开发者需要知道的术语。
- The [API Reference](../api/) is the authority on every public-facing member of the Angular libraries. <li if-docs="ts">The [Change Log](change-log.html) announces what's new and changed in the documentation.</li>
- [API参考手册](../api/)是关于Angular库中每一个公有成员的权威参考资料。 <li if-docs="ts">The [变更日志](change-log.html) 宣布文档新增与更新内容</li>
## We welcome feedback! * The [API Reference](../api/) is the authority on every public-facing member of the Angular libraries.
## 我们期待您的反馈!
- Use the [website GitHub repo](!{_ngDocRepoURL}) for **documentation** issues and pull requests. * [API参考手册](../api/)是关于Angular库中每一个公有成员的权威参考资料。
- 到[angular.io Github库](https://github.com/angular/angular.io)提交**文档相关**的issues和pull requests. ## Feedback
- Use the [Angular GitHub repo](!{_ngRepoURL}) to report issues with **Angular** itself. ## 反馈
- 到[Angular Github库](https://github.com/angular/angular)报告与**Angular本身**有关的issues。 We welcome feedback!
我们期待您的反馈!
* Use the <a href="!{_ngDocRepoURL}" target="_blank" title="angular docs on github">!{_angular_io} Github repository</a> for **documentation** issues and pull requests.
* 到<a href="!{_ngDocRepoURL}" target="_blank" title="angular docs on github">!{_angular_io} Github 库</a>提交**文档相关**的issues和pull requests.
* Use the <a href="!{_ngRepoURL}" target="_blank" title="angular source on github">Angular Github repository</a> to report issues with **Angular** itself.
- 到<a href="!{_ngRepoURL}" target="_blank" title="angular source on github">Angular Github 库</a>报告与**Angular本身**有关的issues。

View File

@ -0,0 +1,74 @@
block includes
include ../_util-fns
figure
img(src="/resources/images/devguide/intro/people.png" width="200px" height="152px" alt="Us" align="left" style="margin-left:-40px;margin-right:10px" )
:marked
Everyone learns differently.
You don't have to read the documentation straight through. Most pages stand on their own.
Those new to Angular may wish to follow this popular learning path.
<br class="l-clear-left">
每个人的学习方式不一样。
你不一定要通读文档。本站大部分页面都是独立的。
Angular 初学者可能希望跟随下面流行的学习路径。
<br class="l-clear-left">
1. [Setup](setup.html "Setup locally withe Quickstart seed") for local Angular development, if you haven't already done so.
如果你还没有这样做,[搭建](setup.html "利用《快速起步》种子搭建本地开发环境")本地 Angular 开发环境。
1. Take the [*Tour of Heroes* tutorial](../tutorial "Tour of Heroes").
学习[*英雄指南*教程](../tutorial "Tour of Heroes")。
The *Tour of Heroes* takes you step-by-step from [setup](setup.html)
to a full-featured example that demonstrates the essential characteristics of a professional application:
a sensible project structure, data binding, master/detail, services, dependency injection, navigation, and remote data access.
*英雄指南*让你一步一步的从[搭建本地开发环境](setup.html)开始,到开发一个完整特征的例子程序,它包含了专业应用最基本的特征:
合理的项目结构、数据绑定、主从视图、服务、依赖注入、导航和远程数据访问等。
1. <a id="architecture"></a>Read the [Architecture](architecture.html) overview for the big picture.
<a id="architecture"></a>阅读[架构](architecture.html)了解大局。
1. [The Root Module](appmodule.html) introduces the `NgModule` class that tells Angular how to compile and run your application.
[根模块](appmodule.html)介绍了`NgModule`类,它为 Angular 描述如何编译和运行应用。
1. [Displaying Data](displaying-data.html) shows how data binding puts component property values on screen.
[显示数据](displaying-data.html)展示了数据绑定如何将组件属性值显示到屏幕上。
1. [User Input](user-input.html) explains how to respond to user-initiated DOM events.
[用户输入](user-input.html)解释了如何处理用户触发的DOM事件。
1. [Forms](forms.html) covers data entry and validation within the UI.
[表单](forms.html)涵盖了 UI 界面中数据的输入和验证。
1. [Dependency Injection](dependency-injection.html) is the way to build large, maintainable applications
from small, single-purpose parts.
[依赖注入](dependency-injection.html)是用小型和单用途部件来构建大型、易维护应用的途径。
1. [Template Syntax](template-syntax.html) is a comprehensive study of Angular template HTML.
[模板语法](template-syntax.html)对 Angular 模板HTML进行了完整的阐述。
After reading the above sections, feel free to skip around among the other pages on this site.
阅读上面这些页面以后,你可以随意在本站的其他页面中挑选阅读。
.l-sub-section
:marked
### Next Step
### 下一步
Try the [tutorial](../tutorial "Tour of Heroes") if you're ready to start coding or
visit the [Architecture](architecture.html "Basic Concepts") page if you prefer to learn the basic concepts first.
如果你想开始编程,那就试试[教程](../tutorial "Tour of Heroes")。或者如果你想要先学习基础概念,那么阅读[架构](architecture.html "基本概念")页面。

View File

@ -24,6 +24,7 @@ figure
A directive has the same set of lifecycle hooks, minus the hooks that are specific to component content and views. A directive has the same set of lifecycle hooks, minus the hooks that are specific to component content and views.
除了那些组件内容和视图相关的钩子外,指令有相同生命周期钩子。 除了那些组件内容和视图相关的钩子外,指令有相同生命周期钩子。
<br class="l-clear-both"> <br class="l-clear-both">
## Table of Contents ## Table of Contents
@ -865,7 +866,7 @@ figure.image-display
:marked :marked
### AfterContent hooks ### AfterContent hooks
### AfterContent钩子 ### AfterContent钩子
*AfterContent* hooks are similar to the *AfterView* hooks. *AfterContent* hooks are similar to the *AfterView* hooks.
The key difference is in the child component The key difference is in the child component
*AfterContent*钩子和*AfterView*相似。关键的不同点是子组件的类型不同。 *AfterContent*钩子和*AfterView*相似。关键的不同点是子组件的类型不同。

File diff suppressed because it is too large Load Diff

View File

@ -30,12 +30,12 @@ include ../_util-fns
:marked :marked
We recommend a comprehensive starter-set of packages as specified in the `dependencies` and `devDependencies` We recommend a comprehensive starter-set of packages as specified in the `dependencies` and `devDependencies`
sections of the QuickStart sections of the <a href="https://docs.npmjs.com/files/package.json" target="_blank">package.json</a> file
<a href="https://docs.npmjs.com/files/package.json" target="_blank">package.json</a> file: installed as described during [Setup](setup.html).
我们在“快速起步”一章中<a href="https://docs.npmjs.com/files/package.json" target="_blank">package.json</a>文件的 我们在[搭建本地开发环境](setup.html)一章中安装并解释的<a href="https://docs.npmjs.com/files/package.json" target="_blank">package.json</a>文件的
`dependencies`和`devDependencies`区中指定了一组适用于新手的综合依赖包。 `dependencies`和`devDependencies`区中指定了一组适用于新手的综合依赖包。
+makeJson('quickstart/ts/package.1.json',{ paths: 'dependencies, devDependencies'}, 'package.json (dependencies)')(format=".")
:marked :marked
You can use other packages but we recommend *this particular set* to start with because (a) they work well together and You can use other packages but we recommend *this particular set* to start with because (a) they work well together and
(b) they include everything you'll need to build and run the sample applications in this series. (b) they include everything you'll need to build and run the sample applications in this series.
@ -243,7 +243,7 @@ a(id="dev-dependencies")
列在`package.json`文件中*devDependencies*区的包会帮助我们开发该应用程序。 列在`package.json`文件中*devDependencies*区的包会帮助我们开发该应用程序。
我们不用把它们部署到产品环境的应用程序中 —— 虽然这样做也没什么坏处。 我们不用把它们部署到产品环境的应用程序中 —— 虽然这样做也没什么坏处。
***[concurrently](https://www.npmjs.com/package/concurrently)*** - ***[concurrently](https://www.npmjs.com/package/concurrently)*** -
A utility to run multiple *npm* commands concurrently on OS/X, Windows, and Linux operating systems. A utility to run multiple *npm* commands concurrently on OS/X, Windows, and Linux operating systems.
***[concurrently](https://www.npmjs.com/package/concurrently)*** - 一个用来在OS/X、Windows和Linux操作系统上同时运行多个*npm*命令的工具 ***[concurrently](https://www.npmjs.com/package/concurrently)*** - 一个用来在OS/X、Windows和Linux操作系统上同时运行多个*npm*命令的工具

View File

@ -1634,8 +1634,7 @@ a#hero-detail-ctor
provide them for the `id` parameter by name and tell the `HeroService` to fetch the hero with that `id`. provide them for the `id` parameter by name and tell the `HeroService` to fetch the hero with that `id`.
然后,在`ngOnInit`方法中,我们用`ActivatedRoute`服务来接收本路由的参数。 然后,在`ngOnInit`方法中,我们用`ActivatedRoute`服务来接收本路由的参数。
由于这些参数是作为`Observable`可观察对象提供的所以我们_订阅`subscribe`_它们通过名字引用`id`参数,并告诉`HeroService`获取指定`id`的英雄。 由于这些参数是作为`Observable`可观察对象提供的所以我们用_`switchMap`来通过名字引用`id`参数,并告诉`HeroService`获取指定`id`的英雄。
我们还要保存这个`Subscription`(订阅的返回值)的引用,供后面做清理工作。
+makeExcerpt('app/heroes/hero-detail.component.ts (ngOnInit)', 'ngOnInit') +makeExcerpt('app/heroes/hero-detail.component.ts (ngOnInit)', 'ngOnInit')

View File

@ -243,7 +243,6 @@ a#HeroListComponent
### The *HeroListComponent* class ### The *HeroListComponent* class
### *HeroListComponent*类 ### *HeroListComponent*类
Here's the component class: Here's the component class:
下面是这个组件类: 下面是这个组件类:
@ -371,7 +370,6 @@ a#HeroService
## RxJS library ## RxJS library
## RxJS库 ## RxJS库
[RxJS](https://github.com/ReactiveX/RxJS) ("Reactive Extensions") is a 3rd party library, endorsed by Angular, [RxJS](https://github.com/ReactiveX/RxJS) ("Reactive Extensions") is a 3rd party library, endorsed by Angular,
that implements the [*asynchronous observable*](https://www.youtube.com/watch?v=UHI0AzD_WfY "Rob Wormald on observables") pattern. that implements the [*asynchronous observable*](https://www.youtube.com/watch?v=UHI0AzD_WfY "Rob Wormald on observables") pattern.
@ -1144,7 +1142,8 @@ a#in-mem-web-api
如果我们只关心获取到的数据我们可以告诉Angular从一个`heroes.json`文件中获取英雄列表,就像这样: 如果我们只关心获取到的数据我们可以告诉Angular从一个`heroes.json`文件中获取英雄列表,就像这样:
+makeJson('server-communication/ts/app/heroes.json', null, 'app/heroes.json')(format=".") - var _heroesJsonPath = (_docsFor == 'dart' ? 'web' : 'app') + '/heroes.json';
+makeJson('server-communication/' + _docsFor + '/' + _heroesJsonPath, null, _heroesJsonPath)(format=".")
.l-sub-section .l-sub-section
:marked :marked
You wrap the heroes array in an object with a `data` property for the same reason that a data server does: You wrap the heroes array in an object with a `data` property for the same reason that a data server does:

View File

@ -0,0 +1,243 @@
block includes
include ../_util-fns
:marked
The documentation [setup](setup.html) procedures install a _lot_ of files,
Most of them can be safely ignored.
在[搭建](setup.html)本地开发环境的过程中会安装*很多*文件。它们大部分都可以被忽略掉。
Application files _inside the_ **`app/`** and **`e2e/`** folders matter most to developers.
对程序员来讲最重要的是在 *`app/`* 和 *`e2e/`* 文件夹*之内*的应用文件。
Files _outside_ those folders condition the development environment.
They rarely change and you may never view or modify them.
If you do, this page can help you understand their purpose.
在这两个文件夹*之外*的文件为开发环境设定条件。
这些文件很少会需要变动,你可能永远都不需要阅览或者修改它们。
style td, th {vertical-align: top}
table(width="100%")
col(width="10%")
col(width="90%")
tr
th
:marked
File
文件
th
:marked
Purpose
用途
tr
td <code>app/...</code>
td
:marked
Your Angular application files.
你的 Angular 应用文件。
Ships with the "Hello Angular" sample's
`AppComponent`, `AppModule`, a component unit test (`app.component.spec.ts`), and
the bootstrap file, `main.ts`.
"Hello Angular" 这个例子中有 `AppComponent`、`AppModule`、 一个组件单元测试 (`app.component.spec.ts`) 以及引导文件 `main.ts`。
tr
td <code>e2e/...</code>
td
:marked
_End-to-end_ (e2e) tests of your application,
written in Jasmine and run by the
<a href="http://www.protractortest.org/" target="_blank" title="Protractor: end-to-end testing for Angular">protractor</a>
e2e test runner.
应用的*端对端*(e2e)测试,用 Jasmine 写成并用 <a href="http://www.protractortest.org/" target="_blank" title="Protractor:Angular 的端对端测试">protractor</a> 端对端测试运行器测试。
Initialized with an e2e test for the "Hello Angular" sample.
初始化后,有个"Hello Angular" 的例子的端对端测试。
tr
td <code>node_modules/...</code>
td
:marked
The _npm_ packages installed with the `npm install` command.
用 `npm install` 命令安装的 *npm* 包。
tr
td
code.
.editorconfig<br>
.git/...<br>
.gitignore<br>
.travis.yml
td
:marked
Tooling configuration files and folders.
配置文件和文件夹的工具。
Ignore them until you have a compelling reason to do otherwise.
除非非常必要,否则可以忽略。
tr
td <code>CHANGELOG.md</code>
td
:marked
The history of changes to the _QuickStart_ repository.
*快速起步*库的更新历史。
Delete or ignore.
删除或忽略。
tr
td <code>favicon.ico</code>
td
:marked
The application icon that appears in the browser tab.
出现在浏览器标签上的应用图标。
tr
td <code>index.html</code>
td
:marked
The application host page.
It loads a few essential scripts in a prescribed order.
Then it boots the application, placing the root `AppComponent`
in the custom `<my-app>` body tag.
应用的宿主页面。
它以特定的顺序加载一些基本脚本。
然后它启动应用,将根`AppComponent`放置到自定义`<my-app>`标签里。
The same `index.html` satisfies all documentation application samples.
同一个 `index.html`满足所有文档应用例子。
tr
td <code>karma.conf.js</code>
td
:marked
Configuration for the <a href="https://karma-runner.github.io/1.0/index.html" target="_blank" title="Karma unit test runner">karma</a>
test runner described in the [Testing](testing.html) guide.
在[测试](testing.html)指南中提到的 <a href="https://karma-runner.github.io/1.0/index.html" target="_blank" title="Karma测试运行器">karma</a> 测试运行器的配置。
tr
td <code>karma-test-shim.js</code>
td
:marked
Script to run <a href="https://karma-runner.github.io/1.0/index.html" target="_blank" title="Karma unit test runner">karma</a>
with SystemJS as described in the [Testing](testing.html) guide.
在[测试](testing.html)指南中提到的 <a href="https://karma-runner.github.io/1.0/index.html" target="_blank" title="Karma测试运行器">karma</a> 测试运行器的脚本。
tr
td <code>LICENSE</code>
td
:marked
The open source MIT license to use this setup code in your application.
应用的搭建代码中用到的开源 MIT 许可证。
tr
td <code>package.json</code>
td
:marked
Identifies `npm `package dependencies for the project.
为项目指定`npm`依赖包。
Contains command scripts for running the application,
running tests, and more. Enter `npm run` for a listing.
<a href="https://github.com/angular/quickstart/blob/master/README.md#npm-scripts"
target="_blank" title="npm scripts for Angular documentation samples">Read more</a> about them.
包含了一些命令脚本,用来运行应用、运行测试与其他。输入`npm run`来查看命令列表。
到<a href="https://github.com/angular/quickstart/blob/master/README.md#npm-scripts"
target="_blank" title="Angular 文档例子的 npm 脚本">这里</a>阅读更多关于它们的说明。
tr
td <code>protractor.config.js</code>
td
:marked
Configuration for the
<a href="http://www.protractortest.org/" target="_blank" title="Protractor: end-to-end testing for Angular">protractor</a>
_end-to-end_ (e2e) test runner.
<a href="http://www.protractortest.org/" target="_blank" title="Protractor: Angular 的端对端测试">protractor</a> *端对端* (e2e) 测试器运行器的配置。
tr
td <code>README.md</code>
td
:marked
Instruction for using this git repository in your project.
Worth reading before deleting.
项目中使用这个 git 库的说明。
在删除前值得阅读。
tr
td <code>styles.css</code>
td
:marked
Global styles for the application. Initialized with an `<h1>` style for the QuickStart demo.
应用的全局样式。初始化后,有个为《快速起步》演示准备的`<h1>`样式。
tr
td <code>systemjs<br>.config.js</code>
td
:marked
Tells the **SystemJS** module loader where to find modules
referenced in JavaScript `import` statements such as
为 **SystemJS** 模块加载器指定去哪儿查找在 JavaScript 的`import`语句中引用的模块。
code-example(language="ts").
import { Component } from '@angular/core;
:marked
Don't touch this file unless you are fully versed in SystemJS configuration.
除非你完全理解 SystemJS 的配置,不要修改它。
tr
td <code>systemjs<br>.config.extras.js</code>
td
:marked
Optional extra SystemJS configuration.
A way to add SystemJS mappings, such as for appliation _barrels_,
without changing the original `system.config.js`.
可选的额外 SystemJS 配置。
是添加 SystemJS 映射的途径,例如在无需修改原始`systemjs.config.js`的情况下为应用映射*封装桶*。
tr
td <code>tsconfig.json</code>
td
:marked
Tells the TypeScript compiler how to transpile TypeScript source files
into JavaScript files that run in all modern browsers.
为 TypeScript 编译器指定如何将 TypeScript 代码转换为 JavaScript 文件,用来在所有现代浏览器中运行。
tr
td <code>tslint.json</code>
td
:marked
The `npm` installed TypeScript linter inspects your TypeScript code
and complains when you violate one of its rules.
利用`npm`安装的 TypeScript 语法检查器 (linter) 检测 TypeScript 代码并在你违反它的规则时提示你。
This file defines linting rules favored by the
[Angular style guide](style-guide.html) and by the authors of the documentation.
该文件定义了 [Angular 风格指南](style-guide.html)与本文档站作者喜爱的语法检查规则。

View File

@ -0,0 +1,295 @@
block includes
include ../_util-fns
- var _prereq = 'node and npm';
- var _playground = 'playground';
- var _Install = 'Install';
//- npm commands
- var _install = 'install';
- var _start = 'start';
a#develop-locally
:marked
## Setup a local development environment
## 搭建本地开发环境
<span if-docs="ts">
The <live-example name=quickstart>QuickStart live-coding</live-example> example is an Angular _playground_.
It's not where you'd develop a real application.
You [should develop locally](#why-locally "Why develop locally") on your own machine ... and that's also how we think you should learn Angular.
<live-example name=quickstart>《快速起步》在线编程</live-example>例子是 Angular 的*游乐场*。
它不是你开发的真实应用的地方。
你应该在自己的电脑上[本地开发](#why-locally "为什么在本地开发?")... 你也应该在本地环境学习 Angular。
</span>
Setting up a new project on your machine is quick and easy with the **QuickStart seed**,
maintained [on github](!{_qsRepo} "Install the github QuickStart repo").
利用 [github 上](!{_qsRepo} "安装 github 《快速起步》库")的**快速起步种子**在你的电脑上搭建一个新项目是很快很容易的。
Make sure you have [!{_prereq} installed](#install-prerequisites "What if you don't have !{_prereq}?").
Then ...
确定你已经安装了[!{_prereq}](#install-prerequisites "如果你没有 !{_prereq} "),然后:
1. Create a project folder (you can call it `quickstart` and rename it later).
新建项目目录(你可以暂时为其取名`quickstart`,以后再重命名)。
1. [Clone](#clone "Clone it from github") or [download](#download "download it from github") the **QuickStart seed** into your project folder.
[克隆](#clone "从 github 克隆")或者[下载](#download "从 github 下载")**《快速起步》种子**到你的项目目录。
1. !{_Install} [!{_npm}](#install-prerequisites "What if you don't have !{_prereq}?") packages.
安装[!{_npm}](#install-prerequisites "如果你没有 !{_prereq} ") 包。
1. Run `!{_npm} !{_start}` to launch the sample application.
运行`!{_npm} !{_start}`来启动例子应用。
a#clone
:marked
### Clone
### 克隆
Perform the _clone-to-launch_ steps with these terminal commands.
运行下列命令来执行*克隆并启动*步骤。
code-example(language="sh" class="code-shell").
git clone !{_qsRepo}.git quickstart
cd quickstart
!{_npm} !{_install}
!{_npm} !{_start}
a#download
:marked
### Download
### 下载
<a href="!{_qsRepoZip}" title="Download the QuickStart seed repository">Download the QuickStart seed</a>
and unzip it into your project folder. Then perform the remaining steps with these terminal commands.
<a href="!{_qsRepoZip}" title="下载《快速起步》种子库">下载《快速起步》种子</a>
并解压到你的项目目录中。然后执行下面的命令完成剩余步骤。
code-example(language="sh" class="code-shell").
cd quickstart
!{_npm} !{_install}
!{_npm} !{_start}
.l-main-section#seed
:marked
## What's in the QuickStart seed?
## 《快速起步》种子库里都有什么?
block qs-seed
:marked
The **QuickStart seed** contains the same application as the QuickStart playground
and even has <live-example>its own _playground_</live-example>
that accomodates development of richer examples in a live coding environment.
**《快速起步》种子** 包含了与《快速起步》游乐场一样的应用,并且它还有<live-example>自己的*游乐场*</live-example>
用来在在线编程的环境里开发更丰富例子。
But it's true purpose is to provide a solid foundation for _local_ development.
Consequently, there are _many more files_ in the project folder on your machine,
most of which you can [learn about later](setup-systemjs-anatomy.html "Setup Anatomy").
但是,它真实的目的是提供坚实的*本地*开发基础。
所以你的电脑里的项目目录里面有*更多文件*,参见[搭建剖析](setup-systemjs-anatomy.html "Setup Anatomy")。
block core-files
:marked
Focus on the following three TypeScript (`.ts`) files in the **`/app`** folder.
注意**`/app`**目录里的下列三个 TypeScript (`.ts`) 文件:
.filetree
.file app
.children
.file app.component.ts
.file app.module.ts
.file main.ts
+makeTabs(`
setup/ts/app/app.component.ts,
setup/ts/app/app.module.ts,
setup/ts/app/main.ts
`, '', `
app/app.component.ts,
app/app.module.ts,
app/main.ts
`)(format='.')
:marked
All guides and cookbooks have _at least these core files_. Each file has a distinct purpose and evolves independently as the application grows.
所有指南和烹饪书都有*至少这几个核心文件*。每个文件都有独特的用途,并且随着应用的成长各自独立演变。
style td, th {vertical-align: top}
table(width="100%")
col(width="20%")
col(width="80%")
tr
th
:marked
File
文件
th
:marked
Purpose
用途
tr
td <ngio-ex>app.component.ts</ngio-ex>
td
:marked
Defines the same `AppComponent` as the one in the QuickStart !{_playground}.
It is the **root** component of what will become a tree of nested components
as the application evolves.
定义与《快速起步》游乐场同样的`AppComponent`。
它是**根**组件,随着应用的演变,它将变成一颗嵌套组件树。
tr(if-docs="ts")
td <code>app.module.ts</code>
td
:marked
Defines `AppModule`, the [root module](appmodule.html "AppModule: the root module") that tells Angular how to assemble the application.
Right now it declares only the `AppComponent`.
Soon there will be more components to declare.
定义`AppModule`[根模块](appmodule.html "AppModule: 根模块")为 Angular 描述如何组装应用。
目前,它只声明了`AppComponent`。
不久,它将声明更多组件。
tr
td <ngio-ex>main.ts</ngio-ex>
td
:marked
Compiles the application with the [JiT compiler](../glossary.html#jit)
and [bootstraps](appmodule.html#main "bootstrap the application") the application to run in the browser.
That's a reasonable choice for the development of most projects and
it's the only viable choice for a sample running in a _live-coding_ environment like Plunker.
You'll learn about alternative compiling and deployment options later in the documentation.
使[即时 (JiT) 编译器](../glossary.html#jit)用编译应用并且在浏览器中[启动](appmodule.html#main "启动应用")并运行应用。
对于大多数项目的开发,这都是合理的选择。而且它是在像 Plunker 这样的*在线编程*环境中运行例子的唯一选择。
你将在本文档中学习其他编译和开发选择。
.l-sub-section
:marked
### Next Step
### 下一步
If you're new to Angular, we recommend staying on the [learning path](learning-angular.html).
如果你是 Angular 初学者,建议根据[学习路径](learning-angular.html)学习。
br
br
a#install-prerequisites
.l-main-section
:marked
## Appendix: !{_prereq}
## 附录node 和 npm
block install-tooling
:marked
Node.js and npm are essential to modern web development with Angular and other platforms.
Node powers client development and build tools.
The _npm_ package manager, itself a _node_ application, installs JavaScript libraries.
Node.js 和 npm 对使用 Angular 和其他平台进行现代网络开发的至关重要。
Node 驱动客户端开发和构建工具。
*npm* 包管理器本身是*node*应用,用于安装 JavaScript 库。
<a href="https://docs.npmjs.com/getting-started/installing-node" target="_blank" title="Installing Node.js and updating npm">
Get them now</a> if they're not already installed on your machine.
如果你的电脑没有安装它们,<a href="https://docs.npmjs.com/getting-started/installing-node" target="_blank" title="安装 Node.js 和更新 npm">
立刻安装它们</a>。
**Verify that you are running node `v4.x.x` or higher and npm `3.x.x` or higher**
by running the commands `node -v` and `npm -v` in a terminal/console window.
Older versions produce errors.
在终端/控制台窗口运行命令`node -v`和`npm -v`,来**确认你运行的 node 是`v4.x.x`或更高npm 为`3.x.x`或更高。**。
老版本产品会产生错误。
We recommend [nvm](https://github.com/creationix/nvm) for managing multiple versions of node and npm.
You may need [nvm](https://github.com/creationix/nvm) if you already have projects running on your machine that
use other versions of node and npm.
我们推荐 [nvm](https://github.com/creationix/nvm) 来管理多版本 node 和 npm。
如果你的电脑上已经有使用其他版本 node 和 npm 的项目,你可能需要[nvm](https://github.com/creationix/nvm)。
+ifDocsFor('ts')
a#why-locally
.l-main-section
:marked
## Appendix: Why develop locally
## 附录:为何在本地开发
<live-example>Live coding</live-example> in the browser is a great way to explore Angular.
在浏览器中<live-example>在线编程</live-example>是很好的探索 Angular 的方法。
Links on almost every documentation page open completed samples in the browser.
You can play with the sample code, share your changes with friends, and download and run the code on your own machine.
几乎每章文档里面的链接都在浏览器中打开完整的例子。
你可以用这些代码做实验,或者与朋友共享你的修改,或者下载并在你自己的电脑上运行这些代码。
The [QuickStart](../quickstart.html "Angular QuickStart Playground") shows just the `AppComponent` file.
It creates the equivalent of `app.module.ts` and `main.ts` internally _for the playground only_.
so the reader can discover Angular without distraction.
The other samples are based on the QuickStart seed.
[快速起步](../quickstart.html "Angular 快速起步游乐场")仅仅展示了`AppComponent`文件。
它在内部创建了只为*游乐场*而准备的等价`app.module.ts`和`main.ts`。
所以读者可以在零干扰的情况下探索 Angular。
其他例子是基于 《快速起步》种子的。
As much fun as this is ...
虽然有这么多的乐趣,但是...
* you can't ship your app in plunker
* 你不能在 plunker 里面发布你的应用
* you aren't always online when writing code
* 编程时你不可能总是在线
* transpiling TypeScript in the browser is slow
* 在浏览器中编译 TypeScript 很慢
* the type support, refactoring, and code completion only work in your local IDE
* 只有本地IDE有类型支持、代码重构和代码自动完成
Use the <live-example><i>live coding</i></live-example> environment as a _playground_,
a place to try the documentation samples and experiment on your own.
It's the perfect place to reproduce a bug when you want to
<a href="https://github.com/angular/angular.io/issues/new" target="_blank" title="File a documentation issue">file a documentation issue</a> or
<a href="https://github.com/angular/angular/issues/new" target="_blank" title="File an Angular issue">file an issue with Angular itself</a>.
把<live-example><i>在线编程</i></live-example>环境当做*游乐场*,一个尝试文档例子和自己做实验的地方。
当你想要<a href="https://github.com/angular/angular.io/issues/new" target="_blank" title="提交关于文档的问题">提交关于文档的问题</a>或者
<a href="https://github.com/angular/angular/issues/new" target="_blank" title="提交关于 Angular 的问题">提交关于 Angular 自身的问题</a>时,
它是重现错误的完美地方。
For real development, we strongly recommend [developing locally](#develop-locally).
对于现实项目开发,我们强烈推荐在[本地开发](#develop-locally)。

Some files were not shown because too many files have changed in this diff Show More