8

我正在尝试在我的一个应用程序中实现一个简单的 Web 浏览器控件。这是为了帮助将 Web 应用程序集成到我正在创建的工具集中。

问题是,这个网络应用程序绝对喜欢弹出窗口......

打开一个弹出窗口时,它会在一个 IE 窗口中打开,该窗口不是我的主窗口所在的 MDI 容器窗体的子窗体。

如何通过单击 WebBrowser 中的链接创建的任何和所有弹出窗口成为我的 MDI 容器的子项(类似于设置表单的 MDIParent 属性)?

提前致谢。

4

4 回答 4

26

Web 浏览器控件支持 NewWindow 事件以获取有关弹出窗口的通知。然而,Winforms 包装器并不能让你做太多事情,你只能取消弹出窗口。本机 COM 包装器允许传回 Web 浏览器的新实例,然后该实例将用于显示弹出窗口。

利用这一点需要一些工作。对于初学者,使用 Project + Add Reference,Browse 选项卡并选择 c:\windows\system32\shdocvw.dll。这增加了对本机 COM 接口的引用。

创建一个充当弹出表单的表单。在其上放置一个 WebBrowser 并使其代码如下所示:

public partial class Form2 : Form {
    public Form2() {
        InitializeComponent();
    }
    public WebBrowser Browser {
        get { return webBrowser1; }
    }
}

Browser 属性允许访问将用于在弹出窗口中显示网页的浏览器。

现在回到主窗体。在其上放置一个 WebBrowser 并使其代码如下所示:

public partial class Form1 : Form {
    public Form1() {
        InitializeComponent();
        webBrowser1.Url = new Uri("http://google.com");
    }
    SHDocVw.WebBrowser nativeBrowser;
    protected override void OnLoad(EventArgs e) {
        base.OnLoad(e);
        nativeBrowser = (SHDocVw.WebBrowser)webBrowser1.ActiveXInstance;
        nativeBrowser.NewWindow2 += nativeBrowser_NewWindow2;
    }
    protected override void OnFormClosing(FormClosingEventArgs e) {
        nativeBrowser.NewWindow2 -= nativeBrowser_NewWindow2;
        base.OnFormClosing(e);
    }

    void nativeBrowser_NewWindow2(ref object ppDisp, ref bool Cancel) {
        var popup = new Form2();
        popup.Show(this);
        ppDisp = popup.Browser.ActiveXInstance;
    }
}

OnLoad 方法获取对本机 COM 接口的引用,然后将事件处理程序订阅到 NewWindow2 事件。我确保在 FormClosing 事件处理程序中取消订阅该事件,而不是 100% 确定是否有必要。最好安全然后抱歉。

NewWindow2 事件处理程序是关键,请注意第一个参数允许传回无类型的引用。那应该是弹出窗口中的本机浏览器。所以我创建了一个 Form2 的实例并 Show() 它。注意 Show() 的参数,它确保弹出窗口是一个拥有的窗口。根据您的应用程序的需要替换它,我假设您希望在您的情况下创建一个 MDI 子窗口。

请注意,当 Javascript 使用 alert() 时显示的窗口不会触发此事件。浏览器不会将该窗口视为 HTML 弹出窗口,也不会使用浏览器窗口来显示它,因此您无法拦截或替换它。

于 2011-06-24T20:30:32.053 回答
3

我发现最好的方法是实现/接收NewWindow3 事件

添加对c:\windows\system32\shdocvw.dll的引用,如此处其他答案中所述。

添加事件处理程序

SHDocVw.WebBrowser wbCOMmain = (SHDocVw.WebBrowser)webbrowser.ActiveXInstance;
wbCOMmain.NewWindow3 += wbCOMmain_NewWindow3;

事件方法

void wbCOMmain_NewWindow3(ref object ppDisp, 
                          ref bool Cancel, 
                          uint dwFlags, 
                          string bstrUrlContext, 
                          string bstrUrl)
{
    // bstrUrl is the url being navigated to
    Cancel = true; // stop the navigation

    // Do whatever else you want to do with that URL
    // open in the same browser or new browser, etc.
}
  • 将“Interop.SHDocVw”程序集的“嵌入互操作类型”设置为 false
  • 将“本地副本”设置为 true。

该帮助的来源MSDN Post

于 2014-05-06T17:24:30.513 回答
2

