All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.seleniumtests.driver.CustomEventFiringWebDriver Maven / Gradle / Ivy

/**
 * Orignal work: Copyright 2015 www.seleniumtests.com
 * Modified work: Copyright 2016 www.infotel.com
 * 				Copyright 2017-2019 B.Hecquet
 *
 * 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.
 */
package com.seleniumtests.driver;

import java.awt.AWTError;
import java.awt.AWTException;
import java.awt.GraphicsDevice;
import java.awt.GraphicsEnvironment;
import java.awt.HeadlessException;
import java.awt.MouseInfo;
import java.awt.PointerInfo;
import java.awt.Rectangle;
import java.awt.Robot;
import java.awt.Toolkit;
import java.awt.datatransfer.StringSelection;
import java.awt.event.InputEvent;
import java.awt.event.KeyEvent;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Date;
import java.util.List;
import java.util.Set;
import java.util.TreeSet;
import java.util.UUID;

import javax.imageio.ImageIO;

import org.apache.commons.codec.binary.Base64;
import org.apache.commons.codec.binary.Base64OutputStream;
import org.apache.commons.io.FileUtils;
import org.apache.log4j.Logger;
import org.openqa.selenium.Alert;
import org.openqa.selenium.Capabilities;
import org.openqa.selenium.Dimension;
import org.openqa.selenium.HasCapabilities;
import org.openqa.selenium.JavascriptException;
import org.openqa.selenium.JavascriptExecutor;
import org.openqa.selenium.MutableCapabilities;
import org.openqa.selenium.NoSuchSessionException;
import org.openqa.selenium.NoSuchWindowException;
import org.openqa.selenium.Point;
import org.openqa.selenium.UnhandledAlertException;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.WebElement;
import org.openqa.selenium.ie.InternetExplorerDriver;
import org.openqa.selenium.remote.FileDetector;
import org.openqa.selenium.remote.RemoteWebDriver;
import org.openqa.selenium.remote.UselessFileDetector;
import org.openqa.selenium.safari.SafariDriver;
import org.openqa.selenium.support.events.EventFiringWebDriver;

import com.neotys.selenium.proxies.NLWebDriver;
import com.seleniumtests.browserfactory.BrowserInfo;
import com.seleniumtests.connectors.selenium.SeleniumGridConnector;
import com.seleniumtests.core.StatisticsStorage;
import com.seleniumtests.core.StatisticsStorage.DriverUsage;
import com.seleniumtests.customexception.DriverExceptions;
import com.seleniumtests.customexception.ScenarioException;
import com.seleniumtests.customexception.WebSessionEndedException;
import com.seleniumtests.util.helper.WaitHelper;
import com.seleniumtests.util.logging.SeleniumRobotLogger;
import com.seleniumtests.util.osutility.OSUtilityFactory;
import com.seleniumtests.util.video.VideoRecorder;

import net.lightbody.bmp.BrowserMobProxy;

/**
 * This class acts as a proxy for everything related to selenium driver actions (mostly a bypass) or with the machine holding
 * the browser
 * - send keys via keyboard
 * - move mouse
 * - upload file to browser
 * - capture video
 * 
 * When action do not need a real driver, static methods are provided
 * It also handles the grid mode, masking it to requester.
 */
public class CustomEventFiringWebDriver extends EventFiringWebDriver implements HasCapabilities {
	
	private static final String OTHER_BROWSER = "other";
	private static final String SAFARI_BROWSER = "safari";
	private static final Logger logger = SeleniumRobotLogger.getLogger(CustomEventFiringWebDriver.class);
    private FileDetector fileDetector = new UselessFileDetector();
    private static final int MAX_DIMENSION = 100000;
    private Set currentHandles;
    private final List driverPids;
	private final WebDriver driver;
	private final NLWebDriver neoloadDriver;
	private final boolean isWebTest;
	private final DriverMode driverMode;
	private final BrowserInfo browserInfo;
	private final BrowserMobProxy mobProxy;
	private final SeleniumGridConnector gridConnector;
	private final Integer attachExistingDriverPort;
	private MutableCapabilities internalCapabilities = new MutableCapabilities();
	
	/*
	 * returns pixel aspect ratio of web site, it depends on OS and browser zoom
	 * We can force to use PAR=1 by calling driver.executeScript("Some JS", false)
	 * Cases which need the real PAR:
	 * - Snapshot capture
	 * - operations on ScreenZon
	 * - more generally, every operation where we interact with screen
	 * Cases which need PAR=1
	 * - operations on PictureElement
	 * - more generally, every operation where we interact with selenium elements which do not take PAR into account
	 */
	private static final String JS_PIXEL_RATIO = "var pixelRatio;"
			+ "try{"
			+ "    if (arguments.length==0 || arguments[0]) {"				
			+ "        pixelRatio = devicePixelRatio"
			+ "    } else {"
			+ "        pixelRatio = 1"
			+ "    }"
			+ "} catch(err){"
			+ "    pixelRatio=1"
			+ "}";
    
    private static final String JS_GET_VIEWPORT_SIZE_WIDTH = String.format(
    		JS_PIXEL_RATIO
    				+ "var width = %d;"
    				+ "if (window.innerWidth) {"
    				+ "   width = Math.min(window.innerWidth, width);"
    				+ "} "
    				+ "if (document.documentElement && document.documentElement.clientWidth) {"
    				+ "   width = Math.min(document.documentElement.clientWidth, width);"
    				+ "} "
    				+ "var b = document.getElementsByTagName('html')[0]; "
    				+ "if (b.clientWidth) {"
    				+ "   width = Math.min(b.clientWidth, width);"
    				+ "}"
    				+ "return width * pixelRatio;", MAX_DIMENSION);
    
    private static final String JS_GET_VIEWPORT_SIZE_HEIGHT = String.format(
    				JS_PIXEL_RATIO
    				+ "var height = %d;"
    				+ "if (window.innerHeight) {"
    				+ "    height = Math.min(window.innerHeight, height);"
    				+ "}"
    				+ "if (document.documentElement && document.documentElement.clientHeight) {"
    				+ "    height = Math.min(document.documentElement.clientHeight, height);"
    				+ "}"
    				+ "var b = document.getElementsByTagName('html')[0]; "
    				+ "if (b.clientHeight) {"
    				+ "    height = Math.min(b.clientHeight, height);"
    				+ "}"
    				+ "return height * pixelRatio;", MAX_DIMENSION);

    private static final String JS_GET_CURRENT_SCROLL_POSITION =
            "var doc = document.documentElement; "
            + "var x = window.scrollX || ((window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0));"
            + "var y = window.scrollY || ((window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0));"
            + "return [x, y];";


    // IMPORTANT: Notice there's a major difference between scrollWidth
    // and scrollHeight. While scrollWidth is the maximum between an
    // element's width and its content width, scrollHeight might be
    // smaller (!) than the clientHeight, which is why we take the
    // maximum between them.
    private static final String JS_GET_CONTENT_ENTIRE_SIZE =
    		JS_PIXEL_RATIO +
    		"var scrollWidth = document.documentElement.scrollWidth; " +
            "var bodyScrollWidth = document.body.scrollWidth; " +
            "var totalWidth = Math.max(scrollWidth, bodyScrollWidth); " +
            "var clientHeight = document.documentElement.clientHeight; " +
            "var bodyClientHeight = document.body.clientHeight; " +
            "var scrollHeight = document.documentElement.scrollHeight; " +
            "var bodyScrollHeight = document.body.scrollHeight; " +
            "var maxDocElementHeight = Math.max(clientHeight, scrollHeight); " +
            "var maxBodyHeight = Math.max(bodyClientHeight, bodyScrollHeight); " +
            "var totalHeight = Math.max(maxDocElementHeight, maxBodyHeight); " +
            "return [totalWidth * pixelRatio, totalHeight * pixelRatio];";
    
    // according to https://www.w3schools.com/jsref/prop_element_scrolltop.asp
    // root scrollable (overflow) element is 'document.body' for safari and 'document.documentElement' for other browsers
    private static final String JS_SCROLL_PARENT = "function getScrollParent(element, includeHidden, browserName) {" + 
    		"    var rootElement = browserName === \"safari\" ? document.body: document.documentElement;" +
    		"    var style = getComputedStyle(element);" + 
    		"    var excludeStaticParent = style.position === \"absolute\";" + 
    		"    var overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/;" + 
    		"" + 
    		"    if (style.position === \"fixed\") return rootElement;" + 
    		"    for (var parent = element; (parent = parent.parentElement);) {" + 
    		"        style = getComputedStyle(parent);" + 
    		"        if (excludeStaticParent && style.position === \"static\") {" + 
    		"            continue;" + 
    		"        }" + 
    		"        if (overflowRegex.test(style.overflow + style.overflowY + style.overflowX)) return parent;" + 
    		"    }" + 
    		"" + 
    		"    return rootElement;" + 
    		"}" +
    		"return getScrollParent(arguments[0], false, arguments[1]);";
    

    /*
	an other methode for finding scolling parent, that computes full xpath and then search a matching parent
    this method will work in case of shadowDom / document fragments / slots
	@param element			the element for which we want to get a scrollable parent
	@param includeHidden	whether to include hidden scrollable parents in search
	@param browserName		name of the browser

	TESTED WITH
	- element does not exist
	- no scrollable element in parents
	- scrollable element in parents
	*/
    private static final String JS_SCROLL_PARENT2 = "getScrollableParent = function(element, includeHidden, browserName) {"
    
    		/**
    			function for getting the full XPath of an element, or null if element cannot be found
    			Based on example found here: https://stackoverflow.com/questions/2661818/javascript-get-xpath-of-a-node/43688599#43688599
    			@param root		root element to search in
    			@param element	the element for which we want to get xpath
    			@param xpath	the base xpath (should be empty on first call, used for recursion)
    			*/
    		
    		+ "path = function(root, element, xpath) {"
    		+ "    _xPathIndex = function (node) {"
    		// Returns -1 in case of error, 0 if no siblings matching the same expression,"
    		//  otherwise."
    		+ "         function areNodesSimilar(left, right) {"
    		+ "             if (left === right) {"
    		+ "                 return true;"
    		+ "             }"
    		+ ""
    		+ "             if (left.nodeType === Node.ELEMENT_NODE && right.nodeType === Node.ELEMENT_NODE) {"
    		+ "                 return left.localName === right.localName;"
    		+ "             }"
    		+ ""
    		+ "             if (left.nodeType === right.nodeType) {"
    		+ "                 return true;"
    		+ "             }"
    		+ ""
    		// XPath treats CDATA as text nodes.
    		+ "             const leftType = left.nodeType === Node.CDATA_SECTION_NODE ? Node.TEXT_NODE : left.nodeType;"
    		+ "             const rightType = right.nodeType === Node.CDATA_SECTION_NODE ? Node.TEXT_NODE : right.nodeType;"
    		+ "             return leftType === rightType;"
    		+ "         }"
    		+ ""
    		+ "         const siblings = node.parentNode ? node.parentNode.children : null;"
    		// Root node - no siblings.
    		+ "         if (!siblings) {"
    		+ "             return 0;"
    		+ "         } "
    		+ "         let hasSameNamedElements;"
    		+ "         for (let i = 0; i < siblings.length; ++i) {"
    		+ "             if (areNodesSimilar(node, siblings[i]) && siblings[i] !== node) {"
    		+ "                 hasSameNamedElements = true;"
    		+ "                 break;"
    		+ "             }"
    		+ "         }"
    		+ "         if (!hasSameNamedElements) {"
    		+ "             return 0;"
    		+ "         }"
    		// XPath indices start with 1.
    		+ "         let ownIndex = 1;  "
    		+ "         for (let i = 0; i < siblings.length; ++i) {"
    		+ "             if (areNodesSimilar(node, siblings[i])) {"
    		+ "                 if (siblings[i] === node) {"
    		+ "                      return ownIndex;"
    		+ "                 }"
    		+ "                 ++ownIndex;"
    		+ "                   }"
    		+ "         }"
    		// An error occurred: |node| not found in parent's children.
    		+ "         return -1;"
    		+ "     };"
    		+ "     "
    		+ "     var elements = Array.from(root.children);"
    		+ "     "
    		// handle shadowRoot
    		+ "     if (root.shadowRoot != null) {"
    		+ "         elements = Array.from(root.shadowRoot.children).concat(elements);"
    		+ "     } "
    		// handle slots which carry some element inside even if children or shadowRoot is null
    		+ "     if (root.localName === 'slot' && root.assignedNodes() != null) {"
    		+ "         elements = Array.from(root.assignedElements()).concat(elements);"
    		+ "     } "
    		+ "     "
    		+ "     for (var i=0; i 0) {"
    		+ "             p += '[' + ownIndex + ']';"
    		+ "         }"
    		+ "         "
    		+ "         if (element === elements[i]) {"
//    		+ "             console.log(\"trouve\");"
    		+ "             return p;"
    		+ "             break;"
    		+ "         }"
    		+ ""
    		+ "         var ret = path(elements[i], element, p);"
    		+ "         if (ret != null) {"
    		+ "             return ret;"
    		+ "         }"
    		+ "     }"
    		+ ""
    		+ "     return null;"
    		+ "   "
    		+ " };"
    		+ " "
    		+ " "
    		+ " let rootElement = browserName === \"safari\" ? document.body: document.documentElement;"
    		+ " if (element == null) {"
    		+ "     return rootElement;"
    		+ " }"
    		+ " "
    		// get full xpath of the element
    		+ " let elementFullXpath = path(document, element, '');"
    		+ " let style = getComputedStyle(element);"
    		+ " let excludeStaticParent = style.position === \"absolute\";"
    		+ " let overflowRegex = includeHidden ? /(auto|scroll|hidden)/ : /(auto|scroll)/;"
    		+ " if (style.position === \"fixed\") return rootElement;"
    		+ " "
    		// parse xpath from end to begin to find a scrollable element
    		+ " if (elementFullXpath != null) {"
    		+ "     while ((elementFullXpath.match(new RegExp(\"/\", \"g\")) || []).length > 1) {"
    		+ "         elementFullXpath = elementFullXpath.substring(0, elementFullXpath.lastIndexOf('/'));"
    		+ "         let elementToEvaluate = document.evaluate(elementFullXpath, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;"
    		+ ""
    		+ "         if (elementToEvaluate === null) {"
    		+ "             return rootElement;"
    		+ "         } else {"
    		+ "             style = getComputedStyle(elementToEvaluate);"
    		+ "             if (excludeStaticParent && style.position === \"static\") {"
    		+ "                 continue;"
    		+ "             } "
    		+ ""
    		+ "             if (overflowRegex.test(style.overflow + style.overflowY + style.overflowX)) return elementToEvaluate;"
    		+ "         }"
    		+ "     }"
    		+ " }"
    		+ " return rootElement;"
    		+ "};" +
    		"return getScrollableParent(arguments[0], false, arguments[1]);";
    
    
    private static final String JS_GET_ELEMENT_STYLE = "function getStyle(dom) {" + 
    		"  var style;" + 
    		"  var returns = {};" + 
    		"  /* FireFox and Chrome way */" + 
    		"  if (window.getComputedStyle) {" + 
    		"    style = window.getComputedStyle(dom, null);" + 
    		"    for (var i = 0, l = style.length; i < l; i++) {" + 
    		"      var prop = style[i];" + 
    		"      var val = style.getPropertyValue(prop);" + 
    		"      returns[prop] = val;" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"" + 
    		"  /* IE and Opera way */" + 
    		"  if (dom.currentStyle) {" + 
    		"    style = dom.currentStyle;" + 
    		"    for (var prop in style) {" + 
    		"      returns[prop] = style[prop];" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"  /* Style from style attribute */" + 
    		"  if ((style = dom.style)) {" + 
    		"    for (var prop in style) {" + 
    		"      if (typeof style[prop] != \"function\") {" + 
    		"        returns[prop] = style[prop];" + 
    		"      }" + 
    		"    }" + 
    		"    return returns;" + 
    		"  }" + 
    		"  return returns;" + 
    		"}";

