angular2
Version:
Angular 2 - a web framework for modern web apps
242 lines • 36 kB
JavaScript
'use strict';var lang_1 = require('angular2/src/facade/lang');
var exceptions_1 = require('angular2/src/facade/exceptions');
var collection_1 = require('angular2/src/facade/collection');
var url_parser_1 = require('./url_parser');
var TouchMap = (function () {
function TouchMap(map) {
var _this = this;
this.map = {};
this.keys = {};
if (lang_1.isPresent(map)) {
collection_1.StringMapWrapper.forEach(map, function (value, key) {
_this.map[key] = lang_1.isPresent(value) ? value.toString() : null;
_this.keys[key] = true;
});
}
}
TouchMap.prototype.get = function (key) {
collection_1.StringMapWrapper.delete(this.keys, key);
return this.map[key];
};
TouchMap.prototype.getUnused = function () {
var _this = this;
var unused = {};
var keys = collection_1.StringMapWrapper.keys(this.keys);
keys.forEach(function (key) { return unused[key] = collection_1.StringMapWrapper.get(_this.map, key); });
return unused;
};
return TouchMap;
})();
function normalizeString(obj) {
if (lang_1.isBlank(obj)) {
return null;
}
else {
return obj.toString();
}
}
var ContinuationSegment = (function () {
function ContinuationSegment() {
this.name = '';
}
ContinuationSegment.prototype.generate = function (params) { return ''; };
ContinuationSegment.prototype.match = function (path) { return true; };
return ContinuationSegment;
})();
var StaticSegment = (function () {
function StaticSegment(path) {
this.path = path;
this.name = '';
}
StaticSegment.prototype.match = function (path) { return path == this.path; };
StaticSegment.prototype.generate = function (params) { return this.path; };
return StaticSegment;
})();
var DynamicSegment = (function () {
function DynamicSegment(name) {
this.name = name;
}
DynamicSegment.prototype.match = function (path) { return path.length > 0; };
DynamicSegment.prototype.generate = function (params) {
if (!collection_1.StringMapWrapper.contains(params.map, this.name)) {
throw new exceptions_1.BaseException("Route generator for '" + this.name + "' was not included in parameters passed.");
}
return normalizeString(params.get(this.name));
};
return DynamicSegment;
})();
var StarSegment = (function () {
function StarSegment(name) {
this.name = name;
}
StarSegment.prototype.match = function (path) { return true; };
StarSegment.prototype.generate = function (params) { return normalizeString(params.get(this.name)); };
return StarSegment;
})();
var paramMatcher = /^:([^\/]+)$/g;
var wildcardMatcher = /^\*([^\/]+)$/g;
function parsePathString(route) {
// normalize route as not starting with a "/". Recognition will
// also normalize.
if (route.startsWith("/")) {
route = route.substring(1);
}
var segments = splitBySlash(route);
var results = [];
var specificity = '';
// a single slash (or "empty segment" is as specific as a static segment
if (segments.length == 0) {
specificity += '2';
}
// The "specificity" of a path is used to determine which route is used when multiple routes match
// a URL. Static segments (like "/foo") are the most specific, followed by dynamic segments (like
// "/:id"). Star segments add no specificity. Segments at the start of the path are more specific
// than proceeding ones.
//
// The code below uses place values to combine the different types of segments into a single
// string that we can sort later. Each static segment is marked as a specificity of "2," each
// dynamic segment is worth "1" specificity, and stars are worth "0" specificity.
var limit = segments.length - 1;
for (var i = 0; i <= limit; i++) {
var segment = segments[i], match;
if (lang_1.isPresent(match = lang_1.RegExpWrapper.firstMatch(paramMatcher, segment))) {
results.push(new DynamicSegment(match[1]));
specificity += '1';
}
else if (lang_1.isPresent(match = lang_1.RegExpWrapper.firstMatch(wildcardMatcher, segment))) {
results.push(new StarSegment(match[1]));
specificity += '0';
}
else if (segment == '...') {
if (i < limit) {
throw new exceptions_1.BaseException("Unexpected \"...\" before the end of the path for \"" + route + "\".");
}
results.push(new ContinuationSegment());
}
else {
results.push(new StaticSegment(segment));
specificity += '2';
}
}
return { 'segments': results, 'specificity': specificity };
}
// this function is used to determine whether a route config path like `/foo/:id` collides with
// `/foo/:name`
function pathDslHash(segments) {
return segments.map(function (segment) {
if (segment instanceof StarSegment) {
return '*';
}
else if (segment instanceof ContinuationSegment) {
return '...';
}
else if (segment instanceof DynamicSegment) {
return ':';
}
else if (segment instanceof StaticSegment) {
return segment.path;
}
})
.join('/');
}
function splitBySlash(url) {
return url.split('/');
}
var RESERVED_CHARS = lang_1.RegExpWrapper.create('//|\\(|\\)|;|\\?|=');
function assertPath(path) {
if (lang_1.StringWrapper.contains(path, '#')) {
throw new exceptions_1.BaseException("Path \"" + path + "\" should not include \"#\". Use \"HashLocationStrategy\" instead.");
}
var illegalCharacter = lang_1.RegExpWrapper.firstMatch(RESERVED_CHARS, path);
if (lang_1.isPresent(illegalCharacter)) {
throw new exceptions_1.BaseException("Path \"" + path + "\" contains \"" + illegalCharacter[0] + "\" which is not allowed in a route config.");
}
}
/**
* Parses a URL string using a given matcher DSL, and generates URLs from param maps
*/
var PathRecognizer = (function () {
function PathRecognizer(path) {
this.path = path;
this.terminal = true;
assertPath(path);
var parsed = parsePathString(path);
this._segments = parsed['segments'];
this.specificity = parsed['specificity'];
this.hash = pathDslHash(this._segments);
var lastSegment = this._segments[this._segments.length - 1];
this.terminal = !(lastSegment instanceof ContinuationSegment);
}
PathRecognizer.prototype.recognize = function (beginningSegment) {
var nextSegment = beginningSegment;
var currentSegment;
var positionalParams = {};
var captured = [];
for (var i = 0; i < this._segments.length; i += 1) {
var segment = this._segments[i];
currentSegment = nextSegment;
if (segment instanceof ContinuationSegment) {
break;
}
if (lang_1.isPresent(currentSegment)) {
// the star segment consumes all of the remaining URL, including matrix params
if (segment instanceof StarSegment) {
positionalParams[segment.name] = currentSegment.toString();
captured.push(currentSegment.toString());
nextSegment = null;
break;
}
captured.push(currentSegment.path);
if (segment instanceof DynamicSegment) {
positionalParams[segment.name] = currentSegment.path;
}
else if (!segment.match(currentSegment.path)) {
return null;
}
nextSegment = currentSegment.child;
}
else if (!segment.match('')) {
return null;
}
}
if (this.terminal && lang_1.isPresent(nextSegment)) {
return null;
}
var urlPath = captured.join('/');
var auxiliary;
var urlParams;
var allParams;
if (lang_1.isPresent(currentSegment)) {
// If this is the root component, read query params. Otherwise, read matrix params.
var paramsSegment = beginningSegment instanceof url_parser_1.RootUrl ? beginningSegment : currentSegment;
allParams = lang_1.isPresent(paramsSegment.params) ?
collection_1.StringMapWrapper.merge(paramsSegment.params, positionalParams) :
positionalParams;
urlParams = url_parser_1.serializeParams(paramsSegment.params);
auxiliary = currentSegment.auxiliary;
}
else {
allParams = positionalParams;
auxiliary = [];
urlParams = [];
}
return { urlPath: urlPath, urlParams: urlParams, allParams: allParams, auxiliary: auxiliary, nextSegment: nextSegment };
};
PathRecognizer.prototype.generate = function (params) {
var paramTokens = new TouchMap(params);
var path = [];
for (var i = 0; i < this._segments.length; i++) {
var segment = this._segments[i];
if (!(segment instanceof ContinuationSegment)) {
path.push(segment.generate(paramTokens));
}
}
var urlPath = path.join('/');
var nonPositionalParams = paramTokens.getUnused();
var urlParams = url_parser_1.serializeParams(nonPositionalParams);
return { urlPath: urlPath, urlParams: urlParams };
};
return PathRecognizer;
})();
exports.PathRecognizer = PathRecognizer;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"path_recognizer.js","sourceRoot":"","sources":["angular2/src/router/path_recognizer.ts"],"names":["TouchMap","TouchMap.constructor","TouchMap.get","TouchMap.getUnused","normalizeString","ContinuationSegment","ContinuationSegment.constructor","ContinuationSegment.generate","ContinuationSegment.match","StaticSegment","StaticSegment.constructor","StaticSegment.match","StaticSegment.generate","DynamicSegment","DynamicSegment.constructor","DynamicSegment.match","DynamicSegment.generate","StarSegment","StarSegment.constructor","StarSegment.match","StarSegment.generate","parsePathString","pathDslHash","splitBySlash","assertPath","PathRecognizer","PathRecognizer.constructor","PathRecognizer.recognize","PathRecognizer.generate"],"mappings":"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MA,CAACA,QAAQA,CAACA,GAAGA,CAACA,UAACA,OAAOA;QACXA,EAAEA,CAACA,CAACA,OAAOA,YAAYA,WAAWA,CAACA,CAACA,CAACA;YACnCA,MAAMA,CAACA,GAAGA,CAACA;QACbA,CAACA;QAACA,IAAIA,CAACA,EAAEA,CAACA,CAACA,OAAOA,YAAYA,mBAAmBA,CAACA,CAACA,CAACA;YAClDA,MAAMA,CAACA,KAAKA,CAACA;QACfA,CAACA;QAACA,IAAIA,CAACA,EAAEA,CAACA,CAACA,OAAOA,YAAYA,cAAcA,CAACA,CAACA,CAACA;YAC7CA,MAAMA,CAACA,GAAGA,CAACA;QACbA,CAACA;QAACA,IAAIA,CAACA,EAAEA,CAACA,CAACA,OAAOA,YAAYA,aAAaA,CAACA,CAACA,CAACA;YAC5CA,MAAMA,CAACA,OAAOA,CAACA,IAAIA,CAACA;QACtBA,CAACA;IACHA,CAACA,CAACA;SACZA,IAAIA,CAACA,GAAGA,CAACA,CAACA;AACjBA,CAACA;AAED,sBAAsB,GAAW;IAC/BC,MAAMA,CAACA,GAAGA,CAACA,KAAKA,CAACA,GAAGA,CAACA,CAACA;AACxBA,CAACA;AAED,IAAI,cAAc,GAAG,oBAAa,CAAC,MAAM,CAAC,oBAAoB,CAAC,CAAC;AAChE,oBAAoB,IAAY;IAC9BC,EAAEA,CAACA,CAACA,oBAAaA,CAACA,QAAQA,CAACA,IAAIA,EAAEA,GAAGA,CAACA,CAACA,CAACA,CAACA;QACtCA,MAAMA,IAAIA,0BAAaA,CACnBA,YAASA,IAAIA,uEAA+DA,CAACA,CAACA;IACpFA,CAACA;IACDA,IAAIA,gBAAgBA,GAAGA,oBAAaA,CAACA,UAAUA,CAACA,cAAcA,EAAEA,IAAIA,CAACA,CAACA;IACtEA,EAAEA,CAACA,CAACA,gBAASA,CAACA,gBAAgBA,CAACA,CAACA,CAACA,CAACA;QAChCA,MAAMA,IAAIA,0BAAaA,CACnBA,YAASA,IAAIA,sBAAeA,gBAAgBA,CAACA,CAACA,CAACA,+CAA2CA,CAACA,CAACA;IAClGA,CAACA;AACHA,CAACA;AAGD;;GAEG;AAC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","sourcesContent":["import {\n  RegExp,\n  RegExpWrapper,\n  RegExpMatcherWrapper,\n  StringWrapper,\n  isPresent,\n  isBlank\n} from 'angular2/src/facade/lang';\nimport {BaseException, WrappedException} from 'angular2/src/facade/exceptions';\nimport {Map, MapWrapper, StringMapWrapper} from 'angular2/src/facade/collection';\n\nimport {Url, RootUrl, serializeParams} from './url_parser';\n\nclass TouchMap {\n  map: {[key: string]: string} = {};\n  keys: {[key: string]: boolean} = {};\n\n  constructor(map: {[key: string]: any}) {\n    if (isPresent(map)) {\n      StringMapWrapper.forEach(map, (value, key) => {\n        this.map[key] = isPresent(value) ? value.toString() : null;\n        this.keys[key] = true;\n      });\n    }\n  }\n\n  get(key: string): string {\n    StringMapWrapper.delete(this.keys, key);\n    return this.map[key];\n  }\n\n  getUnused(): {[key: string]: any} {\n    var unused: {[key: string]: any} = {};\n    var keys = StringMapWrapper.keys(this.keys);\n    keys.forEach(key => unused[key] = StringMapWrapper.get(this.map, key));\n    return unused;\n  }\n}\n\nfunction normalizeString(obj: any): string {\n  if (isBlank(obj)) {\n    return null;\n  } else {\n    return obj.toString();\n  }\n}\n\ninterface Segment {\n  name: string;\n  generate(params: TouchMap): string;\n  match(path: string): boolean;\n}\n\nclass ContinuationSegment implements Segment {\n  name: string = '';\n  generate(params: TouchMap): string { return ''; }\n  match(path: string): boolean { return true; }\n}\n\nclass StaticSegment implements Segment {\n  name: string = '';\n  constructor(public path: string) {}\n  match(path: string): boolean { return path == this.path; }\n  generate(params: TouchMap): string { return this.path; }\n}\n\nclass DynamicSegment implements Segment {\n  constructor(public name: string) {}\n  match(path: string): boolean { return path.length > 0; }\n  generate(params: TouchMap): string {\n    if (!StringMapWrapper.contains(params.map, this.name)) {\n      throw new BaseException(\n          `Route generator for '${this.name}' was not included in parameters passed.`);\n    }\n    return normalizeString(params.get(this.name));\n  }\n}\n\n\nclass StarSegment implements Segment {\n  constructor(public name: string) {}\n  match(path: string): boolean { return true; }\n  generate(params: TouchMap): string { return normalizeString(params.get(this.name)); }\n}\n\n\nvar paramMatcher = /^:([^\\/]+)$/g;\nvar wildcardMatcher = /^\\*([^\\/]+)$/g;\n\nfunction parsePathString(route: string): {[key: string]: any} {\n  // normalize route as not starting with a \"/\". Recognition will\n  // also normalize.\n  if (route.startsWith(\"/\")) {\n    route = route.substring(1);\n  }\n\n  var segments = splitBySlash(route);\n  var results = [];\n\n  var specificity = '';\n\n  // a single slash (or \"empty segment\" is as specific as a static segment\n  if (segments.length == 0) {\n    specificity += '2';\n  }\n\n  // The \"specificity\" of a path is used to determine which route is used when multiple routes match\n  // a URL. Static segments (like \"/foo\") are the most specific, followed by dynamic segments (like\n  // \"/:id\"). Star segments add no specificity. Segments at the start of the path are more specific\n  // than proceeding ones.\n  //\n  // The code below uses place values to combine the different types of segments into a single\n  // string that we can sort later. Each static segment is marked as a specificity of \"2,\" each\n  // dynamic segment is worth \"1\" specificity, and stars are worth \"0\" specificity.\n\n  var limit = segments.length - 1;\n  for (var i = 0; i <= limit; i++) {\n    var segment = segments[i], match;\n\n    if (isPresent(match = RegExpWrapper.firstMatch(paramMatcher, segment))) {\n      results.push(new DynamicSegment(match[1]));\n      specificity += '1';\n    } else if (isPresent(match = RegExpWrapper.firstMatch(wildcardMatcher, segment))) {\n      results.push(new StarSegment(match[1]));\n      specificity += '0';\n    } else if (segment == '...') {\n      if (i < limit) {\n        throw new BaseException(`Unexpected \"...\" before the end of the path for \"${route}\".`);\n      }\n      results.push(new ContinuationSegment());\n    } else {\n      results.push(new StaticSegment(segment));\n      specificity += '2';\n    }\n  }\n\n  return {'segments': results, 'specificity': specificity};\n}\n\n// this function is used to determine whether a route config path like `/foo/:id` collides with\n// `/foo/:name`\nfunction pathDslHash(segments: Segment[]): string {\n  return segments.map((segment) => {\n                   if (segment instanceof StarSegment) {\n                     return '*';\n                   } else if (segment instanceof ContinuationSegment) {\n                     return '...';\n                   } else if (segment instanceof DynamicSegment) {\n                     return ':';\n                   } else if (segment instanceof StaticSegment) {\n                     return segment.path;\n                   }\n                 })\n      .join('/');\n}\n\nfunction splitBySlash(url: string): string[] {\n  return url.split('/');\n}\n\nvar RESERVED_CHARS = RegExpWrapper.create('//|\\\\(|\\\\)|;|\\\\?|=');\nfunction assertPath(path: string) {\n  if (StringWrapper.contains(path, '#')) {\n    throw new BaseException(\n        `Path \"${path}\" should not include \"#\". Use \"HashLocationStrategy\" instead.`);\n  }\n  var illegalCharacter = RegExpWrapper.firstMatch(RESERVED_CHARS, path);\n  if (isPresent(illegalCharacter)) {\n    throw new BaseException(\n        `Path \"${path}\" contains \"${illegalCharacter[0]}\" which is not allowed in a route config.`);\n  }\n}\n\n\n/**\n * Parses a URL string using a given matcher DSL, and generates URLs from param maps\n */\nexport class PathRecognizer {\n  private _segments: Segment[];\n  specificity: string;\n  terminal: boolean = true;\n  hash: string;\n\n  constructor(public path: string) {\n    assertPath(path);\n    var parsed = parsePathString(path);\n\n    this._segments = parsed['segments'];\n    this.specificity = parsed['specificity'];\n    this.hash = pathDslHash(this._segments);\n\n    var lastSegment = this._segments[this._segments.length - 1];\n    this.terminal = !(lastSegment instanceof ContinuationSegment);\n  }\n\n  recognize(beginningSegment: Url): {[key: string]: any} {\n    var nextSegment = beginningSegment;\n    var currentSegment: Url;\n    var positionalParams = {};\n    var captured = [];\n\n    for (var i = 0; i < this._segments.length; i += 1) {\n      var segment = this._segments[i];\n\n      currentSegment = nextSegment;\n      if (segment instanceof ContinuationSegment) {\n        break;\n      }\n\n      if (isPresent(currentSegment)) {\n        // the star segment consumes all of the remaining URL, including matrix params\n        if (segment instanceof StarSegment) {\n          positionalParams[segment.name] = currentSegment.toString();\n          captured.push(currentSegment.toString());\n          nextSegment = null;\n          break;\n        }\n\n        captured.push(currentSegment.path);\n\n        if (segment instanceof DynamicSegment) {\n          positionalParams[segment.name] = currentSegment.path;\n        } else if (!segment.match(currentSegment.path)) {\n          return null;\n        }\n\n        nextSegment = currentSegment.child;\n      } else if (!segment.match('')) {\n        return null;\n      }\n    }\n\n    if (this.terminal && isPresent(nextSegment)) {\n      return null;\n    }\n\n    var urlPath = captured.join('/');\n\n    var auxiliary;\n    var urlParams;\n    var allParams;\n    if (isPresent(currentSegment)) {\n      // If this is the root component, read query params. Otherwise, read matrix params.\n      var paramsSegment = beginningSegment instanceof RootUrl ? beginningSegment : currentSegment;\n\n      allParams = isPresent(paramsSegment.params) ?\n                      StringMapWrapper.merge(paramsSegment.params, positionalParams) :\n                      positionalParams;\n\n      urlParams = serializeParams(paramsSegment.params);\n\n\n      auxiliary = currentSegment.auxiliary;\n    } else {\n      allParams = positionalParams;\n      auxiliary = [];\n      urlParams = [];\n    }\n    return {urlPath, urlParams, allParams, auxiliary, nextSegment};\n  }\n\n\n  generate(params: {[key: string]: any}): {[key: string]: any} {\n    var paramTokens = new TouchMap(params);\n\n    var path = [];\n\n    for (var i = 0; i < this._segments.length; i++) {\n      let segment = this._segments[i];\n      if (!(segment instanceof ContinuationSegment)) {\n        path.push(segment.generate(paramTokens));\n      }\n    }\n    var urlPath = path.join('/');\n\n    var nonPositionalParams = paramTokens.getUnused();\n    var urlParams = serializeParams(nonPositionalParams);\n\n    return {urlPath, urlParams};\n  }\n}\n"]}