Вызов функции C ++ из JavaScript-скрипта, запущенного в веб-браузере - PullRequest
12 голосов
/ 20 сентября 2010

Я встроил элемент управления веб-браузера в свое приложение на С ++. Я хочу, чтобы javascript, запущенный в элементе управления веб-браузера, мог вызывать функцию / метод c ++.

Я нашел упоминания о трех способах сделать это:

  1. Реализация компонента ActiveX, который действует как посредник. (Подробности реализации здесь: http://blogs.msdn.com/b/nicd/archive/2007/04/18/calling-into-your-bho-from-a-client-script.aspx)
  2. Использовать window.external. (Также обсуждается в ссылке выше, но реализация не предусмотрена)
  3. Добавление пользовательского объекта в объект окна

Я хочу перейти к третьему варианту, но я не нашел ни одного рабочего примера того, как это сделать. Может кто-нибудь, пожалуйста, покажите мне, как это сделать, или ссылку на рабочий пример в сети где-то.

Наиболее близким примером, который я нашел, является первый ответ Игоря Тандетника в ветке в группе новостей webbrowser_ctl . Но я боюсь, что мне нужно больше помощи, чем это.

Я встраиваю элемент управления IWebBrowser2 и не использую MFC, ATL или WTL.

EDIT:

Исходя из псевдокода, заданного Игорем в связанном ранее потоке, и кода, найденного в статье проекта кода " Создание массивов JavaScript и других объектов из C ++ " Я создал некоторый код.

void WebForm::AddCustomObject(IDispatch *custObj, std::string name)
{
    IHTMLDocument2 *doc = GetDoc();
    IHTMLWindow2 *win = NULL;
    doc->get_parentWindow(&win);

    if (win == NULL) {
        return;
    }

    IDispatchEx *winEx;
    win->QueryInterface(&winEx);

    if (winEx == NULL) {
        return;
    }

    int lenW = MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, name.c_str(), -1, NULL, 0);
    BSTR objName = SysAllocStringLen(0, lenW);
    MultiByteToWideChar(CP_ACP, MB_PRECOMPOSED, name.c_str(), -1, objName, lenW);

    DISPID dispid; 
    HRESULT hr = winEx->GetDispID(objName, fdexNameEnsure, &dispid);

    SysFreeString(objName);

    if (FAILED(hr)) {
        return;
    }

    DISPID namedArgs[] = {DISPID_PROPERTYPUT};
    DISPPARAMS params;
    params.rgvarg = new VARIANT[1];
    params.rgvarg[0].pdispVal = custObj;
    params.rgvarg[0].vt = VT_DISPATCH;
    params.rgdispidNamedArgs = namedArgs;
    params.cArgs = 1;
    params.cNamedArgs = 1;

    hr = winEx->InvokeEx(dispid, LOCALE_USER_DEFAULT, DISPATCH_PROPERTYPUT, &params, NULL, NULL, NULL); 

    if (FAILED(hr)) {
        return;
    }
}

Приведенный выше код работает до конца, поэтому все выглядит хорошо.

Я вызываю AddCustomObject, когда получаю событие DISPID_NAVIGATECOMPLETE2 DWebBrowserEvents2, передавая это как *custObj:

class JSObject : public IDispatch {
private:
    long ref;

public:
    // IUnknown
    virtual HRESULT STDMETHODCALLTYPE QueryInterface(REFIID riid, void **ppv);
    virtual ULONG STDMETHODCALLTYPE AddRef();
    virtual ULONG STDMETHODCALLTYPE Release();

    // IDispatch
    virtual HRESULT STDMETHODCALLTYPE GetTypeInfoCount(UINT *pctinfo);
    virtual HRESULT STDMETHODCALLTYPE GetTypeInfo(UINT iTInfo, LCID lcid,
        ITypeInfo **ppTInfo);
    virtual HRESULT STDMETHODCALLTYPE GetIDsOfNames(REFIID riid,
        LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId);
    virtual HRESULT STDMETHODCALLTYPE Invoke(DISPID dispIdMember, REFIID riid,
        LCID lcid, WORD wFlags, DISPPARAMS *Params, VARIANT *pVarResult,
        EXCEPINFO *pExcepInfo, UINT *puArgErr);
};

Примечательные реализации могут быть

HRESULT STDMETHODCALLTYPE JSObject::QueryInterface(REFIID riid, void **ppv)
{
    *ppv = NULL;

    if (riid == IID_IUnknown || riid == IID_IDispatch) {
        *ppv = static_cast<IDispatch*>(this);
    }

    if (*ppv != NULL) {
        AddRef();
        return S_OK;
    }

    return E_NOINTERFACE;
}