    private static final String JS_IS_ELEMENT_VISIBLE = "function isVisible(elem) {" + 
    		"  var style = getStyle(elem);" + 
    		"" + 
    		"  if (style[\"display\"] === \"none\") return false;" + 
    		"  if (style[\"visibility\"] !== \"visible\") return false;" + 
    		"  if (style[\"opacity\"] === 0) return false;" + 
    		"" + 
    		"  if (" + 
    		"    elem.offsetWidth +" + 
    		"      elem.offsetHeight +" + 
    		"      elem.getBoundingClientRect().height +" + 
    		"      elem.getBoundingClientRect().width ===" + 
    		"    0" + 
    		"  )" + 
    		"    return false;" + 
    		"" + 
    		"  var elementPoints = {" + 
    		"    center: {" + 
    		"      x: elem.getBoundingClientRect().left + elem.offsetWidth / 2," + 
    		"      y: elem.getBoundingClientRect().top + elem.offsetHeight / 2" + 
    		"    }," + 
    		"    topLeft: {" + 
    		"      x: elem.getBoundingClientRect().left," + 
    		"      y: elem.getBoundingClientRect().top" + 
    		"    }," + 
    		"    topRight: {" + 
    		"      x: elem.getBoundingClientRect().right," + 
    		"      y: elem.getBoundingClientRect().top" + 
    		"    }," + 
    		"    bottomLeft: {" + 
    		"      x: elem.getBoundingClientRect().left," + 
    		"      y: elem.getBoundingClientRect().bottom" + 
    		"    }," + 
    		"    bottomRight: {" + 
    		"      x: elem.getBoundingClientRect().right," + 
    		"      y: elem.getBoundingClientRect().bottom" + 
    		"    }" + 
    		"  };" + 
    		"" + 
    		"  var docWidth = document.documentElement.clientWidth || window.innerWidth;" + 
    		"  var docHeight = document.documentElement.clientHeight || window.innerHeight;" + 
    		"" + 
    		"  if (elementPoints.topLeft.x > docWidth) return false;" + 
    		"  if (elementPoints.topLeft.y > docHeight) return false;" + 
    		"  if (elementPoints.bottomRight.x < 0) return false;" + 
    		"  if (elementPoints.bottomRight.y < 0) return false;" + 
    		"" + 
    		"  for (var index in elementPoints) {" + 
    		"    var point = elementPoints[index];" + 
    		"    var pointContainer = document.elementFromPoint(point.x, point.y);" + 
    		"    if (pointContainer !== null) {" + 
    		"      do {" + 
    		"        if (pointContainer === elem) return true;" + 
    		"      } while ((pointContainer = pointContainer.parentNode));" + 
    		"    }" + 
    		"  }" + 
    		"  return false;" + 
    		"}";
    
    // return true if a modal is present: a 'visible' div whose size is the same as the viewport
    private static final String JS_GET_MODAL = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +
    		"" +
    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +
    		"" +
    		JS_GET_ELEMENT_STYLE +
    		"" + 
    		JS_IS_ELEMENT_VISIBLE + 

    		"var modals;" + 
    		"var nodes = document.querySelectorAll(\"div\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  modals = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    modals.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +
    		"" + 
    		"if (modals && modals.length > 0) {" + 
    		// only keep modals which are wide enough and visible
    		"  modals = modals.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth >= viewportWidth && e.clientHeight >= viewportHeight;" + 
    		"  });" + 
    		"  if (modals.length > 0) {" + 
    		"    return true;" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"return false;";
    
    // returns the top pixel from which fixed positioned headers are not present
    private static final String JS_GET_TOP_HEADER = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +
    		"" +
    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +
    		"" +
    		JS_GET_ELEMENT_STYLE +
    		"" + 
    		JS_IS_ELEMENT_VISIBLE + 
    		"" + 
    		JS_PIXEL_RATIO +
    		"var headers;" + 
    		"var nodes = document.querySelectorAll(\"div,header,ul\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  headers = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    headers.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"var topPixel = 0;" + 
    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +
    		"" + 
    		"if (headers && headers.length > 0) {" + 
    		// only keep headers which are wide enough, not too high and visible
    		"  headers = headers.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth > viewportWidth / 2 && e.clientHeight < viewportHeight / 2;" + 
    		"  });" + 
    		"  headers = headers.sort(function(a, b) {" + 
    		"    if (a == null || a.offsetTop == null) return 1;" + 
    		"    if (b == null || b.offsetTop == null) return -1;" + 
    		"    return a.offsetTop - b.offsetTop;" + 
    		"  });" + 
    		"" + 
    		"  for (var i = 0; i < headers.length; i++) {" + 
    		"    var header = headers[i];" + 
    		"    if (header.offsetTop <= topPixel + 10) {" + 
    		"      topPixel = header.offsetTop + header.scrollHeight;" + 
    		"    } else {" + 
    		"      break;" + 
    		"    }" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"return Math.round(topPixel * pixelRatio);";
    
