The First Step to
Angular 2
data:image/s3,"s3://crabby-images/b7fb1/b7fb1385b1f65ae1c355771750d1004bcd72b3ee" alt=""
@laco0416
2016/9/3 HTML5 Conference
about me
data:image/s3,"s3://crabby-images/f6753/f67538189a55aa4676fe6fd945f695a0ab466364" alt=""
- TOPGATE, Inc.
- Angular 2 Contributor
- Angular2 Info
ng2-info.github.io
data:image/s3,"s3://crabby-images/ff0ee/ff0ee4618bcc6933e38e77defae1ea929858bc8e" alt=""
data:image/s3,"s3://crabby-images/aa3d8/aa3d8e50a7b6004f3be41c4fcf61abcd59edea40" alt=""
Developing an Angular 2 Edge
http://shop.oreilly.com/product/9781939902375.do
data:image/s3,"s3://crabby-images/f6b93/f6b93be319faaf2f5ac25a8edc30f5d251d34e9b" alt=""
Agenda
- Angular 2 Overview
- Final APIs
- Angular 2 Router
Angular 2
- Web Standards
- Simple
- Fast
- Full-stack
- Powered by
data:image/s3,"s3://crabby-images/1a423/1a42390a6f85455b72741c2754c23d49acdf611a" alt=""
data:image/s3,"s3://crabby-images/8362c/8362c61f068e7b4c013f195c0a186f25d3a9a4c9" alt=""
Web Standards Friendly
- ECMAScript 2015
- Web Worker
- Decorators
- Web Animations API
- Web Components
- Fetch API
- Observables
- Zones
Component definition
@Component({
selector: "app-profile",
template: "<p>Name: {{ name }}</p>"
})
class ProfileComponent {
@Input() name: string;
}
<div>
<app-profile name="laco"></app-profile>
</div>
data:image/s3,"s3://crabby-images/f588a/f588a2eba5fdb3e2772a8ac8543570f2ab24c3c8" alt=""
Template Syntax (1)
Data-Binding: [prop]
@Component({
template: `
<input [value]="name">
<div [attr.role]="role"></div>
<div [class.large]="isLarge"></div>
<div [style.width.px]="mySize"></div>
`
})
class MyComponent {
name = "laco";
role = "main"
isLarge = true;
mySize = 16;
}
Template Syntax (2)
Event Handling: (event)
@Component({
template: `
<button (click)="onClick($event)">
`
})
class MyComponent {
onClick(e: MouseEvent) {
}
}
High Performance
- Smart Change Detector
- One-way Checking
- 100000 checks / 10ms
data:image/s3,"s3://crabby-images/ce9ca/ce9ca29aa61516859037d93523a350d54222d860" alt=""
data:image/s3,"s3://crabby-images/bac35/bac353551a4b8598c2883997d340566b480899ed" alt=""
Official Packages
-
@angular/core
-
@angular/forms
-
@angular/http
-
@angular/router
-
Testing
-
Angular Material
-
Angular Universal: Server side
-
AngularFire: Firebase
TypeScript
- Static Type
- Decorators
- Type-base Dependency Injection
data:image/s3,"s3://crabby-images/60745/607455e6628a574ea46c87d30a61128f2f0bba76" alt=""
Static Types
Types as API documents
-
Always correct
-
Compiler check
-
Auto-completions
data:image/s3,"s3://crabby-images/b565c/b565c793c45bc643924c2c1938bd5960cb356633" alt=""
@Decorators()
Next generation standardSpec: javascript-decorators
Status: tc39/proposals
- Class and Property: Stage 2
- Function: Stage 0
- Method: Stage 0
tsc --experimentalDecorators
Type-base DI
IOW, "Typo-safe" DIGood-bye "$http"
import {Http} from "@angular/http";
@Injectable()
class MyService {
constructor(private anyNameForHttp: Http) {
}
}
data:image/s3,"s3://crabby-images/724ac/724acafce0a3ab4c87ece2aa6426203b06c69a30" alt=""
Angular 2 Final APIs
Basic APIs
-
@Component()
-
@Input() / @Output()
-
@Directive()
-
@Pipe()
-
@NgModule()
@Component
It's a component because it composes app
-
Component = Directive + Template
-
CSS Encapsulation
-
Build Component Tree
import {Component} from '@angular/core';
@Component({
selector: 'my-app',
template: `
<h1>{{title}}</h1>
`,
styles: [`h1 { color: blue; }`]
})
export class MyAppComponent {
title = 'Hello World!';
}
Demo
Component Tree
Top-down Data Flow: @Input()
@Component({
selector: 'my-app',
template: `<profile [name]="myName"></profile>`
})
class MyAppComponent {
myName = 'laco';
}
@Component({
selector: 'profile',
template: `<p>{{name}}</p>`
})
class ProfileComponent {
@Input() name;
}
Component Tree
Bottom-up Event System: @Output()
@Component({
selector: 'my-app',
template: `<app-menu (select)="onMenuSelect($event)"></app-menu>`
})
class MyAppComponent {
onMenuSelect(menuItem) { }
}
@Component({
selector: 'app-menu',
template: `...`
})
class MenuComponent {
@Output() select = new EventEmitter();
selectMenu(menuItem) { this.select.emit(menuItem); }
}
@Directive
Extend elements
import {Directive, HostBinding} from '@angular/core';
@Directive({
selector: '[bsBtnPrimary]'
})
class BootstrapBtn {
@HostBinding('class') classNames = "btn btn-primary";
}
<button bsBtnPrimary>Primary Button</button>
<button class="btn btn-primary">Primary Button</button>
@Pipe
Data transformer
import {Pipe} from '@angular/core';
@Pipe({
name: 'quote'
})
class QuotePipe {
transform(value) {
return `"${value}"`;
}
}
<p>{{ message | quote }}</p>
Angular 2 Architecture
Module
Components
Application
Platform
Directives
Pipes
Angular Modules
Application Configuration
-
Import other modules
-
Create Compiler
-
Setup Injector
-
bootstrapping
import {NgModule} from '@angular/core';
import {BrowserModule} from '@angular/platform-browser';
import {MyAppComponent} from './app.component';
import {MyOtherComponent} from './other.component';
import {MyService} from './my.service';
@NgModule({
imports: [BrowserModule],
declarations: [MyOtherComponent, MyAppComponent],
providers: [MyService],
bootstrap: [MyAppComponent]
})
export class MyAppModule {}
Angular Platforms
Application Environment
-
Browser (JIT, AoT)
-
Web Worker
-
Server
import {
platformBrowser
} from '@angular/platform-browser';
import {
platformBrowserDynamic
} from '@angular/platform-browser-dynamic';
import {
platformWorkerApp, platformWorkerUi
} from '@angular/platform-webworker';
import {
platformServer, platformServerDynamic
} from '@angular/platform-server';
import {MyAppModule} from './app.module';
platformBrowserDynamic().bootstrapModule(MyAppModule);
Angular 2 Router
a long time ago...
data:image/s3,"s3://crabby-images/f39bd/f39bd9363c88265e49c674202810375ce8d01a86" alt=""
data:image/s3,"s3://crabby-images/dcc78/dcc78b7dc997e42bbe44a66508177d9cbf199889" alt=""
data:image/s3,"s3://crabby-images/4c42f/4c42fa33ec6fa1fab33f046ff753aad148e0c949" alt=""
2015 ng-conf: New Router
2015 AngularConnect: Component Router
2016 ng-conf Router v2
data:image/s3,"s3://crabby-images/54c2f/54c2fb6f8b6f614c8688935b5ff53e4a4ad5b2fa" alt=""
data:image/s3,"s3://crabby-images/1ab28/1ab280c676a1292b02785da5a5a71847d12c2560" alt=""
data:image/s3,"s3://crabby-images/49122/49122924c24257c271561a90975f6431c8630bd1" alt=""
data:image/s3,"s3://crabby-images/aa3bb/aa3bba710fe1b2278d33decfca939edccf5e6962" alt=""
data:image/s3,"s3://crabby-images/09ee9/09ee9970457628f302103be266601fde077a837a" alt=""
Router v3
data:image/s3,"s3://crabby-images/91e15/91e150ac7286f26e2b41f300db51b0ea303dfa84" alt=""
Route Configuration
path-component patterns
export const appRoutes = [
// .../
{ path: '', pathMatch: 'full', component: HomeCmp },
// .../about
{ path: 'about', component: AboutCmp },
{
// nested routes
path: 'users', children: [
// .../users
{ path: '', pathMatch: 'full', component: UserListCmp },
// .../users/11
{ path: ':id', component: UserDetailCmp }
]
},
// wildcard route
{ path: '**', redirectTo: '/'}
];
Install Router
import RouterModule
import {RouterModule} from '@angular/router';
import {appRoutes} from './app.routes';
@NgModule({
imports: [RouterModule.forRoot(appRoutes)],
})
class MyAppModule {}
Use Router
<router-outlet> & routerLink
@Component({
template: `
<nav>
<a routerLink="/">Home</a> |
<a routerLink="/about">About</a>
</nav>
<main>
<router-outlet></router-outlet>
</main>
`
})
export class MyAppComponent {}
Router v3 Architecture
data:image/s3,"s3://crabby-images/072d9/072d9176ca276087aea12119fb4821778e21e2fe" alt=""
Router State
ActivatedRoute 💖 Observable
@Component({
template: `
<nav>...</nav>
<main>
<router-outlet></router-outlet>
</main>
`
})
export class MyAppComponent {
constructor(private route: ActivatedRoute) {
route.url.subscribe(url => {
console.log(`URL changed: ${url}`);
});
}
}
Router State
Async / Sync
// URL Changes
route.url.subscribe(...);
// Params (/users/:id)
route.params.subscribe(...);
// QueryParams (?foo=bar)
route.queryParams.subscribe(...);
// Fragment (#section1)
route.fragment.subscribe(...);
// Sync Snapshot
route.snapshot.params['id'];
Demo
Conclusion
-
Angular 2 💖 Web Standards
-
Angular 2 💖 TypeScript
-
Simple & Useful APIs
-
Platform/Module/Component Tree
-
Router v3: Easy
Thanks!
data:image/s3,"s3://crabby-images/fd8c2/fd8c247c3505026d4cf35d6de7516295444e5cd4" alt=""
The First Step to Angular 2
By Suguru Inatomi
The First Step to Angular 2
- 7,371