提炼汉斯的答案,您可以在不添加引用的情况下派生用于访问 COM 的 WebBrowser。这是通过使用未发布的 Winforms WebBrowser.AttachInterface 和 DetachInterface 方法。

更详细的在这里

这是代码:

用法(将您的 WebBrowser 实例更改为 WebBrowserNewWindow2)

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();
        this.webBrowser1.NewWindow2 += webBrowser_NewWindow2;
    }

    protected override void OnFormClosing(FormClosingEventArgs e)
    {
        webBrowser1.NewWindow2 -= webBrowser_NewWindow2;
        base.OnFormClosing(e);
    }

    void webBrowser_NewWindow2(object sender, WebBrowserNewWindow2EventArgs e)
    {
        var popup = new Form1();
        popup.Show(this);
        e.PpDisp = popup.Browser.ActiveXInstance;
    }
    public WebBrowserNewWindow2 Browser
    {
        get { return webBrowser1; }
    }
}

代码:

using System;
using System.Security.Permissions;
using System.Runtime.InteropServices;
using System.Windows.Forms;

namespace SHDocVw
{
    public delegate void WebBrowserNewWindow2EventHandler(object sender, WebBrowserNewWindow2EventArgs e);

    public class WebBrowserNewWindow2EventArgs : EventArgs
    {
        public WebBrowserNewWindow2EventArgs(object ppDisp, bool cancel)
        {
            PpDisp = ppDisp;
            Cancel = cancel;
        }

        public object PpDisp { get; set; }
        public bool Cancel { get; set; }
    }

    public class WebBrowserNewWindow2 : WebBrowser
    {
        private AxHost.ConnectionPointCookie _cookie;
        private WebBrowser2EventHelper _helper;

        [PermissionSet(SecurityAction.LinkDemand, Name = "FullTrust")]
        protected override void CreateSink()
        {
            base.CreateSink();

            _helper = new WebBrowser2EventHelper(this);
            _cookie = new AxHost.ConnectionPointCookie(
                this.ActiveXInstance, _helper, typeof(DWebBrowserEvents2));
        }

        [PermissionSetAttribute(SecurityAction.LinkDemand, Name = "FullTrust")]
        protected override void DetachSink()
        {
            if (_cookie != null)
            {
                _cookie.Disconnect();
                _cookie = null;
            }
            base.DetachSink();
        }

        public event WebBrowserNewWindow2EventHandler NewWindow2;

        private class WebBrowser2EventHelper : StandardOleMarshalObject, DWebBrowserEvents2
        {
            private readonly WebBrowserNewWindow2 _parent;

            public WebBrowser2EventHelper(WebBrowserNewWindow2 parent)
            {
                _parent = parent;
            }

            public void NewWindow2(ref object pDisp, ref bool cancel)
            {
                WebBrowserNewWindow2EventArgs arg = new WebBrowserNewWindow2EventArgs(pDisp, cancel);
                _parent.NewWindow2(this, arg);
                if (pDisp != arg.PpDisp)
                    pDisp = arg.PpDisp;
                if (cancel != arg.Cancel)
                    cancel = arg.Cancel;
            }
        }

        [ComImport, Guid("34A715A0-6587-11D0-924A-0020AFC7AC4D"),
        InterfaceType(ComInterfaceType.InterfaceIsIDispatch),
        TypeLibType(TypeLibTypeFlags.FHidden)]
        public interface DWebBrowserEvents2
        {
            [DispId(0xfb)]
            void NewWindow2(
                [In, Out, MarshalAs(UnmanagedType.IDispatch)] ref object ppDisp,
                [In, Out] ref bool cancel);
        }
    }
}
于 2013-12-20T18:28:32.570 回答
0

我知道这个问题很老了,但我这样解决了:添加新引用,在 COM 中选择 Microsoft Internet Controls 并在代码中,在单击打开新窗口之前添加以下内容:

SHDocVw.WebBrowser_V1 axBrowser = (SHDocVw.WebBrowser_V1)webBrowser1.ActiveXInstance;
axBrowser.NewWindow += axBrowser_NewWindow;

然后添加以下方法:

void axBrowser_NewWindow(string URL, int Flags, string TargetFrameName, ref object PostData, string Headers, ref bool Processed)
{
    Processed = true;
    webBrowser1.Navigate(URL);
}
于 2020-06-09T17:22:24.393 回答