    private static final String JS_GET_BOTTOM_FOOTER = 
    		"function getViewportWidth() {" + JS_GET_VIEWPORT_SIZE_WIDTH + "}" +
    		"" +
    		"function getViewportHeight() {" + JS_GET_VIEWPORT_SIZE_HEIGHT + "}" +
    		"" +
    		JS_GET_ELEMENT_STYLE +
    		"" + 
    		JS_IS_ELEMENT_VISIBLE +
    		"" + 
    		JS_PIXEL_RATIO +
    		"" + 
    		"var footers;" + 
    		"var nodes = document.querySelectorAll(\"div,footer,ul\");" + 
    		"if (nodes && nodes.length > 0) {" + 
    		"  var length = nodes.length;" + 
    		"  footers = new Array();" + 
    		"  for (var i = 0; i < length; i++) {" + 
    		"    footers.push(nodes[i]);" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"var bottomPixel = document.documentElement.clientHeight;" + // count from bottom
    		"var viewportWidth = getViewportWidth();" +
    		"var viewportHeight = getViewportHeight();" +
    		"" + 
    		"if (footers && footers.length > 0) {" + 
    		// only keep footers which are wide enough, not too high and visible
    		"  footers = footers.filter(function(e) {" + 
    		"    return getComputedStyle(e)[\"position\"] === \"fixed\" && isVisible(e) && e.clientWidth > viewportWidth / 2 && e.clientHeight < viewportHeight / 2;" + 
    		"  });" + 
    		"  footers = footers.sort(function(a, b) {" + 
    		"    if (a == null || a.offsetBottom == null) return -1;" + 
    		"    if (b == null || b.offsetBottom == null) return 1;" + 
    		"    return b.offsetBottom - a.offsetBottom;" + 
    		"  });" + 
    		"" + 
    		"  for (var i = 0; i < footers.length; i++) {" + 
    		"    var footer = footers[i];" + 
    		"    if (footer.offsetTop + footer.scrollHeight >= bottomPixel - 10) {" + 
    		"      bottomPixel = footer.offsetTop;" + 
    		"    } else {" + 
    		"      break;" + 
    		"    }" + 
    		"  }" + 
    		"}" + 
    		"" + 
    		"return Math.round(document.documentElement.clientHeight * pixelRatio - bottomPixel * pixelRatio);";
    		
    
    public static final String NON_JS_UPLOAD_FILE_THROUGH_POPUP = 
    		"var action = 'upload_file_through_popup';";
    public static final String NON_JS_CAPTURE_DESKTOP = 
    		"var action = 'capture_desktop_snapshot_to_base64_string';return '';";
    
    public CustomEventFiringWebDriver(final WebDriver driver) {
    	this(driver, null, null, true, DriverMode.LOCAL, null, null);
    }

    public CustomEventFiringWebDriver(final WebDriver driver, List driverPids, BrowserInfo browserInfo, Boolean isWebTest, DriverMode localDriver, BrowserMobProxy mobProxy, SeleniumGridConnector gridConnector) {
    	this(driver, driverPids, browserInfo, isWebTest, localDriver, mobProxy, gridConnector, null);
    }
	public CustomEventFiringWebDriver(final WebDriver driver, List driverPids, BrowserInfo browserInfo, Boolean isWebTest, DriverMode localDriver, BrowserMobProxy mobProxy, SeleniumGridConnector gridConnector, Integer attachExistingDriverPort) {
        super(driver);
        this.driverPids = driverPids == null ? new ArrayList<>(): driverPids;
		this.driver = driver;
		this.browserInfo = browserInfo;
		this.isWebTest = isWebTest;
		this.driverMode = localDriver;
		this.mobProxy = mobProxy;
		this.gridConnector = gridConnector;
		this.attachExistingDriverPort = attachExistingDriverPort;
		
		// NEOLOAD //
		if (driver instanceof NLWebDriver) {
			neoloadDriver = (NLWebDriver)driver;
		} else {
			neoloadDriver = null;
		}
    }

    public void setFileDetector(final FileDetector detector) {
        if (detector == null) {
            throw new WebDriverException("file detector is null");
        }

        fileDetector = detector;
    }
    
    /**
     * Method for updating window handles when an operation may create a new window (a click action)
     * This is called for Composite actions, native actions (from DriverListener) and JS actions
     */
    public void updateWindowsHandles() {
    	if (isWebTest) {
    		currentHandles = getWindowHandles();
    			
    	} else {
    		currentHandles = new TreeSet<>();
    	}
    }
    
    public void setCurrentHandles(Set currentHandles) {
		this.currentHandles = currentHandles;
	}

	@Override
    public Set getWindowHandles() {
    	
    	if (!isWebTest) {
    		return new TreeSet<>();
    	}
    	
    	// issue #169: workaround for ios / IE tests where getWindowHandles sometimes fails with: class org.openqa.selenium.WebDriverException: Returned value cannot be converted to List: true
    	for (int i = 0; i < 10; i++) {
			try  {
				return super.getWindowHandles();
			} catch (UnhandledAlertException e) {
	    		logger.info("getWindowHandles: Handling alert");
	    		handleAlert();
	    		return super.getWindowHandles();
			} catch (WebSessionEndedException e) {
				logger.warn("session already terminated");
				return new TreeSet<>();
			} catch (Exception e) {
				logger.info("error getting window handles: " + e.getMessage());
				WaitHelper.waitForSeconds(2);
			}
		}
    	return super.getWindowHandles();
    	
    }
    
    private void handleAlert() {
    	try {
	    	Alert alert = driver.switchTo().alert();
			alert.dismiss();
    	} catch (Exception e) {
    		// do nothing in case of error
    	}
    }
    
    @Override
    public String getWindowHandle() {
    	
    	if (!isWebTest) {
    		return "";
    	}
    	
    	try {
    		return super.getWindowHandle();
    	} catch (UnhandledAlertException e) {
    		logger.info("getWindowHandle: Handling alert");
    		handleAlert();
    		return super.getWindowHandle();
    	}
    }
    
    @Override
    public void close() {
    	try {
    		super.close();
    	} catch (UnhandledAlertException e) {
    		logger.info("close: Handling alert");
    		handleAlert();
    		super.close();
    	}
    }
    
    @Override
    public String getCurrentUrl() {
    	try {
	    	return super.getCurrentUrl();
	    } catch (UnhandledAlertException e) {
    		logger.info("getCurrentUrl: Handling alert");
			handleAlert();
			return super.getCurrentUrl();
		} catch (NoSuchWindowException e) {
			return "";
		}
    }
    
    @Override
    public String getTitle() {
    	try {
    		return super.getTitle();
    	} catch (UnhandledAlertException e) {
    		logger.info("getTitle: Handling alert");
			handleAlert();
			return super.getTitle();
    	} catch (NoSuchWindowException e) {
			return "";
    	}
    }
    
    

    public FileDetector getFileDetector() {
        return fileDetector;
    }

    /**
     * Returns the Selenium driver
     * @return
     */
    public WebDriver getWebDriver() {
        return driver;
    }
    
    /**
     * Returns true if all browser windows are closed and so, no actions can be performed
     * @return
     */
    public boolean isBrowserClosed() {
    	try {
    		getSessionId();
    		getCapabilities();
    		return driver.getWindowHandles().isEmpty();
    	} catch (NoSuchSessionException e) {
    		return true;
    	}
    	
    }
    
