Как заставить окно статистики c увеличивать размер при наведении курсора мыши - PullRequest
0 голосов
/ 02 августа 2020

У меня куча windows внутри другого окна. Дочерние элементы Windows создаются в методе CreateBoard() класса MainWindow. Create() метод MainWindow и GemWindow взят из абстрактного класса MS Documentation BaseWindow, и именно здесь каждый класс регистрируется с вызовом RegisterClass()

template <class DERIVED_TYPE>
class BaseWindow
{
public:
    static LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
    {
        DERIVED_TYPE* pThis = NULL;

        if (uMsg == WM_NCCREATE)
        {
            CREATESTRUCT* pCreate = (CREATESTRUCT*)lParam;
            pThis = (DERIVED_TYPE*)pCreate->lpCreateParams;
            SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)pThis);

            pThis->m_hwnd = hwnd;
        }
        else
        {
            pThis = (DERIVED_TYPE*)GetWindowLongPtr(hwnd, GWLP_USERDATA);
        }
        if (pThis)
        {
            //getting read access violation here why?
            //because on resize you are not initializing tiles
            //but resizing the thing
            return pThis->HandleMessage(uMsg, wParam, lParam);
        }
        else
        {
            return DefWindowProc(hwnd, uMsg, wParam, lParam);
        }
    }

    BaseWindow() : m_hwnd(NULL) { }

    BOOL Create(
        PCWSTR lpWindowName,
        DWORD dwStyle,
        DWORD dwExStyle = 0,
        int x = CW_USEDEFAULT,
        int y = CW_USEDEFAULT,
        int nWidth = CW_USEDEFAULT,
        int nHeight = CW_USEDEFAULT,
        HWND hWndParent = 0,
        HMENU hMenu = 0
    )
    {
        WNDCLASS wc = { 0 };

        wc.lpfnWndProc = DERIVED_TYPE::WindowProc;
        wc.hInstance = GetModuleHandle(NULL);
        wc.lpszClassName = ClassName();

        RegisterClass(&wc);

        m_hwnd = CreateWindowEx(
            dwExStyle, ClassName(), lpWindowName, dwStyle, x, y,
            nWidth, nHeight, hWndParent, hMenu, GetModuleHandle(NULL), this
        );

        return (m_hwnd ? TRUE : FALSE);
    }

    //returns handle
    HWND Window() const { return m_hwnd; }

protected:

    virtual PCWSTR  ClassName() const = 0;
    virtual LRESULT HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam) = 0;

    HWND m_hwnd;
};
Метод

CreateBoard() создает драгоценные камни на доске.

BOOL MainWindow::CreateBoard()
{
    for (unsigned int i = 0; i < GetcGem(); i++)
    {
        for (unsigned int j = 0; j < GetcGem(); j++)
        {
            if (Gems[i][j].Create(L"gem", WS_CHILDWINDOW | WS_VISIBLE, NULL, 
              5 + i * GetsGem().cx + i * 10, 
              5 + j * GetsGem().cy + j * 10, GetsGem().cx, GetsGem().cy, Window(), NULL))
            {
                //pass from main window to Gem object
                Gems[i][j].SetSize(GetsGem().cx, GetsGem().cy);
                Gems[i][j].SetPosition(5 + i * GetsGem().cx + i * 10, 5 + j * GetsGem().cy + j * 10);

                HBRUSH hbrush = CreateSolidBrush(RGB(125,125,125));
                HBRUSH hOldBrush = (HBRUSH)SetClassLongPtr(Gems[i][j].Window(), GCLP_HBRBACKGROUND, (LONG_PTR)hbrush);
                DeleteObject(hOldBrush);
                InvalidateRect(Gems[i][j].Window(), NULL, TRUE);

            }
            else
            {
                return FALSE;
            }
        }
    }
    return TRUE;
}

Я хочу, чтобы каждое дочернее окно увеличивало свой размер на 4 пикселя при наведении курсора мыши отдельно. Однако после увеличения размера они оставляют след, а иногда при наведении курсора на одно окно увеличивается размер всей строки или столбца. Ссылка на изображение, например:

Проблемная демонстрация

У меня есть код обработки для каждого дочернего окна, как показано ниже. tracking - это static BOOL в классе GemWindow

BOOL GemWindow::tracking = false;

LRESULT GemWindow::HandleMessage(UINT uMsg, WPARAM wParam, LPARAM lParam)
{
    switch (uMsg)
    {
    case WM_DESTROY:
        DestroyWindow(Window());
        break;

    case WM_MOUSEMOVE:
    {
        if (!tracking) {
            TrackMouse();
            tracking = true;
        }

    }break;

    case WM_MOUSEHOVER:
    {
        MoveWindow(Window(), GetSize().cx + 4, GetSize().cy + 4, GetPosition().x - 2, GetPosition().y - 2, TRUE);
        InvalidateRect(Window(), NULL, TRUE);
        OutputDebugString(L"MOUSE ENTERED\n");
    }break;

    case WM_MOUSELEAVE:
    {
        MoveWindow(Window(), GetSize().cx, GetSize().cy, GetPosition().x, GetPosition().y, TRUE);
        InvalidateRect(Window(), NULL, TRUE);
        OutputDebugString(L"MOUSE LEFT\n");
        tracking = false;
    }break;


    case WM_PAINT:
    {
        PAINTSTRUCT ps;
        HDC hdc = BeginPaint(m_hwnd, &ps);
        //FillRect(hdc, &ps.rcPaint, CreateSolidBrush(color));
        EndPaint(m_hwnd, &ps);
        //OutputDebugString(L"PAINT\n");
    }
    return 0;

    default:
        return DefWindowProc(m_hwnd, uMsg, wParam, lParam);
    }
    return TRUE;
}

TrackMouse() - это метод GemWindow, который инициализирует TRACKMOUSEEVENT и вызывает TrackMouseEvent()

    //method for tracking mouse
    void TrackMouse(/*reusability enhanced if you add arguments in the future*/)
    {
        TRACKMOUSEEVENT tme;
        tme.cbSize = sizeof(TRACKMOUSEEVENT);
        tme.dwFlags = TME_HOVER | TME_LEAVE; //Type of events to track & trigger.
        tme.dwHoverTime = 1; //How long the mouse has to be in the window to trigger a hover event.
        tme.hwndTrack = Window();
        TrackMouseEvent(&tme);
    }

Я думал, это потому, что я не обновляю регионы, но это ничего не меняло. Я смущен этой ошибкой и не могу придумать для нее какой-либо веской причины.

Спасибо всем, кто нашел время, чтобы помочь.

1 Ответ

1 голос
/ 03 августа 2020
BOOL MoveWindow(
  HWND hWnd,
  int  X,  //The new position of the left side of the window.
  int  Y,  //The new position of the top of the window.
  int  nWidth, //The new width of the window.
  int  nHeight, //The new height of the window.
  BOOL bRepaint
);

Измените следующий код:

   case WM_MOUSEHOVER:
    {
//      MoveWindow(Window(), GetSize().cx + 4, GetSize().cy + 4, GetPosition().x - 2, GetPosition().y - 2, TRUE);
        MoveWindow(Window(), GetPosition().x - 2, GetPosition().y - 2, GetSize().cx + 4, GetSize().cy + 4, TRUE);            
        ...

    case WM_MOUSELEAVE:
    {
 //     MoveWindow(Window(), GetSize().cx, GetSize().cy, GetPosition().x, GetPosition().y, TRUE);
        MoveWindow(Window(), GetPosition().x, GetPosition().y, GetSize().cx, GetSize().cy,  TRUE);
        ...

Отладка:

введите описание изображения здесь

...