2

有一个 div 元素。其大小的 50% 在屏幕上。其他 50% 超出屏幕高度并且不可见。无法滚动。

我尝试使用 Selenium 自动化测试并单击该 div 元素,但有时它可以工作,有时它不能。

为什么 Selenium 不只是单击该 div 的“屏幕区域”?

这个功能是如何实现的?当我告诉 Selenium 单击一个巨大的 div 元素时,它会单击该 div 上的随机位置吗?

4

2 回答 2

1

元素的视图中心点

根据WebDriver W3C 规范,元素的视图中心点是矩形的原点位置,该矩形是元素的第一个DOM客户端矩形和初始视口之间的交点。

给定一个已知可见的元素,可以这样计算:

  1. 让 rectangle 成为通过在 element 上调用 getClientRects 返回的 DOMRect 序列的第一个元素。
  2. 让left为max(0,min(x坐标,x坐标+宽度尺寸))。
  3. 设右为 min(innerWidth, max(x 坐标, x 坐标 + 宽度尺寸))。
  4. 让 top 为 max(0, min(y 坐标,y 坐标 + 高度维度))。
  5. 设底部为 min(innerHeight, max(y 坐标, y 坐标 + 高度维度))。
  6. 设 x 为 floor((left + right) ÷ 2.0)。
  7. 设 y 为 floor((top + bottom) ÷ 2.0)。
  8. 返回 (x, y) 对。

如果一个元素是它自己的指针可交互绘制树的成员,那么它就在视图中,假设它的指针事件没有被禁用。


元素点击

element_click

根据文档,如果元素尚未与指针交互,元素单击命令将滚动到视图中,并单击其视图中心点。

注意:如果元素的中心点被另一个元素遮挡,则返回元素点击拦截错误。如果元素在视口之外,则返回元素不可交互错误。


解决方案

在这种情况下,有以下两种可能的解决方案:

  1. 您可以诱导WebDriverWaitexpected_conditions设置为element_to_be_clickable(). 因此,您的代码行将是:

    WebDriverWait(browser, 20).until(EC.element_to_be_clickable((By.CSS_SELECTOR, ".reply-button"))).click()
    
  2. 可以分别使用move_to_element(to_element)click(on_element=None)方法。因此,您的代码行将是:

    ActionChains(driver).move_to_element(element).click(element).perform()
    

参考

您可以在 selenium.common.exceptions.ElementClickInterceptedException 中找到相关讨论:消息:元素单击被截获:元素无法使用 Selenium 和 Python 进行点击

于 2019-12-01T23:11:42.080 回答
0

如果您基于 xpath、id 等定位器单击,是的,它会单击随机位置。但只要单击该元素并执行该单击的结果动作就没有关系。

关于有时它是否有效,如果您检查了要启用或单击的按钮,它将始终有效。

于 2019-12-01T17:49:43.343 回答