Angular 2
Core concepts
data:image/s3,"s3://crabby-images/01216/012167b9ee34e067806d0dca66c033b4de2b5a4c" alt=""
History
data:image/s3,"s3://crabby-images/0f596/0f5963ba5690c3fcd6b2456be2dd0a01dbf95d57" alt=""
- Started by Miško Hevery in 2009
- Challenged to migrate a project from GWT (3 devs × 6 months) to Angular in 2 weeks ....... and FAILED
- version 1.0 released in 2012
- current version is 1.6.1 (23/12/2016)
- $scope, $http, $resource, two-way binding
- Global API - angular.copy(), angular.equals(), -angular.bind() ...
- $broadcast, $emit, $watch, $eval, $digest, $apply...
- Directives (58 build-in)
- Services, Factories, Providers
data:image/s3,"s3://crabby-images/dd2ed/dd2ed69eaceec6d0ea9d25e797f704e19a600b93" alt=""
Angular 2 vs 1.x
data:image/s3,"s3://crabby-images/2b0a3/2b0a360a41c82c00625c471142c40e943ac8eb57" alt=""
TypeScript
data:image/s3,"s3://crabby-images/f95b1/f95b12780ca1985a4006ab5cfd0590f8d6849527" alt=""
https://www.typescriptlang.org/
- Developed and maintained by Microsoft
- Open source programming language
- Optional static typing and class-based OOP
- Anders Hejlsberg, lead architect of C# and creator of Delphi and Turbo Pascal, has worked on the development of TypeScript.
Text
Text
-The TypeScript compiler is implemented in TypeScript
data:image/s3,"s3://crabby-images/80839/80839d4e260a0e6783708dc8c79dc32d890bb87c" alt=""
TypeScript = ES6 + Types + Annotations
class Animal {
constructor(public name: string) { }
move(distanceInMeters: number = 0) {
console.log(`${this.name} moved ${distanceInMeters}m.`);
}
}
class Snake extends Animal {
constructor(name: string) { super(name); }
move(distanceInMeters = 5) {
console.log("Slithering...");
super.move(distanceInMeters);
}
}
let sam = new Snake("Sammy the Python");
sam.move();
var __extends = (this && this.__extends) || function (d, b) {
for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p];
function __() { this.constructor = d; }
d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __());
};
var Animal = (function () {
function Animal(name) {
this.name = name;
}
Animal.prototype.move = function (distanceInMeters) {
if (distanceInMeters === void 0) { distanceInMeters = 0; }
console.log(this.name + " moved " + distanceInMeters + "m.");
};
return Animal;
}());
var Snake = (function (_super) {
__extends(Snake, _super);
function Snake(name) {
return _super.call(this, name) || this;
}
Snake.prototype.move = function (distanceInMeters) {
if (distanceInMeters === void 0) { distanceInMeters = 5; }
console.log("Slithering...");
_super.prototype.move.call(this, distanceInMeters);
};
return Snake;
}(Animal));
var sam = new Snake("Sammy the Python");
sam.move();
- Component-Based
- Templates
- Reduced and simplified build-in directives
(*ngIf, *ngFor, [ngSwitch], [ngSwitchCase], ngSwitchDefault, [ngClass])
- One-way data binding (two-way is also available)
- New Dependency Injection system
- Improved Change Detection System
- Native support
Angular 2
data:image/s3,"s3://crabby-images/01216/012167b9ee34e067806d0dca66c033b4de2b5a4c" alt=""
- Framework for building client applications in TypeScript and HTML
- Consists of several libraries
(angular/core, angular/platform-browser, angular/http, angular/forms, ...)
- Angular Application =
Modules[(angularized HTML + Directives) + Services]
- Launching the application = Bootstrap the root module
// app.component.ts
import { Component } from '@angular/core';
@Component({
selector: 'my-app',
template: `
<div>
<h1>Hello {{name}}</h1>
</div>
`
})
export class AppComponent { name = 'World'; }
// app.module.ts
import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser';
@NgModule({
imports: [ BrowserModule ],
providers: [ Logger ],
declarations: [ AppComponent ],
exports: [ AppComponent ],
bootstrap: [ AppComponent ]
})
export class AppModule { }
// main.ts
import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app.module';
platformBrowserDynamic().bootstrapModule(AppModule);
// index.html
<body>
<my-app>Starting application...</my-app>
</body>
data:image/s3,"s3://crabby-images/08972/08972fda8c00c19c7f79fd09a3da653027c65b77" alt=""
data:image/s3,"s3://crabby-images/68964/68964526c154a4d1e2ab39ce68151003a031cfb1" alt=""
import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser';
@NgModule({
imports: [ BrowserModule ],
providers: [ Logger ],
declarations: [ AppComponent ],
exports: [ AppComponent ],
bootstrap: [ AppComponent ]
})
export class AppModule { }
export class HeroListComponent implements OnInit {
heroes: Hero[];
selectedHero: Hero;
constructor(private service: HeroService) { }
ngOnInit() {
this.heroes = this.service.getHeroes();
}
selectHero(hero: Hero) { this.selectedHero = hero; }
}
data:image/s3,"s3://crabby-images/73f5c/73f5cf4ccc6c906d494eaf6438bdf3b0b8edf4cc" alt=""
<h2>Hero List</h2>
<p><i>Pick a hero from the list</i></p>
<ul>
<li *ngFor="let hero of heroes" (click)="selectHero(hero)">
{{hero.name}}
</li>
</ul>
<hero-detail *ngIf="selectedHero" [hero]="selectedHero"></hero-detail>
data:image/s3,"s3://crabby-images/a2f89/a2f89b1bc7f77a8497b0b0475c1763095f17fdae" alt=""
@Component({
moduleId: module.id,
selector: 'hero-list',
templateUrl: 'hero-list.component.html',
providers: [ HeroService ]
})
export class HeroListComponent implements OnInit {
/* . . . */
}
data:image/s3,"s3://crabby-images/e0cc4/e0cc41f6f2a247bd71c31a30ae0eb2139b4aad4d" alt=""
<li *ngFor="let hero of heroes"></li>
<hero-detail *ngIf="selectedHero"></hero-detail>
data:image/s3,"s3://crabby-images/83897/83897b34bc8050a743ff3765a512ece523340138" alt=""
Structural
Attributes
// highlight.directive.ts
import { Directive, ElementRef, Input } from '@angular/core';
@Directive({ selector: '[highlight]' })
export class HighlightDirective {
constructor(el: ElementRef) {
el.nativeElement.style.backgroundColor = 'yellow';
}
}
// page.html
<p highlight>Highlight me!</p>
// app/logger.service.ts
@Injectable()
export class Logger {
log(msg: any) {
console.log(msg);
}
error(msg: any) {
console.error(msg);
}
warn(msg: any) {
console.warn(msg);
}
}
data:image/s3,"s3://crabby-images/788d4/788d4071b32a676a036867be721a0ac37f7c1e94" alt=""
// app.module.ts
@NgModule({
imports: [ ... ],
providers: [ Logger ],
exports: [ ... ],
bootstrap: [ ... ]
})
export class AppModule {
}
data:image/s3,"s3://crabby-images/25bc6/25bc66c0c1cb2ca1d85ef63dfb37f89e58a258e8" alt=""
// app/hero-list.component.ts
@Component({
selector: 'hero-list',
templateUrl: 'hero-list.component.html',
providers: [ HeroService ]
})
export class HeroListComponent {
constructor(private service: HeroService) {
}
}
data:image/s3,"s3://crabby-images/2cb69/2cb69fcd40eb28dfe03d7f027708ad60548c53d1" alt=""
// hero-list.component.html
<li>{{hero.name}}</li>
<hero-detail [hero]="selectedHero"></hero-detail>
<li (click)="selectHero(hero)"></li>
// hero-detail.component.html
<input [(ngModel)]="hero.name">
// --- understanding ngModel ---
<input [ngModel]="hero.name" (ngModelChange)="hero.name = $event">
Application state change is caused by three things:
Events - User events like click, change, input, submit, …
XMLHttpRequests - E.g. when fetching data from a remote service
Timers - setTimeout(), setInterval(), because JavaScript
ObservableWrapper.subscribe(this.zone.onTurnDone, () => {
this.zone.run(() => {
this.tick();
});
});
tick() {
// perform change detection
this.changeDetectorRefs.forEach((detector) => {
detector.detectChanges();
});
}
Change detection
data:image/s3,"s3://crabby-images/586a1/586a1638c3774ad61614b2590d0744d3f85dd553" alt=""
data:image/s3,"s3://crabby-images/36c5a/36c5afe9b188ffb567eb7f4937bf1214fd9c1fc2" alt=""
data:image/s3,"s3://crabby-images/28e3a/28e3a0087c729b8511a7eff2296854100e83c56c" alt=""
data:image/s3,"s3://crabby-images/be4b0/be4b0f5baf5449432b9c8f29bdac4779019574c8" alt=""
data:image/s3,"s3://crabby-images/aa5c4/aa5c425086e674584956d814cf7f283a7c0b6e55" alt=""
@Component({
changeDetection: ChangeDetectionStrategy.OnPush
})
export class ImmutableTodoCmp {
...
}
// todos.component.ts
type ObservableTodo = Observable<todo>;
type ObservableTodos = Observable<array>>;
@Component({
selector: 'todos',
template: `<todo *ngFor="let t of todos" todo="t"></todo>`
})
class ObservableTodosCmp {
todos:ObservableTodos;
...
}
// todo.component.ts
@Component({selector:'todo'})
class ObservableTodoCmp {
todo: ObservableTodo;
...
}
data:image/s3,"s3://crabby-images/10c5d/10c5d819865799291ab19be25b4f9f5045c68594" alt=""
data:image/s3,"s3://crabby-images/641aa/641aad87a3de68bdae254f1a4b469134acda0f6c" alt=""
data:image/s3,"s3://crabby-images/641aa/641aad87a3de68bdae254f1a4b469134acda0f6c" alt=""
data:image/s3,"s3://crabby-images/7cbf8/7cbf82bb7a294d3480e6eea880d84a97a468ec16" alt=""
Resources
https://angular.io/
http://learnangular2.com/
http://blog.ng-book.com/
https://angular.io/docs/ts/latest/guide/cheatsheet.html
https://dzone.com/articles/typed-front-end-with-angular-2
http://angular2academy.com/2017/01/08/angular-2-core-components/
https://blog.thoughtram.io/angular/2016/02/01/zones-in-angular-2.html
https://vsavkin.com/change-detection-in-angular-2-4f216b855d4c#.tigfnjcvx
https://blog.thoughtram.io/angular/2016/02/22/angular-2-change-detection-explained.html
https://egghead.io/courses/angular-2-fundamentals
http://courses.angularclass.com/p/angular-2-fundamentals
Thank you!
Angular 2. Core concepts
By Stanislav Oaserele
Angular 2. Core concepts
- 905