技术分享 | 网页 frame 与多窗口处理

本文节选自霍格沃兹测试开发学社内部教材

要定位一个元素时,怎么都定位不到的时候就要考虑是不是浏览器内嵌了一个 frame 窗口或者要找的元素在新打开的窗口里。这时候就需要进行 frame 的切换或者窗口的切换。

frame 类似于在原有主 HTML 的基础上又嵌套一个 HTML,而且嵌套的 HTML 是独立使用的,互不影响。

当打开一个页面时,光标的定位是在主页面中,如果页面是由多个 frame 组成的,那么无法直接定位到具体的元素,需要切换到自己所需要的 frame 中,再查找该元素。

知识点

iframe解析

如图可以看到iframe的标签

iframe 的多种切换方式

HTML 代码示例

<iframe src="1.html" id="hogwarts_id" name="hogwarts_name"></iframe>

那么通过传入 id、name、index 以及 Selenium 的 WebElement 对象来切换 frame

  • Python 版本
# index:传入整型的参数,从 0 开始,这里的 0 就是第一个 frame
driver.switch_to.frame(0)


#id:iframe 的 id
driver.switch_to.frame("hogwarts_id")


#name: iframe 的 name
driver.switch_to.frame("hogwarts_name")


#WebElement: 传入 `selenium.webelement` 对象
driver.switch_to.frame(driver.find_element_by_tag_name("iframe"))

  • Java 版本
// index:传入整型的参数,从 0 开始,这里的 0 就是第一个 frame
driver.switchTo().frame(0);


// id:iframe 的 id
driver.switchTo().frame("hogwarts_id");


// name: iframe 的 name
driver.switchTo().frame("hogwarts_name");


// WebElement: 传入 `selenium.webelement` 对象
driver.switchTo().frame(driver.findElement(By.tagName("iframe")));

iframe 切换回默认页面

在切换页面之后,如果还想操作原页面,则可以使用

  • Python 版本
driver.switch_to.default_content()
  • Java 版本
driver.switchTo().defaultContent();

iframe 多层切换

如图所示多层嵌套的 iframe 从最外部 iframe 切换到 iframe2 则需要层层切换

  • Python 版本
driver.switch_to.frame("iframe1")

driver.switch_to.frame("iframe2")
  • Java 版本
driver.switchTo().frame("iframe1");

driver.switchTo().frame("iframe2");

从 iframe2 切换回 iframe1 可以使用父子切换

  • Python 版本
# 从 iframe2 切换到上一级 iframe1 

driver.switch_to.parent_frame()

# 从 iframe1 切换到上一级 iframe,如果 iframe 已经是最上级,则保持不变

driver.switch_to.parent_frame()
  • Java 版本
// 从 iframe2 切换到上一级 iframe1

driver.switchTo().parentFrame();

// 从 iframe1 切换到上一级 iframe,如果 iframe 已经是最上级,则保持不变

driver.switchTo().parentFrame();

这个方法是 Selenium 提供的直接从子 frame 切换到父 frame,可以使用在嵌套的 frame 框架中。

多窗口处理

元素有属性,浏览器的窗口其实也有属性的,浏览器窗口的属性用句柄(handle)来识别。

当浏览器打开一个窗口时,如果要在新的窗口操作就需要句柄切换。

句柄的获取

当有多个窗口时,可以用 window_handles 打印句柄:

  • Python 版本
driver = webdriver.Chrome()

handles = driver.window_handles

print(handles)
  • Java 版本
driver = new ChromeDriver();

Set<String> handles = driver.getWindowHandles();

System.out.println(handles);

打印出的 window_handles:

['CDwindow-8012E9EF4DC788A58DC1588E7B8A7C44', 'CDwindow-11D52927C71E7C2B9984F2D1E2856049']

句柄的切换

通过打印 handles 可以看出,它是一个列表。

  • Python版本可以通过 switch_to.window() 来切换句柄
  • Java版本可以通过 switchTo().window() 来切换句柄
  • Python 版本

从上面源代码中的说明可以看出,switch_to.window()需要提供一个 windows_name,可以是 name 也可以是 windows handle。

from selenium import webdriver


driver = webdriver.Chrome()
handles = driver.window_handles
print(handles)
driver.switch_to.window(handles[-1])

这里唯一要注意的是 handles 是一个列表,这里的 -1 表示浏览器窗口的倒数第一个。

  • Java 版本
...
Set<String> windowHandles = driver.getWindowHandles();
Iterator<String> it = windowHandles.iterator();   //迭代allhandle里面的句柄
while(it.hasNext()) {                            //用it.hasNext()判断时候有下一个窗口,如果有就切换到下一个窗口
    driver.switchTo().window(it.next());        //切换到新窗口
}
...
}

