将事件从.NET暴露给COM

时间:2009-01-07 18:31:34

标签: .net events com interop iunknown

最近我遇到了将事件从.NET暴露给COM的问题。

我在这个例子中取得了成功(概念上取自http://blogs.msdn.com/andreww/archive/2008/10/13/exposing-events-from-managed-add-in-objects.aspx):

//我们自定义事件的委托类型。

[ComVisible(false)]
public delegate void SomeEventHandler(object sender, EventArgs e);

// Outgoing (source/event) interface.
[ComVisible(true)]
[InterfaceType(ComInterfaceType.InterfaceIsIDispatch)]
public interface IAddInEvents
{
    [DispId(1)]
    void SomeEvent(object sender, EventArgs e);
}

[ComVisible(true)]
[ClassInterface(ClassInterfaceType.None)]
[ComSourceInterfaces(typeof(IAddInEvents))]
public class AddInUtilities :
{       
    // Event field. This is what a COM client will hook up
    // their sink to.
    public event SomeEventHandler SomeEvent;

    inernal void FireEvent(object sender, EventArgs e)
    {
        if (SomeEvent != null)
        {
            SomeEvent(sender, e);
        }
    }
}

这很好用,因为IAddInEvents接口定义为 IDispatch 。但是,我需要发布一个IUnknown 的事件源接口。我无法控制事件界面,因为它来自第三方库(也将是已发布事件的使用者)。每当我尝试挂钩事件时,VB环境(我尝试接收事件的地方)都会崩溃,第三方产品(ESRI ArcMap)使用的VBA环境也会崩溃。

我已经能够(部分)手动实现 IConnectionPointContainer 接口(COM在后台使用它来处理事件),然后我能够接收事件并进入我的IConnectionPointContainer实现。但是,这似乎对我来说太过分了,我认为在.NET中必须有一个隐含的支持。其次,通过这种方法,我立即失去了代表支持。

有没有人有这方面的经验? 提前谢谢。

2 个答案:

答案 0 :(得分:2)

很简单,你将无法做到这一点。 Classic VB不支持非自动化COM(如您所见)。

您必须拥有一个包装器,您可以将自动化实例传递给公共非自动化事件。您实际上必须有两种不同的类型来处理事件的两个独立客户端(启用自动化和启用非自动化)。

答案 1 :(得分:1)

好的,所以我通过实现经典的COM IConnectionPointCointainer,IConnectionPoint和IConnection(以及枚举接口)来实现这一点。它没有集成到.NET委托/事件模型中,但可以工作。