使用带有Java的Selenium WebDriver切换选项卡


70

将Selenium WebDriver与JAVA结合使用。我试图自动执行一项功能,在该功能中我必须打开一个新选项卡,然后在此处执行一些操作,然后返回到上一个选项卡(父项)。我使用了开关手柄,但无法正常工作。奇怪的是,两个选项卡具有相同的窗口句柄,因此我无法在两个选项卡之间切换。

但是,当我尝试使用其他Firefox窗口时,它可以工作,但是对于选项卡,它却无法工作。

请帮我。如何切换标签。或者在不使用窗口句柄的情况下如何切换选项卡,就我而言,这两个选项卡都是相同的。

(我观察到,当您在同一窗口中打开不同的选项卡时,窗口句柄保持不变)


我认为2个标签在大多数情况下与2个窗口相同
Harshavardhan Konakanchi 2013年

Answers:


115
    psdbComponent.clickDocumentLink();
    ArrayList<String> tabs2 = new ArrayList<String> (driver.getWindowHandles());
    driver.switchTo().window(tabs2.get(1));
    driver.close();
    driver.switchTo().window(tabs2.get(0));

这段代码非常适合我。试试看。在执行新选项卡上的操作之前,始终需要将驱动程序切换到新选项卡。


2
标签不是窗口。我猜它在Firefox中不起作用。
Alex78191

3
此答案假定driver.getWindowHandles()中的顺序。该方法返回Set,因此不保证合同级别的订单。也许您的WebDriver实现会返回一个可以保证顺序的Set实现,但是为什么要冒险呢?
吉尔伯特·阿里纳斯·匕首

使用ChromeDriver 2.33,getWindowHandles()的确会返回LinkedHashSet,因此按索引进行访问的效果很好。请注意,您需要使用getAt()而不是get()
SourMonk '18

24

这是打开新标签页,更改焦点,关闭标签页并将焦点返回到旧/原始标签页的简单解决方案:

@Test
public void testTabs() {
    driver.get("https://business.twitter.com/start-advertising");
    assertStartAdvertising();

    // considering that there is only one tab opened in that point.
    String oldTab = driver.getWindowHandle();
    driver.findElement(By.linkText("Twitter Advertising Blog")).click();
    ArrayList<String> newTab = new ArrayList<String>(driver.getWindowHandles());
    newTab.remove(oldTab);
    // change focus to new tab
    driver.switchTo().window(newTab.get(0));
    assertAdvertisingBlog();

    // Do what you want here, you are in the new tab

    driver.close();
    // change focus back to old tab
    driver.switchTo().window(oldTab);
    assertStartAdvertising();

    // Do what you want here, you are in the old tab
}

private void assertStartAdvertising() {
    assertEquals("Start Advertising | Twitter for Business", driver.getTitle());
}

private void assertAdvertisingBlog() {
    assertEquals("Twitter Advertising", driver.getTitle());
}

我正在使用chrome驱动程序,并且在其中使用“ driver.switchTo()。window(newTab.get(0));”行。一个新的窗口随同一个新的选项卡一起打开。。。为什么会这样?
露西

我没有检查过,但我想它是浏览器配置。
乔丹·席尔瓦

13

Web驱动程序如何处理不同的窗口以及如何处理不同的选项卡有所不同。

情况1:
如果有多个窗口,则以下代码可以提供帮助:

//Get the current window handle
String windowHandle = driver.getWindowHandle();

//Get the list of window handles
ArrayList tabs = new ArrayList (driver.getWindowHandles());
System.out.println(tabs.size());
//Use the list of window handles to switch between windows
driver.switchTo().window(tabs.get(0));

//Switch back to original window
driver.switchTo().window(mainWindowHandle);


情况2:
如果同一窗口中有多个选项卡,则只有一个窗口句柄。因此,在窗口句柄之间切换可将控件保留在同一选项卡中。
在这种情况下,使用Ctrl + \ t(Ctrl + Tab)在各选项卡之间切换更为有用。

//Open a new tab using Ctrl + t
driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL +"t");
//Switch between tabs using Ctrl + \t
driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL +"\t");

详细的示例代码可以在这里找到:http :
//design-interviews.blogspot.com/2014/11/switching-between-tabs-in-same-browser-window.html


是。这是第一个响应,表明只有一个窗口句柄,并且与选项卡不同。我不知道其他解决方案如何工作。getWindowHandles如何获取它们的选项卡列表?它仅在IE中为我获得窗口句柄(我知道,IE,很糟糕)
BClaydon

