天天看點

開發IE下js使用的com元件,包含事件處理

很簡單,将事件做成屬性,并設定屬性的bindable, displaybind标記。

在js下面直接将function傳給這個屬性,com元件裡面調用這個IDispatch的函數即可。

idl檔案,注意onevent1這裡。

import "oaidl.idl";
import "ocidl.idl";

[
	object,
	uuid(6A6140E8-9356-4FBD-B0FD-04508D336B17),
	dual,
	nonextensible,
	pointer_default(unique)
]
interface IComClassTest : IDispatch{
	[id(1)] HRESULT Method1(BSTR arg1);
	[propget, id(2), bindable, displaybind] HRESULT onevent1([out, retval] VARIANT* pVal);
	[propput, id(2), bindable, displaybind] HRESULT onevent1([in] VARIANT newVal);
};
[
	uuid(97157E1C-637C-4962-B069-96E70F572A99),
	version(1.0),
]
library ATLProject3Lib
{
	importlib("stdole2.tlb");
	[
		uuid(14D66A8E-899E-4A72-A2FD-3F9720BD1DF7)		
	]
	dispinterface _IComClassTestEvents
	{
		properties:
		methods:
	};
	[
		uuid(887AF328-699E-4750-8FE7-F3D715CA7983)		
	]
	coclass ComClassTest
	{
		[default] interface IComClassTest;
		[default, source] dispinterface _IComClassTestEvents;
	};
};
           

部分實作代碼

STDMETHODIMP CComClassTest::Method1(BSTR arg1)
{
	// TODO:  在此添加實作代碼
	if (m_onevent1) {
		DISPPARAMS params = { nullptr, nullptr, 0, 0 };
		m_onevent1.Invoke0((DISPID)0);
	}
	return S_OK;
}

STDMETHODIMP CComClassTest::get_onevent1(VARIANT* pVal)
{
	// TODO:  在此添加實作代碼
	pVal->vt = VT_DISPATCH;
	V_DISPATCH(pVal) = m_onevent1;
	return S_OK;
}

STDMETHODIMP CComClassTest::put_onevent1(VARIANT newVal)
{
	// TODO:  在此添加實作代碼
	m_onevent1 = V_DISPATCH(&newVal);
	return S_OK;
}
           

m_onevent是CComDispatchDriver對象。

測試例子

<html>
<body>
	<script type="text/javascript">
		var t = new ActiveXObject("ATLProject3.ComClassTest");
		t.onevent1 = function() {
			console.log("sdfsdfsdf");
		}
		t.Method1("123");
	</script>
</body>
</html>
           

在IE中重新整理這個頁面,在控制台裡面能看到效果。

不過,現在會提示ActiveX安全性問題。在ATL技術内幕這本書裡面有講到怎麼不讓它提示,現在找不到在哪裡了,等以後看到了再補上吧。