refactor(router): update link to reuse url segments when possible

This commit is contained in:
vsavkin 2016-05-04 11:46:38 -07:00
parent 12637a761c
commit d00b26d941
4 changed files with 81 additions and 38 deletions

View File

@ -1,11 +1,10 @@
import {Tree, TreeNode, UrlSegment, RouteSegment, rootNode, UrlTree, RouteTree} from './segments'; import {Tree, TreeNode, UrlSegment, RouteSegment, rootNode, UrlTree, RouteTree} from './segments';
import {isBlank, isPresent, isString, isStringMap} from './facade/lang'; import {isBlank, isPresent, isString, isStringMap} from './facade/lang';
import {BaseException} from './facade/exceptions'; import {BaseException} from './facade/exceptions';
import {ListWrapper} from './facade/collection'; import {ListWrapper, StringMapWrapper} from './facade/collection';
// TODO: vsavkin: should reuse segments // TODO: vsavkin: should reuse segments
export function link(segment: RouteSegment, routeTree: RouteTree, urlTree: UrlTree, export function link(segment: RouteSegment, routeTree: RouteTree, urlTree: UrlTree, commands: any[]): UrlTree {
commands: any[]): UrlTree {
if (commands.length === 0) return urlTree; if (commands.length === 0) return urlTree;
let normalizedCommands = _normalizeCommands(commands); let normalizedCommands = _normalizeCommands(commands);
@ -14,22 +13,20 @@ export function link(segment: RouteSegment, routeTree: RouteTree, urlTree: UrlTr
} }
let startingNode = _findStartingNode(normalizedCommands, urlTree, segment, routeTree); let startingNode = _findStartingNode(normalizedCommands, urlTree, segment, routeTree);
let updated = let updated = normalizedCommands.commands.length > 0 ?
normalizedCommands.commands.length > 0 ? _updateMany(ListWrapper.clone(startingNode.children), normalizedCommands.commands) : [];
_updateMany(ListWrapper.clone(startingNode.children), normalizedCommands.commands) :
[];
let newRoot = _constructNewTree(rootNode(urlTree), startingNode, updated); let newRoot = _constructNewTree(rootNode(urlTree), startingNode, updated);
return new UrlTree(newRoot); return new UrlTree(newRoot);
} }
function _navigateToRoot(normalizedChange: _NormalizedNavigationCommands): boolean { function _navigateToRoot(normalizedChange:_NormalizedNavigationCommands):boolean {
return normalizedChange.isAbsolute && normalizedChange.commands.length === 1 && return normalizedChange.isAbsolute && normalizedChange.commands.length === 1 && normalizedChange.commands[0] == "/";
normalizedChange.commands[0] == "/";
} }
class _NormalizedNavigationCommands { class _NormalizedNavigationCommands {
constructor(public isAbsolute: boolean, public numberOfDoubleDots: number, constructor(public isAbsolute: boolean,
public numberOfDoubleDots: number,
public commands: any[]) {} public commands: any[]) {}
} }
@ -67,7 +64,7 @@ function _normalizeCommands(commands: any[]): _NormalizedNavigationCommands {
} }
} else { } else {
if (cc != '') { if (cc != ''){
res.push(cc); res.push(cc);
} }
} }
@ -77,8 +74,7 @@ function _normalizeCommands(commands: any[]): _NormalizedNavigationCommands {
return new _NormalizedNavigationCommands(isAbsolute, numberOfDoubleDots, res); return new _NormalizedNavigationCommands(isAbsolute, numberOfDoubleDots, res);
} }
function _findUrlSegment(segment: RouteSegment, routeTree: RouteTree, urlTree: UrlTree, function _findUrlSegment(segment: RouteSegment, routeTree: RouteTree, urlTree: UrlTree, numberOfDoubleDots: number): UrlSegment {
numberOfDoubleDots: number): UrlSegment {
let s = segment; let s = segment;
while (s.urlSegments.length === 0) { while (s.urlSegments.length === 0) {
s = routeTree.parent(s); s = routeTree.parent(s);
@ -91,13 +87,11 @@ function _findUrlSegment(segment: RouteSegment, routeTree: RouteTree, urlTree: U
return path[path.length - 1 - numberOfDoubleDots]; return path[path.length - 1 - numberOfDoubleDots];
} }
function _findStartingNode(normalizedChange: _NormalizedNavigationCommands, urlTree: UrlTree, function _findStartingNode(normalizedChange:_NormalizedNavigationCommands, urlTree:UrlTree, segment:RouteSegment, routeTree:RouteTree):TreeNode<UrlSegment> {
segment: RouteSegment, routeTree: RouteTree): TreeNode<UrlSegment> {
if (normalizedChange.isAbsolute) { if (normalizedChange.isAbsolute) {
return rootNode(urlTree); return rootNode(urlTree);
} else { } else {
let urlSegment = let urlSegment = _findUrlSegment(segment, routeTree, urlTree, normalizedChange.numberOfDoubleDots);
_findUrlSegment(segment, routeTree, urlTree, normalizedChange.numberOfDoubleDots);
return _findMatchingNode(urlSegment, rootNode(urlTree)); return _findMatchingNode(urlSegment, rootNode(urlTree));
} }
} }
@ -134,7 +128,7 @@ function _update(node: TreeNode<UrlSegment>, commands: any[]): TreeNode<UrlSegme
return new TreeNode<UrlSegment>(urlSegment, children); return new TreeNode<UrlSegment>(urlSegment, children);
} else if (isBlank(node) && isStringMap(next)) { } else if (isBlank(node) && isStringMap(next)) {
let urlSegment = new UrlSegment(segment, next, outlet); let urlSegment = new UrlSegment(segment, _stringify(next), outlet);
return _recurse(urlSegment, node, rest.slice(1)); return _recurse(urlSegment, node, rest.slice(1));
// different outlet => preserve the subtree // different outlet => preserve the subtree
@ -143,23 +137,40 @@ function _update(node: TreeNode<UrlSegment>, commands: any[]): TreeNode<UrlSegme
// params command // params command
} else if (isStringMap(segment)) { } else if (isStringMap(segment)) {
let newSegment = new UrlSegment(node.value.segment, segment, node.value.outlet); let newSegment = new UrlSegment(node.value.segment, _stringify(segment), node.value.outlet);
return _recurse(newSegment, node, rest); return _recurse(newSegment, node, rest);
// next one is a params command // next one is a params command && can reuse the node
} else if (isStringMap(next) && _compare(segment, _stringify(next), node.value)){
return _recurse(node.value, node, rest.slice(1));
// next one is a params command && cannot reuse the node
} else if (isStringMap(next)) { } else if (isStringMap(next)) {
let urlSegment = new UrlSegment(segment, next, outlet); let urlSegment = new UrlSegment(segment, _stringify(next), outlet);
return _recurse(urlSegment, node, rest.slice(1)); return _recurse(urlSegment, node, rest.slice(1));
// next one is not a params command // next one is not a params command && can reuse the node
} else if (_compare(segment, {}, node.value)) {
return _recurse(node.value, node, rest);
// next one is not a params command && cannot reuse the node
} else { } else {
let urlSegment = new UrlSegment(segment, {}, outlet); let urlSegment = new UrlSegment(segment, {}, outlet);
return _recurse(urlSegment, node, rest); return _recurse(urlSegment, node, rest);
} }
} }
function _recurse(urlSegment: UrlSegment, node: TreeNode<UrlSegment>, function _stringify(params: {[key: string]: any}):{[key: string]: string} {
rest: any[]): TreeNode<UrlSegment> { let res = {};
StringMapWrapper.forEach(params, (v, k) => res[k] = v.toString());
return res;
}
function _compare(path: string, params: {[key: string]: any}, segment: UrlSegment): boolean {
return path == segment.segment && StringMapWrapper.equals(params, segment.parameters);
}
function _recurse(urlSegment: UrlSegment, node: TreeNode<UrlSegment>, rest: any[]): TreeNode<UrlSegment> {
if (rest.length === 0) { if (rest.length === 0) {
return new TreeNode<UrlSegment>(urlSegment, []); return new TreeNode<UrlSegment>(urlSegment, []);
} }

View File

@ -1,6 +1,6 @@
import {ComponentFactory, Type} from '@angular/core'; import {ComponentFactory, Type} from '@angular/core';
import {StringMapWrapper, ListWrapper} from './facade/collection'; import {StringMapWrapper, ListWrapper} from './facade/collection';
import {isBlank, isPresent, stringify} from './facade/lang'; import {isBlank, isPresent, stringify, NumberWrapper} from './facade/lang';
export class Tree<T> { export class Tree<T> {
/** @internal */ /** @internal */
@ -80,8 +80,8 @@ function _contains<T>(tree: TreeNode<T>, subtree: TreeNode<T>): boolean {
} }
function _equalValues(a: any, b: any): boolean { function _equalValues(a: any, b: any): boolean {
if (a instanceof RouteSegment) return equalSegments(<any>a, <any>b); // if (a instanceof RouteSegment) return equalSegments(<any>a, <any>b);
if (a instanceof UrlSegment) return equalUrlSegments(<any>a, <any>b); // if (a instanceof UrlSegment) return equalUrlSegments(<any>a, <any>b);
return a === b; return a === b;
} }
@ -90,8 +90,8 @@ export class TreeNode<T> {
} }
export class UrlSegment { export class UrlSegment {
constructor(public segment: any, public parameters: {[key: string]: any}, public outlet: string) { constructor(public segment: any, public parameters: {[key: string]: string},
} public outlet: string) {}
toString(): string { toString(): string {
let outletPrefix = isBlank(this.outlet) ? "" : `${this.outlet}:`; let outletPrefix = isBlank(this.outlet) ? "" : `${this.outlet}:`;
@ -112,7 +112,7 @@ export class RouteSegment {
/** @internal */ /** @internal */
_componentFactory: ComponentFactory<any>; _componentFactory: ComponentFactory<any>;
constructor(public urlSegments: UrlSegment[], public parameters: {[key: string]: any}, constructor(public urlSegments: UrlSegment[], public parameters: {[key: string]: string},
public outlet: string, type: Type, componentFactory: ComponentFactory<any>) { public outlet: string, type: Type, componentFactory: ComponentFactory<any>) {
this._type = type; this._type = type;
this._componentFactory = componentFactory; this._componentFactory = componentFactory;
@ -122,6 +122,10 @@ export class RouteSegment {
return isPresent(this.parameters) ? this.parameters[param] : null; return isPresent(this.parameters) ? this.parameters[param] : null;
} }
getParamAsNumber(param: string): number {
return isPresent(this.parameters) ? NumberWrapper.parseFloat(this.parameters[param]) : null;
}
get type(): Type { return this._type; } get type(): Type { return this._type; }
get stringifiedUrlSegments(): string { return this.urlSegments.map(s => s.toString()).join("/"); } get stringifiedUrlSegments(): string { return this.urlSegments.map(s => s.toString()).join("/"); }

View File

@ -69,6 +69,34 @@ export function main() {
expect(parser.serialize(t)).toEqual("/a/b;aa=22;bb=33"); expect(parser.serialize(t)).toEqual("/a/b;aa=22;bb=33");
}); });
describe("node reuse", () => {
it('should reuse nodes when path is the same', () => {
let p = parser.parse("/a/b");
let tree = s(p.root);
let t = link(tree.root, tree, p, ['/a/c']);
expect(t.root).toBe(p.root);
expect(t.firstChild(t.root)).toBe(p.firstChild(p.root));
expect(t.firstChild(t.firstChild(t.root))).not.toBe(p.firstChild(p.firstChild(p.root)));
});
it("should create new node when params are the same", () => {
let p = parser.parse("/a;x=1");
let tree = s(p.root);
let t = link(tree.root, tree, p, ['/a', {'x': 1}]);
expect(t.firstChild(t.root)).toBe(p.firstChild(p.root));
});
it("should create new node when params are different", () => {
let p = parser.parse("/a;x=1");
let tree = s(p.root);
let t = link(tree.root, tree, p, ['/a', {'x': 2}]);
expect(t.firstChild(t.root)).not.toBe(p.firstChild(p.root));
});
});
describe("relative navigation", () => { describe("relative navigation", () => {
it("should work", () => { it("should work", () => {
let p = parser.parse("/a(ap)/c(cp)"); let p = parser.parse("/a(ap)/c(cp)");
@ -155,7 +183,7 @@ export function main() {
let p = parser.parse("/a(ap)/c(cp)"); let p = parser.parse("/a(ap)/c(cp)");
let c = p.firstChild(p.root); let c = p.firstChild(p.root);
let child = new RouteSegment([], {'one': 1}, null, null, null); let child = new RouteSegment([], {'one':'1'}, null, null, null);
let root = new TreeNode<RouteSegment>(new RouteSegment([c], {}, null, null, null), let root = new TreeNode<RouteSegment>(new RouteSegment([c], {}, null, null, null),
[new TreeNode<RouteSegment>(child, [])]); [new TreeNode<RouteSegment>(child, [])]);
let tree = new RouteTree(root); let tree = new RouteTree(root);

View File

@ -20,7 +20,7 @@ import {DefaultRouterUrlSerializer} from '../src/router_url_serializer';
import {DEFAULT_OUTLET_NAME} from '../src/constants'; import {DEFAULT_OUTLET_NAME} from '../src/constants';
export function main() { export function main() {
describe('recognize', () => { ddescribe('recognize', () => {
it('should handle position args', it('should handle position args',
inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => { inject([AsyncTestCompleter, ComponentResolver], (async, resolver) => {
recognize(resolver, ComponentA, tree("b/paramB/c/paramC/d")) recognize(resolver, ComponentA, tree("b/paramB/c/paramC/d"))