Java 中切换句柄需要使用迭代器,如果有下一个句柄,则切换,没有则不切换,上面代码表示切换到最后一个窗口。

实战演示

百度搜索“霍格沃兹测试学院”,点击“霍格沃兹测试学院_腾讯课堂”,点击“中高级测试开发「名企定向培养」班-霍格沃兹测试学院”。

Python 代码

from selenium import webdriver




class TestHogwarts:
    def setup_method(self, method):
        self.driver = webdriver.Chrome()
        self.driver.implicitly_wait(3)


    def teardown_method(self, method):
        self.driver.quit()


    def test_hogwarts(self):
        self.driver.get('https://www.baidu.com')
        #在输入框中输入霍格沃兹测试学院
        self.driver.find_element_by_id('kw').send_keys('霍格沃兹测试学院')
        #点击搜索
        self.driver.find_element_by_css_selector('.s_btn').click()
        #使用link_text点击
        self.driver.find_element_by_link_text('关于我们 - 霍格沃兹测试学院').click()
        #将获取到的window_handles赋值给一个变量handles
        handles = self.driver.window_handles
        #切换句柄
        self.driver.switch_to.window(handles[-1])
        assert len(self.driver.find_elements_by_css_selector('.ag-title-main')) == 1

Java 代码

import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.openqa.selenium.By;
import org.openqa.selenium.chrome.ChromeDriver;
import java.util.Iterator;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import static org.junit.jupiter.api.Assertions.assertEquals;


public class Web1Test {
    private static ChromeDriver driver;
    @BeforeAll
    public static void setUp()  {
        System.setProperty(
                "webdriver.chrome.driver",
                "/driver/chrome95/chromedriver"
        );
        driver = new ChromeDriver();
        driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
    }
    @AfterAll
    public static void tearDown()  {
        driver.quit();
    }
    @Test
    public void hogwartsTest(){
        driver.get("https://www.baidu.com");
        // 在输入框中输入霍格沃兹测试学院
        driver.findElement(By.id("kw")).sendKeys("霍格沃兹测试学院");
        // 点击搜索
        driver.findElement(By.cssSelector(".s_btn")).click();
        // 使用link_text点击
        driver.findElement(By.linkText("关于我们 - 霍格沃兹测试学院")).click();
        // 将获取到的window_handles赋值给一个变量handles
        Set<String> windowHandles = driver.getWindowHandles();
        // 切换句柄
        //迭代allhandle里面的句柄
        Iterator<String> it = windowHandles.iterator();   
        //用it.hasNext()判断时候有下一个窗口,如果有就切换到下一个窗口
        while(it.hasNext()) {       
            //切换到新窗口                     
            driver.switchTo().window(it.next());       
        }
        int size = driver.findElements(By.cssSelector(".ag-title-main")).size();
        assertEquals(1,size);
    }
}

需要注意的是,把被测浏览器对应版本的 chromedriver 放置到某个路径下,配置到环境变量或者脚本代码中。

获取更多相关资料:请添加vx,ceshiren001
https://qrcode.ceba.ceshiren.com/link?name=article&project_id=qrcode&from=CSDN&timestamp=1652341581&author=MM

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Python 的 tkinter 库中,可以使用 Frame 来创建多个窗口。 Frame 是 tkinter 中的一个容器小部件,可以用于组织和管理其他小部件。 下面是一个简单的示例,演示如何使用 Frame 创建多个窗口: ```python import tkinter as tk class MyApp: def __init__(self, master): self.master = master self.master.title("多窗口示例") # 创建主窗口的 Frame self.main_frame = tk.Frame(self.master) self.main_frame.pack() # 创建按钮,点击按钮时打开新窗口 self.open_button = tk.Button(self.main_frame, text="打开新窗口", command=self.open_new_window) self.open_button.pack() def open_new_window(self): # 创建新窗口的 Frame new_window = tk.Toplevel(self.master) new_frame = tk.Frame(new_window) new_frame.pack() # 在新窗口中添加一些小部件 label = tk.Label(new_frame, text="这是一个新窗口") label.pack() if __name__ == "__main__": root = tk.Tk() app = MyApp(root) root.mainloop() ``` 在这个示例中,我们创建了一个名为 `MyApp` 的类,它表示我们的应用程序。在 `__init__` 方法中,我们创建了主窗口的 Frame,并在其中添加了一个按钮。当按钮被点击时,`open_new_window` 方法会被调用,它会创建一个新的窗口,然后在其中添加一些小部件。 可以看到,在 `open_new_window` 方法中,我们使用 `tk.Toplevel` 来创建新的窗口,然后在这个窗口中创建一个新的 Frame,并在其中添加了一个 Label。 你可以根据自己的需求在每个窗口中添加其他小部件和布局。希望这个示例对你有帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值