2015-04-17 09:59:56 -07:00
|
|
|
import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
|
2015-05-14 13:01:48 -07:00
|
|
|
import {isBlank, isPresent} from 'angular2/src/facade/lang';
|
2015-04-17 09:59:56 -07:00
|
|
|
|
2015-04-30 13:38:40 -07:00
|
|
|
import {Directive} from 'angular2/src/core/annotations_impl/annotations';
|
2015-05-03 20:25:04 -07:00
|
|
|
import {Attribute} from 'angular2/src/core/annotations_impl/di';
|
2015-05-15 15:55:26 -07:00
|
|
|
import {DynamicComponentLoader, ComponentRef, ElementRef} from 'angular2/core';
|
2015-04-17 09:59:56 -07:00
|
|
|
import {Injector, bind} from 'angular2/di';
|
|
|
|
|
|
|
|
import * as routerMod from './router';
|
|
|
|
import {Instruction, RouteParams} from './instruction'
|
|
|
|
|
2015-05-14 13:01:48 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* A router outlet is a placeholder that Angular dynamically fills based on the application's route.
|
|
|
|
*
|
|
|
|
* ## Use
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <router-outlet></router-outlet>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Route outlets can also optionally have a name:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* <router-outlet name="side"></router-outlet>
|
|
|
|
* <router-outlet name="main"></router-outlet>
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
*/
|
2015-04-30 13:38:40 -07:00
|
|
|
@Directive({
|
2015-04-17 09:59:56 -07:00
|
|
|
selector: 'router-outlet'
|
|
|
|
})
|
|
|
|
export class RouterOutlet {
|
|
|
|
_injector:Injector;
|
2015-05-14 13:01:48 -07:00
|
|
|
_parentRouter:routerMod.Router;
|
|
|
|
_childRouter:routerMod.Router;
|
2015-05-15 15:55:26 -07:00
|
|
|
_loader:DynamicComponentLoader;
|
|
|
|
_componentRef:ComponentRef;
|
|
|
|
_elementRef:ElementRef;
|
2015-05-19 10:35:28 -07:00
|
|
|
_currentInstruction:Instruction;
|
2015-04-17 09:59:56 -07:00
|
|
|
|
2015-05-15 15:55:26 -07:00
|
|
|
constructor(elementRef:ElementRef, loader:DynamicComponentLoader, router:routerMod.Router, injector:Injector, @Attribute('name') nameAttr:String) {
|
2015-05-03 20:25:04 -07:00
|
|
|
if (isBlank(nameAttr)) {
|
|
|
|
nameAttr = 'default';
|
|
|
|
}
|
2015-05-15 15:55:26 -07:00
|
|
|
this._loader = loader;
|
2015-05-14 13:01:48 -07:00
|
|
|
this._parentRouter = router;
|
2015-05-15 15:55:26 -07:00
|
|
|
this._elementRef = elementRef;
|
2015-04-17 09:59:56 -07:00
|
|
|
this._injector = injector;
|
2015-05-15 15:55:26 -07:00
|
|
|
|
|
|
|
this._childRouter = null;
|
|
|
|
this._componentRef = null;
|
2015-05-19 10:35:28 -07:00
|
|
|
this._currentInstruction = null;
|
2015-05-14 13:01:48 -07:00
|
|
|
this._parentRouter.registerOutlet(this, nameAttr);
|
2015-04-17 09:59:56 -07:00
|
|
|
}
|
|
|
|
|
2015-05-14 13:01:48 -07:00
|
|
|
/**
|
|
|
|
* Given an instruction, update the contents of this viewport.
|
|
|
|
*/
|
2015-05-14 15:24:35 +02:00
|
|
|
activate(instruction:Instruction): Promise {
|
2015-05-15 15:55:26 -07:00
|
|
|
// if we're able to reuse the component, we just have to pass along the instruction to the component's router
|
|
|
|
// so it can propagate changes to its children
|
2015-05-19 10:35:28 -07:00
|
|
|
if ((instruction == this._currentInstruction) || instruction.reuse && isPresent(this._childRouter)) {
|
2015-05-14 13:01:48 -07:00
|
|
|
return this._childRouter.commit(instruction);
|
|
|
|
}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
2015-05-19 10:35:28 -07:00
|
|
|
this._currentInstruction = instruction;
|
2015-05-15 15:55:26 -07:00
|
|
|
this._childRouter = this._parentRouter.childRouter(instruction.component);
|
|
|
|
var outletInjector = this._injector.resolveAndCreateChild([
|
|
|
|
bind(RouteParams).toValue(new RouteParams(instruction.params)),
|
|
|
|
bind(routerMod.Router).toValue(this._childRouter)
|
|
|
|
]);
|
|
|
|
|
|
|
|
if (isPresent(this._componentRef)) {
|
|
|
|
this._componentRef.dispose();
|
|
|
|
}
|
2015-05-19 10:35:28 -07:00
|
|
|
|
2015-05-15 15:55:26 -07:00
|
|
|
return this._loader.loadNextToExistingLocation(instruction.component, this._elementRef, outletInjector).then((componentRef) => {
|
|
|
|
this._componentRef = componentRef;
|
2015-05-14 13:01:48 -07:00
|
|
|
return this._childRouter.commit(instruction);
|
2015-04-17 09:59:56 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-05-14 13:01:48 -07:00
|
|
|
deactivate():Promise {
|
|
|
|
return (isPresent(this._childRouter) ? this._childRouter.deactivate() : PromiseWrapper.resolve(true))
|
2015-05-15 15:55:26 -07:00
|
|
|
.then((_) =>this._componentRef.dispose());
|
2015-04-17 09:59:56 -07:00
|
|
|
}
|
|
|
|
|
2015-05-14 15:24:35 +02:00
|
|
|
canDeactivate(instruction:Instruction): Promise<boolean> {
|
2015-05-14 13:01:48 -07:00
|
|
|
// TODO: how to get ahold of the component instance here?
|
2015-04-17 09:59:56 -07:00
|
|
|
return PromiseWrapper.resolve(true);
|
|
|
|
}
|
|
|
|
}
|