@christian-bromann/webdriverio
Version:
A nodejs bindings implementation for selenium 2.0/webdriver
145 lines (126 loc) • 4.85 kB
JavaScript
import findStrategy from '../helpers/findElementStrategy'
/**
* Returns the script to execute in the browser, in string format.
* @param {Function|String} fn - function to execute in the browser
* @param {Array.<String>} selectors - the selectors to resolve and pass to fn, each in its own array
* @param {Array} args - the arguments to pass to fn (after resolved selectors)
* @param {Function} callback
* @returns {string}
*/
let createSelectorScript = function (fn, selectors, args) {
let strArgs = []
let foundSel = []
if (typeof fn === 'function' || (this.inMultibrowserMode && fn.indexOf('function (') === 0)) {
// Handle function script
strArgs.push(fn.toString())
} else if (typeof fn === 'string') {
// Handle string script
strArgs.push(`function(){${fn}}`)
}
for (let selector of selectors) {
let val = findStrategy(selector)
foundSel.push(val.using, val.value)
}
strArgs.push(JSON.stringify(foundSel))
var i = -1
while (++i < args.length) {
if (typeof args[i] === 'function' || (this.inMultibrowserMode && (typeof args[i] === 'string' && args[i].indexOf('function (') === 0))) {
args[i] = args[i].toString()
} else {
args[i] = JSON.stringify(args[i])
}
}
strArgs.push('[' + args.join(',') + ']')
return (`return (${executeClientSide})(${strArgs.join(',')}, arguments[arguments.length - 1]);`).replace(/(\s{4}|\t)+/g, ' ')
}
/**
* Helper that resolves selectors client side and returns the result in the given fn.
* Every resolved selector is prepended to the function's arguments.
* Each resolved selector yields a single array.
*
* @param {Function} fn - the function to execute client side that will receive the resolved selectors
* @param {Array.<String>} sArr - a series of usage, value pairs from find-element-strategy
* @param {Array} args - any other arguments to pass to fn
* @returns {*} the return value of fn
* @example
* var helper = require('./executeClientSideSelector');
* // Execute in the browser
* helper(fn, ['xpath', '//body', 'css', '[id="what"]'], [1, 2, 3]);
* // Assuming fn is...
* fn = function(xpathResult, cssResult, one, two, three) {
* console.log(xpathResult.length); // 1
* console.log(xpathResult[0]); // <body.../>
* console.log(cssResult.length); // 0
* console.log(one, two, three); // 1 2 3
* }
*/
var executeClientSide = function (fn, sArr, args) {
var cb = arguments[arguments.length - 1]
var i = 0
var sArgs = []
var use, value, xp, cs, tn, res, j, arg
if (typeof document.querySelectorAll === 'undefined') {
document.querySelectorAll = function (selector) {
var doc = document
var head = doc.documentElement.firstChild
var styleTag = doc.createElement('STYLE')
head.appendChild(styleTag)
doc.__qsaels = []
styleTag.styleSheet.cssText = selector + '{x:expression(document.__qsaels.push(this))}'
window.scrollBy(0, 0)
return doc.__qsaels
}
}
while ((use = sArr[i++]) && (value = sArr[i++])) {
arg = []
xp = cs = tn = null
switch (use) {
case 'partial link text':
xp = '//a[contains(text(),"' + value + '")]'
break
case 'link text':
xp = '//a[text()="' + value + '"]'
break
case 'xpath':
xp = value
break
case 'id':
cs = '#' + value
break
case 'name':
cs = '[name="' + value + '"]'
break
case 'tag name':
tn = value
break
case 'css selector':
cs = value
break
default: throw new Error('Could not evaluate selector: Invalid strategy ' + use)
}
if (xp) {
res = document.evaluate(xp, document, null, 0, null)
value = res.iterateNext()
while (value) {
arg.push(value)
value = res.iterateNext()
}
} else if (tn || cs) {
res = tn ? document.getElementsByTagName(tn) : document.querySelectorAll(cs)
for (j = 0; j < res.length; ++j) {
arg.push(res[j])
}
}
sArgs.push(arg)
}
var parameter = args && sArgs.concat(args) || sArgs
if (parameter.length === 0 || (parameter.length === 1 && parameter[0].length === 0)) {
if (typeof cb === 'function') {
return cb('NoSuchElement') && new Error('NoSuchElement')
}
return new Error('NoSuchElement')
}
parameter.push(arguments[arguments.length - 1])
return fn.apply(window, parameter)
}
export default createSelectorScript