mirror of
https://gitee.com/g1879/DrissionPage.git
synced 2024-12-10 04:00:23 +08:00
click()增加wait_stop参数;启动时自动关闭浏览器隐私声明;动作链移动增加时长参数
This commit is contained in:
parent
27bea5f579
commit
9fb0f84507
@ -666,41 +666,15 @@ class ChromiumElement(DrissionElement):
|
|||||||
def drag_to(self, ele_or_loc, duration=.5):
|
def drag_to(self, ele_or_loc, duration=.5):
|
||||||
"""拖拽当前元素,目标为另一个元素或坐标元组(x, y)
|
"""拖拽当前元素,目标为另一个元素或坐标元组(x, y)
|
||||||
:param ele_or_loc: 另一个元素或坐标元组,坐标为元素中点的坐标
|
:param ele_or_loc: 另一个元素或坐标元组,坐标为元素中点的坐标
|
||||||
:param duration: 拖动用时,传入0即瞬间到j达
|
:param duration: 拖动用时,传入0即瞬间到达
|
||||||
:return: None
|
:return: None
|
||||||
"""
|
"""
|
||||||
# x, y:目标点坐标
|
|
||||||
if isinstance(ele_or_loc, ChromiumElement):
|
if isinstance(ele_or_loc, ChromiumElement):
|
||||||
target_x, target_y = ele_or_loc.locations.midpoint
|
ele_or_loc = ele_or_loc.locations.midpoint
|
||||||
elif isinstance(ele_or_loc, (list, tuple)):
|
elif not isinstance(ele_or_loc, (list, tuple)):
|
||||||
target_x, target_y = ele_or_loc
|
|
||||||
else:
|
|
||||||
raise TypeError('需要ChromiumElement对象或坐标。')
|
raise TypeError('需要ChromiumElement对象或坐标。')
|
||||||
|
|
||||||
current_x, current_y = self.locations.midpoint
|
self.page.actions.hold(self).move_to(ele_or_loc, duration=duration).release()
|
||||||
width = target_x - current_x
|
|
||||||
height = target_y - current_y
|
|
||||||
|
|
||||||
duration = .02 if duration < .02 else duration
|
|
||||||
num = int(duration * 50)
|
|
||||||
|
|
||||||
# 将要经过的点存入列表
|
|
||||||
points = [(int(current_x + i * (width / num)), int(current_y + i * (height / num))) for i in range(1, num)]
|
|
||||||
points.append((target_x, target_y))
|
|
||||||
|
|
||||||
from .._units.action_chains import ActionChains
|
|
||||||
actions = ActionChains(self.page)
|
|
||||||
actions.hold(self)
|
|
||||||
|
|
||||||
# 逐个访问要经过的点
|
|
||||||
for x, y in points:
|
|
||||||
t = perf_counter()
|
|
||||||
actions.move(x - current_x, y - current_y)
|
|
||||||
current_x, current_y = x, y
|
|
||||||
ss = .02 - perf_counter() + t
|
|
||||||
if ss > 0:
|
|
||||||
sleep(ss)
|
|
||||||
actions.release()
|
|
||||||
|
|
||||||
def _get_obj_id(self, node_id=None, backend_id=None):
|
def _get_obj_id(self, node_id=None, backend_id=None):
|
||||||
"""根据传入node id或backend id获取js中的object id
|
"""根据传入node id或backend id获取js中的object id
|
||||||
|
@ -84,7 +84,13 @@ class ChromiumBase(BasePage):
|
|||||||
self._scroll = None
|
self._scroll = None
|
||||||
|
|
||||||
if not tab_id:
|
if not tab_id:
|
||||||
tab_id = self.browser.tabs[0]
|
tabs = self.browser.driver.get(f'http://{self.address}/json').json()
|
||||||
|
tabs = [(i['id'], i['url']) for i in tabs if i['type'] == 'page' and not i['url'].startswith('devtools://')]
|
||||||
|
if tabs[0][1] == 'chrome://privacy-sandbox-dialog/notice' and len(tabs) > 1:
|
||||||
|
tab_id = tabs[1][0]
|
||||||
|
close_privacy_dialog(self, tabs[0][0])
|
||||||
|
else:
|
||||||
|
tab_id = tabs[0][0]
|
||||||
|
|
||||||
self._driver_init(tab_id)
|
self._driver_init(tab_id)
|
||||||
if self.ready_state == 'complete' and self._ready_state is None:
|
if self.ready_state == 'complete' and self._ready_state is None:
|
||||||
@ -100,9 +106,6 @@ class ChromiumBase(BasePage):
|
|||||||
"""
|
"""
|
||||||
self._is_loading = True
|
self._is_loading = True
|
||||||
self._driver = self.browser._get_driver(tab_id)
|
self._driver = self.browser._get_driver(tab_id)
|
||||||
if self._driver.run('Target.getTargetInfo',
|
|
||||||
targetId=self._target_id)['targetInfo']['url'] == 'chrome://privacy-sandbox-dialog/notice':
|
|
||||||
self._driver = close_privacy_dialog(self)
|
|
||||||
|
|
||||||
self._alert = Alert()
|
self._alert = Alert()
|
||||||
self._driver.set_callback('Page.javascriptDialogOpening', self._on_alert_open)
|
self._driver.set_callback('Page.javascriptDialogOpening', self._on_alert_open)
|
||||||
@ -1038,33 +1041,33 @@ class Alert(object):
|
|||||||
self.response_text = None
|
self.response_text = None
|
||||||
|
|
||||||
|
|
||||||
def close_privacy_dialog(page):
|
def close_privacy_dialog(page, tid):
|
||||||
"""关闭隐私声明弹窗
|
"""关闭隐私声明弹窗
|
||||||
:param page: ChromiumBase对象
|
:param page: ChromiumBase对象
|
||||||
|
:param tid: tab id
|
||||||
:return: ChromiumDriver对象
|
:return: ChromiumDriver对象
|
||||||
"""
|
"""
|
||||||
tid = page.tab_id
|
try:
|
||||||
page._driver.run('Runtime.enable')
|
driver = page.browser._get_driver(tid)
|
||||||
page._driver.run('DOM.enable')
|
driver.run('Runtime.enable')
|
||||||
page._driver.run('DOM.getDocument')
|
driver.run('DOM.enable')
|
||||||
sid = page._driver.run('DOM.performSearch', query='//*[name()="privacy-sandbox-notice-dialog-app"]',
|
driver.run('DOM.getDocument')
|
||||||
includeUserAgentShadowDOM=True)['searchId']
|
sid = driver.run('DOM.performSearch', query='//*[name()="privacy-sandbox-notice-dialog-app"]',
|
||||||
r = page._driver.run('DOM.getSearchResults', searchId=sid, fromIndex=0, toIndex=1)['nodeIds'][0]
|
includeUserAgentShadowDOM=True)['searchId']
|
||||||
while True:
|
r = driver.run('DOM.getSearchResults', searchId=sid, fromIndex=0, toIndex=1)['nodeIds'][0]
|
||||||
try:
|
end_time = perf_counter() + 3
|
||||||
r = page._driver.run('DOM.describeNode', nodeId=r)['node']['shadowRoots'][0]['backendNodeId']
|
while perf_counter() < end_time:
|
||||||
break
|
try:
|
||||||
except KeyError:
|
r = driver.run('DOM.describeNode', nodeId=r)['node']['shadowRoots'][0]['backendNodeId']
|
||||||
pass
|
break
|
||||||
page._driver.run('DOM.discardSearchResults', searchId=sid)
|
except KeyError:
|
||||||
r = page._driver.run('DOM.resolveNode', backendNodeId=r)['object']['objectId']
|
pass
|
||||||
r = page._driver.run('Runtime.callFunctionOn', objectId=r,
|
driver.run('DOM.discardSearchResults', searchId=sid)
|
||||||
functionDeclaration='function()'
|
r = driver.run('DOM.resolveNode', backendNodeId=r)['object']['objectId']
|
||||||
'{return this.getElementById("ackButton");}')['result']['objectId']
|
r = driver.run('Runtime.callFunctionOn', objectId=r,
|
||||||
page._driver.run('Runtime.callFunctionOn', objectId=r, functionDeclaration='function(){return this.click();}')
|
functionDeclaration='function(){return this.getElementById("ackButton");}')['result']['objectId']
|
||||||
while True:
|
driver.run('Runtime.callFunctionOn', objectId=r, functionDeclaration='function(){return this.click();}')
|
||||||
new_tid = page.browser.tabs[0]
|
driver.close()
|
||||||
if new_tid != tid:
|
|
||||||
break
|
except:
|
||||||
sleep(.1)
|
pass
|
||||||
return page.browser._get_driver(new_tid)
|
|
||||||
|
@ -33,6 +33,7 @@ class ChromiumFrame(ChromiumBase):
|
|||||||
self.doc_ele: ChromiumElement = ...
|
self.doc_ele: ChromiumElement = ...
|
||||||
self._states: ElementStates = ...
|
self._states: ElementStates = ...
|
||||||
self._ids: FrameIds = ...
|
self._ids: FrameIds = ...
|
||||||
|
self._is_init_get_doc: bool = ...
|
||||||
|
|
||||||
def __call__(self,
|
def __call__(self,
|
||||||
loc_or_str: Union[Tuple[str, str], str],
|
loc_or_str: Union[Tuple[str, str], str],
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
@Author : g1879
|
@Author : g1879
|
||||||
@Contact : g1879@qq.com
|
@Contact : g1879@qq.com
|
||||||
"""
|
"""
|
||||||
from time import sleep
|
from time import sleep, perf_counter
|
||||||
|
|
||||||
from .._commons.keys import modifierBit, keyDescriptionForString
|
from .._commons.keys import modifierBit, keyDescriptionForString
|
||||||
from .._commons.web import location_in_viewport
|
from .._commons.web import location_in_viewport
|
||||||
@ -22,12 +22,13 @@ class ActionChains:
|
|||||||
self.curr_x = 0 # 视口坐标
|
self.curr_x = 0 # 视口坐标
|
||||||
self.curr_y = 0
|
self.curr_y = 0
|
||||||
|
|
||||||
def move_to(self, ele_or_loc, offset_x=0, offset_y=0):
|
def move_to(self, ele_or_loc, offset_x=0, offset_y=0, duration=.5):
|
||||||
"""鼠标移动到元素中点,或页面上的某个绝对坐标。可设置偏移量
|
"""鼠标移动到元素中点,或页面上的某个绝对坐标。可设置偏移量
|
||||||
当带偏移量时,偏移量相对于元素左上角坐标
|
当带偏移量时,偏移量相对于元素左上角坐标
|
||||||
:param ele_or_loc: 元素对象、绝对坐标或文本定位符,坐标为tuple(int, int)形式
|
:param ele_or_loc: 元素对象、绝对坐标或文本定位符,坐标为tuple(int, int)形式
|
||||||
:param offset_x: 偏移量x
|
:param offset_x: 偏移量x
|
||||||
:param offset_y: 偏移量y
|
:param offset_y: 偏移量y
|
||||||
|
:param duration: 拖动用时,传入0即瞬间到达
|
||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
is_loc = False
|
is_loc = False
|
||||||
@ -50,7 +51,7 @@ class ActionChains:
|
|||||||
clientHeight = self.page.run_js('return document.body.clientHeight;')
|
clientHeight = self.page.run_js('return document.body.clientHeight;')
|
||||||
self.page.scroll.to_location(lx - clientWidth // 2, ly - clientHeight // 2)
|
self.page.scroll.to_location(lx - clientWidth // 2, ly - clientHeight // 2)
|
||||||
|
|
||||||
# # 这样设计为了应付那些不随滚动条滚动的元素
|
# 这样设计为了应付那些不随滚动条滚动的元素
|
||||||
if is_loc:
|
if is_loc:
|
||||||
cx, cy = location_to_client(self.page, lx, ly)
|
cx, cy = location_to_client(self.page, lx, ly)
|
||||||
else:
|
else:
|
||||||
@ -59,21 +60,35 @@ class ActionChains:
|
|||||||
cx = x + offset_x
|
cx = x + offset_x
|
||||||
cy = y + offset_y
|
cy = y + offset_y
|
||||||
|
|
||||||
self._dr.run('Input.dispatchMouseEvent', type='mouseMoved', x=cx, y=cy, modifiers=self.modifier)
|
ox = cx - self.curr_x
|
||||||
self.curr_x = cx
|
oy = cy - self.curr_y
|
||||||
self.curr_y = cy
|
self.move(ox, oy, duration)
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def move(self, offset_x=0, offset_y=0):
|
def move(self, offset_x=0, offset_y=0, duration=.5):
|
||||||
"""鼠标相对当前位置移动若干位置
|
"""鼠标相对当前位置移动若干位置
|
||||||
:param offset_x: 偏移量x
|
:param offset_x: 偏移量x
|
||||||
:param offset_y: 偏移量y
|
:param offset_y: 偏移量y
|
||||||
|
:param duration: 拖动用时,传入0即瞬间到达
|
||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
self.curr_x += offset_x
|
duration = .02 if duration < .02 else duration
|
||||||
self.curr_y += offset_y
|
num = int(duration * 50)
|
||||||
self._dr.run('Input.dispatchMouseEvent', type='mouseMoved', x=self.curr_x, y=self.curr_y,
|
|
||||||
modifiers=self.modifier)
|
points = [(self.curr_x + i * (offset_x / num),
|
||||||
|
self.curr_y + i * (offset_y / num)) for i in range(1, num)]
|
||||||
|
points.append((self.curr_x + offset_x, self.curr_y + offset_y))
|
||||||
|
|
||||||
|
for x, y in points:
|
||||||
|
t = perf_counter()
|
||||||
|
self.curr_x = x
|
||||||
|
self.curr_y = y
|
||||||
|
self._dr.run('Input.dispatchMouseEvent', type='mouseMoved', x=self.curr_x, y=self.curr_y,
|
||||||
|
modifiers=self.modifier)
|
||||||
|
ss = .02 - perf_counter() + t
|
||||||
|
if ss > 0:
|
||||||
|
sleep(ss)
|
||||||
|
|
||||||
return self
|
return self
|
||||||
|
|
||||||
def click(self, on_ele=None):
|
def click(self, on_ele=None):
|
||||||
@ -122,7 +137,7 @@ class ActionChains:
|
|||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
if on_ele:
|
if on_ele:
|
||||||
self.move_to(on_ele)
|
self.move_to(on_ele, duration=0)
|
||||||
self._release('left')
|
self._release('left')
|
||||||
return self
|
return self
|
||||||
|
|
||||||
@ -140,7 +155,7 @@ class ActionChains:
|
|||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
if on_ele:
|
if on_ele:
|
||||||
self.move_to(on_ele)
|
self.move_to(on_ele, duration=0)
|
||||||
self._release('right')
|
self._release('right')
|
||||||
return self
|
return self
|
||||||
|
|
||||||
@ -158,7 +173,7 @@ class ActionChains:
|
|||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
if on_ele:
|
if on_ele:
|
||||||
self.move_to(on_ele)
|
self.move_to(on_ele, duration=0)
|
||||||
self._release('middle')
|
self._release('middle')
|
||||||
return self
|
return self
|
||||||
|
|
||||||
@ -170,7 +185,7 @@ class ActionChains:
|
|||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
if on_ele:
|
if on_ele:
|
||||||
self.move_to(on_ele)
|
self.move_to(on_ele, duration=0)
|
||||||
self._dr.run('Input.dispatchMouseEvent', type='mousePressed', button=button, clickCount=count,
|
self._dr.run('Input.dispatchMouseEvent', type='mousePressed', button=button, clickCount=count,
|
||||||
x=self.curr_x, y=self.curr_y, modifiers=self.modifier)
|
x=self.curr_x, y=self.curr_y, modifiers=self.modifier)
|
||||||
return self
|
return self
|
||||||
@ -192,7 +207,7 @@ class ActionChains:
|
|||||||
:return: self
|
:return: self
|
||||||
"""
|
"""
|
||||||
if on_ele:
|
if on_ele:
|
||||||
self.move_to(on_ele)
|
self.move_to(on_ele, duration=0)
|
||||||
self._dr.run('Input.dispatchMouseEvent', type='mouseWheel', x=self.curr_x, y=self.curr_y,
|
self._dr.run('Input.dispatchMouseEvent', type='mouseWheel', x=self.curr_x, y=self.curr_y,
|
||||||
deltaX=delta_x, deltaY=delta_y, modifiers=self.modifier)
|
deltaX=delta_x, deltaY=delta_y, modifiers=self.modifier)
|
||||||
return self
|
return self
|
||||||
|
@ -20,9 +20,9 @@ class ActionChains:
|
|||||||
self.curr_y: int = ...
|
self.curr_y: int = ...
|
||||||
|
|
||||||
def move_to(self, ele_or_loc: Union[ChromiumElement, Tuple[int, int], str],
|
def move_to(self, ele_or_loc: Union[ChromiumElement, Tuple[int, int], str],
|
||||||
offset_x: int = 0, offset_y: int = 0) -> ActionChains: ...
|
offset_x: int = 0, offset_y: int = 0, duration: float = .5) -> ActionChains: ...
|
||||||
|
|
||||||
def move(self, offset_x: int = 0, offset_y: int = 0) -> ActionChains: ...
|
def move(self, offset_x: int = 0, offset_y: int = 0, duration: float = .5) -> ActionChains: ...
|
||||||
|
|
||||||
def click(self, on_ele: Union[ChromiumElement, str] = None) -> ActionChains: ...
|
def click(self, on_ele: Union[ChromiumElement, str] = None) -> ActionChains: ...
|
||||||
|
|
||||||
|
@ -17,19 +17,21 @@ class Clicker(object):
|
|||||||
"""
|
"""
|
||||||
self._ele = ele
|
self._ele = ele
|
||||||
|
|
||||||
def __call__(self, by_js=False, timeout=1):
|
def __call__(self, by_js=False, timeout=2, wait_stop=True):
|
||||||
"""点击元素
|
"""点击元素
|
||||||
如果遇到遮挡,可选择是否用js点击
|
如果遇到遮挡,可选择是否用js点击
|
||||||
:param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击
|
:param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击
|
||||||
:param timeout: 模拟点击的超时时间,等待元素可见、不被遮挡、进入视口
|
:param timeout: 模拟点击的超时时间,等待元素可见、可用、进入视口
|
||||||
|
:param wait_stop: 是否等待元素运动结束再执行点击
|
||||||
:return: 是否点击成功
|
:return: 是否点击成功
|
||||||
"""
|
"""
|
||||||
return self.left(by_js, timeout)
|
return self.left(by_js, timeout, wait_stop)
|
||||||
|
|
||||||
def left(self, by_js=False, timeout=2):
|
def left(self, by_js=False, timeout=2, wait_stop=True):
|
||||||
"""点击元素,可选择是否用js点击
|
"""点击元素,可选择是否用js点击
|
||||||
:param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击
|
:param by_js: 是否用js点击,为None时先用模拟点击,遇到遮挡改用js,为True时直接用js点击,为False时只用模拟点击
|
||||||
:param timeout: 模拟点击的超时时间,等待元素可见、不被遮挡、进入视口
|
:param timeout: 模拟点击的超时时间,等待元素可见、可用、进入视口
|
||||||
|
:param wait_stop: 是否等待元素运动结束再执行点击
|
||||||
:return: 是否点击成功
|
:return: 是否点击成功
|
||||||
"""
|
"""
|
||||||
if not by_js: # 模拟点击
|
if not by_js: # 模拟点击
|
||||||
@ -53,7 +55,8 @@ class Clicker(object):
|
|||||||
rect = self._ele.states.has_rect
|
rect = self._ele.states.has_rect
|
||||||
sleep(.001)
|
sleep(.001)
|
||||||
|
|
||||||
self._ele.wait.stop_moving(timeout=end_time - perf_counter())
|
if wait_stop:
|
||||||
|
self._ele.wait.stop_moving(timeout=end_time - perf_counter())
|
||||||
if rect:
|
if rect:
|
||||||
self._ele.scroll.to_see()
|
self._ele.scroll.to_see()
|
||||||
rect = self._ele.locations.rect
|
rect = self._ele.locations.rect
|
||||||
|
@ -12,9 +12,9 @@ class Clicker(object):
|
|||||||
def __init__(self, ele: ChromiumElement):
|
def __init__(self, ele: ChromiumElement):
|
||||||
self._ele: ChromiumElement = ...
|
self._ele: ChromiumElement = ...
|
||||||
|
|
||||||
def __call__(self, by_js: Union[None, bool] = False, timeout: float = 1) -> bool: ...
|
def __call__(self, by_js: Union[None, bool] = False, timeout: float = 2, wait_stop: bool = True) -> bool: ...
|
||||||
|
|
||||||
def left(self, by_js: Union[None, bool] = False, timeout: float = 1) -> bool: ...
|
def left(self, by_js: Union[None, bool] = False, timeout: float = 2, wait_stop: bool = True) -> bool: ...
|
||||||
|
|
||||||
def right(self) -> None: ...
|
def right(self) -> None: ...
|
||||||
|
|
||||||
|
@ -241,4 +241,4 @@ class SelectElement(object):
|
|||||||
|
|
||||||
def _dispatch_change(self):
|
def _dispatch_change(self):
|
||||||
"""触发修改动作"""
|
"""触发修改动作"""
|
||||||
self._ele.run_js('this.dispatchEvent(new UIEvent("change"));')
|
self._ele.run_js('this.dispatchEvent(new Event("change", {bubbles: true}));')
|
||||||
|
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="4.0.0b6",
|
version="4.0.0b7",
|
||||||
author="g1879",
|
author="g1879",
|
||||||
author_email="g1879@qq.com",
|
author_email="g1879@qq.com",
|
||||||
description="Python based web automation tool. It can control the browser and send and receive data packets.",
|
description="Python based web automation tool. It can control the browser and send and receive data packets.",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user