[WPF - 之一问一答系列] 如何从WPF的WebBrowser控件中获得WebResponse内容?为何WebBrowser控件的Navigated事件参数NavigationEventArgs的WebResponse属性始终为null?
问:
如何从WPF的WebBrowser控件中获得WebResponse内容?为何WebBrowser控件的Navigated事件参数NavigationEventArgs的WebResponse属性始终为null?
我们在一个WPF的WebBrowser的Navigated事件中,尝试去输出NavigationEventArgs e的WebResponse属性,他始终是null。比如,
XAML代码:
<WebBrowser x:Name="browser"/>C#代码:
browser.Navigated += new NavigatedEventHandler(browser_Navigated);
browser.Navigate(new Uri("http://www.microsoft.com"));//省略...
void browser_Navigated(object sender, NavigationEventArgs e)
{
Console.WriteLine(e.WebResponse.Headers); //输出null
}[相关MSDN英文贴:http://social.msdn.microsoft.com/Forums/en-US/wpf/thread/856608db-48ba-4492-bff1-3721618ff3ae]
答:
一般都会认为从WebBrowser中获得WebResponse内容,可以从Navigated事件的NavigationEventArgs参数中获得。但是,事实是这个参数始终在返回null值,导致无法真正获得返回的内容。
解决这个问题,首先想到的是去了解WebBrowser控件的基本组成。我们知道,IE的结构是如下:
而对于一个WebBrowser控件(无论是Winform的还是WPF的),都是和这个架构类似的,不同的只是最上层的UI封装。控件来说,是没有IExplore.exe的包装的,所以一般控件就没有IE的某些特性,但是从底层来说,他们都使用了ShDocVw.dll和MSHTML.dll,控件只是对下方组件的一次封装。那么这个Navigated事件是从谁抛出的呢?答案是ShDocVw.dll,当然,面向不同的封装,抛到上方会遇到不同的处理。Wnform中的WebBrowserNavigatedEventArgs中没有封装WebResponse信息,而从WPF的WebBrowser中,却包含了这个属性。
接下来,从表象很难找到原因了,这个时候就需要借助下工具,比如我使用了Reflector (从这里可以下载到试用版:http://reflector.red-gate.com/download.aspx?TreatAsUpdate=1)去反编译了一些源代码,尝试去分析下为什么会一直返回null。通过工具分析,我们可以找到下面的一个调用列表:
在NavigationEventArgs类中的_webResponse成员实际就是我们要分析的对象,他仅由NavigationEventArgs的构造函数调用。而从他的构造函数的被调用列表中,我们发现了两类方法,一类是被定义在MS.Internal.Controls.WebBrowserEvent中的,还有是定义在System.Windows.Navigation.NavigationService中的。当然,我们可以想到,第一类是所谓WebBrowser控件的内部组件类,这个类实际就是所谓架构中对于ShDocVw.dll的一次.Net封装。所以这个类的方法就是WebBrowser在Navigated的时候调用的。点击显示MS.Internal.Controls.WebBrowserEvent.NavigateComplete2方法的逻辑代码:
[SecurityTreatAsSafe, SecurityCritical]
public void NavigateComplete2(object pDisp, ref object url)
{
//省略…
NavigationEventArgs e = new NavigationEventArgs(uri, null, null, null, null, true);
this._parent.OnNavigated(e);
//省略…
}可以看出,代码中直接使用null值构造了NavigationEventArgs参数,然后触发WebBrowser.Navigated事件。(当然,你也可以看DocumentComplete方法,在这个里面一样的用了null去构造后触发了WebBrowser.LoadCompleted事件)
到此可以解释为什么WebResponse属性总是返回null了。
等等,还没有结束。那么怎么去获得WebBrowser呢? 我们还省略没有去看另外的几个NavigationService的方法,比如System.Windows.Navigation.NavigationService.FireNavigated方法逻辑代码:
private void FireNavigated(object navState)
{
object extraData = (navState is NavigateInfo) ? null : navState;
try
{
NavigationEventArgs e = new NavigationEventArgs(this.CurrentSource, this.Content, extraData, this._webResponse, this.INavigatorHost, this.IsNavigationInitiator);
//省略…
}可以看出,NavigationService在他的事件中封装了WebResponse信息。这样的话,就有了解决方案——
使用NavigationService去Navigate页面,并且在Navigated事件中获得。但是WebBrowser本身没有支持NavigationService,在WPF中只有NavigationWindow和Frame支持了NavigationService,所以我们只需要使用这两者,这里我的代码是用了Frame替代了WebBrowser,
XAML:
<Frame x:Name="frame"/>
C#代码:
frame.Navigated += new NavigatedEventHandler(frame_Navigated);
frame.NavigationService.Navigate(new Uri("http://www.microsoft.com"));
//省略...
void frame_Navigated(object sender, NavigationEventArgs e)
{
Console.WriteLine(e.WebResponse.Headers);
}
[备注:]在WebBrowser和Frame之间,我的第一感觉是,如果你需要一个浏览控件有依赖属性来支持绑定等WPF特性的话,那么就选择Frame吧,它包含你所需要的依赖属性,而WebBrowser没有。 不过我还会推荐下一个第三方的WPF浏览器控件:http://wpfchromium.codeplex.com/ 基于Google的Chromium项目的。