增加点击等待url或title变化方法;一些等待成功时返回对象本身;点击成功返回元素对象

This commit is contained in:
g1879 2024-07-19 14:24:52 +08:00
parent 362566d2b6
commit 71ac85d3bf
6 changed files with 107 additions and 43 deletions

View File

@ -107,6 +107,11 @@ class ChromiumPage(ChromiumBase):
"""返回所控制的浏览器版本号"""
return self._browser.version
@property
def address(self):
"""返回浏览器地址ip:port"""
return self.browser.address
def save(self, path=None, name=None, as_pdf=False, **kwargs):
"""把当前页面保存为文件如果path和name参数都为None只返回文本
:param path: 保存路径为None且name不为None时保存在当前路径

View File

@ -61,6 +61,9 @@ class ChromiumPage(ChromiumBase):
@property
def browser_version(self) -> str: ...
@property
def address(self) -> str: ...
@property
def set(self) -> ChromiumPageSetter: ...

View File

@ -43,7 +43,7 @@ class Clicker(object):
select = self._ele.parent('t:select')
if select.select.is_multi:
self._ele.parent('t:select').select.cancel_by_option(self._ele)
return
return self._ele
if not by_js: # 模拟点击
can_click = False
@ -101,11 +101,11 @@ class Clicker(object):
lx, ly = self._ele.rect._get_page_coord(vx, vy)
self._click(lx, ly, vx, vy)
return True
return self._ele
if by_js is not False:
self._ele._run_js('this.click();')
return True
return self._ele
if Settings.raise_when_click_failed:
raise CanNotClickError
return False
@ -113,7 +113,7 @@ class Clicker(object):
def right(self):
"""右键单击"""
self._ele.owner.scroll.to_see(self._ele)
self._click(*self._ele.rect.click_point, *self._ele.rect.viewport_click_point, button='right')
return self._click(*self._ele.rect.click_point, *self._ele.rect.viewport_click_point, button='right')
def middle(self, get_tab=True):
"""中键单击默认返回新出现的tab对象
@ -143,14 +143,14 @@ class Clicker(object):
w, h = self._ele.rect.size
offset_x = w // 2
offset_y = h // 2
self._click(*offset_scroll(self._ele, offset_x, offset_y), button=button, count=count)
return self._click(*offset_scroll(self._ele, offset_x, offset_y), button=button, count=count)
def multi(self, times=2):
"""多次点击
:param times: 默认双击
:return: None
"""
self.at(count=times)
return self.at(count=times)
def to_download(self, save_path=None, rename=None, suffix=None, new_tab=False, by_js=False, timeout=None):
"""点击触发下载
@ -198,6 +198,34 @@ class Clicker(object):
return (self._ele.tab.browser.get_mix_tab(tid) if self._ele.tab._type == 'MixTab'
else self._ele.tab.browser.get_tab(tid))
def for_url_change(self, text=None, exclude=False, by_js=False, timeout=None):
"""点击并等待tab的url变成包含或不包含指定文本
:param text: 用于识别的文本为None等待当前url变化
:param exclude: 是否排除为True时当url不包含text指定文本时返回Truetext为None时自动设为True
:param by_js: 是否用js点击
:param timeout: 超时时间为None使用页面设置
:return: 是否等待成功
"""
if text is None:
exclude = True
text = self._ele.tab.url
self.left(by_js=by_js)
return True if self._ele.tab.wait.url_change(text=text, exclude=exclude, timeout=timeout) else False
def for_title_change(self, text=None, exclude=False, by_js=False, timeout=None):
"""点击并等待tab的title变成包含或不包含指定文本
:param text: 用于识别的文本为None等待当前title变化
:param exclude: 是否排除为True时当title不包含text指定文本时返回Truetext为None时自动设为True
:param by_js: 是否用js点击
:param timeout: 超时时间为None使用页面设置
:return: 是否等待成功
"""
if text is None:
exclude = True
text = self._ele.tab.title
self.left(by_js=by_js)
return True if self._ele.tab.wait.title_change(text=text, exclude=exclude, timeout=timeout) else False
def _click(self, loc_x, loc_y, view_x, view_y, button='left', count=1):
"""实施点击
:param loc_x: 绝对x坐标
@ -213,3 +241,4 @@ class Clicker(object):
y=view_y, button=button, clickCount=count, _ignore=AlertExistsError)
self._ele.owner._run_cdp('Input.dispatchMouseEvent', type='mouseReleased', x=view_x,
y=view_y, button=button, _ignore=AlertExistsError)
return self._ele

View File

