/*
 * PROJECT:     ReactOS api tests
 * LICENSE:     LGPL-2.0-or-later (https://spdx.org/licenses/LGPL-2.0-or-later)
 * PURPOSE:     Test for SHAppBarMessage
 * COPYRIGHT:   Copyright 2020 Katayama Hirofumi MZ (katayama.hirofumi.mz@gmail.com)
 */

#include "shelltest.h"
#include <windowsx.h>
#include <shlwapi.h>
#include <stdio.h>

/* Based on https://github.com/katahiromz/AppBarSample */

//#define VERBOSE

#define IDT_AUTOHIDE 1
#define IDT_AUTOUNHIDE 2

#define ID_ACTION 100

#define APPBAR_CALLBACK (WM_USER + 100)

#define LEFT_DOWN() mouse_event(MOUSEEVENTF_LEFTDOWN, 0, 0, 0, 0);
#define LEFT_UP() mouse_event(MOUSEEVENTF_LEFTUP, 0, 0, 0, 0);
#define MOVE(x, y) SetCursorPos((x), (y))

static const TCHAR s_szName[] = TEXT("AppBarSample");
static RECT s_rcWorkArea;
static HWND s_hwnd1 = NULL;
static HWND s_hwnd2 = NULL;

#ifdef VERBOSE
static LPCSTR MessageOfAppBar(DWORD dwMessage)
{
    static char s_buf[32];
    switch (dwMessage)
    {
    case ABM_NEW: return "ABM_NEW";
    case ABM_REMOVE: return "ABM_REMOVE";
    case ABM_QUERYPOS: return "ABM_QUERYPOS";
    case ABM_SETPOS: return "ABM_SETPOS";
    case ABM_GETSTATE: return "ABM_GETSTATE";
    case ABM_GETTASKBARPOS: return "ABM_GETTASKBARPOS";
    case ABM_ACTIVATE: return "ABM_ACTIVATE";
    case ABM_GETAUTOHIDEBAR: return "ABM_GETAUTOHIDEBAR";
    case ABM_SETAUTOHIDEBAR: return "ABM_SETAUTOHIDEBAR";
    case ABM_WINDOWPOSCHANGED: return "ABM_WINDOWPOSCHANGED";
    }
    wsprintfA(s_buf, "%lu", dwMessage);
    return s_buf;
}

static UINT WINAPI
SHAppBarMessageWrap(DWORD dwMessage, PAPPBARDATA pData)
{
    trace("SHAppBarMessage entered (dwMessage=%s, rc=(%ld, %ld, %ld, %ld))\n",
          MessageOfAppBar(dwMessage),
          pData->rc.left, pData->rc.top, pData->rc.right, pData->rc.bottom);
    UINT ret = SHAppBarMessage(dwMessage, pData);
    trace("SHAppBarMessage leaved (dwMessage=%s, rc=(%ld, %ld, %ld, %ld))\n",
          MessageOfAppBar(dwMessage),
          pData->rc.left, pData->rc.top, pData->rc.right, pData->rc.bottom);
    return ret;
}
#define SHAppBarMessage SHAppBarMessageWrap

#undef ARRAYSIZE
#define ARRAYSIZE _countof

void appbar_tprintf(const TCHAR *fmt, ...)
{
    TCHAR szText[512];
    va_list va;
    va_start(va, fmt);
    wvsprintf(szText, fmt, va);
#ifdef UNICODE
    printf("%ls", szText);
#else
    printf("%s", szText);
#endif
    va_end(va);
}

#define MSGDUMP_TPRINTF appbar_tprintf
#include "msgdump.h"

#endif  // def VERBOSE

