【问题标题】:ATL COM - EnumWindows callback function not working with IDispatch::InvokeATL COM - EnumWindows 回调函数不适用于 IDispatch::Invoke
【发布时间】:2017-07-29 04:05:26
【问题描述】:

我正在尝试为 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

提前感谢您的帮助。

【问题讨论】:

    标签: c++ callback dispatch


    【解决方案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);
    

    任何好的建议都值得进一步感谢。

    【讨论】:

      猜你喜欢
      • 2017-08-17
      • 2012-03-09
      • 2011-01-07
      • 2014-08-21
      • 1970-01-01
      • 2013-10-15
      • 2021-03-23
      • 1970-01-01
      • 2011-06-22
      相关资源
      最近更新 更多