Selenium WebDriver - определить, является ли элемент кликабельным (т.е. Не затенен dojo модальным лайтбокс)

Я пишу автоматизированные скрипты для тестирования веб-приложений, которые очень тяжелы для ajax. Например, модальный диалог отображается с текстом "Saving..." при сохранении настроек, в то время как лайтбокс отображает оставшуюся часть страницы.

Мои тестовые скрипты пытаются щелкнуть следующую ссылку в тесте, прежде чем сообщение исчезнет. Он почти всегда работает при управлении Firefox, но при управлении Chrome отображается следующая ошибка:

Exception in thread "main" org.openqa.selenium.WebDriverException: Element is not clickable at point (99.5, 118.5). Other element would receive the click: <div class="dijitDialogUnderlay _underlay" dojoattachpoint="node" id="lfn10Dijit_freedom_widget_common_environment_Dialog_8_underlay" style="width: 1034px; height: 1025px; "></div> (WARNING: The server did not provide any stacktrace information)

Это происходит потому, что лайтбокс скрывает элемент, который я хочу щелкнуть. Я хочу подождать, пока лайтбокс исчезнет, ​​прежде чем пытаться щелкнуть ссылку.

Проверка того, что лайтбокс больше не существует, не является допустимым временным решением, поскольку иногда существуют несколько уровней модальных диалогов и лайтбоксов и нет простого способа их отличить.

Есть ли способ в Selenium определить, является ли элемент кликабельным (никаких других элементов, скрывающих его)? Попробуйте/поймать будет обходным путем, но я бы предпочел сделать правильную проверку, если это возможно.

Ответ 1

Используйте условия WebDriverWait.

    WebDriverWait wait = new WebDriverWait(yourWebDriver, 5);
    wait.until(ExpectedConditions.elementToBeClickable(By.xpath("//xpath_to_element")));

Webdriver будет ждать 5 секунд, чтобы ваш элемент мог быть нажат.

Ответ 2

Вы можете использовать метод ExpectedConditions.invisibilityOfElementLocated(By by), который ждет, пока элемент не будет невидим или не будет присутствовать в DOM.

WebDriverWait wait = new WebDriverWait(yourWebDriver, 10);
wait.until(ExpectedConditions.invisibilityOfElementLocated(By.id("yourSavingModalDialogDiv")));

Таким образом, в зависимости от того, сколько времени ваш модальный диалог станет невидимым или уйдет с DOM, webdriver будет ждать. Ожидание - не более 10 секунд.

Ответ 3

Вы можете создать функцию/метод clickUntil, которая заставит WebDriver дождаться момента, когда элемент будет доступен с помощью таймаута. Он попытается щелкнуть элемент и выбросить сообщения об ошибке "Element not clickable" каждый раз, пока он не будет нажат или не истечет время ожидания.

Не уверен, как записать это в dojo, но это идея.

Ответ 4

В Scala:

  • Стандартный код ожидания (видимость/невидимость)

    (new WebDriverWait(remote, 45)).until(
        ExpectedConditions.visibilityOf(remote.findElement(locator))
    )
    Thread.sleep(3000)
    
  • Больше видимости в журналах:

    while (remote.findElement(locator).isDisplayed) {
        println(s"isDisplayed: $ii $a : " + remote.findElement(locator).isDisplayed)
        Thread.sleep(100)
    }
    
  • Если у вас есть асинхронные процессы JavaScript, используйте веб-элементы с меткой времени:

    val oldtimestamp = remote.findElements(locator).get(0).getAttribute("data-time-stamp")
    
    while (oldtimestamp == remote.findElements(locator).get(0).getAttribute("data-time-stamp")) {
        println("Tstamp2:" + remote.findElements(locator).get(0).getAttribute("data-time-stamp"))
        Thread.sleep(200)
    }
    

Ответ 5

У меня также есть те же проблемы, но я тестировал много входных данных на сайте. Один кликабель, который я тестировал, а другой - не кликабельный, просто пропущен. Я сделал это методом try() catch() Просто код:

for(){ // for all my input
try {
    driver.findElement(By.xpath("...."
                                + "//input)["+(i+1)+"]")).click();

  ... tests...


} catch(Exception e) {
     if(e.getMessage().contains("is not clickable at point")) {

          System.out.println(driver.findElement(By.xpath(locator)).
          getAttribute("name")+" are not clicable");
     } else {
          System.err.println(e.getMessage());
     }
}

И еще более элегантно:

 @SuppressWarnings("finally")
       public boolean tryClick(WebDriver driver,String locator, locatorMethods m) {

           boolean result = false;
           switch (m) {

           case xpath:
            try {
                driver.findElement(By.xpath(locator)).click();
                result= true;
            } catch (Exception e) {
                   if(e.getMessage().contains("is not clickable at point")) {
                       System.out.println(driver.findElement(By.xpath(locator)).getAttribute("name")+" are not clicable");
                   } else {
                       System.err.println(e.getMessage());
                   }
            } finally {
                break;
            }
        case id:
            try {   
                driver.findElement(By.id(locator)).click();
                return true;
            } catch (Exception e) {
               if(e.getMessage().contains("is not clickable at point")) {
                   System.out.println(driver.findElement(By.id(locator)).getAttribute("name")+" are not clicable");   
               } else {
                   System.err.println(e.getMessage());
               }
            } finally {
                break;
            }

          default:
              System.err.println("Unknown locator!");

        }
           return result;
  }