cleanup(router): make analyzer happy

Closes #8220
This commit is contained in:
vsavkin 2016-04-28 18:33:48 -07:00 committed by Victor Savkin
parent 602641dffd
commit 30de2db349
17 changed files with 174 additions and 646 deletions

View File

@ -5,7 +5,7 @@
*/ */
export {Router, RouterOutletMap} from './src/alt_router/router'; export {Router, RouterOutletMap} from './src/alt_router/router';
export {RouteSegment} from './src/alt_router/segments'; export {RouteSegment, UrlSegment, Tree} from './src/alt_router/segments';
export {Routes} from './src/alt_router/metadata/decorators'; export {Routes} from './src/alt_router/metadata/decorators';
export {Route} from './src/alt_router/metadata/metadata'; export {Route} from './src/alt_router/metadata/metadata';
export { export {
@ -13,13 +13,7 @@ export {
DefaultRouterUrlSerializer DefaultRouterUrlSerializer
} from './src/alt_router/router_url_serializer'; } from './src/alt_router/router_url_serializer';
export {OnActivate} from './src/alt_router/interfaces'; export {OnActivate} from './src/alt_router/interfaces';
export {ROUTER_PROVIDERS} from './src/alt_router/router_providers';
export {Location} from './src/alt_router/location/location';
export {LocationStrategy} from './src/alt_router/location/location_strategy';
export {PathLocationStrategy} from './src/alt_router/location/path_location_strategy';
export {HashLocationStrategy} from './src/alt_router/location/hash_location_strategy';
export {PlatformLocation} from './src/alt_router/location/platform_location';
export {BrowserPlatformLocation} from './src/alt_router/location/browser_platform_location';
import {RouterOutlet} from './src/alt_router/directives/router_outlet'; import {RouterOutlet} from './src/alt_router/directives/router_outlet';
import {RouterLink} from './src/alt_router/directives/router_link'; import {RouterLink} from './src/alt_router/directives/router_link';

View File

@ -28,7 +28,7 @@ export class RouterLink implements OnDestroy {
@HostBinding() private href: string; @HostBinding() private href: string;
constructor(private _router: Router, private _segment: RouteSegment) { constructor(private _router: Router) {
this._subscription = ObservableWrapper.subscribe(_router.changes, (_) => { this._subscription = ObservableWrapper.subscribe(_router.changes, (_) => {
this._targetUrl = _router.urlTree; this._targetUrl = _router.urlTree;
this._updateTargetUrlAndHref(); this._updateTargetUrlAndHref();
@ -53,7 +53,7 @@ export class RouterLink implements OnDestroy {
} }
private _updateTargetUrlAndHref(): void { private _updateTargetUrlAndHref(): void {
this._targetUrl = link(this._segment, this._router.urlTree, this._changes); this._targetUrl = link(null, this._router.urlTree, this._changes);
this.href = this._router.serializeUrl(this._targetUrl); this.href = this._router.serializeUrl(this._targetUrl);
} }
} }

View File

@ -1,57 +0,0 @@
import {Injectable} from 'angular2/src/core/di/decorators';
import {UrlChangeListener, PlatformLocation} from './platform_location';
import {History, Location} from 'angular2/src/facade/browser';
import {DOM} from 'angular2/src/platform/dom/dom_adapter';
/**
* `PlatformLocation` encapsulates all of the direct calls to platform APIs.
* This class should not be used directly by an application developer. Instead, use
* {@link Location}.
*/
@Injectable()
export class BrowserPlatformLocation extends PlatformLocation {
private _location: Location;
private _history: History;
constructor() {
super();
this._init();
}
// This is moved to its own method so that `MockPlatformLocationStrategy` can overwrite it
/** @internal */
_init() {
this._location = DOM.getLocation();
this._history = DOM.getHistory();
}
/** @internal */
get location(): Location { return this._location; }
getBaseHrefFromDOM(): string { return DOM.getBaseHref(); }
onPopState(fn: UrlChangeListener): void {
DOM.getGlobalEventTarget('window').addEventListener('popstate', fn, false);
}
onHashChange(fn: UrlChangeListener): void {
DOM.getGlobalEventTarget('window').addEventListener('hashchange', fn, false);
}
get pathname(): string { return this._location.pathname; }
get search(): string { return this._location.search; }
get hash(): string { return this._location.hash; }
set pathname(newPath: string) { this._location.pathname = newPath; }
pushState(state: any, title: string, url: string): void {
this._history.pushState(state, title, url);
}
replaceState(state: any, title: string, url: string): void {
this._history.replaceState(state, title, url);
}
forward(): void { this._history.forward(); }
back(): void { this._history.back(); }
}

View File

@ -1,101 +0,0 @@
import {Injectable, Inject, Optional} from 'angular2/core';
import {LocationStrategy, APP_BASE_HREF} from './location_strategy';
import {Location} from './location';
import {UrlChangeListener, PlatformLocation} from './platform_location';
import {isPresent} from 'angular2/src/facade/lang';
/**
* `HashLocationStrategy` is a {@link LocationStrategy} used to configure the
* {@link Location} service to represent its state in the
* [hash fragment](https://en.wikipedia.org/wiki/Uniform_Resource_Locator#Syntax)
* of the browser's URL.
*
* For instance, if you call `location.go('/foo')`, the browser's URL will become
* `example.com#/foo`.
*
* ### Example
*
* ```
* import {Component, provide} from 'angular2/core';
* import {
* Location,
* LocationStrategy,
* HashLocationStrategy
* } from 'angular2/platform/common';
* import {
* ROUTER_DIRECTIVES,
* ROUTER_PROVIDERS,
* RouteConfig
* } from 'angular2/router';
*
* @Component({directives: [ROUTER_DIRECTIVES]})
* @RouteConfig([
* {...},
* ])
* class AppCmp {
* constructor(location: Location) {
* location.go('/foo');
* }
* }
*
* bootstrap(AppCmp, [
* ROUTER_PROVIDERS,
* provide(LocationStrategy, {useClass: HashLocationStrategy})
* ]);
* ```
*/
@Injectable()
export class HashLocationStrategy extends LocationStrategy {
private _baseHref: string = '';
constructor(private _platformLocation: PlatformLocation,
@Optional() @Inject(APP_BASE_HREF) _baseHref?: string) {
super();
if (isPresent(_baseHref)) {
this._baseHref = _baseHref;
}
}
onPopState(fn: UrlChangeListener): void {
this._platformLocation.onPopState(fn);
this._platformLocation.onHashChange(fn);
}
getBaseHref(): string { return this._baseHref; }
path(): string {
// the hash value is always prefixed with a `#`
// and if it is empty then it will stay empty
var path = this._platformLocation.hash;
if (!isPresent(path)) path = '#';
// Dart will complain if a call to substring is
// executed with a position value that extends the
// length of string.
return (path.length > 0 ? path.substring(1) : path);
}
prepareExternalUrl(internal: string): string {
var url = Location.joinWithSlash(this._baseHref, internal);
return url.length > 0 ? ('#' + url) : url;
}
pushState(state: any, title: string, path: string, queryParams: string) {
var url = this.prepareExternalUrl(path + Location.normalizeQueryParams(queryParams));
if (url.length == 0) {
url = this._platformLocation.pathname;
}
this._platformLocation.pushState(state, title, url);
}
replaceState(state: any, title: string, path: string, queryParams: string) {
var url = this.prepareExternalUrl(path + Location.normalizeQueryParams(queryParams));
if (url.length == 0) {
url = this._platformLocation.pathname;
}
this._platformLocation.replaceState(state, title, url);
}
forward(): void { this._platformLocation.forward(); }
back(): void { this._platformLocation.back(); }
}

View File

@ -1,179 +0,0 @@
import {EventEmitter, ObservableWrapper} from 'angular2/src/facade/async';
import {Injectable, Inject} from 'angular2/core';
import {LocationStrategy} from './location_strategy';
/**
* `Location` is a service that applications can use to interact with a browser's URL.
* Depending on which {@link LocationStrategy} is used, `Location` will either persist
* to the URL's path or the URL's hash segment.
*
* Note: it's better to use {@link Router#navigate} service to trigger route changes. Use
* `Location` only if you need to interact with or create normalized URLs outside of
* routing.
*
* `Location` is responsible for normalizing the URL against the application's base href.
* A normalized URL is absolute from the URL host, includes the application's base href, and has no
* trailing slash:
* - `/my/app/user/123` is normalized
* - `my/app/user/123` **is not** normalized
* - `/my/app/user/123/` **is not** normalized
*
* ### Example
*
* ```
* import {Component} from 'angular2/core';
* import {Location} from 'angular2/platform/common';
* import {
* ROUTER_DIRECTIVES,
* ROUTER_PROVIDERS,
* RouteConfig
* } from 'angular2/router';
*
* @Component({directives: [ROUTER_DIRECTIVES]})
* @RouteConfig([
* {...},
* ])
* class AppCmp {
* constructor(location: Location) {
* location.go('/foo');
* }
* }
*
* bootstrap(AppCmp, [ROUTER_PROVIDERS]);
* ```
*/
@Injectable()
export class Location {
/** @internal */
_subject: EventEmitter<any> = new EventEmitter();
/** @internal */
_baseHref: string;
constructor(public platformStrategy: LocationStrategy) {
var browserBaseHref = this.platformStrategy.getBaseHref();
this._baseHref = Location.stripTrailingSlash(_stripIndexHtml(browserBaseHref));
this.platformStrategy.onPopState((ev) => {
ObservableWrapper.callEmit(this._subject, {'url': this.path(), 'pop': true, 'type': ev.type});
});
}
/**
* Returns the normalized URL path.
*/
path(): string { return this.normalize(this.platformStrategy.path()); }
/**
* Given a string representing a URL, returns the normalized URL path without leading or
* trailing slashes
*/
normalize(url: string): string {
return Location.stripTrailingSlash(_stripBaseHref(this._baseHref, _stripIndexHtml(url)));
}
/**
* Given a string representing a URL, returns the platform-specific external URL path.
* If the given URL doesn't begin with a leading slash (`'/'`), this method adds one
* before normalizing. This method will also add a hash if `HashLocationStrategy` is
* used, or the `APP_BASE_HREF` if the `PathLocationStrategy` is in use.
*/
prepareExternalUrl(url: string): string {
if (url.length > 0 && !url.startsWith('/')) {
url = '/' + url;
}
return this.platformStrategy.prepareExternalUrl(url);
}
// TODO: rename this method to pushState
/**
* Changes the browsers URL to the normalized version of the given URL, and pushes a
* new item onto the platform's history.
*/
go(path: string, query: string = ''): void {
this.platformStrategy.pushState(null, '', path, query);
}
/**
* Changes the browsers URL to the normalized version of the given URL, and replaces
* the top item on the platform's history stack.
*/
replaceState(path: string, query: string = ''): void {
this.platformStrategy.replaceState(null, '', path, query);
}
/**
* Navigates forward in the platform's history.
*/
forward(): void { this.platformStrategy.forward(); }
/**
* Navigates back in the platform's history.
*/
back(): void { this.platformStrategy.back(); }
/**
* Subscribe to the platform's `popState` events.
*/
subscribe(onNext: (value: any) => void, onThrow: (exception: any) => void = null,
onReturn: () => void = null): Object {
return ObservableWrapper.subscribe(this._subject, onNext, onThrow, onReturn);
}
/**
* Given a string of url parameters, prepend with '?' if needed, otherwise return parameters as
* is.
*/
public static normalizeQueryParams(params: string): string {
return (params.length > 0 && params.substring(0, 1) != '?') ? ('?' + params) : params;
}
/**
* Given 2 parts of a url, join them with a slash if needed.
*/
public static joinWithSlash(start: string, end: string): string {
if (start.length == 0) {
return end;
}
if (end.length == 0) {
return start;
}
var slashes = 0;
if (start.endsWith('/')) {
slashes++;
}
if (end.startsWith('/')) {
slashes++;
}
if (slashes == 2) {
return start + end.substring(1);
}
if (slashes == 1) {
return start + end;
}
return start + '/' + end;
}
/**
* If url has a trailing slash, remove it, otherwise return url as is.
*/
public static stripTrailingSlash(url: string): string {
if (/\/$/g.test(url)) {
url = url.substring(0, url.length - 1);
}
return url;
}
}
function _stripBaseHref(baseHref: string, url: string): string {
if (baseHref.length > 0 && url.startsWith(baseHref)) {
return url.substring(baseHref.length);
}
return url;
}
function _stripIndexHtml(url: string): string {
if (/\/index.html$/g.test(url)) {
// '/index.html'.length == 11
return url.substring(0, url.length - 11);
}
return url;
}

View File

@ -1,62 +0,0 @@
import {CONST_EXPR} from 'angular2/src/facade/lang';
import {OpaqueToken} from 'angular2/core';
import {UrlChangeListener} from './platform_location';
/**
* `LocationStrategy` is responsible for representing and reading route state
* from the browser's URL. Angular provides two strategies:
* {@link HashLocationStrategy} and {@link PathLocationStrategy} (default).
*
* This is used under the hood of the {@link Location} service.
*
* Applications should use the {@link Router} or {@link Location} services to
* interact with application route state.
*
* For instance, {@link HashLocationStrategy} produces URLs like
* `http://example.com#/foo`, and {@link PathLocationStrategy} produces
* `http://example.com/foo` as an equivalent URL.
*
* See these two classes for more.
*/
export abstract class LocationStrategy {
abstract path(): string;
abstract prepareExternalUrl(internal: string): string;
abstract pushState(state: any, title: string, url: string, queryParams: string): void;
abstract replaceState(state: any, title: string, url: string, queryParams: string): void;
abstract forward(): void;
abstract back(): void;
abstract onPopState(fn: UrlChangeListener): void;
abstract getBaseHref(): string;
}
/**
* The `APP_BASE_HREF` token represents the base href to be used with the
* {@link PathLocationStrategy}.
*
* If you're using {@link PathLocationStrategy}, you must provide a provider to a string
* representing the URL prefix that should be preserved when generating and recognizing
* URLs.
*
* ### Example
*
* ```
* import {Component} from 'angular2/core';
* import {ROUTER_DIRECTIVES, ROUTER_PROVIDERS, RouteConfig} from 'angular2/router';
* import {APP_BASE_HREF} from 'angular2/platform/common';
*
* @Component({directives: [ROUTER_DIRECTIVES]})
* @RouteConfig([
* {...},
* ])
* class AppCmp {
* // ...
* }
*
* bootstrap(AppCmp, [
* ROUTER_PROVIDERS,
* provide(APP_BASE_HREF, {useValue: '/my/app'})
* ]);
* ```
*/
export const APP_BASE_HREF: OpaqueToken = CONST_EXPR(new OpaqueToken('appBaseHref'));

View File

@ -1,105 +0,0 @@
import {Injectable, Inject, Optional} from 'angular2/core';
import {isBlank} from 'angular2/src/facade/lang';
import {BaseException} from 'angular2/src/facade/exceptions';
import {PlatformLocation, UrlChangeListener} from './platform_location';
import {LocationStrategy, APP_BASE_HREF} from './location_strategy';
import {Location} from './location';
/**
* `PathLocationStrategy` is a {@link LocationStrategy} used to configure the
* {@link Location} service to represent its state in the
* [path](https://en.wikipedia.org/wiki/Uniform_Resource_Locator#Syntax) of the
* browser's URL.
*
* `PathLocationStrategy` is the default binding for {@link LocationStrategy}
* provided in {@link ROUTER_PROVIDERS}.
*
* If you're using `PathLocationStrategy`, you must provide a provider for
* {@link APP_BASE_HREF} to a string representing the URL prefix that should
* be preserved when generating and recognizing URLs.
*
* For instance, if you provide an `APP_BASE_HREF` of `'/my/app'` and call
* `location.go('/foo')`, the browser's URL will become
* `example.com/my/app/foo`.
*
* ### Example
*
* ```
* import {Component, provide} from 'angular2/core';
* import {bootstrap} from 'angular2/platform/browser';
* import {
* Location,
* APP_BASE_HREF
* } from 'angular2/platform/common';
* import {
* ROUTER_DIRECTIVES,
* ROUTER_PROVIDERS,
* RouteConfig
* } from 'angular2/router';
*
* @Component({directives: [ROUTER_DIRECTIVES]})
* @RouteConfig([
* {...},
* ])
* class AppCmp {
* constructor(location: Location) {
* location.go('/foo');
* }
* }
*
* bootstrap(AppCmp, [
* ROUTER_PROVIDERS, // includes binding to PathLocationStrategy
* provide(APP_BASE_HREF, {useValue: '/my/app'})
* ]);
* ```
*/
@Injectable()
export class PathLocationStrategy extends LocationStrategy {
private _baseHref: string;
constructor(private _platformLocation: PlatformLocation,
@Optional() @Inject(APP_BASE_HREF) href?: string) {
super();
if (isBlank(href)) {
href = this._platformLocation.getBaseHrefFromDOM();
}
if (isBlank(href)) {
throw new BaseException(
`No base href set. Please provide a value for the APP_BASE_HREF token or add a base element to the document.`);
}
this._baseHref = href;
}
onPopState(fn: UrlChangeListener): void {
this._platformLocation.onPopState(fn);
this._platformLocation.onHashChange(fn);
}
getBaseHref(): string { return this._baseHref; }
prepareExternalUrl(internal: string): string {
return Location.joinWithSlash(this._baseHref, internal);
}
path(): string {
return this._platformLocation.pathname +
Location.normalizeQueryParams(this._platformLocation.search);
}
pushState(state: any, title: string, url: string, queryParams: string) {
var externalUrl = this.prepareExternalUrl(url + Location.normalizeQueryParams(queryParams));
this._platformLocation.pushState(state, title, externalUrl);
}
replaceState(state: any, title: string, url: string, queryParams: string) {
var externalUrl = this.prepareExternalUrl(url + Location.normalizeQueryParams(queryParams));
this._platformLocation.replaceState(state, title, externalUrl);
}
forward(): void { this._platformLocation.forward(); }
back(): void { this._platformLocation.back(); }
}

View File

@ -1,48 +0,0 @@
/**
* This class should not be used directly by an application developer. Instead, use
* {@link Location}.
*
* `PlatformLocation` encapsulates all calls to DOM apis, which allows the Router to be platform
* agnostic.
* This means that we can have different implementation of `PlatformLocation` for the different
* platforms
* that angular supports. For example, the default `PlatformLocation` is {@link
* BrowserPlatformLocation},
* however when you run your app in a WebWorker you use {@link WebWorkerPlatformLocation}.
*
* The `PlatformLocation` class is used directly by all implementations of {@link LocationStrategy}
* when
* they need to interact with the DOM apis like pushState, popState, etc...
*
* {@link LocationStrategy} in turn is used by the {@link Location} service which is used directly
* by
* the {@link Router} in order to navigate between routes. Since all interactions between {@link
* Router} /
* {@link Location} / {@link LocationStrategy} and DOM apis flow through the `PlatformLocation`
* class
* they are all platform independent.
*/
export abstract class PlatformLocation {
abstract getBaseHrefFromDOM(): string;
abstract onPopState(fn: UrlChangeListener): void;
abstract onHashChange(fn: UrlChangeListener): void;
/* abstract */ get pathname(): string { return null; }
/* abstract */ get search(): string { return null; }
/* abstract */ get hash(): string { return null; }
abstract replaceState(state: any, title: string, url: string): void;
abstract pushState(state: any, title: string, url: string): void;
abstract forward(): void;
abstract back(): void;
}
/**
* A serializable version of the event from onPopState or onHashChange
*/
export interface UrlChangeEvent { type: string; }
export interface UrlChangeListener { (e: UrlChangeEvent): any; }

View File

@ -10,17 +10,18 @@ import {reflector} from 'angular2/src/core/reflection/reflection';
export function recognize(componentResolver: ComponentResolver, type: Type, export function recognize(componentResolver: ComponentResolver, type: Type,
url: Tree<UrlSegment>): Promise<Tree<RouteSegment>> { url: Tree<UrlSegment>): Promise<Tree<RouteSegment>> {
return componentResolver.resolveComponent(type).then(factory => { let matched = new _MatchResult(type, [url.root], null, rootNode(url).children, []);
let segment = return _constructSegment(componentResolver, matched)
new RouteSegment([url.root], url.root.parameters, DEFAULT_OUTLET_NAME, type, factory); .then(roots => new Tree<RouteSegment>(roots[0]));
return _recognizeMany(componentResolver, type, rootNode(url).children)
.then(children => new Tree<RouteSegment>(new TreeNode<RouteSegment>(segment, children)));
});
} }
function _recognize(componentResolver: ComponentResolver, parentType: Type, function _recognize(componentResolver: ComponentResolver, parentType: 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(parentType); // should read from the factory instead
if (isBlank(metadata)) {
throw new BaseException(
`Component '${stringify(parentType)}' does not have route configuration`);
}
let match; let match;
try { try {
@ -43,18 +44,45 @@ function _recognizeMany(componentResolver: ComponentResolver, parentType: Type,
function _constructSegment(componentResolver: ComponentResolver, function _constructSegment(componentResolver: ComponentResolver,
matched: _MatchResult): Promise<TreeNode<RouteSegment>[]> { matched: _MatchResult): Promise<TreeNode<RouteSegment>[]> {
return componentResolver.resolveComponent(matched.route.component) return componentResolver.resolveComponent(matched.component)
.then(factory => { .then(factory => {
let urlOutlet = matched.consumedUrlSegments[0].outlet; let urlOutlet = matched.consumedUrlSegments[0].outlet;
let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters, let segment = new RouteSegment(matched.consumedUrlSegments, matched.parameters,
isBlank(urlOutlet) ? DEFAULT_OUTLET_NAME : urlOutlet, isBlank(urlOutlet) ? DEFAULT_OUTLET_NAME : urlOutlet,
matched.route.component, factory); matched.component, factory);
if (matched.leftOverUrl.length > 0) { if (matched.leftOverUrl.length > 0) {
return _recognizeMany(componentResolver, matched.route.component, matched.leftOverUrl) return _recognizeMany(componentResolver, matched.component, matched.leftOverUrl)
.then(children => [new TreeNode<RouteSegment>(segment, children)]); .then(children => [new TreeNode<RouteSegment>(segment, children)]);
} else { } else {
return [new TreeNode<RouteSegment>(segment, [])]; return _recognizeLeftOvers(componentResolver, matched.component)
.then(children => [new TreeNode<RouteSegment>(segment, children)]);
}
});
}
function _recognizeLeftOvers(componentResolver: ComponentResolver,
parentType: Type): Promise<TreeNode<RouteSegment>[]> {
return componentResolver.resolveComponent(parentType)
.then(factory => {
let metadata = _readMetadata(parentType);
if (isBlank(metadata)) {
return [];
}
let r = (<any[]>metadata.routes).filter(r => r.path == "" || r.path == "/");
if (r.length === 0) {
return PromiseWrapper.resolve([]);
} else {
return _recognizeLeftOvers(componentResolver, r[0].component)
.then(children => {
return componentResolver.resolveComponent(r[0].component)
.then(factory => {
let segment =
new RouteSegment([], null, DEFAULT_OUTLET_NAME, r[0].component, factory);
return [new TreeNode<RouteSegment>(segment, children)];
});
});
} }
}); });
} }
@ -66,11 +94,14 @@ function _match(metadata: RoutesMetadata, url: TreeNode<UrlSegment>): _MatchResu
return matchingResult; return matchingResult;
} }
} }
throw new BaseException("Cannot match any routes"); let availableRoutes = metadata.routes.map(r => `'${r.path}'`).join(", ");
throw new BaseException(
`Cannot match any routes. Current segment: '${url.value}'. Available routes: [${availableRoutes}].`);
} }
function _matchWithParts(route: RouteMetadata, url: TreeNode<UrlSegment>): _MatchResult { function _matchWithParts(route: RouteMetadata, url: TreeNode<UrlSegment>): _MatchResult {
let parts = route.path.split("/"); let path = route.path.startsWith("/") ? route.path.substring(1) : route.path;
let parts = path.split("/");
let positionalParams = {}; let positionalParams = {};
let consumedUrlSegments = []; let consumedUrlSegments = [];
@ -113,7 +144,7 @@ function _matchWithParts(route: RouteMetadata, url: TreeNode<UrlSegment>): _Matc
<{[key: string]: string}>StringMapWrapper.merge(isBlank(p) ? {} : p, positionalParams); <{[key: string]: string}>StringMapWrapper.merge(isBlank(p) ? {} : p, positionalParams);
let axuUrlSubtrees = isPresent(lastParent) ? lastParent.children.slice(1) : []; let axuUrlSubtrees = isPresent(lastParent) ? lastParent.children.slice(1) : [];
return new _MatchResult(route, consumedUrlSegments, parameters, lastSegment.children, return new _MatchResult(route.component, consumedUrlSegments, parameters, lastSegment.children,
axuUrlSubtrees); axuUrlSubtrees);
} }
@ -132,16 +163,12 @@ function _checkOutletNameUniqueness(nodes: TreeNode<RouteSegment>[]): TreeNode<R
} }
class _MatchResult { class _MatchResult {
constructor(public route: RouteMetadata, public consumedUrlSegments: UrlSegment[], constructor(public component: Type, 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>[]) {}
} }
function _readMetadata(componentType: Type) { function _readMetadata(componentType: Type) {
let metadata = reflector.annotations(componentType).filter(f => f instanceof RoutesMetadata); let metadata = reflector.annotations(componentType).filter(f => f instanceof RoutesMetadata);
if (metadata.length === 0) { return ListWrapper.first(metadata);
throw new BaseException(
`Component '${stringify(componentType)}' does not have route configuration`);
}
return metadata[0];
} }

View File

@ -1,13 +1,12 @@
import {provide, ReflectiveInjector, ComponentResolver} from 'angular2/core'; import {OnInit, provide, ReflectiveInjector, ComponentResolver} from 'angular2/core';
import {RouterOutlet} from './directives/router_outlet'; import {RouterOutlet} from './directives/router_outlet';
import {Type, isBlank, isPresent} from 'angular2/src/facade/lang'; import {Type, isBlank, isPresent} from 'angular2/src/facade/lang';
import {EventEmitter, Observable} from 'angular2/src/facade/async'; import {EventEmitter, Observable} from 'angular2/src/facade/async';
import {StringMapWrapper} from 'angular2/src/facade/collection'; import {StringMapWrapper} from 'angular2/src/facade/collection';
import {BaseException} from 'angular2/src/facade/exceptions'; import {BaseException} from 'angular2/src/facade/exceptions';
import {BaseException} from 'angular2/src/facade/exceptions';
import {RouterUrlSerializer} from './router_url_serializer'; import {RouterUrlSerializer} from './router_url_serializer';
import {recognize} from './recognize'; import {recognize} from './recognize';
import {Location} from './location/location'; import {Location} from 'angular2/platform/common';
import { import {
equalSegments, equalSegments,
routeSegmentComponentFactory, routeSegmentComponentFactory,
@ -31,15 +30,12 @@ export class Router {
private _prevTree: Tree<RouteSegment>; private _prevTree: Tree<RouteSegment>;
private _urlTree: Tree<UrlSegment>; private _urlTree: Tree<UrlSegment>;
private _location: Location;
private _changes: EventEmitter<void> = new EventEmitter<void>(); private _changes: EventEmitter<void> = new EventEmitter<void>();
constructor(private _componentType: Type, private _componentResolver: ComponentResolver, constructor(private _componentType: Type, private _componentResolver: ComponentResolver,
private _urlSerializer: RouterUrlSerializer, private _urlSerializer: RouterUrlSerializer,
private _routerOutletMap: RouterOutletMap, location: Location) { private _routerOutletMap: RouterOutletMap, private _location: Location) {
this._location = location; this.navigateByUrl(this._location.path());
this.navigateByUrl(location.path());
} }
get urlTree(): Tree<UrlSegment> { return this._urlTree; } get urlTree(): Tree<UrlSegment> { return this._urlTree; }
@ -48,9 +44,7 @@ export class Router {
this._urlTree = url; this._urlTree = url;
return recognize(this._componentResolver, this._componentType, url) return recognize(this._componentResolver, this._componentType, url)
.then(currTree => { .then(currTree => {
let prevRoot = isPresent(this._prevTree) ? rootNode(this._prevTree) : null; new _LoadSegments(currTree, this._prevTree).load(this._routerOutletMap);
new _LoadSegments(currTree, this._prevTree)
.loadSegments(rootNode(currTree), prevRoot, this._routerOutletMap);
this._prevTree = currTree; this._prevTree = currTree;
this._location.go(this._urlSerializer.serialize(this._urlTree)); this._location.go(this._urlSerializer.serialize(this._urlTree));
this._changes.emit(null); this._changes.emit(null);
@ -69,6 +63,12 @@ export class Router {
class _LoadSegments { class _LoadSegments {
constructor(private currTree: Tree<RouteSegment>, private prevTree: Tree<RouteSegment>) {} constructor(private currTree: Tree<RouteSegment>, private prevTree: Tree<RouteSegment>) {}
load(parentOutletMap: RouterOutletMap): void {
let prevRoot = isPresent(this.prevTree) ? rootNode(this.prevTree) : null;
let currRoot = rootNode(this.currTree);
this.loadChildSegments(currRoot, prevRoot, parentOutletMap);
}
loadSegments(currNode: TreeNode<RouteSegment>, prevNode: TreeNode<RouteSegment>, loadSegments(currNode: TreeNode<RouteSegment>, prevNode: TreeNode<RouteSegment>,
parentOutletMap: RouterOutletMap): void { parentOutletMap: RouterOutletMap): void {
let curr = currNode.value; let curr = currNode.value;

View File

@ -0,0 +1,12 @@
import {ROUTER_PROVIDERS_COMMON} from './router_providers_common';
import {Provider} from 'angular2/core';
import {
BrowserPlatformLocation
} from 'angular2/src/platform/browser/location/browser_platform_location';
import {PlatformLocation} from 'angular2/platform/common';
import {CONST_EXPR} from 'angular2/src/facade/lang';
export const ROUTER_PROVIDERS: any[] = CONST_EXPR([
ROUTER_PROVIDERS_COMMON,
CONST_EXPR(new Provider(PlatformLocation, {useClass: BrowserPlatformLocation})),
]);

View File

@ -0,0 +1,35 @@
import {OpaqueToken, ComponentResolver} from 'angular2/core';
import {LocationStrategy, PathLocationStrategy, Location} from 'angular2/platform/common';
import {Router, RouterOutletMap} from './router';
import {RouterUrlSerializer, DefaultRouterUrlSerializer} from './router_url_serializer';
import {CONST_EXPR} from 'angular2/src/facade/lang';
import {ApplicationRef, Provider} from 'angular2/core';
import {BaseException} from 'angular2/src/facade/exceptions';
export const ROUTER_PROVIDERS_COMMON: any[] = CONST_EXPR([
RouterOutletMap,
CONST_EXPR(new Provider(RouterUrlSerializer, {useClass: DefaultRouterUrlSerializer})),
CONST_EXPR(new Provider(LocationStrategy, {useClass: PathLocationStrategy})),
Location,
CONST_EXPR(new Provider(Router,
{
useFactory: routerFactory,
deps: CONST_EXPR([
ApplicationRef,
ComponentResolver,
RouterUrlSerializer,
RouterOutletMap,
Location
])
}))
]);
function routerFactory(app: ApplicationRef, componentResolver: ComponentResolver,
urlSerializer: RouterUrlSerializer, routerOutletMap: RouterOutletMap,
location: Location): Router {
if (app.componentTypes.length == 0) {
throw new BaseException("Bootstrap at least one component before injecting Router.");
}
return new Router(app.componentTypes[0], componentResolver, urlSerializer, routerOutletMap,
location);
}

View File

@ -64,7 +64,7 @@ class _UrlParser {
parse(url: string): TreeNode<UrlSegment> { parse(url: string): TreeNode<UrlSegment> {
this._remaining = url; this._remaining = url;
if (url == '' || url == '/') { if (url == '' || url == '/') {
return new TreeNode<UrlSegment>(new UrlSegment('', {}, null), []); return new TreeNode<UrlSegment>(new UrlSegment('', null, null), []);
} else { } else {
return this.parseRoot(); return this.parseRoot();
} }
@ -72,7 +72,7 @@ class _UrlParser {
parseRoot(): TreeNode<UrlSegment> { parseRoot(): TreeNode<UrlSegment> {
let segments = this.parseSegments(); let segments = this.parseSegments();
let queryParams = this.peekStartsWith('?') ? this.parseQueryParams() : {}; let queryParams = this.peekStartsWith('?') ? this.parseQueryParams() : null;
return new TreeNode<UrlSegment>(new UrlSegment('', queryParams, null), segments); return new TreeNode<UrlSegment>(new UrlSegment('', queryParams, null), segments);
} }

View File

@ -89,7 +89,9 @@ export class RouteSegment {
this._componentFactory = componentFactory; this._componentFactory = componentFactory;
} }
getParam(param: string): string { return this.parameters[param]; } getParam(param: string): string {
return isPresent(this.parameters) ? this.parameters[param] : null;
}
get type(): Type { return this._type; } get type(): Type { return this._type; }

View File

@ -28,10 +28,10 @@ import {
Routes, Routes,
RouterUrlSerializer, RouterUrlSerializer,
DefaultRouterUrlSerializer, DefaultRouterUrlSerializer,
OnActivate, OnActivate
Location
} from 'angular2/alt_router'; } from 'angular2/alt_router';
import {SpyLocation} from 'angular2/src/mock/location_mock'; import {SpyLocation} from 'angular2/src/mock/location_mock';
import {Location} from 'angular2/platform/common';
import {DOM} from 'angular2/src/platform/dom/dom_adapter'; import {DOM} from 'angular2/src/platform/dom/dom_adapter';
export function main() { export function main() {
@ -58,6 +58,7 @@ export function main() {
router.navigateByUrl('/team/33/simple'); router.navigateByUrl('/team/33/simple');
advance(fixture); advance(fixture);
expect(location.path()).toEqual('/team/33/simple'); expect(location.path()).toEqual('/team/33/simple');
}))); })));
@ -123,40 +124,43 @@ export function main() {
expect(fixture.debugElement.nativeElement).toHaveText('team 22 { hello fedor, aux: }'); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { hello fedor, aux: }');
}))); })));
it("should support router links", if (DOM.supportsDOMEvents()) { // this is required to use fakeAsync
fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => {
let fixture = tcb.createFakeAsync(RootCmp);
advance(fixture);
router.navigateByUrl('/team/22/link'); it("should support router links",
advance(fixture); fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => {
expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, aux: }'); let fixture = tcb.createFakeAsync(RootCmp);
advance(fixture);
let native = DOM.querySelector(fixture.debugElement.nativeElement, "a"); router.navigateByUrl('/team/22/link');
expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple"); advance(fixture);
DOM.dispatchEvent(native, DOM.createMouseEvent('click')); expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, aux: }');
advance(fixture);
expect(fixture.debugElement.nativeElement).toHaveText('team 33 { simple, aux: }'); let native = DOM.querySelector(fixture.debugElement.nativeElement, "a");
}))); expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple");
DOM.dispatchEvent(native, DOM.createMouseEvent('click'));
advance(fixture);
it("should update router links when router changes", expect(fixture.debugElement.nativeElement).toHaveText('team 33 { simple, aux: }');
fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => { })));
let fixture = tcb.createFakeAsync(RootCmp);
advance(fixture);
router.navigateByUrl('/team/22/link(simple)'); it("should update router links when router changes",
advance(fixture); fakeAsync(inject([Router, TestComponentBuilder], (router, tcb) => {
expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, aux: simple }'); let fixture = tcb.createFakeAsync(RootCmp);
advance(fixture);
let native = DOM.querySelector(fixture.debugElement.nativeElement, "a"); router.navigateByUrl('/team/22/link(simple)');
expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple(aux:simple)"); advance(fixture);
expect(fixture.debugElement.nativeElement).toHaveText('team 22 { link, aux: simple }');
router.navigateByUrl('/team/22/link(simple2)'); let native = DOM.querySelector(fixture.debugElement.nativeElement, "a");
advance(fixture); expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple(aux:simple)");
expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple(aux:simple2)"); router.navigateByUrl('/team/22/link(simple2)');
}))); advance(fixture);
expect(DOM.getAttribute(native, "href")).toEqual("/team/33/simple(aux:simple2)");
})));
}
}); });
} }

View File

@ -48,6 +48,26 @@ export function main() {
}); });
})); }));
it('should support empty routes',
inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => {
recognize(resolver, ComponentA, tree("f"))
.then(r => {
let a = r.root;
expect(stringifyUrl(a.urlSegments)).toEqual([""]);
expect(a.type).toBe(ComponentA);
let f = r.firstChild(r.root);
expect(stringifyUrl(f.urlSegments)).toEqual(["f"]);
expect(f.type).toBe(ComponentF);
let d = r.firstChild(r.firstChild(r.root));
expect(stringifyUrl(d.urlSegments)).toEqual([]);
expect(d.type).toBe(ComponentD);
async.done();
});
}));
it('should handle aux routes', it('should handle aux routes',
inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => { inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => {
recognize(resolver, ComponentA, tree("b/paramB(/d//right:d)")) recognize(resolver, ComponentA, tree("b/paramB(/d//right:d)"))
@ -133,7 +153,7 @@ export function main() {
inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => { inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => {
recognize(resolver, ComponentA, tree("invalid")) recognize(resolver, ComponentA, tree("invalid"))
.catch(e => { .catch(e => {
expect(e.message).toEqual("Cannot match any routes"); expect(e.message).toContain("Cannot match any routes");
async.done(); async.done();
}); });
})); }));
@ -142,7 +162,7 @@ export function main() {
inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => { inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => {
recognize(resolver, ComponentA, tree("b")) recognize(resolver, ComponentA, tree("b"))
.catch(e => { .catch(e => {
expect(e.message).toEqual("Cannot match any routes"); expect(e.message).toContain("Cannot match any routes");
async.done(); async.done();
}); });
})); }));
@ -175,6 +195,11 @@ class ComponentD {
class ComponentE { class ComponentE {
} }
@Component({selector: 'f', template: 't'})
@Routes([new Route({path: "/", component: ComponentD})])
class ComponentF {
}
@Component({selector: 'c', template: 't'}) @Component({selector: 'c', template: 't'})
@Routes([new Route({path: "d", component: ComponentD})]) @Routes([new Route({path: "d", component: ComponentD})])
class ComponentC { class ComponentC {
@ -193,7 +218,8 @@ class ComponentB {
@Routes([ @Routes([
new Route({path: "b/:b", component: ComponentB}), new Route({path: "b/:b", component: ComponentB}),
new Route({path: "d", component: ComponentD}), new Route({path: "d", component: ComponentD}),
new Route({path: "e", component: ComponentE}) new Route({path: "e", component: ComponentE}),
new Route({path: "f", component: ComponentF})
]) ])
class ComponentA { class ComponentA {
} }

View File

@ -106,26 +106,6 @@ export function main() {
expect(url.serialize(tree)).toEqual("/one;a=true"); expect(url.serialize(tree)).toEqual("/one;a=true");
}); });
// it("should parse key-value query params", () => {
// let tree = url.parse("/one?a=1&b=2");
// expect(tree.root).toEqual(new UrlSegment("", {'a': '1', 'b': '2'}, DEFAULT_OUTLET_NAME));
// });
//
// it("should parse key only query params", () => {
// let tree = url.parse("/one?a");
// expect(tree.root).toEqual(new UrlSegment("", {'a': "true"}, DEFAULT_OUTLET_NAME));
// });
//
// it("should parse a url with only query params", () => {
// let tree = url.parse("?a");
// expect(tree.root).toEqual(new UrlSegment("", {'a': "true"}, DEFAULT_OUTLET_NAME));
// });
//
// it("should allow slashes within query params", () => {
// let tree = url.parse("?a=http://boo");
// expect(tree.root).toEqual(new UrlSegment("", {'a': "http://boo"}, DEFAULT_OUTLET_NAME));
// });
}); });
} }