Angular Router

Gonzalo Ruiz de Villa @gruizdevilla

CTO @gft_es

 

Angular Madrid Meetup

https://github.com/domenic/zones

¿Qué es un Router?

The Angular Router enables navigation from one view to the next as users perform application tasks.

Principales características

  • Navegación por URLs
  • Routing flexible:
    • rutas hijas
    • rutas auxiliares
  • Guardias en la navegación
  • Lazy loading & preloading
  • Resolve

https://leanpub.com/router

Angular 2 Router

by Victor Savkin

Bootstrap

<!doctype html>
<html>
<head>
  <meta charset="utf-8">
  <title>AmRouter</title>
  <base href="/">

  <meta name="viewport" content="width=device-width, initial-scale=1">
  <link rel="icon" type="image/x-icon" href="favicon.ico">
  <link href="https://fonts.googleapis.com/icon?family=Material+Icons" rel="stylesheet">
</head>
<body>
  <app-root>Loading...</app-root>
</body>
</html>

index.html

Bootstrap

import './polyfills.ts';

import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
import { AppModule } from './app/app.module';

platformBrowserDynamic().bootstrapModule(AppModule);

main.ts

Plantillas

<md-toolbar color="accent">¡Bienvenido!</md-toolbar>
<md-card>
  <md-card-content>
    <h2 class="example-h2">Estado actual</h2>
    <md-slide-toggle
        [checked]="authService.logged"
        (change)="authService.logged=!authService.logged">
      Logado
    </md-slide-toggle>
    <!-- ... -->
  </md-card-content>
</md-card>

bienvenido/bienvenido.component.html

Componentes

import { Component, OnInit } from '@angular/core';
import { AuthService } from '../auth.service';

@Component({
  selector: 'am-bienvenido',
  templateUrl: './bienvenido.component.html',
  styleUrls: ['./bienvenido.component.css']
})
export class BienvenidoComponent implements OnInit {

  constructor(private authService: AuthService) { }

  ngOnInit() {
  }

}

bienvenido/bienvenido.component.ts

Servicios

import { Injectable } from '@angular/core';
import { Router, CanActivate} from '@angular/router';
import {Observable} from 'rxjs/Observable';
import 'rxjs/add/observable/of';

@Injectable()
export class AuthService  implements CanActivate {
  logged:boolean;
  constructor() {
    this.logged = false;
  }
  canActivate(){
    return Observable.of(this.logged);
  }
}

auth.service.ts

Servicios

import { Injectable } from '@angular/core';
import { Song } from './entities/song';
import { Album } from './entities/album';

@Injectable()
export class MusicService {

  //...

  selectedSongs(): Song[] { return playList; }

  albums(): Album[]{ return albumList; }

  songs(): Song[] { return songList; }

music.service.ts

Configuración del módulo

@NgModule({
  declarations: [
    AppComponent,
    PlayerComponent,
    SonglistComponent,
    AlbumgridComponent,
    BienvenidoComponent,
    //...
  ],
  imports: [
    BrowserModule,
    FormsModule,
    HttpModule,
    AppRoutingModule,
    MaterialModule.forRoot()
  ],
  entryComponents: [
    DeactivatedialogComponent
  ],
  providers: [AuthService, DeactivateService, MusicService],
  bootstrap: [AppComponent]
})
export class AppModule { }

app.module.ts

Setup

import { NgModule } from '@angular/core';
import { AppRoutingModule }     from './app-routing.module';
//...

@NgModule({
  declarations: [
    //...
  ],
  imports: [
    //...
    AppRoutingModule,
    //...
  ],
  entryComponents: [//... ],
  providers: [//...],
  bootstrap: [AppComponent]
})
export class AppModule { }

Incluir módulo router

RouterModule.forRoot(routes)


RouterModule.forChild(routes)

Definiendo las rutas

import { NgModule }             from '@angular/core';
import { RouterModule, Routes } from '@angular/router';
import { BienvenidoComponent }  from './bienvenido/bienvenido.component';

const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent }
];


@NgModule({
  imports: [ RouterModule.forRoot(routes) ],
  exports: [ RouterModule ]
})
export class AppRoutingModule {}

app-routing.module.ts

Pintamos en outlets

<md-sidenav-layout>

  <md-sidenav #sidenav mode="side" class="app-sidenav"></md-sidenav>

  <md-toolbar color="primary">
    Angular Router Example App
  </md-toolbar>

  <div class="app-content">
    <router-outlet></router-outlet>
  </div>

</md-sidenav-layout>

app.component.html

Rutas jerárquicas

const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent },
  { path: 'albumes',  component: AlbumgridComponent},
  { path: 'albumes/:albumTitle', component: AlbumdetailComponent}
];

app-routing.module.ts

Rutas jerárquicas (refactor)

const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent },
  { path: 'albumes',  
      children: [
        {path:'', component: AlbumgridComponent},
        {path:':albumTitle', component: AlbumdetailComponent}
      ]
  }
];

app-routing.module.ts

Navegando

