环境描述
- python版本: 3.10.0
- selenium版本: 3.141.0
- 浏览器: firefox
- 浏览器版本: 112.0.1 (64 位)
问题描述
我在使用python中的selenium显示等待等到元素可点击后,点击元素,等到元素可点击了,但是元素没有点击成功。示例代码如下。
from selenium import webdriver
from selenium.webdriver import ActionChains
from selenium.webdriver.common.by import By
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
with webdriver.Firefox()as driver:
driver.maximize_window()
url ="https://www.baidu.com/"
driver.get(url)
wait = WebDriverWait(driver,10)
locator = By.CSS_SELECTOR,"input#su"
method = EC.element_to_be_clickable(locator)
element = wait.until(method)
chains = ActionChains(driver)
chains.move_to_element(element).click().perform()
注:以上代码可点击成功,代码仅供参考
问题分析
我认为等到元素可点击,但是没有点击成功有以下几种可能性(仅供参考,欢迎指出问题和补充):
- 元素被禁用了,可使用element的
is_enabled
方法判断元素是否已启用。 - 元素被隐藏了,可通过下面的代码判断元素是否被隐藏。
display_style = element.value_of_css_property("display")visibility_style = element.value_of_css_property("visibility")if display_style =="none"or visibility_style =="hidden":print("元素被隐藏了")else:print("元素可见")
3.元素被其它元素遮挡了,这也是我遇到的问题,可通过下面的代码判断元素是否被其它元素遮挡。is_element_overlapped = self.driver.execute_script(""" function isOverlapping(element) { const rect1 = element.getBoundingClientRect(); const elements = document.querySelectorAll("*"); for (let i = 0; i < elements.length; i++) { const element = elements[i]; if (element === rect1) continue; const rect2 = element.getBoundingClientRect(); if ( rect1.top < rect2.bottom && rect1.bottom > rect2.top && rect1.left < rect2.right && rect1.right > rect2.left ) { return true; } } return false; } return isOverlapping(arguments[0]); """, element)if is_element_overlapped:print("元素被遮挡了")else:print("元素没有被遮挡")
### 元素被遮挡点击方法#### 解决方法1使用js进行点击,代码如下。driver.execute_script("arguments[0].click();", element)
#### 解决方法2定位到上层元素再将鼠标移动到要点击的元素上。代码如下。defclick_by_offset(driver: WebDriver, element: WebElement)->None:""" 鼠标先移动到该元素的父元素上,鼠标再移动到该元素上,然后点击 """# 要点击的元素的位置 x, y = element.location.values()# 要点击的元素的父元素 parent_element = driver.execute_script("return arguments[0].offsetParent;", element)assertisinstance(parent_element, WebElement),f"{parent_element}类型错误,应为{WebElement}"# 要点击的元素的父元素的位置 parent_x, parent_y = parent_element.get_attribute("offsetLeft"), parent_element.get_attribute("offsetTop")# 偏移量 xoffset, yoffset = x -int(parent_x), y -int(parent_y)# 点击 ActionChains(driver).move_to_element_with_offset(parent_element, xoffset, yoffset).click().perform()
完整示例代码如下。from selenium import webdriverfrom selenium.webdriver import ActionChainsfrom selenium.webdriver.common.by import Byfrom selenium.webdriver.remote.webdriver import WebDriverfrom selenium.webdriver.remote.webelement import WebElementfrom selenium.webdriver.support.wait import WebDriverWaitfrom selenium.webdriver.support import expected_conditions as ECdefclick_by_offset(driver: WebDriver, element: WebElement)->None:""" 鼠标先移动到该元素的父元素上,鼠标再移动到该元素上,然后点击 """# 要点击的元素的位置 x, y = element.location.values()# 要点击的元素的父元素 parent_element = driver.execute_script("return arguments[0].offsetParent;", element)assertisinstance(parent_element, WebElement),f"{parent_element}类型错误,应为{WebElement}"# 要点击的元素的父元素的位置 parent_x, parent_y = parent_element.get_attribute("offsetLeft"), parent_element.get_attribute("offsetTop")# 偏移量 xoffset, yoffset = x -int(parent_x), y -int(parent_y)# 点击 ActionChains(driver).move_to_element_with_offset(parent_element, xoffset, yoffset).click().perform()if __name__ =='__main__':with webdriver.Firefox()as driver: driver.maximize_window() url ="https://www.baidu.com" driver.get(url) wait = WebDriverWait(driver,10) locator = By.CSS_SELECTOR,"input#su" method = EC.element_to_be_clickable(locator) element = wait.until(method) click_by_offset(driver, element)
版权归原作者 qingtest85329140 所有, 如有侵权,请联系我们删除。