и

HRESULT STDMETHODCALLTYPE JSObject::Invoke(DISPID dispIdMember, REFIID riid,
    LCID lcid, WORD wFlags, DISPPARAMS *Params, VARIANT *pVarResult,
    EXCEPINFO *pExcepInfo, UINT *puArgErr)
{
    MessageBox(NULL, "Invoke", "JSObject", MB_OK);
    return DISP_E_MEMBERNOTFOUND;
}

К сожалению, я никогда не получаю окно сообщения "Invoke", когда пытаюсь использовать объект "JSObject" из кода javascript.

JSObject.randomFunctionName();  // This should give me the c++ "Invoke" message
                                // box, but it doesn't

РЕДАКТИРОВАТЬ 2:

Я реализовал GetIDsOfNames примерно так:

HRESULT STDMETHODCALLTYPE JSObject::GetIDsOfNames(REFIID riid,
    LPOLESTR *rgszNames, UINT cNames, LCID lcid, DISPID *rgDispId)
{
    HRESULT hr = S_OK;

    for (UINT i = 0; i < cNames; i++) {
        std::map<std::wstring, DISPID>::iterator iter = idMap.find(rgszNames[i]);
        if (iter != idMap.end()) {
            rgDispId[i] = iter->second;
        } else {
            rgDispId[i] = DISPID_UNKNOWN;
            hr = DISP_E_UNKNOWNNAME;
        }
    }

    return hr;
}

а это мой конструктор

JSObject::JSObject() : ref(0)
{
    idMap.insert(std::make_pair(L"execute", DISPID_USER_EXECUTE));
    idMap.insert(std::make_pair(L"writefile", DISPID_USER_WRITEFILE));
    idMap.insert(std::make_pair(L"readfile", DISPID_USER_READFILE));
}

с константами DISPID_USER_ *, определенными как закрытые члены класса

class JSObject : public IDispatch {
private:
    static const DISPID DISPID_USER_EXECUTE = DISPID_VALUE + 1;
    static const DISPID DISPID_USER_WRITEFILE = DISPID_VALUE + 2;
    static const DISPID DISPID_USER_READFILE = DISPID_VALUE + 3;

    // ...
};

РЕДАКТИРОВАТЬ 3, 4 и 5:

Перемещено в отдельный вопрос

РЕДАКТИРОВАТЬ 6:

Сделано отдельным вопросом из правок "Возвращение строки". Таким образом, я могу принять Георг ответ, так как он отвечает на исходный вопрос.

РЕДАКТИРОВАТЬ 7:

Я получил несколько запросов на полностью работающую автономную примерную реализацию. Вот оно: https://github.com/Tobbe/CppIEEmbed. Пожалуйста, раскошелиться и улучшить, если вы можете:)

1 Ответ

5 голосов
/ 20 сентября 2010

Вам нужно реализовать GetIDsOfNames(), чтобы сделать что-то разумное, так как эта функция будет вызываться клиентским кодом до Invoke().
Если у вас есть интерфейсы в библиотеке типов, см. здесь для примера. Если вы хотите использовать позднюю привязку, вы можете использовать DISPID s больше DISPID_VALUE и меньше 0x80010000 (все значения <= 0 и в диапазоне от 0x80010000 до 0x8001FFFF зарезервированы):

HRESULT GetIDsOfNames(REFIID riid, LPOLESTR *rgszNames, UINT cNames, 
                      LCID lcid, DISPID *rgDispId)
{
    HR hr = S_OK;
    for (UINT i=0; i<cNames; ++i) {
        if (validName(rgszNames)) {
            rgDispId[i] = dispIdForName(rgszNames);
        } else {
            rgDispId[i] = DISPID_UNKNOWN;
            hr = DISP_E_UNKNOWNNAME;
        }
    }
    return hr;
}

HRESULT Invoke(DISPID dispIdMember, REFIID riid, LCID lcid, WORD wFlags, 
               DISPPARAMS *Params, VARIANT *pVarResult, EXCEPINFO *pExcepInfo, 
               UINT *puArgErr)
{
    if (wFlags & DISPATCH_METHOD) {
       // handle according to DISPID ...
    }

    // ...

Обратите внимание, что DISPID не должны внезапно меняться, например, следует использовать статические map или постоянные значения.

...