I am using Selenium to test the user interface of a website. 95% of the time, things work fine. I find my elements, execute my clicks, and check the results. Unfortunately, maybe 5% of the time, I'll find the element, verify that it is clickable, run the click event with no error... but nothing actually changes.
Here's an example bit of code:
WebDriverWait waitForTimeout = new WebDriverWait(Driver, TimeSpan.FromSeconds(60));
IWebElement selectMaskButton = waitForTimeout.Until(ExpectedConditions.ElementToBeClickable(By.Id("mask_expander")));
Thread.Sleep(250); // A slight delay ensures that the button really is clickable
selectMaskButton.ClickAndWait(Driver);
I have that sleep statement in there because this is a case where, frequently, that click event goes awry. The quarter second wait works most of the time, but not always. For this case, I could probably surround it with custom code to pause, and then click again if the specified follow-up element doesn't work out, but that's clunky and involves a different target for everywhere that I click on things. And adding delays everywhere is less than optimal too because it leads to a bunch of fixed delays that just keep getting incremented as people run into issues.
I don't know that it's entirely a matter of going too fast because I've had the occasional case where I set a debug statement on a Click command, run the test in debug, and waited a few seconds before executing the Click, and still it doesn't work. As a tester, I don't have access to some of the inner code, so I don't know exactly what gets executed inside the click, and the developers I've spoken to are basically telling me that it works fine when they manually click on it, so they don't see the problem.
Is there a generic way to handle clicking on an element and ensuring that the action actually fires?
Additional code for helper function ClickAndWait:
public static void WaitUntilPageLoads(this IWebDriver driver)
{
var wait = new WebDriverWait(driver, TimeSpan.FromSeconds(30));
wait.Until(d => ((RemoteWebDriver)d).ExecuteScript("return document.readyState").Equals("complete"));
Thread.Sleep(500);
}
public static void ClickAndWait(this IWebElement element, IWebDriver driver)
{
element.Click();
driver.WaitUntilPageLoads();
}
In this case, ClickAndWait actually isn't useful because the page is changing dynamically rather than all at once, so the page claims to be loaded right away. I had just forgotten to replace the call.
Related
Using SeleniumExtras.WaitHelpers nuget pkg
Trying to click a button on a page but as the page is loading there is a sort of blocking div that is causing an error. I noticed that if I just gradually step through the code, things work as expected, but when I just let the code run, I catch an exception that mentions something is blocking it. Browsing the site I can see the blocking div occasionally. First I tried just using a wait, and that was when I discovered the error, but when using ExpectedConditions I am still getting it. What cam I missing?
IWebElement dtLink = new WebDriverWait(Driver, TimeSpan.FromSeconds(45))
.Until(ExpectedConditions.ElementToBeClickable(By.XPath(date_link_path)));
dtLink .Click();
The exception message mentions this:
...is not clickable at point (115,311) ...because another element obscures it
When this error occurs there are 2 main ways to deal with that:
It is your way that you already did it.
Using 'Javascript' executor it will definitely work.
Here it is code example:
IJavaScriptExecutor executor = (IJavaScriptExecutor)driver ;
executor.ExecuteScript("arguments[0].click();", dtLink);
Sometimes, Selenium C# launches chrome as well as the website, but is unable to interact with it.
On close inspection, I noticed that it was because chrome was not in focus and was not loading the UI.
As soon as I focused the chrome, I would see a white screen for half a second, and then UI appeared. And as soon as UI appeared, the test started running well.
I feel like chrome can not get enough system resources without the background. Any help would be appreciated. I never faced the issue in selenium-python.
Website is - web.whatsapp.com
It look like you are not giving enough time for page to load. You can use any of the below wait technique.
Explicit Wait in Selenium C#
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(30));
String ele_xpath = "<xpath of element>"
WebDriverWait wait = new WebDriverWait(driver,30);
IWebElement welcomeMessage =
wait.until(ExpectedConditions.visibilityOfElementLocated(By.XPath(ele_xpath)));
// Here I have assumed first page is having a welcome message, you can use any element present on your page. Your script will wait up to 30 sec for element to appear before it will throw timeoutexception
implicit wait in Selenium C#
driver.manage().timeouts().implicitlyWait(30, TimeUnit.SECONDS);
//Above will make script to wait for each element up to 30 sec
waiting in C#:
You can ask your script to wait for a defined millisecond before executing next line of code:
Thread.Sleep(6000);
# It will for 6 sec
Fluent Wait in Selenium C#
Wait<WebDriver> fluentWait = new FluentWait<WebDriver>(driver)
.withTimeout(30, SECONDS) // this defines the total amount of time to wait for
.pollingEvery(2, SECONDS) // this defines the polling frequency
.ignoring(NoSuchElementException.class); // this defines the exception to ignore
WebElement welcomeMessage= fluentWait.until(new Function<WebDriver, WebElement>() {
public WebElement apply(WebDriver driver) //in this method defined your own subjected conditions for which we need to wait for
{ return driver.findElement(By.xpath("//*[contains(text(),'Welcome')]"));
}});
Note : You can use any of the above wait method. However each of them have their own advantage/ disadvantage. Please read more about them and difference at below link:
https://www.lambdatest.com/blog/selenium-waits-implicit-explicit-fluent-and-sleep/
Please try the following - it will force focus on Chrome (from the information provided I can assume there a JS event which blocks the page loading or trigger onfocus event).
var driver = new ChromeDriver();
driver.Manage().Window.Maximize(); // will focus or at least bring to front
// other things to try - switch
driver.SwitchTo().DefaultContent();
driver.SwitchTo().ActiveElement();
// other things to try - JavaScript
((IJavaScriptExecutor)driver).ExecuteScript("document.querySelector('body').focus();")
Guys, I have started to work on selenium web driver. You can assume I am a beginner. At the moment I am having difficulties in implementing the implicit wait command in my code (C#). it is not working as it should and result in an exception due to Element not found, however when I add the "Thread.Sleep(3000) the code get executed flawlessly. I have been looking for the solution all over the internet but not able to resolve the problem. Below I have mentioned sample code.
class Entrypoint
{
static void Main()
{
IWebDriver driver = new ChromeDriver();
**driver.Manage().Timeouts().ImplicitWait = TimeSpan.FromSeconds(20);**
driver.Navigate().GoToUrl("https://r1.netrevelation.com:8443/mcba-cms/pages/flight-transfer.cab");
driver.Manage().Window.Maximize();
driver.FindElement(By.Id("loginlink")).Click();
driver.FindElement(By.Id("headerSubView:inputUserName:input")).SendKeys("st001");
driver.FindElement(By.Id("headerSubView:inputPassword:input")).SendKeys("hello321" + Keys.Enter);
driver.FindElement(By.Id("dateOfFlight:input")).Click();**//This Step does not get Executed , it throws exception element not found.**
driver.FindElement(By.Id("ui-datepicker-div")).Click();
driver.FindElement(By.XPath(".//*[#id='ui-datepicker-div']/div/a[2]/span")).Click();
driver.FindElement(By.LinkText("28")).Click();
IWebElement Flightno = driver.FindElement(By.Id("selectedFlight:input"));
Flightno.SendKeys("BA901" + Keys.Enter);
IWebElement Flighttick = driver.FindElement(By.Id("flightTickImg"));
driver.Quit();
Please note that at the moment I don't want to use explicit wait, as implicit will serve my need (if it starts working). The above code Run in supersonic speed for somehow it manages to Login into the system but afterward it Fails every time reason being once Login request is made system pauses for 2-3 seconds. Please provide your comment in this regard.
As per the documentation, an Implicit Wait is to tell the WebDriver to poll the HTML DOM for a certain amount of time when trying to find an element() or find all elements() if they are not immediately available. But availability of an element in the DOM Tree doesn't guarantees that the ElementToBeClickable as you have tried in your code block. Hence you face the exception as Element not found.
Solution:
So the solution to your issue is to induce Explicit Wait i.e. WebDriverWait with ExpectedConditions clause as ElementToBeClickable which will not only confirm the availability of an element in the HTML DOM but also ensure that the Element is Clickable i.e. Element is Displayed and Enabled as follows:
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10));
IWebElement element = wait.Until(ExpectedConditions.ElementToBeClickable(By.Id("loginlink")));
I have a website with a jQuery slide down at the top with a list of all the languages I want to use: http://testing.bestshippers.com/net/index.aspx, the slider is "language selection" button at the top.
I can click it, but I get and error when trying to click on the elements inside of the slide down. I believe it is because I need to pause for a few seconds before I try to select them. Could be wrong? I'm relatively new, but I have read all kinds of things about the WebDriverWait and changing focus.
//check spanish
driver.FindElement(By.XPath("//*[#id='openCloseWrap']/img")).Click();
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(100));
driver.FindElement(By.Id("ButtonSPFlag")).Click();
String check = driver.FindElement(By.XPath("html/body/form/div[5]/div/div[1]/div/p")).Text;
Console.Out.WriteLine(check);
The above code clicks on the openCloseWrap (Language Selection Button) and then I am trying to pause for seconds (100) and then trying to click on the SP flag to change the language.
Anyone able to provide any assistance with why my wait will not pause?
You are initiating the Wait, but not waiting for anything. You probably want to do something similar to:
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(100));
// wait.Until(By.Id("ButtonSPFlag"));
IWebElement element = wait.Until(driver => driver.FindElement(By.Id("ButtonSPFlag")));
element.Click();
Alternatively you can set implicit wait, but I would go with the first one.
driver.Manage().Timeouts().ImplicitlyWait(TimeSpan.FromSeconds(10));
I would upvote you both, but I don't have enough rep points yet.
I worked on this for 4 days, issue seems to be with the way that the language bar is displayed. I can't seem to reference anything whether by CSS, XPath or naming. I'm going to keep moving on my project for now. I can check to make sure it exists, and that passes (see below) but the actual interaction fails for some reason. I am going to keep working and I will revisit. Thank you both for your time and effort!
DoesElementExistX(driver, sw, "//*[#id='openCloseWrap']/img");
DoesElementExistX(driver, sw, "//*[#id='ButtonUSFlag']");
DoesElementExistX(driver, sw, "//*[#id='ButtoFRFlag']");
DoesElementExistX(driver, sw, "//*[#id='ImageButton1']");
DoesElementExistX(driver, sw, "//*[#id='ButtonDEFlag']");
DoesElementExistX(driver, sw, "//*[#id='ButtonITFlag']");
DoesElementExistX(driver, sw, "//*[#id='ButtonSPFlag']");
#region[DoesElementExistX]
public static void DoesElementExistX(IWebDriver driver, StreamWriter sw, String id)
{
var wait = new WebDriverWait(driver, TimeSpan.FromSeconds(2));
try
{
wait.Until(ExpectedConditions.ElementExists(By.XPath(id)));
}
catch (WebDriverTimeoutException)
{
sw.WriteLine("FAILED - " + id);
}
}
#endregion
I looked into this some more.
You're trying to click on the Language element by id. This is a problem, because Selenium will click in the center of the element (unless otherwise told differently), and the center of the element is to the left of the Language area.
I think this will accomplish what you need:
driver.FindElement(By.CssSelector(".topMenuAction")).Click();
driver.FindElement(By.Id("ButtonSPFlag")).Click();
String Check = driver.FindElement(By.CssSelector("#loginBoxInner>p")).Text;
You may need to add a Sleep(), or use WebDriverWait().
Could it be that the language selection bar is always in the DOM, and that when you click "expand" it is simply showing the html rather than creating or injecting it?
If so, then your "wait" will return straight away because you are saying "wait until the element" is in the HTML, and in your case, it is always there.
You need to make your waiting smarter such as "wait until displayed" or perhaps even "wait until displayed and stationary" as you also need to wait for the animation to finish
I have the following scenario:
I want to navigate to a page. Then click a button as soon as it appears(not wait for page to load ). I don't want to wait for the initial page to load as it takes a long time. My program currently is stuck until the page loads and then clicks the button.
I basically want to navigate to link and then have no wait for page and continue with my code.
Is there anyway round this?
With the latest version of the .NET bindings, you can set a page load timeout. However, there are some caveats you'll need to be aware of. First, this hasn't been implemented by all browsers. It should work for IE and Firefox, if memory serves. Secondly, you'll need to catch an exception to make that work properly, but it can be done.
// WARNING! Completely untested code written without
// the benefit of an IDE!
IWebDriver driver = new InternetExplorerDriver();
driver.Manage().Timeouts().SetPageLoadTimeout(TimeSpan.FromSeconds(1));
try
{
driver.Url = "http://your.long.loading.page.com";
}
catch (TimeoutException)
{
// NOTE: In 2.26 or later, this will be WebDriverTimeoutException
}
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10));
IWebElement element = wait.Until<IWebElement>((d) =>
{
return d.FindElement(By.Id("yourId"));
}
IWebDriver driver = new FirefoxDriver();
driver.Url = "http://somedomain/url_that_delays_loading";
WebDriverWait wait = new WebDriverWait(driver, TimeSpan.FromSeconds(10));
IWebElement myDynamicElement = wait.Until<IWebElement>((d) =>
{
return d.FindElement(By.Id("someDynamicElement"));
});
It does exactly what you want. It queries the page to find the element, catches exceptions if no element found and returns an element when found.
After that, you can manipulate the button ignoring the fact that page can be not loaded
http://seleniumhq.org/docs/04_webdriver_advanced.html
Short answer: No.
Long answer: Selenium is by default trying to simulate the real user - I know that it kinda sucks, especially at a times you need to test a deploy after deploy after deploy...
But: If real user has to wait for button to appear, then the program has to do it also...
BTW: If the button really appears among first on the page, you can try this:
search for the button by ID, xpath ... the way you do it
Catch exception (I am Java guy, so dont ask me how)
If there was error, wait short time (e.g. 200 milliseconds) and then go to 1
If not, click the button
The cycle should end also if certain amount of time passes (10s) and if so, throw exception
As stated above: Language of my choice is Java. There I can do that. But I dont know how to write that in C#