nightwatch
Version:
Easy to use Node.js based end-to-end testing solution for web applications using the W3C WebDriver API.
114 lines (88 loc) • 3.47 kB
JavaScript
const {By, RelativeBy} = require('selenium-webdriver');
const Utils = require('../../utils');
const Element = require('../../element/index.js');
const Locator = require('../../element/locator.js');
const NightwatchLocator = require('../../element/locator-factory');
class ElementLocator {
constructor(selector, options = {}) {
this.index = ElementLocator.getOrDefault(selector, ['index', '__index'], 0);
this.timeout = ElementLocator.getOrDefault(selector, 'timeout', options.waitForConditionTimeout);
this.retryInterval = ElementLocator.getOrDefault(selector, 'retryInterval', options.waitForConditionPollInterval);
this.locateStrategy = ElementLocator.getLocateStrategy(selector, options.locateStrategy);
this.abortOnFailure = ElementLocator.getOrDefault(selector, 'abortOnFailure', true);
this.suppressNotFoundErrors = ElementLocator.getOrDefault(selector, 'suppressNotFoundErrors', false);
this.condition = ElementLocator.getCondition(selector, this.locateStrategy);
}
static create(selector, nightwatchInstance = {}) {
return new ElementLocator(selector, {
waitForConditionTimeout: nightwatchInstance.settings.globals.waitForConditionTimeout,
waitForConditionPollInterval: nightwatchInstance.settings.globals.waitForConditionPollInterval,
locateStrategy: nightwatchInstance.locateStrategy
});
}
static getOrDefault(obj, props, defaultValue) {
if (!Utils.isObject(obj)) {
return defaultValue;
}
const propArray = Array.isArray(props) ? props : [props];
for (const prop of propArray) {
// eslint-disable-next-line no-prototype-builtins
if (obj.hasOwnProperty(prop) && Utils.isDefined(obj[prop])) {
return obj[prop];
}
}
return defaultValue;
}
static getLocateStrategy(element, strategy) {
if (element instanceof By) {
return element.using;
}
if (Utils.isString(element)) {
return strategy;
}
return ElementLocator.getOrDefault(element, 'locateStrategy', strategy);
}
static isElementDescriptor(selector) {
if (!Utils.isObject(selector)) {
return false;
}
if (selector.__nightwatchScopedWebElement__) {
return true;
}
// eslint-disable-next-line no-prototype-builtins
return ['selector', 'xpath', 'css'].some(prop => selector.hasOwnProperty(prop));
}
static getSelectorValue(descriptor) {
if (descriptor.selector instanceof By || descriptor.selector instanceof RelativeBy) {
return descriptor.selector;
}
if (descriptor.xpath) {
return By.xpath(descriptor.xpath);
}
if (descriptor.css) {
return By.css(descriptor.css);
}
return null;
}
static getCondition(element, strategy) {
const locateStrategy = ElementLocator.getLocateStrategy(element, strategy);
if (element instanceof By || element instanceof RelativeBy) {
return element;
}
if (element instanceof Element) {
if (element.usingRecursion) {
return element;
}
return NightwatchLocator.create(element);
}
if (!ElementLocator.isElementDescriptor(element)) {
return By[Locator.AVAILABLE_LOCATORS[locateStrategy]](element);
}
const selector = ElementLocator.getSelectorValue(element);
if (selector) {
return selector;
}
return By[Locator.AVAILABLE_LOCATORS[locateStrategy]](element.selector);
}
}
module.exports.ScopedElementLocator = ElementLocator;