单击文本框时发送WM_Quit消息

时间:2014-10-23 15:43:55

标签: c++ c winapi

所以标题说明了一切。我想也许是因为有81个文本框它与图层有关,但坦率地说我不知道​​..刚开始学习windows api就像2天前一直在学习直接从msdn库中获取功能..我google了这个问题多次,没有运气所以我在这里。非常感谢帮助^。^

// 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 ..我编码它做一个消息框并给用户一个机会接受或不....所以当这种情况发生后点击一个文本框我点击否和另一个消息框apperas再次询问,我点击否,它消失但当我点击x按钮,我点击没有窗口仍然消失.....

2 个答案:

答案 0 :(得分:4)

问题可能是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碰巧有一个等于IDM_EXIT的ID。由于您在此控件上收到HIWORD(wParam)时未查看EN_SETFOCUS您要放弃的内容。

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

答案 1 :(得分:4)

case WM_COMMAND:
    switch (LOWORD(wParam))

这还不够好。编辑控件还发送WM_COMMAND消息以通知其父窗口有关正在进行的操作。和EN_UPDATE一样,无论何时键入一个字符。或者EN_SETFOCUS当他们获得焦点时,听起来就像你点击它们时出现问题一样。这些通知包含在WM_COMMAND消息中。

因此,您必须注意WM_COMMAND消息的来源。 LPARAM参数告诉你。如果IDM_EXIT来自菜单项,则必须验证LPARAM为0.检查MSDN library以获取详细信息。