How can I overlop ie's BeforeNavigate2 when us

2019-05-25 08:07发布

问题:

As far as I understand, WPF's Brwoser control is a wrapper of the ie Active-X control. The later has a BeforeNavigate2 Method, while I don't find this in the WPF WebBrowser control. Is there a way I can work around this?

Thx! Marc

回答1:

Yes. The WebBrowser control of WPF is really braindead. On top of that it is even sealed.

You have to use the WebBrowser control of Windows.Forms and embed it in a WindowsFormsHost.

Additionally you have to derive a class from WebBrowser and do some COM magic in it.

Make a UserControl like this:

<UserControl x:Class="MP.Assistant.WpfClient.DialogContentControls.WebBrowserHost"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
         xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
         xmlns:DialogContentControls="clr-namespace:MP.Assistant.WpfClient.DialogContentControls"
         mc:Ignorable="d"
         d:DesignHeight="300"
         d:DesignWidth="300">
<Grid Name="BrowserHost">
    <WindowsFormsHost>
        <DialogContentControls:ExtendedWinFormsWebBrowser x:Name="WebBrowser" />
    </WindowsFormsHost>
</Grid>

Then modify Windows.Forms WebBrowser like this:

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

namespace MP.Assistant.WpfClient.DialogContentControls
{
    /// Imports the BeforeNavigate2 method from the OLE DWebBrowserEvents2 
    /// interface. 
    [ComImport]
    [Guid("34A715A0-6587-11D0-924A-0020AFC7AC4D")]
    [InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
    [TypeLibType(TypeLibTypeFlags.FHidden)]
    public interface DWebBrowserEvents2
    {
        [PreserveSig]
        [MethodImpl(MethodImplOptions.InternalCall, MethodCodeType =    MethodCodeType.Runtime)]
    [DispId(250)] 

    void BeforeNavigate2([In] [MarshalAs(UnmanagedType.IDispatch)] object pDisp,
                         [In] [MarshalAs(UnmanagedType.Struct)] ref object URL,
                         [In] [MarshalAs(UnmanagedType.Struct)] ref object Flags,
                         [In] [MarshalAs(UnmanagedType.Struct)] ref object TargetFrameName,
                         [In] [MarshalAs(UnmanagedType.Struct)] ref object PostData,
                         [In] [MarshalAs(UnmanagedType.Struct)] ref object Headers,
                         [In] [Out] ref bool Cancel);
}

public class ExtendedWinFormsWebBrowser : WebBrowser
{
    // Handles the NavigateError event from the underlying ActiveX 
    // control by raising the NavigateError event defined in this class.
    AxHost.ConnectionPointCookie cookie;
    ExtendedWinFormsWebBrowserEventHelper helper;
    bool renavigating;

    public ExtendedWinFormsWebBrowser()
    {
        AdditionalHeaders = new string[] {};
    }

    public string[] AdditionalHeaders { get; set; }

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

        // Create an instance of the client that will handle the event
        // and associate it with the underlying ActiveX control.
        helper = new ExtendedWinFormsWebBrowserEventHelper(this);
        cookie = new AxHost.ConnectionPointCookie(ActiveXInstance, helper, typeof (DWebBrowserEvents2));
    }

    [PermissionSetAttribute(SecurityAction.LinkDemand, Name = "FullTrust")]
    protected override void DetachSink()
    {
        // Disconnect the client that handles the event
        // from the underlying ActiveX control.
        if (cookie != null)
        {
            cookie.Disconnect();
            cookie = null;
        }
        base.DetachSink();
    }

    void OnBeforeNavigate2(object pDisp,
                           ref object url,
                           ref object flags,
                           ref object targetFrameName,
                           ref object postData,
                           ref object headers,
                           ref bool cancel)
    {
        if (!renavigating)
        {
            if (AdditionalHeaders.Length > 0)
            {
                headers += string.Join("\r\n", AdditionalHeaders) + "\r\n";
                renavigating = true;
                cancel = true;
                Navigate((string)url, (string)targetFrameName, (byte[])postData, (string)headers);
            }
        }
        else
        {
            renavigating = false;
        }
    }

    #region Nested type: ExtendedWinFormsWebBrowserEventHelper

    class ExtendedWinFormsWebBrowserEventHelper : StandardOleMarshalObject, DWebBrowserEvents2
    {
        readonly ExtendedWinFormsWebBrowser parent;

        public ExtendedWinFormsWebBrowserEventHelper(ExtendedWinFormsWebBrowser parent)
        {
            this.parent = parent;
        }

        #region DWebBrowserEvents2 Members

        public void BeforeNavigate2(object pDisp,
                                    ref object URL,
                                    ref object Flags,
                                    ref object TargetFrameName,
                                    ref object PostData,
                                    ref object Headers,
                                    ref bool Cancel)
        {
            parent.OnBeforeNavigate2(pDisp,
                ref URL,
                ref Flags,
                ref TargetFrameName,
                ref PostData,
                ref Headers,
                ref Cancel);
        }

        #endregion
    }

    #endregion
}
}

Then use WebBrowser (its named like that in XAML) from your code-behind and provide AdditionalHeaders with the required value.