我想编写一个程序来控制使用WINAPI移动的方块

时间:2013-10-22 15:44:06

标签: winapi

这是到目前为止尝试的代码:

#include <windows.h>

LRESULT CALLBACK WndProc (HWND, UINT, WPARAM, LPARAM) ;

int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance,
                    PSTR szCmdLine, int iCmdShow)
{
    static TCHAR szAppname[] = TEXT ("win0") ;
    WNDCLASS wndclass ;
    MSG msg ;
    HWND hwnd ;

    wndclass.style = CS_HREDRAW | CS_VREDRAW ;
    wndclass.cbWndExtra = 0 ;
    wndclass.cbClsExtra = 0 ;
    wndclass.lpfnWndProc = WndProc ;
    wndclass.lpszClassName = szAppname ;
    wndclass.lpszMenuName = NULL ;
    wndclass.hbrBackground = (HBRUSH) GetStockObject (WHITE_BRUSH) ;
    wndclass.hCursor = LoadCursor (NULL, IDC_ARROW) ;
    wndclass.hIcon = LoadIcon (NULL, IDI_APPLICATION) ;
    wndclass.hInstance = hInstance ;

    if (!RegisterClass (&wndclass))
    {
        MessageBox (NULL, TEXT ("Register fail"), szAppname, 0) ;
        return 0 ;
    }
    hwnd = CreateWindow (   szAppname,
                            TEXT ("mywin"),
                            WS_OVERLAPPEDWINDOW,
                            CW_USEDEFAULT,
                            CW_USEDEFAULT,
                            CW_USEDEFAULT,
                            CW_USEDEFAULT,
                            NULL,
                            NULL,
                            hInstance,
                            NULL) ;

    ShowWindow (hwnd, iCmdShow) ;
    UpdateWindow (hwnd) ;

    while (GetMessage (&msg, NULL, 0, 0))
    {
        TranslateMessage (&msg) ;
        DispatchMessage (&msg) ;
    }

    return msg.wParam ;
}

LRESULT CALLBACK WndProc (HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
    static int cxClient, cyClient, Left, Top, Right, Down ;
    PAINTSTRUCT ps ;
    HDC hdc ;
    RECT rect ;

    Right = 20 ; Down = 20 ;

    switch (message)
    {
    case WM_SIZE :
        cxClient = LOWORD (lParam) ;
        cyClient = HIWORD (lParam) ;
        return 0 ;
    case WM_PAINT :
        hdc = BeginPaint (hwnd, &ps) ;

        SetRect (&rect, Left, Top, Right, Down) ;

        FillRect (hdc, &rect, CreateSolidBrush (RGB (100, 100, 100))) ;

        EndPaint (hwnd, &ps) ;
        return 0 ;
    case WM_KEYDOWN :
        InvalidateRect (hwnd, &rect, TRUE) ;

        switch (wParam)
        {
        case VK_UP :
            if (Top - 20 < 0)
            {
                Top = 0 ;
                Down = 20 ;
            }
            else
            {
                Top -= 20 ;
                Down -= 20 ;
            }
            SendMessage (hwnd, WM_PAINT, wParam, lParam) ;
            break ;
        case VK_DOWN :
            if (Down + 20 > cyClient)
            {
                Down = cyClient ;
                Top = Down - 20 ;
            }
            else
            {
                Down += 20 ;
                Top += 20 ;
            }SendMessage (hwnd, WM_PAINT, wParam, lParam) ;
            break ;
        case VK_LEFT :
            if (Left - 20 < 0)
            {
                Left = 0 ;
                Right = 20 ;
            }
            else
            {
                Left -= 20 ;
                Right -= 20 ;
            }SendMessage (hwnd, WM_PAINT, wParam, lParam) ;
            break ;
        case VK_RIGHT :
            if (Right + 20 > cxClient)
            {
                Right = cxClient ;
                Left = Right - 20 ;
            }
            else
            {
                Right += 20 ;
                Left += 20 ;
            }SendMessage (hwnd, WM_PAINT, wParam, lParam) ;
            break ;
        default :
            break ;
        }
        return 0 ;
    case WM_DESTROY :
        PostQuitMessage (0) ;
        return 0 ;

    }
    return DefWindowProc (hwnd, message, wParam, lParam);
}

我认为我没有很好地处理这个消息,所以我无法控制它。

1 个答案:

答案 0 :(得分:0)

除了你应该在Window Proc中使用一些静态变量以保持程序的状态(你应该使用HWND作为键→值映射的键来定位正确的数据结构),以实际制作窗口重绘你必须“使它”无效,即告诉Windows需要重新绘制内容。发送WM_PAINT消息不会执行此操作。 InvalidateRect会这样做,因此您应该在更改变量而不是发送消息后调用该函数

相反,您应该在设置变量后立即调用InvalidateRect