7

解决

假设:通过单击网页上的某些内容可以打开一个新选项卡。

使用以下逻辑切换到第二个选项卡。

new Actions(driver).sendKeys(driver.findElement(By.tagName("html")), Keys.CONTROL).sendKeys(driver.findElement(By.tagName("html")),Keys.NUMPAD2).build().perform();

同样,您可以再次切换回第一个标签。

new Actions(driver).sendKeys(driver.findElement(By.tagName("html")), Keys.CONTROL).sendKeys(driver.findElement(By.tagName("html")),Keys.NUMPAD1).build().perform();

仅适用于Chrome
Jury

5

由于driver.window_handles顺序不正确,因此是一个更好的解决方案。

首先使用快捷方式切换到第一个标签页,Control + X以切换到浏览器窗口中的第'x'个标签页。

driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "1");
# goes to 1st tab

driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "4");
# goes to 4th tab if its exists or goes to last tab.

4
String selectLinkOpeninNewTab = Keys.chord(Keys.CONTROL, Keys.RETURN);
    WebElement e = driver.findElement(By
            .xpath("html/body/header/div/div[1]/nav/a"));
e.sendKeys(selectLinkOpeninNewTab);//to open the link in a current page in to the browsers new tab

    e.sendKeys(Keys.CONTROL + "\t");//to move focus to next tab in same browser
    try {
        Thread.sleep(8000);
    } catch (InterruptedException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    }
    //to wait some time in that tab
    e.sendKeys(Keys.CONTROL + "\t");//to switch the focus to old tab again

希望对您有帮助。


3

您需要做的第一件事是打开一个新标签并保存它的句柄名称。最好使用javascript而不是keys(ctrl + t)来完成此操作,因为密钥在自动化服务器上​​并不总是可用。例:

public static String openNewTab(String url) {
    executeJavaScript("window.parent = window.open('parent');");
    ArrayList<String> tabs = new ArrayList<String>(bot.driver.getWindowHandles());
    String handleName = tabs.get(1);
    bot.driver.switchTo().window(handleName);
    System.setProperty("current.window.handle", handleName);
    bot.driver.get(url);
    return handleName;
}

您需要做的第二件事是在选项卡之间切换。仅通过切换窗口手柄来执行此操作将不会始终有效,因为您将要使用的选项卡不会始终处于焦点状态,并且Selenium会不时失败。就像我说的那样,使用键有点问题,而javascript并不真正支持切换选项卡,因此我使用了警报来切换选项卡,它的工作原理很吸引人:

public static void switchTab(int tabNumber, String handleName) {
        driver.switchTo().window(handleName);
        System.setProperty("current.window.handle", handleName);
        if (tabNumber==1)
            executeJavaScript("alert(\"alert\");");
        else
            executeJavaScript("parent.alert(\"alert\");");
        bot.wait(1000);
        driver.switchTo().alert().accept();
    }

2
driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL,Keys.SHIFT,Keys.TAB);

此方法有助于在多个窗口之间切换。这种方法的局限性在于它只能使用多次,直到达到所需的窗口。希望能帮助到你。


也许在多个标签之间?
Alex78191

2

在使用firefox 47.0.1作为Java中的WebDriver的Selenium 2.53.1中:无论我打开了多少选项卡,“ driver.getWindowHandles()”将仅返回一个句柄,因此无法在选项卡之间进行切换。

一旦开始使用Chrome 51.0,我将获得所有处理权。以下代码显示了如何访问多个驱动程序以及每个驱动程序中的多个选项卡。

// INITIALIZE TWO DRIVERS (THESE REPRESENT SEPARATE CHROME WINDOWS)
driver1 = new ChromeDriver();
driver2 = new ChromeDriver();

// LOOP TO OPEN AS MANY TABS AS YOU WISH
for(int i = 0; i < TAB_NUMBER; i++) {
   driver1.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "t");
   // SLEEP FOR SPLIT SECOND TO ALLOW DRIVER TIME TO OPEN TAB
   Thread.sleep(100);

// STORE TAB HANDLES IN ARRAY LIST FOR EASY ACCESS
ArrayList tabs1 = new ArrayList<String> (driver1.getWindowHandles());

// REPEAT FOR THE SECOND DRIVER (SECOND CHROME BROWSER WINDOW)

