These wrappers are not natively understood by ts2dart. Removing them will improve Dart2JS compilation due to fewer megamorphic calls to List functions. It also makes Angular code more succinct and improves type safety in Angular due to better type inference of the Array component type. This change exposed several bugs in Angular.
121 lines
3.5 KiB
TypeScript
121 lines
3.5 KiB
TypeScript
import {
|
|
RegExp,
|
|
RegExpWrapper,
|
|
StringWrapper,
|
|
isPresent,
|
|
BaseException
|
|
} from 'angular2/src/facade/lang';
|
|
import {
|
|
Map,
|
|
MapWrapper,
|
|
List,
|
|
ListWrapper,
|
|
StringMap,
|
|
StringMapWrapper
|
|
} from 'angular2/src/facade/collection';
|
|
|
|
import {PathRecognizer} from './path_recognizer';
|
|
|
|
/**
|
|
* `RouteRecognizer` is responsible for recognizing routes for a single component.
|
|
* It is consumed by `RouteRegistry`, which knows how to recognize an entire hierarchy of
|
|
* components.
|
|
*/
|
|
export class RouteRecognizer {
|
|
names: Map<string, PathRecognizer>;
|
|
redirects: Map<string, string>;
|
|
matchers: Map<RegExp, PathRecognizer>;
|
|
|
|
constructor() {
|
|
this.names = MapWrapper.create();
|
|
this.matchers = MapWrapper.create();
|
|
this.redirects = MapWrapper.create();
|
|
}
|
|
|
|
addRedirect(path: string, target: string): void { MapWrapper.set(this.redirects, path, target); }
|
|
|
|
addConfig(path: string, handler: any, alias: string = null): void {
|
|
var recognizer = new PathRecognizer(path, handler);
|
|
MapWrapper.forEach(this.matchers, (matcher, _) => {
|
|
if (recognizer.regex.toString() == matcher.regex.toString()) {
|
|
throw new BaseException(
|
|
`Configuration '${path}' conflicts with existing route '${matcher.path}'`);
|
|
}
|
|
});
|
|
MapWrapper.set(this.matchers, recognizer.regex, recognizer);
|
|
if (isPresent(alias)) {
|
|
MapWrapper.set(this.names, alias, recognizer);
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
* Given a URL, returns a list of `RouteMatch`es, which are partial recognitions for some route.
|
|
*
|
|
*/
|
|
recognize(url: string): List<RouteMatch> {
|
|
var solutions = [];
|
|
|
|
MapWrapper.forEach(this.redirects, (target, path) => {
|
|
// "/" redirect case
|
|
if (path == '/' || path == '') {
|
|
if (path == url) {
|
|
url = target;
|
|
}
|
|
} else if (StringWrapper.startsWith(url, path)) {
|
|
url = target + StringWrapper.substring(url, path.length);
|
|
}
|
|
});
|
|
|
|
MapWrapper.forEach(this.matchers, (pathRecognizer, regex) => {
|
|
var match;
|
|
if (isPresent(match = RegExpWrapper.firstMatch(regex, url))) {
|
|
// TODO(btford): determine a good generic way to deal with terminal matches
|
|
var matchedUrl = '/';
|
|
var unmatchedUrl = '';
|
|
if (url != '/') {
|
|
matchedUrl = match[0];
|
|
unmatchedUrl = StringWrapper.substring(url, match[0].length);
|
|
}
|
|
solutions.push(new RouteMatch({
|
|
specificity: pathRecognizer.specificity,
|
|
handler: pathRecognizer.handler,
|
|
params: pathRecognizer.parseParams(url),
|
|
matchedUrl: matchedUrl,
|
|
unmatchedUrl: unmatchedUrl
|
|
}));
|
|
}
|
|
});
|
|
|
|
return solutions;
|
|
}
|
|
|
|
hasRoute(name: string): boolean { return MapWrapper.contains(this.names, name); }
|
|
|
|
generate(name: string, params: any): string {
|
|
var pathRecognizer = MapWrapper.get(this.names, name);
|
|
return isPresent(pathRecognizer) ? pathRecognizer.generate(params) : null;
|
|
}
|
|
}
|
|
|
|
export class RouteMatch {
|
|
specificity: number;
|
|
handler: StringMap<string, any>;
|
|
params: StringMap<string, string>;
|
|
matchedUrl: string;
|
|
unmatchedUrl: string;
|
|
constructor({specificity, handler, params, matchedUrl, unmatchedUrl}: {
|
|
specificity?: number,
|
|
handler?: StringMap<string, any>,
|
|
params?: StringMap<string, string>,
|
|
matchedUrl?: string,
|
|
unmatchedUrl?: string
|
|
} = {}) {
|
|
this.specificity = specificity;
|
|
this.handler = handler;
|
|
this.params = params;
|
|
this.matchedUrl = matchedUrl;
|
|
this.unmatchedUrl = unmatchedUrl;
|
|
}
|
|
}
|