Selenium WebDriver: How to wait for iFrames to load completely?
Asked Answered
A

3

18

I'm testing a page with an iFrame whose contents are generated by JavaScript dynamically. I have to wait for the iFrame loaded completely to make sure that all the elements are present. I tried the following code, it didn't work.

WebDriver frame = wait.until(ExpectedConditions.frameToBeAvailableAndSwitchToIt("frmMain"));

I also tried to wait for some element in the iFrame to be present. It didn't work, neither.

Any help would be greatly appreciated, thanks!

Achernar answered 28/8, 2013 at 11:23 Comment(0)
S
19

Select any element on IFrame which takes maximum time to load e.g. any button or image and the wait using the following code.

WebDriverWait wait = new WebDriverWait(driver, 30);
wait.until(ExpectedConditions.visibilityOfElementLocated((By
                .name("element"))));

Or rather you can wait for for iFrame to appear and then switch to it and then use the above statement !

Shyster answered 28/8, 2013 at 12:21 Comment(5)
I tested the page with the following code: public void SelectSomething(WebDriver driver, String s) { driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS); WebDriverWait wait = new WebDriverWait(driver, 20); WebDriver frame = wait.until( ExpectedConditions.frameToBeAvailableAndSwitchToIt("frmMain")); wait = new WebDriverWait(frame, 30); System.out.println("#######################" + (new Date())); System.out.println(frame.getPageSource()); System.out.println("#######################" + (new Date())); //////To be continued...Achernar
try { wait.until(ExpectedConditions.presenceOfElementLocated(By.tagName("img"))); } catch (TimeoutException e) { System.out.println(frame.getPageSource()); } System.out.println("#######################" + (new Date())); System.out.println(frame.getPageSource()); System.out.println("#######################" + (new Date())); } //////To be continued...Achernar
And get the following results sometimes: #######################Thu Aug 29 10:30:36 CST 2013 <HTML><HEAD></HEAD> <BODY></BODY></HTML> #######################Thu Aug 29 10:30:37 CST 2013 PageSource Section#2 #######################Thu Aug 29 10:31:09 CST 2013 PageSource Section#3 #######################Thu Aug 29 10:31:09 CST 2013 //////To be continued...Achernar
Here are two problems I couldn't work out: 1. frame.getPageSource() got the source of the whole page, other than that of the iframe. 2. I had waited long enough for elements in the iframe to be present, but the source got with method frame.getPageSource() didn't include anyone. Sorry for the ugly formatting.Achernar
Have you tried switching to the Iframe first and then waiting for an element in it to loading? This is what I do in my code. In C# this is done with driver.SwitchTo().Frame(driver.FindElement(By.CssSelector(IFrameCss)));Antelope
I
4

The most robust way to verify a page reload is to verify that an element went stale and then to verify that a new element has loaded.

Choose an element, elementToBeStale, which exists on the "old" page. This is the element you expect to go stale when the page reloads.

Now choose an element you expect to appear on the page once it has reloaded, xPathOfElementToLoad. Note that this can be the same element you expected to go stale.

Note that xPathOfElementToLoad can't be a WebElement because the it doesn't yet exist on the page so it can't be represented by one.

WebDriverWait wait = new WebDriverWait(driver, WAIT_TIME, POLL_INTERVAL);

ExpectedCondition<Boolean> cond1 = ExpectedConditions.stalenessOf(elementToBeStale);
ExpectedCondition<WebElement> cond2 = ExpectedConditions.presenceOfElementLocated(By.xpath(xPathOfElementToLoad));
ExpectedCondition<Boolean> cond = ExpectedConditions.and(cond1, cond2);

wait.until(cond);

This is useful for page reloads because on a page reload waiting for a WebElement to go stale guarantees the element no longer exists and then waiting for a new element to exist guarantees the new page has loaded

Intreat answered 26/11, 2019 at 22:16 Comment(2)
Thanks! This helped me a lot. Just needed to change second condition (cond2) to more general one without putting locator there: ExpectedCondition<Boolean> stalenessOfOldElement = ExpectedConditions.stalenessOf(elementOld); ExpectedCondition<WebElement> visibilityOfNewElement = ExpectedConditions.visibilityOf(elementNew); ExpectedCondition<Boolean> readyToSwitch = ExpectedConditions.and(stalenessOfOldElement, visibilityOfNewElement); wait.until(readyToSwitch);Pharaoh
@Pharaoh the reason I use an xpath for the new element is because how would you have found the element you're looking for if the page hasn't loaded yet?Intreat
V
0

I don't know if you already solved this, but here it goes.

Your code is missing the type of search for the element, and I think is "By.ID", so your code should be:

WebDriver frame = wait.until(ExpectedConditions.frameToBeAvailableAndSwitchToIt((By.ID, "frmMain")))
Victuals answered 9/1, 2023 at 12:49 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.