当控制键关闭然后我滚动鼠标滚轮时,应用程序无缘无故终止.我在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)
扩展我在评论中所说的内容:
WM_MOUSEWHEEL到编辑控件WM_MOUSEWHEEL消息并将其传递给DefWindowProc.DefWindowProc将消息传递给父链(此行为记录在WM_MOUSEWHEEL文档中).最终,您的堆栈耗尽,您的流程终止.
有三种方法可以解决这个问题:
第一个(也可能是最安全的)是使用一个标志来防止递归循环; 例如:
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)