UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

128 lines 22.3 kB
'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 async_1 = require('angular2/src/facade/async'); var route_recognizer_1 = require('./route_recognizer'); var route_config_impl_1 = require('./route_config_impl'); var async_route_handler_1 = require('./async_route_handler'); var sync_route_handler_1 = require('./sync_route_handler'); /** * `ComponentRecognizer` is responsible for recognizing routes for a single component. * It is consumed by `RouteRegistry`, which knows how to recognize an entire hierarchy of * components. */ var ComponentRecognizer = (function () { function ComponentRecognizer() { this.names = new collection_1.Map(); // map from name to recognizer this.auxNames = new collection_1.Map(); // map from starting path to recognizer this.auxRoutes = new collection_1.Map(); // TODO: optimize this into a trie this.matchers = []; this.defaultRoute = null; } /** * returns whether or not the config is terminal */ ComponentRecognizer.prototype.config = function (config) { var handler; if (lang_1.isPresent(config.name) && config.name[0].toUpperCase() != config.name[0]) { var suggestedName = config.name[0].toUpperCase() + config.name.substring(1); throw new exceptions_1.BaseException("Route \"" + config.path + "\" with name \"" + config.name + "\" does not begin with an uppercase letter. Route names should be CamelCase like \"" + suggestedName + "\"."); } if (config instanceof route_config_impl_1.AuxRoute) { handler = new sync_route_handler_1.SyncRouteHandler(config.component, config.data); var path = config.path.startsWith('/') ? config.path.substring(1) : config.path; var recognizer = new route_recognizer_1.RouteRecognizer(config.path, handler); this.auxRoutes.set(path, recognizer); if (lang_1.isPresent(config.name)) { this.auxNames.set(config.name, recognizer); } return recognizer.terminal; } var useAsDefault = false; if (config instanceof route_config_impl_1.Redirect) { var redirector = new route_recognizer_1.RedirectRecognizer(config.path, config.redirectTo); this._assertNoHashCollision(redirector.hash, config.path); this.matchers.push(redirector); return true; } if (config instanceof route_config_impl_1.Route) { handler = new sync_route_handler_1.SyncRouteHandler(config.component, config.data); useAsDefault = lang_1.isPresent(config.useAsDefault) && config.useAsDefault; } else if (config instanceof route_config_impl_1.AsyncRoute) { handler = new async_route_handler_1.AsyncRouteHandler(config.loader, config.data); useAsDefault = lang_1.isPresent(config.useAsDefault) && config.useAsDefault; } var recognizer = new route_recognizer_1.RouteRecognizer(config.path, handler); this._assertNoHashCollision(recognizer.hash, config.path); if (useAsDefault) { if (lang_1.isPresent(this.defaultRoute)) { throw new exceptions_1.BaseException("Only one route can be default"); } this.defaultRoute = recognizer; } this.matchers.push(recognizer); if (lang_1.isPresent(config.name)) { this.names.set(config.name, recognizer); } return recognizer.terminal; }; ComponentRecognizer.prototype._assertNoHashCollision = function (hash, path) { this.matchers.forEach(function (matcher) { if (hash == matcher.hash) { throw new exceptions_1.BaseException("Configuration '" + path + "' conflicts with existing route '" + matcher.path + "'"); } }); }; /** * Given a URL, returns a list of `RouteMatch`es, which are partial recognitions for some route. */ ComponentRecognizer.prototype.recognize = function (urlParse) { var solutions = []; this.matchers.forEach(function (routeRecognizer) { var pathMatch = routeRecognizer.recognize(urlParse); if (lang_1.isPresent(pathMatch)) { solutions.push(pathMatch); } }); // handle cases where we are routing just to an aux route if (solutions.length == 0 && lang_1.isPresent(urlParse) && urlParse.auxiliary.length > 0) { return [async_1.PromiseWrapper.resolve(new route_recognizer_1.PathMatch(null, null, urlParse.auxiliary))]; } return solutions; }; ComponentRecognizer.prototype.recognizeAuxiliary = function (urlParse) { var routeRecognizer = this.auxRoutes.get(urlParse.path); if (lang_1.isPresent(routeRecognizer)) { return [routeRecognizer.recognize(urlParse)]; } return [async_1.PromiseWrapper.resolve(null)]; }; ComponentRecognizer.prototype.hasRoute = function (name) { return this.names.has(name); }; ComponentRecognizer.prototype.componentLoaded = function (name) { return this.hasRoute(name) && lang_1.isPresent(this.names.get(name).handler.componentType); }; ComponentRecognizer.prototype.loadComponent = function (name) { return this.names.get(name).handler.resolveComponentType(); }; ComponentRecognizer.prototype.generate = function (name, params) { var pathRecognizer = this.names.get(name); if (lang_1.isBlank(pathRecognizer)) { return null; } return pathRecognizer.generate(params); }; ComponentRecognizer.prototype.generateAuxiliary = function (name, params) { var pathRecognizer = this.auxNames.get(name); if (lang_1.isBlank(pathRecognizer)) { return null; } return pathRecognizer.generate(params); }; return ComponentRecognizer; })(); exports.ComponentRecognizer = ComponentRecognizer; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component_recognizer.js","sourceRoot":"","sources":["angular2/src/router/component_recognizer.ts"],"names":["ComponentRecognizer","ComponentRecognizer.constructor","ComponentRecognizer.config","ComponentRecognizer._assertNoHashCollision","ComponentRecognizer.recognize","ComponentRecognizer.recognizeAuxiliary","ComponentRecognizer.hasRoute","ComponentRecognizer.componentLoaded","ComponentRecognizer.loadComponent","ComponentRecognizer.generate","ComponentRecognizer.generateAuxiliary"],"mappings":"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,CAACA,IAAIA,MAAGA,CAACA,CAACA;YACjFA,CAACA;QACHA,CAACA,CAACA,CAACA;IACLA,CAACA;IAGDH;;OAEGA;IACHA,uCAASA,GAATA,UAAUA,QAAaA;QACrBI,IAAIA,SAASA,GAAGA,EAAEA,CAACA;QAEnBA,IAAIA,CAACA,QAAQA,CAACA,OAAOA,CAACA,UAACA,eAAmCA;YACxDA,IAAIA,SAASA,GAAGA,eAAeA,CAACA,SAASA,CAACA,QAAQA,CAACA,CAACA;YAEpDA,EAAEA,CAACA,CAACA,gBAASA,CAACA,SAASA,CAACA,CAACA,CAACA,CAACA;gBACzBA,SAASA,CAACA,IAAIA,CAACA,SAASA,CAACA,CAACA;YAC5BA,CAACA;QACHA,CAACA,CAACA,CAACA;QAEHA,yDAAyDA;QACzDA,EAAEA,CAACA,CAACA,SAASA,CAACA,MAAMA,IAAIA,CAACA,IAAIA,gBAASA,CAACA,QAAQA,CAACA,IAAIA,QAAQA,CAACA,SAASA,CAACA,MAAMA,GAAGA,CAACA,CAACA,CAACA,CAACA;YAClFA,MAAMA,CAACA,CAACA,sBAAcA,CAACA,OAAOA,CAACA,IAAIA,4BAASA,CAACA,IAAIA,EAAEA,IAAIA,EAAEA,QAAQA,CAACA,SAASA,CAACA,CAACA,CAACA,CAACA;QACjFA,CAACA;QAEDA,MAAMA,CAACA,SAASA,CAACA;IACnBA,CAACA;IAEDJ,gDAAkBA,GAAlBA,UAAmBA,QAAaA;QAC9BK,IAAIA,eAAeA,GAAoBA,IAAIA,CAACA,SAASA,CAACA,GAAGA,CAACA,QAAQA,CAACA,IAAIA,CAACA,CAACA;QACzEA,EAAEA,CAACA,CAACA,gBAASA,CAACA,eAAeA,CAACA,CAACA,CAACA,CAACA;YAC/BA,MAAMA,CAACA,CAACA,eAAeA,CAACA,SAASA,CAACA,QAAQA,CAACA,CAACA,CAACA;QAC/CA,CAACA;QAEDA,MAAMA,CAACA,CAACA,sBAAcA,CAACA,OAAOA,CAACA,IAAIA,CAACA,CAACA,CAACA;IACxCA,CAACA;IAEDL,sCAAQA,GAARA,UAASA,IAAYA,IAAaM,MAAMA,CAACA,IAAIA,CAACA,KAAKA,CAACA,GAAGA,CAACA,IAAIA,CAACA,CAACA,CAACA,CAACA;IAEhEN,6CAAeA,GAAfA,UAAgBA,IAAYA;QAC1BO,MAAMA,CAACA,IAAIA,CAACA,QAAQA,CAACA,IAAIA,CAACA,IAAIA,gBAASA,CAACA,IAAIA,CAACA,KAAKA,CAACA,GAAGA,CAACA,IAAIA,CAACA,CAACA,OAAOA,CAACA,aAAaA,CAACA,CAACA;IACtFA,CAACA;IAEDP,2CAAaA,GAAbA,UAAcA,IAAYA;QACxBQ,MAAMA,CAACA,IAAIA,CAACA,KAAKA,CAACA,GAAGA,CAACA,IAAIA,CAACA,CAACA,OAAOA,CAACA,oBAAoBA,EAAEA,CAACA;IAC7DA,CAACA;IAEDR,sCAAQA,GAARA,UAASA,IAAYA,EAAEA,MAAWA;QAChCS,IAAIA,cAAcA,GAAoBA,IAAIA,CAACA,KAAKA,CAACA,GAAGA,CAACA,IAAIA,CAACA,CAACA;QAC3DA,EAAEA,CAACA,CAACA,cAAOA,CAACA,cAAcA,CAACA,CAACA,CAACA,CAACA;YAC5BA,MAAMA,CAACA,IAAIA,CAACA;QACdA,CAACA;QACDA,MAAMA,CAACA,cAAcA,CAACA,QAAQA,CAACA,MAAMA,CAACA,CAACA;IACzCA,CAACA;IAEDT,+CAAiBA,GAAjBA,UAAkBA,IAAYA,EAAEA,MAAWA;QACzCU,IAAIA,cAAcA,GAAoBA,IAAIA,CAACA,QAAQA,CAACA,GAAGA,CAACA,IAAIA,CAACA,CAACA;QAC9DA,EAAEA,CAACA,CAACA,cAAOA,CAACA,cAAcA,CAACA,CAACA,CAACA,CAACA;YAC5BA,MAAMA,CAACA,IAAIA,CAACA;QACdA,CAACA;QACDA,MAAMA,CAACA,cAAcA,CAACA,QAAQA,CAACA,MAAMA,CAACA,CAACA;IACzCA,CAACA;IACHV,0BAACA;AAADA,CAACA,AA1ID,IA0IC;AA1IY,2BAAmB,sBA0I/B,CAAA","sourcesContent":["import {isBlank, isPresent} from 'angular2/src/facade/lang';\nimport {BaseException, WrappedException} from 'angular2/src/facade/exceptions';\nimport {Map, MapWrapper, ListWrapper, StringMapWrapper} from 'angular2/src/facade/collection';\nimport {PromiseWrapper} from 'angular2/src/facade/async';\n\nimport {\n  AbstractRecognizer,\n  RouteRecognizer,\n  RedirectRecognizer,\n  RouteMatch,\n  PathMatch\n} from './route_recognizer';\nimport {Route, AsyncRoute, AuxRoute, Redirect, RouteDefinition} from './route_config_impl';\nimport {AsyncRouteHandler} from './async_route_handler';\nimport {SyncRouteHandler} from './sync_route_handler';\nimport {Url} from './url_parser';\nimport {ComponentInstruction} from './instruction';\n\n\n/**\n * `ComponentRecognizer` is responsible for recognizing routes for a single component.\n * It is consumed by `RouteRegistry`, which knows how to recognize an entire hierarchy of\n * components.\n */\nexport class ComponentRecognizer {\n  names = new Map<string, RouteRecognizer>();\n\n  // map from name to recognizer\n  auxNames = new Map<string, RouteRecognizer>();\n\n  // map from starting path to recognizer\n  auxRoutes = new Map<string, RouteRecognizer>();\n\n  // TODO: optimize this into a trie\n  matchers: AbstractRecognizer[] = [];\n\n  defaultRoute: RouteRecognizer = null;\n\n  /**\n   * returns whether or not the config is terminal\n   */\n  config(config: RouteDefinition): boolean {\n    var handler;\n\n    if (isPresent(config.name) && config.name[0].toUpperCase() != config.name[0]) {\n      var suggestedName = config.name[0].toUpperCase() + config.name.substring(1);\n      throw new BaseException(\n          `Route \"${config.path}\" with name \"${config.name}\" does not begin with an uppercase letter. Route names should be CamelCase like \"${suggestedName}\".`);\n    }\n\n    if (config instanceof AuxRoute) {\n      handler = new SyncRouteHandler(config.component, config.data);\n      let path = config.path.startsWith('/') ? config.path.substring(1) : config.path;\n      var recognizer = new RouteRecognizer(config.path, handler);\n      this.auxRoutes.set(path, recognizer);\n      if (isPresent(config.name)) {\n        this.auxNames.set(config.name, recognizer);\n      }\n      return recognizer.terminal;\n    }\n\n    var useAsDefault = false;\n\n    if (config instanceof Redirect) {\n      let redirector = new RedirectRecognizer(config.path, config.redirectTo);\n      this._assertNoHashCollision(redirector.hash, config.path);\n      this.matchers.push(redirector);\n      return true;\n    }\n\n    if (config instanceof Route) {\n      handler = new SyncRouteHandler(config.component, config.data);\n      useAsDefault = isPresent(config.useAsDefault) && config.useAsDefault;\n    } else if (config instanceof AsyncRoute) {\n      handler = new AsyncRouteHandler(config.loader, config.data);\n      useAsDefault = isPresent(config.useAsDefault) && config.useAsDefault;\n    }\n    var recognizer = new RouteRecognizer(config.path, handler);\n\n    this._assertNoHashCollision(recognizer.hash, config.path);\n\n    if (useAsDefault) {\n      if (isPresent(this.defaultRoute)) {\n        throw new BaseException(`Only one route can be default`);\n      }\n      this.defaultRoute = recognizer;\n    }\n\n    this.matchers.push(recognizer);\n    if (isPresent(config.name)) {\n      this.names.set(config.name, recognizer);\n    }\n    return recognizer.terminal;\n  }\n\n\n  private _assertNoHashCollision(hash: string, path) {\n    this.matchers.forEach((matcher) => {\n      if (hash == matcher.hash) {\n        throw new BaseException(\n            `Configuration '${path}' conflicts with existing route '${matcher.path}'`);\n      }\n    });\n  }\n\n\n  /**\n   * Given a URL, returns a list of `RouteMatch`es, which are partial recognitions for some route.\n   */\n  recognize(urlParse: Url): Promise<RouteMatch>[] {\n    var solutions = [];\n\n    this.matchers.forEach((routeRecognizer: AbstractRecognizer) => {\n      var pathMatch = routeRecognizer.recognize(urlParse);\n\n      if (isPresent(pathMatch)) {\n        solutions.push(pathMatch);\n      }\n    });\n\n    // handle cases where we are routing just to an aux route\n    if (solutions.length == 0 && isPresent(urlParse) && urlParse.auxiliary.length > 0) {\n      return [PromiseWrapper.resolve(new PathMatch(null, null, urlParse.auxiliary))];\n    }\n\n    return solutions;\n  }\n\n  recognizeAuxiliary(urlParse: Url): Promise<RouteMatch>[] {\n    var routeRecognizer: RouteRecognizer = this.auxRoutes.get(urlParse.path);\n    if (isPresent(routeRecognizer)) {\n      return [routeRecognizer.recognize(urlParse)];\n    }\n\n    return [PromiseWrapper.resolve(null)];\n  }\n\n  hasRoute(name: string): boolean { return this.names.has(name); }\n\n  componentLoaded(name: string): boolean {\n    return this.hasRoute(name) && isPresent(this.names.get(name).handler.componentType);\n  }\n\n  loadComponent(name: string): Promise<any> {\n    return this.names.get(name).handler.resolveComponentType();\n  }\n\n  generate(name: string, params: any): ComponentInstruction {\n    var pathRecognizer: RouteRecognizer = this.names.get(name);\n    if (isBlank(pathRecognizer)) {\n      return null;\n    }\n    return pathRecognizer.generate(params);\n  }\n\n  generateAuxiliary(name: string, params: any): ComponentInstruction {\n    var pathRecognizer: RouteRecognizer = this.auxNames.get(name);\n    if (isBlank(pathRecognizer)) {\n      return null;\n    }\n    return pathRecognizer.generate(params);\n  }\n}\n"]}