WM_单击文本框时退出正在发送的消息

WM_Quit message being sent when i click on textbox

本文关键字:消息 退出 单击 文本 WM      更新时间:2023-10-16

所以title说明了一切。我在想,也许是因为有81个文本框,这与层有关,但坦率地说,我不知道。。两天前刚开始学习windowsapi,我一直在学习msdn函数库。。我在谷歌上搜索了很多次这个问题,但没有运气,所以我来了。非常感谢你的帮助

// Win32Project9.cpp : Defines the entry point for the application.
//
#include "stdafx.h"
#include "Win32Project9.h"
#include "Resource.h"
#include <Windows.h>
#include <vector>
#include <cstring>
using namespace std;
HWND Hwnd;
HMENU hMenu;
HWND boxes[81];
int x, y;
vector<LPWSTR> BoxNum;

LRESULT CALLBACK WindowProcedure(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
{
switch(msg)
{
case WM_COMMAND:
    switch (LOWORD(wParam))
    {
    case IDM_EXIT:
        PostQuitMessage(0);
        return 0;
    break;
    case ID_SOLVE:
        for (int i = 0; i < 81; i++)
        {
            GetWindowText(boxes[i], BoxNum[i], NULL);
        }
    break;
    }
break;
}
if (msg == WM_COMMAND)
{
    if (LOWORD(wParam) > 199 && LOWORD(wParam) < 281)
    {
        if (HIWORD(wParam) == EN_SETFOCUS | HIWORD(wParam) == EN_UPDATE)
        {
            return DefWindowProc(hwnd, msg, wParam, lParam);
        }
    }
}
else if (msg == WM_CLOSE)
{
    PostQuitMessage(0);
    return 0;
}
return DefWindowProc(hwnd, msg, wParam, lParam);
}
void DrawBoard()
{
x = 10;
y = 10;
int count = 0;
for (int i = 0; i < 81; i++)
{
    int BOX_ID = 200 + i;
    boxes[i] = CreateWindow(TEXT("Edit"), NULL, WS_CHILD | WS_BORDER | WS_VISIBLE, x, y, 20, 20, Hwnd, (HMENU)BOX_ID, NULL, NULL);
    x += 30;
    count++;
    if (count == 9)
    {
        y += 30;
        x = 10;
        count = 0;
    }
}
}
int WINAPI WinMain(HINSTANCE hInstance,
               HINSTANCE hPrevInstance,
               LPSTR lpCmdLine,
               int nCmdShow)
{   
//structure to hold window specs
WNDCLASSEX Wc;
//allocate memory for window class
ZeroMemory(&Wc, sizeof(WNDCLASSEX));
//fill in neccessary info
Wc.cbSize = sizeof(WNDCLASSEX);
Wc.hbrBackground = (HBRUSH)COLOR_WINDOW;
Wc.hCursor = LoadCursor(hInstance, IDC_ARROW);
Wc.hInstance = hInstance;
Wc.lpfnWndProc = WindowProcedure;
Wc.lpszClassName = L"MyClass";
Wc.style = CS_HREDRAW | CS_VREDRAW;
//register class
RegisterClassEx(&Wc);
//load menu into handle
hMenu = LoadMenu(hInstance, MAKEINTRESOURCE(ID_MENU));
//Create Window with class and create handle
Hwnd = CreateWindow(L"MyClass", L"Sudoku", WS_OVERLAPPEDWINDOW, 0, 0, 300, 340, NULL, hMenu, hInstance, NULL);
//DisplayWindow
ShowWindow(Hwnd, nCmdShow); 
DrawBoard();
//structure to hold input stream
MSG msg;
//listen for input
while(GetMessage(&msg, Hwnd, NULL, NULL))
{
    TranslateMessage(&msg);
    DispatchMessage(&msg);
}
return 0;
}

我还读到,当我按下x按钮时,wm_close会被处理。好吧,即使我点击文本框,也会收到这条消息。如果你看看我的案例WM_Close。。我把它编码成一个消息框,让用户有机会接受或不接受。。。。因此,当这种情况发生在点击一个文本框后,我点击"否",另一个消息框似乎再次询问,我单击"否"后,它消失了,但当我点击x按钮并点击"否》时,窗口仍然消失。。。。。

问题可能是WM_COMMAND处理不当。

收到的论据如下:

WORD code = HIWORD(wParam);
WORD id = LOWORD(lParam);

问题是code取决于您正在使用的控件类型。例如,如果是按钮,它将是一些BTN_*值,如果是编辑,它将为EN_*,依此类推。但这些值重叠严重,因此不能在单个开关中使用它们。

例如CBN_KILLFOCUS==4,还有LBN_SETFOCUS==4。。。此外,菜单项将在这里得到一个0和加速器一个1。顺便说一句,BN_CLICKED==0和它看起来没有其他通知消息使用0,所以你可以在菜单和按钮中使用相同的ID,它就会正常工作。还有加速器,只要小心一点。。。BN_PAINT==1,我认为这个已经不存在了,但你明白了。。。

不管怎样,解决你的问题。我的猜测是,您有一个EDIT,它的ID恰好等于IDM_EXIT。由于您没有检查HIWORD(wParam),因此当您收到此控件上的EN_SETFOCUS时,您将退出。

解决方案是:首先,始终检查wParam中的两个WORD。其次,避免菜单选项和ID控件之间的冲突,除了按钮。

case WM_COMMAND:
    switch (LOWORD(wParam))

这还不够好。编辑控件还会发送WM_COMMAND消息,通知其父窗口正在发生的事情。比如每当您键入字符时都要发送EN_UPDATE。或者EN_SETFOCUS,当它们获得焦点时,当你点击它们时看到它出错时,听起来就像是你的情况。这些通知被包装在WM_COMMAND消息中。

因此,您必须注意WM_COMMAND消息的来源。LPARAM参数告诉您。如果IDM_EXIT来自菜单项,则必须验证LPARAM是否为0。有关详细信息,请查看MSDN库。