Con directiva routerLink:

  • routerLink="/licencia"
  • [routerLink]="['/albumes', album.title]"
  • [routerLink]="['/',{ outlets: { 'side': ['player'] } }]"

Navegando

Programáticamente:

  • this.router.navigate(['bienvenido']);
  • this.router.navigateByUrl('/bienvenido');

routerLink

  <md-card-content>
    <h2 class="example-h2">Estado actual</h2>
    <md-slide-toggle [checked]="authService.logged" (change)="authService.logged=!authService.logged">
      Logado
    </md-slide-toggle>
    <p>Debes estar logado para acceder al listado de álbumes</p>
    <p><button md-raised-button color="primary" routerLink="/albumes">Ver Álbumes</button></p>
    <p>Puedes consultar la <a routerLink="/licencia">licencia</a></p>
    <p>Puedes ver algo <a routerLink="/lazy">lazy</a></p>
  </md-card-content>

bienvenido.component.html

routerLink


<p>
  <button md-raised-button
    [routerLinkActive]="['active']" 
    [routerLink]="['/',{ outlets: { 'side': ['player'] } }]" 
    [queryParams]="{ play: 1 }">Play</button>
  <button md-raised-button
    [routerLinkActive]="['active']" 
    [routerLink]="['/',{ outlets: { 'side': ['player'] } }]" 
    [queryParams]="{ play: 0 }">Stop</button>
</p>

albumgrid.component.html

routerLink


<!-- params and queryParams -->
<a href="#/albumes/Blue-again;flag=true?q=1">Blue again</a>
<a [routerLink]="['albumes', 'Blue again', {flag: true}]" 
   [queryParams]="{q: 1}">Blue again</a>

más opciones

Para acceder a la url usamos Router

@Component({
  selector: 'am-albumdetail'
})
export class AlbumdetailComponent implements OnInit { 
  constructor(private router: Router){ }

  ngOnInit() {
    console.log(this.router.url); // /albumes/Blue-again
  }
}

Accediendo a los datos de la url

import { ActivatedRoute } from '@angular/router';

@Component({
  selector: 'am-albumdetail',
})
export class AlbumdetailComponent implements OnInit {

  constructor(
    private musicService:MusicService, 
    private route: ActivatedRoute
  ) {}

  ngOnInit() {
    let title = this.route.snapshot.params['albumTitle'];
    //...
  }

}

albumdetail.component.ts

Observando los datos de la url 

import { Component, OnInit, OnDestroy } from '@angular/core';
import { ActivatedRoute } from '@angular/router';

@Component({
  selector: 'am-navmusic'
})
export class NavmusicComponent implements OnInit, OnDestroy {
  play: string;

  constructor( private route: ActivatedRoute) { }

  ngOnInit() {
    this.sub = this.route
      .queryParams
      .subscribe(params => {
        this.play = params["play"];
       //...
      });
  }

  ngOnDestroy() {
    this.sub.unsubscribe();
  }
}

navmusic.component.ts

Guardias: preguntar al salir

import { LicenciaComponent }    from './licencia/licencia.component';
import { DeactivateService }    from './deactivate.service';


const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent },
  { 
    path: 'licencia',  
    component: LicenciaComponent, 
    canDeactivate:[DeactivateService]
  },
  
];

app-routing.module.ts

import { Injectable } from '@angular/core';
import { CanDeactivate } from '@angular/router';
import { MdDialog} from '@angular/material';
import { DeactivatedialogComponent } from './deactivatedialog/deactivatedialog.component';
import { LicenciaComponent } from './licencia/licencia.component';


@Injectable()
export class DeactivateService implements CanDeactivate<LicenciaComponent>  {
  constructor(private dialog: MdDialog) { }

  canDeactivate(target: LicenciaComponent) {
    return this.dialog.open(DeactivatedialogComponent).afterClosed();
  }

}

deactivate.service.ts

import { Component, OnInit } from '@angular/core';
import { MdDialogRef } from '@angular/material';


@Component({
  selector: 'app-deactivatedialog',
  templateUrl: './deactivatedialog.component.html',
  styleUrls: ['./deactivatedialog.component.css']
})
export class DeactivatedialogComponent implements OnInit {

  constructor(public dialogRef: MdDialogRef<DeactivatedialogComponent>) { }

  ngOnInit() {
  }

}

deactivatedialog.component.ts

<h1 md-dialog-title>¿Estas seguro?</h1>
<div md-dialog-content>
  <p>¿Seguro que quieres salir? Si es la mar de interesante.</p>
</div>
<div md-dialog-actions>
  <button md-raised-button
    color="primary"
    (click)="dialogRef.close(true)">Por supuesto</button>
  <button md-raised-button
    color="warn"
    (click)="dialogRef.close(false)">Bueno... me quedo</button>
</div>

deactivatedialog.component.html

Restringiendo accesos: validando al entrar

const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent },
  { path: 'licencia',  component: LicenciaComponent, canDeactivate:[DeactivateService] },
  { path: 'albumes',  canActivate:[AuthService] ,
      children: [
        {path:'', component: AlbumgridComponent},
        {path:':albumTitle', component: AlbumdetailComponent}
      ]
  }
];

