1 | /**
|
2 | * @license
|
3 | * Copyright Google Inc. All Rights Reserved.
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style license that can be
|
6 | * found in the LICENSE file at https://angular.io/license
|
7 | */
|
8 | import { getHtmlTagDefinition } from './ml_parser/html_tags';
|
9 | var _SELECTOR_REGEXP = new RegExp('(\\:not\\()|' + //":not("
|
10 | '([-\\w]+)|' + // "tag"
|
11 | '(?:\\.([-\\w]+))|' + // ".class"
|
12 | // "-" should appear first in the regexp below as FF31 parses "[.-\w]" as a range
|
13 | '(?:\\[([-.\\w*]+)(?:=([\"\']?)([^\\]\"\']*)\\5)?\\])|' + // "[name]", "[name=value]",
|
14 | // "[name="value"]",
|
15 | // "[name='value']"
|
16 | '(\\))|' + // ")"
|
17 | '(\\s*,\\s*)', // ","
|
18 | 'g');
|
19 | /**
|
20 | * A css selector contains an element name,
|
21 | * css classes and attribute/value pairs with the purpose
|
22 | * of selecting subsets out of them.
|
23 | */
|
24 | var CssSelector = /** @class */ (function () {
|
25 | function CssSelector() {
|
26 | this.element = null;
|
27 | this.classNames = [];
|
28 | /**
|
29 | * The selectors are encoded in pairs where:
|
30 | * - even locations are attribute names
|
31 | * - odd locations are attribute values.
|
32 | *
|
33 | * Example:
|
34 | * Selector: `[key1=value1][key2]` would parse to:
|
35 | * ```
|
36 | * ['key1', 'value1', 'key2', '']
|
37 | * ```
|
38 | */
|
39 | this.attrs = [];
|
40 | this.notSelectors = [];
|
41 | }
|
42 | CssSelector.parse = function (selector) {
|
43 | var results = [];
|
44 | var _addResult = function (res, cssSel) {
|
45 | if (cssSel.notSelectors.length > 0 && !cssSel.element && cssSel.classNames.length == 0 &&
|
46 | cssSel.attrs.length == 0) {
|
47 | cssSel.element = '*';
|
48 | }
|
49 | res.push(cssSel);
|
50 | };
|
51 | var cssSelector = new CssSelector();
|
52 | var match;
|
53 | var current = cssSelector;
|
54 | var inNot = false;
|
55 | _SELECTOR_REGEXP.lastIndex = 0;
|
56 | while (match = _SELECTOR_REGEXP.exec(selector)) {
|
57 | if (match[1]) {
|
58 | if (inNot) {
|
59 | throw new Error('Nesting :not is not allowed in a selector');
|
60 | }
|
61 | inNot = true;
|
62 | current = new CssSelector();
|
63 | cssSelector.notSelectors.push(current);
|
64 | }
|
65 | if (match[2]) {
|
66 | current.setElement(match[2]);
|
67 | }
|
68 | if (match[3]) {
|
69 | current.addClassName(match[3]);
|
70 | }
|
71 | if (match[4]) {
|
72 | current.addAttribute(match[4], match[6]);
|
73 | }
|
74 | if (match[7]) {
|
75 | inNot = false;
|
76 | current = cssSelector;
|
77 | }
|
78 | if (match[8]) {
|
79 | if (inNot) {
|
80 | throw new Error('Multiple selectors in :not are not supported');
|
81 | }
|
82 | _addResult(results, cssSelector);
|
83 | cssSelector = current = new CssSelector();
|
84 | }
|
85 | }
|
86 | _addResult(results, cssSelector);
|
87 | return results;
|
88 | };
|
89 | CssSelector.prototype.isElementSelector = function () {
|
90 | return this.hasElementSelector() && this.classNames.length == 0 && this.attrs.length == 0 &&
|
91 | this.notSelectors.length === 0;
|
92 | };
|
93 | CssSelector.prototype.hasElementSelector = function () { return !!this.element; };
|
94 | CssSelector.prototype.setElement = function (element) {
|
95 | if (element === void 0) { element = null; }
|
96 | this.element = element;
|
97 | };
|
98 | /** Gets a template string for an element that matches the selector. */
|
99 | CssSelector.prototype.getMatchingElementTemplate = function () {
|
100 | var tagName = this.element || 'div';
|
101 | var classAttr = this.classNames.length > 0 ? " class=\"" + this.classNames.join(' ') + "\"" : '';
|
102 | var attrs = '';
|
103 | for (var i = 0; i < this.attrs.length; i += 2) {
|
104 | var attrName = this.attrs[i];
|
105 | var attrValue = this.attrs[i + 1] !== '' ? "=\"" + this.attrs[i + 1] + "\"" : '';
|
106 | attrs += " " + attrName + attrValue;
|
107 | }
|
108 | return getHtmlTagDefinition(tagName).isVoid ? "<" + tagName + classAttr + attrs + "/>" :
|
109 | "<" + tagName + classAttr + attrs + "></" + tagName + ">";
|
110 | };
|
111 | CssSelector.prototype.getAttrs = function () {
|
112 | var result = [];
|
113 | if (this.classNames.length > 0) {
|
114 | result.push('class', this.classNames.join(' '));
|
115 | }
|
116 | return result.concat(this.attrs);
|
117 | };
|
118 | CssSelector.prototype.addAttribute = function (name, value) {
|
119 | if (value === void 0) { value = ''; }
|
120 | this.attrs.push(name, value && value.toLowerCase() || '');
|
121 | };
|
122 | CssSelector.prototype.addClassName = function (name) { this.classNames.push(name.toLowerCase()); };
|
123 | CssSelector.prototype.toString = function () {
|
124 | var res = this.element || '';
|
125 | if (this.classNames) {
|
126 | this.classNames.forEach(function (klass) { return res += "." + klass; });
|
127 | }
|
128 | if (this.attrs) {
|
129 | for (var i = 0; i < this.attrs.length; i += 2) {
|
130 | var name_1 = this.attrs[i];
|
131 | var value = this.attrs[i + 1];
|
132 | res += "[" + name_1 + (value ? '=' + value : '') + "]";
|
133 | }
|
134 | }
|
135 | this.notSelectors.forEach(function (notSelector) { return res += ":not(" + notSelector + ")"; });
|
136 | return res;
|
137 | };
|
138 | return CssSelector;
|
139 | }());
|
140 | export { CssSelector };
|
141 | /**
|
142 | * Reads a list of CssSelectors and allows to calculate which ones
|
143 | * are contained in a given CssSelector.
|
144 | */
|
145 | var SelectorMatcher = /** @class */ (function () {
|
146 | function SelectorMatcher() {
|
147 | this._elementMap = new Map();
|
148 | this._elementPartialMap = new Map();
|
149 | this._classMap = new Map();
|
150 | this._classPartialMap = new Map();
|
151 | this._attrValueMap = new Map();
|
152 | this._attrValuePartialMap = new Map();
|
153 | this._listContexts = [];
|
154 | }
|
155 | SelectorMatcher.createNotMatcher = function (notSelectors) {
|
156 | var notMatcher = new SelectorMatcher();
|
157 | notMatcher.addSelectables(notSelectors, null);
|
158 | return notMatcher;
|
159 | };
|
160 | SelectorMatcher.prototype.addSelectables = function (cssSelectors, callbackCtxt) {
|
161 | var listContext = null;
|
162 | if (cssSelectors.length > 1) {
|
163 | listContext = new SelectorListContext(cssSelectors);
|
164 | this._listContexts.push(listContext);
|
165 | }
|
166 | for (var i = 0; i < cssSelectors.length; i++) {
|
167 | this._addSelectable(cssSelectors[i], callbackCtxt, listContext);
|
168 | }
|
169 | };
|
170 | /**
|
171 | * Add an object that can be found later on by calling `match`.
|
172 | * @param cssSelector A css selector
|
173 | * @param callbackCtxt An opaque object that will be given to the callback of the `match` function
|
174 | */
|
175 | SelectorMatcher.prototype._addSelectable = function (cssSelector, callbackCtxt, listContext) {
|
176 | var matcher = this;
|
177 | var element = cssSelector.element;
|
178 | var classNames = cssSelector.classNames;
|
179 | var attrs = cssSelector.attrs;
|
180 | var selectable = new SelectorContext(cssSelector, callbackCtxt, listContext);
|
181 | if (element) {
|
182 | var isTerminal = attrs.length === 0 && classNames.length === 0;
|
183 | if (isTerminal) {
|
184 | this._addTerminal(matcher._elementMap, element, selectable);
|
185 | }
|
186 | else {
|
187 | matcher = this._addPartial(matcher._elementPartialMap, element);
|
188 | }
|
189 | }
|
190 | if (classNames) {
|
191 | for (var i = 0; i < classNames.length; i++) {
|
192 | var isTerminal = attrs.length === 0 && i === classNames.length - 1;
|
193 | var className = classNames[i];
|
194 | if (isTerminal) {
|
195 | this._addTerminal(matcher._classMap, className, selectable);
|
196 | }
|
197 | else {
|
198 | matcher = this._addPartial(matcher._classPartialMap, className);
|
199 | }
|
200 | }
|
201 | }
|
202 | if (attrs) {
|
203 | for (var i = 0; i < attrs.length; i += 2) {
|
204 | var isTerminal = i === attrs.length - 2;
|
205 | var name_2 = attrs[i];
|
206 | var value = attrs[i + 1];
|
207 | if (isTerminal) {
|
208 | var terminalMap = matcher._attrValueMap;
|
209 | var terminalValuesMap = terminalMap.get(name_2);
|
210 | if (!terminalValuesMap) {
|
211 | terminalValuesMap = new Map();
|
212 | terminalMap.set(name_2, terminalValuesMap);
|
213 | }
|
214 | this._addTerminal(terminalValuesMap, value, selectable);
|
215 | }
|
216 | else {
|
217 | var partialMap = matcher._attrValuePartialMap;
|
218 | var partialValuesMap = partialMap.get(name_2);
|
219 | if (!partialValuesMap) {
|
220 | partialValuesMap = new Map();
|
221 | partialMap.set(name_2, partialValuesMap);
|
222 | }
|
223 | matcher = this._addPartial(partialValuesMap, value);
|
224 | }
|
225 | }
|
226 | }
|
227 | };
|
228 | SelectorMatcher.prototype._addTerminal = function (map, name, selectable) {
|
229 | var terminalList = map.get(name);
|
230 | if (!terminalList) {
|
231 | terminalList = [];
|
232 | map.set(name, terminalList);
|
233 | }
|
234 | terminalList.push(selectable);
|
235 | };
|
236 | SelectorMatcher.prototype._addPartial = function (map, name) {
|
237 | var matcher = map.get(name);
|
238 | if (!matcher) {
|
239 | matcher = new SelectorMatcher();
|
240 | map.set(name, matcher);
|
241 | }
|
242 | return matcher;
|
243 | };
|
244 | /**
|
245 | * Find the objects that have been added via `addSelectable`
|
246 | * whose css selector is contained in the given css selector.
|
247 | * @param cssSelector A css selector
|
248 | * @param matchedCallback This callback will be called with the object handed into `addSelectable`
|
249 | * @return boolean true if a match was found
|
250 | */
|
251 | SelectorMatcher.prototype.match = function (cssSelector, matchedCallback) {
|
252 | var result = false;
|
253 | var element = cssSelector.element;
|
254 | var classNames = cssSelector.classNames;
|
255 | var attrs = cssSelector.attrs;
|
256 | for (var i = 0; i < this._listContexts.length; i++) {
|
257 | this._listContexts[i].alreadyMatched = false;
|
258 | }
|
259 | result = this._matchTerminal(this._elementMap, element, cssSelector, matchedCallback) || result;
|
260 | result = this._matchPartial(this._elementPartialMap, element, cssSelector, matchedCallback) ||
|
261 | result;
|
262 | if (classNames) {
|
263 | for (var i = 0; i < classNames.length; i++) {
|
264 | var className = classNames[i];
|
265 | result =
|
266 | this._matchTerminal(this._classMap, className, cssSelector, matchedCallback) || result;
|
267 | result =
|
268 | this._matchPartial(this._classPartialMap, className, cssSelector, matchedCallback) ||
|
269 | result;
|
270 | }
|
271 | }
|
272 | if (attrs) {
|
273 | for (var i = 0; i < attrs.length; i += 2) {
|
274 | var name_3 = attrs[i];
|
275 | var value = attrs[i + 1];
|
276 | var terminalValuesMap = this._attrValueMap.get(name_3);
|
277 | if (value) {
|
278 | result =
|
279 | this._matchTerminal(terminalValuesMap, '', cssSelector, matchedCallback) || result;
|
280 | }
|
281 | result =
|
282 | this._matchTerminal(terminalValuesMap, value, cssSelector, matchedCallback) || result;
|
283 | var partialValuesMap = this._attrValuePartialMap.get(name_3);
|
284 | if (value) {
|
285 | result = this._matchPartial(partialValuesMap, '', cssSelector, matchedCallback) || result;
|
286 | }
|
287 | result =
|
288 | this._matchPartial(partialValuesMap, value, cssSelector, matchedCallback) || result;
|
289 | }
|
290 | }
|
291 | return result;
|
292 | };
|
293 | /** @internal */
|
294 | SelectorMatcher.prototype._matchTerminal = function (map, name, cssSelector, matchedCallback) {
|
295 | if (!map || typeof name !== 'string') {
|
296 | return false;
|
297 | }
|
298 | var selectables = map.get(name) || [];
|
299 | var starSelectables = map.get('*');
|
300 | if (starSelectables) {
|
301 | selectables = selectables.concat(starSelectables);
|
302 | }
|
303 | if (selectables.length === 0) {
|
304 | return false;
|
305 | }
|
306 | var selectable;
|
307 | var result = false;
|
308 | for (var i = 0; i < selectables.length; i++) {
|
309 | selectable = selectables[i];
|
310 | result = selectable.finalize(cssSelector, matchedCallback) || result;
|
311 | }
|
312 | return result;
|
313 | };
|
314 | /** @internal */
|
315 | SelectorMatcher.prototype._matchPartial = function (map, name, cssSelector, matchedCallback) {
|
316 | if (!map || typeof name !== 'string') {
|
317 | return false;
|
318 | }
|
319 | var nestedSelector = map.get(name);
|
320 | if (!nestedSelector) {
|
321 | return false;
|
322 | }
|
323 | // TODO(perf): get rid of recursion and measure again
|
324 | // TODO(perf): don't pass the whole selector into the recursion,
|
325 | // but only the not processed parts
|
326 | return nestedSelector.match(cssSelector, matchedCallback);
|
327 | };
|
328 | return SelectorMatcher;
|
329 | }());
|
330 | export { SelectorMatcher };
|
331 | var SelectorListContext = /** @class */ (function () {
|
332 | function SelectorListContext(selectors) {
|
333 | this.selectors = selectors;
|
334 | this.alreadyMatched = false;
|
335 | }
|
336 | return SelectorListContext;
|
337 | }());
|
338 | export { SelectorListContext };
|
339 | // Store context to pass back selector and context when a selector is matched
|
340 | var SelectorContext = /** @class */ (function () {
|
341 | function SelectorContext(selector, cbContext, listContext) {
|
342 | this.selector = selector;
|
343 | this.cbContext = cbContext;
|
344 | this.listContext = listContext;
|
345 | this.notSelectors = selector.notSelectors;
|
346 | }
|
347 | SelectorContext.prototype.finalize = function (cssSelector, callback) {
|
348 | var result = true;
|
349 | if (this.notSelectors.length > 0 && (!this.listContext || !this.listContext.alreadyMatched)) {
|
350 | var notMatcher = SelectorMatcher.createNotMatcher(this.notSelectors);
|
351 | result = !notMatcher.match(cssSelector, null);
|
352 | }
|
353 | if (result && callback && (!this.listContext || !this.listContext.alreadyMatched)) {
|
354 | if (this.listContext) {
|
355 | this.listContext.alreadyMatched = true;
|
356 | }
|
357 | callback(this.selector, this.cbContext);
|
358 | }
|
359 | return result;
|
360 | };
|
361 | return SelectorContext;
|
362 | }());
|
363 | export { SelectorContext };
|
364 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selector.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler/src/selector.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {getHtmlTagDefinition} from './ml_parser/html_tags';\n\nconst _SELECTOR_REGEXP = new RegExp(\n    '(\\\\:not\\\\()|' +           //\":not(\"\n        '([-\\\\w]+)|' +         // \"tag\"\n        '(?:\\\\.([-\\\\w]+))|' +  // \".class\"\n        // \"-\" should appear first in the regexp below as FF31 parses \"[.-\\w]\" as a range\n        '(?:\\\\[([-.\\\\w*]+)(?:=([\\\"\\']?)([^\\\\]\\\"\\']*)\\\\5)?\\\\])|' +  // \"[name]\", \"[name=value]\",\n                                                                   // \"[name=\"value\"]\",\n                                                                   // \"[name='value']\"\n        '(\\\\))|' +                                                 // \")\"\n        '(\\\\s*,\\\\s*)',                                             // \",\"\n    'g');\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 = null;\n  classNames: string[] = [];\n  /**\n   * The selectors are encoded in pairs where:\n   * - even locations are attribute names\n   * - odd locations are attribute values.\n   *\n   * Example:\n   * Selector: `[key1=value1][key2]` would parse to:\n   * ```\n   * ['key1', 'value1', 'key2', '']\n   * ```\n   */\n  attrs: string[] = [];\n  notSelectors: CssSelector[] = [];\n\n  static parse(selector: string): CssSelector[] {\n    const results: CssSelector[] = [];\n    const _addResult = (res: CssSelector[], cssSel: CssSelector) => {\n      if (cssSel.notSelectors.length > 0 && !cssSel.element && cssSel.classNames.length == 0 &&\n          cssSel.attrs.length == 0) {\n        cssSel.element = '*';\n      }\n      res.push(cssSel);\n    };\n    let cssSelector = new CssSelector();\n    let match: string[]|null;\n    let current = cssSelector;\n    let inNot = false;\n    _SELECTOR_REGEXP.lastIndex = 0;\n    while (match = _SELECTOR_REGEXP.exec(selector)) {\n      if (match[1]) {\n        if (inNot) {\n          throw new Error('Nesting :not is not allowed in a selector');\n        }\n        inNot = true;\n        current = new CssSelector();\n        cssSelector.notSelectors.push(current);\n      }\n      if (match[2]) {\n        current.setElement(match[2]);\n      }\n      if (match[3]) {\n        current.addClassName(match[3]);\n      }\n      if (match[4]) {\n        current.addAttribute(match[4], match[6]);\n      }\n      if (match[7]) {\n        inNot = false;\n        current = cssSelector;\n      }\n      if (match[8]) {\n        if (inNot) {\n          throw new Error('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 this.hasElementSelector() && this.classNames.length == 0 && this.attrs.length == 0 &&\n        this.notSelectors.length === 0;\n  }\n\n  hasElementSelector(): boolean { return !!this.element; }\n\n  setElement(element: string|null = null) { this.element = element; }\n\n  /** Gets a template string for an element that matches the selector. */\n  getMatchingElementTemplate(): string {\n    const tagName = this.element || 'div';\n    const 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      const attrName = this.attrs[i];\n      const attrValue = this.attrs[i + 1] !== '' ? `=\"${this.attrs[i + 1]}\"` : '';\n      attrs += ` ${attrName}${attrValue}`;\n    }\n\n    return getHtmlTagDefinition(tagName).isVoid ? `<${tagName}${classAttr}${attrs}/>` :\n                                                  `<${tagName}${classAttr}${attrs}></${tagName}>`;\n  }\n\n  getAttrs(): string[] {\n    const result: string[] = [];\n    if (this.classNames.length > 0) {\n      result.push('class', this.classNames.join(' '));\n    }\n    return result.concat(this.attrs);\n  }\n\n  addAttribute(name: string, value: string = '') {\n    this.attrs.push(name, value && value.toLowerCase() || '');\n  }\n\n  addClassName(name: string) { this.classNames.push(name.toLowerCase()); }\n\n  toString(): string {\n    let res: string = this.element || '';\n    if (this.classNames) {\n      this.classNames.forEach(klass => res += `.${klass}`);\n    }\n    if (this.attrs) {\n      for (let i = 0; i < this.attrs.length; i += 2) {\n        const name = this.attrs[i];\n        const value = this.attrs[i + 1];\n        res += `[${name}${value ? '=' + value : ''}]`;\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<T = any> {\n  static createNotMatcher(notSelectors: CssSelector[]): SelectorMatcher<null> {\n    const notMatcher = new SelectorMatcher<null>();\n    notMatcher.addSelectables(notSelectors, null);\n    return notMatcher;\n  }\n\n  private _elementMap = new Map<string, SelectorContext<T>[]>();\n  private _elementPartialMap = new Map<string, SelectorMatcher<T>>();\n  private _classMap = new Map<string, SelectorContext<T>[]>();\n  private _classPartialMap = new Map<string, SelectorMatcher<T>>();\n  private _attrValueMap = new Map<string, Map<string, SelectorContext<T>[]>>();\n  private _attrValuePartialMap = new Map<string, Map<string, SelectorMatcher<T>>>();\n  private _listContexts: SelectorListContext[] = [];\n\n  addSelectables(cssSelectors: CssSelector[], callbackCtxt?: T) {\n    let listContext: SelectorListContext = null !;\n    if (cssSelectors.length > 1) {\n      listContext = new SelectorListContext(cssSelectors);\n      this._listContexts.push(listContext);\n    }\n    for (let i = 0; i < cssSelectors.length; i++) {\n      this._addSelectable(cssSelectors[i], callbackCtxt as T, 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(\n      cssSelector: CssSelector, callbackCtxt: T, listContext: SelectorListContext) {\n    let matcher: SelectorMatcher<T> = this;\n    const element = cssSelector.element;\n    const classNames = cssSelector.classNames;\n    const attrs = cssSelector.attrs;\n    const selectable = new SelectorContext(cssSelector, callbackCtxt, listContext);\n\n    if (element) {\n      const 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 (classNames) {\n      for (let i = 0; i < classNames.length; i++) {\n        const isTerminal = attrs.length === 0 && i === classNames.length - 1;\n        const className = classNames[i];\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 (attrs) {\n      for (let i = 0; i < attrs.length; i += 2) {\n        const isTerminal = i === attrs.length - 2;\n        const name = attrs[i];\n        const value = attrs[i + 1];\n        if (isTerminal) {\n          const terminalMap = matcher._attrValueMap;\n          let terminalValuesMap = terminalMap.get(name);\n          if (!terminalValuesMap) {\n            terminalValuesMap = new Map<string, SelectorContext<T>[]>();\n            terminalMap.set(name, terminalValuesMap);\n          }\n          this._addTerminal(terminalValuesMap, value, selectable);\n        } else {\n          const partialMap = matcher._attrValuePartialMap;\n          let partialValuesMap = partialMap.get(name);\n          if (!partialValuesMap) {\n            partialValuesMap = new Map<string, SelectorMatcher<T>>();\n            partialMap.set(name, partialValuesMap);\n          }\n          matcher = this._addPartial(partialValuesMap, value);\n        }\n      }\n    }\n  }\n\n  private _addTerminal(\n      map: Map<string, SelectorContext<T>[]>, name: string, selectable: SelectorContext<T>) {\n    let terminalList = map.get(name);\n    if (!terminalList) {\n      terminalList = [];\n      map.set(name, terminalList);\n    }\n    terminalList.push(selectable);\n  }\n\n  private _addPartial(map: Map<string, SelectorMatcher<T>>, name: string): SelectorMatcher<T> {\n    let matcher = map.get(name);\n    if (!matcher) {\n      matcher = new SelectorMatcher<T>();\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: T) => void)|null): boolean {\n    let result = false;\n    const element = cssSelector.element !;\n    const classNames = cssSelector.classNames;\n    const attrs = cssSelector.attrs;\n\n    for (let 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 (classNames) {\n      for (let i = 0; i < classNames.length; i++) {\n        const className = classNames[i];\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 (attrs) {\n      for (let i = 0; i < attrs.length; i += 2) {\n        const name = attrs[i];\n        const value = attrs[i + 1];\n\n        const terminalValuesMap = this._attrValueMap.get(name) !;\n        if (value) {\n          result =\n              this._matchTerminal(terminalValuesMap, '', cssSelector, matchedCallback) || result;\n        }\n        result =\n            this._matchTerminal(terminalValuesMap, value, cssSelector, matchedCallback) || result;\n\n        const partialValuesMap = this._attrValuePartialMap.get(name) !;\n        if (value) {\n          result = this._matchPartial(partialValuesMap, '', cssSelector, matchedCallback) || result;\n        }\n        result =\n            this._matchPartial(partialValuesMap, value, cssSelector, matchedCallback) || result;\n      }\n    }\n    return result;\n  }\n\n  /** @internal */\n  _matchTerminal(\n      map: Map<string, SelectorContext<T>[]>, name: string, cssSelector: CssSelector,\n      matchedCallback: ((c: CssSelector, a: any) => void)|null): boolean {\n    if (!map || typeof name !== 'string') {\n      return false;\n    }\n\n    let selectables: SelectorContext<T>[] = map.get(name) || [];\n    const starSelectables: SelectorContext<T>[] = map.get('*') !;\n    if (starSelectables) {\n      selectables = selectables.concat(starSelectables);\n    }\n    if (selectables.length === 0) {\n      return false;\n    }\n    let selectable: SelectorContext<T>;\n    let result = false;\n    for (let i = 0; i < selectables.length; i++) {\n      selectable = selectables[i];\n      result = selectable.finalize(cssSelector, matchedCallback) || result;\n    }\n    return result;\n  }\n\n  /** @internal */\n  _matchPartial(\n      map: Map<string, SelectorMatcher<T>>, name: string, cssSelector: CssSelector,\n      matchedCallback: ((c: CssSelector, a: any) => void)|null): boolean {\n    if (!map || typeof name !== 'string') {\n      return false;\n    }\n\n    const nestedSelector = map.get(name);\n    if (!nestedSelector) {\n      return false;\n    }\n    // TODO(perf): get rid of recursion and measure again\n    // TODO(perf): don't pass the whole selector into the recursion,\n    // but only the not processed parts\n    return nestedSelector.match(cssSelector, matchedCallback);\n  }\n}\n\n\nexport class SelectorListContext {\n  alreadyMatched: boolean = false;\n\n  constructor(public selectors: CssSelector[]) {}\n}\n\n// Store context to pass back selector and context when a selector is matched\nexport class SelectorContext<T = any> {\n  notSelectors: CssSelector[];\n\n  constructor(\n      public selector: CssSelector, public cbContext: T, public listContext: SelectorListContext) {\n    this.notSelectors = selector.notSelectors;\n  }\n\n  finalize(cssSelector: CssSelector, callback: ((c: CssSelector, a: T) => void)|null): boolean {\n    let result = true;\n    if (this.notSelectors.length > 0 && (!this.listContext || !this.listContext.alreadyMatched)) {\n      const notMatcher = SelectorMatcher.createNotMatcher(this.notSelectors);\n      result = !notMatcher.match(cssSelector, null);\n    }\n    if (result && callback && (!this.listContext || !this.listContext.alreadyMatched)) {\n      if (this.listContext) {\n        this.listContext.alreadyMatched = true;\n      }\n      callback(this.selector, this.cbContext);\n    }\n    return result;\n  }\n}\n"]} |
\ | No newline at end of file |