Selenium 2.0b3 IE WebDriver,点击不触发

Posted

技术标签:

【中文标题】Selenium 2.0b3 IE WebDriver,点击不触发【英文标题】:Selenium 2.0b3 IE WebDriver, Click not firing 【发布时间】:2011-07-31 07:04:58 【问题描述】:

在使用IE9的IE驱动时,有时Click方法只会选择一个按钮,它不会做Click()的动作。请注意,这只是偶尔发生,所以我认为问题不是代码。在 Firefox4 中使用 Firefox 驱动程序没有问题。我也遇到了偶尔也找不到元素的问题,但只是在 IE 中,而不是在 Firefox 中。

if (Driver.FindElement(By.Name("username")) == null) 
    //sometimes gets here in IE, never gets here in Firefox

Driver.FindElement(By.Name("username")).SendKeys(username);
Driver.FindElement(By.Name("surname")).SendKeys(surname);
Driver.FindElement(By.Name("firstname")).SendKeys(firstname);
string url = Driver.Url;
Driver.FindElement(By.Name("cmd")).Click();
if (Driver.Url == url) 
    //if the page didnt change, click the link again
    Driver.FindElement(By.Name("cmd")).Click();

我见过类似的问题 (http://***.com/questions/4737205/selenium-webdriver-ie-button-issue),但我没有动态生成的 id。

【问题讨论】:

【参考方案1】:

经过一番搜索后,我发现了两件事似乎对可重复测试有所帮助:

首先我添加了 5 秒的 ImplicitlyWait。不确定这是否适用于所有 FindElement 函数,但我已经停止获取大部分 NoSuchElementException。

OpenQA.Selenium.IE.InternetExplorerDriver driver = new OpenQA.Selenium.IE.InternetExplorerDriver();
driver.Manage().Timeouts().ImplicitlyWait(new TimeSpan(0, 0, 0, 5, 0));
//driver.Manage().Speed = Speed.Medium;

其次,我遇到了注销功能问题,并将代码更改为:

public LoginPageObject Logout() 
    Driver.FindElement(By.LinkText("Logout")).Click();

    OpenQA.Selenium.Support.UI.IWait<IWebDriver> wait = new OpenQA.Selenium.Support.UI.WebDriverWait(Driver, TimeSpan.FromSeconds(5));
    IWebElement element = wait.Until(driver => driver.FindElement(By.Name("username")));

    LoginPageObject lpage = new LoginPageObject(Driver);
    return lpage;

显式等待似乎可以处理 ImplicitlyWait 没有捕获的内容(我认为是因为重定向)。

http://code.google.com/p/selenium/source/browse/trunk/support/src/csharp/webdriver-support/UI/WebDriverWait.cs?r=10855

【讨论】:

【参考方案2】:

当我尝试使用.Click() 单击链接时,我在 Internet Explorer 8 上发现了同样的情况——即使我可以看到 Selenium 在链接上单击。根据我的经验,如果浏览器没有焦点,那么初始点击将不起作用。

解决方法是在尝试单击链接之前将.Click() 发送到页面上的另一个元素,以便浏览器获得焦点,例如它是父母:

Driver.FindElement(By.Id("Logout")).FindElement(By.XPath("..")).Click();
Driver.FindElement(By.Id("Logout")).Click();

【讨论】:

我很欣赏这个简单的解决方案,以及原因。这有助于我找出问题的最佳解决方案。 我遇到了一个问题,我无法单击我选择的元素的父级。单击同一元素两次有效,但我担心如果窗口已经获得焦点,它可能会导致不需要的行为。对所选解决方案的这种细微更改对我有用:(在 C# 中)driver.FindElement(By.TagName("body")).Click(); 还要检查你的 IE 的缩放系数。重要的是 zomm 因子始终设置为 100%! @Naishy:你用什么语言点击使用WebDriver?是Java吗?还是什么? @RiponAlWasim:我一直在使用 C#。【参考方案3】:

遇到同样的问题,点击在我的 IE 中不起作用。我找到了一种解决方法,我在其中执行 Driver.FindElement(By.Name("...")).sendKeys("\n") 来执行单击(基本上我只需按按钮上的 enter)。不是很干净,但在问题解决之前它可以工作!

【讨论】:

【参考方案4】:

我基于在我的项目中引用 Selenium WebDriver 2.15.0 并使用 Selenium WebDriver Server 2.16.0 重构了我的解决方案,并且我做了以下观察:

使用FirefoxDriver 时,点击事件确实正确触发 将RemoteWebDriver 用于DesiredCapabilities.Firefox 时,某些控件的点击事件不会正确触发 将RemoteWebDriver 用于DesiredCapabilities.htmlUnitDesiredCapabilities.HtmlUnitWithjavascript 时,点击事件确实正确触发 InternetExplorerDriverRemoteWebDriverDesiredCapabilities.InternetExplorer(实际上是同一件事)仍然给我带来不一致的结果,我发现很难确定。

我对前三点的解决方案是创建我自己的扩展 RemoteWebDriverRemoteWebElement 的类,以便我可以在继续引用 IRemoteWebDriverIWebElement 的测试代码中隐藏我的自定义行为。

我有以下我当前的“调整”,但如果您使用这些自定义类,您将能够根据您的需要调整您的驱动程序和 Web 元素行为,而无需更改您的测试代码。

public class MyRemoteWebDriver : RemoteWebDriver

    //Constructors...

    protected override RemoteWebElement CreateElement(string elementId)
    
        return new MyWebElement(this, elementId);
    


public class MyWebElement : RemoteWebElement, IWebElement

    //Constructor...

    void IWebElement.Click()
    
        if (Settings.Default.WebDriver.StartsWith("HtmlUnit"))
        
            Click();
            return;
        

        if (TagName == "a")
        
            SendKeys("\n");
            Thread.Sleep(100);
            return;
        

        if (TagName == "input")
        
            switch (GetAttribute("type"))
            
                case "submit":
                case "image":
                    Submit();
                    return;
                case "checkbox":
                case "radio":
                    //Send the 'spacebar' keystroke
                    SendKeys(" ");
                    return;
            
        

        //If no special conditions are detected, just run the normal click
        Click();
    

【讨论】:

(TagName == "input") 并不涵盖所有情况 - 还有<button> tag。您可能还想检查case "button",因为&lt;input&gt;/&lt;button&gt; 可能有自定义脚本逻辑。 另外,if(...HtmlUnit...) 函数为每个元素发送Click(),而在其他情况下它不会回退到Click()【参考方案5】:

我在 2.0rc2、IE8、Java 中也遇到过这种情况。我在实现一个可以发送多次点击的解决方案时遇到的问题是,有时它确实有效。在这些情况下,单击我的对象两次不会让我的其余测试继续进行。在我们的控件上发送“Enter”键也不起作用。

对此有类似的问题logged,但我的对象不一定靠近“视点”。任何更多建议将不胜感激。

【讨论】:

【参考方案6】:

尝试将Internet 选项-> 安全-> 启用保护模式 设置为所有区域的设置相同,请参阅http://www.mail-archive.com/watir-general@googlegroups.com/msg13482.html。这来自 Watir googlegroup,但在我的 Selenium 2 测试中,IE 按钮点击似乎在应用此功能后效果更好。

【讨论】:

您可以通过功能做到这一点,但它仍然无法始终如一地工作。【参考方案7】:

我发现 IE 驱动程序有问题,并且相同版本的代码在具有相同 IE 版本的不同机器上表现不同。

为了在每个动作之前保持一致,我会执行以下操作。

   driver.SwitchTo().Window(driver.CurrentWindowHandle);//Force Focus

对我来说,这使得 IE 驱动程序的行为更加符合预期。

【讨论】:

很高兴知道,但我认为每次需要焦点时调用 SwitchTo() 有点贵。我认为有“更便宜”的方式来强制集中注意力。 昂贵与否这远远优于作为公认答案的廉价黑客。 这对我来说并不成功。 Java 版本:driver.switchTo().window(driver.getWindowHandle());// Force Focus【参考方案8】:

以上解决方案都不适合我。这成功了:

Java

element.sendKeys(org.openqa.selenium.Keys.CONTROL);
element.click();

时髦

element << org.openqa.selenium.Keys.CONTROL
element.click()

盖布

或者,如果您使用的是Geb,还有一个更更好的解决方案,完全不引人注目:

(用IE7和Geb 0.7.0测试)

abstract class BaseSpec extends geb.spock.GebSpec

    static
    
        def oldClick = geb.navigator.NonEmptyNavigator.metaClass.getMetaMethod("click")
        def metaclass = new geb.navigator.AttributeAccessingMetaClass(new ExpandoMetaClass(geb.navigator.NonEmptyNavigator))

        // Wrap the original click method
        metaclass.click = ->
            delegate << org.openqa.selenium.Keys.CONTROL
            oldClick.invoke(delegate)
        

        metaclass.initialize()

        geb.navigator.NonEmptyNavigator.metaClass = metaclass
    


class ClickSpec extends BaseSpec

    def "verify click"()
    
        given:
        to HomePage

        expect:
        waitFor  at HomePage 

        when:
        dialog.dismiss()
        // Call the wrapped .click() method normally
        $('#someLink').click()

        then:
        waitFor  at SomePage 
    


class HomePage extends geb.Page

    static url = "index.html"
    static at =  title == "Home - Example.com" 
    static content = 
        dialog  module DialogModule 
    


class SomePage extends geb.Page  ... 
class DialogModule extends geb.Module  def dismiss()  ...  

在我的例子中,在 IE7 中点击似乎会在关闭动画模式覆盖之前失败(我们使用的是jQuery Tools Overlay Modal Dialog)。上面的Geb方法解决了这个问题。

【讨论】:

在 C# 中,这对我来说适用于 IE 驱动程序(但不适用于 chrome 等其他驱动程序): Element.SendKeys(Keys.CONTROL); Element.Click(); 您能详细说明为什么要发送控制权吗?这不会在新窗口中打开每个点击的链接吗? 上面的代码模拟了目标元素上的按键press(按键down后跟按键up)。在调用oldClick 时,模拟的 Ctrl 键已经被释放,因此链接不会在新的选项卡/窗口中打开。 请注意,我的回答是基于 Emmanuel 的解决方案:向目标元素发送按键事件似乎足以使其获得焦点。我选择了Ctrl 键,因为它不会触发任何默认浏览器行为,而且任何 JavaScript 事件处理程序都不太可能只查找该键。 适用于 java 和 IE。你能解释一下指令 element.sendKeys(org.openqa.selenium.Keys.CONTROL); 的作用吗?做什么?【参考方案9】:

将焦点集中在元素上的更好方法是使用 Javascript。当您的元素使用 id 属性标记时,此方法有效。如果不是,请让开发人员进行更改。

使用您需要的任何定位器/属性来查找元素。检索到元素后,检查它是否包含 ID 属性。如果是,则执行以下代码,将焦点强制到元素:

JavascriptExecutor executor = (JavascriptExecutor) webDriver();
executor.executeScript("document.getElementById('" + element.GetAttribute("id") + "').focus()");

使用这个,几乎所有点击丢失的问题都在使用 InternetExplorerDriver 时得到解决。

【讨论】:

【参考方案10】:

我使用的是 IE 版本:9 面临同样的问题。以下在我的情况下有效

element.sendKeys(Keys.ENTER);
element.click();

【讨论】:

【参考方案11】:

绝对没有其他事情对我有用。 有些 InternetExplorerDriver click() 对我有用,有些则没有。 然后我发现我在the documentation中漏掉了一行:浏览器的缩放级别必须设置为100%。

我确信所有其他答案都涉及缩放级别已经为 100% 的情况,但它确实解决了我的情况。所以请先检查一下。

【讨论】:

【参考方案12】:

我通过 .click() 解决了下一个问题。我使用 JS 和 executeScript(JS, WebElement el) 而不是 .click()。 示例:

protected void clickForIE(WebElement element)  
        ((JavascriptExecutor)wd).executeScript("var tmp = arguments[0];
          tmp.click()", element);  
    

但是使用这种方法后,我们应该等待页面加载。这就是我使用下一个方法的原因:

protected synchronized void pageWaitLoad()   
        String str = null;
        try 
            str = (String)((JavascriptExecutor)wd).executeScript("return document.readyState");
        
        catch (Exception e) 
// it's need when JS isn't worked
            pageWaitLoad();
            return;
        
        System.out.println("ttt " + str);
        while(!str.equals("complete"))
            try 
                Thread.currentThread().sleep(2000);
             catch (InterruptedException e) 
                e.printStackTrace();
            
            str = (String)((JavascriptExecutor)wd).executeScript("return document.readyState");
        
    

您必须在每次 clickForIE() 之后调用 pageWaitLoad()。

【讨论】:

【参考方案13】:

我的解决方案是:

Selenium WebDriver 2.29.0 (JAVA),通过 FF16 和 IE9 测试

在制作 findElement 之前,我做了一个最大化浏览器屏幕。它工作正常。

public void maximeBrowser() 
        Toolkit toolkit = Toolkit.getDefaultToolkit();
        Dimension screenResolution = new Dimension((int)toolkit.getScreenSize().getWidth(), (int)toolkit.getScreenSize().getHeight());

        //Maximize the browser
        logger.info("Maximizing the browser : getWidth ["+screenResolution.getWidth()+"] - getHeight ["+screenResolution.getHeight()+"]");
        getDriver().manage().window().maximize();
    

【讨论】:

【参考方案14】:

另一个:

v2.29.0

网络驱动程序: * 火狐 18 支持。 * IEDriver 支持“requireWindowFocus”所需的功能。什么时候 使用此事件和本机事件,IE 驱动程序将需要焦点和 用户交互将使用 SendInput() 来模拟用户 互动。请注意,这意味着您不得使用 在测试运行时运行 IE 的机器。

【讨论】:

【参考方案15】:

作为解决方法,我在每次单击之前使用了带有空字符串的 SendKey:

element.SendKeys("");
element.Click();

【讨论】:

【参考方案16】:

phpUnit + facebook / php-webdriver 有时函数 click() 不检查复选框元素。

我的解决办法是:

$Element = $WebDriver->findElement(
    WebDriverBy::id('checkbox_id')
);

if(false === $Element->isSelected())

    $Element->sendKeys(WebDriverKeys::SPACE);

【讨论】:

【参考方案17】:

简答: 如果您在 触摸屏 显示器(例如 Windows 8 触摸笔记本电脑)上打开浏览器窗口在 IE11 中运行自动化 Selenium 测试,请尝试在 中打开浏览器窗口运行测试非触摸屏。 原始的 .click() 方法应该可以在没有所有代码变通方法的情况下正常工作。

答案背景: 我与我们 QA 团队的一名测试人员一起调查了一个类似的问题。在尝试了此处和selenium webdriver IE button issue 的大部分代码解决方案后,我们最终发现问题仅发生在测试人员的 Windows 8 笔记本电脑触摸屏上的 IE(我们的版本 11)中。 使用在外部戴尔显示器上运行的 IE 窗口运行 Selenium 测试可以让测试每次都运行良好,即使只使用标准的 .click() 调用也是如此。 我们在 Magnific Popup (http://dimsemenov.com/plugins/magnific-popup/) 对话框中的按钮单击事件上也失败了。

我的假设:IE11(不确定其他版本)如何处理触摸事件与触摸屏上的鼠标单击事件之间的转换存在问题。

【讨论】:

【参考方案18】:

WebdriverJS

在IE 中,当您尝试执行 click() 操作时,URL 在状态栏上一直闪烁。这意味着驱动程序正在关注元素并尝试执行 click() 操作。为了完成它的 click() 动作,我在每次点击动作前后都使用了 sleep() 方法。

试试这个例子。

var webdriver = require('..'), By = webdriver.By, until = webdriver.until;
var driver = new webdriver.Builder().usingServer().withCapabilities('browserName': 'ie' ).build();

driver.get('http://www.google.com')
.then(function()
    driver.manage().window().maximize();
    driver.manage().timeouts().implicitlyWait(1000 * 3);
    driver.findElement(By.name('q')).sendKeys('webdriver');
    driver.findElement(By.name('btnG')).then(function(button)
    button.click();
    );
)
.then(function()
    driver.findElement(By.css('div[class="gb_Zb"] a[title="Google Apps"]')).then(function(apps)
        apps.click();       
        driver.findElements(By.css('a[class="gb_O"]')).then(function(appsList)
        console.log("apps : "+appsList.length);
        for(var i = 0; i < appsList.length; i++)
        console.log('applications : '+i);
        if(i == 5) 
        var element = appsList[i];
        driver.sleep(1000 * 5);
        driver.executeScript("var tmp = arguments[0]; tmp.click()", element);
        driver.sleep(1000 * 5);
          )  
    )
)
.then(null, function(err) 
  console.error("An error was thrown! By Promise... " + err);
);
driver.quit();

要执行点击,我们可以使用其中任何一个,在 IE 上测试过

element.click(); 
driver.actions().click(element).perform();
driver.executeScript("var tmp = arguments[0]; tmp.click()", element);

【讨论】:

以上是关于Selenium 2.0b3 IE WebDriver,点击不触发的主要内容,如果未能解决你的问题,请参考以下文章

Selenium:启动服务器时出现无效的“已经运行”错误

等待在 Firefox 浏览器中不起作用(Webdriver selenium 2.0 +Java)

selenium爬取网易云

爬虫之selenium模块

Selenium 入门到精通系列:五

selenium登录 京东滑动验证码