mirror of
https://gitee.com/g1879/DrissionPage.git
synced 2024-12-10 04:00:23 +08:00
增加select.all();优化select逻辑
This commit is contained in:
parent
c593981b13
commit
ad7f871e8e
@ -1838,7 +1838,7 @@ class ChromiumSelect(object):
|
|||||||
@property
|
@property
|
||||||
def options(self):
|
def options(self):
|
||||||
"""返回所有选项元素组成的列表"""
|
"""返回所有选项元素组成的列表"""
|
||||||
return self._ele.eles('tag:option')
|
return self._ele.eles('xpath://option')
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def selected_option(self):
|
def selected_option(self):
|
||||||
@ -1855,58 +1855,56 @@ class ChromiumSelect(object):
|
|||||||
"""
|
"""
|
||||||
return [x for x in self.options if x.states.is_selected]
|
return [x for x in self.options if x.states.is_selected]
|
||||||
|
|
||||||
|
def all(self):
|
||||||
|
"""全选"""
|
||||||
|
if not self.is_multi:
|
||||||
|
raise TypeError("只能在多选菜单执行此操作。")
|
||||||
|
return self._by_loc('tag:option', 1, False)
|
||||||
|
|
||||||
|
def invert(self):
|
||||||
|
"""反选"""
|
||||||
|
if not self.is_multi:
|
||||||
|
raise TypeError("只能对多项选框执行反选。")
|
||||||
|
for i in self.options:
|
||||||
|
i.click(by_js=True)
|
||||||
|
|
||||||
def clear(self):
|
def clear(self):
|
||||||
"""清除所有已选项"""
|
"""清除所有已选项"""
|
||||||
if not self.is_multi:
|
if not self.is_multi:
|
||||||
raise NotImplementedError("只能在多选菜单执行此操作。")
|
raise TypeError("只能在多选菜单执行此操作。")
|
||||||
for opt in self.options:
|
return self._by_loc('tag:option', 1, True)
|
||||||
if opt.states.is_selected:
|
|
||||||
opt.click(by_js=True)
|
|
||||||
|
|
||||||
def by_text(self, text, timeout=None):
|
def by_text(self, text, timeout=None):
|
||||||
"""此方法用于根据text值选择项。当元素是多选列表时,可以接收list或tuple
|
"""此方法用于根据text值选择项。当元素是多选列表时,可以接收list或tuple
|
||||||
:param text: text属性值,传入list或tuple可选择多项
|
:param text: text属性值,传入list或tuple可选择多项
|
||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,为None默认使用页面超时时间
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(text, 'text', False, timeout)
|
return self._select(text, 'text', False, timeout)
|
||||||
|
|
||||||
def by_value(self, value, timeout=None):
|
def by_value(self, value, timeout=None):
|
||||||
"""此方法用于根据value值选择项。当元素是多选列表时,可以接收list或tuple
|
"""此方法用于根据value值选择项。当元素是多选列表时,可以接收list或tuple
|
||||||
:param value: value属性值,传入list或tuple可选择多项
|
:param value: value属性值,传入list或tuple可选择多项
|
||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,为None默认使用页面超时时间
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(value, 'value', False, timeout)
|
return self._select(value, 'value', False, timeout)
|
||||||
|
|
||||||
def by_index(self, index, timeout=None):
|
def by_index(self, index, timeout=None):
|
||||||
"""此方法用于根据index值选择项。当元素是多选列表时,可以接收list或tuple
|
"""此方法用于根据index值选择项。当元素是多选列表时,可以接收list或tuple
|
||||||
:param index: 序号,0开始,传入list或tuple可选择多项
|
:param index: 序号,0开始,传入list或tuple可选择多项
|
||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,为None默认使用页面超时时间
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(index, 'index', False, timeout)
|
return self._select(index, 'index', False, timeout)
|
||||||
|
|
||||||
def by_loc(self, loc, timeout=None):
|
def by_loc(self, loc, timeout=None):
|
||||||
"""用定位符选择要选择的项
|
"""用定位符选择指定的项
|
||||||
:param loc: 定位符
|
:param loc: 定位符
|
||||||
:param timeout: 超时时间
|
:param timeout: 超时时间
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
eles = self._ele.eles(loc, timeout)
|
return self._by_loc(loc, timeout)
|
||||||
if not eles:
|
|
||||||
return False
|
|
||||||
|
|
||||||
if self.is_multi:
|
|
||||||
for ele in eles:
|
|
||||||
ele.run_js(f'this.selected=true;')
|
|
||||||
return True
|
|
||||||
|
|
||||||
eles[0].run_js(f'this.selected=true;')
|
|
||||||
return True
|
|
||||||
|
|
||||||
def cancel_by_text(self, text, timeout=None):
|
def cancel_by_text(self, text, timeout=None):
|
||||||
"""此方法用于根据text值取消选择项。当元素是多选列表时,可以接收list或tuple
|
"""此方法用于根据text值取消选择项。当元素是多选列表时,可以接收list或tuple
|
||||||
@ -1914,7 +1912,6 @@ class ChromiumSelect(object):
|
|||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,不输入默认实用页面超时时间
|
||||||
:return: 是否取消成功
|
:return: 是否取消成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(text, 'text', True, timeout)
|
return self._select(text, 'text', True, timeout)
|
||||||
|
|
||||||
def cancel_by_value(self, value, timeout=None):
|
def cancel_by_value(self, value, timeout=None):
|
||||||
@ -1923,7 +1920,6 @@ class ChromiumSelect(object):
|
|||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,不输入默认实用页面超时时间
|
||||||
:return: 是否取消成功
|
:return: 是否取消成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(value, 'value', True, timeout)
|
return self._select(value, 'value', True, timeout)
|
||||||
|
|
||||||
def cancel_by_index(self, index, timeout=None):
|
def cancel_by_index(self, index, timeout=None):
|
||||||
@ -1932,85 +1928,105 @@ class ChromiumSelect(object):
|
|||||||
:param timeout: 超时时间,不输入默认实用页面超时时间
|
:param timeout: 超时时间,不输入默认实用页面超时时间
|
||||||
:return: 是否取消成功
|
:return: 是否取消成功
|
||||||
"""
|
"""
|
||||||
timeout = timeout if timeout is not None else self._ele.page.timeout
|
|
||||||
return self._select(index, 'index', True, timeout)
|
return self._select(index, 'index', True, timeout)
|
||||||
|
|
||||||
def invert(self):
|
def cancel_by_loc(self, loc, timeout=None):
|
||||||
"""反选"""
|
"""用定位符取消选择指定的项
|
||||||
if not self.is_multi:
|
:param loc: 定位符
|
||||||
raise NotImplementedError("只能对多项选框执行反选。")
|
:param timeout: 超时时间
|
||||||
|
|
||||||
for i in self.options:
|
|
||||||
i.click(by_js=True)
|
|
||||||
|
|
||||||
def _select(self, text_value_index=None, para_type='text', deselect=False, timeout=None):
|
|
||||||
"""选定或取消选定下拉列表中子元素
|
|
||||||
:param text_value_index: 根据文本、值选或序号择选项,若允许多选,传入list或tuple可多选
|
|
||||||
:param para_type: 参数类型,可选 'text'、'value'、'index'
|
|
||||||
:param deselect: 是否取消选择
|
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
if not self.is_multi and isinstance(text_value_index, (list, tuple)):
|
return self._by_loc(loc, timeout, True)
|
||||||
raise TypeError('单选下拉列表不能传入list和tuple')
|
|
||||||
|
|
||||||
def do_select():
|
def _by_loc(self, loc, timeout=None, cancel=False):
|
||||||
if para_type == 'text':
|
"""用定位符取消选择指定的项
|
||||||
ele = self._ele._ele(f'tx={text_value_index}', timeout=0, raise_err=False)
|
:param loc: 定位符
|
||||||
elif para_type == 'value':
|
:param timeout: 超时时间
|
||||||
ele = self._ele._ele(f'@value={text_value_index}', timeout=0, raise_err=False)
|
:param cancel: 是否取消选择
|
||||||
elif para_type == 'index':
|
:return: 是否选择成功
|
||||||
ele = self._ele._ele(f'x:.//option[{int(text_value_index)}]', timeout=0, raise_err=False)
|
"""
|
||||||
else:
|
eles = self._ele.eles(loc, timeout)
|
||||||
raise ValueError('para_type参数只能传入"text"、"value"或"index"。')
|
if not eles:
|
||||||
|
return False
|
||||||
if not ele:
|
|
||||||
return False
|
|
||||||
|
|
||||||
js = 'false' if deselect else 'true'
|
|
||||||
ele.run_js(f'this.selected={js};')
|
|
||||||
|
|
||||||
|
mode = 'false' if cancel else 'true'
|
||||||
|
if self.is_multi:
|
||||||
|
for ele in eles:
|
||||||
|
ele.run_js(f'this.selected={mode};')
|
||||||
return True
|
return True
|
||||||
|
|
||||||
if isinstance(text_value_index, (str, int)):
|
eles[0].run_js(f'this.selected={mode};')
|
||||||
ok = do_select()
|
return True
|
||||||
end_time = perf_counter() + timeout
|
|
||||||
while not ok and perf_counter() < end_time:
|
|
||||||
sleep(.2)
|
|
||||||
ok = do_select()
|
|
||||||
return ok
|
|
||||||
|
|
||||||
elif isinstance(text_value_index, (list, tuple)):
|
def _select(self, condition, para_type='text', cancel=False, timeout=None):
|
||||||
return self._select_multi(text_value_index, para_type, deselect)
|
"""选定或取消选定下拉列表中子元素
|
||||||
|
:param condition: 根据文本、值选或序号择选项,若允许多选,传入list或tuple可多选
|
||||||
else:
|
|
||||||
raise TypeError('只能传入str、int、list和tuple类型。')
|
|
||||||
|
|
||||||
def _select_multi(self,
|
|
||||||
text_value_index=None,
|
|
||||||
para_type='text',
|
|
||||||
deselect=False):
|
|
||||||
"""选定或取消选定下拉列表中多个子元素
|
|
||||||
:param text_value_index: 根据文本、值选或序号择选多项
|
|
||||||
:param para_type: 参数类型,可选 'text'、'value'、'index'
|
:param para_type: 参数类型,可选 'text'、'value'、'index'
|
||||||
:param deselect: 是否取消选择
|
:param cancel: 是否取消选择
|
||||||
:return: 是否选择成功
|
:return: 是否选择成功
|
||||||
"""
|
"""
|
||||||
if para_type not in ('text', 'value', 'index'):
|
if not self.is_multi and isinstance(condition, (list, tuple)):
|
||||||
raise ValueError('para_type参数只能传入“text”、“value”或“index”')
|
raise TypeError('单选列表只能传入str格式。')
|
||||||
|
|
||||||
if not isinstance(text_value_index, (list, tuple)):
|
mode = 'false' if cancel else 'true'
|
||||||
raise TypeError('只能传入list或tuple类型。')
|
timeout = timeout if timeout is not None else self._ele.page.timeout
|
||||||
|
condition = {condition} if isinstance(condition, str) else set(condition)
|
||||||
|
|
||||||
success = True
|
if para_type in ('text', 'value'):
|
||||||
for i in text_value_index:
|
return self._text_value(condition, para_type, mode, timeout)
|
||||||
if not isinstance(i, (int, str)):
|
elif para_type == 'index':
|
||||||
raise TypeError('列表只能由str或int组成')
|
return self._index(condition, mode, timeout)
|
||||||
|
|
||||||
p = 'index' if isinstance(i, int) else para_type
|
def _text_value(self, condition, para_type, mode, timeout):
|
||||||
if not self._select(i, p, deselect):
|
"""执行text和value搜索
|
||||||
success = False
|
:param condition: 条件set
|
||||||
|
:param para_type: 参数类型,可选 'text'、'value'
|
||||||
|
:param mode: 'true' 或 'false'
|
||||||
|
:param timeout: 超时时间
|
||||||
|
:return: 是否选择成功
|
||||||
|
"""
|
||||||
|
ok = False
|
||||||
|
text_len = len(condition)
|
||||||
|
eles = []
|
||||||
|
end_time = perf_counter() + timeout
|
||||||
|
while perf_counter() < end_time:
|
||||||
|
if para_type == 'text':
|
||||||
|
eles = [i for i in self.options if i.text in condition]
|
||||||
|
elif para_type == 'value':
|
||||||
|
eles = [i for i in self.options if i.attr('value') in condition]
|
||||||
|
|
||||||
return success
|
if len(eles) >= text_len:
|
||||||
|
ok = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if ok:
|
||||||
|
for i in eles:
|
||||||
|
i.run_js(f'this.selected={mode};')
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
def _index(self, condition, mode, timeout):
|
||||||
|
"""执行index搜索
|
||||||
|
:param condition: 条件set
|
||||||
|
:param mode: 'true' 或 'false'
|
||||||
|
:param timeout: 超时时间
|
||||||
|
:return: 是否选择成功
|
||||||
|
"""
|
||||||
|
ok = False
|
||||||
|
condition = [int(i) for i in condition]
|
||||||
|
text_len = max(condition)
|
||||||
|
end_time = perf_counter() + timeout
|
||||||
|
while perf_counter() < end_time:
|
||||||
|
if len(self.options) >= text_len:
|
||||||
|
ok = True
|
||||||
|
break
|
||||||
|
|
||||||
|
if ok:
|
||||||
|
eles = self.options
|
||||||
|
for i in condition:
|
||||||
|
eles[i - 1].run_js(f'this.selected={mode};')
|
||||||
|
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
||||||
class ChromiumElementWaiter(object):
|
class ChromiumElementWaiter(object):
|
||||||
|
@ -505,6 +505,8 @@ class ChromiumSelect(object):
|
|||||||
|
|
||||||
def clear(self) -> None: ...
|
def clear(self) -> None: ...
|
||||||
|
|
||||||
|
def all(self) -> None: ...
|
||||||
|
|
||||||
def by_text(self, text: Union[str, list, tuple], timeout: float = None) -> bool: ...
|
def by_text(self, text: Union[str, list, tuple], timeout: float = None) -> bool: ...
|
||||||
|
|
||||||
def by_value(self, value: Union[str, list, tuple], timeout: float = None) -> bool: ...
|
def by_value(self, value: Union[str, list, tuple], timeout: float = None) -> bool: ...
|
||||||
@ -519,18 +521,21 @@ class ChromiumSelect(object):
|
|||||||
|
|
||||||
def cancel_by_index(self, index: Union[int, list, tuple], timeout: float = None) -> bool: ...
|
def cancel_by_index(self, index: Union[int, list, tuple], timeout: float = None) -> bool: ...
|
||||||
|
|
||||||
|
def cancel_by_loc(self, loc: Union[str, Tuple[str, str]], timeout: float = None) -> bool: ...
|
||||||
|
|
||||||
def invert(self) -> None: ...
|
def invert(self) -> None: ...
|
||||||
|
|
||||||
|
def _by_loc(self, loc: Union[str, Tuple[str, str]], timeout: float = None, cancel: bool = False) -> bool: ...
|
||||||
|
|
||||||
def _select(self,
|
def _select(self,
|
||||||
text_value_index: Union[str, int, list, tuple] = None,
|
condition: Union[str, int, list, tuple] = None,
|
||||||
para_type: str = 'text',
|
para_type: str = 'text',
|
||||||
deselect: bool = False,
|
cancel: bool = False,
|
||||||
timeout: float = None) -> bool: ...
|
timeout: float = None) -> bool: ...
|
||||||
|
|
||||||
def _select_multi(self,
|
def _text_value(self, condition: set, para_type: str, mode: str, timeout: float) -> bool: ...
|
||||||
text_value_index: Union[list, tuple] = None,
|
|
||||||
para_type: str = 'text',
|
def _index(self, condition: set, mode: str, timeout: float) -> bool: ...
|
||||||
deselect: bool = False) -> bool: ...
|
|
||||||
|
|
||||||
|
|
||||||
class ChromiumElementWaiter(object):
|
class ChromiumElementWaiter(object):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user