前段時間開發了一個COM組件配合web前端使用,遇到了C++中調用JS代碼的問題,在網上查了很多資料,現 總結一下,留作以後察看。
C++中調用JS代碼主要有兩種情況:1. IE線程中調用;2. 其他線程調用
1. IE線程中調用:這種情況網上已經有很多資料,下面列出示列代碼:
C++代碼
STDMETHODIMP CJsInvoker::InvokeJsFunc(LONG para1, LONG para2, VARIANT jsFunction, LONG*
retValue)
{
CComPtr<IDispatch> jsCallback;
if (jsFunction.vt == VT_DISPATCH)
jsCallback = jsFunction.pdispVal;
VARIANT arg[2];
arg[0].vt = VT_I4;
arg[1].vt = VT_I4;
arg[0].lVal = para1;
arg[1].lVal = para2;
VARIANT pvarRet;
jsCallback.InvokeN(static_cast<DISPID>(DISPID_VALUE), arg, 2, &pvarRet);
*retValue = pvarRet.lVal;
return S_OK;
}
JS代碼
<script type="text/javascript">
// 兩個參數的回調方法
function jsCallbackFunc(a, b)
{
return a + b;
}
var obj = new ActiveXObject("ComCallJsFunction.JsInvoker");
var retValue = objA.InvokeJsFunc(1, 2, jsCallbackFunc);
alert(retValue); // 返回值為3
</script>
從代碼中可以看出,Js方法作為IDispatch指針傳入COM,C++通過調用其InvokeN 方法實現。
2. 其他線程調用:與IE線程直接調用的區別在於需要列集與反列集,原因是JS代碼是運行 在自己的套間線程裡的,其他線程是不能直接訪問的,只能通過代理進入消息循環中。
C++代碼
STDMETHODIMP CJsInvoker::InvokeJsFunc3(LONG para1, LONG para2, VARIANT jsFunction,
LONG* retValue)
{
// Check whether is valid Dispatch interface.
if (V_VT(&jsFunction) != VT_DISPATCH || jsFunction.pdispVal == NULL) {
return E_INVALIDARG;
}
// 對IDispatch指針列集
CoMarshalInterThreadInterfaceInStream(IID_IDispatch, jsFunction.pdispVal,
&m_stream_jsfunc);
m_hTread = CreateThread(NULL, 0, ThreadFunction, this, NULL, NULL);
return S_OK;
}
DWORD WINAPI ThreadFunction(LPVOID pParam)
{
::CoInitialize(NULL);
CJsInvoker* pJsInvoker = (CJsInvoker*)pParam;
CComPtr<IDispatch> script;
// 反列集得到IDisPatch指針
CoGetInterfaceAndReleaseStream(pJsInvoker->m_stream_jsfunc, IID_IDispatch, (LPVOID *)
&script);
VARIANT arg[2];
arg[0].vt = VT_I4;
arg[1].vt = VT_I4;
arg[0].lVal = 1;
arg[1].lVal = 2;
VARIANT pvarRet;
script.InvokeN(static_cast<DISPID>(DISPID_VALUE), arg, 2, &pvarRet);
::CoUninitialize();
return S_OK;
}
JS代碼
<script type="text/javascript">
// 兩個參數的回調方法
function jsCallbackFunc(a, b)
{
return a + b;
}
var obj = new ActiveXObject("ComCallJsFunction.JsInvoker");
var retValue = objA.InvokeJsFunc(1, 2, jsCallbackFunc);
alert(retValue); // 此時返回值沒有意義
</script>