diff --git a/DrissionPage/chromium_base.py b/DrissionPage/chromium_base.py index af7d051..4ee6b28 100644 --- a/DrissionPage/chromium_base.py +++ b/DrissionPage/chromium_base.py @@ -706,9 +706,7 @@ class ChromiumBase(BasePage): sleep(interval) while self.ready_state != 'complete': sleep(.1) - if self._debug: - print('重试') - if show_errmsg: + if self._debug or show_errmsg: print(f'重试 {to_url}') if err: diff --git a/DrissionPage/chromium_element.py b/DrissionPage/chromium_element.py index a8f4b3f..311b3f9 100644 --- a/DrissionPage/chromium_element.py +++ b/DrissionPage/chromium_element.py @@ -1638,26 +1638,37 @@ class Click(object): """ self._ele = ele - def __call__(self, by_js=None, wait_loading=0): + def __call__(self, by_js=False, timeout=1, wait_loading=0): """点击元素 如果遇到遮挡,可选择是否用js点击 :param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击 + :param timeout: 模拟点击的超时时间,等待元素可见、不被遮挡、进入视口 :param wait_loading: 等待页面进入加载状态超时时间 :return: 是否点击成功 """ return self.left(by_js, wait_loading) - def left(self, by_js=None, wait_loading=0): + def left(self, by_js=False, timeout=1, wait_loading=0): """点击元素 如果遇到遮挡,可选择是否用js点击 :param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击 + :param timeout: 模拟点击的超时时间,等待元素可见、不被遮挡、进入视口 :param wait_loading: 等待页面进入加载状态超时时间 :return: 是否点击成功 """ if not by_js: try: self._ele.scroll.to_see() - if (self._ele.states.is_in_viewport and not self._ele.states.is_covered) or by_js is False: + can_click = False + + timeout = self._ele.page.timeout if timeout is None else timeout + end_time = perf_counter() + timeout + while perf_counter() < end_time: + if self._ele.states.is_in_viewport and self._ele.states.is_enabled and self._ele.states.is_displayed: + can_click = True + break + + if by_js is False or (can_click and not self._ele.states.is_covered): client_x, client_y = self._ele.locations.viewport_midpoint if self._ele.tag == 'input' \ else self._ele.locations.viewport_click_point self._click(client_x, client_y) @@ -1874,7 +1885,8 @@ class ChromiumSelect(object): if not self.is_multi: raise TypeError("只能对多项选框执行反选。") for i in self.options: - i.click(by_js=True) + mode = 'false' if i.states.is_selected else 'true' + i.run_js(f'this.selected={mode};') def clear(self): """清除所有已选项""" diff --git a/DrissionPage/chromium_element.pyi b/DrissionPage/chromium_element.pyi index c14315c..123af53 100644 --- a/DrissionPage/chromium_element.pyi +++ b/DrissionPage/chromium_element.pyi @@ -433,9 +433,9 @@ class Click(object): def __init__(self, ele: ChromiumElement): self._ele: ChromiumElement = ... - def __call__(self, by_js: bool = None, wait_loading: Union[bool, float] = 0) -> bool: ... + def __call__(self, by_js: bool = False, timeout: float = None, wait_loading: Union[bool, float] = 0) -> bool: ... - def left(self, by_js: bool = None, wait_loading: Union[bool, float] = 0) -> bool: ... + def left(self, by_js: bool = False, timeout: float = None, wait_loading: Union[bool, float] = 0) -> bool: ... def right(self): ... diff --git a/DrissionPage/chromium_page.py b/DrissionPage/chromium_page.py index efa2836..7d83300 100644 --- a/DrissionPage/chromium_page.py +++ b/DrissionPage/chromium_page.py @@ -310,7 +310,7 @@ class ChromiumPage(ChromiumBase): :param timeout: 等待提示框出现的超时时间,为None则使用self.timeout属性的值 :return: 提示框内容文本,未等到提示框则返回None """ - timeout = timeout or self.timeout + timeout = self.timeout if timeout is None else timeout timeout = .1 if timeout <= 0 else timeout end_time = perf_counter() + timeout while not self._alert.activated and perf_counter() < end_time: diff --git a/setup.py b/setup.py index 7329fd6..a48f68a 100644 --- a/setup.py +++ b/setup.py @@ -6,7 +6,7 @@ with open("README.md", "r", encoding='utf-8') as fh: setup( name="DrissionPage", - version="3.2.16", + version="3.2.17", author="g1879", author_email="g1879@qq.com", description="Python based web automation tool. It can control the browser and send and receive data packets.",