    public String getSessionId() {
    	try {
    		return ((RemoteWebDriver)driver).getSessionId().toString();
    	} catch (ClassCastException e) {
    		return UUID.randomUUID().toString();
    	} catch (NullPointerException e) {
    		return null;
    	}
    }
    
    /**
     * Handle WebDriver exception when this method is not implemented
     */
    @Override
    public String getPageSource() {
    	try {
    		return super.getPageSource();
    	} catch (UnhandledAlertException e) {
    		logger.info("getPageSource: Handling alert");
			handleAlert();
			return super.getPageSource();
    	} catch (WebDriverException e) {
    		logger.info("page source not get: " + e.getMessage());
    		return null;
    	}
    }
    
    public Set getCurrentHandles() {
    	if (currentHandles == null) {
    		updateWindowsHandles();
    	}
		return currentHandles;
	}
    
    /**
     * Return the device aspect ratio
     * @return
     */
    public double getDeviceAspectRatio() {
    	if (isWebTest) {
    		Number devicePixelRatio = (Number) ((JavascriptExecutor)driver).executeScript(JS_PIXEL_RATIO + " return pixelRatio;", true);
    		return devicePixelRatio.doubleValue();
    	} else {
    		return 1;
    	}
    }
    
    public int getViewPortWidthWithoutScrollbar() {
    	return getViewPortWidthWithoutScrollbar(true);
    }
    /**
	 * Get viewport width, do not take scrollbar into account
	 * @param usePixelAspectRatio	if true, take zoom level into account to compute size
	 * @return
	 */
	public int getViewPortWidthWithoutScrollbar(boolean usePixelAspectRatio) {
    	if (isWebTest) {
    		try {
    			
	    		Number width = (Number)((JavascriptExecutor)driver).executeScript(JS_GET_VIEWPORT_SIZE_WIDTH, usePixelAspectRatio);
	    		
	    		// issue #238: check we get a non max size
	    		if (width.intValue() == MAX_DIMENSION) {
	    			driver.switchTo().defaultContent();
	    			width = (Number)((JavascriptExecutor)driver).executeScript(JS_GET_VIEWPORT_SIZE_WIDTH, usePixelAspectRatio);
	    		}
	    		return width.intValue();
	    		
    		} catch (Exception e) {
    			return driver.manage().window().getSize().getWidth();
    		}
    	} else {
    		return driver.manage().window().getSize().getWidth();
    	}
    }
	
	/**
	 * Get viewport size, do not take scrollbar into account
	 * @return
	 */
	public int getViewPortHeighthWithoutScrollbar() {
		return getViewPortHeighthWithoutScrollbar(true);
	}
	
	/**
	 * Get viewport size, do not take scrollbar into account
	 * @param usePixelAspectRatio	if true, take zoom level into account to compute size
	 * @return
	 */
	public int getViewPortHeighthWithoutScrollbar(boolean usePixelAspectRatio) {
		if (isWebTest) {
			try {
				
				Number height = (Number)((JavascriptExecutor)driver).executeScript(JS_GET_VIEWPORT_SIZE_HEIGHT, usePixelAspectRatio);
				
				// issue #238: check we get a non max size
				if (height.intValue() == MAX_DIMENSION) {
					driver.switchTo().defaultContent();
					height = (Number)((JavascriptExecutor)driver).executeScript(JS_GET_VIEWPORT_SIZE_HEIGHT, usePixelAspectRatio);
				}
				return height.intValue();
				
			} catch (Exception e) {
				return driver.manage().window().getSize().getHeight();
			}
		} else {
			return driver.manage().window().getSize().getHeight();
		}
	}
    
    /**
     * get dimensions of the visible part of the page
     * TODO: handle mobile app case
     * @return
     */
    @SuppressWarnings("unchecked")
    public Dimension getViewPortDimensionWithoutScrollbar() {
    	return getViewPortDimensionWithoutScrollbar(true);
    }
    
    /**
     * get dimensions of the visible part of the page
     * 
     * @return
     */
	public Dimension getViewPortDimensionWithoutScrollbar(boolean usePixelAspectRatio) {
    	Dimension foundDimension = new Dimension(getViewPortWidthWithoutScrollbar(usePixelAspectRatio), getViewPortHeighthWithoutScrollbar(usePixelAspectRatio));
		
		// issue #233: prevent too big size at it may be used to process images and Buffer
		if (foundDimension.width * foundDimension.height * 8L > Integer.MAX_VALUE) {
			return new Dimension(2000, 10000);
		} else {
			return foundDimension;
		}
    }
    
	
	public Dimension getContentDimension() {
		return getContentDimension(true);
	}
    /**
     * Get the whole webpage dimension
     * TODO: handle mobile app case
     * @return
     */

	@SuppressWarnings("unchecked")
    public Dimension getContentDimension(boolean usePixelAspectRatio) {
    	if (isWebTest) {
    		try {
				List dims = (List)((JavascriptExecutor)driver).executeScript(JS_GET_CONTENT_ENTIRE_SIZE, usePixelAspectRatio);
		    	
		    	// issue #238: check we get a non zero size
		    	if (dims.get(0).intValue() == 0 || dims.get(1).intValue() == 0) {
		    		driver.switchTo().defaultContent();
		    		dims = (List)((JavascriptExecutor)driver).executeScript(JS_GET_CONTENT_ENTIRE_SIZE, usePixelAspectRatio);
		    	}
		    	
		    	return new Dimension(dims.get(0).intValue(), dims.get(1).intValue());
		    	
    		} catch (Exception e) {
    			return driver.manage().window().getSize();
    		}
    	} else {
    		return driver.manage().window().getSize();
    	}
    }
	
	/**
	 * TODO: handle mobile app case
	 */
	public void scrollTop() {
		if (isWebTest) {
			((JavascriptExecutor) driver).executeScript("window.top.scroll(0, 0)");
		} 
	}
	
	public void scrollTo(int x, int y) {
		if (isWebTest) {
			((JavascriptExecutor) driver).executeScript(String.format("window.top.scroll(%d, %d)", x, y));
			
			// wait for scrolling end
			Point previousScrollPosition = getScrollPosition();
			int i = 0;
			boolean fixed = false;
			do {
				Point scrollPosition = getScrollPosition();
				if (scrollPosition.x == x && scrollPosition.y == y) {
					break;
				} else if (scrollPosition.x == previousScrollPosition.x && scrollPosition.y == previousScrollPosition.y) {
					if (!fixed) {
						fixed = true;
					} else {
						break;
					}
				}
				previousScrollPosition = scrollPosition;
				WaitHelper.waitForMilliSeconds(100);
				i++;
			} while (i < 10);
		} 
	}
	