@ -17,11 +17,13 @@ class Clicker(object):
def __init__(self, ele: ChromiumElement):
self._ele: ChromiumElement = ...
def __call__(self, by_js: Union[bool, str, None] = False, timeout: float = 1.5, wait_stop: bool = True) -> bool: ...
def __call__(self, by_js: Union[bool, str, None] = False,
timeout: float = 1.5, wait_stop: bool = True) -> Union[ChromiumElement, False]: ...
def left(self, by_js: Union[bool, str, None] = False, timeout: float = 1.5, wait_stop: bool = True) -> bool: ...
def left(self, by_js: Union[bool, str, None] = False,
timeout: float = 1.5, wait_stop: bool = True) -> Union[ChromiumElement, False]: ...
def right(self) -> None: ...
def right(self) -> ChromiumElement: ...
def middle(self, get_tab: bool = True) -> Union[ChromiumTab, MixTab, None]: ...
@ -29,9 +31,9 @@ class Clicker(object):
offset_x: float = None,
offset_y: float = None,
button: str = 'left',
count: int = 1) -> None: ...
count: int = 1) -> ChromiumElement: ...
def multi(self, times: int = 2) -> None: ...
def multi(self, times: int = 2) -> ChromiumElement: ...
def to_download(self,
save_path: Union[str, Path] = None,
@ -45,9 +47,15 @@ class Clicker(object):
def for_new_tab(self, by_js: bool = False, timeout: float = 3) -> Union[ChromiumTab, MixTab]: ...
def for_url_change(self, text: str = None, exclude: bool = False,
by_js: bool = False, timeout: float = None) -> bool: ...
def for_title_change(self, text: str = None, exclude: bool = False,
by_js: bool = False, timeout: float = None) -> bool: ...
def _click(self, loc_x: float,
loc_y: float,
view_x: float,
view_y: float,
button: str = 'left',
count: int = 1) -> None: ...
count: int = 1) -> ChromiumElement: ...

View File

@ -262,19 +262,19 @@ class BaseWaiter(OriginWaiter):
:param exclude: 是否排除为True时当url不包含text指定文本时返回True
:param timeout: 超时时间
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 等待成功返回页面对象否则返回False
"""
return self._change('url', text, exclude, timeout, raise_err)
return self._owner if self._change('url', text, exclude, timeout, raise_err) else False
def title_change(self, text, exclude=False, timeout=None, raise_err=None):
"""等待title变成包含或不包含指定文本
:param text: 用于识别的文本
:param exclude: 是否排除为True时当title不包含text指定文本时返回True
:param timeout: 超时时间
:param timeout: 超时时间为None使用页面设置
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 等待成功返回页面对象否则返回False
"""
return self._change('title', text, exclude, timeout, raise_err)
return self._owner if self._change('title', text, exclude, timeout, raise_err) else False
def _change(self, arg, text, exclude=False, timeout=None, raise_err=None):
"""等待指定属性变成包含或不包含指定文本
@ -403,7 +403,7 @@ class ElementWaiter(OriginWaiter):
"""等待元素从dom删除
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_alive', False, timeout, raise_err, err_text='等待元素被删除失败。')
@ -411,7 +411,7 @@ class ElementWaiter(OriginWaiter):
"""等待元素从dom显示
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_displayed', True, timeout, raise_err, err_text='等待元素显示失败。')
@ -419,7 +419,7 @@ class ElementWaiter(OriginWaiter):
"""等待元素从dom隐藏
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_displayed', False, timeout, raise_err, err_text='等待元素隐藏失败。')
@ -435,7 +435,7 @@ class ElementWaiter(OriginWaiter):
"""等待当前元素不被遮盖
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_covered', False, timeout, raise_err, err_text='等待元素不被覆盖失败。')
@ -443,7 +443,7 @@ class ElementWaiter(OriginWaiter):
"""等待当前元素变成可用
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_enabled', True, timeout, raise_err, err_text='等待元素变成可用失败。')
@ -451,7 +451,7 @@ class ElementWaiter(OriginWaiter):
"""等待当前元素变成不可用
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
return self._wait_state('is_enabled', False, timeout, raise_err, err_text='等待元素变成不可用失败。')
@ -459,14 +459,14 @@ class ElementWaiter(OriginWaiter):
"""等待当前元素变成不可用或从DOM移除
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
if timeout is None:
timeout = self._timeout
end_time = perf_counter() + timeout
while perf_counter() < end_time:
if not self._ele.states.is_enabled or not self._ele.states.is_alive:
return True
return self._ele
sleep(.05)
if raise_err is True or Settings.raise_when_wait_failed is True:
@ -479,7 +479,7 @@ class ElementWaiter(OriginWaiter):
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param gap: 检测间隔时间
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
if timeout is None:
timeout = self._timeout
@ -498,7 +498,7 @@ class ElementWaiter(OriginWaiter):
while perf_counter() < end_time:
sleep(gap)
if self._ele.rect.size == size and self._ele.rect.location == location:
return True
return self._ele
size = self._ele.rect.size
location = self._ele.rect.location
@ -512,7 +512,7 @@ class ElementWaiter(OriginWaiter):
:param wait_moved: 是否等待元素运动结束
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
t1 = perf_counter()
r = self._wait_state('is_clickable', True, timeout, raise_err, err_text='等待元素可点击失败(等{}秒)。')
@ -536,11 +536,11 @@ class ElementWaiter(OriginWaiter):
:param timeout: 超时时间为None使用元素所在页面timeout属性
:param raise_err: 等待失败时是否报错为None时根据Settings设置
:param err_text: 抛出错误时显示的信息
:return: 是否等待成功
:return: 成功返回元素对象失败返回False
"""
a = self._ele.states.__getattribute__(attr)
if (a and mode) or (not a and not mode):
return True if isinstance(a, bool) else a
return self._ele if isinstance(a, bool) else a
if timeout is None:
timeout = self._timeout
@ -548,7 +548,7 @@ class ElementWaiter(OriginWaiter):
while perf_counter() < end_time:
a = self._ele.states.__getattribute__(attr)
if (a and mode) or (not a and not mode):
return True if isinstance(a, bool) else a
return self._ele if isinstance(a, bool) else a
sleep(.05)
err_text = err_text or '等待元素状态改变失败(等待{}秒)。'