app-routing.module.ts

import { Injectable } from '@angular/core';
import { Router, CanActivate} from '@angular/router';
import {Observable} from 'rxjs/Observable';
import 'rxjs/add/observable/of';

@Injectable()
export class AuthService  implements CanActivate {
  logged:boolean;
  constructor() {
    this.logged = false;
  }
  canActivate(){
    return Observable.of(this.logged);
  }
}

auth.service.ts

Rutas auxiliares

const routes: Routes = [
  { path: '', redirectTo: '/bienvenido', pathMatch: 'full' },
  { path: 'bienvenido',  component: BienvenidoComponent },
  { path: 'licencia',  component: LicenciaComponent, canDeactivate:[DeactivateService] },
  { path: 'albumes',  canActivate:[AuthService] ,
      children: [
        {path:'', component: AlbumgridComponent},
        {path:':albumTitle', component: AlbumdetailComponent}
      ]
  },
  { path: 'player', component: NavmusicComponent, outlet: 'side'}
]

app-routing.module.ts

  <md-sidenav #sidenav mode="side" class="app-sidenav">

    <router-outlet name="side"></router-outlet>

    <p>
        <button md-raised-button color="primary" 
            [routerLinkActive]="['hidebutton']" 
            [routerLink]="[{ outlets: { 'side': ['player'] } }]"
        >Load player</button>
    </p>

  </md-sidenav>

app.component.html

<p>
  <button md-raised-button
    [routerLinkActive]="['active']" 
    [routerLink]="['/',{ outlets: { 'side': ['player'] } }]" 
    [queryParams]="{ play: 1 }">Play</button>
  <button md-raised-button
    [routerLinkActive]="['active']" 
    [routerLink]="['/',{ outlets: { 'side': ['player'] } }]" 
    [queryParams]="{ play: 0 }">Stop</button>
</p>

albumgrid.component.html

albumgrid.component.css

.active{font-style:italic;}

Lazy loading

Lazy loading

const routes: Routes = [
  //...
  { path: 'lazy', loadChildren: 'app/lazy/lazy.module#LazyModule' }
];

app-routing.module.ts

Lazy loading

import { NgModule } from '@angular/core';
import { MaterialModule } from '@angular/material';

import { LazyComponent }   from './lazy.component';
import { routing } from './lazy.routing';

@NgModule({
  imports: [routing,   MaterialModule.forRoot()],
  declarations: [LazyComponent]
})
export class LazyModule {}

lazy.module.ts

Lazy loading

import { ModuleWithProviders } from '@angular/core';
import { Routes, RouterModule } from '@angular/router';

import { LazyComponent } from './lazy.component';

const routes: Routes = [
  { path: '', component: LazyComponent }
];

export const routing: ModuleWithProviders = RouterModule.forChild(routes);

lazy.routing.ts

Lazy loading

import { Component } from '@angular/core';

@Component({
  template: `<md-card>
    <md-card-content>
      <p>¡Esto es lazy!</p>
    </md-card-content>
  </md-card>`
})
export class LazyComponent {}

lazy.component.ts

Estrategias de Lazy Loading

@NgModule({
  bootstrap: [AppCmp],
  imports: [
    RouterModule.forRoot(ROUTES, 
    {preloadingStrategy: PreloadAllModules})
  ]
})
class AppModule {}

Preloading

@NgModule({
  bootstrap: [AppCmp],
  imports: [
    RouterModule.forRoot(ROUTES, 
    {preloadingStrategy: NoPreloading})
  ]
})
class AppModule {}

Sin preloading

Estrategias a medida de Lazy Loading

[
  {
    path: 'moduleA',
    loadChildren: './moduleA.module',
    data: {preload: true}
  },
  {
    path: 'moduleB',
    loadChildren: './moduleB.module'
  }
]

Agregamos un nuevo atributo a los datos de las rutas

Estrategias a medida de Lazy Loading

export class CustomPreloadingStrategy implements PreloadingStrategy {
  preload(route: Route, load: Function): Observable<any> {
    return route.data && route.data.preload ? load() : of(null);
  }
}

Implementamos la estrategia usando la interfaz PreloadingStrategy

Configuramos el módulo con la nueva estrategia

@NgModule({
  bootstrap: [AppCmp],
  providers: [CustomPreloadingStrategy],
  imports: [
    RouterModule.forRoot(
        ROUTES, 
        {preloadingStrategy: CustomPreloadingStrategy}
    )
  ]
})
class AppModule {}

Revisión de características

  • Navegación por URLs
  • Routing flexible:
    • rutas hijas
    • rutas auxiliares
  • Guardias en la navegación
  • Lazy loading & preloading

Pendiente para agregar al ejemplo

  • Resolve
  • Inner outlets

¡Gracias!

 

@gruizdevilla

@gft_es

Angular Router

By Gonzalo Ruiz de Villa

Angular Router

  • 3,233