void SlideWindow(HWND hwnd, LPRECT prc)
{
#define SLIDE_HIDE 400
#define SLIDE_SHOW 150
    RECT rcOld, rcNew = *prc;
    GetWindowRect(hwnd, &rcOld);

    BOOL fShow = (rcNew.bottom - rcNew.top > rcOld.bottom - rcOld.top) ||
                 (rcNew.right - rcNew.left > rcOld.right - rcOld.left);

    INT dx = (rcNew.right - rcOld.right) + (rcNew.left - rcOld.left);
    INT dy = (rcNew.bottom - rcOld.bottom) + (rcNew.top - rcOld.top);

    LONG dt = SLIDE_HIDE;
    if (fShow)
    {
        dt = SLIDE_SHOW;
        rcOld = rcNew;
        OffsetRect(&rcOld, -dx, -dy);
        SetWindowPos(hwnd, NULL, rcOld.left, rcOld.top,
                     rcOld.right - rcOld.left, rcOld.bottom - rcOld.top,
                     SWP_NOZORDER | SWP_NOACTIVATE | SWP_DRAWFRAME);
    }

    HANDLE hThread = GetCurrentThread();
    INT priority = GetThreadPriority(hThread);
    SetThreadPriority(hThread, THREAD_PRIORITY_HIGHEST);

    LONG t, t0 = GetTickCount();
    while ((t = GetTickCount()) < t0 + dt)
    {
        INT x = rcOld.left + dx * (t - t0) / dt;
        INT y = rcOld.top + dy * (t - t0) / dt;
        SetWindowPos(hwnd, NULL, x, y, 0, 0, SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE);

        UpdateWindow(hwnd);
        UpdateWindow(GetDesktopWindow());
    }

    SetThreadPriority(hThread, priority);
    SetWindowPos(hwnd, NULL, rcNew.left, rcNew.top,
                 rcNew.right - rcNew.left, rcNew.bottom - rcNew.top,
                 SWP_NOZORDER | SWP_NOACTIVATE | SWP_DRAWFRAME);
#undef SLIDE_HIDE
#undef SLIDE_SHOW
}

class Window
{
public:
    Window(INT cx, INT cy, BOOL fAutoHide = FALSE)
        : m_hwnd(NULL)
        , m_fAutoHide(fAutoHide)
        , m_cxWidth(cx)
        , m_cyHeight(cy)
    {
    }

    virtual ~Window()
    {
    }

    static BOOL DoRegisterClass(HINSTANCE hInstance)
    {
        WNDCLASS wc;
        ZeroMemory(&wc, sizeof(wc));
        wc.lpfnWndProc = Window::WindowProc;
        wc.hInstance = hInstance;
        wc.hIcon = LoadIcon(NULL, IDI_APPLICATION);
        wc.hCursor = LoadCursor(NULL, IDC_ARROW);
        wc.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
        wc.lpszClassName = s_szName;
        return !!RegisterClass(&wc);
    }

    static HWND DoCreateMainWnd(HINSTANCE hInstance, LPCTSTR pszText, INT cx, INT cy,
                                DWORD style = WS_POPUP | WS_THICKFRAME | WS_CLIPCHILDREN,
                                DWORD exstyle = WS_EX_WINDOWEDGE | WS_EX_TOOLWINDOW | WS_EX_TOPMOST,
                                BOOL fAutoHide = FALSE)
    {
        Window *this_ = new Window(cx, cy, fAutoHide);
        HWND hwnd = CreateWindowEx(exstyle, s_szName, pszText, style,
                                   CW_USEDEFAULT, CW_USEDEFAULT, 50, 50,
                                   NULL, NULL, hInstance, this_);
        ShowWindow(hwnd, SW_SHOWNORMAL);
        UpdateWindow(hwnd);
        return hwnd;
    }

    static INT DoMainLoop()
    {
        MSG msg;
        while (GetMessage(&msg, NULL, 0, 0))
        {
            TranslateMessage(&msg);
            DispatchMessage(&msg);
        }
        return (INT)msg.wParam;
    }

    static Window *GetAppbarData(HWND hwnd)
    {
        return (Window *)GetWindowLongPtr(hwnd, GWLP_USERDATA);
    }

    virtual LRESULT CALLBACK
    WindowProcDx(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
    {
#ifdef VERBOSE
        MD_msgdump(hwnd, uMsg, wParam, lParam);
#endif
        switch (uMsg)
        {
        HANDLE_MSG(hwnd, WM_CREATE, OnCreate);
        HANDLE_MSG(hwnd, WM_COMMAND, OnCommand);
        HANDLE_MSG(hwnd, WM_ACTIVATE, OnActivate);
        HANDLE_MSG(hwnd, WM_WINDOWPOSCHANGED, OnWindowPosChanged);
        HANDLE_MSG(hwnd, WM_SIZE, OnSize);
        HANDLE_MSG(hwnd, WM_MOVE, OnMove);
        HANDLE_MSG(hwnd, WM_NCDESTROY, OnNCDestroy);
        HANDLE_MSG(hwnd, WM_TIMER, OnTimer);
        HANDLE_MSG(hwnd, WM_NCHITTEST, OnNCHitTest);
        HANDLE_MSG(hwnd, WM_LBUTTONDOWN, OnLButtonDown);
        HANDLE_MSG(hwnd, WM_MOUSEMOVE, OnMouseMove);
        HANDLE_MSG(hwnd, WM_LBUTTONUP, OnLButtonUp);
        HANDLE_MSG(hwnd, WM_RBUTTONDOWN, OnRButtonDown);
        HANDLE_MSG(hwnd, WM_KEYDOWN, OnKey);
        HANDLE_MSG(hwnd, WM_PAINT, OnPaint);

        case APPBAR_CALLBACK:
            OnAppBarCallback(hwnd, uMsg, wParam, lParam);
            break;

        default:
            return DefWindowProc(hwnd, uMsg, wParam, lParam);
        }
        return 0;
    }

    static LRESULT CALLBACK
    WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
    {
        Window *this_ = GetAppbarData(hwnd);
        if (uMsg == WM_CREATE)
        {
            LPCREATESTRUCT pCS = (LPCREATESTRUCT)lParam;
            this_ = (Window *)pCS->lpCreateParams;
            SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)this_);
        }
        if (this_)
            return this_->WindowProcDx(hwnd, uMsg, wParam, lParam);
        return DefWindowProc(hwnd, uMsg, wParam, lParam);
    }

