谨慎注意WebBrowser控件的DocumentCompleted事件

简介:

引言

WebBrowser控件的DocumentCompleted事件一般就被认定为是在页面完全加载完毕后产生,而注释中也是这么写的:

image

但事实却并非如此。

首先它不一定会在完全加载完毕时才触发,有时就会在加载过程中就会触发。

其次按照“完全加载完毕后”来理解,会认为通常一次页面跳转只会引发一次该事件,事实也并非如此,某些页面加载时会引发十多次乃至更多。

试验

做一个简单试验,首先设计这样的界面:

image

然后为那个转到按钮添加单击事件处理:

private void button1_Click(object sender, EventArgs e)

{

    webBrowser1.Navigate(textBox1.Text);

}

再为WebBrowser控件的DocumentCompleted事件添加处理:

private void webBrowser1_DocumentCompleted(object sender, WebBrowserDocumentCompletedEventArgs e)

{

    listBox1.Items.Insert(0, webBrowser1.ReadyState);

}

在这里就是输出WebBrowser控件的ReadyState属性到列表中。

运行并测试:

image

 

 

可以看到加载某些页面时会引发很多次该事件,并且状态都是一连串的Interactive。

究其原因是与页面复杂度有关的,猜想是因iFrame或Ajax加载完毕而触发的。

而触发DocumentCompleted事件时ReadyState为Complete的情况通常只在每次加载页面时出现一次,所以这时我们才应认为其已完全加载。但也并非一个页面100%只会出现一次这种情况,比如页面中Google的“更多”链接点击后仅仅是弹出一个列表,但这时又会出现一次Complete。

这会导致什么问题?

假如在ReadyState为Interactive甚至是Loading时对页面实施操控,那很可能无效或引发异常。

而假如忽视这种一页多发事件的情况,会导致大量重复操作,从而引发异常或逻辑问题。

该如何解决?

以下代码可以确保你不会在未加载完毕时执行操作:

private void webBrowser1_DocumentCompleted(object sender, WebBrowserDocumentCompletedEventArgs e)

{

    if (webBrowser1.ReadyState < WebBrowserReadyState.Complete) return;

    //执行正常流程代码…………

}

如果需要严格控制每页只能执行一次代码,那么需要附加一个属性或变量来记录上次访问的网址,并据此进行判断:

private void webBrowser1_DocumentCompleted(object sender, WebBrowserDocumentCompletedEventArgs e)

{

    if (webBrowser1.ReadyState < WebBrowserReadyState.Complete || webBrowser1.Url.ToString() == LastUrl) return;

    LastUrl = webBrowser1.Url.ToString();

    //执行正常流程代码…………

}

 

public string LastUrl

{

    get

    {

        return _LastUrl;

    }

    set

    {

        _LastUrl = value;

    }

}

private string _LastUrl;

结语

WebBrowser是非常实用的控件,但也可说是粗制滥造的典范,各种问题层出不穷,从注释对不上实际功能这一点来看,就知道开发者多么漫不经心了。还有就是ReadyState属性其实是一个非常关键的属性,而开发者却没有设置一个事件来监控其状态变更,真是太失败了。

更多相关经验请参看:WebBrowser控件使用技巧分享 在WebBrowser中通过模拟键盘鼠标操控网页中的文件上传控件

 

下载本文的示例源代码:http://www.uushare.com/user/icesee/file/1869325

下载本文的XPS版本:http://www.uushare.com/user/icesee/file/1869320


本文转自斯克迪亚博客园博客,原文链接:http://www.cnblogs.com/SkyD/archive/2009/08/06/1540028.html,如需转载请自行联系原作者

相关文章
|
8月前
|
C#
C# 禁止ComboBox滚动事件细谈
C# 禁止ComboBox滚动事件细谈
|
8月前
|
JavaScript 前端开发
js基础——事件(鼠标事件、键盘事件、表单事件......)
js基础——事件(鼠标事件、键盘事件、表单事件......)
64 0
|
C# 前端开发
wpf中的datagrid绑定操作按钮是否显示或者隐藏
如图,需要在wpf中的datagrid的操作那列有个确认按钮,然后在某些条件下确认按钮可见,某些情况下不可见的,放在mvc里直接在cshtml页面中if..else就行了。 但是在wpf里不行。。网上搜索了好久才找到解决方法,原来只是binding那个visiable属性就行了,
6906 0
|
C#
WPF,强制捕获鼠标事件,鼠标移出控件外依然可以执行强制捕获的鼠标事件
原文:WPF,强制捕获鼠标事件,鼠标移出控件外依然可以执行强制捕获的鼠标事件 在WPF中,只有鼠标位置在某个控件上的时候才会触发该控件的鼠标事件。例如,有两个控件都注册了MouseDown和MouseUp事件,在控件1上按下鼠标,不要放开,移动到控件2上再放开。
2332 0
|
C#
WPF Label控件在数据绑定Content属性变化触发TargetUpdated事件简单实现类似TextChanged 事件效果
原文:WPF Label控件在数据绑定Content属性变化触发TargetUpdated事件简单实现类似TextChanged 事件效果   本以为Label也有TextChanged 事件,但在使用的时候却没找到,网友说Label的Content属性改变肯定是使用赋值操作,赋值的时候就可以对其进行相应的操作所以不需TextChanged 事件。
2081 0
|
C# 前端开发
WPF当属性值改变时利用PropertyChanged事件来加载动画
原文:WPF当属性值改变时利用PropertyChanged事件来加载动画     在我们的程序中,有时我们需要当绑定到UI界面上的属性值发生变化从而引起数据更新的时候能够加载一些动画,从而使数据更新的效果更佳绚丽,在我们的程序中尽量将动画作为一种资源放在xaml中,而不是在后台中通过写代码的这种方式来加载动画,在我们的这篇博客中我们尽量使用简洁的语言来阐述这一方法。
1510 0
|
JavaScript C# 前端开发
c#控制IE浏览器自动点击等事件WebBrowser,mshtml.IHTMLDocument2
原文:c#控制IE浏览器自动点击等事件WebBrowser,mshtml.IHTMLDocument2 可以实现例如通过应用程序操作google搜索,用户输入要搜索的内容,然后在google中搜索;可以自动点击网页上的按钮等功能     1.
2549 0
WebBrowser 的 DocumentCompleted事件不执行的解决方法
原文:WebBrowser 的 DocumentCompleted事件不执行的解决方法 WebBrowser 的 DocumentCompleted事件不执行的解决方法: 使用WebBrowser的ProgressChanged事件,在时间中判断((WebBrowser)sender).ReadyState == WebBrowserReadyState.Complete是否成立,若成立则执行DocumentCompleted的处理。
1537 0