| 
									
										
											  
											
												refactor(router): improve recognition and generation pipeline
This is a big change. @matsko also deserves much of the credit for the implementation.
Previously, `ComponentInstruction`s held all the state for async components.
Now, we introduce several subclasses for `Instruction` to describe each type of navigation.
BREAKING CHANGE:
Redirects now use the Link DSL syntax. Before:
```
@RouteConfig([
	{ path: '/foo', redirectTo: '/bar' },
	{ path: '/bar', component: BarCmp }
])
```
After:
```
@RouteConfig([
	{ path: '/foo', redirectTo: ['Bar'] },
	{ path: '/bar', component: BarCmp, name: 'Bar' }
])
```
BREAKING CHANGE:
This also introduces `useAsDefault` in the RouteConfig, which makes cases like lazy-loading
and encapsulating large routes with sub-routes easier.
Previously, you could use `redirectTo` like this to expand a URL like `/tab` to `/tab/posts`:
@RouteConfig([
	{ path: '/tab', redirectTo: '/tab/users' }
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
Now the recommended way to handle this is case is to use `useAsDefault` like so:
```
@RouteConfig([
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
@RouteConfig([
	{ path: '/posts', component: PostsCmp, useAsDefault: true, name: 'Posts' },
	{ path: '/users', component: UsersCmp, name: 'Users' }
])
TabsCmp { ... }
```
In the above example, you can write just `['/Tab']` and the route `Users` is automatically selected as a child route.
Closes #4728
Closes #4228
Closes #4170
Closes #4490
Closes #4694
Closes #5200
Closes #5475
											
										 
											2015-11-23 18:07:37 -08:00
										 |  |  | import {Component} from 'angular2/core'; | 
					
						
							| 
									
										
										
										
											2016-04-12 09:40:37 -07:00
										 |  |  | import { | 
					
						
							|  |  |  |   AsyncRoute, | 
					
						
							|  |  |  |   Route, | 
					
						
							|  |  |  |   Redirect, | 
					
						
							|  |  |  |   RouteConfig, | 
					
						
							|  |  |  |   RouteParams, | 
					
						
							|  |  |  |   RouteData, | 
					
						
							|  |  |  |   ROUTER_DIRECTIVES | 
					
						
							|  |  |  | } from 'angular2/router'; | 
					
						
							| 
									
										
											  
											
												refactor(router): improve recognition and generation pipeline
This is a big change. @matsko also deserves much of the credit for the implementation.
Previously, `ComponentInstruction`s held all the state for async components.
Now, we introduce several subclasses for `Instruction` to describe each type of navigation.
BREAKING CHANGE:
Redirects now use the Link DSL syntax. Before:
```
@RouteConfig([
	{ path: '/foo', redirectTo: '/bar' },
	{ path: '/bar', component: BarCmp }
])
```
After:
```
@RouteConfig([
	{ path: '/foo', redirectTo: ['Bar'] },
	{ path: '/bar', component: BarCmp, name: 'Bar' }
])
```
BREAKING CHANGE:
This also introduces `useAsDefault` in the RouteConfig, which makes cases like lazy-loading
and encapsulating large routes with sub-routes easier.
Previously, you could use `redirectTo` like this to expand a URL like `/tab` to `/tab/posts`:
@RouteConfig([
	{ path: '/tab', redirectTo: '/tab/users' }
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
Now the recommended way to handle this is case is to use `useAsDefault` like so:
```
@RouteConfig([
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
@RouteConfig([
	{ path: '/posts', component: PostsCmp, useAsDefault: true, name: 'Posts' },
	{ path: '/users', component: UsersCmp, name: 'Users' }
])
TabsCmp { ... }
```
In the above example, you can write just `['/Tab']` and the route `Users` is automatically selected as a child route.
Closes #4728
Closes #4228
Closes #4170
Closes #4490
Closes #4694
Closes #5200
Closes #5475
											
										 
											2015-11-23 18:07:37 -08:00
										 |  |  | import {PromiseWrapper} from 'angular2/src/facade/async'; | 
					
						
							| 
									
										
										
										
											2016-02-26 10:37:43 -08:00
										 |  |  | import {isPresent} from 'angular2/src/facade/lang'; | 
					
						
							| 
									
										
										
										
											2016-04-12 09:40:37 -07:00
										 |  |  | import { | 
					
						
							|  |  |  |   DynamicComponentLoader, | 
					
						
							|  |  |  |   ComponentRef | 
					
						
							|  |  |  | } from 'angular2/src/core/linker/dynamic_component_loader'; | 
					
						
							| 
									
										
										
										
											2016-02-26 10:37:43 -08:00
										 |  |  | import {ElementRef} from 'angular2/src/core/linker/element_ref'; | 
					
						
							| 
									
										
											  
											
												refactor(router): improve recognition and generation pipeline
This is a big change. @matsko also deserves much of the credit for the implementation.
Previously, `ComponentInstruction`s held all the state for async components.
Now, we introduce several subclasses for `Instruction` to describe each type of navigation.
BREAKING CHANGE:
Redirects now use the Link DSL syntax. Before:
```
@RouteConfig([
	{ path: '/foo', redirectTo: '/bar' },
	{ path: '/bar', component: BarCmp }
])
```
After:
```
@RouteConfig([
	{ path: '/foo', redirectTo: ['Bar'] },
	{ path: '/bar', component: BarCmp, name: 'Bar' }
])
```
BREAKING CHANGE:
This also introduces `useAsDefault` in the RouteConfig, which makes cases like lazy-loading
and encapsulating large routes with sub-routes easier.
Previously, you could use `redirectTo` like this to expand a URL like `/tab` to `/tab/posts`:
@RouteConfig([
	{ path: '/tab', redirectTo: '/tab/users' }
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
Now the recommended way to handle this is case is to use `useAsDefault` like so:
```
@RouteConfig([
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
@RouteConfig([
	{ path: '/posts', component: PostsCmp, useAsDefault: true, name: 'Posts' },
	{ path: '/users', component: UsersCmp, name: 'Users' }
])
TabsCmp { ... }
```
In the above example, you can write just `['/Tab']` and the route `Users` is automatically selected as a child route.
Closes #4728
Closes #4228
Closes #4170
Closes #4490
Closes #4694
Closes #5200
Closes #5475
											
										 
											2015-11-23 18:07:37 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-15 15:58:04 -08:00
										 |  |  | @Component({selector: 'goodbye-cmp', template: `{{farewell}}`}) | 
					
						
							|  |  |  | export class GoodbyeCmp { | 
					
						
							|  |  |  |   farewell: string; | 
					
						
							|  |  |  |   constructor() { this.farewell = 'goodbye'; } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												refactor(router): improve recognition and generation pipeline
This is a big change. @matsko also deserves much of the credit for the implementation.
Previously, `ComponentInstruction`s held all the state for async components.
Now, we introduce several subclasses for `Instruction` to describe each type of navigation.
BREAKING CHANGE:
Redirects now use the Link DSL syntax. Before:
```
@RouteConfig([
	{ path: '/foo', redirectTo: '/bar' },
	{ path: '/bar', component: BarCmp }
])
```
After:
```
@RouteConfig([
	{ path: '/foo', redirectTo: ['Bar'] },
	{ path: '/bar', component: BarCmp, name: 'Bar' }
])
```
BREAKING CHANGE:
This also introduces `useAsDefault` in the RouteConfig, which makes cases like lazy-loading
and encapsulating large routes with sub-routes easier.
Previously, you could use `redirectTo` like this to expand a URL like `/tab` to `/tab/posts`:
@RouteConfig([
	{ path: '/tab', redirectTo: '/tab/users' }
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
Now the recommended way to handle this is case is to use `useAsDefault` like so:
```
@RouteConfig([
	{ path: '/tab', component: TabsCmp, name: 'Tab' }
])
AppCmp { ... }
@RouteConfig([
	{ path: '/posts', component: PostsCmp, useAsDefault: true, name: 'Posts' },
	{ path: '/users', component: UsersCmp, name: 'Users' }
])
TabsCmp { ... }
```
In the above example, you can write just `['/Tab']` and the route `Users` is automatically selected as a child route.
Closes #4728
Closes #4228
Closes #4170
Closes #4490
Closes #4694
Closes #5200
Closes #5475
											
										 
											2015-11-23 18:07:37 -08:00
										 |  |  | @Component({selector: 'hello-cmp', template: `{{greeting}}`}) | 
					
						
							|  |  |  | export class HelloCmp { | 
					
						
							|  |  |  |   greeting: string; | 
					
						
							|  |  |  |   constructor() { this.greeting = 'hello'; } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function helloCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(HelloCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({selector: 'user-cmp', template: `hello {{user}}`}) | 
					
						
							|  |  |  | export class UserCmp { | 
					
						
							|  |  |  |   user: string; | 
					
						
							|  |  |  |   constructor(params: RouteParams) { this.user = params.get('name'); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function userCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(UserCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'parent-cmp', | 
					
						
							|  |  |  |   template: `inner { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig([new Route({path: '/b', component: HelloCmp, name: 'Child'})]) | 
					
						
							|  |  |  | export class ParentCmp { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function parentCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(ParentCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'parent-cmp', | 
					
						
							|  |  |  |   template: `inner { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig([new AsyncRoute({path: '/b', loader: helloCmpLoader, name: 'Child'})]) | 
					
						
							|  |  |  | export class AsyncParentCmp { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function asyncParentCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(AsyncParentCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'parent-cmp', | 
					
						
							|  |  |  |   template: `inner { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig( | 
					
						
							|  |  |  |     [new AsyncRoute({path: '/b', loader: helloCmpLoader, name: 'Child', useAsDefault: true})]) | 
					
						
							|  |  |  | export class AsyncDefaultParentCmp { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function asyncDefaultParentCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(AsyncDefaultParentCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'parent-cmp', | 
					
						
							|  |  |  |   template: `inner { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig([new Route({path: '/b', component: HelloCmp, name: 'Child', useAsDefault: true})]) | 
					
						
							|  |  |  | export class ParentWithDefaultCmp { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function parentWithDefaultCmpLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(ParentWithDefaultCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'team-cmp', | 
					
						
							|  |  |  |   template: `team {{id}} | user { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig([new Route({path: '/user/:name', component: UserCmp, name: 'User'})]) | 
					
						
							|  |  |  | export class TeamCmp { | 
					
						
							|  |  |  |   id: string; | 
					
						
							|  |  |  |   constructor(params: RouteParams) { this.id = params.get('id'); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'team-cmp', | 
					
						
							|  |  |  |   template: `team {{id}} | user { <router-outlet></router-outlet> }`, | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES], | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | @RouteConfig([new AsyncRoute({path: '/user/:name', loader: userCmpLoader, name: 'User'})]) | 
					
						
							|  |  |  | export class AsyncTeamCmp { | 
					
						
							|  |  |  |   id: string; | 
					
						
							|  |  |  |   constructor(params: RouteParams) { this.id = params.get('id'); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function asyncTeamLoader() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(AsyncTeamCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({selector: 'data-cmp', template: `{{myData}}`}) | 
					
						
							|  |  |  | export class RouteDataCmp { | 
					
						
							|  |  |  |   myData: boolean; | 
					
						
							|  |  |  |   constructor(data: RouteData) { this.myData = data.get('isAdmin'); } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export function asyncRouteDataCmp() { | 
					
						
							|  |  |  |   return PromiseWrapper.resolve(RouteDataCmp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({selector: 'redirect-to-parent-cmp', template: 'redirect-to-parent'}) | 
					
						
							|  |  |  | @RouteConfig([new Redirect({path: '/child-redirect', redirectTo: ['../HelloSib']})]) | 
					
						
							|  |  |  | export class RedirectToParentCmp { | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-02-26 10:37:43 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({selector: 'dynamic-loader-cmp', template: `{ <div #viewport></div> }`}) | 
					
						
							|  |  |  | @RouteConfig([new Route({path: '/', component: HelloCmp})]) | 
					
						
							|  |  |  | export class DynamicLoaderCmp { | 
					
						
							|  |  |  |   private _componentRef: ComponentRef = null; | 
					
						
							| 
									
										
										
										
											2016-04-12 09:40:37 -07:00
										 |  |  |   constructor(private _dynamicComponentLoader: DynamicComponentLoader, | 
					
						
							|  |  |  |               private _elementRef: ElementRef) {} | 
					
						
							| 
									
										
										
										
											2016-02-26 10:37:43 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   onSomeAction(): Promise<any> { | 
					
						
							|  |  |  |     if (isPresent(this._componentRef)) { | 
					
						
							|  |  |  |       this._componentRef.dispose(); | 
					
						
							|  |  |  |       this._componentRef = null; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-04-12 09:40:37 -07:00
										 |  |  |     return this._dynamicComponentLoader.loadIntoLocation(DynamicallyLoadedComponent, | 
					
						
							|  |  |  |                                                          this._elementRef, 'viewport') | 
					
						
							| 
									
										
										
										
											2016-02-26 10:37:43 -08:00
										 |  |  |         .then((cmp) => { this._componentRef = cmp; }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @Component({ | 
					
						
							|  |  |  |   selector: 'loaded-cmp', | 
					
						
							|  |  |  |   template: '<router-outlet></router-outlet>', | 
					
						
							|  |  |  |   directives: [ROUTER_DIRECTIVES] | 
					
						
							|  |  |  | }) | 
					
						
							|  |  |  | class DynamicallyLoadedComponent { | 
					
						
							|  |  |  | } |