/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ "use strict"; ChromeUtils.import("resource://gre/modules/Preferences.jsm"); ChromeUtils.import("resource://gre/modules/XPCOMUtils.jsm"); ChromeUtils.import("chrome://marionette/content/accessibility.js"); ChromeUtils.import("chrome://marionette/content/atom.js"); ChromeUtils.import("chrome://marionette/content/element.js"); const { ElementClickInterceptedError, ElementNotInteractableError, InvalidArgumentError, InvalidElementStateError, } = ChromeUtils.import("chrome://marionette/content/error.js", {}); ChromeUtils.import("chrome://marionette/content/event.js"); const {pprint} = ChromeUtils.import("chrome://marionette/content/format.js", {}); const {TimedPromise} = ChromeUtils.import("chrome://marionette/content/sync.js", {}); XPCOMUtils.defineLazyGlobalGetters(this, ["File"]); this.EXPORTED_SYMBOLS = ["interaction"]; /** XUL elements that support disabled attribute. */ const DISABLED_ATTRIBUTE_SUPPORTED_XUL = new Set([ "ARROWSCROLLBOX", "BUTTON", "CHECKBOX", "COLORPICKER", "COMMAND", "DESCRIPTION", "KEY", "KEYSET", "LABEL", "LISTBOX", "LISTCELL", "LISTHEAD", "LISTHEADER", "LISTITEM", "MENU", "MENUITEM", "MENULIST", "MENUSEPARATOR", "PREFERENCE", "RADIO", "RADIOGROUP", "RICHLISTBOX", "RICHLISTITEM", "SCALE", "TAB", "TABS", "TEXTBOX", "TOOLBARBUTTON", "TREE", ]); /** * Common form controls that user can change the value property * interactively. */ const COMMON_FORM_CONTROLS = new Set([ "input", "textarea", "select", ]); /** * Input elements that do not fire input and change * events when value property changes. */ const INPUT_TYPES_NO_EVENT = new Set([ "checkbox", "radio", "file", "hidden", "image", "reset", "button", "submit", ]); /** @namespace */ this.interaction = {}; /** * Interact with an element by clicking it. * * The element is scrolled into view before visibility- or interactability * checks are performed. * * Selenium-style visibility checks will be performed * if specCompat is false (default). Otherwise * pointer-interactability checks will be performed. If either of these * fail an {@link ElementNotInteractableError} is thrown. * * If strict is enabled (defaults to disabled), further * accessibility checks will be performed, and these may result in an * {@link ElementNotAccessibleError} being returned. * * When el is not enabled, an {@link InvalidElementStateError} * is returned. * * @param {(DOMElement|XULElement)} el * Element to click. * @param {boolean=} [strict=false] strict * Enforce strict accessibility tests. * @param {boolean=} [specCompat=false] specCompat * Use WebDriver specification compatible interactability definition. * * @throws {ElementNotInteractableError} * If either Selenium-style visibility check or * pointer-interactability check fails. * @throws {ElementClickInterceptedError} * If el is obscured by another element and a click would * not hit, in specCompat mode. * @throws {ElementNotAccessibleError} * If strict is true and element is not accessible. * @throws {InvalidElementStateError} * If el is not enabled. */ interaction.clickElement = async function( el, strict = false, specCompat = false) { const a11y = accessibility.get(strict); if (element.isXULElement(el)) { await chromeClick(el, a11y); } else if (specCompat) { await webdriverClickElement(el, a11y); } else { await seleniumClickElement(el, a11y); } }; async function webdriverClickElement(el, a11y) { const win = getWindow(el); // step 3 if (el.localName == "input" && el.type == "file") { throw new InvalidArgumentError( "Cannot click elements"); } let containerEl = element.getContainer(el); // step 4 if (!element.isInView(containerEl)) { element.scrollIntoView(containerEl); } // step 5 // TODO(ato): wait for containerEl to be in view // step 6 // if we cannot bring the container element into the viewport // there is no point in checking if it is pointer-interactable if (!element.isInView(containerEl)) { throw new ElementNotInteractableError( pprint`Element ${el} could not be scrolled into view`); } // step 7 let rects = containerEl.getClientRects(); let clickPoint = element.getInViewCentrePoint(rects[0], win); if (element.isObscured(containerEl)) { throw new ElementClickInterceptedError(containerEl, clickPoint); } let acc = await a11y.getAccessible(el, true); a11y.assertVisible(acc, el, true); a11y.assertEnabled(acc, el, true); a11y.assertActionable(acc, el); // step 8 if (el.localName == "option") { interaction.selectOption(el); } else { // step 9 let clicked = interaction.flushEventLoop(containerEl); event.synthesizeMouseAtPoint(clickPoint.x, clickPoint.y, {}, win); await clicked; } // step 10 // if the click causes navigation, the post-navigation checks are // handled by the load listener in listener.js } async function chromeClick(el, a11y) { if (!atom.isElementEnabled(el)) { throw new InvalidElementStateError("Element is not enabled"); } let acc = await a11y.getAccessible(el, true); a11y.assertVisible(acc, el, true); a11y.assertEnabled(acc, el, true); a11y.assertActionable(acc, el); if (el.localName == "option") { interaction.selectOption(el); } else { el.click(); } } async function seleniumClickElement(el, a11y) { let win = getWindow(el); let visibilityCheckEl = el; if (el.localName == "option") { visibilityCheckEl = element.getContainer(el); } if (!element.isVisible(visibilityCheckEl)) { throw new ElementNotInteractableError(); } if (!atom.isElementEnabled(el)) { throw new InvalidElementStateError("Element is not enabled"); } let acc = await a11y.getAccessible(el, true); a11y.assertVisible(acc, el, true); a11y.assertEnabled(acc, el, true); a11y.assertActionable(acc, el); if (el.localName == "option") { interaction.selectOption(el); } else { let rects = el.getClientRects(); let centre = element.getInViewCentrePoint(rects[0], win); let opts = {}; event.synthesizeMouseAtPoint(centre.x, centre.y, opts, win); } } /** * Select <option> element in a <select> * list. * * Because the dropdown list of select elements are implemented using * native widget technology, our trusted synthesised events are not able * to reach them. Dropdowns are instead handled mimicking DOM events, * which for obvious reasons is not ideal, but at the current point in * time considered to be good enough. * * @param {HTMLOptionElement} option * Option element to select. * * @throws {TypeError} * If el is a XUL element or not an <option> * element. * @throws {Error} * If unable to find el's parent <select> * element. */ interaction.selectOption = function(el) { if (element.isXULElement(el)) { throw new TypeError("XUL dropdowns not supported"); } if (el.localName != "option") { throw new TypeError(pprint`Expected