2015-05-15 02:05:57 -07:00
|
|
|
|
import {RouteRecognizer, RouteMatch} from './route_recognizer';
|
2015-05-29 14:58:41 -07:00
|
|
|
|
import {Instruction} from './instruction';
|
|
|
|
|
import {
|
|
|
|
|
List,
|
|
|
|
|
ListWrapper,
|
|
|
|
|
Map,
|
|
|
|
|
MapWrapper,
|
|
|
|
|
StringMap,
|
|
|
|
|
StringMapWrapper
|
|
|
|
|
} from 'angular2/src/facade/collection';
|
2015-05-21 13:59:14 -07:00
|
|
|
|
import {Promise, PromiseWrapper} from 'angular2/src/facade/async';
|
|
|
|
|
import {
|
|
|
|
|
isPresent,
|
|
|
|
|
isBlank,
|
|
|
|
|
isType,
|
2015-06-30 13:18:51 -07:00
|
|
|
|
isString,
|
2015-06-11 19:32:55 +02:00
|
|
|
|
isStringMap,
|
2015-05-21 13:59:14 -07:00
|
|
|
|
isFunction,
|
|
|
|
|
StringWrapper,
|
|
|
|
|
BaseException
|
|
|
|
|
} from 'angular2/src/facade/lang';
|
2015-05-04 15:39:14 -07:00
|
|
|
|
import {RouteConfig} from './route_config_impl';
|
2015-04-17 09:59:56 -07:00
|
|
|
|
import {reflector} from 'angular2/src/reflection/reflection';
|
2015-06-29 10:22:00 +02:00
|
|
|
|
import {Injectable} from 'angular2/di';
|
2015-04-17 09:59:56 -07:00
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
/**
|
|
|
|
|
* The RouteRegistry holds route configurations for each component in an Angular app.
|
2015-05-29 14:58:41 -07:00
|
|
|
|
* It is responsible for creating Instructions from URLs, and generating URLs based on route and
|
|
|
|
|
* parameters.
|
2015-05-15 02:05:57 -07:00
|
|
|
|
*/
|
2015-06-29 10:22:00 +02:00
|
|
|
|
@Injectable()
|
2015-04-17 09:59:56 -07:00
|
|
|
|
export class RouteRegistry {
|
2015-06-30 13:18:51 -07:00
|
|
|
|
private _rules: Map<any, RouteRecognizer> = new Map();
|
|
|
|
|
|
|
|
|
|
constructor(private _rootHostComponent: any) {}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
/**
|
|
|
|
|
* Given a component and a configuration object, add the route to this registry
|
|
|
|
|
*/
|
2015-05-29 14:58:41 -07:00
|
|
|
|
config(parentComponent, config: StringMap<string, any>): void {
|
2015-05-21 13:59:14 -07:00
|
|
|
|
assertValidConfig(config);
|
2015-05-01 15:50:12 -07:00
|
|
|
|
|
2015-06-17 16:21:40 -07:00
|
|
|
|
var recognizer: RouteRecognizer = this._rules.get(parentComponent);
|
2015-05-14 15:24:35 +02:00
|
|
|
|
|
|
|
|
|
if (isBlank(recognizer)) {
|
2015-04-17 09:59:56 -07:00
|
|
|
|
recognizer = new RouteRecognizer();
|
2015-06-17 16:21:40 -07:00
|
|
|
|
this._rules.set(parentComponent, recognizer);
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-05-03 20:25:26 -07:00
|
|
|
|
if (StringMapWrapper.contains(config, 'redirectTo')) {
|
2015-05-15 02:05:57 -07:00
|
|
|
|
recognizer.addRedirect(config['path'], config['redirectTo']);
|
2015-05-03 20:25:26 -07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-21 13:59:14 -07:00
|
|
|
|
config = StringMapWrapper.merge(
|
|
|
|
|
config, {'component': normalizeComponentDeclaration(config['component'])});
|
|
|
|
|
|
|
|
|
|
var component = config['component'];
|
2015-06-17 11:57:38 -07:00
|
|
|
|
var terminal = recognizer.addConfig(config['path'], config, config['as']);
|
2015-04-17 09:59:56 -07:00
|
|
|
|
|
2015-06-17 11:57:38 -07:00
|
|
|
|
if (component['type'] == 'constructor') {
|
|
|
|
|
if (terminal) {
|
|
|
|
|
assertTerminalComponent(component['constructor'], config['path']);
|
|
|
|
|
} else {
|
|
|
|
|
this.configFromComponent(component['constructor']);
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
/**
|
|
|
|
|
* Reads the annotations of a component and configures the registry based on them
|
|
|
|
|
*/
|
2015-05-14 15:24:35 +02:00
|
|
|
|
configFromComponent(component): void {
|
2015-04-17 09:59:56 -07:00
|
|
|
|
if (!isType(component)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Don't read the annotations from a type more than once –
|
|
|
|
|
// this prevents an infinite loop if a component routes recursively.
|
2015-06-17 21:42:56 -07:00
|
|
|
|
if (this._rules.has(component)) {
|
2015-04-17 09:59:56 -07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
var annotations = reflector.annotations(component);
|
|
|
|
|
if (isPresent(annotations)) {
|
2015-05-29 14:58:41 -07:00
|
|
|
|
for (var i = 0; i < annotations.length; i++) {
|
2015-04-17 09:59:56 -07:00
|
|
|
|
var annotation = annotations[i];
|
|
|
|
|
|
|
|
|
|
if (annotation instanceof RouteConfig) {
|
2015-05-15 02:05:57 -07:00
|
|
|
|
ListWrapper.forEach(annotation.configs, (config) => this.config(component, config));
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
/**
|
|
|
|
|
* Given a URL and a parent component, return the most specific instruction for navigating
|
|
|
|
|
* the application into the state specified by the
|
|
|
|
|
*/
|
2015-05-21 13:59:14 -07:00
|
|
|
|
recognize(url: string, parentComponent): Promise<Instruction> {
|
2015-06-17 16:21:40 -07:00
|
|
|
|
var componentRecognizer = this._rules.get(parentComponent);
|
2015-04-17 09:59:56 -07:00
|
|
|
|
if (isBlank(componentRecognizer)) {
|
2015-05-21 13:59:14 -07:00
|
|
|
|
return PromiseWrapper.resolve(null);
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
// Matches some beginning part of the given URL
|
|
|
|
|
var possibleMatches = componentRecognizer.recognize(url);
|
2015-05-21 13:59:14 -07:00
|
|
|
|
var matchPromises =
|
|
|
|
|
ListWrapper.map(possibleMatches, (candidate) => this._completeRouteMatch(candidate));
|
2015-05-15 02:05:57 -07:00
|
|
|
|
|
2015-05-21 13:59:14 -07:00
|
|
|
|
return PromiseWrapper.all(matchPromises)
|
2015-06-26 11:10:52 -07:00
|
|
|
|
.then((solutions: List<Instruction>) => {
|
2015-05-21 13:59:14 -07:00
|
|
|
|
// remove nulls
|
|
|
|
|
var fullSolutions = ListWrapper.filter(solutions, (solution) => isPresent(solution));
|
|
|
|
|
|
|
|
|
|
if (fullSolutions.length > 0) {
|
|
|
|
|
return mostSpecific(fullSolutions);
|
2015-05-12 14:53:13 -07:00
|
|
|
|
}
|
2015-05-21 13:59:14 -07:00
|
|
|
|
return null;
|
|
|
|
|
});
|
|
|
|
|
}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
|
|
|
|
|
|
2015-06-30 13:18:51 -07:00
|
|
|
|
_completeRouteMatch(partialMatch: RouteMatch): Promise<Instruction> {
|
|
|
|
|
var recognizer = partialMatch.recognizer;
|
|
|
|
|
var handler = recognizer.handler;
|
|
|
|
|
return handler.resolveComponentType().then((componentType) => {
|
|
|
|
|
this.configFromComponent(componentType);
|
2015-05-14 15:38:16 +02:00
|
|
|
|
|
2015-06-30 13:18:51 -07:00
|
|
|
|
if (partialMatch.unmatchedUrl.length == 0) {
|
|
|
|
|
return new Instruction(componentType, partialMatch.matchedUrl, recognizer);
|
|
|
|
|
}
|
2015-05-12 14:53:13 -07:00
|
|
|
|
|
2015-06-30 13:18:51 -07:00
|
|
|
|
return this.recognize(partialMatch.unmatchedUrl, componentType)
|
|
|
|
|
.then(childInstruction => {
|
|
|
|
|
if (isBlank(childInstruction)) {
|
|
|
|
|
return null;
|
|
|
|
|
} else {
|
|
|
|
|
return new Instruction(componentType, partialMatch.matchedUrl, recognizer,
|
|
|
|
|
childInstruction);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
});
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
|
2015-06-30 13:18:51 -07:00
|
|
|
|
/**
|
|
|
|
|
* Given a list with component names and params like: `['./user', {id: 3 }]`
|
|
|
|
|
* generates a url with a leading slash relative to the provided `parentComponent`.
|
|
|
|
|
*/
|
|
|
|
|
generate(linkParams: List<any>, parentComponent): string {
|
|
|
|
|
let normalizedLinkParams = splitAndFlattenLinkParams(linkParams);
|
|
|
|
|
let url = '/';
|
|
|
|
|
|
|
|
|
|
let componentCursor = parentComponent;
|
|
|
|
|
|
|
|
|
|
// The first segment should be either '.' (generate from parent) or '' (generate from root).
|
|
|
|
|
// When we normalize above, we strip all the slashes, './' becomes '.' and '/' becomes ''.
|
|
|
|
|
if (normalizedLinkParams[0] == '') {
|
|
|
|
|
componentCursor = this._rootHostComponent;
|
|
|
|
|
} else if (normalizedLinkParams[0] != '.') {
|
|
|
|
|
throw new BaseException(
|
|
|
|
|
`Link "${ListWrapper.toJSON(linkParams)}" must start with "/" or "./"`);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (normalizedLinkParams[normalizedLinkParams.length - 1] == '') {
|
|
|
|
|
ListWrapper.removeLast(normalizedLinkParams);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (normalizedLinkParams.length < 2) {
|
2015-06-30 20:38:08 -07:00
|
|
|
|
let msg = `Link "${ListWrapper.toJSON(linkParams)}" must include a route name.`;
|
|
|
|
|
throw new BaseException(msg);
|
2015-06-30 13:18:51 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (let i = 1; i < normalizedLinkParams.length; i += 1) {
|
|
|
|
|
let segment = normalizedLinkParams[i];
|
|
|
|
|
if (!isString(segment)) {
|
|
|
|
|
throw new BaseException(`Unexpected segment "${segment}" in link DSL. Expected a string.`);
|
|
|
|
|
}
|
|
|
|
|
let params = null;
|
|
|
|
|
if (i + 1 < normalizedLinkParams.length) {
|
|
|
|
|
let nextSegment = normalizedLinkParams[i + 1];
|
|
|
|
|
if (isStringMap(nextSegment)) {
|
|
|
|
|
params = nextSegment;
|
|
|
|
|
i += 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var componentRecognizer = this._rules.get(componentCursor);
|
|
|
|
|
if (isBlank(componentRecognizer)) {
|
|
|
|
|
throw new BaseException(`Could not find route config for "${segment}".`);
|
|
|
|
|
}
|
|
|
|
|
var response = componentRecognizer.generate(segment, params);
|
|
|
|
|
url += response['url'];
|
|
|
|
|
componentCursor = response['nextComponent'];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return url;
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-05-21 13:59:14 -07:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* A config should have a "path" property, and exactly one of:
|
|
|
|
|
* - `component`
|
|
|
|
|
* - `redirectTo`
|
|
|
|
|
*/
|
|
|
|
|
var ALLOWED_TARGETS = ['component', 'redirectTo'];
|
|
|
|
|
function assertValidConfig(config: StringMap<string, any>): void {
|
|
|
|
|
if (!StringMapWrapper.contains(config, 'path')) {
|
|
|
|
|
throw new BaseException(`Route config should contain a "path" property`);
|
|
|
|
|
}
|
|
|
|
|
var targets = 0;
|
|
|
|
|
ListWrapper.forEach(ALLOWED_TARGETS, (target) => {
|
|
|
|
|
if (StringMapWrapper.contains(config, target)) {
|
|
|
|
|
targets += 1;
|
|
|
|
|
}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
});
|
2015-05-21 13:59:14 -07:00
|
|
|
|
if (targets != 1) {
|
|
|
|
|
throw new BaseException(
|
|
|
|
|
`Route config should contain exactly one 'component', or 'redirectTo' property`);
|
|
|
|
|
}
|
2015-04-17 09:59:56 -07:00
|
|
|
|
}
|
2015-04-29 15:47:12 -07:00
|
|
|
|
|
2015-05-15 02:05:57 -07:00
|
|
|
|
/*
|
2015-05-21 13:59:14 -07:00
|
|
|
|
* Returns a StringMap like: `{ 'constructor': SomeType, 'type': 'constructor' }`
|
2015-05-15 02:05:57 -07:00
|
|
|
|
*/
|
2015-05-21 13:59:14 -07:00
|
|
|
|
var VALID_COMPONENT_TYPES = ['constructor', 'loader'];
|
|
|
|
|
function normalizeComponentDeclaration(config: any): StringMap<string, any> {
|
|
|
|
|
if (isType(config)) {
|
|
|
|
|
return {'constructor': config, 'type': 'constructor'};
|
2015-06-11 19:32:55 +02:00
|
|
|
|
} else if (isStringMap(config)) {
|
2015-05-21 13:59:14 -07:00
|
|
|
|
if (isBlank(config['type'])) {
|
|
|
|
|
throw new BaseException(
|
|
|
|
|
`Component declaration when provided as a map should include a 'type' property`);
|
|
|
|
|
}
|
|
|
|
|
var componentType = config['type'];
|
|
|
|
|
if (!ListWrapper.contains(VALID_COMPONENT_TYPES, componentType)) {
|
|
|
|
|
throw new BaseException(`Invalid component type '${componentType}'`);
|
|
|
|
|
}
|
2015-05-15 02:05:57 -07:00
|
|
|
|
return config;
|
2015-05-21 13:59:14 -07:00
|
|
|
|
} else {
|
|
|
|
|
throw new BaseException(`Component declaration should be either a Map or a Type`);
|
2015-05-15 02:05:57 -07:00
|
|
|
|
}
|
2015-05-21 13:59:14 -07:00
|
|
|
|
}
|
2015-04-29 15:47:12 -07:00
|
|
|
|
|
2015-05-21 13:59:14 -07:00
|
|
|
|
/*
|
|
|
|
|
* Given a list of instructions, returns the most specific instruction
|
|
|
|
|
*/
|
|
|
|
|
function mostSpecific(instructions: List<Instruction>): Instruction {
|
|
|
|
|
var mostSpecificSolution = instructions[0];
|
|
|
|
|
for (var solutionIndex = 1; solutionIndex < instructions.length; solutionIndex++) {
|
|
|
|
|
var solution = instructions[solutionIndex];
|
|
|
|
|
if (solution.specificity > mostSpecificSolution.specificity) {
|
|
|
|
|
mostSpecificSolution = solution;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return mostSpecificSolution;
|
2015-04-29 15:47:12 -07:00
|
|
|
|
}
|
2015-06-17 11:57:38 -07:00
|
|
|
|
|
|
|
|
|
function assertTerminalComponent(component, path) {
|
|
|
|
|
if (!isType(component)) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var annotations = reflector.annotations(component);
|
|
|
|
|
if (isPresent(annotations)) {
|
|
|
|
|
for (var i = 0; i < annotations.length; i++) {
|
|
|
|
|
var annotation = annotations[i];
|
|
|
|
|
|
|
|
|
|
if (annotation instanceof RouteConfig) {
|
|
|
|
|
throw new BaseException(
|
|
|
|
|
`Child routes are not allowed for "${path}". Use "..." on the parent's route path.`);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-30 13:18:51 -07:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Given: ['/a/b', {c: 2}]
|
|
|
|
|
* Returns: ['', 'a', 'b', {c: 2}]
|
|
|
|
|
*/
|
|
|
|
|
var SLASH = new RegExp('/');
|
|
|
|
|
function splitAndFlattenLinkParams(linkParams: List<any>): List<any> {
|
|
|
|
|
return ListWrapper.reduce(linkParams, (accumulation, item) => {
|
|
|
|
|
if (isString(item)) {
|
|
|
|
|
return ListWrapper.concat(accumulation, StringWrapper.split(item, SLASH));
|
|
|
|
|
}
|
|
|
|
|
accumulation.push(item);
|
|
|
|
|
return accumulation;
|
|
|
|
|
}, []);
|
|
|
|
|
}
|