C++代碼中回調JS方法


前段時間開發了一個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>

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM