feat(router): update router to support lazy loading

This commit is contained in:
vsavkin 2016-05-03 16:45:30 -07:00
parent c0cfd3c6ed
commit 0f1465b899
5 changed files with 43 additions and 24 deletions

View File

@ -4,6 +4,7 @@ import {
IS_DART, IS_DART,
Type, Type,
isBlank, isBlank,
isString
} from '../src/facade/lang'; } from '../src/facade/lang';
import {BaseException} from '../src/facade/exceptions'; import {BaseException} from '../src/facade/exceptions';
import { import {
@ -49,7 +50,12 @@ export class RuntimeCompiler implements ComponentResolver {
private _viewCompiler: ViewCompiler, private _xhr: XHR, private _viewCompiler: ViewCompiler, private _xhr: XHR,
private _genConfig: CompilerConfig) {} private _genConfig: CompilerConfig) {}
resolveComponent(componentType: Type): Promise<ComponentFactory<any>> { resolveComponent(component: Type|string): Promise<ComponentFactory<any>> {
if (isString(component)) {
return PromiseWrapper.reject(new BaseException(`Cannot resolve component using '${component}'.`), null);
}
let componentType = <Type>component;
var compMeta: CompileDirectiveMetadata = var compMeta: CompileDirectiveMetadata =
this._metadataResolver.getDirectiveMetadata(componentType); this._metadataResolver.getDirectiveMetadata(componentType);
var hostCacheKey = this._hostCacheKeys.get(componentType); var hostCacheKey = this._hostCacheKeys.get(componentType);

View File

@ -1,4 +1,4 @@
import {Type, isBlank, stringify} from '../../src/facade/lang'; import {Type, isBlank, isString, stringify} from '../../src/facade/lang';
import {BaseException} from '../../src/facade/exceptions'; import {BaseException} from '../../src/facade/exceptions';
import {PromiseWrapper} from '../../src/facade/async'; import {PromiseWrapper} from '../../src/facade/async';
import {reflector} from '../reflection/reflection'; import {reflector} from '../reflection/reflection';
@ -10,7 +10,7 @@ import {Injectable} from '../di/decorators';
* can later be used to create and render a Component instance. * can later be used to create and render a Component instance.
*/ */
export abstract class ComponentResolver { export abstract class ComponentResolver {
abstract resolveComponent(componentType: Type): Promise<ComponentFactory<any>>; abstract resolveComponent(component: Type|string): Promise<ComponentFactory<any>>;
abstract clearCache(); abstract clearCache();
} }
@ -20,12 +20,16 @@ function _isComponentFactory(type: any): boolean {
@Injectable() @Injectable()
export class ReflectorComponentResolver extends ComponentResolver { export class ReflectorComponentResolver extends ComponentResolver {
resolveComponent(componentType: Type): Promise<ComponentFactory<any>> { resolveComponent(component: Type|string): Promise<ComponentFactory<any>> {
var metadatas = reflector.annotations(componentType); if (isString(component)) {
return PromiseWrapper.reject(new BaseException(`Cannot resolve component using '${component}'.`), null);
}
var metadatas = reflector.annotations(<Type>component);
var componentFactory = metadatas.find(_isComponentFactory); var componentFactory = metadatas.find(_isComponentFactory);
if (isBlank(componentFactory)) { if (isBlank(componentFactory)) {
throw new BaseException(`No precompiled component ${stringify(componentType)} found`); throw new BaseException(`No precompiled component ${stringify(component)} found`);
} }
return PromiseWrapper.resolve(componentFactory); return PromiseWrapper.resolve(componentFactory);
} }

View File

@ -41,6 +41,15 @@ export function main() {
return null; return null;
}); });
})); }));
it('should throw when given a string',
inject([AsyncTestCompleter, ComponentResolver], (async, compiler: ComponentResolver) => {
compiler.resolveComponent("someString")
.catch((e) => {
expect(e.message).toContain("Cannot resolve component using 'someString'.")
async.done();
});
}));
}); });
} }

View File

