gpt4 book ai didi

c - 为什么在按下控制键的同时滚动鼠标滚轮时应用程序无故终止?

转载 作者:太空狗 更新时间:2023-10-29 16:06:20 29 4
gpt4 key购买 nike

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

最佳答案

扩展我在评论中所说的话:

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

第二种解决方案依赖于这样一个事实,即编辑控件查看 wParam 值以查看控制键是否按下(顺便说一句,它也检查 shift)。这是内部未记录的行为,可能会发生变化,因此您不应依赖它,但您应该能够通过不转发 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/

29 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com