当我按下控制键时滚动鼠标滚轮时,为什么app无缘无故终止?

Mik*_*2ab 3 c winapi

当控制键关闭然后我滚动鼠标滚轮时,应用程序无缘无故终止.我在Windows XP上测试它.只有在滚动时按下控制键时才会发生这种情况.如果在滚动时未按下控制键,则不会发生.不知道它是如何与其他操作系统.使用下面的代码来测试它

#include <windows.h>
#include <tchar.h>


LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
    static HWND hEdit = 0;

    switch(msg)
    {
    case WM_CREATE:
        hEdit = CreateWindowEx(WS_EX_CLIENTEDGE, TEXT("Edit"), 0, WS_VISIBLE | WS_CHILD | WS_HSCROLL | WS_VSCROLL | ES_MULTILINE | ES_READONLY,
            0, 0, 0, 0, hwnd, 0, GetModuleHandle(0), 0);
        break;

    case WM_SIZE:
        MoveWindow(hEdit, 0, 0, LOWORD(lParam), HIWORD(lParam), TRUE);
        break;

    case WM_MOUSEWHEEL:
        SendMessage(hEdit, msg, wParam, lParam);
        break;

    case WM_DESTROY:
        PostQuitMessage(0);
        break;

    default:
        return DefWindowProc(hwnd, msg, wParam, lParam);
    }
    return 0;
}

int WINAPI _tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)
{
    WNDCLASSEX wc = {0};
    HWND hwnd;
    MSG msg;

    wc.cbSize = sizeof wc;
    wc.hbrBackground = 0;
    wc.hCursor = LoadCursor(0, IDC_ARROW);
    wc.hIcon = LoadIcon(0, IDI_APPLICATION);
    wc.hInstance = hInstance;
    wc.lpfnWndProc = WndProc;
    wc.lpszClassName = TEXT("MainClass");

    if(!RegisterClassEx(&wc)) 
        return 0;

    hwnd = CreateWindowEx(0, wc.lpszClassName, TEXT("Hello"), WS_OVERLAPPEDWINDOW, 40, 20, 400, 200,
        0, 0, hInstance, 0);

    if(!hwnd) 
        return 0;

    ShowWindow(hwnd, nCmdShow);

    while(GetMessage(&msg, 0, 0, 0) > 0)
    {
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }

    return (int)msg.wParam;
}
Run Code Online (Sandbox Code Playgroud)

如果我注释掉下面的3行,并且在滚动时按下控制键,则不会发生

case WM_MOUSEWHEEL:
    SendMessage(hEdit, msg, wParam, lParam);
    break;
Run Code Online (Sandbox Code Playgroud)

Jon*_*ter 6

扩展我在评论中所说的内容:

  1. 您的窗口类转发WM_MOUSEWHEEL到编辑控件
  2. 如果按下控制键,编辑控件将忽略该WM_MOUSEWHEEL消息并将其传递给DefWindowProc.
  3. DefWindowProc将消息传递给父链(此行为记录在WM_MOUSEWHEEL文档中).
  4. 您的窗口收到转发的消息并循环回到步骤#1

最终,您的堆栈耗尽,您的流程终止.

有三种方法可以解决这个问题:

第一个(也可能是最安全的)是使用一个标志来防止递归循环; 例如:

static bool fInForwardMsg; // if you have multiple windows you would want to make this a local variable

case WM_MOUSEWHEEL:
    if (!fInForwardMsg) {
        fInForwardMsg = true;
        SendMessage(hEdit, uMsg, wParam, lParam);
        fInForwardMsg = false;
    }
    break;
Run Code Online (Sandbox Code Playgroud)

第二种解决方案依赖于以下事实:编辑控件查看该wParam值以查看控制键是否已关闭(它还会偶然检查移位).这是内部未记录的行为并且可能会发生变化,因此您不应该依赖它,但您应该能够通过不转发原始值来防止此问题wParam.例如:

case WM_MOUSEWHEEL:
    SendMessage(hEdit, msg, wParam & ~0xffff, lParam);
    break;
Run Code Online (Sandbox Code Playgroud)

第三种解决方案也是最简单的; 因为如果按住shift或control,编辑控件实际上什么都不做,所以在这些情况下根本不转发消息:

case WM_MOUSEWHEEL:
    if (!(wParam & (MK_SHIFT | MK_CONTROL))
        SendMessage(hEdit, msg, wParam, lParam);
    break;
Run Code Online (Sandbox Code Playgroud)