<=======================
作者: 谭剑
博客:   http://hi.baidu.com/tanjian
关键字:C#,.NET,WebBrowser,alert,ExtendedWebBrowser,浏览器,对话框
=======================>
截获"浏览器信息对话框"弹出消息:
    基于某些特殊需求的需要,程序需要知道浏览器控件信息对话框何时弹出了,消息的内容是什么,以及其它相关的信息.
    思路一:
    向浏览的页面中添加如下代码(为避免与页面中原有的元素相冲突,可以将下面代码内的方法名,元素名等替换成不同的guid)

 

    并在宿主程序中响应该button的click事件消息:
((ExtendedWebBrowser)sender).Document.All["myButton"].Click +=
                new HtmlElementEventHandler(myAlertElement_Click);
    这样一来,便可以一种迂回的方式巧妙获取到alert消息框弹出事件.
    但是通过使用HtmlElement.AppendChild(..) / HtmlDocument.Write(..) 以及 HTMLWindow2.execScript(...) 等方法向载入的页面中添加上述代码,均未能得到理想的结果,遂暂时放弃(哪位知道该怎么弄的话,请不吝赐教!!!)
    思路二:
    额外开一监视线程,实时枚举所有子窗口.以达到获取该消息框弹出事件的目的.
    可以预计,该方法会影响到整个系统的性能,会影响到程序的稳定性,同时其可靠性也不高(不能保证每次弹出消息框时刚好能够枚举到).
    所以这只能作为备用方法,在没有找到其它更好方法的情况下,勉强用用.
    思路三:
    再一次扩展浏览器控件.
    查询MSDN,可以找到对WebBrowser.CreateWebBrowserSiteBase 方法的描述:
    『返回对非托管 WebBrowser ActiveX 控件站点的引用,扩展该站点可以对托管 WebBrowser 控件进行自定义。
    若要使用此功能,请实现从   WebBrowser    WebBrowser.WebBrowserSite  类继承的类。非托管   WebBrowser  ActiveX 控件使用受保护的 CreateWebBrowserSiteBase  方法检索由   WebBrowser.WebBrowserSite  类实现的扩展性接口。重写   CreateWebBrowserSiteBase  方法以返回自己的从 WebBrowser.WebBrowserSite  类继承的类的实例。 WebBrowser.WebBrowserSite  类提供 OLE   IDocHostUIHandler  的默认实现。您可以为此接口提供自己的实现,或者实现任何其他   WebBrowser  ActiveX 控件接口,从而对控件行为进行自定义。』
    可以看出,我们只需重写 CreateWebBrowserSiteBase  方法,扩展 WebBrowserSite  ,便可实现对浏览器控件的扩展。
    我们知道,IDocHostUIHandler ,IDocHostUIHandler2 和 IDocHostShowUI 三个 接口是浏览器控件用户界面的自定义核心。当一个容器提供对ActiveX   控件支持 时候 ,   当浏览器控件被实例化 ,如果 可能的话,它尝试找来自 宿主的 IDocHostUIHandler , IDocHostUIHandler2 和 IDocHostShowUI   实现。当我们修改 浏览器控件的时候 , 这些是我们应该在程序中实现的 接口
    在这里,我们需要的仅仅是 截获"浏览器信息 对话框"弹出消息,所以我们也没有必要把所有这些接口全部实现一遍。
    IDocHostShowUI 这个接口提供给我们对浏览器控件显示信息对话框和帮助的控制。我们实现它,这样在浏览器控件显示它自己的任何的信息或帮助之前 ,能调用我们定义的IDocHostShowUI方法。这样我们便有一个机会捕获该信息对话框弹出的消息。IDocHostShowUI有两个方法,IDocHostShowUI::ShowMessage和IDocHostShowUI::ShowHelp。
    已经准备得很充分了。现在开工!!!
    在文件UnsafeNativeMethods.cs的    public class UnsafeNativeMethods里面添加如下代码:
        #region IDocHostShowUI
        [StructLayout(LayoutKind.Explicit, Pack = 4)]
        public struct __MIDL_IWinTypes_0009
        {
            // Fields
            [FieldOffset(0)]
            public int hInproc;
            [FieldOffset(0)]
            public int hRemote;
        }
        [StructLayout(LayoutKind.Sequential, Pack = 4)]
        public struct _RemotableHandle
        {
            public int fContext;
            public __MIDL_IWinTypes_0009 u;
        }
        [StructLayout(LayoutKind.Sequential, Pack = 4)]
        public struct tagPOINT
        {
            public int x;
            public int y;
        }
        [ComImport, Guid("C4D244B0-D43E-11CF-893B-00AA00BDCE1A"), InterfaceType((short)1)]
        public interface IDocHostShowUI
        {
            [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
            void ShowMessage([In, ComAliasName("ExtendedWebBrowser2.UnsafeNativeMethods.wireHWND")] ref _RemotableHandle hwnd, [In, MarshalAs(UnmanagedType.LPWStr)] string lpstrText, [In, MarshalAs(UnmanagedType.LPWStr)] string lpstrCaption, [In] uint dwType, [In, MarshalAs(UnmanagedType.LPWStr)] string lpstrHelpFile, [In] uint dwHelpContext, [ComAliasName("ExtendedWebBrowser2.UnsafeNativeMethods.LONG_PTR")] out int plResult);
            [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType = MethodCodeType.Runtime)]
            void ShowHelp([In, ComAliasName("ExtendedWebBrowser2.UnsafeNativeMethods.wireHWND")] ref _RemotableHandle hwnd, [In, MarshalAs(UnmanagedType.LPWStr)] string pszHelpFile, [In] uint uCommand, [In] uint dwData, [In] tagPOINT ptMouse, [Out, MarshalAs(UnmanagedType.IDispatch)] object pDispatchObjectHit);
        }
        #endregion
    在文件ExtendedWebBrowser.cs的    public class ExtendedWebBrowser : System.Windows.Forms.WebBrowser里面添加如下代码:
        #region ExtendedWebBrowserSite
        class ExtendedWebBrowserSite : WebBrowser.WebBrowserSite, UnsafeNativeMethods.IDocHostShowUI
        {
            public ExtendedWebBrowserSite(WebBrowser host)
                : base(host)
            {
            }
            void UnsafeNativeMethods.IDocHostShowUI.ShowMessage(ref UnsafeNativeMethods._RemotableHandle hwnd, string lpstrText, string lpstrCaption, uint dwType, string lpstrHelpFile, uint dwHelpContext, out int plResult)
            {
                //TODO:自定义
            }
            void UnsafeNativeMethods.IDocHostShowUI.ShowHelp(ref UnsafeNativeMethods._RemotableHandle hwnd, string pszHelpFile, uint uCommand, uint dwData, UnsafeNativeMethods.tagPOINT ptMouse, object pDispatchObjectHit)
            {
                //TODO:自定义
            }
        }
        protected override WebBrowserSiteBase CreateWebBrowserSiteBase()
        {
            return new ExtendedWebBrowserSite(this);
        }
        #endregion
    OK,现在便可以添加相应的自定义事件了,这个轻车熟路,不再赘述。
说明:本文为《对ExtendedWebBrowser的再扩展》的续,某些地方请自行参看前文。