protected:
    HWND m_hwnd;
    BOOL m_fAutoHide;
    BOOL m_fOnTop;
    BOOL m_fHiding;
    UINT m_uSide;
    LONG m_cxWidth;
    LONG m_cyHeight;
    LONG m_cxSave;
    LONG m_cySave;
    BOOL m_fAppBarRegd;
    BOOL m_fMoving;
    BOOL m_bDragged;
    POINT m_ptDragOn;
    RECT m_rcAppBar;
    RECT m_rcDrag;

    void OnCommand(HWND hwnd, int id, HWND hwndCtl, UINT codeNotify)
    {
        HANDLE hThread;
        switch (id)
        {
        case ID_ACTION:
            PostMessage(s_hwnd2, WM_COMMAND, ID_ACTION + 1, 0);
            break;
        case ID_ACTION + 1:
            hThread = CreateThread(NULL, 0, ActionThreadFunc, this, 0, NULL);
            if (!hThread)
            {
                skip("failed to create thread\n");
                PostMessage(s_hwnd1, WM_CLOSE, 0, 0);
                PostMessage(s_hwnd2, WM_CLOSE, 0, 0);
                return;
            }
            CloseHandle(hThread);
        }
    }

    void OnPaint(HWND hwnd)
    {
        PAINTSTRUCT ps;

        TCHAR szText[64];
        GetWindowText(hwnd, szText, 64);

        RECT rc;
        GetClientRect(hwnd, &rc);

        if (HDC hdc = BeginPaint(hwnd, &ps))
        {
            DrawText(hdc, szText, -1, &rc, DT_SINGLELINE | DT_CENTER | DT_VCENTER);
            EndPaint(hwnd, &ps);
        }
    }

    void OnRButtonDown(HWND hwnd, BOOL fDoubleClick, int x, int y, UINT keyFlags)
    {
        m_fAutoHide = !m_fAutoHide;
        AppBar_SetAutoHide(hwnd, m_fAutoHide);
    }

    void OnKey(HWND hwnd, UINT vk, BOOL fDown, int cRepeat, UINT flags)
    {
        if (vk == VK_ESCAPE)
            DestroyWindow(hwnd);
    }

    void OnAppBarCallback(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
    {
        static HWND s_hwndZOrder = NULL;

        switch (wParam)
        {
        case ABN_STATECHANGE:
            break;

        case ABN_FULLSCREENAPP:
            if (lParam)
            {
                s_hwndZOrder = GetWindow(hwnd, GW_HWNDPREV);
                SetWindowPos(hwnd, HWND_BOTTOM, 0, 0, 0, 0,
                             SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
            }
            else
            {
                SetWindowPos(hwnd, m_fOnTop ? HWND_TOPMOST : s_hwndZOrder,
                             0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
                s_hwndZOrder = NULL;
            }
            break;

        case ABN_POSCHANGED:
            {
                APPBARDATA abd = { sizeof(abd) };
                abd.hWnd = hwnd;
                AppBar_PosChanged(&abd);
            }
            break;
        }
    }

    BOOL AppBar_Register(HWND hwnd)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        abd.uCallbackMessage = APPBAR_CALLBACK;

        m_fAppBarRegd = (BOOL)SHAppBarMessage(ABM_NEW, &abd);
        return m_fAppBarRegd;
    }

    BOOL AppBar_UnRegister(HWND hwnd)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;

        m_fAppBarRegd = !SHAppBarMessage(ABM_REMOVE, &abd);
        return !m_fAppBarRegd;
    }

    BOOL AppBar_SetAutoHide(HWND hwnd, BOOL fHide)
    {
        if (fHide)
            return AppBar_AutoHide(hwnd);
        else
            return AppBar_NoAutoHide(hwnd);
    }

    BOOL AppBar_AutoHide(HWND hwnd)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        abd.uEdge = m_uSide;

        HWND hwndAutoHide = (HWND)SHAppBarMessage(ABM_GETAUTOHIDEBAR, &abd);
        if (hwndAutoHide)
            return FALSE;

        abd.lParam = TRUE;
        if (!(BOOL)SHAppBarMessage(ABM_SETAUTOHIDEBAR, &abd))
            return FALSE;

        m_fAutoHide = TRUE;
        m_cxSave = m_cxWidth;
        m_cySave = m_cyHeight;

        RECT rc = m_rcAppBar;
        switch (m_uSide)
        {
        case ABE_TOP:
            rc.bottom = rc.top + 2;
            break;
        case ABE_BOTTOM:
            rc.top = rc.bottom - 2;
            break;
        case ABE_LEFT:
            rc.right = rc.left + 2;
            break;
        case ABE_RIGHT:
            rc.left = rc.right - 2;
            break;
        }

        AppBar_QueryPos(hwnd, &rc);
        abd.rc = rc;
        SHAppBarMessage(ABM_SETPOS, &abd);
        rc = abd.rc;

        m_fHiding = TRUE;
        SlideWindow(hwnd, &rc);

        AppBar_SetAutoHideTimer(hwnd);
        return TRUE;
    }

    BOOL AppBar_NoAutoHide(HWND hwnd)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        abd.uEdge = m_uSide;
        HWND hwndAutoHide = (HWND)SHAppBarMessage(ABM_GETAUTOHIDEBAR, &abd);
        if (hwndAutoHide != hwnd)
            return FALSE;

        abd.lParam = FALSE;
        if (!(BOOL)SHAppBarMessage(ABM_SETAUTOHIDEBAR, &abd))
            return FALSE;

        m_fAutoHide = FALSE;
        m_cxWidth = m_cxSave;
        m_cyHeight = m_cySave;
        AppBar_SetSide(hwnd, m_uSide);
        return TRUE;
    }

    BOOL AppBar_SetSide(HWND hwnd, UINT uSide)
    {
        RECT rc;
        SetRect(&rc, 0, 0, GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN));

        BOOL fAutoHide = FALSE;
        if (m_fAutoHide)
        {
            fAutoHide = m_fAutoHide;
            SetWindowRedraw(GetDesktopWindow(), FALSE);
            AppBar_SetAutoHide(hwnd, FALSE);
            m_fHiding = FALSE;
        }

        switch (uSide)
        {
        case ABE_TOP:
            rc.bottom = rc.top + m_cyHeight;
            break;
        case ABE_BOTTOM:
            rc.top = rc.bottom - m_cyHeight;
            break;
        case ABE_LEFT:
            rc.right = rc.left + m_cxWidth;
            break;
        case ABE_RIGHT:
            rc.left = rc.right - m_cxWidth;
            break;
        }

        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        AppBar_QuerySetPos(uSide, &rc, &abd, TRUE);

        if (fAutoHide)
        {
            AppBar_SetAutoHide(hwnd, TRUE);
            m_fHiding = TRUE;

            SetWindowRedraw(GetDesktopWindow(), TRUE);
            RedrawWindow(GetDesktopWindow(), NULL, NULL,
                         RDW_INVALIDATE | RDW_UPDATENOW | RDW_ALLCHILDREN);
        }

        return TRUE;
    }

    void AppBar_SetAlwaysOnTop(HWND hwnd, BOOL fOnTop)
    {
        SetWindowPos(hwnd, (fOnTop ? HWND_TOPMOST : HWND_NOTOPMOST),
                     0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
        m_fOnTop = fOnTop;
    }

    void AppBar_Hide(HWND hwnd)
    {
        if (!m_fAutoHide)
            return;

        RECT rc = m_rcAppBar;
        switch (m_uSide)
        {
        case ABE_TOP:
            rc.bottom = rc.top + 2;
            break;
        case ABE_BOTTOM:
            rc.top = rc.bottom - 2;
            break;
        case ABE_LEFT:
            rc.right = rc.left + 2;
            break;
        case ABE_RIGHT:
            rc.left = rc.right - 2;
            break;
        }

        m_fHiding = TRUE;
        SlideWindow(hwnd, &rc);
    }

    void AppBar_UnHide(HWND hwnd)
    {
        SlideWindow(hwnd, &m_rcAppBar);
        m_fHiding = FALSE;

        AppBar_SetAutoHideTimer(hwnd);
    }

    void AppBar_SetAutoHideTimer(HWND hwnd)
    {
        if (m_fAutoHide)
        {
            SetTimer(hwnd, IDT_AUTOHIDE, 500, NULL);
        }
    }

    void AppBar_SetAutoUnhideTimer(HWND hwnd)
    {
        if (m_fAutoHide && m_fHiding)
        {
            SetTimer(hwnd, IDT_AUTOUNHIDE, 50, NULL);
        }
    }

    void AppBar_Size(HWND hwnd)
    {
        if (m_fAppBarRegd)
        {
            APPBARDATA abd = { sizeof(abd) };
            abd.hWnd = hwnd;

            RECT rc;
            GetWindowRect(hwnd, &rc);
            AppBar_QuerySetPos(m_uSide, &rc, &abd, TRUE);
        }
    }

    void AppBar_QueryPos(HWND hwnd, LPRECT lprc)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        abd.rc = *lprc;
        abd.uEdge = m_uSide;

        INT cx = 0, cy = 0;
        if (ABE_LEFT == abd.uEdge || ABE_RIGHT == abd.uEdge)
        {
            cx = abd.rc.right - abd.rc.left;
            abd.rc.top = 0;
            abd.rc.bottom = GetSystemMetrics(SM_CYSCREEN);
        }
        else
        {
            cy = abd.rc.bottom - abd.rc.top;
            abd.rc.left = 0;
            abd.rc.right = GetSystemMetrics(SM_CXSCREEN);
        }

        SHAppBarMessage(ABM_QUERYPOS, &abd);

        switch (abd.uEdge)
        {
        case ABE_LEFT:
            abd.rc.right = abd.rc.left + cx;
            break;
        case ABE_RIGHT:
            abd.rc.left = abd.rc.right - cx;
            break;
        case ABE_TOP:
            abd.rc.bottom = abd.rc.top + cy;
            break;
        case ABE_BOTTOM:
            abd.rc.top = abd.rc.bottom - cy;
            break;
        }

        *lprc = abd.rc;
    }

    void AppBar_QuerySetPos(UINT uEdge, LPRECT lprc, PAPPBARDATA pabd, BOOL fMove)
    {
        pabd->rc = *lprc;
        pabd->uEdge = uEdge;
        m_uSide = uEdge;

        AppBar_QueryPos(pabd->hWnd, &pabd->rc);

        SHAppBarMessage(ABM_SETPOS, pabd);

        if (fMove)
        {
            RECT rc = pabd->rc;
            MoveWindow(pabd->hWnd, rc.left, rc.top,
                       rc.right - rc.left, rc.bottom - rc.top, TRUE);
        }

        if (!m_fAutoHide)
        {
            m_rcAppBar = pabd->rc;
        }
    }

    void AppBar_PosChanged(PAPPBARDATA pabd)
    {
        RECT rc;
        SetRect(&rc, 0, 0, GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN));

        if (m_fAutoHide)
        {
            m_rcAppBar = rc;
            switch (m_uSide)
            {
            case ABE_TOP:
                m_rcAppBar.bottom = m_rcAppBar.top + m_cySave;
                break;
            case ABE_BOTTOM:
                m_rcAppBar.top = m_rcAppBar.bottom - m_cySave;
                break;
            case ABE_LEFT:
                m_rcAppBar.right = m_rcAppBar.left + m_cxSave;
                break;
            case ABE_RIGHT:
                m_rcAppBar.left = m_rcAppBar.right - m_cxSave;
                break;
            }
        }

        RECT rcWindow;
        GetWindowRect(pabd->hWnd, &rcWindow);
        INT cx = rcWindow.right - rcWindow.left;
        INT cy = rcWindow.bottom - rcWindow.top;
        switch (m_uSide)
        {
        case ABE_TOP:
            rc.bottom = rc.top + cy;
            break;
        case ABE_BOTTOM:
            rc.top = rc.bottom - cy;
            break;
        case ABE_LEFT:
            rc.right = rc.left + cx;
            break;
        case ABE_RIGHT:
            rc.left = rc.right - cx;
            break;
        }
        AppBar_QuerySetPos(m_uSide, &rc, pabd, TRUE);
    }

    BOOL OnCreate(HWND hwnd, LPCREATESTRUCT lpCreateStruct)
    {
        m_hwnd = hwnd;
        m_fOnTop = TRUE;
        m_uSide = ABE_TOP;

        m_fAppBarRegd = FALSE;
        m_fMoving = FALSE;
        m_cxSave = m_cxWidth;
        m_cySave = m_cyHeight;
        m_bDragged = FALSE;

        AppBar_Register(hwnd);
        AppBar_SetSide(hwnd, ABE_TOP);

        return TRUE;
    }

    void OnActivate(HWND hwnd, UINT state, HWND hwndActDeact, BOOL fMinimized)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        SHAppBarMessage(ABM_ACTIVATE, &abd);

        switch (state)
        {
            case WA_ACTIVE:
            case WA_CLICKACTIVE:
                AppBar_UnHide(hwnd);
                KillTimer(hwnd, IDT_AUTOHIDE);
                break;

            case WA_INACTIVE:
                AppBar_Hide(hwnd);
                break;
        }
    }

    void OnWindowPosChanged(HWND hwnd, const LPWINDOWPOS lpwpos)
    {
        APPBARDATA abd = { sizeof(abd) };
        abd.hWnd = hwnd;
        SHAppBarMessage(ABM_WINDOWPOSCHANGED, &abd);

        FORWARD_WM_WINDOWPOSCHANGED(hwnd, lpwpos, DefWindowProc);
    }

    void OnSize(HWND hwnd, UINT state, int cx, int cy)
    {
        RECT rcWindow;

        if (m_fMoving || (m_fAutoHide && m_fHiding))
            return;

        if (!m_fHiding)
        {
            if (!m_fAutoHide)
                AppBar_Size(hwnd);

            GetWindowRect(hwnd, &rcWindow);
            m_rcAppBar = rcWindow;

            if (m_uSide == ABE_TOP || m_uSide == ABE_BOTTOM)
            {
                m_cyHeight = m_cySave = rcWindow.bottom - rcWindow.top;
            }
            else
            {
                m_cxWidth = m_cxSave = rcWindow.right - rcWindow.left;
            }
        }

        InvalidateRect(hwnd, NULL, TRUE);
    }

    void OnMove(HWND hwnd, int x, int y)
    {
        if (m_fMoving || m_fAutoHide)
            return;

        if (!m_fHiding)
            AppBar_Size(hwnd);
    }

    void OnNCDestroy(HWND hwnd)
    {
        AppBar_UnRegister(hwnd);

        m_hwnd = NULL;
        SetWindowLongPtr(hwnd, GWLP_USERDATA, 0);
        delete this;
    }

    void OnTimer(HWND hwnd, UINT id)
    {
        POINT pt;
        RECT rc;
        HWND hwndActive;

        switch (id)
        {
        case IDT_AUTOHIDE:
            if (m_fAutoHide && !m_fHiding && !m_fMoving)
            {
                GetCursorPos(&pt);
                GetWindowRect(hwnd, &rc);
                hwndActive = GetForegroundWindow();

                if (!PtInRect(&rc, pt) &&
                    hwndActive != hwnd &&
                    hwndActive != NULL &&
                    GetWindowOwner(hwndActive) != hwnd)
                {
                    KillTimer(hwnd, id);
                    AppBar_Hide(hwnd);
                }
            }
            break;

        case IDT_AUTOUNHIDE:
            KillTimer(hwnd, id);

            if (m_fAutoHide && m_fHiding)
            {
                GetCursorPos(&pt);
                GetWindowRect(hwnd, &rc);
                if (PtInRect(&rc, pt))
                {
                    AppBar_UnHide(hwnd);
                }
            }
            break;
        }
    }

    UINT OnNCHitTest(HWND hwnd, int x, int y)
    {
        AppBar_SetAutoUnhideTimer(hwnd);

        UINT uHitTest = FORWARD_WM_NCHITTEST(hwnd, x, y, DefWindowProc);

        if (m_uSide == ABE_TOP && uHitTest == HTBOTTOM)
            return HTBOTTOM;

        if (m_uSide == ABE_BOTTOM && uHitTest == HTTOP)
            return HTTOP;

        if (m_uSide == ABE_LEFT && uHitTest == HTRIGHT)
            return HTRIGHT;

        if (m_uSide == ABE_RIGHT && uHitTest == HTLEFT)
            return HTLEFT;

        return HTCLIENT;
    }

    void OnLButtonDown(HWND hwnd, BOOL fDoubleClick, int x, int y, UINT keyFlags)
    {
        m_fMoving = TRUE;
        m_bDragged = FALSE;
        SetCapture(hwnd);
        GetCursorPos(&m_ptDragOn);
    }

    void OnMouseMove(HWND hwnd, int x, int y, UINT keyFlags)
    {
        if (!m_fMoving)
            return;

        POINT pt;
        GetCursorPos(&pt);
        if (labs(pt.x - m_ptDragOn.x) > GetSystemMetrics(SM_CXDRAG) ||
            labs(pt.y - m_ptDragOn.y) > GetSystemMetrics(SM_CYDRAG))
        {
            m_bDragged = TRUE;
        }

        INT cxScreen = GetSystemMetrics(SM_CXSCREEN);
        INT cyScreen = GetSystemMetrics(SM_CYSCREEN);

        DWORD dx, dy;
        UINT ix, iy;
        if (pt.x < cxScreen / 2)
        {
            dx = pt.x;
            ix = ABE_LEFT;
        }
        else
        {
            dx = cxScreen - pt.x;
            ix = ABE_RIGHT;
        }

        if (pt.y < cyScreen / 2)
        {
            dy = pt.y;
            iy = ABE_TOP;
        }
        else
        {
            dy = cyScreen - pt.y;
            iy = ABE_BOTTOM;
        }

        if (cxScreen * dy > cyScreen * dx)
        {
            m_rcDrag.top = 0;
            m_rcDrag.bottom = cyScreen;
            if (ix == ABE_LEFT)
            {
                m_uSide = ABE_LEFT;
                m_rcDrag.left = 0;
                m_rcDrag.right = m_rcDrag.left + m_cxWidth;
            }
            else
            {
                m_uSide = ABE_RIGHT;
                m_rcDrag.right = cxScreen;
                m_rcDrag.left = m_rcDrag.right - m_cxWidth;
            }
        }
        else
        {
            m_rcDrag.left = 0;
            m_rcDrag.right = cxScreen;
            if (iy == ABE_TOP)
            {
                m_uSide = ABE_TOP;
                m_rcDrag.top = 0;
                m_rcDrag.bottom = m_rcDrag.top + m_cyHeight;
            }
            else
            {
                m_uSide = ABE_BOTTOM;
                m_rcDrag.bottom = cyScreen;
                m_rcDrag.top = m_rcDrag.bottom - m_cyHeight;
            }
        }

        AppBar_QueryPos(hwnd, &m_rcDrag);

        if (m_bDragged)
        {
            MoveWindow(hwnd, m_rcDrag.left, m_rcDrag.top,
                       m_rcDrag.right - m_rcDrag.left,
                       m_rcDrag.bottom - m_rcDrag.top,
                       TRUE);
        }
    }

    void OnLButtonUp(HWND hwnd, int x, int y, UINT keyFlags)
    {
        if (!m_fMoving)
            return;

        OnMouseMove(hwnd, x, y, keyFlags);

        m_rcAppBar = m_rcDrag;

        ReleaseCapture();

        if (m_fAutoHide)
        {
            switch (m_uSide)
            {
            case ABE_TOP:
                m_rcDrag.bottom = m_rcDrag.top + 2;
                break;
            case ABE_BOTTOM:
                m_rcDrag.top = m_rcDrag.bottom - 2;
                break;
            case ABE_LEFT:
                m_rcDrag.right = m_rcDrag.left + 2;
                break;
            case ABE_RIGHT:
                m_rcDrag.left = m_rcDrag.right - 2;
                break;
            }
        }

        if (m_bDragged)
        {
            if (m_fAutoHide)
            {
                AppBar_AutoHide(hwnd);
            }
            else
            {
                APPBARDATA abd = { sizeof(abd) };
                abd.hWnd = hwnd;
                AppBar_QuerySetPos(m_uSide, &m_rcDrag, &abd, FALSE);
            }
        }

        m_fMoving = FALSE;
    }

    void GetWorkArea(LPRECT prc) const
    {
        SystemParametersInfoW(SPI_GETWORKAREA, 0, prc, 0);
    }

