2020-01-07 03:22:35 +01:00
|
|
|
/**
|
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
|
*
|
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
|
*
|
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
*
|
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
|
* limitations under the License.
|
|
|
|
|
*/
|
2019-11-23 00:36:17 +01:00
|
|
|
|
|
|
|
|
import { SelectorEngine, SelectorRoot } from './selectorEngine';
|
|
|
|
|
|
2019-12-17 05:49:18 +01:00
|
|
|
export const CSSEngine: SelectorEngine = {
|
2019-11-23 00:36:17 +01:00
|
|
|
name: 'css',
|
|
|
|
|
|
|
|
|
|
create(root: SelectorRoot, targetElement: Element): string | undefined {
|
|
|
|
|
const tokens: string[] = [];
|
|
|
|
|
|
|
|
|
|
function uniqueCSSSelector(prefix?: string): string | undefined {
|
|
|
|
|
const path = tokens.slice();
|
|
|
|
|
if (prefix)
|
|
|
|
|
path.unshift(prefix);
|
|
|
|
|
const selector = path.join(' > ');
|
|
|
|
|
const nodes = Array.from(root.querySelectorAll(selector));
|
|
|
|
|
return nodes[0] === targetElement ? selector : undefined;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (let element: Element | null = targetElement; element && element !== root; element = element.parentElement) {
|
|
|
|
|
const nodeName = element.nodeName.toLowerCase();
|
|
|
|
|
|
|
|
|
|
// Element ID is the strongest signal, use it.
|
|
|
|
|
let bestTokenForLevel: string = '';
|
|
|
|
|
if (element.id) {
|
|
|
|
|
const token = /^[a-zA-Z][a-zA-Z0-9\-\_]+$/.test(element.id) ? '#' + element.id : `[id="${element.id}"]`;
|
|
|
|
|
const selector = uniqueCSSSelector(token);
|
|
|
|
|
if (selector)
|
|
|
|
|
return selector;
|
|
|
|
|
bestTokenForLevel = token;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const parent = element.parentElement;
|
|
|
|
|
|
|
|
|
|
// Combine class names until unique.
|
|
|
|
|
const classes = Array.from(element.classList);
|
|
|
|
|
for (let i = 0; i < classes.length; ++i) {
|
|
|
|
|
const token = '.' + classes.slice(0, i + 1).join('.');
|
|
|
|
|
const selector = uniqueCSSSelector(token);
|
|
|
|
|
if (selector)
|
|
|
|
|
return selector;
|
|
|
|
|
// Even if not unique, does this subset of classes uniquely identify node as a child?
|
|
|
|
|
if (!bestTokenForLevel && parent) {
|
|
|
|
|
const sameClassSiblings = parent.querySelectorAll(token);
|
|
|
|
|
if (sameClassSiblings.length === 1)
|
|
|
|
|
bestTokenForLevel = token;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Ordinal is the weakest signal.
|
|
|
|
|
if (parent) {
|
|
|
|
|
const siblings = Array.from(parent.children);
|
|
|
|
|
const sameTagSiblings = siblings.filter(sibling => (sibling as Element).nodeName.toLowerCase() === nodeName);
|
|
|
|
|
const token = sameTagSiblings.length === 1 ? nodeName : `${nodeName}:nth-child(${1 + siblings.indexOf(element)})`;
|
|
|
|
|
const selector = uniqueCSSSelector(token);
|
|
|
|
|
if (selector)
|
|
|
|
|
return selector;
|
|
|
|
|
if (!bestTokenForLevel)
|
|
|
|
|
bestTokenForLevel = token;
|
|
|
|
|
} else if (!bestTokenForLevel) {
|
|
|
|
|
bestTokenForLevel = nodeName;
|
|
|
|
|
}
|
|
|
|
|
tokens.unshift(bestTokenForLevel);
|
|
|
|
|
}
|
|
|
|
|
return uniqueCSSSelector();
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
query(root: SelectorRoot, selector: string): Element | undefined {
|
|
|
|
|
return root.querySelector(selector) || undefined;
|
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
queryAll(root: SelectorRoot, selector: string): Element[] {
|
|
|
|
|
return Array.from(root.querySelectorAll(selector));
|
|
|
|
|
}
|
|
|
|
|
};
|