0

我正在尝试使用 Windows C API 制作自定义 Windows 8 样式按钮。但是,当我重绘按钮后尝试在按钮上绘制文本时,什么都没有发生!我什至尝试创建一个静态窗口作为按钮的子窗口,并在我重绘按钮时向它发送 WS_ERASEBKGND 消息,但是一旦我第一次重绘按钮,它就会消失!这是我的自定义按钮的窗口过程:

LRESULT CALLBACK ButtonWindowProc(HWND hwnd,UINT msg,WPARAM wParam,LPARAM lParam)
{
    //static HWND static_text_handle;
    HDC hdc;
    PAINTSTRUCT ps;
    DWORD color;
    HFONT font,holdFont;
    static RECT rect;
    wchar_t test[] = L"test";
    TRACKMOUSEEVENT tme = {sizeof(TRACKMOUSEEVENT),TME_LEAVE,hwnd,HOVER_DEFAULT};
    /*static*/ HBRUSH brush = CreateSolidBrush(RGB(20,30,40));
    /*static*/ HBRUSH clicked_brush = CreateSolidBrush(RGB(40,50,60));
    /*static*/ HBRUSH hover_brush = CreateSolidBrush(RGB(70,80,90));
    //TrackMouseEvent(&tme);
    switch(msg)
    {
    case WM_CREATE:
        //static_text_handle = CreateWindowW(L"Static",L"test",WS_CHILD | WS_VISIBLE,0,0,20,20,hwnd,NULL,NULL,NULL);
        //brush = ((struct custom_button_colors*)(lParam))->default_color;
        //clicked_brush = ((struct custom_button_colors*)(lParam))->push_color;
        //hover_brush = ((struct custom_button_colors*)(lParam))->hover_color;
        break;
    case WM_LBUTTONDOWN:
            SendMessage(hwnd,WM_ERASEBKGND,(WPARAM)GetDC(hwnd),(LPARAM)LPARAM_L_DOWN);
            break;
    case WM_LBUTTONUP:
        SendMessage(hwnd,WM_ERASEBKGND,(WPARAM)GetDC(hwnd),(LPARAM)LPARAM_L_UP);
        break;
    case WM_MOUSEMOVE:
        //Beep(1000,1000);
        TrackMouseEvent(&tme);
        if (GetAsyncKeyState(VK_LBUTTON))
        {
            SendMessage(hwnd,WM_ERASEBKGND,(WPARAM)GetDC(hwnd),(LPARAM)LPARAM_L_DOWN);
        }
        else
        {
            SendMessage(hwnd,WM_ERASEBKGND,(WPARAM)GetDC(hwnd),(LPARAM)LPARAM_MOUSEHOVER);
        }
        break;
    case WM_MOUSELEAVE:
        SendMessage(hwnd,WM_ERASEBKGND,(WPARAM)GetDC(hwnd),(LPARAM)-1);
        break;
    case WM_ERASEBKGND:
        GetClientRect(hwnd,&rect);
        if (lParam == LPARAM_L_DOWN)
        {
            FillRect((HDC)wParam,&rect,clicked_brush);
        }
        else if (lParam == LPARAM_MOUSEHOVER)
        {
            FillRect((HDC)wParam,&rect,hover_brush);
            break;
        }
        else
        {
            POINT mousepos;
            GetCursorPos(&mousepos);
            if (WindowFromPoint(mousepos) == hwnd)
            {
                FillRect((HDC)wParam,&rect,hover_brush);
            }
            else
            {
                FillRect((HDC)wParam,&rect,brush);
            }
        }
        hdc = BeginPaint(hwnd,&ps);
        color = GetSysColor(COLOR_BTNFACE);
        SetBkColor(hdc,color);
        font = CreateFontW(25, 0, 0, 0, FW_MEDIUM, 0, 0, 0, 0,0, 0, ANTIALIASED_QUALITY, 0, L"Tahoma");
        holdFont = SelectObject(hdc, font);
        TextOutW(hdc,200,200,test,lstrlenW(test));
        SelectObject(hdc,holdFont);
        DeleteObject(font);
        EndPaint(hwnd,&ps);
        break;
    default:
        return DefWindowProc(hwnd,msg,wParam,lParam);
    }
    return 0;
}

我是 Windows 编程的绝对初学者,如果我的问题微不足道,我很抱歉。

4

1 回答 1

1

1) 不要在 WM_ERASEBKGND 中使用 BeginPaint/EndPaint。在 WM_PAINT 处理程序中移动该代码。

2) 使用 SetWindowLong 和 GWL_USERDATA 索引存储按钮状态(UP 或 DOWN)并使用 GetWindowLong 检索当前状态。

3) 使用 SetCapture 捕获鼠标,从 WM_LBUTTONDOWN 开始。根据鼠标位置更新按钮的状态(UP 或 DOWN)

4) 当你需要绘画时使用 RedrawWindow

于 2013-06-27T16:47:09.047 回答