public:
    void DoAction()
    {
#define INTERVAL 250
        POINT pt;
        RECT rc1, rc2, rcWork;
        DWORD dwTID = GetWindowThreadProcessId(s_hwnd1, NULL);

        GetWindowRect(s_hwnd1, &rc1);
        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc1.left, s_rcWorkArea.left);
        ok_long(rc1.top, s_rcWorkArea.top);
        ok_long(rc1.right, s_rcWorkArea.right);
        ok_long(rc1.bottom, s_rcWorkArea.top + 80);
        ok_long(rc2.left, s_rcWorkArea.left);
        ok_long(rc2.top, s_rcWorkArea.top + 80);
        ok_long(rc2.right, s_rcWorkArea.right);
        ok_long(rc2.bottom, s_rcWorkArea.top + 110);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top + 110);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        PostMessage(s_hwnd1, WM_CLOSE, 0, 0);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.left);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.right);
        ok_long(rc2.bottom, s_rcWorkArea.top + 30);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top + 30);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        AppBar_SetSide(s_hwnd2, ABE_LEFT);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.left);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.left + 30);
        ok_long(rcWork.left, s_rcWorkArea.left + 30);
        ok_long(rcWork.top, s_rcWorkArea.top);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        AppBar_SetSide(s_hwnd2, ABE_TOP);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.left);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.right);
        ok_long(rc2.bottom, s_rcWorkArea.top + 30);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top + 30);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        AppBar_SetSide(s_hwnd2, ABE_RIGHT);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.right - 30);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.right);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top);
        ok_long(rcWork.right, s_rcWorkArea.right - 30);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        pt.x = (rc2.left + rc2.right) / 2;
        pt.y = (rc2.top + rc2.bottom) / 2;
        MOVE(pt.x, pt.y);
        LEFT_DOWN();
        MOVE(pt.x + 64, pt.y + 64);
        Sleep(INTERVAL);

        pt.x = s_rcWorkArea.left + 80;
        pt.y = (s_rcWorkArea.top + s_rcWorkArea.bottom) / 2;
        MOVE(pt.x, pt.y);
        LEFT_UP();
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.left);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.left + 30);
        ok_long(rcWork.left, s_rcWorkArea.left + 30);
        ok_long(rcWork.top, s_rcWorkArea.top);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        pt.x = (rc2.left + rc2.right) / 2;
        pt.y = (rc2.top + rc2.bottom) / 2;
        MOVE(pt.x, pt.y);
        LEFT_DOWN();
        MOVE(pt.x + 64, pt.y + 64);
        Sleep(INTERVAL);

        pt.x = s_rcWorkArea.right - 80;
        pt.y = (s_rcWorkArea.top + s_rcWorkArea.bottom) / 2;
        MOVE(pt.x, pt.y);
        LEFT_UP();
        Sleep(INTERVAL);

        GetWindowRect(s_hwnd2, &rc2);
        GetWorkArea(&rcWork);
        ok_long(rc2.left, s_rcWorkArea.right - 30);
        ok_long(rc2.top, s_rcWorkArea.top);
        ok_long(rc2.right, s_rcWorkArea.right);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top);
        ok_long(rcWork.right, s_rcWorkArea.right - 30);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);
        Sleep(INTERVAL);

        SendMessage(s_hwnd2, WM_CLOSE, 0, 0);
        Sleep(INTERVAL);

        GetWorkArea(&rcWork);
        ok_long(rcWork.left, s_rcWorkArea.left);
        ok_long(rcWork.top, s_rcWorkArea.top);
        ok_long(rcWork.right, s_rcWorkArea.right);
        ok_long(rcWork.bottom, s_rcWorkArea.bottom);

        PostMessage(s_hwnd2, WM_QUIT, 0, 0);
        PostThreadMessage(dwTID, WM_QUIT, 0, 0);