// LOOP TO OPEN AS MANY TABS AS YOU WISH
for(int i = 0; i < TAB_NUMBER; i++) {
   driver2.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "t");
   // SLEEP FOR SPLIT SECOND TO ALLOW DRIVER TIME TO OPEN TAB
   Thread.sleep(100);

// STORE TAB HANDLES IN ARRAY LIST FOR EASY ACCESS
ArrayList tabs2 = new ArrayList<String> (driver1.getWindowHandles());

// NOW PERFORM DESIRED TASKS WITH FIRST BROWSER IN ANY TAB
for(int ii = 0; ii <= TAB_NUMBER; ii++) {
   driver1.switchTo().window(tabs1.get(ii));
   // LOGIC FOR THAT DRIVER'S CURRENT TAB
}

// PERFORM DESIRED TASKS WITH SECOND BROWSER IN ANY TAB
for(int ii = 0; ii <= TAB_NUMBER; ii++) {
   drvier2.switchTo().window(tabs2.get(ii));
   // LOGIC FOR THAT DRIVER'S CURRENT TAB
}

希望这使您对如何在多个浏览器窗口中操作多个选项卡有了一个好主意。



1

我最近遇到了问题,该链接在新标签页中打开,但是硒仍集中在初始标签页上。

我使用的是Chromedriver,而专注于标签的唯一方法是让我使用switch_to_window()

这是Python代码:

driver.switch_to_window(driver.window_handles[-1])

因此,提示是找出所需窗口句柄的名称,它们以列表形式存储在

driver.window_handles

4
索引到窗口句柄的集合时要非常非常小心。不能保证以任何顺序显示这些句柄,并且打开的第一个窗口在getWindowhandles方法返回的列表中(或与您的语言等效的窗口)可能具有最后一个窗口句柄。
JimEvans

1

这是一个非常简单的过程:假定您有两个选项卡,因此您需要先使用来关闭当前选项卡,client.window(callback)因为switch命令“切换到第一个可用选项卡”。然后,您可以使用轻松切换标签页client.switchTab


1

一个简短的示例,说明如何在浏览器中的选项卡之间切换(如果只有一个窗口):

// open the first tab
driver.get("https://www.google.com");
Thread.sleep(2000);

// open the second tab
driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "t");
driver.get("https://www.google.com");
Thread.sleep(2000);

// switch to the previous tab
driver.findElement(By.cssSelector("body")).sendKeys(Keys.CONTROL + "" + Keys.SHIFT + "" + Keys.TAB);
Thread.sleep(2000);

我写Thread.sleep(2000)只是为了让超时,以查看选项卡之间的切换。

您可以使用CTRL + TAB切换到下一个选项卡,并使用CTRL + SHIFT + TAB切换到上一个选项卡。


1

这将适用于Firefox和Chrome的MacOS:

// opens the default browser tab with the first webpage
driver.get("the url 1");
thread.sleep(2000);

// opens the second tab
driver.findElement(By.cssSelector("Body")).sendKeys(Keys.COMMAND + "t");
driver.get("the url 2");
Thread.sleep(2000);

// comes back to the first tab
driver.findElement(By.cssSelector("Body")).sendKeys(Keys.COMMAND, Keys.SHIFT, "{");

1

获取父窗口句柄。

String parentHandle = driverObj.getWindowHandle();
public String switchTab(String parentHandle){
    String currentHandle ="";
    Set<String> win  = ts.getDriver().getWindowHandles();   

    Iterator<String> it =  win.iterator();
    if(win.size() > 1){
        while(it.hasNext()){
            String handle = it.next();
            if (!handle.equalsIgnoreCase(parentHandle)){
                ts.getDriver().switchTo().window(handle);
                currentHandle = handle;
            }
        }
    }
    else{
        System.out.println("Unable to switch");
    }
    return currentHandle;
}

1

所选答案的缺陷在于,它不必要地采用webDriver.getWindowHandles()。该getWindowHandles()方法返回Set,不保证顺序。

我使用以下代码来更改选项卡,该选项卡不进行任何排序。

String currentTabHandle = driver.getWindowHandle();
String newTabHandle = driver.getWindowHandles()
       .stream()
       .filter(handle -> !handle.equals(currentTabHandle ))
       .findFirst()
       .get();
driver.switchTo().window(newTabHandle);

