Sterowanie Windows Forms Webbrowswer za pomocą IDownloadManager

UżywamSystems.Windows.Forms.Webbrowser kontrola i należy zastąpić menedżera pobierania. Postępowałem zgodnie z instrukcjamitutaj podklasować formularz i nadpisaćCreateWebBrowserSiteBase()

/// <summary>
/// Browser with download manager
/// </summary>
public class MyBrowser : WebBrowser  
{
    /// <summary>
    /// Returns a reference to the unmanaged WebBrowser ActiveX control site,
    /// which you can extend to customize the managed <see ref="T:System.Windows.Forms.WebBrowser"/> control.
    /// </summary>
    /// <returns>
    /// A <see cref="T:System.Windows.Forms.WebBrowser.WebBrowserSite"/> that represents the WebBrowser ActiveX control site.
    /// </returns>
    protected override WebBrowserSiteBase CreateWebBrowserSiteBase()
    {
        var manager = new DownloadWebBrowserSite(this);
        manager.FileDownloading += (sender, args) =>
            {
                if (FileDownloading != null)
                {
                    FileDownloading(this, args);
                }
            };
        return manager;
    }
}

wDownloadWebBrowserSite, WdrażamIServiceProvider zapewnićIDownloadManager na żądanie.

        /// <summary>
        /// Queries for a service
        /// </summary>
        /// <param name="guidService">the service GUID</param>
        /// <param name="riid"></param>
        /// <param name="ppvObject"></param>
        /// <returns></returns>
        public int QueryService(ref Guid guidService, ref Guid riid, out IntPtr ppvObject)
        {
            if ( (guidService == Constants.IID_IDownloadManager && riid == Constants.IID_IDownloadManager ))
            {
                ppvObject = Marshal.GetComInterfaceForObject(_manager, typeof(IDownloadManager));
                return Constants.S_OK;
            }
            ppvObject = IntPtr.Zero;
            return Constants.E_NOINTERFACE;
        }

TheDownloadManager pochodzi z powyższego przykładu.

/// <summary>
/// Intercepts downloads of files, to add as PDFs or suppliments
/// </summary>
[ComVisible(true)]
[Guid("bdb9c34c-d0ca-448e-b497-8de62e709744")]
[CLSCompliant(false)]
public class DownloadManager : IDownloadManager
{
    /// <summary>
    /// event called when the browser is about to download a file
    /// </summary>
    public event EventHandler<FileDownloadEventArgs> FileDownloading;

    /// <summary>
    /// Return S_OK (0) so that IE will stop to download the file itself. 
    /// Else the default download user interface is used.
    /// </summary>
    public int Download(IMoniker pmk, IBindCtx pbc, uint dwBindVerb, int grfBINDF, IntPtr pBindInfo,
                        string pszHeaders, string pszRedir, uint uiCP)
    {
        // Get the display name of the pointer to an IMoniker interface that specifies the object to be downloaded.
        string name;
        pmk.GetDisplayName(pbc, null, out name);
        if (!string.IsNullOrEmpty(name))
        {
            Uri url;
            if (Uri.TryCreate(name, UriKind.Absolute, out url))
            {
                if ( FileDownloading != null )
                {
                     FileDownloading(this, new FileDownloadEventArgs(url));
                }
                return Constants.S_OK;
            }
        }
        return 1;
    }
}

Problem polega na tym, żepmk.GetDisplayName zwraca początkowy adres URL, a nie adres URL elementu do pobrania. Jeśli identyfikator URI wskazuje dynamiczną stronę, taką jakhttp://www.example.com/download.php, Nie pobieram rzeczywistego pliku. Muszę pobrać adres URL z nagłówka, aby uzyskać rzeczywisty plik, który powinienem pobierać.

Google wskazuje, że muszę utworzyćIBindStatusCallback implementacja, która również implementujeIHttpNegotiate iIServiceProvider odpowiedziećIID_IHttpNegotiate, abym mógł zobaczyćIHttpNegotiate.OnResponse. Udało mi się to jednak wdrożyćQueryService wydaje się, że kiedykolwiek o to prosiłIID_IInternetProtocol i nigdyIID_IHttpNegotiate.

Każda rada byłaby świetna.

questionAnswers(1)

yourAnswerToTheQuestion