LE WEB DE DEMAIN ! C’EST DÉJÀ AUJOURD’HUI
Angular 2
Meetup N°2
24/09/2016
#ngMorocco
#ngMorocco
data:image/s3,"s3://crabby-images/59cba/59cba40b930e4f805e7207b385b52d3eaf7a30be" alt=""
Ouadie Lahdioui
Consultant IT
lahdiouiouadie
ouadie-lahdioui
Pour vous servir ...
Regardez la présentation en temps réel : https://slides.com/ouadielahdioui/angular2/live
data:image/s3,"s3://crabby-images/2d788/2d78804b69dfbc5e94c55e2dff35ae3634d66ffa" alt=""
data:image/s3,"s3://crabby-images/035c4/035c400dfefaf79c039ee6ba48f998266bd7d09b" alt=""
lahdiouiouadie
data:image/s3,"s3://crabby-images/a05ec/a05ec5c297344ac1b2e45eeb4bef628a5cc167b1" alt=""
Bataille de pouces géante
data:image/s3,"s3://crabby-images/97bbe/97bbe9ea8ca0fcebc97a5d52852f921633206237" alt=""
Ce qu'on va voir !
Angular Universal
Module Loaders
Material design
Composantes
NativeScript
Angular CLI
TypeScript
Services
Zone.JS
Augury
Mobile
Pipes
Ce qu'on NE va pas voir !
Functional Reactive Programming
i18n (internationalization)
Change detection
WebWorkers
Observables
Animations
Routage
Tests
RxJS
data:image/s3,"s3://crabby-images/84c25/84c256ce1a103aa683ada175682419899dbe9c0f" alt=""
Angular 2 has finally been released !
has finally been released!
Wake up
Avez-vous déjà utilisé Angular ?
Avez-vous un retour d'expérience
Sur Angular 1.x ?
data:image/s3,"s3://crabby-images/00690/006908dc72f26e1bb03bd083e78e535508de4b70" alt=""
Angular 1.x, Piqûre de rappel
Model-View-Whatever (a.k.a MV* )
Injection de dépendance
Liaison bidirectionnelle
Expressions
Directives
Services
Module
Filtres
Scope
...
Mais c'est quoi le problème avec Angular 1.x ?
l'injection de dépendance
function MyController($scope, $window) {}
someModule.controller(‘MyController’, [‘$scope’, function($scope) { }]);
MyController.$inject = [‘$scope’, ‘$window’];
Pour faire la minification !
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
Débogage
<div ng-repeat=”phone in user.phones”> {{ phone }} </div>
Impossible de faire un points d'arrêt dans {{ }}
C'est quoi la source du problème ?
data:image/s3,"s3://crabby-images/c66d2/c66d2ba6e5680fb622b0886e6a0d6f9a30622875" alt=""
C'est quoi le résultat si "user" est undefined ?
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
Le dirty checking
data:image/s3,"s3://crabby-images/2ebef/2ebef41f4e73b697f22ec273e6075f95a3df1d99" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
Le support d'une limite théorique de 2000 bindings/page web
Et la dégradation des performances
Héritage des scopes
<div ng-init="phone = 11"> {{ phone }} </div>
<div ng-if="true">
<button ng-click=”phone = 22">change phone</button>
</div>
<div ng-init="phone = {value: 11}"> {{ phone.value }} </div>
<div ng-if="true">
<button ng-click=”phone.value = 22">change phone</button>
</div>
C'est quoi la résultat si je clique sur "change phone" ?
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
Cas 1 :
Cas 2 :
Est la source de la plupart des problèmes
Search engine optimization
http://monsite.com/#/clients/edit
http://monsite.com/#/clients/send/2
Le moteur de recherche ne sait pas que ce sont deux pages différentes :
Pas d'évalution JS dans les robot d'indexation :
data:image/s3,"s3://crabby-images/bce85/bce85830ec02f3720a18fb49906135f0b7adba1d" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
La complexité des DIRECTIVES
Trasnclude, Compile, Link, Controller ...
data:image/s3,"s3://crabby-images/66894/668941018a616062c71c79419155092ea13feec2" alt=""
data:image/s3,"s3://crabby-images/de048/de04817de0a5294c4c01ccf448a29f89405f60a1" alt=""
data:image/s3,"s3://crabby-images/c5076/c5076306d7b7f93a2c76f3bb85f76d94688f416a" alt=""
Tous les chemins mènent à Rome
You have a factory, wich is a service, and you have a service wich is a service, both have providers, and when you write a factory as your service, you actually write a provider wich returns a factory wich is basically A SERVICE
data:image/s3,"s3://crabby-images/11221/11221d1fa5014bb241105f2edc456f8a5e313e06" alt=""
- Extrait de la session de Shai Reznik à ngConf
data:image/s3,"s3://crabby-images/58a2a/58a2a0f0458a51f174a17aabb4ea4713649ff6cc" alt=""
Pourquoi Angular 2 ?
Né pour résoudre une problèmatique
ONE TEAM + CORE COMPETENCE + ONE CODE BASE
=
LOTS OF STUFF
Multi platformes
data:image/s3,"s3://crabby-images/35ea7/35ea7f4182e1e9eac3d39ca7625ae047a5d9a637" alt=""
WEB
MOBILE WEB
MOBILE natif
android - ios - windows
desktop
windows - mac - linux
data:image/s3,"s3://crabby-images/dcffe/dcffea9d433f34c4b13758e461993ee3268872f9" alt=""
HTTP 2
data:image/s3,"s3://crabby-images/d3010/d301094deb9efdd031a683de0baa6df56c6e03bc" alt=""
data:image/s3,"s3://crabby-images/00c5f/00c5f8a49db6ee5e9471d7e8acf2dcce21617454" alt=""
VM Turns
Push Notification
Zones
DI
intellisense
CLI
Débogage
Putting the pieces together
Server rendering
Isomorphique
Mobile
le fruit d'un effort communautaire
data:image/s3,"s3://crabby-images/5c432/5c432e1b63338c0438a252fc3ac111fcdd30ca8a" alt=""
data:image/s3,"s3://crabby-images/c710a/c710a60227b8491b48cc1ab6ca1a5c67ecade1b3" alt=""
data:image/s3,"s3://crabby-images/54818/5481827eab4d0eb94ebf25dc80a665030f923f6a" alt=""
data:image/s3,"s3://crabby-images/f58df/f58dfbe9c6ec40407bc3644153bbba2c3fb33f86" alt=""
data:image/s3,"s3://crabby-images/fc145/fc145dcbe97f79ca8208a57052e6c08fe406f8ca" alt=""
data:image/s3,"s3://crabby-images/58a03/58a034db76927857731c1d2042e71213808805a0" alt=""
data:image/s3,"s3://crabby-images/9ab12/9ab12215db9f80bfefe6e28f2d86873b6b8fe29f" alt=""
data:image/s3,"s3://crabby-images/17f6f/17f6f909884f459057869d8464d78f9f3116de09" alt=""
data:image/s3,"s3://crabby-images/84e27/84e27a8ffd2f595225ab2e54ffa9e83b32bae492" alt=""
data:image/s3,"s3://crabby-images/253e8/253e8966a366948e8bc9650dd4acf5a0d29d1bb8" alt=""
... D'un framwork à une plateforme
data:image/s3,"s3://crabby-images/3adbc/3adbcc4f4c2a423577ddf79c4902bcaa480ff4ae" alt=""
data:image/s3,"s3://crabby-images/3327f/3327fc6da7a8a28d85e899ea4616a8f4304734c4" alt=""
Angular 1.x
Angular 2
Angular 2, le renouveau
data:image/s3,"s3://crabby-images/edb24/edb2493f00fc4c76c31f6503e7a23a29e9c1a149" alt=""
Popularité d'Angular selon libscore.com
data:image/s3,"s3://crabby-images/b1b87/b1b874ae7e7f2e46f7834ea932e59b11b814924b" alt=""
Evolution de l'intérêt de recherche
Feedback sur Stack Overflow
data:image/s3,"s3://crabby-images/ae894/ae8946729df0ab52b221e3e825ac3efa85323a4d" alt=""
data:image/s3,"s3://crabby-images/e1dbb/e1dbb74c9dc1f3e2ee77c087d22992a195439a8e" alt=""
Source : Keynote ng-conf 2016
Taille de la communauté
RIP
Il était une fois dans angular 1 ...
data:image/s3,"s3://crabby-images/13c9c/13c9cfaca6f0ec73e3151603d813bef23c89adda" alt=""
PREMIÈRE IMPRESSION DE LA COMMUNAUTÉ
WHAT ! Et mes web apps Angular 1.x ?
data:image/s3,"s3://crabby-images/81a42/81a422ca6c712a3efb49eabb2da91f95e3a31225" alt=""
data:image/s3,"s3://crabby-images/72703/72703523a43c1cd592392020e0af04e320b4a2eb" alt=""
data:image/s3,"s3://crabby-images/c889c/c889c4c38782191b78e762aaa0fddf7985f7338f" alt=""
data:image/s3,"s3://crabby-images/bbccf/bbccf2b429ac1070351cba2e9e9bfcc00915f1aa" alt=""
data:image/s3,"s3://crabby-images/80881/808817a0034583d6c6cba5ada05f7a89603c72ab" alt=""
data:image/s3,"s3://crabby-images/16889/1688942c8b92dd41db910816742936ca0344d193" alt=""
-Brad Green, Engineering director at Google and AngularJS team manager
Angular 1 will continue to receive bugfix and security patch support for 18-24 months after the release of version 2.0.
data:image/s3,"s3://crabby-images/e64e4/e64e4247b830ebd6b64e1d05df8462e106e858df" alt=""
data:image/s3,"s3://crabby-images/11221/11221d1fa5014bb241105f2edc456f8a5e313e06" alt=""
data:image/s3,"s3://crabby-images/5044a/5044a10c4e43ac841ab9b82164c70561cfecd0b2" alt=""
Concepts-clés
TypeScript
data:image/s3,"s3://crabby-images/26dc4/26dc40c33b5c1d113f86308350af928b66c45e94" alt=""
Source code : *.ts
Source code : *.js
Transpiling
interface Pokemon {
name: string;
size: string;
}
POO - typage statique - ANNOTATIONS
class Player {
fullName: string;
constructor(name: string, pokemon: pokemon) {
this.name = name;
this.pokemons.push(pokemon);
}
}
class PokemonBall() {
open() {...}
close() {...}
}
var sacha = new Player("Sacha", pikachu);
data:image/s3,"s3://crabby-images/25bba/25bbabd5be0e5078885d73ac9683099a9648c3c3" alt=""
data:image/s3,"s3://crabby-images/12f49/12f4900aac5c6592dad02c823928496e9c2f2761" alt=""
data:image/s3,"s3://crabby-images/1c37e/1c37ead7792450b9527228f18fefeedcd9b628dc" alt=""
data:image/s3,"s3://crabby-images/460c5/460c585864e181b9e6421ee9ca1219d04ee649a4" alt=""
vue globale de l'architecture angular 2
Modularité
Modularité
- Minimiser les risques d'interaction involontaire avec l'extérieur
- Rendre les bibliothèques :
- Plus faciles à maintenir
- Mieux structurées
function getPlayer(){
return this.player;
}
function getBall(){
return this.ball;
}
module.exports = {
player: getPlayer,
ball: getBall
};
var pokemon = require('./pokemon.js');
console.log(pokemon.player());
console.log(pokemon.ball());
Exporter
Modularité avant l'ES 6
define(
['pokemonPlayer','pokemonBall'],
function(player, ball
) {
// Use player and ball
return myFunction;
});
var player = require('pokemonPlayer');
var ball = require('pokemonBall');
// use player and ball
module.exports = myFunction;
Deux écoles ont vu le jour grace à la communauté
CommonJS
AMD
modularité à l'aire de l'ES 6
En fin, ce pattern d'écriture modulaire est intégré dans le langage lui-même
export const maxAge = this.maxAge;
export function player() {
return this.player;
}
export function ball() {
return this.ball;
}
import { player, ball } from 'pokemon';
console.log(player());
console.log(ball());
Exporter
modularité avec angular 2
SystemJS : Un chargeur de modules universel pour JavaScript
data:image/s3,"s3://crabby-images/8caeb/8caeb43e1746ffff1b900e6953cbd61e61e0709d" alt=""
AMD
(Asynchronous Module Definition)
CJS
(CommonJS)
REGISTER
(System.register)
ESM
(ECMAScript Module)
GLOBAL
(Global shim module format)
qu'est qu'un module peut exporter ?
data:image/s3,"s3://crabby-images/d848c/d848c68b38c867a14d66de6eb114fd8c16f8f12a" alt=""
// app/app.component.ts
export class AppComponent { }
// app/main.ts
import { AppComponent } from './app.component';
import { Directive } from '@angular/core';
@Directive({ selector: '[pokemon]' })
export class PokemonDirective { /*...*/ }
export class Logger {
log(msg: any) { /*...*/ }
error(msg: any) { /*...*/ }
}
Composantes
Le respect des nouveaux standards du web
data:image/s3,"s3://crabby-images/6ac7a/6ac7af2533ea55158a93e8e927486b94245c7486" alt=""
data:image/s3,"s3://crabby-images/61e73/61e73e8754ea5371e49669c0412ae4c7b0bb0f76" alt=""
Sans Composant
AVEC Composant
import { Component } from '@angular/core';
@Component({
selector: 'pokemon'
template: `...`
})
export class Pokemon implements OnInit {
ngOnInit() {
/* ... */
}
}
Déclaration d'un composant
data:image/s3,"s3://crabby-images/07944/07944a73fcba16727a1eaa7e4e5502f06f4b37bd" alt=""
OnChanges
OnInit
DoCheck
AfterContentInit
AfterContentChecked
AfterViewChecked
AfterViewInit
OnDestroy
Le cycle de vie d'un composant :
<ul>
<li *ngFor="let pokemon of pokemons">
<img src="/assets/pokemons/{{pokemon.id}}.png">
<p>{{pokemon.name}}</p>
<p>{{pokemon.category}}</p>
<img src="/assets/pokemons/{{pokemon.id}}.png">
<p>{{pokemon.description}}</p>
</li>
<ul>
Template d'un composant
data:image/s3,"s3://crabby-images/787af/787afe24d84d3ff9201fa794c27200590ce09d02" alt=""
arbre de composants imbriqués
Composant
principal
Composant
PokemonList
Composant
Navbar
Composant
SideBar
Composant 5
Composant
PokemonDetail
Application Angular 2
metadata d'un composant
@Component({
selector: 'pokemon-list',
templateUrl: 'app/pekemon-list.component.html',
directives: [PekemonDetailComponent],
providers: [PokemonService]
})
export class PokemonListComponent implements OnInit {
/* . . . */
}
data:image/s3,"s3://crabby-images/91e0e/91e0ed35e2bffe5ce099138793316cf43841a3fe" alt=""
liaison de données (a.k.a Data binding)
data:image/s3,"s3://crabby-images/18d5f/18d5f12e633a8b19fa45d0867cece7592bb9db8c" alt=""
<p>{{pokemon.name}}</p>
<pokemon-detail [pokemon]="selectedPokemon"></pokemon-detail>
<button (click)="selectPokemon(pokemon)"></button>
data:image/s3,"s3://crabby-images/7769a/7769a401ae45ac460ea9eb369d1905f35b3f0ca2" alt=""
data:image/s3,"s3://crabby-images/644fd/644fd78b3ff77db7c76ddf948fa267dcdf3dbb0a" alt=""
[()] BANANA IN A BOX
data:image/s3,"s3://crabby-images/8fec9/8fec92016ff84707ec3b527e4acefafd281b25a6" alt=""
43 Directives dans Angular 1 VS [()] dans Angular 2
= "value"
<input [(ngModel)]="pokemon.name">
Directives
data:image/s3,"s3://crabby-images/707c0/707c098eb2aa06238e8df13548674f10a5f98e4e" alt=""
1/ Composants :
2/ Directives structurelles :
3/ Directives attributs :
Une directive avec son sélecteur et son template
Agissent sur le DOM en ajoutant/retirant des éléments
Changent l’apparence ou le comportement d’un élément
<pokemon-detail></pokemon-detail>
<div [ngStyle]="setStyles()">
This div is italic, normal weight, and extra large (24px).
</div>
<div *ngIf="isPokemon">{{pokemon}}</div>
<div *ngFor="let pokemon of pokemons">{{pokemon}}</div>
Services
data:image/s3,"s3://crabby-images/28442/28442b79fcdc35b0a215734b98defd6bdc53ec15" alt=""
@Injectable()
export class PokemonService {
getPokemonById(id: number): Pokemon {
return this.pokemons.filter(pokemon => pokemon.id === id).pop();
}
}
PIPES
import { Pipe, PipeTransform } from '@angular/core';
@Pipe({ name: 'sample' })
export class SamplePipe implements PipeTransform {
transform(input: string): string {
return `${input} from sample pipe`;
}
}
UpperCasePipe - LowerCasePipe - CurrencyPipe - PercentPipe - DatePipe
...
- Effectuer des transformations directement dans le template :
<p>The hero's birthday is {{ birthday | date:'fullDate' | uppercase}}</p>
Injection de dépendances
data:image/s3,"s3://crabby-images/70f1c/70f1cf03434b0868bddc73a5c86ab2ef89946299" alt=""
constructor(private service: PokemonService) { }
data:image/s3,"s3://crabby-images/34b67/34b6711174036d12caf40c475011e0d58b29af03" alt=""
PROVIDERS pour guider l'injector
import { Component } from '@angular/core';
@Component({
selector: 'home',
providers: [ PokemonService ]
})
export class HomeComponent {}
@NgModule({
declarations: [AppComponent, PokemonListComponent],
providers: [MdIconRegistry],
bootstrap: [AppComponent]
})
export class AppModule { }
Déclaration globale d'un provider au moment du bootstrap :
Déclaration locale d'un provider dans le composant :
L'injector crée un service à partir d'un objet moule : Provider
ZONE.JS
Monkey-patch des fonctions asynchrones
Zone.JS
- Similaire aux domains dans NodeJS et ThreadLocal dans Java
- l'implémentation des Zones inspirée par "Dart language"
A Zone is an execution context that persists across async tasks.
data:image/s3,"s3://crabby-images/11221/11221d1fa5014bb241105f2edc456f8a5e313e06" alt=""
Async JavaScript
start()
showBulbasaur()
setTimeout(showPikachu, 0);
setTimeout(showCharmander, 0);
showSnorlax()
stop()
Temps
Temps
showCharmander
Queue :
Et si on veut savoir le temps d'execution ?
showBulbasaur()
setTimeout(showPikachu, 0);
setTimeout(showCharmander, 0);
showSnorlax()
Temps
showPikachu
data:image/s3,"s3://crabby-images/ce755/ce755849394ce31b54a4bf088d48a7e3e4353814" alt=""
data:image/s3,"s3://crabby-images/07944/07944a73fcba16727a1eaa7e4e5502f06f4b37bd" alt=""
data:image/s3,"s3://crabby-images/ed3a8/ed3a8d67bd48f8525e6d6defa801276f592a01bd" alt=""
data:image/s3,"s3://crabby-images/10a58/10a58052a36a287ce26d076976d412a0159c7724" alt=""
(async)
(async)
Temps
data:image/s3,"s3://crabby-images/ce755/ce755849394ce31b54a4bf088d48a7e3e4353814" alt=""
data:image/s3,"s3://crabby-images/07944/07944a73fcba16727a1eaa7e4e5502f06f4b37bd" alt=""
data:image/s3,"s3://crabby-images/ed3a8/ed3a8d67bd48f8525e6d6defa801276f592a01bd" alt=""
data:image/s3,"s3://crabby-images/10a58/10a58052a36a287ce26d076976d412a0159c7724" alt=""
(exclu)
(exclu)
ASYNC JavaScript avec Zone.js
Temps
showBulbasaur()
showSnorlax()
showPikachu()
showCharmander()
start()
stop()
start()
stop()
start()
stop()
showBulbasaur()
showSnorlax()
showPikachu()
showCharmander()
onZoneEnter()
onZoneEnter()
onZoneEnter()
onZoneLeave()
onZoneLeave()
onZoneLeave()
- Exécute du code avant et après
- Testing / Timing / Gestion des erreurs / ...
- Un jour, ça vas devenir un standard (Ecma TC39)
ZONE.JS
Zone.current.fork({}).run(function () {
Zone.current.inTheZone = true;
setTimeout(function () {
console.log('in the zone: ' + !!Zone.current.inTheZone);
}, 0);
});
console.log('in the zone: ' + !!Zone.current.inTheZone);
/* OutPut */
'in the zone: false'
'in the zone: true'
RENDERS
Renderers
- Remplaçable selon l'environnement
- Précompilation (taille de l'app + temps de bootstrap)
data:image/s3,"s3://crabby-images/cebce/cebce6a290548097eb58339545b0ce49cface648" alt=""
Server side render
Angular universal
Angular Mobile Toolkit
Progressive web apps :
Instant loading
Offline
Installable
Notifications
App natives
data:image/s3,"s3://crabby-images/90c05/90c050ad0b525f760a546ef2f34ced4b34fc55e0" alt=""
data:image/s3,"s3://crabby-images/81cef/81cef3ce412908aee065538a771df9a14dbef9e3" alt=""
data:image/s3,"s3://crabby-images/3adbc/3adbcc4f4c2a423577ddf79c4902bcaa480ff4ae" alt=""
data:image/s3,"s3://crabby-images/d1889/d1889252610c3b31676558b6fe05c15b1245f6cf" alt=""
data:image/s3,"s3://crabby-images/81cef/81cef3ce412908aee065538a771df9a14dbef9e3" alt=""
NativeScript
- Un framework JavaScript opensource pour créer des applications mobiles natives (pas de WebView)
- Basé sur une machine virtuelle JS (eg: JavaScriptCore, V8, ...)
var myAlert = new UIAlertView();
myAlert.message = "NativeScript rocks!";
myAlert.show();
data:image/s3,"s3://crabby-images/8c787/8c787587c3556897b1f513e9bf13fd8683acd4e5" alt=""
data:image/s3,"s3://crabby-images/2eb6e/2eb6e7cf3efab7cf96ce25f1b4a831b967e25882" alt=""
data:image/s3,"s3://crabby-images/86c1b/86c1b68de887541ed8056d991be6e39588892a4e" alt=""
qu'est ce que je peux faire avec ?
MATERIAL DESIGN
Le web design selon Google
Material design
Angular material 1
Angular material 2
data:image/s3,"s3://crabby-images/c19c9/c19c93dd1b11d3e446e81bcd2064a5be8bee9514" alt=""
Angular-CLI
Angular Augury
data:image/s3,"s3://crabby-images/06e0e/06e0e3c883611fc1facd4a46fe26f70c92677e1c" alt=""
Style Guide
data:image/s3,"s3://crabby-images/8b7a0/8b7a085c602072fe4f4886d74bbd694dd22bb703" alt=""
Officiel et communautaire
Migration
Angular 1.x -----------> Angular 2
data:image/s3,"s3://crabby-images/f984f/f984f56b6ccb73650d1964b5e9b4647134f5d3ca" alt=""
It's really hard for us to build a bridge to a city that doesn't exist yet.
- Brad Green
data:image/s3,"s3://crabby-images/11221/11221d1fa5014bb241105f2edc456f8a5e313e06" alt=""
Solution N° 1
sudo rm -rf yourAngularProject
data:image/s3,"s3://crabby-images/db413/db4131e7dd5f6b066d9801531782b969bd8a374b" alt=""
commencez par :
- Lire le Styleguide
- Adopter une approche composants
- Utiliser TypeScript
- Utiliser un module loader
- Utiliser un adapteur
data:image/s3,"s3://crabby-images/07c58/07c5897f78d66fc34e2ce21d5424eb29bd0e4d39" alt=""
data:image/s3,"s3://crabby-images/75446/754463322b32cde8ac9c7dc0b17ec26d7caf2ae4" alt=""
data:image/s3,"s3://crabby-images/51a0e/51a0ee27eaca5c15871bb914a6c3cfea653771b7" alt=""
data:image/s3,"s3://crabby-images/1c37e/1c37ead7792450b9527228f18fefeedcd9b628dc" alt=""
Déja en PRODUCTION
Capital One
data:image/s3,"s3://crabby-images/340dd/340dd645612bc3ab82580c3be769685ade673705" alt=""
Lucidchart
data:image/s3,"s3://crabby-images/e9717/e971771941818a6c0b9134eddb81c155a8158ffb" alt=""
Kiva
data:image/s3,"s3://crabby-images/32451/32451ae8619c8d1922c6bdba7b2298eb04071c91" alt=""
Fidelity
data:image/s3,"s3://crabby-images/18c07/18c07a35e0389be155f11e12bee3541f4169989f" alt=""
... we must find a fine balance between supporting and improving the existing Angular, while building the best Angular yet. ... This is why I need you to step up ! Help each other. Help us triage issues and review pull requests. Answer questions on the mailing list, stackoverflow etc.
- Igor Minar, Lead on the Angular project
data:image/s3,"s3://crabby-images/11221/11221d1fa5014bb241105f2edc456f8a5e313e06" alt=""
data:image/s3,"s3://crabby-images/bab45/bab45dedd3d7f50363dd8037099c3a3af025c3b7" alt=""
STEP UP WITH US
POUR ALLER PLUS LOIN
- Le site officiel Angular.io
- Awesome Angular 2 - Angular Class
- Getting started with Angular 2 - Wassim Chegham
- Creating Cross-Platform Apps with Angular 2 @ Microsoft Build - Brad Green
- Angular 2 and the future of HTML5 apps @ Fluent conf - Brad Green
N'oubliez pas de partager avec moi vos feedbacks sur Angular 2 !
lahdiouiouadie
ouadie-lahdioui
Merci !
data:image/s3,"s3://crabby-images/2d788/2d78804b69dfbc5e94c55e2dff35ae3634d66ffa" alt=""
data:image/s3,"s3://crabby-images/035c4/035c400dfefaf79c039ee6ba48f998266bd7d09b" alt=""
lahdiouiouadie
data:image/s3,"s3://crabby-images/a05ec/a05ec5c297344ac1b2e45eeb4bef628a5cc167b1" alt=""
Angular 2
By Ouadie LAHDIOUI
Angular 2
Le web de demain ! c'est déjà aujourd'hui
- 1,781