mirror of
https://gitee.com/g1879/DrissionPage.git
synced 2024-12-10 04:00:23 +08:00
3.0.6改进点击逻辑
This commit is contained in:
parent
3c5a1205fd
commit
410839b23f
@ -125,54 +125,40 @@ class ChromiumElement(DrissionElement):
|
|||||||
return {'height': 0, 'width': 0}
|
return {'height': 0, 'width': 0}
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def location(self) -> dict:
|
def client_location(self) -> Union[dict, None]:
|
||||||
"""返回元素左上角的绝对坐标"""
|
"""返回元素左上角在视口中的坐标"""
|
||||||
js = '''function(){
|
m = self._get_client_rect('border')
|
||||||
function getElementPagePosition(element){
|
return {'x': m[0], 'y': m[1]} if m else None
|
||||||
var actualLeft = element.offsetLeft;
|
|
||||||
var current = element.offsetParent;
|
|
||||||
while (current !== null){
|
|
||||||
actualLeft += current.offsetLeft;
|
|
||||||
current = current.offsetParent;
|
|
||||||
}
|
|
||||||
var actualTop = element.offsetTop;
|
|
||||||
var current = element.offsetParent;
|
|
||||||
while (current !== null){
|
|
||||||
actualTop += (current.offsetTop+current.clientTop);
|
|
||||||
current = current.offsetParent;
|
|
||||||
}
|
|
||||||
return actualLeft.toString() +' '+actualTop.toString();
|
|
||||||
}
|
|
||||||
return getElementPagePosition(this);}'''
|
|
||||||
xy = self.run_script(js)
|
|
||||||
x, y = xy.split(' ')
|
|
||||||
return {'x': int(x.split('.')[0]), 'y': int(y.split('.')[0])}
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def client_location(self) -> dict:
|
def client_midpoint(self) -> Union[dict, None]:
|
||||||
"""返回元素左上角在视口中的坐标"""
|
"""返回元素中间点在视口中的坐标"""
|
||||||
js = 'return this.getBoundingClientRect().left.toString()+" "+this.getBoundingClientRect().top.toString();'
|
m = self._get_client_rect('border')
|
||||||
xy = self.run_script(js)
|
return {'x': m[2] - m[0], 'y': m[5] - m[1]} if m else None
|
||||||
x, y = xy.split(' ')
|
|
||||||
return {'x': int(x.split('.')[0]), 'y': int(y.split('.')[0])}
|
@property
|
||||||
|
def location(self) -> Union[dict, None]:
|
||||||
|
"""返回元素左上角的绝对坐标"""
|
||||||
|
cl = self.client_location
|
||||||
|
return self._get_absolute_rect(cl['x'], cl['y']) if cl else None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def midpoint(self) -> dict:
|
def midpoint(self) -> dict:
|
||||||
"""返回元素中间点的绝对坐标"""
|
"""返回元素中间点的绝对坐标"""
|
||||||
loc = self.location
|
cl = self.client_midpoint
|
||||||
size = self.size
|
return self._get_absolute_rect(cl['x'], cl['y']) if cl else None
|
||||||
lx = loc['x'] + size['width'] // 2
|
|
||||||
ly = loc['y'] + size['height'] // 2
|
|
||||||
return {'x': lx, 'y': ly}
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def client_midpoint(self) -> dict:
|
def _client_click_point(self) -> Union[dict, None]:
|
||||||
"""返回元素中间点在视口中的坐标"""
|
"""返回元素左上角可接受点击的点视口坐标"""
|
||||||
loc = self.client_location
|
m = self._get_client_rect('padding')
|
||||||
size = self.size
|
return {'x': m[0], 'y': m[1]} if m else None
|
||||||
cx = loc['x'] + size['width'] // 2
|
|
||||||
cy = loc['y'] + size['height'] // 2
|
@property
|
||||||
return {'x': cx, 'y': cy}
|
def _click_point(self) -> Union[dict, None]:
|
||||||
|
"""返回元素左上角可接受点击的点的绝对坐标"""
|
||||||
|
cl = self._client_click_point
|
||||||
|
return self._get_absolute_rect(cl['x'], cl['y']) if cl else None
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def shadow_root(self) -> Union[None, 'ChromiumShadowRootElement']:
|
def shadow_root(self) -> Union[None, 'ChromiumShadowRootElement']:
|
||||||
@ -336,8 +322,8 @@ class ChromiumElement(DrissionElement):
|
|||||||
|
|
||||||
@property
|
@property
|
||||||
def is_in_viewport(self) -> bool:
|
def is_in_viewport(self) -> bool:
|
||||||
"""返回元素是否出现在视口中,以元素中点为判断"""
|
"""返回元素是否出现在视口中,以元素可以接受点击的点为判断"""
|
||||||
loc = self.midpoint
|
loc = self.location
|
||||||
return _location_in_viewport(self.page, loc['x'], loc['y'])
|
return _location_in_viewport(self.page, loc['x'], loc['y'])
|
||||||
|
|
||||||
def attr(self, attr: str) -> Union[str, None]:
|
def attr(self, attr: str) -> Union[str, None]:
|
||||||
@ -611,17 +597,23 @@ class ChromiumElement(DrissionElement):
|
|||||||
else:
|
else:
|
||||||
self.input(('\ue009', 'a', '\ue017'), clear=False)
|
self.input(('\ue009', 'a', '\ue017'), clear=False)
|
||||||
|
|
||||||
def click(self, by_js: bool = None, timeout: float = .2) -> bool:
|
def click(self, by_js: bool = None, retry: bool = False, timeout: float = .2) -> bool:
|
||||||
"""点击元素 \n
|
"""点击元素 \n
|
||||||
如果遇到遮挡,会重新尝试点击直到超时,若都失败就改用js点击 \n
|
如果遇到遮挡,会重新尝试点击直到超时,若都失败就改用js点击 \n
|
||||||
:param by_js: 是否用js点击,为True时直接用js点击,为False时重试失败也不会改用js
|
:param by_js: 是否用js点击,为True时直接用js点击,为False时重试失败也不会改用js
|
||||||
:param timeout: 尝试点击的超时时间,不指定则使用父页面的超时时间
|
:param retry: 遇到其它元素遮挡时,是否重试
|
||||||
|
:param timeout: 尝试点击的超时时间,不指定则使用父页面的超时时间,retry为True时才生效
|
||||||
:return: 是否点击成功
|
:return: 是否点击成功
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def do_it(cx, cy, lx, ly) -> bool:
|
def do_it(cx, cy, lx, ly) -> Union[None, bool]:
|
||||||
r = self.page.driver.DOM.getNodeForLocation(x=lx + 1, y=ly + 1)
|
"""无遮挡返回True,有遮挡返回False,无元素返回None"""
|
||||||
if r.get('nodeId') != self._node_id:
|
try:
|
||||||
|
r = self.page.driver.DOM.getNodeForLocation(x=lx, y=ly)
|
||||||
|
except Exception:
|
||||||
|
return None
|
||||||
|
|
||||||
|
if retry and r.get('nodeId') != self._node_id:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
self._click(cx, cy)
|
self._click(cx, cy)
|
||||||
@ -630,25 +622,28 @@ class ChromiumElement(DrissionElement):
|
|||||||
if not by_js:
|
if not by_js:
|
||||||
self.page.scroll_to_see(self)
|
self.page.scroll_to_see(self)
|
||||||
if self.is_in_viewport:
|
if self.is_in_viewport:
|
||||||
midpoint = self.midpoint
|
client_point = self._client_click_point
|
||||||
client_midpoint = self.client_midpoint
|
if client_point:
|
||||||
client_x = client_midpoint['x']
|
loc_point = self._click_point
|
||||||
client_y = client_midpoint['y']
|
client_x = client_point['x']
|
||||||
loc_x = midpoint['x']
|
client_y = client_point['y']
|
||||||
loc_y = midpoint['y']
|
loc_x = loc_point['x']
|
||||||
|
loc_y = loc_point['y']
|
||||||
|
|
||||||
timeout = timeout if timeout is not None else self.page.timeout
|
|
||||||
end_time = perf_counter() + timeout
|
|
||||||
click = do_it(client_x, client_y, loc_x, loc_y)
|
|
||||||
while not click and perf_counter() < end_time:
|
|
||||||
click = do_it(client_x, client_y, loc_x, loc_y)
|
click = do_it(client_x, client_y, loc_x, loc_y)
|
||||||
|
if click:
|
||||||
|
return True
|
||||||
|
|
||||||
if click:
|
timeout = timeout if timeout is not None else self.page.timeout
|
||||||
return True
|
end_time = perf_counter() + timeout
|
||||||
|
while click is False and perf_counter() < end_time:
|
||||||
|
click = do_it(client_x, client_y, loc_x, loc_y)
|
||||||
|
|
||||||
|
if click is not None:
|
||||||
|
return True
|
||||||
|
|
||||||
if by_js is not False:
|
if by_js is not False:
|
||||||
js = 'this.click();'
|
self.run_script('this.click();')
|
||||||
self.run_script(js)
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
return False
|
return False
|
||||||
@ -657,7 +652,7 @@ class ChromiumElement(DrissionElement):
|
|||||||
offset_x: Union[int, str] = None,
|
offset_x: Union[int, str] = None,
|
||||||
offset_y: Union[int, str] = None,
|
offset_y: Union[int, str] = None,
|
||||||
button: str = 'left') -> None:
|
button: str = 'left') -> None:
|
||||||
"""带偏移量点击本元素,相对于左上角坐标。不传入x或y值时点击元素中点 \n
|
"""带偏移量点击本元素,相对于左上角坐标。不传入x或y值时点击元素左上角可接受点击的点 \n
|
||||||
:param offset_x: 相对元素左上角坐标的x轴偏移量
|
:param offset_x: 相对元素左上角坐标的x轴偏移量
|
||||||
:param offset_y: 相对元素左上角坐标的y轴偏移量
|
:param offset_y: 相对元素左上角坐标的y轴偏移量
|
||||||
:param button: 左键还是右键
|
:param button: 左键还是右键
|
||||||
@ -669,10 +664,10 @@ class ChromiumElement(DrissionElement):
|
|||||||
def r_click(self) -> None:
|
def r_click(self) -> None:
|
||||||
"""右键单击"""
|
"""右键单击"""
|
||||||
self.page.scroll_to_see(self)
|
self.page.scroll_to_see(self)
|
||||||
xy = self.client_midpoint
|
xy = self._client_click_point
|
||||||
self._click(xy['x'], xy['y'], 'right')
|
self._click(xy['x'], xy['y'], 'right')
|
||||||
|
|
||||||
def r_click_at(self, offset_x: Union[int, str], offset_y: Union[int, str]) -> None:
|
def r_click_at(self, offset_x: Union[int, str] = None, offset_y: Union[int, str] = None) -> None:
|
||||||
"""带偏移量右键单击本元素,相对于左上角坐标。不传入x或y值时点击元素中点 \n
|
"""带偏移量右键单击本元素,相对于左上角坐标。不传入x或y值时点击元素中点 \n
|
||||||
:param offset_x: 相对元素左上角坐标的x轴偏移量
|
:param offset_x: 相对元素左上角坐标的x轴偏移量
|
||||||
:param offset_y: 相对元素左上角坐标的y轴偏移量
|
:param offset_y: 相对元素左上角坐标的y轴偏移量
|
||||||
@ -815,6 +810,23 @@ class ChromiumElement(DrissionElement):
|
|||||||
t = self.run_script(js)
|
t = self.run_script(js)
|
||||||
return f':root{t}' if mode == 'css' else t
|
return f':root{t}' if mode == 'css' else t
|
||||||
|
|
||||||
|
def _get_client_rect(self, quad: str) -> Union[dict, None]:
|
||||||
|
"""按照类型返回坐标
|
||||||
|
:param quad: 方框类型,margin border padding
|
||||||
|
:return: 四个角坐标,大小为0时返回None
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
return self.page.run_cdp('DOM.getBoxModel', nodeId=self.node_id)['model'][quad]
|
||||||
|
except:
|
||||||
|
return None
|
||||||
|
|
||||||
|
def _get_absolute_rect(self, x, y) -> dict:
|
||||||
|
"""根据绝对坐标获取窗口坐标"""
|
||||||
|
js = 'return document.documentElement.scrollLeft+" "+document.documentElement.scrollTop;'
|
||||||
|
xy = self.run_script(js)
|
||||||
|
sx, sy = xy.split(' ')
|
||||||
|
return {'x': x + int(sx), 'y': y + int(sy)}
|
||||||
|
|
||||||
|
|
||||||
class ChromiumShadowRootElement(BaseElement):
|
class ChromiumShadowRootElement(BaseElement):
|
||||||
"""ChromiumShadowRootElement是用于处理ShadowRoot的类,使用方法和ChromiumElement基本一致"""
|
"""ChromiumShadowRootElement是用于处理ShadowRoot的类,使用方法和ChromiumElement基本一致"""
|
||||||
@ -1199,6 +1211,11 @@ class ChromiumBase(BasePage):
|
|||||||
self._wait_loading()
|
self._wait_loading()
|
||||||
return self._tab_obj
|
return self._tab_obj
|
||||||
|
|
||||||
|
@property
|
||||||
|
def is_loading(self) -> bool:
|
||||||
|
"""返回页面是否正在加载状态"""
|
||||||
|
return self._is_loading
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def url(self) -> str:
|
def url(self) -> str:
|
||||||
"""返回当前页面url"""
|
"""返回当前页面url"""
|
||||||
@ -1992,7 +2009,10 @@ def _run_script(page_or_ele, script: str, as_expr: bool = False, timeout: float
|
|||||||
if exceptionDetails:
|
if exceptionDetails:
|
||||||
raise RuntimeError(f'Evaluation failed: {exceptionDetails}')
|
raise RuntimeError(f'Evaluation failed: {exceptionDetails}')
|
||||||
|
|
||||||
return _parse_js_result(page, page_or_ele, res.get('result'))
|
try:
|
||||||
|
return _parse_js_result(page, page_or_ele, res.get('result'))
|
||||||
|
except Exception:
|
||||||
|
return res
|
||||||
|
|
||||||
|
|
||||||
def _parse_js_result(page, ele, result: dict):
|
def _parse_js_result(page, ele, result: dict):
|
||||||
@ -2076,7 +2096,7 @@ def _send_key(ele: ChromiumElement, modifier: int, key: str) -> None:
|
|||||||
ele.page.run_cdp('Input.dispatchKeyEvent', **data)
|
ele.page.run_cdp('Input.dispatchKeyEvent', **data)
|
||||||
|
|
||||||
|
|
||||||
def _offset_scroll(ele, offset_x: int, offset_y: int) -> tuple:
|
def _offset_scroll(ele: ChromiumElement, offset_x: int, offset_y: int) -> tuple:
|
||||||
"""接收元素及偏移坐标,滚动到偏移坐标,返回该点在视口中的坐标
|
"""接收元素及偏移坐标,滚动到偏移坐标,返回该点在视口中的坐标
|
||||||
:param ele: 元素对象
|
:param ele: 元素对象
|
||||||
:param offset_x: 偏移量x
|
:param offset_x: 偏移量x
|
||||||
@ -2084,14 +2104,14 @@ def _offset_scroll(ele, offset_x: int, offset_y: int) -> tuple:
|
|||||||
:return: 视口中的坐标
|
:return: 视口中的坐标
|
||||||
"""
|
"""
|
||||||
location = ele.location
|
location = ele.location
|
||||||
midpoint = ele.midpoint
|
midpoint = ele._click_point
|
||||||
lx = location['x'] + offset_x if offset_x else midpoint['x']
|
lx = location['x'] + offset_x if offset_x else midpoint['x']
|
||||||
ly = location['y'] + offset_y if offset_y else midpoint['y']
|
ly = location['y'] + offset_y if offset_y else midpoint['y']
|
||||||
|
|
||||||
if not _location_in_viewport(ele.page, lx, ly):
|
if not _location_in_viewport(ele.page, lx, ly):
|
||||||
ele.page.scroll.to_location(lx, ly)
|
ele.page.scroll.to_location(lx, ly)
|
||||||
cl = ele.client_location
|
cl = ele.client_location
|
||||||
cm = ele.client_midpoint
|
cm = ele._client_click_point
|
||||||
cx = cl['x'] + offset_x if offset_x else cm['x']
|
cx = cl['x'] + offset_x if offset_x else cm['x']
|
||||||
cy = cl['y'] + offset_y if offset_y else cm['y']
|
cy = cl['y'] + offset_y if offset_y else cm['y']
|
||||||
return cx, cy
|
return cx, cy
|
||||||
|
@ -217,10 +217,10 @@ class ChromiumPage(ChromiumBase):
|
|||||||
"""
|
"""
|
||||||
tabs = self.tabs
|
tabs = self.tabs
|
||||||
if not tab_id:
|
if not tab_id:
|
||||||
|
tab_id = self.main_tab
|
||||||
|
if tab_id not in tabs:
|
||||||
tab_id = tabs[0]
|
tab_id = tabs[0]
|
||||||
elif tab_id == 'main':
|
if tab_id == self.tab_id:
|
||||||
tab_id = self._main_tab
|
|
||||||
if tab_id == self.tab_id or tab_id not in tabs:
|
|
||||||
return
|
return
|
||||||
|
|
||||||
if activate:
|
if activate:
|
||||||
@ -238,7 +238,13 @@ class ChromiumPage(ChromiumBase):
|
|||||||
:return: None
|
:return: None
|
||||||
"""
|
"""
|
||||||
all_tabs = set(self.tabs)
|
all_tabs = set(self.tabs)
|
||||||
tabs = set(tab_ids) if tab_ids else {self.tab_id}
|
if isinstance(tab_ids, str):
|
||||||
|
tabs = {tab_ids}
|
||||||
|
elif tab_ids is None:
|
||||||
|
tabs = {self.tab_id}
|
||||||
|
else:
|
||||||
|
tabs = set(tab_ids)
|
||||||
|
|
||||||
if others:
|
if others:
|
||||||
tabs = all_tabs - tabs
|
tabs = all_tabs - tabs
|
||||||
|
|
||||||
@ -255,6 +261,9 @@ class ChromiumPage(ChromiumBase):
|
|||||||
while len(self.tabs) != end_len:
|
while len(self.tabs) != end_len:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
if self.main_tab in tabs:
|
||||||
|
self.main_tab = self.tabs[0]
|
||||||
|
|
||||||
self.to_tab()
|
self.to_tab()
|
||||||
|
|
||||||
def close_other_tabs(self, tab_ids: Union[str, List[str], Tuple[str]] = None) -> None:
|
def close_other_tabs(self, tab_ids: Union[str, List[str], Tuple[str]] = None) -> None:
|
||||||
|
2
setup.py
2
setup.py
@ -6,7 +6,7 @@ with open("README.md", "r", encoding='utf-8') as fh:
|
|||||||
|
|
||||||
setup(
|
setup(
|
||||||
name="DrissionPage",
|
name="DrissionPage",
|
||||||
version="3.0.5",
|
version="3.0.6",
|
||||||
author="g1879",
|
author="g1879",
|
||||||
author_email="g1879@qq.com",
|
author_email="g1879@qq.com",
|
||||||
description="A module that integrates selenium and requests session, encapsulates common page operations.",
|
description="A module that integrates selenium and requests session, encapsulates common page operations.",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user