	/**
	 * Return the size of top header when it is 'fixed' positionned
	 * @return
	 */
	public Long getTopFixedHeaderSize() {
		return getTopFixedHeaderSize(true);
	}
	public Long getTopFixedHeaderSize(boolean usePixelAspectRatio) {
		if (isWebTest) {
			return (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_TOP_HEADER, usePixelAspectRatio);
		} else { 
			return 0L;
		}
	}
	
	
	/**
	 * Return the size of bottom footer when it is 'fixed' positionned
	 * @return
	 */
	public Long getBottomFixedFooterSize() {
		return getBottomFixedFooterSize(true);
	}
	public Long getBottomFixedFooterSize(boolean usePixelAspectRatio) {
		if (isWebTest) {
			return (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_BOTTOM_FOOTER, usePixelAspectRatio);
		} else { 
			return 0L;
		}
	}
	
	/**
	 * Check if a modal is displayed
	 * @return
	 */
	public boolean isModalDisplayed() {
		if (isWebTest) {
			return (Boolean) ((JavascriptExecutor) driver).executeScript(JS_GET_MODAL);
		} else { 
			return false;
		}
	}
	
	/**
	 * scroll to the given element
	 * we scroll 200 px to the left of the element so that we see all of it
	 * @param element
	 */
	public void scrollToElement(WebElement element, int yOffset) {
		if (isWebTest) {
			try {
				WebElement parentScrollableElement = (WebElement) ((JavascriptExecutor) driver).executeScript(JS_SCROLL_PARENT, element, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
				Long topHeaderSize = (Long) ((JavascriptExecutor) driver).executeScript(JS_GET_TOP_HEADER);

				// try a second method (the first one is quicker but does not work when element is inside a document fragment, slot or shadow DOM
				if ((parentScrollableElement == null || "html".equalsIgnoreCase(parentScrollableElement.getTagName())) && !(driver instanceof InternetExplorerDriver)) {
					parentScrollableElement = (WebElement) ((JavascriptExecutor) driver).executeScript(JS_SCROLL_PARENT2, element, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
				}
				
				if (parentScrollableElement != null) {
					String parentTagName = parentScrollableElement.getTagName();
					
					// When the scrollable element is the document itself (html or body), use the legacy behavior scrolling the window itself
					if ("html".equalsIgnoreCase(parentTagName) || "body".equalsIgnoreCase(parentTagName)) {
						if (yOffset == Integer.MAX_VALUE) {// equivalent to HtmlElement.OPTIMAL_SCROLLING but, for grid, we do not want dependency between the 2 classes
							yOffset = (int) Math.min(-200, -topHeaderSize);
						}
						
						scrollWindowToElement(element, yOffset);
					
					// When scrollable element is a "div" or anything else, use scrollIntoView because it's the easiest way to make the element visible
					} else {
						((JavascriptExecutor) driver).executeScript("arguments[0].scrollIntoView(true);", element);
						
						// check if scrollbar of parent is at bottom. In this case, going upside (yOffset < 0), could hide searched element.
						// scrollIntoView is configured to position scrollbar to top position of the searched element.
						// in case offset is > 0, this has no impact as scrollBottom will always be < 0
						WaitHelper.waitForMilliSeconds(500); // wait a bit so that scrolling is complete
						
						// if top header is present, scroll up so that our element is not hidden behind it (scrollIntoView scrolls so that element is at the top of the view even if header masks it)
						if (topHeaderSize > 0) {// equivalent to HtmlElement.OPTIMAL_SCROLLING but, for grid, we do not want dependency between the 2 classes
							Integer scrollOffset = (int) (yOffset == Integer.MAX_VALUE ? topHeaderSize: topHeaderSize - yOffset);
							((JavascriptExecutor) driver).executeScript(
									"if((arguments[0].scrollHeight - arguments[0].scrollTop - arguments[0].clientHeight) > 0) {" +
									"   var rootElement = arguments[1] === \"safari\" ? document.body: document.documentElement;" +
									"   rootElement.scrollTop -= " + scrollOffset + ";" +
									"}", parentScrollableElement, (driver instanceof SafariDriver) ? SAFARI_BROWSER: OTHER_BROWSER);
						}
						
						WaitHelper.waitForMilliSeconds(500); // wait a bit so that scrolling is complete
					}
				} else {
					// go to default behavior
					throw new JavascriptException("No parent found");
				}
				
			} catch (Exception e) {
				// fall back to legacy behavior
				if (yOffset == Integer.MAX_VALUE) {
					yOffset = -200;
				}
				try {
					scrollWindowToElement(element, yOffset);
				} catch (Exception e1) {
					logger.info(String.format("Cannot scroll to element %s: %s", element.toString(), e1.getMessage()));
				}
			}
		}
	}
	
	private void scrollWindowToElement(WebElement element, int yOffset) {
		((JavascriptExecutor) driver).executeScript("window.top.scroll(" + Math.max(element.getLocation().x - 200, 0) + "," + Math.max(element.getLocation().y + yOffset, 0) + ")");
	}
	
	/**
	 * TODO: handle mobile app case
	 */
	@SuppressWarnings("unchecked")
	public Point getScrollPosition() {
		if (isWebTest) {
			try {
				List dims = (List)((JavascriptExecutor) driver).executeScript(JS_GET_CURRENT_SCROLL_POSITION);
				return new Point(dims.get(0).intValue(), dims.get(1).intValue());
			} catch (Exception e) {
    			return new Point(0, 0);
    		}
		} else {
			throw new WebDriverException("scroll position can only be get for web");
		}
	}
	
	/**
	 * Returns the rectangle of all screens on the system
	 * @return
	 */
	private static Rectangle getScreensRectangle() {
		GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
		
		Rectangle screenRect = new Rectangle(0, 0, 0, 0);
		for (GraphicsDevice gd : ge.getScreenDevices()) {
		    screenRect = screenRect.union(gd.getDefaultConfiguration().getBounds());
		}
		return screenRect;
	}
	private static Rectangle getMainScreenRectangle() {
		GraphicsEnvironment ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
		
		GraphicsDevice gd = ge.getDefaultScreenDevice();
		return gd.getDefaultConfiguration().getBounds();

	}
	
	/**
	 * Take screenshot of the desktop and put it in a file
	 * Do not expose this method because we need to check that we have a graphical environment. 
	 */
	private static BufferedImage captureDesktopToBuffer() {
		
		
		try {
			Rectangle screenRect = getScreensRectangle();
			return new Robot().createScreenCapture(screenRect);
		} catch (HeadlessException | AWTError | AWTException e) {
			throw new ScenarioException("Cannot capture image", e);
		}
	}
	
	private static BufferedImage captureMainDesktopScreenToBuffer() {
		
		
		try {
			Rectangle screenRect = getMainScreenRectangle();
			return new Robot().createScreenCapture(screenRect);
		} catch (HeadlessException | AWTError | AWTException e) {
			throw new ScenarioException("Cannot capture image", e);
		}
	}

	/**
	 * After quitting driver, if it fails, some pids may remain. Kill them
	 */
	@Override
	public void quit() {
		
		// get list of pids we could have to kill. Sometimes, Chrome does not close all its processes
		// so we have to know which processes to kill when driver is still active
		List pidsToKill = new ArrayList<>();
		if (browserInfo != null && driverMode == DriverMode.LOCAL) {
			pidsToKill.addAll(browserInfo.getAllBrowserSubprocessPids(driverPids));
			
			// issue #401: also get the browser process in case of attached browser (chrome only)
			if (attachExistingDriverPort != null && attachExistingDriverPort > 1000) {
				Integer browserProcess = OSUtilityFactory.getInstance().getProcessIdByListeningPort(attachExistingDriverPort);
				if (browserProcess != null) {
					pidsToKill.add(browserProcess.longValue());
				}
			}
		}
		
		Capabilities caps;
		try {
			caps = getCapabilities();
		} catch (WebDriverException e) {
			caps = new MutableCapabilities();
		}
		
		// close windows before quitting (this is the only way to close chrome attached browser when it's not started by selenium)
		try {
			List handles = new ArrayList<>(driver.getWindowHandles());
	        Collections.reverse(handles);

	        for (String handle: handles) {
				driver.switchTo().window(handle);
				driver.close();
			}
		} catch (Exception e) {
			// nothing to do
		}
		
		
		Long duration = 0L;
		try {
			duration = new Date().getTime() - (Long)internalCapabilities.getCapability(DriverUsage.START_TIME);
		} catch (Exception e) {
			// catch error
		}
		String gridHub = caps.getCapability(DriverUsage.GRID_HUB) != null ? caps.getCapability(DriverUsage.GRID_HUB).toString(): null;
		String sessionId = caps.getCapability(DriverUsage.SESSION_ID) != null ? caps.getCapability(DriverUsage.SESSION_ID).toString(): null;
		
		// store driver stats
		DriverUsage usage = new DriverUsage(gridHub, 
				(String) caps.getCapability(DriverUsage.GRID_NODE), 
				(Long) internalCapabilities.getCapability(DriverUsage.START_TIME), 
				duration, 
				sessionId, 
				caps.getBrowserName(), 
				(Long) internalCapabilities.getCapability(DriverUsage.STARTUP_DURATION), 
				(String) internalCapabilities.getCapability(DriverUsage.TEST_NAME));
		StatisticsStorage.addDriverUsage(usage);
		
		try {
			driver.quit();
		} catch (WebDriverException e) {
			caps = new MutableCapabilities();
			logger.error("Error while quitting driver: " + e.getMessage());
		} finally {
			
			// wait for browser processes to stop
			WaitHelper.waitForSeconds(2);
			
			// only kill processes in local mode
			if (!pidsToKill.isEmpty()) {
		    	for (Long pid: pidsToKill) {
		    		OSUtilityFactory.getInstance().killProcess(pid.toString(), true);
		    	}
			}
			
		}
	}
	
	/**
	 * Use copy to clipboard and copy-paste keyboard shortcut to write something on upload window
	 */
	public static void uploadFileUsingClipboard(File tempFile) {

		// Copy to clipboard
		Toolkit.getDefaultToolkit().getSystemClipboard().setContents(new StringSelection(tempFile.getAbsolutePath()), null);
		Robot robot;
		try {
			robot = new Robot();
		
			WaitHelper.waitForSeconds(1);
	
//			// Press Enter
//			robot.keyPress(KeyEvent.VK_ENTER);
//	
//			// Release Enter
//			robot.keyRelease(KeyEvent.VK_ENTER);
	
			// Press CTRL+V
			robot.keyPress(KeyEvent.VK_CONTROL);
			robot.keyPress(KeyEvent.VK_V);
	
			// Release CTRL+V
			robot.keyRelease(KeyEvent.VK_CONTROL);
			robot.keyRelease(KeyEvent.VK_V);
			WaitHelper.waitForSeconds(1);
	
			// Press Enter
			robot.keyPress(KeyEvent.VK_ENTER);
			robot.keyRelease(KeyEvent.VK_ENTER);
		} catch (AWTException e) {
			throw new ScenarioException("could not initialize robot to upload file: " + e.getMessage());
		}
	}
	
	/**
	 * Upload file typing file path directly
	 * @param tempFile
	 */
	public static void uploadFileUsingKeyboardTyping(File tempFile) {
		try {
			Keyboard keyboard = new Keyboard();
			Robot robot = keyboard.getRobot();
			
			WaitHelper.waitForSeconds(1);
			
//			// Press Enter
//			robot.keyPress(KeyEvent.VK_ENTER);
//	
//			// Release Enter
//			robot.keyRelease(KeyEvent.VK_ENTER);
			
			keyboard.typeKeys(tempFile.getAbsolutePath());
			
			WaitHelper.waitForSeconds(1);
			
			// Press Enter
			robot.keyPress(KeyEvent.VK_ENTER);
			robot.keyRelease(KeyEvent.VK_ENTER);
			
		} catch (AWTException e) {
			throw new ScenarioException("could not initialize robot to upload file typing keys: " + e.getMessage());
		}
	}
	
	public static void uploadFile(String fileName, String base64Content, DriverMode driverMode, SeleniumGridConnector gridConnector) throws IOException {
		
		if (driverMode == DriverMode.LOCAL) {
			byte[] byteArray = base64Content.getBytes();
	        File tempFile = new File("tmp/" + fileName);
	        byte[] decodeBuffer = Base64.decodeBase64(byteArray);
	        FileUtils.writeByteArrayToFile(tempFile, decodeBuffer);
	
	        try {
	        	uploadFileUsingClipboard(tempFile);
	        } catch (IllegalStateException e) {
	        	uploadFileUsingKeyboardTyping(tempFile);
	        }
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.uploadFileToBrowser(fileName, base64Content);
		} else {
			throw new ScenarioException("driver supports uploadFile only in local and grid mode");
		}
	}
	
	/**
	 * move mouse taking screen placing into account. Sometimes, coordinates may be negative if first screen has an other screen on the left 
	 * @param robot
	 */
	private static void moveMouse(Robot robot, int x, int y) {
		Rectangle screenRectangle = getScreensRectangle();
		robot.mouseMove(x + screenRectangle.x, y + screenRectangle.y);
	}
	private static void moveMouseMainScreen(Robot robot, int x, int y) {
		Rectangle screenRectangle = getMainScreenRectangle();
		robot.mouseMove(x + screenRectangle.x, y + screenRectangle.y);
	}
	
	/**
	 * Left clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void leftClicOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		leftClicOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void leftClicOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("leftClicOnDesktopAt: problem using Robot: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.leftClic(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports leftClicOnDesktopAt only in local and grid mode");
		}
	}
	
	/**
	 * Get mouse coordinates
	 * @param x
	 * @param y
	 */
	public static java.awt.Point getMouseCoordinates(DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				PointerInfo pointerInfo = MouseInfo.getPointerInfo();
				return pointerInfo.getLocation();
			} catch (HeadlessException e) {
				throw new ScenarioException("getMouseCoordinades: problem using Robot: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			return gridConnector.getMouseCoordinates();
		} else {
			throw new ScenarioException("driver supports getMouseCoordinades only in local and grid mode");
		}
	}
	
	/**
	 * Left clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void doubleClickOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		doubleClickOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void doubleClickOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
				robot.delay(10);
				robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("doubleClickOnDesktopAt: problem using Robot: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.doubleClick(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports doubleClickOnDesktopAt only in local and grid mode");
		}
	}
	
	/**
	 * right clic at coordinates on desktop. Coordinates are from screen point of view
	 * @param x
	 * @param y
	 */
	public static void rightClicOnDesktopAt(int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		rightClicOnDesktopAt(false, x, y, driverMode, gridConnector);
	}
	public static void rightClicOnDesktopAt(boolean onlyMainScreen, int x, int y, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		
		if (driverMode == DriverMode.LOCAL) {
			try {
				Robot robot = new Robot();
				if (onlyMainScreen) {
					moveMouseMainScreen(robot, x, y);
				} else {
					moveMouse(robot, x, y);
				}
				robot.mousePress(InputEvent.BUTTON3_DOWN_MASK);
				robot.mouseRelease(InputEvent.BUTTON3_DOWN_MASK);
			} catch (AWTException e) {
				throw new ScenarioException("rightClicOnDesktopAt: problem using Robot: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.rightClic(onlyMainScreen, x, y);
		} else {
			throw new ScenarioException("driver supports sendKeysToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * write text to desktop.
	 * @param textToWrite	text to write
	 * @return
	 */
	public static void writeToDesktop(String textToWrite, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
	
			try {
				Keyboard keyboard = new Keyboard();
				keyboard.typeKeys(textToWrite);
			} catch (AWTException e) {
				throw new ScenarioException("writeToDesktop: could not initialize robot to type keys: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.writeText(textToWrite);
		} else {
			throw new ScenarioException("driver supports writeToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * send keys to desktop
	 * This is useful for typing special keys like ENTER
	 * @param keys	List of KeyEvent 
	 */
	public static void sendKeysToDesktop(List keyCodes, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
			try {
				Robot robot = new Robot();
				
				WaitHelper.waitForSeconds(1);
				
				for (Integer key: keyCodes) {
					robot.keyPress(key);
					robot.keyRelease(key);
				}
			} catch (AWTException e) {
				throw new ScenarioException("could not initialize robot to type keys: " + e.getMessage());
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.sendKeysWithKeyboard(keyCodes);
		} else {
			throw new ScenarioException("driver supports sendKeysToDesktop only in local and grid mode");
		}
	}
	
	/**
	 * Returns a Base64 string of the desktop
	 * @param driverMode
	 * @param gridConnector
	 * @return
	 */
	public static String captureDesktopToBase64String(DriverMode driverMode, SeleniumGridConnector gridConnector) {
		return captureDesktopToBase64String(false, driverMode, gridConnector);
	}
	
	/**
	 * Returns a Base64 string of the desktop
	 * @param onlyMainScreen	if true, only capture the default (or main) screen
	 * @param driverMode
	 * @param gridConnector
	 * @return
	 */
	public static String captureDesktopToBase64String(boolean onlyMainScreen, DriverMode driverMode, SeleniumGridConnector gridConnector) {
		if (driverMode == DriverMode.LOCAL) {
			BufferedImage bi;
			if (onlyMainScreen) {
				bi = captureMainDesktopScreenToBuffer();
			} else {
				bi = captureDesktopToBuffer();
			}
			
			ByteArrayOutputStream os = new ByteArrayOutputStream();
			OutputStream b64 = new Base64OutputStream(os);
			try {
				ImageIO.write(bi, "png", b64);
				return os.toString("UTF-8");
			} catch (IOException e) {
				return "";
			}
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			return gridConnector.captureDesktopToBuffer();
		} else {
			throw new ScenarioException("driver supports captureDesktopToBase64String only in local and grid mode");
		}
	}
	
	/**
	 * Start video capture using VideoRecorder class
	 * @param driverMode
	 * @param gridConnector
	 * @param videoName		name of the video to record so that it's unique. Only used locally. In remote, grid sessionId is used
	 */
	public static VideoRecorder startVideoCapture(DriverMode driverMode, SeleniumGridConnector gridConnector, File videoFolder, String videoName) {
		if (driverMode == DriverMode.LOCAL) {
			try {
				VideoRecorder recorder = new VideoRecorder(videoFolder, videoName);
				recorder.start();
				return recorder;
			} catch (HeadlessException e) {
				throw new ScenarioException("could not initialize video capture with headless robot: " + e.getMessage());
			}
			
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.startVideoCapture();
			return new VideoRecorder(videoFolder, videoName, false);
		} else {
			throw new ScenarioException("driver supports startVideoCapture only in local and grid mode");
		}
	}
	
	/**
	 * Stop video capture using VideoRecorder class
	 * @param driverMode
	 * @param gridConnector
	 * @throws IOException 
	 */
	public static File stopVideoCapture(DriverMode driverMode, SeleniumGridConnector gridConnector, VideoRecorder recorder) throws IOException {
		if (driverMode == DriverMode.LOCAL && recorder != null) {
			return recorder.stop();
		} else if (driverMode == DriverMode.GRID && gridConnector != null && recorder != null) {
			return gridConnector.stopVideoCapture(Paths.get(recorder.getFolderPath().getAbsolutePath(), recorder.getFileName()).toString());
			
		} else {
			throw new ScenarioException("driver supports stopVideoCapture only in local and grid mode");
		}
	}
	
	/**
	 * Intercept specific scripts to do some non selenium actions
	 * 
	 * @deprecated: should be removed (kept here for compatibility with old robots)
	 */
	@Deprecated
	@Override
	public Object executeScript(String script, Object... args) {
		
		// to we know this command ?
		if (driverMode == DriverMode.LOCAL && NON_JS_UPLOAD_FILE_THROUGH_POPUP.equals(script)) {
			if (args.length != 2) {
				throw new DriverExceptions("Upload feature through executeScript needs 2 string arguments (file name, base64 content)");
			}
			try {
				uploadFile((String)args[0], (String)args[1], driverMode, gridConnector);
				return null; 
			} catch (IOException e) {
				return null;
			}
			
		} else if (driverMode == DriverMode.LOCAL && NON_JS_CAPTURE_DESKTOP.equals(script)) {
			return captureDesktopToBase64String(driverMode, gridConnector);
		} else {
			return super.executeScript(script, args);
		}
	}
	
	public static void displayStepOnScreen(String textToWrite, DriverMode driverMode, SeleniumGridConnector gridConnector, VideoRecorder recorder) {
		if (driverMode == DriverMode.LOCAL) {
			
			recorder.displayRunningStep(textToWrite);
		} else if (driverMode == DriverMode.GRID && gridConnector != null) {
			gridConnector.displayRunningStep(textToWrite);
		} else {
			throw new ScenarioException("driver supports writeToDesktop only in local and grid mode");
		}
	}
	

    public List getDriverPids() {
		return driverPids;
	}

	public BrowserInfo getBrowserInfo() {
		return browserInfo;
	}

	@Override
    public Capabilities getCapabilities() {
		try {
			return ((HasCapabilities)driver).getCapabilities();
		} catch (ClassCastException e) {
			return new MutableCapabilities();
		}
    }

	public BrowserMobProxy getMobProxy() {
		return mobProxy;
	}

	// NEOLOAD //
	public NLWebDriver getNeoloadDriver() {
		return neoloadDriver;
	}

	public MutableCapabilities getInternalCapabilities() {
		return internalCapabilities;
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy