The First Step to Angular
data:image/s3,"s3://crabby-images/b7fb1/b7fb1385b1f65ae1c355771750d1004bcd72b3ee" alt=""
@laco2net
2017-09-24 HTML5 Conference
about me
data:image/s3,"s3://crabby-images/59ad1/59ad186996b13c7d47429c6a8a59048a3c0528c4" alt=""
- 稲富 駿 (laco)
- Kaizen Platform
- ng-japan Organizer
- Angular Contributor
Why is Angular popular?
Angular Overview
Angular
- Web Standards
- Simple
- Easy
- Fast
- Full-stack
- Powered by
data:image/s3,"s3://crabby-images/1a423/1a42390a6f85455b72741c2754c23d49acdf611a" alt=""
data:image/s3,"s3://crabby-images/8362c/8362c61f068e7b4c013f195c0a186f25d3a9a4c9" alt=""
Angular Web Standards
- ECMAScript 2015+
- ES Modules
- Web Animations API
- Web Components
-
ES Proposal
- Decorators
- Observables
- Zones
data:image/s3,"s3://crabby-images/166f9/166f9aedadeeea52b9ca874be1bfb0d862ab0c37" alt=""
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) {
}
}
Angular CLI
# Install Angular CLI
> npm install -g @angular/cli
# Create new app
> ng new hello-angular
# Test
> ng test
# Build
> ng build --prod
# Debug
> ng serve
Easy to develop Angular apps
High Performance
- Smart Change Detector
- Code Generation
- 100k checks / 10ms
- Reduce detection
- Triggered by Zone.js
- Code Generation
- Smart Renderer
- Reduce DOM manipulations
data:image/s3,"s3://crabby-images/ce9ca/ce9ca29aa61516859037d93523a350d54222d860" alt=""
Official Packages
-
@angular/core
-
@angular/common
-
@angular/forms
-
@angular/http
-
@angular/router
-
@angular/material
-
@angular/platform-server
-
@angular/service-worker
-
@angular/language-service
-
...
TypeScript
-
Static Type
-
Decorators
-
Type-base Dependency Injection
-
Language Services
data:image/s3,"s3://crabby-images/60745/607455e6628a574ea46c87d30a61128f2f0bba76" alt=""
Static Types
-
Benefit of Type
-
Code as Living API doc
-
Always correct
-
-
-
Compiler check
-
Compilation is the easiest test
-
-
Auto-completions
-
Language Services Helps your editor
-
data:image/s3,"s3://crabby-images/b565c/b565c793c45bc643924c2c1938bd5960cb356633" alt=""
@Decorators()
Next generation standardSpec: javascript-decorators
Status: tc39/proposals
- Class and Property: Stage 2
- Method: Stage 0
tsc --experimentalDecorators
Type-based DI
"Typo-safe" DI
Good-bye "$http"
import {Http} from "@angular/http";
@Injectable()
class MyService {
constructor(private anyNameForHttp: Http) {
}
}
data:image/s3,"s3://crabby-images/724ac/724acafce0a3ab4c87ece2aa6426203b06c69a30" alt=""
Language Service
Types in the template
data:image/s3,"s3://crabby-images/e0441/e04415c3859378d36842c0470daeb874f582b203" alt=""
data:image/s3,"s3://crabby-images/0d87a/0d87aa8d691872f863e761b962524f7eddbd88d4" alt=""
Angular Basic APIs
Basic APIs
-
@Component()
-
@Input() / @Output()
-
@Directive()
-
@Pipe()
-
@NgModule()
@Component
It's a component because it composes app
import {Component} from '@angular/core';
@Component({
selector: 'my-app',
template: `
<h1>{{title}}</h1>
`,
styles: [`h1 { color: blue; }`]
})
export class MyAppComponent {
title = 'Hello World!';
}
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: '[underline]'
})
class BootstrapBtn {
@HostBinding('style') hostStyle = "text-decoration: underline";
}
<span underline>Text</span>
<span style="text-decoration: underline">Text</span>
@Pipe
Data transformer
import {Pipe} from '@angular/core';
@Pipe({
name: 'quote'
})
class QuotePipe {
transform(value) {
return `"${value}"`;
}
}
<p>{{ message | quote }}</p>
Angular Architecture
Module
Components
Application
Platform
Directives
Pipes
Module
Module
Module
Services
Angular Modules
Application Configuration
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
import {
platformBrowser
} from '@angular/platform-browser';
import {
platformBrowserDynamic
} from '@angular/platform-browser-dynamic';
import {
platformServer
} from '@angular/platform-server';
import {MyAppModule} from './app.module';
platformBrowserDynamic().bootstrapModule(MyAppModule);
Angular Ecosystem
Angular is a Platform
data:image/s3,"s3://crabby-images/f60db/f60db7764ace7b5bcd2f8bb800c1b659c30e6681" alt=""
data:image/s3,"s3://crabby-images/b8ae2/b8ae231c20284990df506819f9b73d660f6a492c" alt=""
What is Angular's job?
data:image/s3,"s3://crabby-images/fd68d/fd68d1d813aa0f3675c4caea60480cdad339d1bb" alt=""
Responsibility
-
Render View
-
Watch model's change
-
Transit View
-
Access to Server
Render Views
Data Model
View
Rendering
Data Model
View
@Component({
template: "<h1>{{title}}</h1>"
})
class MyComponent {
title = "Hello!";
}
<h1>Hello!</h1>
Watch model's change
Change Detection
data:image/s3,"s3://crabby-images/8dbb7/8dbb78d9ab308229a1eee45e0336ceb8df92cdf0" alt=""
data:image/s3,"s3://crabby-images/2d980/2d980d6a71989a162b0871c9282654132914483b" alt=""
Routing
Transit View
data:image/s3,"s3://crabby-images/4fb91/4fb91815426f3fa08fe17c408377298762704d50" alt=""
data:image/s3,"s3://crabby-images/d74c6/d74c62f838e86be91cb97701e4535986de826ec8" alt=""
Angular Router
-
URL-Component Mapping
-
Guard & Resolve
-
Nesting
-
Lazy-Loading
Access to Server
AJAX
@Injectable()
export class HeroService {
constructor (private http: HttpClient) {}
getHeroes (): Observable<Hero[]> {
return this.http.get<Hero[]>("/api/heroes");
}
addHero (hero: Hero): Observable<Hero> {
return this.http.post<Hero>("/api/heroes", hero);
}
}
Http Service
Angular Http
-
Wrap XHR/JSONP
-
Return Observable<T>
-
Lazy-Execution
-
Aborting
-
Multi-Responses
-
-
Interceptor
-
Progress Event
@Injectable()
export class MyInterceptor implements HttpInterceptor {
intercept(req: HttpRequest<any>, next: HttpHandler) {
newReq = req.clone({
headers: new HttpHeaders({ 'X-AUTHOR': 'laco' })
});
return next.handle(newReq);
}
}
Http Interceptor
http.post('/upload/file', file, { reportProgress: true })
.subscribe(event => {
if (event.type === HttpEventType.UploadProgress) {
const percentDone =
Math.round(100 * event.loaded / event.total);
console.log(`File is ${percentDone}% uploaded.`);
} else if (event instanceof HttpResponse) {
console.log('File is completely uploaded!');
}
});
Http Progress
Additional Features
-
Animations
-
Server-side Rendering
-
UI Components
-
Debugger
-
Editor Plugins
-
Developer Tool
Angular Animations
@Component({
animations: [
trigger("visibility", [
state("shown", style({ opacity: 1 })),
state("hidden", style({ opacity: 0 })),
transition("* => *", animate(".5s"))
])
],
template: `
<div [@visibility]="visibility">
`
})
class MyComponent() {
visibility = 'hidden';
}
Server-side Rendering
import { renderModule } from '@angular/platform-server';
renderModule(AppServerModule, {
document: fs.readFileSync('src/index.html', 'utf8'),
url: '/'
}).then(html => {
fs.writeFileSync('dist/index.html', html);
});
@angular/platform-server
Angular Material
data:image/s3,"s3://crabby-images/11fa0/11fa0f61650864141f3ce64e6f8c8931f0f26bb4" alt=""
Angular CLI
-
Scaffold Angular project
-
Tasks
-
Generate code
-
Debug
-
Test
-
Lint
-
-
Full of Best Practices
data:image/s3,"s3://crabby-images/fe45b/fe45b1e9273d4eaefa683ea6858eb6deaa165c2e" alt=""
Angular Language Service
-
VSCode Plugin (Official)
-
WebStorm (Built-in)
data:image/s3,"s3://crabby-images/d4f87/d4f87c9898191b5fbd70ef277d0fa4328b35ea7f" alt=""
@angular/language-service
Augury
-
Chrome Extension for Angular Debugging
data:image/s3,"s3://crabby-images/346be/346be2d5f8ce9198dbbbc076e6f8180215fc77a7" alt=""
Missing Piece?
Documentation
angular.io
data:image/s3,"s3://crabby-images/7cffc/7cffc4a54499fe25338c9601024ef254c39277cb" alt=""
angular.io as a doc
- Tutorial
- Fundamentals
- Techniques
- API References
- Generated from source
- Always correct
data:image/s3,"s3://crabby-images/8362c/8362c61f068e7b4c013f195c0a186f25d3a9a4c9" alt=""
angular.io as an app
-
Built on Latest Angular
-
Progressive Web App
-
Offline
-
Add to Homescreen
-
-
Source Code
-
angular/angular: /aio
-
How to learn Angular?
Learn Angular
-
Read angular.io
-
Official documentation
-
-
Use Angular CLI
-
Follow the Best Practice
-
- ES2015+
- ES Modules
- TypeScript
- RxJS / Observables
- (webpack)
- (Component Design)
Required Knowledges
Modern Web => Angular, React, Vue, ...
"The Web has moved.
Frameworks must move"
Understand the Web
Semantic Versioning
Angular x.y.z
-
x: major
-
Update can contain breaking changes
-
-
y: minor
-
Update cannot breaking changes
-
-
z: patch
-
Only bug fix
-
Time-base Release
-
Every weak: patch update
-
Every month: minor update
-
Every 6 months: major update
data:image/s3,"s3://crabby-images/400fe/400fee546c2c6d6986d9e4c5477d4b761d2adee4" alt=""
Keep an eye
on changes
data:image/s3,"s3://crabby-images/9d5ee/9d5ee611e1bcca7494cbf377ed5aec67b724ce28" alt=""
data:image/s3,"s3://crabby-images/2bc99/2bc9950616d35d5b7e4c6839e6f1ad8d00999021" alt=""
Watch CHANGELOG
Read Release Note
data:image/s3,"s3://crabby-images/6463d/6463d708b87f2f8d69f6f686f2910542639ba41d" alt=""
Read Release Note
data:image/s3,"s3://crabby-images/f8d3d/f8d3df26e36f21f6bd38dec7db3c70d57e98d9df" alt=""
Join communities
data:image/s3,"s3://crabby-images/0bdf0/0bdf0bcf88a6c0645f95568baa8f2ee0352f0270" alt=""
Conclusion
-
Angular is a Platform for Developer Experience
-
Basic APIs are Simple & Useful
-
How to learn Angular
-
Read angular.io
-
Use Angular CLI
-
Follow the Best Practice
-
Understand the Web
-
Keep an eye on changes
-
Thanks!
data:image/s3,"s3://crabby-images/59ad1/59ad186996b13c7d47429c6a8a59048a3c0528c4" alt=""
The First Step to Angular
By Suguru Inatomi
The First Step to Angular
HTML5 Conference 2017
- 5,443