Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Resolve a value *before* activating a route in Angular 2

I'd like to prevent a route from being accessed as long as a promise hasn't been resolved. Also, I'd like to pass the return value of that promise to the route component.

Several posts on SO recommend using the OnActivate interface for this. The documentation says: "If routerOnActivate returns a promise, the route change will wait until the promise settles to instantiate and activate child components." Sounds perfect, only the route still gets activated immediately... (Is it because it's the child components of Route2 that will wait for the promise, not the Route2 component itself??)

See https://plnkr.co/edit/ipKrOgfRj0vKk8ZejH5v?p=preview. The Route2 component implements the OnActivate interface, but when you click the Route2 link, the Route2 component is activated immediately. However, the URL does update to /route2 only after the promise has resolved. (Launch the Plunker in a separate window to see what I mean.) Now I don't care so much about the URL, the Route2 component should not instantiate at all until the promise is resolved.

My other strategy was to use the @CanActivate decorator. It better suits my purpose as "it is called by the router to determine if a component can be instantiated as part of a navigation".

The problem here is how to pass data from the promise in the @CanActivate decorator to the component?

I have seen people write their data into a property of the next parameter of the decorator, and retrieve it in the next parameter of the routerOnActivate method.

For instance in next.params (here):

@CanActivate((next) => {
  return messageService.getMessage()
      .then((message) => {
         next.params.message = message;
         return true;
      });
})
export class MyComponent implements OnActivate {
  routerOnActivate(next) {
    this.message = next.params.message;
  }
}

Or in next.routeData.data (here).

But the documentation says ComponentInstruction objects "should be treated as immutable". This is confusing.

Is this the best way to do this?

Maybe the router in its current state isn't finalized and there will be a better way when a stable Angular 2 is released?

like image 558
AngularChef Avatar asked Apr 09 '16 14:04

AngularChef


People also ask

What does the * * path in Angular router do?

The two asterisks, ** , indicate to Angular that this routes definition is a wildcard route. For the component property, you can define any component in your application.

Can activate vs CanDeactivate?

CanActivate - Decides if a route can be activated. CanActivateChild - Decides if children routes of a route can be activated. CanDeactivate - Decides if a route can be deactivated.

What is the purpose of wildcard route in Angular?

The Wildcard Route is basically used in Angular Application to handle the invalid URLs. Whenever the user enter some invalid URL or if you have deleted some existing URL from your application, then by default 404 page not found error page is displayed.

Where do you set the route parameter value in Angular?

The first way is through the route snapshot. The route snapshot provides the initial value of the route parameter map (called the paramMap ). You can access the parameters directly without subscribing or adding observable operators. The paramMap provides methods to handle parameter access like get , getAll , and has .


1 Answers

Update

This is implemented in the new router >= RC.4

import { Injectable }             from '@angular/core';
import { Router, Resolve,
         ActivatedRouteSnapshot } from '@angular/router';
import { Observable }             from 'rxjs/Observable';
import { Crisis, CrisisService } from './crisis.service';
@Injectable()
export class CrisisDetailResolve implements Resolve<Crisis> {
  constructor(private cs: CrisisService, private router: Router) {}
  resolve(route: ActivatedRouteSnapshot): Observable<any> | Promise<any> | any {
    let id = +route.params['id'];
    return this.cs.getCrisis(id).then(crisis => {
      if (crisis) {
        return crisis;
      } else { // id not found
        this.router.navigate(['/crisis-center']);
        return false;
      }
    });
  }
}
  • https://angular.io/docs/ts/latest/guide/router.html#!#resolve-guard
  • https://angular.io/docs/ts/latest/api/router/index/Resolve-interface.html
  • https://angular.io/docs/ts/latest/api/router/index/Routes-type-alias.html

Original

Maybe the router in its current state isn't finalized and there will be a better way when a stable Angular 2 is released?

That's actually the case. There were lots of discussions recently about further progress and several changes are planned.

See also
- Angular Weekly Meeting (April 4)
- Router: Design for routerCanActivate, an injectable alternative to @CanActivate

I have seen people write their data into a property of the next parameter of the decorator, and retrieve it in the next parameter of the routerOnActivate method.

AFAIK the params is supposed to be immutable and should not be modified.

The discussions in [router] CanActivate and DI show how to use DI with CanActivate. This is the approach I would use. Share a service globally and inject it in CanActivate to update the values and inject it also in the components where you want to get access to the value.

like image 107
Günter Zöchbauer Avatar answered Oct 31 '22 19:10

Günter Zöchbauer