@ -22,7 +22,7 @@ import {stringify} from "../facade/lang";
*/ */
export abstract class RouteMetadata { export abstract class RouteMetadata {
abstract get path(): string; abstract get path(): string;
abstract get component(): Type; abstract get component(): Type|string;
} }
/** /**
@ -31,8 +31,8 @@ export abstract class RouteMetadata {
*/ */
export class Route implements RouteMetadata { export class Route implements RouteMetadata {
path: string; path: string;
component: Type; component: Type|string;
constructor({path, component}: {path?: string, component?: Type} = {}) { constructor({path, component}: {path?: string, component?: Type|string} = {}) {
this.path = path; this.path = path;
this.component = component; this.component = component;
} }

View File

@ -9,18 +9,18 @@ import {DEFAULT_OUTLET_NAME} from './constants';
import {reflector} from '@angular/core'; import {reflector} from '@angular/core';
// TODO: vsavkin: recognize should take the old tree and merge it // TODO: vsavkin: recognize should take the old tree and merge it
export function recognize(componentResolver: ComponentResolver, type: Type, export function recognize(componentResolver: ComponentResolver, rootComponent: Type,
url: UrlTree): Promise<RouteTree> { url: UrlTree): Promise<RouteTree> {
let matched = new _MatchResult(type, [url.root], {}, rootNode(url).children, []); let matched = new _MatchResult(rootComponent, [url.root], null, rootNode(url).children, []);
return _constructSegment(componentResolver, matched).then(roots => new RouteTree(roots[0])); return _constructSegment(componentResolver, matched).then(roots => new RouteTree(roots[0]));
} }
function _recognize(componentResolver: ComponentResolver, parentType: Type, function _recognize(componentResolver: ComponentResolver, parentComponent: Type,
url: TreeNode<UrlSegment>): Promise<TreeNode<RouteSegment>[]> { url: TreeNode<UrlSegment>): Promise<TreeNode<RouteSegment>[]> {
let metadata = _readMetadata(parentType); // should read from the factory instead let metadata = _readMetadata(parentComponent); // should read from the factory instead
if (isBlank(metadata)) { if (isBlank(metadata)) {
throw new BaseException( throw new BaseException(
`Component '${stringify(parentType)}' does not have route configuration`); `Component '${stringify(parentComponent)}' does not have route configuration`);
} }
let match; let match;
@ -32,13 +32,13 @@ function _recognize(componentResolver: ComponentResolver, parentType: Type,
let main = _constructSegment(componentResolver, match); let main = _constructSegment(componentResolver, match);
let aux = let aux =
_recognizeMany(componentResolver, parentType, match.aux).then(_checkOutletNameUniqueness); _recognizeMany(componentResolver, parentComponent, match.aux).then(_checkOutletNameUniqueness);
return PromiseWrapper.all([main, aux]).then(ListWrapper.flatten); return PromiseWrapper.all([main, aux]).then(ListWrapper.flatten);
} }
function _recognizeMany(componentResolver: ComponentResolver, parentType: Type, function _recognizeMany(componentResolver: ComponentResolver, parentComponent: Type,
urls: TreeNode<UrlSegment>[]): Promise<TreeNode<RouteSegment>[]> { urls: TreeNode<UrlSegment>[]): Promise<TreeNode<RouteSegment>[]> {
let recognized = urls.map(u => _recognize(componentResolver, parentType, u)); let recognized = urls.map(u => _recognize(componentResolver, parentComponent, u));
return PromiseWrapper.all(recognized).then(ListWrapper.flatten); return PromiseWrapper.all(recognized).then(ListWrapper.flatten);
} }
@ -52,23 +52,23 @@ function _constructSegment(componentResolver: ComponentResolver,
matched.consumedUrlSegments[0].outlet; matched.consumedUrlSegments[0].outlet;
let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters, urlOutlet, let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters, urlOutlet,
matched.component, factory); factory.componentType, factory);
if (matched.leftOverUrl.length > 0) { if (matched.leftOverUrl.length > 0) {
return _recognizeMany(componentResolver, matched.component, matched.leftOverUrl) return _recognizeMany(componentResolver, factory.componentType, matched.leftOverUrl)
.then(children => [new TreeNode<RouteSegment>(segment, children)]); .then(children => [new TreeNode<RouteSegment>(segment, children)]);
} else { } else {
return _recognizeLeftOvers(componentResolver, matched.component) return _recognizeLeftOvers(componentResolver, factory.componentType)
.then(children => [new TreeNode<RouteSegment>(segment, children)]); .then(children => [new TreeNode<RouteSegment>(segment, children)]);
} }
}); });
} }
function _recognizeLeftOvers(componentResolver: ComponentResolver, function _recognizeLeftOvers(componentResolver: ComponentResolver,
parentType: Type): Promise<TreeNode<RouteSegment>[]> { parentComponent: Type): Promise<TreeNode<RouteSegment>[]> {
return componentResolver.resolveComponent(parentType) return componentResolver.resolveComponent(parentComponent)
.then(factory => { .then(factory => {
let metadata = _readMetadata(parentType); let metadata = _readMetadata(factory.componentType);
if (isBlank(metadata)) { if (isBlank(metadata)) {
return []; return [];
} }
@ -165,7 +165,7 @@ function _checkOutletNameUniqueness(nodes: TreeNode<RouteSegment>[]): TreeNode<R
} }
class _MatchResult { class _MatchResult {
constructor(public component: Type, public consumedUrlSegments: UrlSegment[], constructor(public component: Type|string, public consumedUrlSegments: UrlSegment[],
public parameters: {[key: string]: string}, public parameters: {[key: string]: string},
public leftOverUrl: TreeNode<UrlSegment>[], public aux: TreeNode<UrlSegment>[]) {} public leftOverUrl: TreeNode<UrlSegment>[], public aux: TreeNode<UrlSegment>[]) {}
} }