import {Component, Injector} from '@angular/core'; import { describe, it, iit, xit, expect, beforeEach, beforeEachProviders, inject, fakeAsync, tick } from '@angular/core/testing'; import {TestComponentBuilder, ComponentFixture} from '@angular/compiler/testing'; import { ComponentResolver } from '@angular/core'; import { SpyLocation } from '@angular/common/testing'; import { UrlSerializer, DefaultUrlSerializer, RouterOutletMap, Router, ActivatedRoute, ROUTER_DIRECTIVES, Params } from '../src/index'; import { Observable } from 'rxjs/Observable'; import 'rxjs/add/operator/map'; describe("Integration", () => { beforeEachProviders(() => [ RouterOutletMap, {provide: UrlSerializer, useClass: DefaultUrlSerializer}, {provide: Location, useClass: SpyLocation}, { provide: Router, useFactory: (resolver, urlSerializer, outletMap, location, injector) => new Router(RootCmp, resolver, urlSerializer, outletMap, location, injector), deps: [ComponentResolver, UrlSerializer, RouterOutletMap, Location, Injector] }, {provide: ActivatedRoute, useFactory: (r) => r.routerState.root, deps: [Router]}, ]); it('should update location when navigating', fakeAsync(inject([Router, TestComponentBuilder, Location], (router, tcb, location) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22'); advance(fixture); expect(location.path()).toEqual('/team/22'); router.navigateByUrl('/team/33'); advance(fixture); expect(location.path()).toEqual('/team/33'); }))); xit('should navigate back and forward', fakeAsync(inject([Router, TestComponentBuilder, Location], (router, tcb, location) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'simple', component: SimpleCmp }, { path: 'user/:name', component: UserCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/33/simple'); advance(fixture); expect(location.path()).toEqual('/team/33/simple'); router.navigateByUrl('/team/22/user/victor'); advance(fixture); location.back(); advance(fixture); expect(location.path()).toEqual('/team/33/simple'); location.forward(); advance(fixture); expect(location.path()).toEqual('/team/22/user/victor'); }))); it('should navigate when locations changes', fakeAsync(inject([Router, TestComponentBuilder, Location], (router, tcb, location) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'user/:name', component: UserCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/22/user/victor'); advance(fixture); location.simulateHashChange("/team/22/user/fedor"); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { user fedor, right: }'); }))); it('should support secondary routes', fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'user/:name', component: UserCmp }, { path: 'simple', component: SimpleCmp, outlet: 'right' } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/22/user/victor(right:simple)'); advance(fixture); expect(fixture.debugElement.nativeElement) .toHaveText('team 22 { user victor, right: simple }'); }))); it('should deactivate outlets', fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'user/:name', component: UserCmp }, { path: 'simple', component: SimpleCmp, outlet: 'right' } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/22/user/victor(right:simple)'); advance(fixture); router.navigateByUrl('/team/22/user/victor'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { user victor, right: }'); }))); it('should deactivate nested outlets', fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'user/:name', component: UserCmp }, { path: 'simple', component: SimpleCmp, outlet: 'right' } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/22/user/victor(right:simple)'); advance(fixture); router.navigateByUrl('/'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText(''); }))); it('should set query params and fragment', fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'query', component: QueryParamsAndFragmentCmp } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/query?name=1#fragment1'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('query: 1 fragment: fragment1'); router.navigateByUrl('/query?name=2#fragment2'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('query: 2 fragment: fragment2'); }))); it('should push params only when they change', fakeAsync(inject([Router, TestComponentBuilder], (router, tcb:TestComponentBuilder) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'user/:name', component: UserCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); router.navigateByUrl('/team/22/user/victor'); advance(fixture); const team = fixture.debugElement.children[1].componentInstance; const user = fixture.debugElement.children[1].children[1].componentInstance; expect(team.recordedParams).toEqual([{id: '22'}]); expect(user.recordedParams).toEqual([{name: 'victor'}]); router.navigateByUrl('/team/22/user/fedor'); advance(fixture); expect(team.recordedParams).toEqual([{id: '22'}]); expect(user.recordedParams).toEqual([{name: 'victor'}, {name: 'fedor'}]); }))); describe("router links", () => { it("should support string router links", fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'link', component: StringLinkCmp }, { path: 'simple', component: SimpleCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22/link'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, right: }'); const native = fixture.debugElement.nativeElement.querySelector("a"); expect(native.getAttribute("href")).toEqual("/team/33/simple"); native.click(); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 33 { simple, right: }'); }))); it("should support absolute router links", fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'link', component: AbsoluteLinkCmp }, { path: 'simple', component: SimpleCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22/link'); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, right: }'); const native = fixture.debugElement.nativeElement.querySelector("a"); expect(native.getAttribute("href")).toEqual("/team/33/simple"); native.click(); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('team 33 { simple, right: }'); }))); it("should support relative router links", fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, children: [ { path: 'link', component: RelativeLinkCmp }, { path: 'simple', component: SimpleCmp } ] } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22/link'); advance(fixture); expect(fixture.debugElement.nativeElement) .toHaveText('team 22 { link, right: }'); const native = fixture.debugElement.nativeElement.querySelector("a"); expect(native.getAttribute("href")).toEqual("/team/22/simple"); native.click(); advance(fixture); expect(fixture.debugElement.nativeElement) .toHaveText('team 22 { simple, right: }'); }))); it("should support top-level link", fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { let fixture = tcb.createFakeAsync(AbsoluteLinkCmp); advance(fixture); expect(fixture.debugElement.nativeElement).toHaveText('link'); }))); }); describe("guards", () => { describe("CanActivate", () => { describe("should not activate a route when CanActivate returns false", () => { beforeEachProviders(() => [ {provide: 'alwaysFalse', useValue: (a, b) => false} ]); it('works', fakeAsync(inject([Router, TestComponentBuilder, Location], (router, tcb, location) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, canActivate: ["alwaysFalse"] } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22'); advance(fixture); expect(location.path()).toEqual(''); }))); }); describe("should activate a route when CanActivate returns true", () => { beforeEachProviders(() => [ {provide: 'alwaysFalse', useValue: (a, b) => true} ]); it('works', fakeAsync(inject([Router, TestComponentBuilder, Location], (router, tcb, location) => { router.resetConfig([ { path: 'team/:id', component: TeamCmp, canActivate: ["alwaysFalse"] } ]); const fixture = tcb.createFakeAsync(RootCmp); advance(fixture); router.navigateByUrl('/team/22'); advance(fixture); expect(location.path()).toEqual('/team/22'); }))); }); }); }); }); @Component({ selector: 'link-cmp', template: `link`, directives: ROUTER_DIRECTIVES }) class StringLinkCmp {} @Component({ selector: 'link-cmp', template: `link`, directives: ROUTER_DIRECTIVES }) class AbsoluteLinkCmp {} @Component({ selector: 'link-cmp', template: `link`, directives: ROUTER_DIRECTIVES }) class RelativeLinkCmp {} @Component({ selector: 'simple-cmp', template: `simple`, directives: ROUTER_DIRECTIVES }) class SimpleCmp { } @Component({ selector: 'team-cmp', template: `team {{id | async}} { , right: }`, directives: ROUTER_DIRECTIVES }) class TeamCmp { id: Observable; recordedParams: Params[] = []; constructor(route: ActivatedRoute) { this.id = route.params.map(p => p['id']); route.params.forEach(_ => this.recordedParams.push(_)); } } @Component({ selector: 'user-cmp', template: `user {{name | async}}`, directives: [ROUTER_DIRECTIVES] }) class UserCmp { name: Observable; recordedParams: Params[] = []; constructor(route: ActivatedRoute) { this.name = route.params.map(p => p['name']); route.params.forEach(_ => this.recordedParams.push(_)); } } @Component({ selector: 'query-cmp', template: `query: {{name | async}} fragment: {{fragment | async}}`, directives: [ROUTER_DIRECTIVES] }) class QueryParamsAndFragmentCmp { name: Observable; fragment: Observable; constructor(router: Router) { this.name = router.routerState.queryParams.map(p => p['name']); this.fragment = router.routerState.fragment; } } @Component({ selector: 'root-cmp', template: ``, directives: [ROUTER_DIRECTIVES] }) class RootCmp {} function advance(fixture: ComponentFixture): void { tick(); fixture.detectChanges(); }