diff --git a/DrissionPage/chromium_element.py b/DrissionPage/chromium_element.py index d365af3..a7964f8 100644 --- a/DrissionPage/chromium_element.py +++ b/DrissionPage/chromium_element.py @@ -125,54 +125,40 @@ class ChromiumElement(DrissionElement): return {'height': 0, 'width': 0} @property - def location(self) -> dict: - """返回元素左上角的绝对坐标""" - js = '''function(){ - function getElementPagePosition(element){ - 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])} + def client_location(self) -> Union[dict, None]: + """返回元素左上角在视口中的坐标""" + m = self._get_client_rect('border') + return {'x': m[0], 'y': m[1]} if m else None @property - def client_location(self) -> dict: - """返回元素左上角在视口中的坐标""" - js = 'return this.getBoundingClientRect().left.toString()+" "+this.getBoundingClientRect().top.toString();' - xy = self.run_script(js) - x, y = xy.split(' ') - return {'x': int(x.split('.')[0]), 'y': int(y.split('.')[0])} + def client_midpoint(self) -> Union[dict, None]: + """返回元素中间点在视口中的坐标""" + m = self._get_client_rect('border') + return {'x': m[2] - m[0], 'y': m[5] - m[1]} if m else None + + @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 def midpoint(self) -> dict: """返回元素中间点的绝对坐标""" - loc = self.location - size = self.size - lx = loc['x'] + size['width'] // 2 - ly = loc['y'] + size['height'] // 2 - return {'x': lx, 'y': ly} + cl = self.client_midpoint + return self._get_absolute_rect(cl['x'], cl['y']) if cl else None @property - def client_midpoint(self) -> dict: - """返回元素中间点在视口中的坐标""" - loc = self.client_location - size = self.size - cx = loc['x'] + size['width'] // 2 - cy = loc['y'] + size['height'] // 2 - return {'x': cx, 'y': cy} + def _client_click_point(self) -> Union[dict, None]: + """返回元素左上角可接受点击的点视口坐标""" + m = self._get_client_rect('padding') + return {'x': m[0], 'y': m[1]} if m else None + + @property + 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 def shadow_root(self) -> Union[None, 'ChromiumShadowRootElement']: @@ -336,8 +322,8 @@ class ChromiumElement(DrissionElement): @property def is_in_viewport(self) -> bool: - """返回元素是否出现在视口中,以元素中点为判断""" - loc = self.midpoint + """返回元素是否出现在视口中,以元素可以接受点击的点为判断""" + loc = self.location return _location_in_viewport(self.page, loc['x'], loc['y']) def attr(self, attr: str) -> Union[str, None]: @@ -611,17 +597,23 @@ class ChromiumElement(DrissionElement): else: 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 如果遇到遮挡,会重新尝试点击直到超时,若都失败就改用js点击 \n :param by_js: 是否用js点击,为True时直接用js点击,为False时重试失败也不会改用js - :param timeout: 尝试点击的超时时间,不指定则使用父页面的超时时间 + :param retry: 遇到其它元素遮挡时,是否重试 + :param timeout: 尝试点击的超时时间,不指定则使用父页面的超时时间,retry为True时才生效 :return: 是否点击成功 """ - def do_it(cx, cy, lx, ly) -> bool: - r = self.page.driver.DOM.getNodeForLocation(x=lx + 1, y=ly + 1) - if r.get('nodeId') != self._node_id: + def do_it(cx, cy, lx, ly) -> Union[None, bool]: + """无遮挡返回True,有遮挡返回False,无元素返回None""" + 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 self._click(cx, cy) @@ -630,25 +622,28 @@ class ChromiumElement(DrissionElement): if not by_js: self.page.scroll_to_see(self) if self.is_in_viewport: - midpoint = self.midpoint - client_midpoint = self.client_midpoint - client_x = client_midpoint['x'] - client_y = client_midpoint['y'] - loc_x = midpoint['x'] - loc_y = midpoint['y'] + client_point = self._client_click_point + if client_point: + loc_point = self._click_point + client_x = client_point['x'] + client_y = client_point['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) + if click: + return True - if click: - return True + timeout = timeout if timeout is not None else self.page.timeout + 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: - js = 'this.click();' - self.run_script(js) + self.run_script('this.click();') return True return False @@ -657,7 +652,7 @@ class ChromiumElement(DrissionElement): offset_x: Union[int, str] = None, offset_y: Union[int, str] = None, button: str = 'left') -> None: - """带偏移量点击本元素,相对于左上角坐标。不传入x或y值时点击元素中点 \n + """带偏移量点击本元素,相对于左上角坐标。不传入x或y值时点击元素左上角可接受点击的点 \n :param offset_x: 相对元素左上角坐标的x轴偏移量 :param offset_y: 相对元素左上角坐标的y轴偏移量 :param button: 左键还是右键 @@ -669,10 +664,10 @@ class ChromiumElement(DrissionElement): def r_click(self) -> None: """右键单击""" self.page.scroll_to_see(self) - xy = self.client_midpoint + xy = self._client_click_point 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 :param offset_x: 相对元素左上角坐标的x轴偏移量 :param offset_y: 相对元素左上角坐标的y轴偏移量 @@ -815,6 +810,23 @@ class ChromiumElement(DrissionElement): t = self.run_script(js) 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): """ChromiumShadowRootElement是用于处理ShadowRoot的类,使用方法和ChromiumElement基本一致""" @@ -1199,6 +1211,11 @@ class ChromiumBase(BasePage): self._wait_loading() return self._tab_obj + @property + def is_loading(self) -> bool: + """返回页面是否正在加载状态""" + return self._is_loading + @property def url(self) -> str: """返回当前页面url""" @@ -1992,7 +2009,10 @@ def _run_script(page_or_ele, script: str, as_expr: bool = False, timeout: float if 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): @@ -2076,7 +2096,7 @@ def _send_key(ele: ChromiumElement, modifier: int, key: str) -> None: 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 offset_x: 偏移量x @@ -2084,14 +2104,14 @@ def _offset_scroll(ele, offset_x: int, offset_y: int) -> tuple: :return: 视口中的坐标 """ location = ele.location - midpoint = ele.midpoint + midpoint = ele._click_point lx = location['x'] + offset_x if offset_x else midpoint['x'] ly = location['y'] + offset_y if offset_y else midpoint['y'] if not _location_in_viewport(ele.page, lx, ly): ele.page.scroll.to_location(lx, ly) cl = ele.client_location - cm = ele.client_midpoint + cm = ele._client_click_point cx = cl['x'] + offset_x if offset_x else cm['x'] cy = cl['y'] + offset_y if offset_y else cm['y'] return cx, cy diff --git a/DrissionPage/chromium_page.py b/DrissionPage/chromium_page.py index 02fac21..715522f 100644 --- a/DrissionPage/chromium_page.py +++ b/DrissionPage/chromium_page.py @@ -217,10 +217,10 @@ class ChromiumPage(ChromiumBase): """ tabs = self.tabs if not tab_id: + tab_id = self.main_tab + if tab_id not in tabs: tab_id = tabs[0] - elif tab_id == 'main': - tab_id = self._main_tab - if tab_id == self.tab_id or tab_id not in tabs: + if tab_id == self.tab_id: return if activate: @@ -238,7 +238,13 @@ class ChromiumPage(ChromiumBase): :return: None """ 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: tabs = all_tabs - tabs @@ -255,6 +261,9 @@ class ChromiumPage(ChromiumBase): while len(self.tabs) != end_len: pass + if self.main_tab in tabs: + self.main_tab = self.tabs[0] + self.to_tab() def close_other_tabs(self, tab_ids: Union[str, List[str], Tuple[str]] = None) -> None: diff --git a/setup.py b/setup.py index f6caf52..1842bbc 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.0.5", + version="3.0.6", author="g1879", author_email="g1879@qq.com", description="A module that integrates selenium and requests session, encapsulates common page operations.",