「ATLシンプルオブジェクト」ウィザードを使用して、ATLを使用してC++でCOMサーバーDLLを構築しました。MicrosoftのATLDLLCOMServerの例に従いました。VBScriptでCOMイベントを受信しないという1つのことを除いて、すべてがうまく機能します。私はC#でイベントを受け取ります。以前のMFCベースの実装のVBScriptでActiveXコントロールとして機能するイベントがありました。
私のコントロールは次のように定義されています。
class ATL_NO_VTABLE CSetACLCOMServer :
public CComObjectRootEx<CComSingleThreadModel>,
public CComCoClass<CSetACLCOMServer, &CLSID_SetACLCOMServer>,
public IConnectionPointContainerImpl<CSetACLCOMServer>,
public CProxy_ISetACLCOMServerEvents<CSetACLCOMServer>,
public IDispatchImpl<ISetACLCOMServer, &IID_ISetACLCOMServer, &LIBID_SetACLCOMLibrary, /*wMajor =*/ 1, /*wMinor =*/ 0>,
public IProvideClassInfo2Impl<&CLSID_SetACLCOMServer, &DIID__ISetACLCOMServerEvents, &LIBID_SetACLCOMLibrary> /* Required for event support in VBS */
{
public:
BEGIN_COM_MAP(CSetACLCOMServer)
COM_INTERFACE_ENTRY(ISetACLCOMServer)
COM_INTERFACE_ENTRY(IDispatch)
COM_INTERFACE_ENTRY(IConnectionPointContainer)
COM_INTERFACE_ENTRY(IProvideClassInfo)
COM_INTERFACE_ENTRY(IProvideClassInfo2)
END_COM_MAP()
BEGIN_CONNECTION_POINT_MAP(CSetACLCOMServer)
CONNECTION_POINT_ENTRY(__uuidof(_ISetACLCOMServerEvents))
END_CONNECTION_POINT_MAP()
[...]
VBScriptでは、次のようなCOMオブジェクトを使用します。
set objSetACL = WScript.CreateObject("SetACL.SetACL", "SetACL_")
' Catch and print messages from the SetACL COM server which are passed (fired) as events.
' The name postfix of this function (MessageEvent) must be identical to the event name
' as defined by SetACL.
' The prefix (SetACL_) can be set freely in the call to WScript.CreateObject
sub SetACL_MessageEvent (Message)
WScript.Echo Message
end sub
IDLの関連部分は次のようになります。
[
object,
uuid(E1B57CA5-FD7F-4304-BC0A-8BEBDE231D53),
dual,
nonextensible,
pointer_default(unique),
helpstring("SetACL COM Interface")
]
interface ISetACLCOMServer : IDispatch
{
};
[
uuid(00D4DCD3-02B9-4A71-AB61-2283504620C8),
version(1.0),
helpstring ("SetACL Type Library")
]
library SetACLCOMLibrary
{
importlib("stdole2.tlb");
[
uuid(35F76182-7F52-4D6A-BD6E-1317345F98FB),
helpstring ("SetACL Event Interface")
]
dispinterface _ISetACLCOMServerEvents
{
properties:
methods:
[id(1), helpstring("Receives string messages that would appear on the screen in the command line version")]
void MessageEvent([in] BSTR message);
};
[
uuid(13379563-8F21-4579-8AC7-CBCD488735DB),
helpstring ("SetACL COM Server"),
]
coclass SetACLCOMServer
{
[default] interface ISetACLCOMServer;
[default, source] dispinterface _ISetACLCOMServerEvents;
};
};
問題:SetACL_MessageEventが呼び出されることはありません。
私が試したこと:
- このKB記事を読んだ後、IProvideClassInfo2の実装を追加しましたが、それは役に立ちませんでした。
- このFAQには、発信インターフェイスをデュアルインターフェイスとして定義するべきではないことが記載されています。インターフェイス定義から「デュアル」を削除しましたが、それは役に立ちませんでした。
- 同様の問題を説明しているように見えるこのSOの質問を見つけました。答えは決して与えられず、回避策だけでした。
更新1:
どこで失敗しているのかはわかりましたが、理由はわかりません。_ISetACLCOMServerEvents_CP.hのfire eventメソッドで、Invokeの呼び出しがE_UNEXPECTEDで失敗します。これは次の行です。
hr = pConnection->Invoke(1, IID_NULL, LOCALE_USER_DEFAULT, DISPATCH_METHOD, ¶ms, NULL, NULL, NULL);
解決:
問題は、COMオブジェクトのバックグラウンドスレッドから、つまり間違ったアパートメントからイベント発生関数を呼び出したことです。