ATL COM - EnumWindows回调函数不能与IDispatch :: Invoke

时间:2017-07-29 04:05:27

标签: c++ callback dispatch

我正在尝试将EnumWindows API函数的包装函数实现为包含在我的包装器dll中,因此它可以与许多脚本语言一起使用。

首先,我必须使用VBScript进行测试。在我的有用研究的帮助下,我为EnumWindows实现了一个包装函数和一个回调函数,但它没有按照我的意愿工作。

以下是我目前的代码:

EnumWindowsEnumWindowsProc回调的包装函数:

BOOL CALLBACK EnumWindowsProc(__in HWND hWnd, __in LPARAM lParam) {

    LPENUMWINDOWSPARAMS pewParams;
    VARIANT vhWnd, vResult; HRESULT HR = S_OK;

    VariantInit(&vhWnd);
    VariantInit(&vResult);

    vhWnd.vt = VT_I4;
    vhWnd.lVal = (LONG)(LONG_PTR)hWnd;

    pewParams = reinterpret_cast<LPENUMWINDOWSPARAMS>(lParam);
    // ^ PASSING MY PARAMETERS THROUGH LPARAM

    CComVariant varArgs[2] = { &vResult, &vhWnd };
    DISPPARAMS Parameters = { &varArgs[0], NULL, 2, 0 };

    pewParams->DISPATCH->Invoke(0, IID_NULL, LOCALE_USER_DEFAULT, DISPATCH_METHOD, &Parameters, &vResult, NULL, NULL);
    // ^ I SUSPECT SOMETHING IS WRONG WITH THIS INVOKE METHOD. BUT IT RETURNS S_OK.

    if (vResult.vt != VT_BOOL) { HR = DISP_E_TYPEMISMATCH; }

    pewParams->CallbackResult.vt = VT_ERROR;
    pewParams->CallbackResult.scode = HR;

    if (HR == S_OK)
        return (vResult.boolVal == VARIANT_TRUE ? TRUE : FALSE);
    else
        return FALSE;
}

STDMETHODIMP CWinAPI::WinAPI_EnumWindows(VARIANT EnumFunc, int lParam, int *Result) {

    ENUMWINDOWSPARAMS ewParams; HRESULT HR = S_OK;

    switch (EnumFunc.vt)
    {
        case VT_DISPATCH:
            ewParams.DISPATCH = EnumFunc.pdispVal;
            break;

        case VT_VARIANT | VT_BYREF:
            if (EnumFunc.pvarVal->vt == VT_DISPATCH) { ewParams.DISPATCH = EnumFunc.pvarVal->pdispVal; }
            break;

        default: return DISP_E_TYPEMISMATCH;
    }

    ewParams.lParam = reinterpret_cast<LPVARIANT>(&lParam);

    *Result = (int)EnumWindows(EnumWindowsProc, reinterpret_cast<LPARAM>(&ewParams));

    HR = ewParams.CallbackResult.scode;

    return HR;
}

ENUMWINDOWSPARAMS结构:

typedef struct tagENUMWINDOWSPARAMS {
    LPDISPATCH DISPATCH;
    LPVARIANT lParam;
    VARIANT CallbackResult;
} ENUMWINDOWSPARAMS, *PENUMWINDOWSPARAMS, *LPENUMWINDOWSPARAMS;

以下是我目前使用的测试VBScript:

Dim WINAPI: Set WINAPI = WScript.CreateObject("WinAPIWrapperLib.WINAPI")

Function EnumWindowsProc(HWND, lParam)
    WScript.Echo "Handle to the window: 0x" + CStr(UCase(Hex(HWND)))
    EnumWindowsProc = True
End Function

Dim Result: Result = WINAPI.WinAPI_EnumWindows(GetRef("EnumWindowsProc"), 0)
WScript.Echo "EnumWindows returned " + CStr(Result)

有一件事有效,EnumWindows正在返回True。 但是,相反,我所期待的,我没有得到窗口回应的句柄。 当我致电GetLastError时,它会返回ERROR_SUCCESS

提前感谢您的帮助。

1 个答案:

答案 0 :(得分:1)

我解决了这个问题,没有做任何特别的事情,只是一个小小的改变:

varArgs的类型从CComVariant更改为VARIANT,如下所示:

VARIANT varArgs[2] = { vResult, vhWnd };

更改了DISPPARAMS,如下所示:

DISPPARAMS Parameters = {};
Parameters.cArgs = 2;
Parameters.rgvarg = varArgs;

Invoke方法仍然相同:

pewParams->DISPATCH->Invoke(0, IID_NULL, LOCALE_USER_DEFAULT, DISPATCH_METHOD, &Parameters, &vResult, NULL, NULL);

进一步赞赏任何好的建议。