当按下控制键然后我滚动鼠标滚轮时,应用程序无缘无故终止。我正在 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;
}

如果我注释掉下面的 3 行,并且在滚动时按下控制键,则不会发生
case WM_MOUSEWHEEL:
    SendMessage(hEdit, msg, wParam, lParam);
    break;

最佳答案

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

  • 您的窗口类将 WM_MOUSEWHEEL 转发到编辑控件
  • 如果按下控制键,编辑控件将忽略 WM_MOUSEWHEEL 消息并将其传递给 DefWindowProc
  • DefWindowProc 将消息向上传递到父链(此行为记录在 WM_MOUSEWHEEL documentation 中)。
  • 您的窗口接收转发的消息并循环回到步骤 #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;
    

    第二种解决方案依赖于这样一个事实,即编辑控件查看 wParam 值以查看控制键是否按下(顺便说一下,它还检查移位)。这是内部未记录的行为并且可能会发生变化,因此您不应依赖它,但您应该能够通过不转发 wParam 的原始值来防止出现问题。例如。:
    case WM_MOUSEWHEEL:
        SendMessage(hEdit, msg, wParam & ~0xffff, lParam);
        break;
    

    第三种方案也是最简单的;由于如果按住 shift 或 control 编辑控件实际上不会执行任何操作,因此在这些情况下根本不要转发消息:
    case WM_MOUSEWHEEL:
        if (!(wParam & (MK_SHIFT | MK_CONTROL))
            SendMessage(hEdit, msg, wParam, lParam);
        break;
    

    关于c - 为什么在按下控制键的同时滚动鼠标滚轮时应用程序会无故终止?,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/32691773/

    10-11 22:50