UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

350 lines 54.4 kB
import { Map, ListWrapper } from 'angular2/src/facade/collection'; import { isPresent, isBlank, RegExpWrapper, RegExpMatcherWrapper, StringWrapper } from 'angular2/src/facade/lang'; import { BaseException } from 'angular2/src/facade/exceptions'; const _EMPTY_ATTR_VALUE = ''; // TODO: Can't use `const` here as // in Dart this is not transpiled into `final` yet... var _SELECTOR_REGEXP = RegExpWrapper.create('(\\:not\\()|' + '([-\\w]+)|' + '(?:\\.([-\\w]+))|' + '(?:\\[([-\\w*]+)(?:=([^\\]]*))?\\])|' + '(\\))|' + '(\\s*,\\s*)'); // "," /** * A css selector contains an element name, * css classes and attribute/value pairs with the purpose * of selecting subsets out of them. */ export class CssSelector { constructor() { this.element = null; this.classNames = []; this.attrs = []; this.notSelectors = []; } static parse(selector) { var results = []; var _addResult = (res, cssSel) => { if (cssSel.notSelectors.length > 0 && isBlank(cssSel.element) && ListWrapper.isEmpty(cssSel.classNames) && ListWrapper.isEmpty(cssSel.attrs)) { cssSel.element = "*"; } res.push(cssSel); }; var cssSelector = new CssSelector(); var matcher = RegExpWrapper.matcher(_SELECTOR_REGEXP, selector); var match; var current = cssSelector; var inNot = false; while (isPresent(match = RegExpMatcherWrapper.next(matcher))) { if (isPresent(match[1])) { if (inNot) { throw new BaseException('Nesting :not is not allowed in a selector'); } inNot = true; current = new CssSelector(); cssSelector.notSelectors.push(current); } if (isPresent(match[2])) { current.setElement(match[2]); } if (isPresent(match[3])) { current.addClassName(match[3]); } if (isPresent(match[4])) { current.addAttribute(match[4], match[5]); } if (isPresent(match[6])) { inNot = false; current = cssSelector; } if (isPresent(match[7])) { if (inNot) { throw new BaseException('Multiple selectors in :not are not supported'); } _addResult(results, cssSelector); cssSelector = current = new CssSelector(); } } _addResult(results, cssSelector); return results; } isElementSelector() { return isPresent(this.element) && ListWrapper.isEmpty(this.classNames) && ListWrapper.isEmpty(this.attrs) && this.notSelectors.length === 0; } setElement(element = null) { if (isPresent(element)) { element = element.toLowerCase(); } this.element = element; } /** Gets a template string for an element that matches the selector. */ getMatchingElementTemplate() { let tagName = isPresent(this.element) ? this.element : 'div'; let classAttr = this.classNames.length > 0 ? ` class="${this.classNames.join(' ')}"` : ''; let attrs = ''; for (let i = 0; i < this.attrs.length; i += 2) { let attrName = this.attrs[i]; let attrValue = this.attrs[i + 1] !== '' ? `="${this.attrs[i + 1]}"` : ''; attrs += ` ${attrName}${attrValue}`; } return `<${tagName}${classAttr}${attrs}></${tagName}>`; } addAttribute(name, value = _EMPTY_ATTR_VALUE) { this.attrs.push(name.toLowerCase()); if (isPresent(value)) { value = value.toLowerCase(); } else { value = _EMPTY_ATTR_VALUE; } this.attrs.push(value); } addClassName(name) { this.classNames.push(name.toLowerCase()); } toString() { var res = ''; if (isPresent(this.element)) { res += this.element; } if (isPresent(this.classNames)) { for (var i = 0; i < this.classNames.length; i++) { res += '.' + this.classNames[i]; } } if (isPresent(this.attrs)) { for (var i = 0; i < this.attrs.length;) { var attrName = this.attrs[i++]; var attrValue = this.attrs[i++]; res += '[' + attrName; if (attrValue.length > 0) { res += '=' + attrValue; } res += ']'; } } this.notSelectors.forEach(notSelector => res += `:not(${notSelector})`); return res; } } /** * Reads a list of CssSelectors and allows to calculate which ones * are contained in a given CssSelector. */ export class SelectorMatcher { constructor() { this._elementMap = new Map(); this._elementPartialMap = new Map(); this._classMap = new Map(); this._classPartialMap = new Map(); this._attrValueMap = new Map(); this._attrValuePartialMap = new Map(); this._listContexts = []; } static createNotMatcher(notSelectors) { var notMatcher = new SelectorMatcher(); notMatcher.addSelectables(notSelectors, null); return notMatcher; } addSelectables(cssSelectors, callbackCtxt) { var listContext = null; if (cssSelectors.length > 1) { listContext = new SelectorListContext(cssSelectors); this._listContexts.push(listContext); } for (var i = 0; i < cssSelectors.length; i++) { this._addSelectable(cssSelectors[i], callbackCtxt, listContext); } } /** * Add an object that can be found later on by calling `match`. * @param cssSelector A css selector * @param callbackCtxt An opaque object that will be given to the callback of the `match` function */ _addSelectable(cssSelector, callbackCtxt, listContext) { var matcher = this; var element = cssSelector.element; var classNames = cssSelector.classNames; var attrs = cssSelector.attrs; var selectable = new SelectorContext(cssSelector, callbackCtxt, listContext); if (isPresent(element)) { var isTerminal = attrs.length === 0 && classNames.length === 0; if (isTerminal) { this._addTerminal(matcher._elementMap, element, selectable); } else { matcher = this._addPartial(matcher._elementPartialMap, element); } } if (isPresent(classNames)) { for (var index = 0; index < classNames.length; index++) { var isTerminal = attrs.length === 0 && index === classNames.length - 1; var className = classNames[index]; if (isTerminal) { this._addTerminal(matcher._classMap, className, selectable); } else { matcher = this._addPartial(matcher._classPartialMap, className); } } } if (isPresent(attrs)) { for (var index = 0; index < attrs.length;) { var isTerminal = index === attrs.length - 2; var attrName = attrs[index++]; var attrValue = attrs[index++]; if (isTerminal) { var terminalMap = matcher._attrValueMap; var terminalValuesMap = terminalMap.get(attrName); if (isBlank(terminalValuesMap)) { terminalValuesMap = new Map(); terminalMap.set(attrName, terminalValuesMap); } this._addTerminal(terminalValuesMap, attrValue, selectable); } else { var parttialMap = matcher._attrValuePartialMap; var partialValuesMap = parttialMap.get(attrName); if (isBlank(partialValuesMap)) { partialValuesMap = new Map(); parttialMap.set(attrName, partialValuesMap); } matcher = this._addPartial(partialValuesMap, attrValue); } } } } _addTerminal(map, name, selectable) { var terminalList = map.get(name); if (isBlank(terminalList)) { terminalList = []; map.set(name, terminalList); } terminalList.push(selectable); } _addPartial(map, name) { var matcher = map.get(name); if (isBlank(matcher)) { matcher = new SelectorMatcher(); map.set(name, matcher); } return matcher; } /** * Find the objects that have been added via `addSelectable` * whose css selector is contained in the given css selector. * @param cssSelector A css selector * @param matchedCallback This callback will be called with the object handed into `addSelectable` * @return boolean true if a match was found */ match(cssSelector, matchedCallback) { var result = false; var element = cssSelector.element; var classNames = cssSelector.classNames; var attrs = cssSelector.attrs; for (var i = 0; i < this._listContexts.length; i++) { this._listContexts[i].alreadyMatched = false; } result = this._matchTerminal(this._elementMap, element, cssSelector, matchedCallback) || result; result = this._matchPartial(this._elementPartialMap, element, cssSelector, matchedCallback) || result; if (isPresent(classNames)) { for (var index = 0; index < classNames.length; index++) { var className = classNames[index]; result = this._matchTerminal(this._classMap, className, cssSelector, matchedCallback) || result; result = this._matchPartial(this._classPartialMap, className, cssSelector, matchedCallback) || result; } } if (isPresent(attrs)) { for (var index = 0; index < attrs.length;) { var attrName = attrs[index++]; var attrValue = attrs[index++]; var terminalValuesMap = this._attrValueMap.get(attrName); if (!StringWrapper.equals(attrValue, _EMPTY_ATTR_VALUE)) { result = this._matchTerminal(terminalValuesMap, _EMPTY_ATTR_VALUE, cssSelector, matchedCallback) || result; } result = this._matchTerminal(terminalValuesMap, attrValue, cssSelector, matchedCallback) || result; var partialValuesMap = this._attrValuePartialMap.get(attrName); if (!StringWrapper.equals(attrValue, _EMPTY_ATTR_VALUE)) { result = this._matchPartial(partialValuesMap, _EMPTY_ATTR_VALUE, cssSelector, matchedCallback) || result; } result = this._matchPartial(partialValuesMap, attrValue, cssSelector, matchedCallback) || result; } } return result; } /** @internal */ _matchTerminal(map, name, cssSelector, matchedCallback) { if (isBlank(map) || isBlank(name)) { return false; } var selectables = map.get(name); var starSelectables = map.get("*"); if (isPresent(starSelectables)) { selectables = selectables.concat(starSelectables); } if (isBlank(selectables)) { return false; } var selectable; var result = false; for (var index = 0; index < selectables.length; index++) { selectable = selectables[index]; result = selectable.finalize(cssSelector, matchedCallback) || result; } return result; } /** @internal */ _matchPartial(map, name, cssSelector, matchedCallback /*: (c: CssSelector, a: any) => void*/) { if (isBlank(map) || isBlank(name)) { return false; } var nestedSelector = map.get(name); if (isBlank(nestedSelector)) { return false; } // TODO(perf): get rid of recursion and measure again // TODO(perf): don't pass the whole selector into the recursion, // but only the not processed parts return nestedSelector.match(cssSelector, matchedCallback); } } export class SelectorListContext { constructor(selectors) { this.selectors = selectors; this.alreadyMatched = false; } } // Store context to pass back selector and context when a selector is matched export class SelectorContext { constructor(selector, cbContext, listContext) { this.selector = selector; this.cbContext = cbContext; this.listContext = listContext; this.notSelectors = selector.notSelectors; } finalize(cssSelector, callback) { var result = true; if (this.notSelectors.length > 0 && (isBlank(this.listContext) || !this.listContext.alreadyMatched)) { var notMatcher = SelectorMatcher.createNotMatcher(this.notSelectors); result = !notMatcher.match(cssSelector, null); } if (result && isPresent(callback) && (isBlank(this.listContext) || !this.listContext.alreadyMatched)) { if (isPresent(this.listContext)) { this.listContext.alreadyMatched = true; } callback(this.selector, this.cbContext); } return result; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selector.js","sourceRoot":"","sources":["angular2/src/compiler/selector.ts"],"names":["CssSelector","CssSelector.constructor","CssSelector.parse","CssSelector.isElementSelector","CssSelector.setElement","CssSelector.getMatchingElementTemplate","CssSelector.addAttribute","CssSelector.addClassName","CssSelector.toString","SelectorMatcher","SelectorMatcher.constructor","SelectorMatcher.createNotMatcher","SelectorMatcher.addSelectables","SelectorMatcher._addSelectable","SelectorMatcher._addTerminal","SelectorMatcher._addPartial","SelectorMatcher.match","SelectorMatcher._matchTerminal","SelectorMatcher._matchPartial","SelectorListContext","SelectorListContext.constructor","SelectorContext","SelectorContext.constructor","SelectorContext.finalize"],"mappings":"OAAO,EAAC,GAAG,EAAE,WAAW,EAAa,MAAM,gCAAgC;OACpE,EACL,SAAS,EACT,OAAO,EACP,aAAa,EACb,oBAAoB,EACpB,aAAa,EACd,MAAM,0BAA0B;OAC1B,EAAC,aAAa,EAAmB,MAAM,gCAAgC;AAE9E,MAAM,iBAAiB,GAAG,EAAE,CAAC;AAE7B,kCAAkC;AAClC,qDAAqD;AACrD,IAAI,gBAAgB,GAAG,aAAa,CAAC,MAAM,CACvC,cAAc;IACd,YAAY;IACZ,mBAAmB;IACnB,sCAAsC;IACtC,QAAQ;IACR,aAAa,CAAC,CAAC,CAA2B,MAAM;AAEpD;;;;GAIG;AACH;IAAAA;QACEC,YAAOA,GAAWA,IAAIA,CAACA;QACvBA,eAAUA,GAAaA,EAAEA,CAACA;QAC1BA,UAAKA,GAAaA,EAAEA,CAACA;QACrBA,iBAAYA,GAAkBA,EAAEA,CAACA;IAiHnCA,CAACA;IA/GCD,OAAOA,KAAKA,CAACA,QAAgBA;QAC3BE,IAAIA,OAAOA,GAAkBA,EAAEA,CAACA;QAChCA,IAAIA,UAAUA,GAAGA,CAACA,GAAkBA,EAAEA,MAAMA;YAC1CA,EAAEA,CAACA,CAACA,MAAMA,CAACA,YAAYA,CAACA,MAAMA,GAAGA,CAACA,IAAIA,OAAOA,CAACA,MAAMA,CAACA,OAAOA,CAACA;gBACzDA,WAAWA,CAACA,OAAOA,CAACA,MAAMA,CAACA,UAAUA,CAACA,IAAIA,WAAWA,CAACA,OAAOA,CAACA,MAAMA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA;gBAChFA,MAAMA,CAACA,OAAOA,GAAGA,GAAGA,CAACA;YACvBA,CAACA;YACDA,GAAGA,CAACA,IAAIA,CAACA,MAAMA,CAACA,CAACA;QACnBA,CAACA,CAACA;QACFA,IAAIA,WAAWA,GAAGA,IAAIA,WAAWA,EAAEA,CAACA;QACpCA,IAAIA,OAAOA,GAAGA,aAAaA,CAACA,OAAOA,CAACA,gBAAgBA,EAAEA,QAAQA,CAACA,CAACA;QAChEA,IAAIA,KAAKA,CAACA;QACVA,IAAIA,OAAOA,GAAGA,WAAWA,CAACA;QAC1BA,IAAIA,KAAKA,GAAGA,KAAKA,CAACA;QAClBA,OAAOA,SAASA,CAACA,KAAKA,GAAGA,oBAAoBA,CAACA,IAAIA,CAACA,OAAOA,CAACA,CAACA,EAAEA,CAACA;YAC7DA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBACxBA,EAAEA,CAACA,CAACA,KAAKA,CAACA,CAACA,CAACA;oBACVA,MAAMA,IAAIA,aAAaA,CAACA,2CAA2CA,CAACA,CAACA;gBACvEA,CAACA;gBACDA,KAAKA,GAAGA,IAAIA,CAACA;gBACbA,OAAOA,GAAGA,IAAIA,WAAWA,EAAEA,CAACA;gBAC5BA,WAAWA,CAACA,YAAYA,CAACA,IAAIA,CAACA,OAAOA,CAACA,CAACA;YACzCA,CAACA;YACDA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBACxBA,OAAOA,CAACA,UAAUA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA;YAC/BA,CAACA;YACDA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBACxBA,OAAOA,CAACA,YAAYA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA;YACjCA,CAACA;YACDA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBACxBA,OAAOA,CAACA,YAAYA,CAACA,KAAKA,CAACA,CAACA,CAACA,EAAEA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA;YAC3CA,CAACA;YACDA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA,CAACA;gBACxBA,KAAKA,GAAGA,KAAKA,CAACA;gBACdA,OAAOA,GAAGA,WAAWA,CAACA;YACxBA,CAACA;YACDA,EAAEA,CAACA,CAACA,SAASA,CAACA,KAAKA,CAACA,CAACA,CAACA,CAAC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","sourcesContent":["import {Map, ListWrapper, MapWrapper} from 'angular2/src/facade/collection';\nimport {\n  isPresent,\n  isBlank,\n  RegExpWrapper,\n  RegExpMatcherWrapper,\n  StringWrapper\n} from 'angular2/src/facade/lang';\nimport {BaseException, WrappedException} from 'angular2/src/facade/exceptions';\n\nconst _EMPTY_ATTR_VALUE = '';\n\n// TODO: Can't use `const` here as\n// in Dart this is not transpiled into `final` yet...\nvar _SELECTOR_REGEXP = RegExpWrapper.create(\n    '(\\\\:not\\\\()|' +                          //\":not(\"\n    '([-\\\\w]+)|' +                            // \"tag\"\n    '(?:\\\\.([-\\\\w]+))|' +                     // \".class\"\n    '(?:\\\\[([-\\\\w*]+)(?:=([^\\\\]]*))?\\\\])|' +  // \"[name]\", \"[name=value]\" or \"[name*=value]\"\n    '(\\\\))|' +                                // \")\"\n    '(\\\\s*,\\\\s*)');                           // \",\"\n\n/**\n * A css selector contains an element name,\n * css classes and attribute/value pairs with the purpose\n * of selecting subsets out of them.\n */\nexport class CssSelector {\n  element: string = null;\n  classNames: string[] = [];\n  attrs: string[] = [];\n  notSelectors: CssSelector[] = [];\n\n  static parse(selector: string): CssSelector[] {\n    var results: CssSelector[] = [];\n    var _addResult = (res: CssSelector[], cssSel) => {\n      if (cssSel.notSelectors.length > 0 && isBlank(cssSel.element) &&\n          ListWrapper.isEmpty(cssSel.classNames) && ListWrapper.isEmpty(cssSel.attrs)) {\n        cssSel.element = \"*\";\n      }\n      res.push(cssSel);\n    };\n    var cssSelector = new CssSelector();\n    var matcher = RegExpWrapper.matcher(_SELECTOR_REGEXP, selector);\n    var match;\n    var current = cssSelector;\n    var inNot = false;\n    while (isPresent(match = RegExpMatcherWrapper.next(matcher))) {\n      if (isPresent(match[1])) {\n        if (inNot) {\n          throw new BaseException('Nesting :not is not allowed in a selector');\n        }\n        inNot = true;\n        current = new CssSelector();\n        cssSelector.notSelectors.push(current);\n      }\n      if (isPresent(match[2])) {\n        current.setElement(match[2]);\n      }\n      if (isPresent(match[3])) {\n        current.addClassName(match[3]);\n      }\n      if (isPresent(match[4])) {\n        current.addAttribute(match[4], match[5]);\n      }\n      if (isPresent(match[6])) {\n        inNot = false;\n        current = cssSelector;\n      }\n      if (isPresent(match[7])) {\n        if (inNot) {\n          throw new BaseException('Multiple selectors in :not are not supported');\n        }\n        _addResult(results, cssSelector);\n        cssSelector = current = new CssSelector();\n      }\n    }\n    _addResult(results, cssSelector);\n    return results;\n  }\n\n  isElementSelector(): boolean {\n    return isPresent(this.element) && ListWrapper.isEmpty(this.classNames) &&\n           ListWrapper.isEmpty(this.attrs) && this.notSelectors.length === 0;\n  }\n\n  setElement(element: string = null) {\n    if (isPresent(element)) {\n      element = element.toLowerCase();\n    }\n    this.element = element;\n  }\n\n  /** Gets a template string for an element that matches the selector. */\n  getMatchingElementTemplate(): string {\n    let tagName = isPresent(this.element) ? this.element : 'div';\n    let classAttr = this.classNames.length > 0 ? ` class=\"${this.classNames.join(' ')}\"` : '';\n\n    let attrs = '';\n    for (let i = 0; i < this.attrs.length; i += 2) {\n      let attrName = this.attrs[i];\n      let attrValue = this.attrs[i + 1] !== '' ? `=\"${this.attrs[i + 1]}\"` : '';\n      attrs += ` ${attrName}${attrValue}`;\n    }\n\n    return `<${tagName}${classAttr}${attrs}></${tagName}>`;\n  }\n\n  addAttribute(name: string, value: string = _EMPTY_ATTR_VALUE) {\n    this.attrs.push(name.toLowerCase());\n    if (isPresent(value)) {\n      value = value.toLowerCase();\n    } else {\n      value = _EMPTY_ATTR_VALUE;\n    }\n    this.attrs.push(value);\n  }\n\n  addClassName(name: string) { this.classNames.push(name.toLowerCase()); }\n\n  toString(): string {\n    var res = '';\n    if (isPresent(this.element)) {\n      res += this.element;\n    }\n    if (isPresent(this.classNames)) {\n      for (var i = 0; i < this.classNames.length; i++) {\n        res += '.' + this.classNames[i];\n      }\n    }\n    if (isPresent(this.attrs)) {\n      for (var i = 0; i < this.attrs.length;) {\n        var attrName = this.attrs[i++];\n        var attrValue = this.attrs[i++];\n        res += '[' + attrName;\n        if (attrValue.length > 0) {\n          res += '=' + attrValue;\n        }\n        res += ']';\n      }\n    }\n    this.notSelectors.forEach(notSelector => res += `:not(${notSelector})`);\n    return res;\n  }\n}\n\n/**\n * Reads a list of CssSelectors and allows to calculate which ones\n * are contained in a given CssSelector.\n */\nexport class SelectorMatcher {\n  static createNotMatcher(notSelectors: CssSelector[]): SelectorMatcher {\n    var notMatcher = new SelectorMatcher();\n    notMatcher.addSelectables(notSelectors, null);\n    return notMatcher;\n  }\n\n  private _elementMap = new Map<string, SelectorContext[]>();\n  private _elementPartialMap = new Map<string, SelectorMatcher>();\n  private _classMap = new Map<string, SelectorContext[]>();\n  private _classPartialMap = new Map<string, SelectorMatcher>();\n  private _attrValueMap = new Map<string, Map<string, SelectorContext[]>>();\n  private _attrValuePartialMap = new Map<string, Map<string, SelectorMatcher>>();\n  private _listContexts: SelectorListContext[] = [];\n\n  addSelectables(cssSelectors: CssSelector[], callbackCtxt?: any) {\n    var listContext = null;\n    if (cssSelectors.length > 1) {\n      listContext = new SelectorListContext(cssSelectors);\n      this._listContexts.push(listContext);\n    }\n    for (var i = 0; i < cssSelectors.length; i++) {\n      this._addSelectable(cssSelectors[i], callbackCtxt, listContext);\n    }\n  }\n\n  /**\n   * Add an object that can be found later on by calling `match`.\n   * @param cssSelector A css selector\n   * @param callbackCtxt An opaque object that will be given to the callback of the `match` function\n   */\n  private _addSelectable(cssSelector: CssSelector, callbackCtxt: any,\n                         listContext: SelectorListContext) {\n    var matcher = this;\n    var element = cssSelector.element;\n    var classNames = cssSelector.classNames;\n    var attrs = cssSelector.attrs;\n    var selectable = new SelectorContext(cssSelector, callbackCtxt, listContext);\n\n    if (isPresent(element)) {\n      var isTerminal = attrs.length === 0 && classNames.length === 0;\n      if (isTerminal) {\n        this._addTerminal(matcher._elementMap, element, selectable);\n      } else {\n        matcher = this._addPartial(matcher._elementPartialMap, element);\n      }\n    }\n\n    if (isPresent(classNames)) {\n      for (var index = 0; index < classNames.length; index++) {\n        var isTerminal = attrs.length === 0 && index === classNames.length - 1;\n        var className = classNames[index];\n        if (isTerminal) {\n          this._addTerminal(matcher._classMap, className, selectable);\n        } else {\n          matcher = this._addPartial(matcher._classPartialMap, className);\n        }\n      }\n    }\n\n    if (isPresent(attrs)) {\n      for (var index = 0; index < attrs.length;) {\n        var isTerminal = index === attrs.length - 2;\n        var attrName = attrs[index++];\n        var attrValue = attrs[index++];\n        if (isTerminal) {\n          var terminalMap = matcher._attrValueMap;\n          var terminalValuesMap = terminalMap.get(attrName);\n          if (isBlank(terminalValuesMap)) {\n            terminalValuesMap = new Map<string, SelectorContext[]>();\n            terminalMap.set(attrName, terminalValuesMap);\n          }\n          this._addTerminal(terminalValuesMap, attrValue, selectable);\n        } else {\n          var parttialMap = matcher._attrValuePartialMap;\n          var partialValuesMap = parttialMap.get(attrName);\n          if (isBlank(partialValuesMap)) {\n            partialValuesMap = new Map<string, SelectorMatcher>();\n            parttialMap.set(attrName, partialValuesMap);\n          }\n          matcher = this._addPartial(partialValuesMap, attrValue);\n        }\n      }\n    }\n  }\n\n  private _addTerminal(map: Map<string, SelectorContext[]>, name: string,\n                       selectable: SelectorContext) {\n    var terminalList = map.get(name);\n    if (isBlank(terminalList)) {\n      terminalList = [];\n      map.set(name, terminalList);\n    }\n    terminalList.push(selectable);\n  }\n\n  private _addPartial(map: Map<string, SelectorMatcher>, name: string): SelectorMatcher {\n    var matcher = map.get(name);\n    if (isBlank(matcher)) {\n      matcher = new SelectorMatcher();\n      map.set(name, matcher);\n    }\n    return matcher;\n  }\n\n  /**\n   * Find the objects that have been added via `addSelectable`\n   * whose css selector is contained in the given css selector.\n   * @param cssSelector A css selector\n   * @param matchedCallback This callback will be called with the object handed into `addSelectable`\n   * @return boolean true if a match was found\n  */\n  match(cssSelector: CssSelector, matchedCallback: (c: CssSelector, a: any) => void): boolean {\n    var result = false;\n    var element = cssSelector.element;\n    var classNames = cssSelector.classNames;\n    var attrs = cssSelector.attrs;\n\n    for (var i = 0; i < this._listContexts.length; i++) {\n      this._listContexts[i].alreadyMatched = false;\n    }\n\n    result = this._matchTerminal(this._elementMap, element, cssSelector, matchedCallback) || result;\n    result = this._matchPartial(this._elementPartialMap, element, cssSelector, matchedCallback) ||\n             result;\n\n    if (isPresent(classNames)) {\n      for (var index = 0; index < classNames.length; index++) {\n        var className = classNames[index];\n        result =\n            this._matchTerminal(this._classMap, className, cssSelector, matchedCallback) || result;\n        result =\n            this._matchPartial(this._classPartialMap, className, cssSelector, matchedCallback) ||\n            result;\n      }\n    }\n\n    if (isPresent(attrs)) {\n      for (var index = 0; index < attrs.length;) {\n        var attrName = attrs[index++];\n        var attrValue = attrs[index++];\n\n        var terminalValuesMap = this._attrValueMap.get(attrName);\n        if (!StringWrapper.equals(attrValue, _EMPTY_ATTR_VALUE)) {\n