#undef INTERVAL
    }

    static DWORD WINAPI ActionThreadFunc(LPVOID args)
    {
        Window *this_ = (Window *)args;
        this_->DoAction();
        return 0;
    }
};

START_TEST(SHAppBarMessage)
{
    HINSTANCE hInstance = GetModuleHandle(NULL);

    if (!Window::DoRegisterClass(hInstance))
    {
        skip("Window::DoRegisterClass failed\n");
        return;
    }

    SystemParametersInfo(SPI_GETWORKAREA, 0, &s_rcWorkArea, FALSE);

    HWND hwnd1 = Window::DoCreateMainWnd(hInstance, TEXT("Test1"), 80, 80,
                                         WS_POPUP | WS_THICKFRAME | WS_CLIPCHILDREN);
    if (!hwnd1)
    {
        skip("CreateWindowExW failed\n");
        return;
    }

    HWND hwnd2 = Window::DoCreateMainWnd(hInstance, TEXT("Test2"), 30, 30,
                                         WS_POPUP | WS_BORDER | WS_CLIPCHILDREN);
    if (!hwnd2)
    {
        skip("CreateWindowExW failed\n");
        return;
    }

    s_hwnd1 = hwnd1;
    s_hwnd2 = hwnd2;

    PostMessage(hwnd1, WM_COMMAND, ID_ACTION, 0);

    Window::DoMainLoop();
}