UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

260 lines 37 kB
import { RegExpWrapper, StringWrapper, isPresent, isBlank } from 'angular2/src/facade/lang'; import { BaseException } from 'angular2/src/facade/exceptions'; import { StringMapWrapper } from 'angular2/src/facade/collection'; import { TouchMap, normalizeString } from '../../utils'; import { RootUrl, convertUrlParamsToArray } from '../../url_parser'; import { GeneratedUrl, MatchedUrl } from './route_path'; /** * Identified by a `...` URL segment. This indicates that the * Route will continue to be matched by child `Router`s. */ class ContinuationPathSegment { constructor() { this.name = ''; this.specificity = ''; this.hash = '...'; } generate(params) { return ''; } match(path) { return true; } } /** * Identified by a string not starting with a `:` or `*`. * Only matches the URL segments that equal the segment path */ class StaticPathSegment { constructor(path) { this.path = path; this.name = ''; this.specificity = '2'; this.hash = path; } match(path) { return path == this.path; } generate(params) { return this.path; } } /** * Identified by a string starting with `:`. Indicates a segment * that can contain a value that will be extracted and provided to * a matching `Instruction`. */ class DynamicPathSegment { constructor(name) { this.name = name; this.specificity = '1'; this.hash = ':'; } match(path) { return path.length > 0; } generate(params) { if (!StringMapWrapper.contains(params.map, this.name)) { throw new BaseException(`Route generator for '${this.name}' was not included in parameters passed.`); } return encodeDynamicSegment(normalizeString(params.get(this.name))); } } DynamicPathSegment.paramMatcher = /^:([^\/]+)$/g; /** * Identified by a string starting with `*` Indicates that all the following * segments match this route and that the value of these segments should * be provided to a matching `Instruction`. */ class StarPathSegment { constructor(name) { this.name = name; this.specificity = '0'; this.hash = '*'; } match(path) { return true; } generate(params) { return normalizeString(params.get(this.name)); } } StarPathSegment.wildcardMatcher = /^\*([^\/]+)$/g; /** * Parses a URL string using a given matcher DSL, and generates URLs from param maps */ export class ParamRoutePath { /** * Takes a string representing the matcher DSL */ constructor(routePath) { this.routePath = routePath; this.terminal = true; this._assertValidPath(routePath); this._parsePathString(routePath); this.specificity = this._calculateSpecificity(); this.hash = this._calculateHash(); var lastSegment = this._segments[this._segments.length - 1]; this.terminal = !(lastSegment instanceof ContinuationPathSegment); } matchUrl(url) { var nextUrlSegment = url; var currentUrlSegment; var positionalParams = {}; var captured = []; for (var i = 0; i < this._segments.length; i += 1) { var pathSegment = this._segments[i]; currentUrlSegment = nextUrlSegment; if (pathSegment instanceof ContinuationPathSegment) { break; } if (isPresent(currentUrlSegment)) { // the star segment consumes all of the remaining URL, including matrix params if (pathSegment instanceof StarPathSegment) { positionalParams[pathSegment.name] = currentUrlSegment.toString(); captured.push(currentUrlSegment.toString()); nextUrlSegment = null; break; } captured.push(currentUrlSegment.path); if (pathSegment instanceof DynamicPathSegment) { positionalParams[pathSegment.name] = decodeDynamicSegment(currentUrlSegment.path); } else if (!pathSegment.match(currentUrlSegment.path)) { return null; } nextUrlSegment = currentUrlSegment.child; } else if (!pathSegment.match('')) { return null; } } if (this.terminal && isPresent(nextUrlSegment)) { return null; } var urlPath = captured.join('/'); var auxiliary = []; var urlParams = []; var allParams = positionalParams; if (isPresent(currentUrlSegment)) { // If this is the root component, read query params. Otherwise, read matrix params. var paramsSegment = url instanceof RootUrl ? url : currentUrlSegment; if (isPresent(paramsSegment.params)) { allParams = StringMapWrapper.merge(paramsSegment.params, positionalParams); urlParams = convertUrlParamsToArray(paramsSegment.params); } else { allParams = positionalParams; } auxiliary = currentUrlSegment.auxiliary; } return new MatchedUrl(urlPath, urlParams, allParams, auxiliary, nextUrlSegment); } generateUrl(params) { var paramTokens = new TouchMap(params); var path = []; for (var i = 0; i < this._segments.length; i++) { let segment = this._segments[i]; if (!(segment instanceof ContinuationPathSegment)) { path.push(segment.generate(paramTokens)); } } var urlPath = path.join('/'); var nonPositionalParams = paramTokens.getUnused(); var urlParams = nonPositionalParams; return new GeneratedUrl(urlPath, urlParams); } toString() { return this.routePath; } _parsePathString(routePath) { // normalize route as not starting with a "/". Recognition will // also normalize. if (routePath.startsWith("/")) { routePath = routePath.substring(1); } var segmentStrings = routePath.split('/'); this._segments = []; var limit = segmentStrings.length - 1; for (var i = 0; i <= limit; i++) { var segment = segmentStrings[i], match; if (isPresent(match = RegExpWrapper.firstMatch(DynamicPathSegment.paramMatcher, segment))) { this._segments.push(new DynamicPathSegment(match[1])); } else if (isPresent(match = RegExpWrapper.firstMatch(StarPathSegment.wildcardMatcher, segment))) { this._segments.push(new StarPathSegment(match[1])); } else if (segment == '...') { if (i < limit) { throw new BaseException(`Unexpected "..." before the end of the path for "${routePath}".`); } this._segments.push(new ContinuationPathSegment()); } else { this._segments.push(new StaticPathSegment(segment)); } } } _calculateSpecificity() { // 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 i, length = this._segments.length, specificity; if (length == 0) { // a single slash (or "empty segment" is as specific as a static segment specificity += '2'; } else { specificity = ''; for (i = 0; i < length; i++) { specificity += this._segments[i].specificity; } } return specificity; } _calculateHash() { // this function is used to determine whether a route config path like `/foo/:id` collides with // `/foo/:name` var i, length = this._segments.length; var hashParts = []; for (i = 0; i < length; i++) { hashParts.push(this._segments[i].hash); } return hashParts.join('/'); } _assertValidPath(path) { if (StringWrapper.contains(path, '#')) { throw new BaseException(`Path "${path}" should not include "#". Use "HashLocationStrategy" instead.`); } var illegalCharacter = RegExpWrapper.firstMatch(ParamRoutePath.RESERVED_CHARS, path); if (isPresent(illegalCharacter)) { throw new BaseException(`Path "${path}" contains "${illegalCharacter[0]}" which is not allowed in a route config.`); } } } ParamRoutePath.RESERVED_CHARS = RegExpWrapper.create('//|\\(|\\)|;|\\?|='); let REGEXP_PERCENT = /%/g; let REGEXP_SLASH = /\//g; let REGEXP_OPEN_PARENT = /\(/g; let REGEXP_CLOSE_PARENT = /\)/g; let REGEXP_SEMICOLON = /;/g; function encodeDynamicSegment(value) { if (isBlank(value)) { return null; } value = StringWrapper.replaceAll(value, REGEXP_PERCENT, '%25'); value = StringWrapper.replaceAll(value, REGEXP_SLASH, '%2F'); value = StringWrapper.replaceAll(value, REGEXP_OPEN_PARENT, '%28'); value = StringWrapper.replaceAll(value, REGEXP_CLOSE_PARENT, '%29'); value = StringWrapper.replaceAll(value, REGEXP_SEMICOLON, '%3B'); return value; } let REGEXP_ENC_SEMICOLON = /%3B/ig; let REGEXP_ENC_CLOSE_PARENT = /%29/ig; let REGEXP_ENC_OPEN_PARENT = /%28/ig; let REGEXP_ENC_SLASH = /%2F/ig; let REGEXP_ENC_PERCENT = /%25/ig; function decodeDynamicSegment(value) { if (isBlank(value)) { return null; } value = StringWrapper.replaceAll(value, REGEXP_ENC_SEMICOLON, ';'); value = StringWrapper.replaceAll(value, REGEXP_ENC_CLOSE_PARENT, ')'); value = StringWrapper.replaceAll(value, REGEXP_ENC_OPEN_PARENT, '('); value = StringWrapper.replaceAll(value, REGEXP_ENC_SLASH, '/'); value = StringWrapper.replaceAll(value, REGEXP_ENC_PERCENT, '%'); return value; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"param_route_path.js","sourceRoot":"","sources":["diffing_plugin_wrapper-output_path-xBLIBrVR.tmp/angular2/src/router/rules/route_paths/param_route_path.ts"],"names":[],"mappings":"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","sourcesContent":["import {RegExpWrapper, StringWrapper, isPresent, isBlank} from 'angular2/src/facade/lang';\nimport {BaseException} from 'angular2/src/facade/exceptions';\nimport {StringMapWrapper} from 'angular2/src/facade/collection';\n\nimport {TouchMap, normalizeString} from '../../utils';\nimport {Url, RootUrl, convertUrlParamsToArray} from '../../url_parser';\nimport {RoutePath, GeneratedUrl, MatchedUrl} from './route_path';\n\n\n\n/**\n * `ParamRoutePath`s are made up of `PathSegment`s, each of which can\n * match a segment of a URL. Different kind of `PathSegment`s match\n * URL segments in different ways...\n */\ninterface PathSegment {\n  name: string;\n  generate(params: TouchMap): string;\n  match(path: string): boolean;\n  specificity: string;\n  hash: string;\n}\n\n/**\n * Identified by a `...` URL segment. This indicates that the\n * Route will continue to be matched by child `Router`s.\n */\nclass ContinuationPathSegment implements PathSegment {\n  name: string = '';\n  specificity = '';\n  hash = '...';\n  generate(params: TouchMap): string { return ''; }\n  match(path: string): boolean { return true; }\n}\n\n/**\n * Identified by a string not starting with a `:` or `*`.\n * Only matches the URL segments that equal the segment path\n */\nclass StaticPathSegment implements PathSegment {\n  name: string = '';\n  specificity = '2';\n  hash: string;\n  constructor(public path: string) { this.hash = path; }\n  match(path: string): boolean { return path == this.path; }\n  generate(params: TouchMap): string { return this.path; }\n}\n\n/**\n * Identified by a string starting with `:`. Indicates a segment\n * that can contain a value that will be extracted and provided to\n * a matching `Instruction`.\n */\nclass DynamicPathSegment implements PathSegment {\n  static paramMatcher = /^:([^\\/]+)$/g;\n  specificity = '1';\n  hash = ':';\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 encodeDynamicSegment(normalizeString(params.get(this.name)));\n  }\n}\n\n/**\n * Identified by a string starting with `*` Indicates that all the following\n * segments match this route and that the value of these segments should\n * be provided to a matching `Instruction`.\n */\nclass StarPathSegment implements PathSegment {\n  static wildcardMatcher = /^\\*([^\\/]+)$/g;\n  specificity = '0';\n  hash = '*';\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/**\n * Parses a URL string using a given matcher DSL, and generates URLs from param maps\n */\nexport class ParamRoutePath implements RoutePath {\n  specificity: string;\n  terminal: boolean = true;\n  hash: string;\n\n  private _segments: PathSegment[];\n\n  /**\n   * Takes a string representing the matcher DSL\n   */\n  constructor(public routePath: string) {\n    this._assertValidPath(routePath);\n\n    this._parsePathString(routePath);\n    this.specificity = this._calculateSpecificity();\n    this.hash = this._calculateHash();\n\n    var lastSegment = this._segments[this._segments.length - 1];\n    this.terminal = !(lastSegment instanceof ContinuationPathSegment);\n  }\n\n  matchUrl(url: Url): MatchedUrl {\n    var nextUrlSegment = url;\n    var currentUrlSegment: Url;\n    var positionalParams = {};\n    var captured: string[] = [];\n\n    for (var i = 0; i < this._segments.length; i += 1) {\n      var pathSegment = this._segments[i];\n\n      currentUrlSegment = nextUrlSegment;\n      if (pathSegment instanceof ContinuationPathSegment) {\n        break;\n      }\n\n      if (isPresent(currentUrlSegment)) {\n        // the star segment consumes all of the remaining URL, including matrix params\n        if (pathSegment instanceof StarPathSegment) {\n          positionalParams[pathSegment.name] = currentUrlSegment.toString();\n          captured.push(currentUrlSegment.toString());\n          nextUrlSegment = null;\n          break;\n        }\n\n        captured.push(currentUrlSegment.path);\n\n        if (pathSegment instanceof DynamicPathSegment) {\n          positionalParams[pathSegment.name] = decodeDynamicSegment(currentUrlSegment.path);\n        } else if (!pathSegment.match(currentUrlSegment.path)) {\n          return null;\n        }\n\n        nextUrlSegment = currentUrlSegment.child;\n      } else if (!pathSegment.match('')) {\n        return null;\n      }\n    }\n\n    if (this.terminal && isPresent(nextUrlSegment)) {\n      return null;\n    }\n\n    var urlPath = captured.join('/');\n\n    var auxiliary = [];\n    var urlParams = [];\n    var allParams = positionalParams;\n    if (isPresent(currentUrlSegment)) {\n      // If this is the root component, read query params. Otherwise, read matrix params.\n      var paramsSegment = url instanceof RootUrl ? url : currentUrlSegment;\n\n      if (isPresent(paramsSegment.params)) {\n        allParams = StringMapWrapper.merge(paramsSegment.params, positionalParams);\n        urlParams = convertUrlParamsToArray(paramsSegment.params);\n      } else {\n        allParams = positionalParams;\n      }\n      auxiliary = currentUrlSegment.auxiliary;\n    }\n\n    return new MatchedUrl(urlPath, urlParams, allParams, auxiliary, nextUrlSegment);\n  }\n\n\n  generateUrl(params: {[key: string]: any}): GeneratedUrl {\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 ContinuationPathSegment)) {\n        path.push(segment.generate(paramTokens));\n      }\n    }\n    var urlPath = path.join('/');\n\n    var nonPositionalParams = paramTokens.getUnused();\n    var urlParams = nonPositionalParams;\n\n    return new GeneratedUrl(urlPath, urlParams);\n  }\n\n\n  toString(): string { return this.routePath; }\n\n  private _parsePathString(routePath: string) {\n    // normalize route as not starting with a \"/\". Recognition will\n    // also normalize.\n    if (routePath.startsWith(\"/\")) {\n      routePath = routePath.substring(1);\n    }\n\n    var segmentStrings = routePath.split('/');\n    this._segments = [];\n\n    var limit = segmentStrings.length - 1;\n    for (var i = 0; i <= limit; i++) {\n      var segment = segmentStrings[i], match;\n\n      if (isPresent(match = RegExpWrapper.firstMatch(DynamicPathSegment.paramMatcher, segment))) {\n        this._segments.push(new DynamicPathSegment(match[1]));\n      } else if (isPresent(\n                     match = RegExpWrapper.firstMatch(StarPathSegment.wildcardMatcher, segment))) {\n        this._segments.push(new StarPathSegment(match[1]));\n      } else if (segment == '...') {\n        if (i < limit) {\n          throw new BaseException(\n              `Unexpected \"...\" before the end of the path for \"${routePath}\".`);\n        }\n        this._segments.push(new ContinuationPathSegment());\n      } else {\n        this._segments.push(new StaticPathSegment(segment));\n      }\n    }\n  }\n\n  private _calculateSpecificity(): string {\n    // The \"specificity\" of a path is used to determine which route is used when multiple routes\n    // match\n    // a URL. Static segments (like \"/foo\") are the most specific, followed by dynamic segments\n    // (like\n    // \"/:id\"). Star segments add no specificity. Segments at the start of the path are more\n    // 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    var i, length = this._segments.length, specificity;\n    if (length == 0) {\n      // a single slash (or \"empty segment\" is as specific as a static segment\n      specificity += '2';\n    } else {\n      specificity = '';\n      for (i = 0; i < length; i++) {\n        specificity += this._segments[i].specificity;\n      }\n    }\n    return specificity;\n  }\n\n  private _calculateHash(): string {\n    // this function is used to determine whether a route config path like `/foo/:id` collides with\n    // `/foo/:name`\n    var i, length = this._segments.length;\n    var hashParts = [];\n    for (i = 0; i < length; i++) {\n      hashParts.push(this._segments[i].hash);\n    }\n    return hashParts.join('/');\n  }\n\n  private _assertValidPath(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(ParamRoutePath.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  static RESERVED_CHARS = RegExpWrapper.create('//|\\\\(|\\\\)|;|\\\\?|=');\n}\n\nlet REGEXP_PERCENT = /%/g;\nlet REGEXP_SLASH = /\\//g;\nlet REGEXP_OPEN_PARENT = /\\(/g;\nlet REGEXP_CLOSE_PARENT = /\\)/g;\nlet REGEXP_SEMICOLON = /;/g;\n\nfunction encodeDynamicSegment(value: string): string {\n  if (isBlank(value)) {\n    return null;\n  }\n\n  value = StringWrapper.replaceAll(value, REGEXP_PERCENT, '%25');\n  value = StringWrapper.replaceAll(value, REGEXP_SLASH, '%2F');\n  value = StringWrapper.replaceAll(value, REGEXP_OPEN_PARENT, '%28');\n  value = StringWrapper.replaceAll(value, REGEXP_CLOSE_PARENT, '%29');\n  value = StringWrapper.replaceAll(value, REGEXP_SEMICOLON, '%3B');\n\n  return value;\n}\n\nlet REGEXP_ENC_SEMICOLON = /%3B/ig;\nlet REGEXP_ENC_CLOSE_PARENT = /%29/ig;\nlet REGEXP_ENC_OPEN_PARENT = /%28/ig;\nlet REGEXP_ENC_SLASH = /%2F/ig;\nlet REGEXP_ENC_PERCENT = /%25/ig;\n\nfunction decodeDynamicSegment(value: string): string {\n  if (isBlank(value)) {\n    return null;\n  }\n\n  value = StringWrapper.replaceAll(value, REGEXP_ENC_SEMICOLON, ';');\n  value = StringWrapper.replaceAll(value, REGEXP_ENC_CLOSE_PARENT, ')');\n  value = StringWrapper.replaceAll(value, REGEXP_ENC_OPEN_PARENT, '(');\n  value = StringWrapper.replaceAll(value, REGEXP_ENC_SLASH, '/');\n  value = StringWrapper.replaceAll(value, REGEXP_ENC_PERCENT, '%');\n\n  return value;\n}\n"]}