net.serenitybdd.core.pages.WebElementFacadeImpl Maven / Gradle / Ivy
package net.serenitybdd.core.pages;
import com.google.common.base.Splitter;
import net.serenitybdd.core.Serenity;
import net.serenitybdd.core.SystemTimeouts;
import net.serenitybdd.model.time.InternalSystemClock;
import net.thucydides.model.ThucydidesSystemProperty;
import net.thucydides.core.annotations.locators.MethodTiming;
import net.thucydides.core.annotations.locators.WithConfigurableTimeout;
import net.thucydides.model.environment.SystemEnvironmentVariables;
import net.thucydides.core.steps.StepEventBus;
import net.thucydides.model.util.EnvironmentVariables;
import net.thucydides.core.webdriver.ConfigurableTimeouts;
import net.thucydides.core.webdriver.TemporalUnitConverter;
import net.thucydides.core.webdriver.WebDriverFacade;
import net.thucydides.core.webdriver.exceptions.*;
import net.thucydides.core.webdriver.javascript.JavascriptExecutorFacade;
import net.thucydides.core.webdriver.stubs.WebElementFacadeStub;
import org.apache.commons.lang3.StringUtils;
import org.openqa.selenium.NoSuchElementException;
import org.openqa.selenium.StaleElementReferenceException;
import org.openqa.selenium.*;
import org.openqa.selenium.interactions.Actions;
import org.openqa.selenium.interactions.Coordinates;
import org.openqa.selenium.interactions.Locatable;
import org.openqa.selenium.support.pagefactory.ElementLocator;
import org.openqa.selenium.support.ui.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.time.Clock;
import java.time.Duration;
import java.time.temporal.TemporalUnit;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import static net.serenitybdd.core.pages.ParameterisedLocator.withArguments;
import static net.serenitybdd.core.pages.WebElementExpectations.*;
import static net.serenitybdd.core.selectors.Selectors.isXPath;
import static net.thucydides.model.ThucydidesSystemProperty.LEGACY_WAIT_FOR_TEXT;
import static org.openqa.selenium.support.ui.ExpectedConditions.elementToBeClickable;
/**
* A proxy class for a web element, providing some more methods.
*/
public class WebElementFacadeImpl implements WebElementFacade, net.thucydides.core.pages.WebElementFacade {
public static final int NUMBER_OF_TRIES_FOR_UNREADY_ELEMENTS = 12;
private final WebElement webElement;
private final WebDriver driver;
private final long implicitTimeoutInMilliseconds;
private final long waitForTimeoutInMilliseconds;
private static final int WAIT_FOR_ELEMENT_PAUSE_LENGTH = 100;
private final Sleeper sleeper;
private final Clock webdriverClock;
private final By bySelector;
private final JavascriptExecutorFacade javascriptExecutorFacade;
private final InternalSystemClock clock = new InternalSystemClock();
private final EnvironmentVariables environmentVariables;
private String foundBy;
private final ElementLocator locator;
private WebElement resolvedELement;
private static final Logger LOGGER = LoggerFactory.getLogger(WebElementFacadeImpl.class);
public WebElementFacadeImpl(final WebDriver driver, final ElementLocator locator, final WebElement webElement, final long implicitTimeoutInMilliseconds, final long waitForTimeoutInMilliseconds, final By bySelector) {
this.webElement = webElement;
this.driver = driver;
this.locator = locator;
this.bySelector = bySelector;
this.webdriverClock = Clock.systemDefaultZone();
this.sleeper = Sleeper.SYSTEM_SLEEPER;
this.javascriptExecutorFacade = new JavascriptExecutorFacade(driver);
this.environmentVariables = SystemEnvironmentVariables.currentEnvironmentVariables();
this.implicitTimeoutInMilliseconds = implicitTimeoutInMilliseconds;
this.waitForTimeoutInMilliseconds = (waitForTimeoutInMilliseconds >= 0) ? waitForTimeoutInMilliseconds : defaultWaitForTimeout();
}
public WebElementFacadeImpl(final WebDriver driver, final WebElement webElement, final long implicitTimeoutInMilliseconds, final long waitForTimeoutInMilliseconds, final By bySelector) {
this.webElement = webElement;
this.driver = driver;
this.locator = null;
this.bySelector = bySelector;
this.webdriverClock = Clock.systemDefaultZone();
this.sleeper = Sleeper.SYSTEM_SLEEPER;
this.javascriptExecutorFacade = new JavascriptExecutorFacade(driver);
this.environmentVariables = SystemEnvironmentVariables.currentEnvironmentVariables();
this.implicitTimeoutInMilliseconds = implicitTimeoutInMilliseconds;
this.waitForTimeoutInMilliseconds = (waitForTimeoutInMilliseconds >= 0) ? waitForTimeoutInMilliseconds : defaultWaitForTimeout();
}
public WebElementFacadeImpl(final WebDriver driver, final ElementLocator locator, final WebElement webElement, final long implicitTimeoutInMilliseconds, final long waitForTimeoutInMilliseconds) {
this.webElement = webElement;
this.driver = driver;
this.locator = locator;
this.bySelector = null;
this.webdriverClock = Clock.systemDefaultZone();
this.sleeper = Sleeper.SYSTEM_SLEEPER;
this.javascriptExecutorFacade = new JavascriptExecutorFacade(driver);
this.environmentVariables = SystemEnvironmentVariables.currentEnvironmentVariables();
this.implicitTimeoutInMilliseconds = implicitTimeoutInMilliseconds;
this.waitForTimeoutInMilliseconds = (waitForTimeoutInMilliseconds >= 0) ? waitForTimeoutInMilliseconds : defaultWaitForTimeout();
}
public WebElementFacadeImpl(final WebDriver driver, final ElementLocator locator, final WebElement webElement, final long implicitTimeoutInMilliseconds) {
this(driver, locator, webElement, implicitTimeoutInMilliseconds, implicitTimeoutInMilliseconds);
}
public WebElementFacadeImpl(WebDriver driver, ElementLocator locator, WebElement webElement, WebElement resolvedELement, By bySelector, long timeoutInMilliseconds, long waitForTimeoutInMilliseconds) {
this.webElement = webElement;
this.resolvedELement = resolvedELement;
this.driver = driver;
this.locator = locator;
this.bySelector = bySelector;
this.webdriverClock = Clock.systemDefaultZone();
this.sleeper = Sleeper.SYSTEM_SLEEPER;
this.javascriptExecutorFacade = new JavascriptExecutorFacade(driver);
this.environmentVariables = SystemEnvironmentVariables.currentEnvironmentVariables();
this.implicitTimeoutInMilliseconds = timeoutInMilliseconds;
this.waitForTimeoutInMilliseconds = (waitForTimeoutInMilliseconds >= 0) ? waitForTimeoutInMilliseconds : defaultWaitForTimeout();
}
private long defaultWaitForTimeout() {
return ThucydidesSystemProperty.WEBDRIVER_WAIT_FOR_TIMEOUT.integerFrom(environmentVariables, (int) DefaultTimeouts.DEFAULT_WAIT_FOR_TIMEOUT.toMillis());
}
private WebElementFacadeImpl copy() {
return BuildWebElementFacade.from(driver, locator, webElement, implicitTimeoutInMilliseconds, waitForTimeoutInMilliseconds);
}
public WebElementFacadeImpl(final WebDriver driver, final ElementLocator locator, final long implicitTimeoutInMilliseconds) {
this(driver, locator, null, implicitTimeoutInMilliseconds, implicitTimeoutInMilliseconds);
}
public WebElementFacadeImpl(final WebDriver driver, final ElementLocator locator, final long implicitTimeoutInMilliseconds, final long waitForTimeoutInMilliseconds) {
this(driver, locator, null, implicitTimeoutInMilliseconds, waitForTimeoutInMilliseconds);
}
public static T wrapWebElement(final WebDriver driver, final WebElement element, final long timeoutInMilliseconds, final long waitForTimeoutInMilliseconds) {
return BuildWebElementFacade.from(driver, element, timeoutInMilliseconds, waitForTimeoutInMilliseconds);
}
public static T wrapWebElement(final WebDriver driver, final WebElement element, final long timeoutInMilliseconds, final long waitForTimeoutInMilliseconds, final String foundBy) {
return BuildWebElementFacade.from(driver, element, timeoutInMilliseconds, waitForTimeoutInMilliseconds, foundBy);
}
public static T wrapWebElement(WebDriver driver, WebElement resolvedELement, WebElement element, By bySelector, ElementLocator locator, long timeoutInMilliseconds, long waitForTimeoutInMilliseconds, String foundBy) {
return BuildWebElementFacade.from(driver, resolvedELement, element, bySelector, locator, timeoutInMilliseconds, waitForTimeoutInMilliseconds, foundBy);
}
public static T wrapWebElement(final WebDriver driver, final By bySelector, final long timeoutInMilliseconds, final long waitForTimeoutInMilliseconds, final String foundBy) {
return BuildWebElementFacade.from(driver, bySelector, timeoutInMilliseconds, waitForTimeoutInMilliseconds, foundBy);
}
public static WebElementFacade wrapWebElement(final WebDriver driver, final WebElement element, final long timeout) {
return BuildWebElementFacade.from(driver, element, timeout);
}
public static WebElementFacade wrapWebElement(final WebDriver driver, final WebElement element) {
return wrapWebElement(driver, element, SystemTimeouts.forTheCurrentTest().getImplicitTimeout());
}
private WebElementResolver getElementResolver() {
if (webElement != null) {
return WebElementResolver.forWebElement(webElement);
}
if (bySelector != null) {
return WebElementResolver.by(bySelector);
}
return WebElementResolver.byLocator(locator).withImplicitTimeout(implicitTimeoutInMilliseconds);
}
public WebElement getElement() {
if (driverIsDisabled()) {
return new WebElementFacadeStub();
}
return getResolvedElement();
}
private WebElement getResolvedElement() {
if (resolvedELement == null) {
resolvedELement = getElementResolver().resolveForDriver(driver);
}
return resolvedELement;
}
protected JavascriptExecutorFacade getJavascriptExecutorFacade() {
return javascriptExecutorFacade;
}
protected InternalSystemClock getClock() {
return clock;
}
@Override
public WebElementFacade then(String xpathOrCssSelector, Object... arguments) {
return findBy(xpathOrCssSelector, arguments);
}
@Override
public WebElementFacade thenFind(String xpathOrCssSelector, Object... arguments) {
return findBy(xpathOrCssSelector, arguments);
}
@Override
public WebElementFacade thenFind(String xpathOrCssSelector) {
return findBy(xpathOrCssSelector);
}
@Override
public WebElementFacade then(String xpathOrCssSelector) {
return findBy(xpathOrCssSelector);
}
@Override
public T findBy(String xpathOrCssSelector) {
logIfVerbose("findBy " + xpathOrCssSelector);
WebElement nestedElement;
if (driverIsDisabled()) {
nestedElement = this;
} else if (isXPath(xpathOrCssSelector)) {
nestedElement = getElement().findElement((By.xpath(xpathOrCssSelector)));
} else {
nestedElement = getElement().findElement((By.cssSelector(xpathOrCssSelector)));
}
return wrapWebElement(driver, nestedElement, timeoutInMilliseconds(), waitForTimeoutInMilliseconds, "element located by " + xpathOrCssSelector);
}
public T findBy(String xpathOrCssSelector, Object... arguments) {
return findBy(withArguments(xpathOrCssSelector, arguments));
}
public long timeoutInMilliseconds() {
if (driver instanceof WebDriverFacade) {
return ((WebDriverFacade) driver).getCurrentImplicitTimeout().toMillis();
} else {
return implicitTimeoutInMilliseconds;
}
}
@Override
public ListOfWebElementFacades thenFindAll(String xpathOrCssSelector) {
logIfVerbose("findAll " + xpathOrCssSelector);
if (driverIsDisabled()) {
return new ListOfWebElementFacades(new ArrayList<>());
}
List nestedElements;
if (isXPath(xpathOrCssSelector)) {
nestedElements = findElements((By.xpath(xpathOrCssSelector)));
} else {
nestedElements = findElements((By.cssSelector(xpathOrCssSelector)));
}
return webElementFacadesFrom(nestedElements);
}
public ListOfWebElementFacades thenFindAll(String xpathOrCssSelector, Object... arguments) {
return thenFindAll(withArguments(xpathOrCssSelector, arguments));
}
private ListOfWebElementFacades webElementFacadesFrom(List nestedElements) {
List results = new ArrayList<>();
for (WebElement element : nestedElements) {
results.add(wrapWebElement(driver, element, timeoutInMilliseconds(), waitForTimeoutInMilliseconds, element.toString()));
}
return new ListOfWebElementFacades(results);
}
@Override
public WebElementFacade findBy(By selector) {
logIfVerbose("findBy " + selector);
if (driverIsDisabled()) {
return this;
}
WebElement nestedElement = getElement().findElement(selector);
return wrapWebElement(driver, nestedElement, timeoutInMilliseconds(), waitForTimeoutInMilliseconds, "element located by " + selector.toString());
}
@Override
public WebElementFacade find(By bySelector) {
return findBy(bySelector);
}
@Override
public WebElementFacade then(By bySelector) {
return findBy(bySelector);
}
@Override
public String getAttribute(String name) {
return getElement().getAttribute(name);
}
@Override
public ListOfWebElementFacades thenFindAll(By... selectors) {
logIfVerbose("findAll " + selectors);
if (driverIsDisabled()) {
return new ListOfWebElementFacades(new ArrayList<>());
}
List nestedElements = new ArrayList<>();
for (By selector : selectors) {
nestedElements = findElements(selector);
if (!nestedElements.isEmpty()) {
break;
}
}
return webElementFacadesFrom(nestedElements);
}
@Override
public long getImplicitTimeoutInMilliseconds() {
return implicitTimeoutInMilliseconds;
}
public Duration getImplicitTimeout() {
return ((ConfigurableTimeouts) driver).getCurrentImplicitTimeout();
}
@Override
public WebElementFacade withTimeoutOf(int timeout, TimeUnit unit) {
return withTimeoutOf(timeout, TemporalUnitConverter.fromTimeUnit(unit));
}
public WebElementFacade withTimeoutOf(int timeout, TemporalUnit unit) {
return withTimeoutOf(Duration.of(timeout, unit));
}
public WebElementFacade withTimeoutOf(Duration duration) {
return wrapWebElement(driver, resolvedELement, webElement, bySelector, locator, duration.toMillis(), duration.toMillis(), foundBy);
}
/**
* Is this web element present and visible on the screen
* This method will not throw an exception if the element is not on the screen at all.
* If the element is not visible, the method will wait a bit to see if it appears later on.
*/
@Override
public boolean isVisible() {
if (driverIsDisabled()) {
return false;
}
try {
WebElement element = getElement();
if (element == null) {
return false;
}
if (shouldWaitForResult()) {
return waitForCondition().until(ExpectedConditions.visibilityOf(element)).isDisplayed();
} else {
return element.isDisplayed();
}
} catch (ElementNotInteractableException | NoSuchElementException | StaleElementReferenceException |
TimeoutException e) {
return false;
}
}
private boolean shouldWaitForResult() {
return !MethodTiming.forThisThread().isInQuickMethod();
}
/**
* Convenience method to chain method calls more fluently.
*/
@Override
public WebElementFacade and() {
return this;
}
/**
* Convenience method to chain method calls more fluently.
*/
@Override
public WebElementFacade then() {
return this;
}
/**
* Is this web element present and visible on the screen
* This method will not throw an exception if the element is not on the screen at all.
* The method will fail immediately if the element is not visible on the screen.
* There is a little black magic going on here - the web element class will detect if it is being called
* by a method called "isCurrently*" and, if so, fail immediately without waiting as it would normally do.
*/
@Override
public boolean isCurrentlyVisible() {
return isVisible();
}
@Override
public boolean isCurrentlyEnabled() {
if (driverIsDisabled()) {
return false;
}
try {
return (getElement() != null) && getElement().isEnabled();
} catch (NoSuchElementException | StaleElementReferenceException e) {
return false;
}
}
/**
* Checks whether a web element is visible.
* Throws an AssertionError if the element is not rendered.
*/
@Override
public WebElementState shouldBeVisible() {
if (!isVisible()) {
failWithMessage("Element should be visible");
}
return this;
}
/**
* Checks whether a web element is visible.
* Throws an AssertionError if the element is not rendered.
*/
@Override
public WebElementState shouldBeCurrentlyVisible() {
if (!isCurrentlyVisible()) {
failWithMessage("Element should be visible");
}
return this;
}
/**
* Checks whether a web element is not visible.
* Throws an AssertionError if the element is not rendered.
*/
@Override
public WebElementState shouldNotBeVisible() {
if (isCurrentlyVisible()) {
failWithMessage("Element should not be visible");
}
return this;
}
/**
* Checks whether a web element is not visible straight away.
* Throws an AssertionError if the element is not rendered.
*/
@Override
public WebElementState shouldNotBeCurrentlyVisible() {
if (isCurrentlyVisible()) {
failWithMessage("Element should not be visible");
}
return this;
}
/**
* Does this element currently have the focus.
*/
@Override
public boolean hasFocus() {
if (driverIsDisabled()) {
return false;
}
JavascriptExecutorFacade js = new JavascriptExecutorFacade(driver);
WebElement activeElement = (WebElement) js.executeScript("return window.document.activeElement");
return getElement().equals(activeElement);
}
/**
* Does this element contain a given text?
*/
@Override
public boolean containsText(final String value) {
if (driverIsDisabled()) {
return false;
}
WebElement element = getElement();
return ((element != null) && (element.getText().contains(value)));
}
@Override
public boolean containsValue(String value) {
if (driverIsDisabled()) {
return false;
}
WebElement element = getElement();
return ((element != null) && (element.getAttribute("value").contains(value)));
}
/**
* Does this element exactly match given text?
*/
@Override
public boolean containsOnlyText(final String value) {
if (driverIsDisabled()) {
return false;
}
WebElement element = getElement();
if (element == null) return false;
String text = element.getText();
if (text.isEmpty()) {
// https://github.com/serenity-bdd/serenity-core/issues/2134
// maybe it's an input element?
text = element.getAttribute("value");
}
return value.equals(text);
}
/**
* Does this dropdown contain the specified value.
*/
@Override
public boolean containsSelectOption(final String value) {
if (driverIsDisabled()) {
return false;
}
return getSelectOptions().contains(value);
}
@Override
public List getSelectOptions() {
if (driverIsDisabled()) {
return new ArrayList<>();
}
if (getElement() != null) {
return findElements(By.tagName("option")).stream().map(WebElement::getText).collect(Collectors.toList());
}
return Collections.emptyList();
}
@Override
public List getSelectOptionValues() {
if (driverIsDisabled()) {
return new ArrayList<>();
}
if (getElement() != null) {
return findElements(By.tagName("option")).stream().map(elt -> elt.getAttribute("value")).collect(Collectors.toList());
}
return Collections.emptyList();
}
public ElementLocator getLocator() {
if ((locator instanceof WithConfigurableTimeout) && (driver instanceof ConfigurableTimeouts)) {
((WithConfigurableTimeout) locator).setTimeOutInSeconds((int) getLocatorTimeout());
}
return locator;
}
private long getLocatorTimeout() {
if (StepEventBus.getParallelEventBus().webdriverCallsAreSuspended() || (MethodTiming.forThisThread().isInQuickMethod())) {
return 0;
} else {
return TimeUnit.SECONDS.convert(implicitTimeoutInMilliseconds, TimeUnit.MILLISECONDS);
}
}
@Override
public void setImplicitTimeout(Duration implicitTimeout) {
if (driverIsDisabled()) {
return;
}
if (driver instanceof ConfigurableTimeouts) {
((ConfigurableTimeouts) driver).setImplicitTimeout(implicitTimeout);
}
}
@Override
public Duration getCurrentImplicitTimeout() {
if (driverIsDisabled()) {
return Duration.ofSeconds(0);
}
if (driver instanceof ConfigurableTimeouts) {
return ((ConfigurableTimeouts) driver).getCurrentImplicitTimeout();
}
return DefaultTimeouts.DEFAULT_IMPLICIT_WAIT_TIMEOUT;
}
@Override
public Duration resetTimeouts() {
if (driverIsDisabled()) {
return Duration.ofSeconds(0);
}
if (driver instanceof ConfigurableTimeouts) {
return ((ConfigurableTimeouts) driver).resetTimeouts();
}
return DefaultTimeouts.DEFAULT_IMPLICIT_WAIT_TIMEOUT;
}
public String getFoundBy() {
return foundBy;
}
/**
* Check that an element contains a text value
*
* @param textValue
*/
@Override
public WebElementState shouldContainText(String textValue) {
if (!containsText(textValue)) {
String errorMessage = String.format("The text '%s' was not found in the web element. Element text '%s'.", textValue, getElement().getText());
failWithMessage(errorMessage);
}
return this;
}
/**
* Check that an element exactly matches a text value
*
* @param textValue
*/
@Override
public WebElementState shouldContainOnlyText(String textValue) {
if (!containsOnlyText(textValue)) {
String errorMessage = String.format("The text '%s' does not match the elements text '%s'.", textValue, getElement().getText());
failWithMessage(errorMessage);
}
return this;
}
@Override
public WebElementState shouldContainSelectedOption(String textValue) {
if (!containsSelectOption(textValue)) {
String errorMessage = String.format("The list element '%s' was not found in the web element", textValue);
failWithMessage(errorMessage);
}
return this;
}
/**
* Check that an element does not contain a text value
*
* @param textValue
*/
@Override
public WebElementState shouldNotContainText(String textValue) {
if (containsText(textValue)) {
String errorMessage = String.format("The text '%s' was found in the web element when it should not have. Element text '%s'.", textValue, getElement().getText());
failWithMessage(errorMessage);
}
return this;
}
@Override
public WebElementState shouldBeEnabled() {
if (!isCurrentlyEnabled()) {
String errorMessage = String.format("Field '%s' should be enabled", this);
failWithMessage(errorMessage);
}
return this;
}
@Override
public boolean isEnabled() {
if (driverIsDisabled()) {
return false;
}
if (getElement() == null) {
return false;
}
if (shouldWaitForResult()) {
try {
waitForCondition().until(webDriver -> getElement().isEnabled());
} catch (TimeoutException timeout) {
return false;
}
}
return getElement().isEnabled();
}
@Override
public WebElementState shouldNotBeEnabled() {
if (isCurrentlyEnabled()) {
String errorMessage = String.format("Field '%s' should not be enabled", this);
failWithMessage(errorMessage);
}
return this;
}
/**
* Check to see if the element is clickable
*/
public boolean isClickable() {
try {
if (!driverIsDisabled() && shouldWaitForResult() && getElement() != null) {
waitForCondition().until(elementToBeClickable(getElement()));
return true;
}
} catch (ElementNotInteractableException | NoSuchElementException | StaleElementReferenceException |
TimeoutException e) {
return false;
}
return false;
}
/**
* Type a value into a field, making sure that the field is empty first.
*
* @param keysToSend
*/
@Override
public WebElementFacade type(final CharSequence... keysToSend) {
final CharSequence[] keyValue = nonNullCharSequenceFrom(keysToSend);
logIfVerbose("Type '" + keysToSend + "'");
if (driverIsDisabled()) {
return this;
}
WithRetries.on(this).perform(elementFacade -> {
elementFacade.getElement().clear();
elementFacade.getElement().sendKeys(nonNullCharSequenceFrom(keyValue));
}, 12);
notifyScreenChange();
return this;
}
private CharSequence[] nonNullCharSequenceFrom(CharSequence... charSequences) {
return Arrays.stream(charSequences).filter(chars -> chars != null).collect(Collectors.toList()).toArray(new CharSequence[]{});
}
/**
* Type a value into a field and then press Enter, making sure that the field is empty first.
*
* @param value
*/
@Override
public WebElementFacade typeAndEnter(final String value) {
logIfVerbose("Type and enter '" + value + "'");
if (driverIsDisabled()) {
return this;
}
WithRetries.on(this).perform(elementFacade -> {
elementFacade.getElement().clear();
elementFacade.getElement().sendKeys(nonNullCharSequenceFrom(value, Keys.ENTER));
}, NUMBER_OF_TRIES_FOR_UNREADY_ELEMENTS);
notifyScreenChange();
return this;
}
/**
* Type a value into a field and then press TAB, making sure that the field is empty first.
* This currently is not supported by all browsers, notably Firefox.
*
* @param value
*/
@Override
public WebElementFacade typeAndTab(final String value) {
logIfVerbose("Type and tab '" + value + "'");
if (driverIsDisabled()) {
return this;
}
WithRetries.on(this).perform(elementFacade -> {
elementFacade.getElement().clear();
elementFacade.getElement().sendKeys(value);
elementFacade.getElement().sendKeys(Keys.TAB);
}, NUMBER_OF_TRIES_FOR_UNREADY_ELEMENTS);
notifyScreenChange();
return this;
}
@Override
public void setWindowFocus() {
if (driverIsDisabled()) {
return;
}
getJavascriptExecutorFacade().executeScript("window.focus()");
}
@Override
public FluentDropdownSelect select() {
return new FluentDropdownSelect(this);
}
@Override
public FluentDropdownDeselect deselect() {
return new FluentDropdownDeselect(this);
}
private DropdownSelector dropdownSelect() {
return new DropdownSelector(this);
}
private DropdownDeselector dropdownDeselect() {
return new DropdownDeselector(this);
}
@Override
public WebElementFacade deselectAll() {
return dropdownDeselect().all();
}
@Override
public WebElementFacade deselectByIndex(int indexValue) {
return dropdownDeselect().byIndex(indexValue);
}
@Override
public WebElementFacade deselectByVisibleText(String label) {
return dropdownDeselect().byVisibleText(label);
}
@Override
public WebElementFacade deselectByValue(String value) {
return dropdownDeselect().byValue(value);
}
@Override
public WebElementFacade selectByVisibleText(final String label) {
return dropdownSelect().byVisibleText(label);
}
@Override
public String getSelectedVisibleTextValue() {
return dropdownSelect().visibleTextValue();
}
@Override
public String getFirstSelectedOptionVisibleText() {
return new Select(this).getFirstSelectedOption().getText();
}
@Override
public List getSelectedVisibleTexts() {
return dropdownSelect().visibleTextValues();
}
@Override
public String getFirstSelectedOptionValue() {
return new Select(this).getFirstSelectedOption().getAttribute("value");
}
@Override
@Deprecated
public WebElementFacade selectByValue(String value) {
return dropdownSelect().byValue(value);
}
@Override
public String getSelectedValue() {
return dropdownSelect().value();
}
@Override
public List getSelectedValues() {
return dropdownSelect().values();
}
@Override
@Deprecated
public WebElementFacade selectByIndex(int indexValue) {
return dropdownSelect().byIndex(indexValue);
}
public X getScreenshotAs(OutputType target) throws WebDriverException {
if (driverIsDisabled()) {
return null;
}
return getElement().getScreenshotAs(target);
}
protected void waitUntilElementAvailable() {
if (driverIsDisabled()) {
return;
}
withTimeoutOf((int) waitForTimeoutInMilliseconds, TimeUnit.MILLISECONDS).waitUntilEnabled();
}
protected void waitUntilElementPresent() {
if (driverIsDisabled()) {
return;
}
withTimeoutOf((int) waitForTimeoutInMilliseconds, TimeUnit.MILLISECONDS).waitUntilPresent();
}
protected boolean driverIsDisabled() {
return StepEventBus.getParallelEventBus().webdriverCallsAreSuspended();
}
private boolean elementIsPresent() {
try {
WebElement element = getElement();
if (getElement() == null) {
return false;
}
element.isDisplayed();
return true;
} catch (ElementNotInteractableException e) {
return true;
} catch (NotFoundException | ElementNotFoundAfterTimeoutError e) {
return false;
}
}
/**
* Returns true if an element is present on the screen, whether visible or not.
*/
public boolean isPresent() {
if (driverIsDisabled()) {
return false;
}
return elementIsPresent();
}
@Override
public WebElementState shouldBePresent() {
if (!isPresent()) {
failWithMessage("Field should be present");
}
return this;
}
@Override
public WebElementState shouldNotBePresent() {
if (isPresent()) {
failWithMessage("Field should not be present");
}
return this;
}
@Override
public WebElementState shouldBeSelected() {
if (!isSelected()) {
failWithMessage("Field should be selected");
}
return this;
}
@Override
public WebElementState shouldNotBeSelected() {
if (isSelected()) {
failWithMessage("Field should not be selected");
}
return this;
}
private void failWithMessage(String errorMessage) {
throw new AssertionError(getErrorMessage(errorMessage));
}
private void checkPresenceOfWebElement() {
try {
if (!driverIsDisabled() && shouldWaitForResult()) {
waitForCondition().until(elementIsDisplayed(this));
}
} catch (Throwable error) {
if (webElement != null) {
throwShouldBeVisibleErrorWithCauseIfPresent(error, error.getMessage());
} else {
throwNoSuchElementExceptionWithCauseIfPresent(error, error.getMessage());
}
}
}
@Override
public WebElementFacade waitUntilVisible() {
checkPresenceOfWebElement();
return this;
}
@Override
public WebElementFacade waitUntilPresent() {
try {
if (!driverIsDisabled()) {
waitForCondition().until(WebElementExpectations.elementIsPresent(this));
}
} catch (TimeoutException timeout) {
throwShouldBePresentErrorWithCauseIfPresent(timeout, timeout.getMessage());
}
return this;
}
private void throwNoSuchElementExceptionWithCauseIfPresent(final Throwable timeout, final String defaultMessage) {
String timeoutMessage = (timeout.getCause() != null) ? timeout.getCause().getMessage() : timeout.getMessage();
String finalMessage = (StringUtils.isNotEmpty(timeoutMessage)) ? timeoutMessage : defaultMessage;
throw new NoSuchElementException(finalMessage, timeout);
}
private void throwShouldBeVisibleErrorWithCauseIfPresent(final Throwable timeout, final String defaultMessage) {
String timeoutMessage = (timeout.getCause() != null) ? timeout.getCause().getMessage() : timeout.getMessage();
String finalMessage = (StringUtils.isNotEmpty(timeoutMessage)) ? timeoutMessage : defaultMessage;
throw new ElementShouldBeVisibleException(finalMessage, timeout);
}
private void throwShouldBeInvisibleErrorWithCauseIfPresent(final Throwable timeout, final String defaultMessage) {
String timeoutMessage = (timeout.getCause() != null) ? timeout.getCause().getMessage() : timeout.getMessage();
String finalMessage = (StringUtils.isNotEmpty(timeoutMessage)) ? timeoutMessage : defaultMessage;
throw new ElementShouldBeInvisibleException(finalMessage, timeout);
}
private void throwShouldBePresentErrorWithCauseIfPresent(final Throwable timeout, final String defaultMessage) {
String timeoutMessage = (timeout.getCause() != null) ? timeout.getCause().getMessage() : timeout.getMessage();
String finalMessage = (StringUtils.isNotEmpty(timeoutMessage)) ? timeoutMessage : defaultMessage;
throw new ElementShouldBePresentException(finalMessage, timeout);
}
private boolean hasValueAttribute(WebElement element) {
try {
return element.getAttribute("value") != null;
} catch (UnsupportedCommandException exception) {
return false;
}
}
@Override
public Wait waitForCondition() {
return new FluentWait<>(driver, webdriverClock, sleeper)
.withTimeout(Duration.ofMillis(waitForTimeoutInMilliseconds))
.pollingEvery(Duration.ofMillis(WAIT_FOR_ELEMENT_PAUSE_LENGTH))
.ignoreAll(Arrays.asList(NoSuchElementException.class,
NoSuchFrameException.class,
StaleElementReferenceException.class));
}
@Override
public WebElementFacade waitUntilNotVisible() {
if (driverIsDisabled()) {
return this;
}
if (!isCurrentlyVisible()) {
return this;
}
try {
waitForCondition().until(elementIsNotDisplayed(this));
} catch (TimeoutException timeout) {
throwShouldBeInvisibleErrorWithCauseIfPresent(timeout, "Expected hidden element was displayed");
}
return this;
}
@Override
public String getValue() {
checkPresenceOfWebElement();
return getElement().getAttribute("value");
}
@Override
public boolean isSelected() {
return getElement().isSelected();
}
@Override
public String getText() {
if (driverIsDisabled()) {
return "";
}
if (LEGACY_WAIT_FOR_TEXT.booleanFrom(environmentVariables, false)) {
checkPresenceOfWebElement();
}
return getElement().getText();
}
@Override
public String getTextContent() {
if (driverIsDisabled()) {
return "";
}
return getElement().getAttribute("textContent");
}
@Override
public String getAriaLabel() {
return getAttribute("aria-label");
}
@Override
public String getRole() {
return getAttribute("role");
}
@Override
public boolean isDisabled() {
if (driverIsDisabled()) {
return true;
}
return !getElement().isEnabled();
}
private boolean driverIsActive() {
return ((driver != null) && (!driverIsDisabled()));
}
@Override
public WebElementFacade waitUntilEnabled() {
try {
if (driverIsActive()) {
waitForCondition().until(elementIsEnabled(this));
}
} catch (TimeoutException timeout) {
throw new ElementShouldBeEnabledException("Expected enabled element was not enabled", timeout);
}
return this;
}
@Override
public WebElementFacade waitUntilClickable() {
try {
if (driverIsActive()) {
waitForCondition().until(elementIsClickable(this));
}
} catch (TimeoutException timeout) {
throw new ElementShouldBeEnabledException("Expected enabled element was not enabled", timeout);
}
return this;
}
@Override
public WebElementFacade waitUntilDisabled() {
try {
if (driverIsActive()) {
waitForCondition().until(elementIsNotEnabled(this));
}
} catch (TimeoutException timeout) {
throw new ElementShouldBeDisabledException("Expected disabled element was not disabled", timeout);
}
return this;
}
@Override
public String getTextValue() {
if (driverIsDisabled()) {
return "";
}
waitUntilPresent();
if (!isVisible()) {
return "";
}
if (valueAttributeSupportedAndDefinedIn(getElement())) {
return getValue();
}
if (!StringUtils.isEmpty(getElement().getText())) {
return getElement().getText();
}
return "";
}
@Override
public WebElementState expect(String errorMessage) {
return copy().expectingErrorMessage(errorMessage);
}
private Optional expectedErrorMessage = Optional.empty();
protected WebElementState expectingErrorMessage(String errorMessage) {
this.expectedErrorMessage = Optional.of(errorMessage);
return this;
}
protected String getErrorMessage(String defaultErrorMessage) {
return expectedErrorMessage.orElse(defaultErrorMessage);
}
private boolean valueAttributeSupportedAndDefinedIn(final WebElement element) {
return hasValueAttribute(element) && StringUtils.isNotEmpty(getValue());
}
/**
* Click on an element, with or without waiting for it to be visible and enabled
*/
@Override
public void click(ClickStrategy clickStrategy) {
if (driverIsDisabled()) {
return;
}
switch (clickStrategy) {
case WAIT_UNTIL_ENABLED:
waitUntilElementAvailable();
break;
case WAIT_UNTIL_PRESENT:
waitUntilElementPresent();
break;
case IMMEDIATE:
break;
}
WithRetries.on(this).perform(elementFacade -> elementFacade.getElement().click(), 12);
logClick();
notifyScreenChange();
}
/**
* Wait for an element to be visible and enabled, and then click on it.
*/
@Override
public void click() {
click(ClickStrategy.WAIT_UNTIL_ENABLED);
}
public void doubleClick() {
Actions actions = new Actions(driver);
WithRetries.on(this).perform(actions::doubleClick, 12);
}
public void contextClick() {
Actions actions = new Actions(driver);
WithRetries.on(this).perform(actions::contextClick, 12);
}
private void logClick() {
logIfVerbose("click");
}
private void logIfVerbose(String logMessage) {
if (useVerboseLogging()) {
LOGGER.debug(logMessage + " : " + this);
}
}
private boolean useVerboseLogging() {
return ThucydidesSystemProperty.SERENITY_VERBOSE_STEPS.booleanFrom(environmentVariables);
}
@Override
public void clear() {
if (driverIsDisabled()) {
return;
}
WithRetries.on(this).perform(elementFacade -> elementFacade.getElement().clear(), 12);
}
protected void notifyScreenChange() {
StepEventBus.getParallelEventBus().notifyScreenChange();
}
@Override
public String toString() {
if (webElement != null && !driverIsDisabled()) {
return webElement.toString();
}
if (foundBy != null) {
return foundBy;
}
if (bySelector != null) {
return bySelector.toString();
}
if (locator != null) {
return locator.toString();
}
return "";
}
public void submit() {
getElement().submit();
}
public void sendKeys(CharSequence... keysToSend) {
if (!allElementsAreNull(keysToSend)) {
WithRetries.on(this).perform(elementFacade -> {
elementFacade.getElement().sendKeys(nonNullCharSequenceFrom(keysToSend));
}, 12);
}
}
private boolean allElementsAreNull(CharSequence[] keysToSend) {
return Arrays.stream(keysToSend).allMatch(Objects::isNull);
}
public String getTagName() {
return getElement().getTagName();
}
public List findElements(By by) {
return getElement().findElements(by);
}
public WebElement findElement(By by) {
return getElement().findElement(by);
}
/**
* Is this element displayed or not? This method avoids the problem of having to parse an
* element's "style" attribute.
* This method respects the semantics of the Selenium isDisplayed() method, and will throw an
* exception if the element is not found. To simply return true or false depending on whether an element
* is
*
* @return Whether or not the element is displayed
*/
public boolean isDisplayed() {
return getElement().isDisplayed();
}
public Point getLocation() {
return getElement().getLocation();
}
public Dimension getSize() {
return getElement().getSize();
}
@Override
public Rectangle getRect() {
return getElement().getRect();
}
public String getCssValue(String propertyName) {
return getElement().getCssValue(propertyName);
}
@Override
public boolean containsElements(By bySelector) {
return !findElements(bySelector).isEmpty();
}
@Override
public boolean containsElements(String xpathOrCssSelector) {
return !thenFindAll(xpathOrCssSelector).isEmpty();
}
@Override
public WebElementState shouldContainElements(By bySelector) {
if (!containsElements(bySelector)) {
String errorMessage = String.format("Could not find contained elements %s in %s", bySelector, getElement().toString());
failWithMessage(errorMessage);
}
return this;
}
@Override
public WebElementState shouldContainElements(String xpathOrCssSelector) {
if (!containsElements(xpathOrCssSelector)) {
String errorMessage = String.format("Could not find contained elements %s in %s", xpathOrCssSelector, getElement().toString());
failWithMessage(errorMessage);
}
return this;
}
@Override
public boolean hasClass(String cssClassName) {
String cssClassValue = getAttribute("class").toLowerCase();
List cssClasses = Splitter.on(" ").omitEmptyStrings().trimResults().splitToList(cssClassValue);
return cssClasses.contains(cssClassName.toLowerCase());
}
public WebElementFacade foundBy(String foundBy) {
this.foundBy = foundBy;
return this;
}
@Override
public Coordinates getCoordinates() {
return ((Locatable) getElement()).getCoordinates();
}
@Override
public WebElement getWrappedElement() {
return getResolvedElement();
}
@Override
public String getDomProperty(String name) {
return getResolvedElement().getDomProperty(name);
}
@Override
public String getDomAttribute(String name) {
return getResolvedElement().getDomAttribute(name);
}
@Override
public String getAriaRole() {
return getResolvedElement().getAriaRole();
}
@Override
public String getAccessibleName() {
return getResolvedElement().getAccessibleName();
}
@Override
public SearchContext getShadowRoot() {
return getResolvedElement().getShadowRoot();
}
@Override
public ListOfWebElementFacades findNestedElementsMatching(ResolvableElement nestedElement) {
return nestedElement.resolveAllFor(this);
}
public static ListOfWebElementFacades fromWebElements(List elements) {
List facades = elements.stream().map(element -> WebElementFacadeImpl.wrapWebElement(Serenity.getDriver(), element)).collect(Collectors.toList());
return new ListOfWebElementFacades(facades);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy