找回密码
 会员注册
查看: 41|回复: 0

Playwright+Python系列之元素定位

[复制链接]

2万

主题

0

回帖

7万

积分

超级版主

积分
73425
发表于 2024-9-4 09:47:34 | 显示全部楼层 |阅读模式
一、常见元素定位定位器是Playwright自动等待和重试能力的核心部分。简而言之,定位器代表了一种随时在页面上查找元素的方法,以下是常用的内置定位器。 1、按角色定位按显式和隐式可访问性属性进行定位语法:page.get_by_role() Dom结构示例1:示例代码1:page.get_by_role("button", name="Sign in").click()说明:按名称为“Signin”button的角色找到元素。Dom结构示例2: 示例代码2expect(page.get_by_role("heading", name="Sign up")).to_be_visible()page.get_by_role("checkbox", name="Subscribe").check()page.get_by_role("button", name=re.compile("submit", re.IGNORECASE)).click()说明:角色定位器包括按钮、复选框、、链接、列表、表格等,并遵循ARIA角色、ARIA属性和可访问名称的 W3C 规范。请注意,许多html元素(如)都具有隐式定义的角色,该角色可由角色定位器识别。建议优先考虑角色定位器来定位元素,因为这是最接近用户和辅助技术感知页面的方式。page.get_by_label()通过关联标签的文本查找表单控件。2、按标签定位通过关联标签的文本查找表单控件语法:page.get_by_label() Dom结构示例:示例代码:page.get_by_label("Password").fill("secret") 3、按占位符定位语法:page.get_by_placeholder()Dom结构示例:示例代码:page.get_by_placeholder("name@example.com").fill("playwright@microsoft.com") 4、通过文本定位按占位符查找输入语法:page.get_by_text()Dom结构示例:示例代码:# 可以通过元素包含的文本找到该元素page.get_by_text("Welcome, John")# 设置完全匹配page.get_by_text("Welcome, John", exact=True)# 正则表达式匹配page.get_by_text(re.compile("welcome, john", re.IGNORECASE))说明:按文本匹配始终会规范化空格,即使完全匹配也是如此。例如,它将多个空格转换为一个空格,将换行符转换为空格,并忽略前导和尾随空格。建议使用文本定位器来查找非交互式元素,如div、span、p等。对于button、a、input等交互式元素,请使用角色定位器。5、通过替代文本定位通过其文本替代来定位元素(通常是图像),所有图片都应具有描述图像的alt属性。可以使用page.get_by_alt_text() 根据替代文本查找图片。语法:page.get_by_alt_text()Dom结构示例:示例代码:page.get_by_alt_text("playwright logo").click()说明:当元素支持替代文本(如img和area元素)时,建议使用此定位器 6、按定位按元素的title属性查找元素语法:page.get_by_title()Dom结构示例:示例代码:expect(page.get_by_title("Issues count")).to_have_text("25 issues")说明:当元素具有title属性时,建议使用此定位器7、按测试ID查找根据元素data-testid属性来定位元素(可以配置其他属性)语法:page.get_by_title()Dom结构示例:示例代码:page.get_by_test_id("directions").click()8、设置自定义测试ID属性默认情况下,page.get_by_test_id()将根据data-testid属性查找元素,但您可以在测试配置中或通过调用selectors.set_test_id_attribute()来配置它。将测试ID设置为对测试使用自定义数据属性,示例代码:playwright.selectors.set_test_id_attribute("data-pw")Dom结构:然后像往常一样找到该元素,示例代码如下:page.get_by_test_id("directions").click() 9、通过CSS或XPath定位如果绝对必须使用CSS或XPath定位器,则可以使用page.locator()创建一个定位器,该定位器采用一个选择器来描述如何在页面中查找元素。Playwright支持CSS和XPath选择器,如果省略css=或xpath=前缀,则会自动检测它们。示例代码:page.locator("css=button").click()page.locator("xpath=//button").click()page.locator("button").click()page.locator("//button").click()说明:XPath和CSS选择器可以绑定到DOM结构或实现。当DOM结构更改时,这些选择器可能会中断。不建议使用CSS和XPath,因为DOM经常会更改,从而导致无法复原的测试。相反,请尝试提供一个接近用户感知页面的定位器,例如角色定位器,或者使用测试ID定义显式测试协定。 二、在ShadowDOM中定位1、什么是ShadowDOM?ShadowDOM是 WebComponents 技术的一部分,它提供了一种将HTML结构、样式和行为封装在一个独立的、封闭的DOM中的机制。以下是一个使用ShadowDOM的例子,该例子展示了如何创建一个简单的自定义组件,并将内容、样式封装在ShadowDOM中。示例代码:            Shadow Host (这里不会显示 Shadow DOM 的内容)            dom结构: 2、如何查看ShadowDOM首先打开浏览器控制台的设置选项 然后再找到Preference->Elements,把showuseranentshadowdom勾上这时候我们再来看一下此时的dom元素发生了什么变化我们会发现这些标签内部都大有乾坤,在这些标签下面都多了一个shadowroot,在它里面才是这些标签的真实布局。3、在ShadowDOM中定位默认情况下,Playwright中的所有定位器都使用ShadowDOM中的元素。例外情况包括:通过XPath定位不会刺穿阴影根不支持闭合模式阴影根要定位,使用page.get_by_text("")或page.locator("",has_text="")都可以,要确保包含文本“ThisisinsidetheShadowDOM.”,示例代码如下:page.goto("http://localhost:8080/shadowDOM.html")expect(page.get_by_text("This is inside the Shadow DOM.")).to_contain_text("Shadow DOM")expect(page.locator("div", has_text="This is inside the Shadow DOM.")).to_contain_text("This is inside") 三、筛选定位1、dom结构2、按文本筛选定位可以使用locator.filter()方法按文本过滤定位器。它将在不区分大小写的情况下搜索元素内部的某个特定字符串,可能在后代元素中。示例代码:page.get_by_role("listitem").filter(has_text="Product 2").get_by_role(    "button", name="Add to cart").click()#传递正则表达式。page.get_by_role("listitem").filter(has_text=re.compile("Product 2")).get_by_role(    "button", name="Add to cart").click()2.1、按没有文本进行筛选expect(page.get_by_role("listitem").filter(has_not_text="Out of stock")).to_have_count(2)2.2、按子项/后代筛选定位器支持仅选择具有或没有与另一个定位器匹配的后代的元素的选项。因此,您可以按任何其他定位器进行过滤,例如 locator.get_by_role()、locator.get_by_test_id()、locator.get_by_text() 等。示例代码:page.get_by_role("listitem").filter(    has=page.get_by_role("heading", name="Product 2")).get_by_role("button", name="Add to cart").click()断言产品卡,确保只有一个,示例代码如下:expect(    page.get_by_role("listitem").filter(        has=page.get_by_role("heading", name="Product 2")    )).to_have_count(1)过滤定位器必须相对于原始定位器,并且从原始定位器匹配项(而不是文档根节点)开始查询。2.3、按没有子/后代进行筛选通过内部没有匹配的元素进行过滤,示例代码:expect(    page.get_by_role("listitem").filter(        has_not=page.get_by_role("heading", name="Product 2")    )).to_have_count(1)注意,内部定位器是从外部定位器开始匹配的,而不是从文档根目节点开始匹配的。 四、使用约束条件定位1、在定位器内匹配就先定位元素,再去定位子节点元素,以将搜索范围缩小到页面的特定部分。示例代码product = page.get_by_role("listitem").filter(has_text="Product 2")product.get_by_role("button", name="Add to cart").click()也可以将两个元素定位组合在一起使用,示例代码如下:save_button = page.get_by_role("button", name="Save")# ...dialog = page.get_by_test_id("settings-dialog")dialog.locator(save_button).click()2、使用and条件匹配方法locator.and_()通过匹配其他定位器来缩小现有定位器的范围,可以理解为xpath的and使用方法,都是定位一个元素,示例代码如下:page.get_by_role("link").and_(page.get_by_text("新闻")).click()3、使用or条件匹配如果您想定位两个或多个元素中的一个,但不知道会是哪一个,请使用locator.or_()创建与所有备选项匹配的定位器。示例代码如下:def test_navigationCnblogs(page: age):    page.goto("https://www.baidu.com/")    login=page.get_by_role("link").and_(page.get_by_text("登录"))    message=page.get_by_text("短信登录")    expect(message.or_(login).first).to_be_visible()    if (login.is_visible()):        login.click()    message.click() 4、仅匹配可见元素考虑一个有两个按钮的页面,第一个不可见,第二个可见,这时候就可以进行约束,示例代码如下:page.locator("button").locator("visible=true").click()五、列表元素操作dom结构:1、使用count断言使用count断言确保列表有3个项目,示例代码如下:expect(page.get_by_role("listitem")).to_have_count(3)2、断言列表中的所有文本断言定位器以查找列表中的所有文本,示例代码如下:expect(page.get_by_role("listitem")).to_have_text(["apple", "banana", "orange"])3、定位某个列表元素使用page.get_by_text()方法按文本内容在列表中查找元素,示例代码如下:page.get_by_text("orange").click()也可以使用locator.filter()查找列表中的特定元素,示例代码如下:page.get_by_role("listitem").filter(has_text="orange").click()4、按下标定位指定元素您有一个相同元素的列表,并且区分它们的唯一方法是顺序,则可以使用 locator.first、locator.last 或 locator.nth() 从列表中选择特定元素。    banana = page.get_by_role("listitem").nth(1)    expect(banana).to_have_text('banana')5、链接筛选器定位元素当您有具有各种相似性的元素时,使用locator.filter()方法选择正确的元素。还可以链接多个筛选器以缩小选择范,就是层级定位,个人感觉。DOM结构示例代码row_locator = page.get_by_role("listitem")row_locator.filter(has_text="Mary").filter(    has=page.get_by_role("button", name="Say goodbye")).screenshot(path="screenshot.png")6、遍历每个元素对列表中的每个元素执行操作,示例代码如下:for row in page.get_by_role("listitem").all():    print(row.text_content())rows = page.get_by_role("listitem")count = rows.count()for i in range(count):    print(rows.nth(i).text_content())7、Evaluateinthepage在页面中评估我觉得这个就是很ES6呀,示例代码如下:rows = page.get_by_role("listitem")# 很前端了吧texts = rows.evaluate_all("list => list.map(element => element.textContent)")8、检查定位元素的个数定位元素如果出现定位多个元素,这个就很好用了,可以作为检验是否定位到唯一元素检测,示例代码如下:print(page.get_by_role("button").count()) #2
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 会员注册

本版积分规则

QQ|手机版|心飞设计-版权所有:微度网络信息技术服务中心 ( 鲁ICP备17032091号-12 )|网站地图

GMT+8, 2025-1-13 03:19 , Processed in 0.556586 second(s), 26 queries .

Powered by Discuz! X3.5

© 2001-2025 Discuz! Team.

快速回复 返回顶部 返回列表