View File

@ -5,7 +5,7 @@
@Copyright: (c) 2024 by g1879, Inc. All Rights Reserved.
@License : BSD 3-Clause.
"""
from typing import Union, Tuple, Literal, List
from typing import Union, Tuple, List
from .downloader import DownloadMission
from .._base.browser import Chromium
@ -92,6 +92,12 @@ class TabWaiter(BaseWaiter):
def alert_closed(self) -> None: ...
def url_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumTab, MixTab]: ...
def title_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumTab, MixTab]: ...
class PageWaiter(TabWaiter):
_owner: Union[ChromiumPage, MixPage] = ...
@ -104,6 +110,12 @@ class PageWaiter(TabWaiter):
def all_downloads_done(self, timeout: float = None, cancel_if_timeout: bool = True) -> bool: ...
def url_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumPage, MixPage]: ...
def title_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumPage, MixPage]: ...
class ElementWaiter(OriginWaiter):
_owner: ChromiumElement = ...
@ -116,36 +128,37 @@ class ElementWaiter(OriginWaiter):
@property
def _timeout(self) -> float: ...
def deleted(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def deleted(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def displayed(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def displayed(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def hidden(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def hidden(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def covered(self, timeout: float = None, raise_err: bool = None) -> Union[Literal[False], int]: ...
def covered(self, timeout: float = None, raise_err: bool = None) -> Union[False, int]: ...
def not_covered(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def not_covered(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def enabled(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def enabled(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def disabled(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def disabled(self, timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def clickable(self, wait_moved: bool = True, timeout: float = None, raise_err: bool = None) -> bool: ...
def clickable(self, wait_moved: bool = True,
timeout: float = None, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def has_rect(self,
timeout: float = None,
raise_err: bool = None) -> Union[Literal[False], List[Tuple[float, float]]]: ...
raise_err: bool = None) -> Union[False, List[Tuple[float, float]]]: ...
def disabled_or_deleted(self, timeout: float = None, raise_err: bool = None) -> bool: ...
def stop_moving(self, timeout: float = None, gap: float = .1, raise_err: bool = None) -> bool: ...
def stop_moving(self, timeout: float = None, gap: float = .1, raise_err: bool = None) -> Union[ChromiumElement, False]: ...
def _wait_state(self,
attr: str,
mode: bool = False,
timeout: float = None,
raise_err: bool = None,
err_text: str = None) -> bool: ...
err_text: str = None) -> Union[ChromiumElement, False]: ...
class FrameWaiter(BaseWaiter, ElementWaiter):
@ -154,3 +167,9 @@ class FrameWaiter(BaseWaiter, ElementWaiter):
def __init__(self, owner: ChromiumFrame): ...
def __call__(self, second: float, scope: float = None) -> ChromiumFrame: ...
def url_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumFrame]: ...
def title_change(self, text: str, exclude: bool = False,
timeout: float = None, raise_err: bool = None) -> Union[False, ChromiumFrame]: ...