[WPF - 之一问一答系列] 如何从WPF的WebBrowser控件中获得WebResponse内容?为何WebBrowser控件的Navigated事件参数NavigationEventArgs的WebResponse属性始终为null?
2011-04-07 02:33
831 查看
问:
如何从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方法的逻辑代码:
可以看出,代码中直接使用null值构造了NavigationEventArgs参数,然后触发WebBrowser.Navigated事件。(当然,你也可以看DocumentComplete方法,在这个里面一样的用了null去构造后触发了WebBrowser.LoadCompleted事件)
到此可以解释为什么WebResponse属性总是返回null了。
等等,还没有结束。那么怎么去获得WebBrowser呢? 我们还省略没有去看另外的几个NavigationService的方法,比如System.Windows.Navigation.NavigationService.FireNavigated方法逻辑代码:
可以看出,NavigationService在他的事件中封装了WebResponse信息。这样的话,就有了解决方案——
使用NavigationService去Navigate页面,并且在Navigated事件中获得。但是WebBrowser本身没有支持NavigationService,在WPF中只有NavigationWindow和Frame支持了NavigationService,所以我们只需要使用这两者,这里我的代码是用了Frame替代了WebBrowser,
XAML:
<Frame x:Name="frame"/>
C#代码:
[备注:]在WebBrowser和Frame之间,我的第一感觉是,如果你需要一个浏览控件有依赖属性来支持绑定等WPF特性的话,那么就选择Frame吧,它包含你所需要的依赖属性,而WebBrowser没有。 不过我还会推荐下一个第三方的WPF浏览器控件:http://wpfchromium.codeplex.com/ 基于Google的Chromium项目的。
如何从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项目的。
相关文章推荐
- [WPF - 之一问一答系列] 如何从WPF的WebBrowser控件中获得WebResponse内容?为何WebBrowser控件的Navigated事件参数NavigationEventArgs的
- Flex中如何利用getTextField事件和numLines属性,计算出TextArea控件中内容的行数的例子
- Flex中如何利用getTextField事件和numLines属性,计算出TextArea控件中内容的行数的例子
- web页,如何按回车获得焦点,触发web控件事件..?
- android中如何获得webView中的内容
- 如何将webbrowser控件的Cookie倒入CookieContainer供WebRequest使用
- [c#]如何在form的webbrowser控件中获得鼠标坐标
- 浅析c#中如何在form的webbrowser控件中获得鼠标坐标
- 如何在webform.aspx.cs中控制用户控件的属性
- 在asp.net中为Web用户控件添加属性和事件
- web控件开发系列(五) 控件事件
- WPF 如何在后台进行控件属性的绑定MVVM
- 关于使用WinHtmlEditor控件获得html代码加载到wpf中webbrowser控件上乱码问题
- 如何获得web.xml里面的<context-param>内容
- 如何获得webview的内容
- C#后台如何获得web控件的值
- 如何通过javascript操作web控件的自定义属性
- WPF利用通过父控件属性来获得绑定数据源RelativeSource
- 【WPF学习笔记】之如何点击“新建”按钮,在面板中加载一条条的“用户控件”的信息:动画系列之(四)
- Android深入探究笔记之二 -- 打开一个新的 Activity 并传递参数与如何响应控件的点击事件