feat(router): introduce matrix params

Closes #2774
Closes #2989
This commit is contained in:
Matias Niemelä 2015-07-08 10:57:38 -07:00
parent 97ef1c27df
commit 5677bf73ca
4 changed files with 270 additions and 14 deletions

View File

@ -26,6 +26,33 @@ import {RouteHandler} from './route_handler';
export class Segment { export class Segment {
name: string; name: string;
regex: string; regex: string;
generate(params: TouchMap): string { return ''; }
}
class TouchMap {
map: StringMap<string, string> = StringMapWrapper.create();
keys: StringMap<string, boolean> = StringMapWrapper.create();
constructor(map: StringMap<string, any>) {
if (isPresent(map)) {
StringMapWrapper.forEach(map, (value, key) => {
this.map[key] = isPresent(value) ? value.toString() : null;
this.keys[key] = true;
});
}
}
get(key: string): string {
StringMapWrapper.delete(this.keys, key);
return this.map[key];
}
getUnused(): StringMap<string, any> {
var unused: StringMap<string, any> = StringMapWrapper.create();
var keys = StringMapWrapper.keys(this.keys);
ListWrapper.forEach(keys, (key) => { unused[key] = StringMapWrapper.get(this.map, key); });
return unused;
}
} }
function normalizeString(obj: any): string { function normalizeString(obj: any): string {
@ -36,10 +63,21 @@ function normalizeString(obj: any): string {
} }
} }
class ContinuationSegment extends Segment { function parseAndAssignMatrixParams(keyValueMap, matrixString) {
generate(params): string { return ''; } if (matrixString[0] == ';') {
matrixString = matrixString.substring(1);
}
matrixString.split(';').forEach((entry) => {
var tuple = entry.split('=');
var key = tuple[0];
var value = tuple.length > 1 ? tuple[1] : true;
keyValueMap[key] = value;
});
} }
class ContinuationSegment extends Segment {}
class StaticSegment extends Segment { class StaticSegment extends Segment {
regex: string; regex: string;
name: string = ''; name: string = '';
@ -47,9 +85,14 @@ class StaticSegment extends Segment {
constructor(public string: string) { constructor(public string: string) {
super(); super();
this.regex = escapeRegex(string); this.regex = escapeRegex(string);
// we add this property so that the route matcher still sees
// this segment as a valid path even if do not use the matrix
// parameters
this.regex += '(;[^\/]+)?';
} }
generate(params): string { return this.string; } generate(params: TouchMap): string { return this.string; }
} }
@IMPLEMENTS(Segment) @IMPLEMENTS(Segment)
@ -58,23 +101,22 @@ class DynamicSegment {
constructor(public name: string) {} constructor(public name: string) {}
generate(params: StringMap<string, string>): string { generate(params: TouchMap): string {
if (!StringMapWrapper.contains(params, this.name)) { if (!StringMapWrapper.contains(params.map, this.name)) {
throw new BaseException( throw new BaseException(
`Route generator for '${this.name}' was not included in parameters passed.`); `Route generator for '${this.name}' was not included in parameters passed.`);
} }
return normalizeString(StringMapWrapper.get(params, this.name)); return normalizeString(params.get(this.name));
} }
} }
class StarSegment { class StarSegment {
regex: string = "(.+)"; regex: string = "(.+)";
constructor(public name: string) {} constructor(public name: string) {}
generate(params: StringMap<string, string>): string { generate(params: TouchMap): string { return normalizeString(params.get(this.name)); }
return normalizeString(StringMapWrapper.get(params, this.name));
}
} }
@ -168,9 +210,27 @@ export class PathRecognizer {
} }
parseParams(url: string): StringMap<string, string> { parseParams(url: string): StringMap<string, string> {
// the last segment is always the star one since it's terminal
var segmentsLimit = this.segments.length - 1;
var containsStarSegment =
segmentsLimit >= 0 && this.segments[segmentsLimit] instanceof StarSegment;
var matrixString;
if (!containsStarSegment) {
var matches =
RegExpWrapper.firstMatch(RegExpWrapper.create('^(.*\/[^\/]+?)(;[^\/]+)?\/?$'), url);
if (isPresent(matches)) {
url = matches[1];
matrixString = matches[2];
}
url = StringWrapper.replaceAll(url, /(;[^\/]+)(?=(\/|\Z))/g, '');
}
var params = StringMapWrapper.create(); var params = StringMapWrapper.create();
var urlPart = url; var urlPart = url;
for (var i = 0; i < this.segments.length; i++) {
for (var i = 0; i <= segmentsLimit; i++) {
var segment = this.segments[i]; var segment = this.segments[i];
if (segment instanceof ContinuationSegment) { if (segment instanceof ContinuationSegment) {
continue; continue;
@ -179,16 +239,45 @@ export class PathRecognizer {
var match = RegExpWrapper.firstMatch(RegExpWrapper.create('/' + segment.regex), urlPart); var match = RegExpWrapper.firstMatch(RegExpWrapper.create('/' + segment.regex), urlPart);
urlPart = StringWrapper.substring(urlPart, match[0].length); urlPart = StringWrapper.substring(urlPart, match[0].length);
if (segment.name.length > 0) { if (segment.name.length > 0) {
StringMapWrapper.set(params, segment.name, match[1]); params[segment.name] = match[1];
} }
} }
if (isPresent(matrixString) && matrixString.length > 0 && matrixString[0] == ';') {
parseAndAssignMatrixParams(params, matrixString);
}
return params; return params;
} }
generate(params: StringMap<string, string>): string { generate(params: StringMap<string, any>): string {
return ListWrapper.join(ListWrapper.map(this.segments, (segment) => segment.generate(params)), var paramTokens = new TouchMap(params);
'/'); var applyLeadingSlash = false;
var url = '';
for (var i = 0; i < this.segments.length; i++) {
let segment = this.segments[i];
let s = segment.generate(paramTokens);
applyLeadingSlash = applyLeadingSlash || (segment instanceof ContinuationSegment);
if (s.length > 0) {
url += (i > 0 ? '/' : '') + s;
}
}
var unusedParams = paramTokens.getUnused();
StringMapWrapper.forEach(unusedParams, (value, key) => {
url += ';' + key;
if (isPresent(value)) {
url += '=' + value;
}
});
if (applyLeadingSlash) {
url += '/';
}
return url;
} }
resolveComponentType(): Promise<any> { return this.handler.resolveComponentType(); } resolveComponentType(): Promise<any> { return this.handler.resolveComponentType(); }

View File

@ -0,0 +1,72 @@
import {
AsyncTestCompleter,
describe,
it,
iit,
ddescribe,
expect,
inject,
beforeEach,
SpyObject
} from 'angular2/test_lib';
import {PathRecognizer} from 'angular2/src/router/path_recognizer';
import {SyncRouteHandler} from 'angular2/src/router/sync_route_handler';
class DummyClass {
constructor() {}
}
var mockRouteHandler = new SyncRouteHandler(DummyClass);
export function main() {
describe('PathRecognizer', () => {
describe('matrix params', () => {
it('should recognize a trailing matrix value on a path value and assign it to the params return value',
() => {
var rec = new PathRecognizer('/hello/:id', mockRouteHandler);
var params = rec.parseParams('/hello/matias;key=value');
expect(params['id']).toEqual('matias');
expect(params['key']).toEqual('value');
});
it('should recognize and parse multiple matrix params separated by a colon value', () => {
var rec = new PathRecognizer('/jello/:sid', mockRouteHandler);
var params = rec.parseParams('/jello/man;color=red;height=20');
expect(params['sid']).toEqual('man');
expect(params['color']).toEqual('red');
expect(params['height']).toEqual('20');
});
it('should recognize a matrix param value on a static path value', () => {
var rec = new PathRecognizer('/static/man', mockRouteHandler);
var params = rec.parseParams('/static/man;name=dave');
expect(params['name']).toEqual('dave');
});
it('should not parse matrix params when a wildcard segment is used', () => {
var rec = new PathRecognizer('/wild/*everything', mockRouteHandler);
var params = rec.parseParams('/wild/super;variable=value');
expect(params['everything']).toEqual('super;variable=value');
});
it('should set matrix param values to true when no value is present within the path string',
() => {
var rec = new PathRecognizer('/path', mockRouteHandler);
var params = rec.parseParams('/path;one;two;three=3');
expect(params['one']).toEqual(true);
expect(params['two']).toEqual(true);
expect(params['three']).toEqual('3');
});
it('should ignore earlier instances of matrix params and only consider the ones at the end of the path',
() => {
var rec = new PathRecognizer('/one/two/three', mockRouteHandler);
var params = rec.parseParams('/one;a=1/two;b=2/three;c=3');
expect(params).toEqual({'c': '3'});
});
});
});
}

View File

@ -10,6 +10,8 @@ import {
SpyObject SpyObject
} from 'angular2/test_lib'; } from 'angular2/test_lib';
import {Map, StringMap, StringMapWrapper} from 'angular2/src/facade/collection';
import {RouteRecognizer, RouteMatch} from 'angular2/src/router/route_recognizer'; import {RouteRecognizer, RouteMatch} from 'angular2/src/router/route_recognizer';
export function main() { export function main() {
@ -122,6 +124,79 @@ export function main() {
expect(() => recognizer.generate('user', {})['url']) expect(() => recognizer.generate('user', {})['url'])
.toThrowError('Route generator for \'name\' was not included in parameters passed.'); .toThrowError('Route generator for \'name\' was not included in parameters passed.');
}); });
describe('matrix params', () => {
it('should recognize matrix parameters within the URL path', () => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('profile/:name', handler, 'user');
var solution = recognizer.recognize('/profile/matsko;comments=all')[0];
var params = solution.params();
expect(params['name']).toEqual('matsko');
expect(params['comments']).toEqual('all');
});
it('should recognize multiple matrix params and set parameters that contain no value to true',
() => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('/profile/hello', handler, 'user');
var solution =
recognizer.recognize('/profile/hello;modal;showAll=true;hideAll=false')[0];
var params = solution.params();
expect(params['modal']).toEqual(true);
expect(params['showAll']).toEqual('true');
expect(params['hideAll']).toEqual('false');
});
it('should only consider the matrix parameters at the end of the path handler', () => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('/profile/hi/:name', handler, 'user');
var solution = recognizer.recognize('/profile;a=1/hi;b=2;c=3/william;d=4')[0];
var params = solution.params();
expect(params).toEqual({'name': 'william', 'd': '4'});
});
it('should generate and populate the given static-based route with matrix params', () => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('forum/featured', handler, 'forum-page');
var params = StringMapWrapper.create();
params['start'] = 10;
params['end'] = 100;
var result = recognizer.generate('forum-page', params);
expect(result['url']).toEqual('forum/featured;start=10;end=100');
});
it('should generate and populate the given dynamic-based route with matrix params', () => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('forum/:topic', handler, 'forum-page');
var params = StringMapWrapper.create();
params['topic'] = 'crazy';
params['total-posts'] = 100;
params['moreDetail'] = null;
var result = recognizer.generate('forum-page', params);
expect(result['url']).toEqual('forum/crazy;total-posts=100;moreDetail');
});
it('should not apply any matrix params if a dynamic route segment takes up the slot when a path is generated',
() => {
var recognizer = new RouteRecognizer();
recognizer.addConfig('hello/:name', handler, 'profile-page');
var params = StringMapWrapper.create();
params['name'] = 'matsko';
var result = recognizer.generate('profile-page', params);
expect(result['url']).toEqual('hello/matsko');
});
});
}); });
} }

View File

@ -115,6 +115,26 @@ export function main() {
expect(router.generate(['/firstCmp', 'secondCmp'])).toEqual('/first/second'); expect(router.generate(['/firstCmp', 'secondCmp'])).toEqual('/first/second');
expect(router.generate(['/firstCmp/secondCmp'])).toEqual('/first/second'); expect(router.generate(['/firstCmp/secondCmp'])).toEqual('/first/second');
}); });
describe('matrix params', () => {
it('should apply inline matrix params for each router path within the generated URL', () => {
router.config({'path': '/first/...', 'component': DummyParentComp, 'as': 'firstCmp'});
var path =
router.generate(['/firstCmp', {'key': 'value'}, 'secondCmp', {'project': 'angular'}]);
expect(path).toEqual('/first;key=value/second;project=angular');
});
it('should apply inline matrix params for each router path within the generated URL and also include named params',
() => {
router.config(
{'path': '/first/:token/...', 'component': DummyParentComp, 'as': 'firstCmp'});
var path =
router.generate(['/firstCmp', {'token': 'min'}, 'secondCmp', {'author': 'max'}]);
expect(path).toEqual('/first/min/second;author=max');
});
});
}); });
} }