feat(router): update recognize to support aux routes

This commit is contained in:
vsavkin
2016-04-25 16:57:15 -07:00
committed by Victor Savkin
parent fad3b6434c
commit d35c109cb9
3 changed files with 186 additions and 43 deletions

View File

@ -0,0 +1 @@
export const DEFAULT_OUTLET_NAME = "__DEFAULT";

View File

@ -1,46 +1,66 @@
import {RouteSegment, UrlSegment, Tree} from './segments';
import {RouteSegment, UrlSegment, Tree, TreeNode, rootNode} from './segments';
import {RoutesMetadata, RouteMetadata} from './metadata/metadata';
import {Type, isPresent, stringify} from 'angular2/src/facade/lang';
import {Type, isBlank, isPresent, stringify} from 'angular2/src/facade/lang';
import {ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';
import {PromiseWrapper} from 'angular2/src/facade/promise';
import {BaseException} from 'angular2/src/facade/exceptions';
import {ComponentResolver} from 'angular2/core';
import {DEFAULT_OUTLET_NAME} from './constants';
import {reflector} from 'angular2/src/core/reflection/reflection';
export function recognize(componentResolver: ComponentResolver, type: Type,
url: Tree<UrlSegment>): Promise<Tree<RouteSegment>> {
return _recognize(componentResolver, type, url, url.root)
.then(nodes => new Tree<RouteSegment>(nodes));
return componentResolver.resolveComponent(type).then(factory => {
let segment =
new RouteSegment([url.root], url.root.parameters, DEFAULT_OUTLET_NAME, type, factory);
return _recognizeMany(componentResolver, type, rootNode(url).children)
.then(children => new Tree<RouteSegment>(new TreeNode<RouteSegment>(segment, children)));
});
}
function _recognize(componentResolver: ComponentResolver, type: Type, url: Tree<UrlSegment>,
current: UrlSegment): Promise<RouteSegment[]> {
let metadata = _readMetadata(type); // should read from the factory instead
function _recognize(componentResolver: ComponentResolver, parentType: Type,
url: TreeNode<UrlSegment>): Promise<TreeNode<RouteSegment>[]> {
let metadata = _readMetadata(parentType); // should read from the factory instead
let matched;
let match;
try {
matched = _match(metadata, url, current);
match = _match(metadata, url);
} catch (e) {
return PromiseWrapper.reject(e, null);
}
let main = _constructSegment(componentResolver, match);
let aux =
_recognizeMany(componentResolver, parentType, match.aux).then(_checkOutletNameUniqueness);
return PromiseWrapper.all([main, aux]).then(ListWrapper.flatten);
}
function _recognizeMany(componentResolver: ComponentResolver, parentType: Type,
urls: TreeNode<UrlSegment>[]): Promise<TreeNode<RouteSegment>[]> {
let recognized = urls.map(u => _recognize(componentResolver, parentType, u));
return PromiseWrapper.all(recognized).then(ListWrapper.flatten);
}
function _constructSegment(componentResolver: ComponentResolver,
matched: _MatchResult): Promise<TreeNode<RouteSegment>[]> {
return componentResolver.resolveComponent(matched.route.component)
.then(factory => {
let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters, "",
let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters,
matched.consumedUrlSegments[0].outlet,
matched.route.component, factory);
if (isPresent(matched.leftOver)) {
return _recognize(componentResolver, matched.route.component, url, matched.leftOver)
.then(children => [segment].concat(children));
if (isPresent(matched.leftOverUrl)) {
return _recognize(componentResolver, matched.route.component, matched.leftOverUrl)
.then(children => [new TreeNode<RouteSegment>(segment, children)]);
} else {
return [segment];
return [new TreeNode<RouteSegment>(segment, [])];
}
});
}
function _match(metadata: RoutesMetadata, url: Tree<UrlSegment>,
current: UrlSegment): _MatchingResult {
function _match(metadata: RoutesMetadata, url: TreeNode<UrlSegment>): _MatchResult {
for (let r of metadata.routes) {
let matchingResult = _matchWithParts(r, url, current);
let matchingResult = _matchWithParts(r, url);
if (isPresent(matchingResult)) {
return matchingResult;
}
@ -48,30 +68,63 @@ function _match(metadata: RoutesMetadata, url: Tree<UrlSegment>,
throw new BaseException("Cannot match any routes");
}
function _matchWithParts(route: RouteMetadata, url: Tree<UrlSegment>,
current: UrlSegment): _MatchingResult {
function _matchWithParts(route: RouteMetadata, url: TreeNode<UrlSegment>): _MatchResult {
let parts = route.path.split("/");
let parameters = {};
let positionalParams = {};
let consumedUrlSegments = [];
let u = current;
let lastParent: TreeNode<UrlSegment> = null;
let lastSegment: TreeNode<UrlSegment> = null;
let current = url;
for (let i = 0; i < parts.length; ++i) {
consumedUrlSegments.push(u);
let p = parts[i];
if (p.startsWith(":")) {
let segment = u.segment;
parameters[p.substring(1)] = segment;
} else if (p != u.segment) {
return null;
let isLastSegment = i === parts.length - 1;
let isLastParent = i === parts.length - 2;
let isPosParam = p.startsWith(":");
if (isBlank(current)) return null;
if (!isPosParam && p != current.value.segment) return null;
if (isLastSegment) {
lastSegment = current;
}
u = url.firstChild(u);
if (isLastParent) {
lastParent = current;
}
if (isPosParam) {
positionalParams[p.substring(1)] = current.value.segment;
}
consumedUrlSegments.push(current.value);
current = ListWrapper.first(current.children);
}
return new _MatchingResult(route, consumedUrlSegments, parameters, u);
let parameters = <{[key: string]: string}>StringMapWrapper.merge(lastSegment.value.parameters,
positionalParams);
let axuUrlSubtrees = isPresent(lastParent) ? lastParent.children.slice(1) : [];
return new _MatchResult(route, consumedUrlSegments, parameters, current, axuUrlSubtrees);
}
class _MatchingResult {
function _checkOutletNameUniqueness(nodes: TreeNode<RouteSegment>[]): TreeNode<RouteSegment>[] {
let names = {};
nodes.forEach(n => {
let segmentWithSameOutletName = names[n.value.outlet];
if (isPresent(segmentWithSameOutletName)) {
let p = segmentWithSameOutletName.stringifiedUrlSegments;
let c = n.value.stringifiedUrlSegments;
throw new BaseException(`Two segments cannot have the same outlet name: '${p}' and '${c}'.`);
}
names[n.value.outlet] = n.value;
});
return nodes;
}
class _MatchResult {
constructor(public route: RouteMetadata, public consumedUrlSegments: UrlSegment[],
public parameters: {[key: string]: string}, public leftOver: UrlSegment) {}
public parameters: {[key: string]: string}, public leftOverUrl: TreeNode<UrlSegment>,
public aux: TreeNode<UrlSegment>[]) {}
}
function _readMetadata(componentType: Type) {