1
protected void switchTabsUsingPartOfUrl(String platform) {
    String currentHandle = null;
    try {
        final Set<String> handles = driver.getWindowHandles();
        if (handles.size() > 1) {
            currentHandle = driver.getWindowHandle();
        }
        if (currentHandle != null) {
            for (final String handle : handles) {
                driver.switchTo().window(handle);
                if (currentUrl().contains(platform) && !currentHandle.equals(handle)) {
                    break;
                }
            }
        } else {
            for (final String handle : handles) {
                driver.switchTo().window(handle);
                if (currentUrl().contains(platform)) {
                    break;
                }
            }
        }
    } catch (Exception e) {
        System.out.println("Switching tabs failed");
    }
}

调用此方法并将您要切换到的标签页的URL子字符串作为参数传递给参数


1

公共类TabBrowserDemo {

public static void main(String[] args) throws InterruptedException {
    System.out.println("Main Started");
    System.setProperty("webdriver.gecko.driver", "driver//geckodriver.exe");
    WebDriver driver = new FirefoxDriver();
    driver.get("https://www.irctc.co.in/eticketing/userSignUp.jsf");
    driver.manage().timeouts().implicitlyWait(30, TimeUnit.SECONDS);

    driver.findElement(By.xpath("//a[text()='Flights']")).click();
    waitForLoad(driver);
    Set<String> ids = driver.getWindowHandles();
    Iterator<String> iterator = ids.iterator();
    String parentID = iterator.next();
    System.out.println("Parent WIn id " + parentID);
    String childID = iterator.next();
    System.out.println("child win id " + childID);

    driver.switchTo().window(childID);
    List<WebElement> hyperlinks = driver.findElements(By.xpath("//a"));

    System.out.println("Total links in tabbed browser " + hyperlinks.size());

    Thread.sleep(3000);
//  driver.close();
    driver.switchTo().window(parentID);
    List<WebElement> hyperlinksOfParent = driver.findElements(By.xpath("//a"));

    System.out.println("Total links " + hyperlinksOfParent.size());

}

public static void waitForLoad(WebDriver driver) {
    ExpectedCondition<Boolean> pageLoadCondition = new
            ExpectedCondition<Boolean>() {
                public Boolean apply(WebDriver driver) {
                    return ((JavascriptExecutor)driver).executeScript("return document.readyState").equals("complete");
                }
            };
    WebDriverWait wait = new WebDriverWait(driver, 30);
    wait.until(pageLoadCondition);
}

4
欢迎使用Stack Overflow!请不要只将您的源代码放在这里。保持友善,并尝试对您的答案给出一个很好的描述,以便其他人会喜欢并赞成。请参阅:我如何写一个好的答案?
sɐunıɔןɐqɐp

1
    public void closeAndSwitchToNextTab() {
        driver.close();
        ArrayList<String> tab = new ArrayList<>(driver.getWindowHandles());
        driver.switchTo().window(tab.get(1));
    }

    public void switchToPreviousTab() {
        ArrayList<String> tab = new ArrayList<>(driver.getWindowHandles());
        driver.switchTo().window(tab.get(0));
    }

    public void closeTabAndReturn() {
        driver.close();
        ArrayList<String> tab = new ArrayList<>(driver.getWindowHandles());
        driver.switchTo().window(tab.get(0));
    }

    public void switchToPreviousTabAndClose() {
        ArrayList<String> tab = new ArrayList<>(driver.getWindowHandles());
        driver.switchTo().window(tab.get(1));
        driver.close();
    }

1
Set<String> tabs = driver.getWindowHandles();
Iterator<String> it = tabs.iterator();
tab1 = it.next();
tab2 = it.next();
driver.switchTo().window(tab1);
driver.close();
driver.switchTo().window(tab2);

尝试这个。它应该工作


0

请看下面:

WebDriver driver = new FirefoxDriver();

driver.manage().window().maximize();
driver.get("https://www.irctc.co.in/");
String oldTab = driver.getWindowHandle();

//For opening window in New Tab
String selectLinkOpeninNewTab = Keys.chord(Keys.CONTROL,Keys.RETURN); 
driver.findElement(By.linkText("Hotels & Lounge")).sendKeys(selectLinkOpeninNewTab);

// Perform Ctrl + Tab to focus on new Tab window
new Actions(driver).sendKeys(Keys.chord(Keys.CONTROL, Keys.TAB)).perform();

// Switch driver control to focused tab window
driver.switchTo().window(oldTab);

driver.findElement(By.id("textfield")).sendKeys("bangalore");

希望这会有所帮助!

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.