Вызов методов С# BHO из Javascript

Я пытаюсь выяснить, как вызвать методы С# в моем объекте BHO из Javascript внутри страницы. Я нашел многочисленные страницы о том, как это сделать в С++/ATL/Com, например:

Параметры и параметры скриптов помощника браузера

Вызов метода BHO из Javascript?

Я попытался следовать и правильно реализовать его на С#, но я не могу заставить его работать, вероятно, из-за некоторых очевидных проблем COM, которые у меня есть, которые я не совсем понимаю.

Я использую С# 4.0.

Вот соответствующие части кода:

using SHDocVw;
using mshtml;
using System.Runtime.InteropServices;

[ComVisible(true),
 Guid("300736C4-DCDA-4DB0-90AD-4510A12EBBC6"),
 ClassInterface(ClassInterfaceType.None),
 ProgId("My Extension")]
public class BrowserHelperObject : IObjectWithSite
{
    const int DISPATCH_PROPERTYPUT = 4;
    const int FDEX_NAME_ENSURE = 2;
    const uint LOCALE_USER_DEFAULT = 0x0400;

    WebBrowser browser;

    ...
    public void OnDocumentComplete(dynamic frame, ref dynamic url)
    {
        ...
        var window = browser.Document.parentWindow;

        int pid = 0;
        window.GetDispId("myExtension", FDEX_NAME_ENSURE, ref pid);

        System.Runtime.InteropServices.ComTypes.DISPPARAMS dispParms = new System.Runtime.InteropServices.ComTypes.DISPPARAMS();
        dispParms.cArgs = 1;
        dispParms.cNamedArgs = 0;
        dispParms.rgvarg = ???;
        dispParms.rgdispidNamedArgs = IntPtr.Zero;
        System.Runtime.InteropServices.ComTypes.EXCEPINFO einfo = new System.Runtime.InteropServices.ComTypes.EXCEPINFO();
        window.Invoke(pid, LOCALE_USER_DEFAULT, DISPATCH_PROPERTYPUT, ref dispParms, this, ref einfo);            
        ...
    }

Ответ 1

Ненавижу отвечать на свой вопрос, но я действительно чувствую, что сообщество должно знать ответ, потому что это короткий, простой и красивый с С# 4.0, и многие люди, похоже, имеют эту проблему.

Убедитесь, что вы правильно раскрываете объект-помощник браузера:

[ComVisible(true),
 Guid("DA8EA345-02AE-434E-82E9-448E3DB7629E"),
 ClassInterface(ClassInterfaceType.None), ProgId("MyExtension"),
 ComDefaultInterface(typeof(IExtension))]
public class BrowserHelperObject : IObjectWithSite, IExtension
{
    ...
    public int Foo(string s) { ... }
    ...
    public void OnDocumentComplete(dynamic frame, ref dynamic url)
    {
        ...
        dynamic window = browser.Document.parentWindow;
        IExpando windowEx = (IExpando)window;
        windowEx.AddProperty("myExtension");
        window.myExtension = this;
        ...
    }
    ...
}

И вам понадобится определение для ваших расширений:

[ComVisible(true),
 Guid("4C1D2E51-018B-4A7C-8A07-618452573E42"),
 InterfaceType(ComInterfaceType.InterfaceIsDual)]
public interface IExtension
{
    [DispId(1)]
    int Foo(string s);
    ...
}

Вы можете получить доступ к своему помощнику браузера в javascript, таким образом:

var result = window.myExtension.Foo("bar");

или просто

var result = myExtension.Foo("bar");

Что это. Перестаньте стучать головой о стену и отправляйтесь праздновать!

Ответ 2

Кроме того, после того, как вы добавите свойство, вам нужно будет освободить COM-ссылки на window и windowEx