[EXPLORER] Show the notification settings dialog from it's context menu entry
[reactos.git] / reactos / base / shell / explorer / traywnd.cpp
index dbbb0a4..b0b8dba 100644 (file)
@@ -55,7 +55,7 @@ HRESULT TrayWindowCtxMenuCreator(ITrayWindow * TrayWnd, IN HWND hWndOwner, ICont
 
 static LONG TrayWndCount = 0;
 
-static const WCHAR szTrayWndClass [] = TEXT("Shell_TrayWnd");
+static const WCHAR szTrayWndClass[] = L"Shell_TrayWnd";
 
 /*
  * ITrayWindow
@@ -64,16 +64,15 @@ static const WCHAR szTrayWndClass [] = TEXT("Shell_TrayWnd");
 const GUID IID_IShellDesktopTray = { 0x213e2df9, 0x9a14, 0x4328, { 0x99, 0xb1, 0x69, 0x61, 0xf9, 0x14, 0x3c, 0xe9 } };
 
 class CStartButton
-    : public CContainedWindow
+    : public CWindow
 {
     HIMAGELIST m_ImageList;
     SIZE       m_Size;
     HFONT      m_Font;
 
 public:
-    CStartButton(CMessageMap *pObject, DWORD dwMsgMapID)
-        : CContainedWindow(pObject, dwMsgMapID),
-          m_ImageList(NULL),
+    CStartButton()
+        : m_ImageList(NULL),
           m_Font(NULL)
     {
         m_Size.cx = 0;
@@ -106,250 +105,67 @@ public:
         if (m_ImageList == NULL ||
             !SendMessageW(BCM_GETIDEALSIZE, 0, (LPARAM) &Size))
         {
-            Size.cx = 2 * GetSystemMetrics(SM_CXEDGE);
-            Size.cy = 2 * GetSystemMetrics(SM_CYEDGE);
-
-            if (hbmStart == NULL)
-            {
-                hbmStart = (HBITMAP) SendMessageW(BM_GETIMAGE, IMAGE_BITMAP, 0);
-            }
-
-            if (hbmStart != NULL)
-            {
-                BITMAP bmp;
-
-                if (GetObject(hbmStart, sizeof(bmp), &bmp) != 0)
-                {
-                    Size.cx += bmp.bmWidth;
-                    Size.cy += max(bmp.bmHeight, GetSystemMetrics(SM_CYCAPTION));
-                }
-                else
-                {
-                    /* Huh?! Shouldn't happen... */
-                    goto DefSize;
-                }
-            }
-            else
-            {
-DefSize:
-                Size.cx += GetSystemMetrics(SM_CXMINIMIZED);
-                Size.cy += GetSystemMetrics(SM_CYCAPTION);
-            }
+            Size.cx = 2 * GetSystemMetrics(SM_CXEDGE) + GetSystemMetrics(SM_CYCAPTION) * 3;
+            Size.cy = 2 * GetSystemMetrics(SM_CYEDGE) + GetSystemMetrics(SM_CYCAPTION);
         }
 
         /* Save the size of the start button */
         m_Size = Size;
     }
 
-    BOOL CreateImageList()
-    {
-        HICON hIconStart;
-        SIZE IconSize;
-
-        if (m_ImageList != NULL)
-            return TRUE;
-
-        IconSize.cx = GetSystemMetrics(SM_CXSMICON);
-        IconSize.cy = GetSystemMetrics(SM_CYSMICON);
-
-        /* Load the start button icon and create a image list for it */
-        hIconStart = (HICON) LoadImage(hExplorerInstance,
-                                       MAKEINTRESOURCE(IDI_START),
-                                       IMAGE_ICON,
-                                       IconSize.cx,
-                                       IconSize.cy,
-                                       LR_SHARED | LR_DEFAULTCOLOR);
-
-        if (hIconStart == NULL)
-            return FALSE;
-
-        m_ImageList = ImageList_Create(IconSize.cx,
-                                        IconSize.cy,
-                                        ILC_COLOR32 | ILC_MASK,
-                                        1, 1);
-        if (m_ImageList == NULL)
-            return FALSE;
-
-        int s = ImageList_ReplaceIcon(m_ImageList, -1, hIconStart);
-        if (s < 0)
-        {
-            /* Failed to add the icon! */
-            ImageList_Destroy(m_ImageList);
-            m_ImageList = NULL;
-
-            return FALSE;
-        }
-
-        return TRUE;
-    }
-
-    HBITMAP CreateBitmap()
+    VOID Initialize()
     {
-        WCHAR szStartCaption[32];
-        HFONT hFontOld;
-        HDC hDC = NULL;
-        HDC hDCScreen = NULL;
-        SIZE Size, SmallIcon;
-        HBITMAP hbmpOld, hbmp = NULL;
-        HBITMAP hBitmap = NULL;
-        HICON hIconStart;
-        BOOL Ret;
-        UINT Flags;
-        RECT rcButton;
-
-        /* NOTE: this is the backwards compatibility code that is used if the
-        Common Controls Version 6.0 are not available! */
-
-        if (!LoadString(hExplorerInstance,
-            IDS_START,
-            szStartCaption,
-            sizeof(szStartCaption) / sizeof(szStartCaption[0])))
-        {
-            return NULL;
-        }
-
-        /* Load the start button icon */
-        SmallIcon.cx = GetSystemMetrics(SM_CXSMICON);
-        SmallIcon.cy = GetSystemMetrics(SM_CYSMICON);
-        hIconStart = (HICON) LoadImage(hExplorerInstance,
-                                       MAKEINTRESOURCE(IDI_START),
-                                       IMAGE_ICON,
-                                       SmallIcon.cx,
-                                       SmallIcon.cy,
-                                       LR_SHARED | LR_DEFAULTCOLOR);
-
-        hDCScreen = GetDC(NULL);
-        if (hDCScreen == NULL)
-            goto Cleanup;
-
-        hDC = CreateCompatibleDC(hDCScreen);
-        if (hDC == NULL)
-            goto Cleanup;
-
-        hFontOld = (HFONT) SelectObject(hDC, m_Font);
-
-        Ret = GetTextExtentPoint32(hDC,
-                                   szStartCaption,
-                                   _tcslen(szStartCaption),
-                                   &Size);
-
-        SelectObject(hDC, hFontOld);
-        if (!Ret)
-            goto Cleanup;
-
-        /* Make sure the height is at least the size of a caption icon. */
-        if (hIconStart != NULL)
-            Size.cx += SmallIcon.cx + 4;
-        Size.cy = max(Size.cy, SmallIcon.cy);
-
-        /* Create the bitmap */
-        hbmp = CreateCompatibleBitmap(hDCScreen,
-                                      Size.cx,
-                                      Size.cy);
-        if (hbmp == NULL)
-            goto Cleanup;
-
-        /* Caluclate the button rect */
-        rcButton.left = 0;
-        rcButton.top = 0;
-        rcButton.right = Size.cx;
-        rcButton.bottom = Size.cy;
-
-        /* Draw the button */
-        hbmpOld = (HBITMAP) SelectObject(hDC, hbmp);
-
-        Flags = DC_TEXT | DC_INBUTTON;
-        if (hIconStart != NULL)
-            Flags |= DC_ICON;
-
-        DrawCaptionTemp(NULL,
-                        hDC,
-                        &rcButton,
-                        m_Font,
-                        hIconStart,
-                        szStartCaption,
-                        Flags);
-
-        SelectObject(hDC, hbmpOld);
-
-        if (!Ret)
-            goto Cleanup;
-
-        /* We successfully created the bitmap! */
-        hBitmap = hbmp;
-        hbmp = NULL;
+        SetWindowTheme(m_hWnd, L"Start", NULL);
 
-Cleanup:
-        if (hDCScreen != NULL)
+        /* Get the system fonts, we use the caption font, always bold, though. */
+        NONCLIENTMETRICS ncm = {sizeof(ncm)};
+        if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE))
         {
-            ReleaseDC(NULL, hDCScreen);
+            ncm.lfCaptionFont.lfWeight = FW_BOLD;
+            m_Font = CreateFontIndirect(&ncm.lfCaptionFont);
         }
 
-        if (hbmp != NULL)
-            DeleteObject(hbmp);
+        SetFont(m_Font, FALSE);
 
-        if (hDC != NULL)
-            DeleteDC(hDC);
+        m_ImageList = ImageList_LoadImageW(hExplorerInstance,
+                                           MAKEINTRESOURCEW(IDB_START),
+                                           0, 0, 0,
+                                           IMAGE_BITMAP,
+                                           LR_LOADTRANSPARENT | LR_CREATEDIBSECTION);
 
-        return hBitmap;
+        BUTTON_IMAGELIST bil = {m_ImageList, {1,1,1,1}, BUTTON_IMAGELIST_ALIGN_LEFT};
+        SendMessageW(BCM_SETIMAGELIST, 0, (LPARAM) &bil);
+        UpdateSize();
     }
 
-    VOID Initialize()
+    HWND Create(HWND hwndParent)
     {
-        NONCLIENTMETRICS ncm;
-
-        SetWindowTheme(m_hWnd, L"Start", NULL);
-
-        if (m_Font == NULL)
+        WCHAR szStartCaption[32];
+        if (!LoadStringW(hExplorerInstance,
+                         IDS_START,
+                         szStartCaption,
+                         _countof(szStartCaption)))
         {
-            /* Get the system fonts, we use the caption font, always bold, though. */
-            ncm.cbSize = sizeof(ncm);
-            if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE))
-            {
-                ncm.lfCaptionFont.lfWeight = FW_BOLD;
-                m_Font = CreateFontIndirect(&ncm.lfCaptionFont);
-            }
+            wcscpy(szStartCaption, L"Start");
         }
 
-        SendMessage(WM_SETFONT, (WPARAM) m_Font, FALSE);
-
-        if (CreateImageList())
-        {
-            BUTTON_IMAGELIST bil;
-
-            /* Try to set the start button image. this requires the Common
-            Controls 6.0 to be present (XP and later) */
-            bil.himl = m_ImageList;
-            bil.margin.left = bil.margin.right = 1;
-            bil.margin.top = bil.margin.bottom = 1;
-            bil.uAlign = BUTTON_IMAGELIST_ALIGN_LEFT;
-
-            if (SendMessageW(BCM_SETIMAGELIST, 0, (LPARAM) &bil))
-            {
-                /* We're using the image list, remove the BS_BITMAP style and
-                don't center it horizontally */
-                SetWindowStyle(m_hWnd, BS_BITMAP | BS_RIGHT, 0);
-
-                UpdateSize();
-                return;
-            }
+        DWORD dwStyle = WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | BS_PUSHBUTTON | BS_LEFT | BS_VCENTER;
 
-            /* Fall back to the deprecated method on older systems that don't
-            support Common Controls 6.0 */
-            ImageList_Destroy(m_ImageList);
-            m_ImageList = NULL;
-        }
-
-        HBITMAP hbmStart = CreateBitmap();
-        if (hbmStart != NULL)
-        {
-            UpdateSize(hbmStart);
+        m_hWnd = CreateWindowEx(
+            0,
+            WC_BUTTON,
+            szStartCaption,
+            dwStyle,
+            0, 0, 0, 0,
+            hwndParent,
+            (HMENU) IDC_STARTBTN,
+            hExplorerInstance,
+            NULL);
 
-            HBITMAP hbmOld = (HBITMAP) SendMessageW(BM_SETIMAGE, IMAGE_BITMAP, (LPARAM) hbmStart);
+        if (m_hWnd)
+            Initialize();
 
-            if (hbmOld != NULL)
-                DeleteObject(hbmOld);
-        }
+        return m_hWnd;
     }
 };
 
@@ -418,7 +234,7 @@ public:
 
 public:
     CTrayWindow() :
-        m_StartButton(this, 1),
+        m_StartButton(),
         m_Theme(NULL),
         m_CaptionFont(NULL),
         m_Font(NULL),
@@ -473,718 +289,325 @@ public:
             PostQuitMessage(0);
     }
 
-    /*
-     * ITrayWindow
-     */
 
-    BOOL UpdateNonClientMetrics()
-    {
-        NONCLIENTMETRICS ncm;
-        ncm.cbSize = sizeof(ncm);
-        if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, 0))
-        {
-            if (m_Font != NULL)
-                DeleteObject(m_Font);
 
-            m_Font = CreateFontIndirect(&ncm.lfMessageFont);
-            return TRUE;
-        }
 
-        return FALSE;
-    }
 
-    VOID SetWindowsFont()
-    {
-        if (m_TrayNotify != NULL)
-        {
-            SendMessage(m_TrayNotify, WM_SETFONT, (WPARAM) m_Font, TRUE);
-        }
-    }
+    /**********************************************************
+     *    ##### command handling #####
+     */
 
-    HMONITOR GetScreenRectFromRect(
-        IN OUT RECT *pRect,
-        IN DWORD dwFlags)
+    HRESULT ExecResourceCmd(int id)
     {
-        MONITORINFO mi;
-        HMONITOR hMon;
+        WCHAR szCommand[256];
+        WCHAR *pszParameters;
 
-        mi.cbSize = sizeof(mi);
-        hMon = MonitorFromRect(pRect,
-                               dwFlags);
-        if (hMon != NULL &&
-            GetMonitorInfo(hMon,
-            &mi))
+        if (!LoadStringW(hExplorerInstance,
+                         id,
+                         szCommand,
+                         _countof(szCommand)))
         {
-            *pRect = mi.rcMonitor;
+            return E_FAIL;
         }
-        else
-        {
-            pRect->left = 0;
-            pRect->top = 0;
-            pRect->right = GetSystemMetrics(SM_CXSCREEN);
-            pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
 
-            hMon = NULL;
+        pszParameters = wcschr(szCommand, L'>');
+        if (pszParameters)
+        {
+            *pszParameters = 0;
+            pszParameters++;
         }
 
-        return hMon;
+        ShellExecuteW(m_hWnd, NULL, szCommand, pszParameters, NULL, SW_SHOWNORMAL);
+        return S_OK;
     }
 
-    HMONITOR GetMonitorFromRect(
-        IN const RECT *pRect)
+    LRESULT DoExitWindows()
     {
-        HMONITOR hMon;
+        ExitWindowsDialog(m_hWnd);
+        return 0;
+    }
 
-        /* In case the monitor sizes or saved sizes differ a bit (probably
-           not a lot, only so the tray window overlaps into another monitor
-           now), minimize the risk that we determine a wrong monitor by
-           using the center point of the tray window if we can't determine
-           it using the rectangle. */
-        hMon = MonitorFromRect(pRect,
-                               MONITOR_DEFAULTTONULL);
-        if (hMon == NULL)
-        {
-            POINT pt;
+    DWORD WINAPI RunFileDlgThread()
+    {
+        HWND hwnd;
+        RECT posRect;
 
-            pt.x = pRect->left + ((pRect->right - pRect->left) / 2);
-            pt.y = pRect->top + ((pRect->bottom - pRect->top) / 2);
+        m_StartButton.GetWindowRect(&posRect);
 
-            /* be less error-prone, find the nearest monitor */
-            hMon = MonitorFromPoint(pt,
-                                    MONITOR_DEFAULTTONEAREST);
-        }
+        hwnd = CreateWindowEx(0,
+                              WC_STATIC,
+                              NULL,
+                              WS_OVERLAPPED | WS_DISABLED | WS_CLIPSIBLINGS | WS_BORDER | SS_LEFT,
+                              posRect.left,
+                              posRect.top,
+                              posRect.right - posRect.left,
+                              posRect.bottom - posRect.top,
+                              NULL,
+                              NULL,
+                              NULL,
+                              NULL);
 
-        return hMon;
+        m_RunFileDlgOwner = hwnd;
+
+        RunFileDlg(hwnd, NULL, NULL, NULL, NULL, RFF_CALCDIRECTORY);
+
+        m_RunFileDlgOwner = NULL;
+        ::DestroyWindow(hwnd);
+
+        return 0;
     }
 
-    HMONITOR GetScreenRect(
-        IN HMONITOR hMonitor,
-        IN OUT RECT *pRect)
+    static DWORD WINAPI s_RunFileDlgThread(IN OUT PVOID pParam)
     {
-        HMONITOR hMon = NULL;
+        CTrayWindow * This = (CTrayWindow*) pParam;
+        return This->RunFileDlgThread();
+    }
 
-        if (hMonitor != NULL)
+    void DisplayRunFileDlg()
+    {
+        HWND hRunDlg;
+        if (m_RunFileDlgOwner)
         {
-            MONITORINFO mi;
-
-            mi.cbSize = sizeof(mi);
-            if (!GetMonitorInfo(hMonitor,
-                &mi))
+            hRunDlg = ::GetLastActivePopup(m_RunFileDlgOwner);
+            if (hRunDlg != NULL &&
+                hRunDlg != m_RunFileDlgOwner)
             {
-                /* Hm, the monitor is gone? Try to find a monitor where it
-                   could be located now */
-                hMon = GetMonitorFromRect(
-                    pRect);
-                if (hMon == NULL ||
-                    !GetMonitorInfo(hMon,
-                    &mi))
-                {
-                    hMon = NULL;
-                    goto GetPrimaryRect;
-                }
+                SetForegroundWindow(hRunDlg);
+                return;
             }
-
-            *pRect = mi.rcMonitor;
-        }
-        else
-        {
-GetPrimaryRect:
-            pRect->left = 0;
-            pRect->top = 0;
-            pRect->right = GetSystemMetrics(SM_CXSCREEN);
-            pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
         }
 
-        return hMon;
+        CloseHandle(CreateThread(NULL, 0, s_RunFileDlgThread, this, 0, NULL));
     }
 
-    VOID MakeTrayRectWithSize(IN DWORD Position,
-                              IN const SIZE *pTraySize,
-                              IN OUT RECT *pRect)
+    DWORD WINAPI TrayPropertiesThread()
     {
-        switch (Position)
-        {
-        case ABE_LEFT:
-            pRect->right = pRect->left + pTraySize->cx;
-            break;
+        HWND hwnd;
+        RECT posRect;
 
-        case ABE_TOP:
-            pRect->bottom = pRect->top + pTraySize->cy;
-            break;
+        m_StartButton.GetWindowRect(&posRect);
+        hwnd = CreateWindowEx(0,
+                              WC_STATIC,
+                              NULL,
+                              WS_OVERLAPPED | WS_DISABLED | WS_CLIPSIBLINGS | WS_BORDER | SS_LEFT,
+                              posRect.left,
+                              posRect.top,
+                              posRect.right - posRect.left,
+                              posRect.bottom - posRect.top,
+                              NULL,
+                              NULL,
+                              NULL,
+                              NULL);
 
-        case ABE_RIGHT:
-            pRect->left = pRect->right - pTraySize->cx;
-            break;
+        m_TrayPropertiesOwner = hwnd;
 
-        case ABE_BOTTOM:
-        default:
-            pRect->top = pRect->bottom - pTraySize->cy;
-            break;
-        }
-    }
-
-    VOID GetTrayRectFromScreenRect(IN DWORD Position,
-                                   IN const RECT *pScreen,
-                                   IN const SIZE *pTraySize OPTIONAL,
-                                   OUT RECT *pRect)
-    {
-        if (pTraySize == NULL)
-            pTraySize = &m_TraySize;
-
-        *pRect = *pScreen;
-
-        /* Move the border outside of the screen */
-        InflateRect(pRect,
-                    GetSystemMetrics(SM_CXEDGE),
-                    GetSystemMetrics(SM_CYEDGE));
-
-        MakeTrayRectWithSize(Position, pTraySize, pRect);
-    }
-
-    BOOL IsPosHorizontal()
-    {
-        return m_Position == ABE_TOP || m_Position == ABE_BOTTOM;
-    }
-
-    HMONITOR CalculateValidSize(
-        IN DWORD Position,
-        IN OUT RECT *pRect)
-    {
-        RECT rcScreen;
-        //BOOL Horizontal;
-        HMONITOR hMon;
-        SIZE szMax, szWnd;
-
-        //Horizontal = IsPosHorizontal();
-
-        szWnd.cx = pRect->right - pRect->left;
-        szWnd.cy = pRect->bottom - pRect->top;
-
-        rcScreen = *pRect;
-        hMon = GetScreenRectFromRect(
-            &rcScreen,
-            MONITOR_DEFAULTTONEAREST);
-
-        /* Calculate the maximum size of the tray window and limit the window
-           size to half of the screen's size. */
-        szMax.cx = (rcScreen.right - rcScreen.left) / 2;
-        szMax.cy = (rcScreen.bottom - rcScreen.top) / 2;
-        if (szWnd.cx > szMax.cx)
-            szWnd.cx = szMax.cx;
-        if (szWnd.cy > szMax.cy)
-            szWnd.cy = szMax.cy;
-
-        /* FIXME - calculate */
+        DisplayTrayProperties(hwnd);
 
-        GetTrayRectFromScreenRect(
-            Position,
-            &rcScreen,
-            &szWnd,
-            pRect);
+        m_TrayPropertiesOwner = NULL;
+        ::DestroyWindow(hwnd);
 
-        return hMon;
+        return 0;
     }
 
-#if 0
-    VOID
-        GetMinimumWindowSize(
-        OUT RECT *pRect)
+    static DWORD WINAPI s_TrayPropertiesThread(IN OUT PVOID pParam)
     {
-        RECT rcMin = {0};
-
-        AdjustWindowRectEx(&rcMin,
-                           GetWindowLong(m_hWnd,
-                           GWL_STYLE),
-                           FALSE,
-                           GetWindowLong(m_hWnd,
-                           GWL_EXSTYLE));
+        CTrayWindow *This = (CTrayWindow*) pParam;
 
-        *pRect = rcMin;
+        return This->TrayPropertiesThread();
     }
-#endif
 
-
-    DWORD GetDraggingRectFromPt(
-        IN POINT pt,
-        OUT RECT *pRect,
-        OUT HMONITOR *phMonitor)
+    HWND STDMETHODCALLTYPE DisplayProperties()
     {
-        HMONITOR hMon, hMonNew;
-        DWORD PosH, PosV, Pos;
-        SIZE DeltaPt, ScreenOffset;
-        RECT rcScreen;
-
-        rcScreen.left = 0;
-        rcScreen.top = 0;
-
-        /* Determine the screen rectangle */
-        hMon = MonitorFromPoint(pt,
-                                MONITOR_DEFAULTTONULL);
+        HWND hTrayProp;
 
-        if (hMon != NULL)
+        if (m_TrayPropertiesOwner)
         {
-            MONITORINFO mi;
-
-            mi.cbSize = sizeof(mi);
-            if (!GetMonitorInfo(hMon,
-                &mi))
+            hTrayProp = ::GetLastActivePopup(m_TrayPropertiesOwner);
+            if (hTrayProp != NULL &&
+                hTrayProp != m_TrayPropertiesOwner)
             {
-                hMon = NULL;
-                goto GetPrimaryScreenRect;
+                SetForegroundWindow(hTrayProp);
+                return NULL;
             }
-
-            /* make left top corner of the screen zero based to
-               make calculations easier */
-            pt.x -= mi.rcMonitor.left;
-            pt.y -= mi.rcMonitor.top;
-
-            ScreenOffset.cx = mi.rcMonitor.left;
-            ScreenOffset.cy = mi.rcMonitor.top;
-            rcScreen.right = mi.rcMonitor.right - mi.rcMonitor.left;
-            rcScreen.bottom = mi.rcMonitor.bottom - mi.rcMonitor.top;
-        }
-        else
-        {
-GetPrimaryScreenRect:
-            ScreenOffset.cx = 0;
-            ScreenOffset.cy = 0;
-            rcScreen.right = GetSystemMetrics(SM_CXSCREEN);
-            rcScreen.bottom = GetSystemMetrics(SM_CYSCREEN);
         }
 
-        /* Calculate the nearest screen border */
-        if (pt.x < rcScreen.right / 2)
-        {
-            DeltaPt.cx = pt.x;
-            PosH = ABE_LEFT;
-        }
-        else
-        {
-            DeltaPt.cx = rcScreen.right - pt.x;
-            PosH = ABE_RIGHT;
-        }
-
-        if (pt.y < rcScreen.bottom / 2)
-        {
-            DeltaPt.cy = pt.y;
-            PosV = ABE_TOP;
-        }
-        else
-        {
-            DeltaPt.cy = rcScreen.bottom - pt.y;
-            PosV = ABE_BOTTOM;
-        }
-
-        Pos = (DeltaPt.cx * rcScreen.bottom < DeltaPt.cy * rcScreen.right) ? PosH : PosV;
-
-        /* Fix the screen origin to be relative to the primary monitor again */
-        OffsetRect(&rcScreen,
-                   ScreenOffset.cx,
-                   ScreenOffset.cy);
-
-        RECT rcPos = m_TrayRects[Pos];
-
-        hMonNew = GetMonitorFromRect(&rcPos);
-        if (hMon != hMonNew)
-        {
-            SIZE szTray;
+        CloseHandle(CreateThread(NULL, 0, s_TrayPropertiesThread, this, 0, NULL));
+        return NULL;
+    }
 
-            /* Recalculate the rectangle, we're dragging to another monitor.
-               We don't need to recalculate the rect on single monitor systems. */
-            szTray.cx = rcPos.right - rcPos.left;
-            szTray.cy = rcPos.bottom - rcPos.top;
+    VOID OpenCommonStartMenuDirectory(IN HWND hWndOwner, IN LPCTSTR lpOperation)
+    {
+        WCHAR szDir[MAX_PATH];
 
-            GetTrayRectFromScreenRect(Pos, &rcScreen, &szTray, pRect);
-            if (AutoHide)
-            {
-                pRect->left += m_AutoHideOffset.cx;
-                pRect->right += m_AutoHideOffset.cx;
-                pRect->top += m_AutoHideOffset.cy;
-                pRect->bottom += m_AutoHideOffset.cy;
-            }
-            hMon = hMonNew;
-        }
-        else
+        if (SHGetSpecialFolderPath(hWndOwner,
+            szDir,
+            CSIDL_COMMON_STARTMENU,
+            FALSE))
         {
-            /* The user is dragging the tray window on the same monitor. We don't need
-               to recalculate the rectangle */
-            *pRect = rcPos;
-            if (AutoHide)
-            {
-                pRect->left += m_AutoHideOffset.cx;
-                pRect->right += m_AutoHideOffset.cx;
-                pRect->top += m_AutoHideOffset.cy;
-                pRect->bottom += m_AutoHideOffset.cy;
-            }
+            ShellExecute(hWndOwner,
+                         lpOperation,
+                         szDir,
+                         NULL,
+                         NULL,
+                         SW_SHOWNORMAL);
         }
-
-        *phMonitor = hMon;
-
-        return Pos;
     }
 
-    DWORD GetDraggingRectFromRect(
-        IN OUT RECT *pRect,
-        OUT HMONITOR *phMonitor)
+    VOID OpenTaskManager(IN HWND hWndOwner)
     {
-        POINT pt;
-
-        /* Calculate the center of the rectangle. We call
-           GetDraggingRectFromPt to calculate a valid
-           dragging rectangle */
-        pt.x = pRect->left + ((pRect->right - pRect->left) / 2);
-        pt.y = pRect->top + ((pRect->bottom - pRect->top) / 2);
-
-        return GetDraggingRectFromPt(
-            pt,
-            pRect,
-            phMonitor);
+        ShellExecute(hWndOwner,
+                     TEXT("open"),
+                     TEXT("taskmgr.exe"),
+                     NULL,
+                     NULL,
+                     SW_SHOWNORMAL);
     }
 
-    VOID ChangingWinPos(IN OUT LPWINDOWPOS pwp)
+    BOOL STDMETHODCALLTYPE ExecContextMenuCmd(IN UINT uiCmd)
     {
-        RECT rcTray;
-
-        if (IsDragging)
-        {
-            rcTray.left = pwp->x;
-            rcTray.top = pwp->y;
-            rcTray.right = rcTray.left + pwp->cx;
-            rcTray.bottom = rcTray.top + pwp->cy;
-            if (AutoHide)
-            {
-                rcTray.left -= m_AutoHideOffset.cx;
-                rcTray.right -= m_AutoHideOffset.cx;
-                rcTray.top -= m_AutoHideOffset.cy;
-                rcTray.bottom -= m_AutoHideOffset.cy;
-            }
-
-            if (!EqualRect(&rcTray,
-                &m_TrayRects[m_DraggingPosition]))
-            {
-                /* Recalculate the rectangle, the user dragged the tray
-                   window to another monitor or the window was somehow else
-                   moved or resized */
-                m_DraggingPosition = GetDraggingRectFromRect(
-                    &rcTray,
-                    &m_DraggingMonitor);
-                //m_TrayRects[DraggingPosition] = rcTray;
-            }
-
-            //Monitor = CalculateValidSize(
-            //                                                   DraggingPosition,
-            //                                                   &rcTray);
-
-            m_Monitor = m_DraggingMonitor;
-            m_Position = m_DraggingPosition;
-            IsDragging = FALSE;
-
-            m_TrayRects[m_Position] = rcTray;
-            goto ChangePos;
-        }
-        else if (GetWindowRect(m_hWnd, &rcTray))
+        switch (uiCmd)
         {
-            if (InSizeMove)
-            {
-                if (!(pwp->flags & SWP_NOMOVE))
-                {
-                    rcTray.left = pwp->x;
-                    rcTray.top = pwp->y;
-                }
-
-                if (!(pwp->flags & SWP_NOSIZE))
-                {
-                    rcTray.right = rcTray.left + pwp->cx;
-                    rcTray.bottom = rcTray.top + pwp->cy;
-                }
-
-                m_Position = GetDraggingRectFromRect(&rcTray, &m_Monitor);
-
-                if (!(pwp->flags & (SWP_NOMOVE | SWP_NOSIZE)))
-                {
-                    SIZE szWnd;
-
-                    szWnd.cx = pwp->cx;
-                    szWnd.cy = pwp->cy;
-
-                    MakeTrayRectWithSize(m_Position, &szWnd, &rcTray);
-                }
-
-                if (AutoHide)
-                {
-                    rcTray.left -= m_AutoHideOffset.cx;
-                    rcTray.right -= m_AutoHideOffset.cx;
-                    rcTray.top -= m_AutoHideOffset.cy;
-                    rcTray.bottom -= m_AutoHideOffset.cy;
-                }
-                m_TrayRects[m_Position] = rcTray;
-            }
-            else
-            {
-                /* If the user isn't resizing the tray window we need to make sure the
-                   new size or position is valid. this is to prevent changes to the window
-                   without user interaction. */
-                rcTray = m_TrayRects[m_Position];
-            }
-
-ChangePos:
-            m_TraySize.cx = rcTray.right - rcTray.left;
-            m_TraySize.cy = rcTray.bottom - rcTray.top;
-
-            if (AutoHide)
-            {
-                rcTray.left += m_AutoHideOffset.cx;
-                rcTray.right += m_AutoHideOffset.cx;
-                rcTray.top += m_AutoHideOffset.cy;
-                rcTray.bottom += m_AutoHideOffset.cy;
-            }
-
-            pwp->flags &= ~(SWP_NOMOVE | SWP_NOSIZE);
-            pwp->x = rcTray.left;
-            pwp->y = rcTray.top;
-            pwp->cx = m_TraySize.cx;
-            pwp->cy = m_TraySize.cy;
-        }
-    }
+        case ID_SHELL_CMD_PROPERTIES:
+            DisplayProperties();
+            break;
 
-    VOID ApplyClipping(IN BOOL Clip)
-    {
-        RECT rcClip, rcWindow;
-        HRGN hClipRgn;
+        case ID_SHELL_CMD_OPEN_ALL_USERS:
+            OpenCommonStartMenuDirectory(m_hWnd,
+                                         TEXT("open"));
+            break;
 
-        if (GetWindowRect(m_hWnd, &rcWindow))
-        {
-            /* Disable clipping on systems with only one monitor */
-            if (GetSystemMetrics(SM_CMONITORS) <= 1)
-                Clip = FALSE;
+        case ID_SHELL_CMD_EXPLORE_ALL_USERS:
+            OpenCommonStartMenuDirectory(m_hWnd,
+                                         TEXT("explore"));
+            break;
 
-            if (Clip)
+        case ID_LOCKTASKBAR:
+            if (SHRestricted(REST_CLASSICSHELL) == 0)
             {
-                rcClip = rcWindow;
-
-                GetScreenRect(m_Monitor, &rcClip);
-
-                if (!IntersectRect(&rcClip, &rcClip, &rcWindow))
-                {
-                    rcClip = rcWindow;
-                }
-
-                OffsetRect(&rcClip,
-                           -rcWindow.left,
-                           -rcWindow.top);
-
-                hClipRgn = CreateRectRgnIndirect(&rcClip);
+                Lock(!Locked);
             }
-            else
-                hClipRgn = NULL;
-
-            /* Set the clipping region or make sure the window isn't clipped
-               by disabling it explicitly. */
-            SetWindowRgn(m_hWnd, hClipRgn, TRUE);
-        }
-    }
+            break;
 
-    VOID ResizeWorkArea()
-    {
-#if !WIN7_COMPAT_MODE
-        RECT rcTray, rcWorkArea;
+        case ID_SHELL_CMD_OPEN_TASKMGR:
+            OpenTaskManager(m_hWnd);
+            break;
 
-        /* If monitor has changed then fix the previous monitors work area */
-        if (m_PreviousMonitor != m_Monitor)
-        {
-            GetScreenRect(m_PreviousMonitor, &rcWorkArea);
-            SystemParametersInfo(SPI_SETWORKAREA,
-                                 1,
-                                 &rcWorkArea,
-                                 SPIF_SENDCHANGE);
-        }
+        case ID_SHELL_CMD_UNDO_ACTION:
+            break;
 
-        rcTray = m_TrayRects[m_Position];
+        case ID_SHELL_CMD_SHOW_DESKTOP:
+            break;
 
-        GetScreenRect(m_Monitor, &rcWorkArea);
-        m_PreviousMonitor = m_Monitor;
+        case ID_SHELL_CMD_TILE_WND_H:
+            TileWindows(NULL, MDITILE_HORIZONTAL, NULL, 0, NULL);
+            break;
 
-        /* If AutoHide is false then change the workarea to exclude the area that
-           the taskbar covers. */
-        if (!AutoHide)
-        {
-            switch (m_Position)
-            {
-            case ABE_TOP:
-                rcWorkArea.top = rcTray.bottom;
-                break;
-            case ABE_LEFT:
-                rcWorkArea.left = rcTray.right;
-                break;
-            case ABE_RIGHT:
-                rcWorkArea.right = rcTray.left;
-                break;
-            case ABE_BOTTOM:
-                rcWorkArea.bottom = rcTray.top;
-                break;
-            }
-        }
+        case ID_SHELL_CMD_TILE_WND_V:
+            TileWindows(NULL, MDITILE_VERTICAL, NULL, 0, NULL);
+            break;
 
-        SystemParametersInfo(SPI_SETWORKAREA,
-                             1,
-                             &rcWorkArea,
-                             SPIF_SENDCHANGE);
-#endif
-    }
+        case ID_SHELL_CMD_CASCADE_WND:
+            CascadeWindows(NULL, MDITILE_SKIPDISABLED, NULL, 0, NULL);
+            break;
 
-    VOID CheckTrayWndPosition()
-    {
-        RECT rcTray;
+        case ID_SHELL_CMD_CUST_NOTIF:
+            ShowCustomizeNotifyIcons(hExplorerInstance, m_hWnd);
+            break;
 
-        rcTray = m_TrayRects[m_Position];
+        case ID_SHELL_CMD_ADJUST_DAT:
+            //FIXME: Use SHRunControlPanel
+            ShellExecuteW(m_hWnd, NULL, L"timedate.cpl", NULL, NULL, SW_NORMAL);
+            break;
 
-        if (AutoHide)
-        {
-            rcTray.left += m_AutoHideOffset.cx;
-            rcTray.right += m_AutoHideOffset.cx;
-            rcTray.top += m_AutoHideOffset.cy;
-            rcTray.bottom += m_AutoHideOffset.cy;
+        default:
+            TRACE("ITrayWindow::ExecContextMenuCmd(%u): Unhandled Command ID!\n", uiCmd);
+            return FALSE;
         }
 
-        /* Move the tray window */
-        SetWindowPos(NULL,
-                     rcTray.left,
-                     rcTray.top,
-                     rcTray.right - rcTray.left,
-                     rcTray.bottom - rcTray.top,
-                     SWP_NOZORDER);
-
-        ResizeWorkArea();
-
-        ApplyClipping(TRUE);
+        return TRUE;
     }
 
-    typedef struct _TW_STUCKRECTS2
-    {
-        DWORD cbSize;
-        LONG Unknown;
-        DWORD dwFlags;
-        DWORD Position;
-        SIZE Size;
-        RECT Rect;
-    } TW_STRUCKRECTS2, *PTW_STUCKRECTS2;
-
-    VOID RegLoadSettings()
+    LRESULT HandleHotKey(DWORD id)
     {
-        DWORD Pos;
-        TW_STRUCKRECTS2 sr;
-        RECT rcScreen;
-        SIZE WndSize, EdgeSize, DlgFrameSize;
-        DWORD cbSize = sizeof(sr);
-        SIZE StartBtnSize = m_StartButton.GetSize();
-
-        EdgeSize.cx = GetSystemMetrics(SM_CXEDGE);
-        EdgeSize.cy = GetSystemMetrics(SM_CYEDGE);
-        DlgFrameSize.cx = GetSystemMetrics(SM_CXDLGFRAME);
-        DlgFrameSize.cy = GetSystemMetrics(SM_CYDLGFRAME);
-
-        if (SHGetValue(hkExplorer,
-            TEXT("StuckRects2"),
-            TEXT("Settings"),
-            NULL,
-            &sr,
-            &cbSize) == ERROR_SUCCESS &&
-            sr.cbSize == sizeof(sr))
+        switch (id)
         {
-            AutoHide = (sr.dwFlags & ABS_AUTOHIDE) != 0;
-            AlwaysOnTop = (sr.dwFlags & ABS_ALWAYSONTOP) != 0;
-            SmSmallIcons = (sr.dwFlags & 0x4) != 0;
-            HideClock = (sr.dwFlags & 0x8) != 0;
-
-            /* FIXME: Are there more flags? */
+        case IDHK_RUN:
+            DisplayRunFileDlg();
+            break;
+        case IDHK_HELP:
+            ExecResourceCmd(IDS_HELP_COMMAND);
+            break;
+        case IDHK_EXPLORE:
+            //FIXME: We don't support this yet:
+            //ShellExecuteW(0, L"explore", NULL, NULL, NULL, 1);
+            ShellExecuteW(0, NULL, L"explorer.exe", NULL, NULL, 1); 
+            break;
+        case IDHK_FIND:
+            SHFindFiles(NULL, NULL);
+            break;
+        case IDHK_FIND_COMPUTER:
+            SHFindComputer(NULL, NULL);
+            break;
+        case IDHK_SYS_PROPERTIES:
+            //FIXME: Use SHRunControlPanel
+            ShellExecuteW(m_hWnd, NULL, L"sysdm.cpl", NULL, NULL, SW_NORMAL);
+            break;
+        case IDHK_NEXT_TASK:
+            break;
+        case IDHK_PREV_TASK:
+            break;
+        case IDHK_MINIMIZE_ALL:
+            break;
+        case IDHK_RESTORE_ALL:
+            break;
+        case IDHK_DESKTOP:
+            break;
+        case IDHK_PAGER:
+            break;
+        }
 
-#if WIN7_COMPAT_MODE
-            m_Position = ABE_LEFT;
-#else
-            if (sr.Position > ABE_BOTTOM)
-                m_Position = ABE_BOTTOM;
-            else
-                m_Position = sr.Position;
-#endif
+        return 0;
+    }
 
-            /* Try to find out which monitor the tray window was located on last.
-               Here we're only interested in the monitor screen that we think
-               is the last one used. We're going to determine on which monitor
-               we really are after calculating the docked position. */
-            rcScreen = sr.Rect;
-            GetScreenRectFromRect(
-                &rcScreen,
-                MONITOR_DEFAULTTONEAREST);
-        }
-        else
+    LRESULT HandleCommand(UINT uCommand)
+    {
+        switch (uCommand)
         {
-            m_Position = ABE_BOTTOM;
-            AlwaysOnTop = TRUE;
-
-            /* Use the minimum size of the taskbar, we'll use the start
-               button as a minimum for now. Make sure we calculate the
-               entire window size, not just the client size. However, we
-               use a thinner border than a standard thick border, so that
-               the start button and bands are not stuck to the screen border. */
-            sr.Size.cx = StartBtnSize.cx + (2 * (EdgeSize.cx + DlgFrameSize.cx));
-            sr.Size.cy = StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
+        case IDM_TASKBARANDSTARTMENU:
+            DisplayProperties();
+            break;
 
-            /* Use the primary screen by default */
-            rcScreen.left = 0;
-            rcScreen.top = 0;
-            rcScreen.right = GetSystemMetrics(SM_CXSCREEN);
-            rcScreen.bottom = GetSystemMetrics(SM_CYSCREEN);
-            GetScreenRectFromRect(
-                &rcScreen,
-                MONITOR_DEFAULTTOPRIMARY);
-        }
+        case IDM_SEARCH:
+            SHFindFiles(NULL, NULL);
+            break;
 
-        if (m_hWnd != NULL)
-            SetWindowPos(
-            AlwaysOnTop ? HWND_TOPMOST : HWND_NOTOPMOST,
-            0,
-            0,
-            0,
-            0,
-            SWP_NOMOVE | SWP_NOSIZE);
+        case IDM_HELPANDSUPPORT:
+            ExecResourceCmd(IDS_HELP_COMMAND);
+            break;
 
-        /* Determine a minimum tray window rectangle. The "client" height is
-           zero here since we cannot determine an optimal minimum width when
-           loaded as a vertical tray window. We just need to make sure the values
-           loaded from the registry are at least. The windows explorer behaves
-           the same way, it allows the user to save a zero width vertical tray
-           window, but not a zero height horizontal tray window. */
-        WndSize.cx = 2 * (EdgeSize.cx + DlgFrameSize.cx);
-        WndSize.cy = StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
+        case IDM_RUN:
+            DisplayRunFileDlg();
+            break;
 
-        if (WndSize.cx < sr.Size.cx)
-            WndSize.cx = sr.Size.cx;
-        if (WndSize.cy < sr.Size.cy)
-            WndSize.cy = sr.Size.cy;
+        /* FIXME: Handle these commands as well */
+        case IDM_SYNCHRONIZE:
+        case IDM_DISCONNECT:
+        case IDM_UNDOCKCOMPUTER:
+            break;
 
-        /* Save the calculated size */
-        m_TraySize = WndSize;
+        case IDM_LOGOFF:
+            LogoffWindowsDialog(m_hWnd); // FIXME: Maybe handle it in a similar way as DoExitWindows?
+            break;
 
-        /* Calculate all docking rectangles. We need to do this here so they're
-           initialized and dragging the tray window to another position gives
-           usable results */
-        for (Pos = ABE_LEFT;
-             Pos <= ABE_BOTTOM;
-             Pos++)
-        {
-            GetTrayRectFromScreenRect(
-                Pos,
-                &rcScreen,
-                &m_TraySize,
-                &m_TrayRects[Pos]);
-            //        TRACE("m_TrayRects[%d(%d)]: %d,%d,%d,%d\n", Pos, Position, m_TrayRects[Pos].left, m_TrayRects[Pos].top, m_TrayRects[Pos].right, m_TrayRects[Pos].bottom);
+        case IDM_SHUTDOWN:
+            DoExitWindows();
+            break;
         }
 
-        /* Determine which monitor we are on. It shouldn't matter which docked
-           position rectangle we use */
-        m_Monitor = GetMonitorFromRect(&m_TrayRects[ABE_LEFT]);
+        return FALSE;
     }
 
+
     UINT TrackMenu(
         IN HMENU hMenu,
         IN POINT *ppt OPTIONAL,
@@ -1202,7 +625,7 @@ ChangePos:
             /* Get the client rectangle and map it to screen coordinates */
             if (::GetClientRect(hwndExclude,
                 &tmp.rcExclude) &&
-                MapWindowPoints(hwndExclude,
+                ::MapWindowPoints(hwndExclude,
                 NULL,
                 (LPPOINT) &tmp.rcExclude,
                 2) != 0)
@@ -1215,7 +638,7 @@ ChangePos:
         {
             if (ptmp == NULL &&
                 GetClientRect(&tmp.rcExclude) &&
-                MapWindowPoints(m_hWnd,
+                MapWindowPoints(
                 NULL,
                 (LPPOINT) &tmp.rcExclude,
                 2) != 0)
@@ -1280,7 +703,7 @@ ChangePos:
             DestroyMenu(popup);
             return hr;
         }
-        
+
         TRACE("Before Tracking\n");
         uCommand = ::TrackPopupMenuEx(popup, TPM_RETURNCMD, x, y, m_hWnd, NULL);
 
@@ -1303,672 +726,885 @@ ChangePos:
         return hr;
     }
 
-    VOID AlignControls(IN PRECT prcClient OPTIONAL)
-    {
-        RECT rcClient;
-        SIZE TraySize, StartSize;
-        POINT ptTrayNotify = { 0, 0 };
-        BOOL Horizontal;
-        HDWP dwp;
-
-        m_StartButton.UpdateSize();
-        if (prcClient != NULL)
-        {
-            rcClient = *prcClient;
-        }
-        else
-        {
-            if (!GetClientRect(&rcClient))
-            {
-                ERR("Could not get client rect lastErr=%d\n", GetLastError());
-                return;
-            }
-        }
-
-        Horizontal = IsPosHorizontal();
-
-        /* We're about to resize/move the start button, the rebar control and
-           the tray notification control */
-        dwp = BeginDeferWindowPos(3);
-        if (dwp == NULL)
-        {
-            ERR("BeginDeferWindowPos failed. lastErr=%d\n", GetLastError());
-            return;
-        }
-
-        /* Limit the Start button width to the client width, if neccessary */
-        StartSize = m_StartButton.GetSize();
-        if (StartSize.cx > rcClient.right)
-            StartSize.cx = rcClient.right;
 
-        if (m_StartButton.m_hWnd != NULL)
-        {
-            /* Resize and reposition the button */
-            dwp = DeferWindowPos(dwp,
-                                 m_StartButton.m_hWnd,
-                                 NULL,
-                                 0,
-                                 0,
-                                 StartSize.cx,
-                                 StartSize.cy,
-                                 SWP_NOZORDER | SWP_NOACTIVATE);
-            if (dwp == NULL)
-            {
-                ERR("DeferWindowPos for start button failed. lastErr=%d\n", GetLastError());
-                return;
-            }
-        }
 
-        /* Determine the size that the tray notification window needs */
-        if (Horizontal)
-        {
-            TraySize.cx = 0;
-            TraySize.cy = rcClient.bottom;
-        }
-        else
-        {
-            TraySize.cx = rcClient.right;
-            TraySize.cy = 0;
-        }
 
-        if (m_TrayNotify != NULL &&
-            SendMessage(m_TrayNotify,
-            TNWM_GETMINIMUMSIZE,
-            (WPARAM) Horizontal,
-            (LPARAM) &TraySize))
-        {
-            /* Move the tray notification window to the desired location */
-            if (Horizontal)
-                ptTrayNotify.x = rcClient.right - TraySize.cx;
-            else
-                ptTrayNotify.y = rcClient.bottom - TraySize.cy;
 
-            dwp = DeferWindowPos(dwp,
-                                 m_TrayNotify,
-                                 NULL,
-                                 ptTrayNotify.x,
-                                 ptTrayNotify.y,
-                                 TraySize.cx,
-                                 TraySize.cy,
-                                 SWP_NOZORDER | SWP_NOACTIVATE);
-            if (dwp == NULL)
-            {
-                ERR("DeferWindowPos for notification area failed. lastErr=%d\n", GetLastError());
-                return;
-            }
-        }
+    /**********************************************************
+     *    ##### moving and sizing handling #####
+     */
 
-        /* Resize/Move the rebar control */
-        if (m_Rebar != NULL)
+    BOOL UpdateNonClientMetrics()
+    {
+        NONCLIENTMETRICS ncm;
+        ncm.cbSize = sizeof(ncm);
+        if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, 0))
         {
-            POINT ptRebar = { 0, 0 };
-            SIZE szRebar;
-
-            SetWindowStyle(m_Rebar, CCS_VERT, Horizontal ? 0 : CCS_VERT);
-
-            if (Horizontal)
-            {
-                ptRebar.x = StartSize.cx + GetSystemMetrics(SM_CXSIZEFRAME);
-                szRebar.cx = ptTrayNotify.x - ptRebar.x;
-                szRebar.cy = rcClient.bottom;
-            }
-            else
-            {
-                ptRebar.y = StartSize.cy + GetSystemMetrics(SM_CYSIZEFRAME);
-                szRebar.cx = rcClient.right;
-                szRebar.cy = ptTrayNotify.y - ptRebar.y;
-            }
+            if (m_Font != NULL)
+                DeleteObject(m_Font);
 
-            dwp = DeferWindowPos(dwp,
-                                 m_Rebar,
-                                 NULL,
-                                 ptRebar.x,
-                                 ptRebar.y,
-                                 szRebar.cx,
-                                 szRebar.cy,
-                                 SWP_NOZORDER | SWP_NOACTIVATE);
+            m_Font = CreateFontIndirect(&ncm.lfMessageFont);
+            return TRUE;
         }
 
-        if (dwp != NULL)
-            EndDeferWindowPos(dwp);
+        return FALSE;
+    }
 
-        if (m_TaskSwitch != NULL)
+    VOID SetWindowsFont()
+    {
+        if (m_TrayNotify != NULL)
         {
-            /* Update the task switch window configuration */
-            SendMessage(m_TaskSwitch, TSWM_UPDATETASKBARPOS, 0, 0);
+            SendMessage(m_TrayNotify, WM_SETFONT, (WPARAM) m_Font, TRUE);
         }
     }
 
-    LRESULT OnThemeChanged()
+    HMONITOR GetScreenRectFromRect(
+        IN OUT RECT *pRect,
+        IN DWORD dwFlags)
     {
-        if (m_Theme)
-            CloseThemeData(m_Theme);
-
-        if (IsThemeActive())
-            m_Theme = OpenThemeData(m_hWnd, L"TaskBar");
-        else
-            m_Theme = NULL;
+        MONITORINFO mi;
+        HMONITOR hMon;
 
-        if (Locked && m_Theme)
+        mi.cbSize = sizeof(mi);
+        hMon = MonitorFromRect(pRect, dwFlags);
+        if (hMon != NULL &&
+            GetMonitorInfo(hMon, &mi))
         {
-            SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, 0);
+            *pRect = mi.rcMonitor;
         }
         else
         {
-            SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, WS_THICKFRAME | WS_BORDER);
+            pRect->left = 0;
+            pRect->top = 0;
+            pRect->right = GetSystemMetrics(SM_CXSCREEN);
+            pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
+
+            hMon = NULL;
         }
-        SetWindowPos(NULL, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER);
 
-        return TRUE;
+        return hMon;
     }
 
-    LRESULT OnThemeChanged(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    HMONITOR GetMonitorFromRect(
+        IN const RECT *pRect)
     {
-        return OnThemeChanged();
-    }
+        HMONITOR hMon;
 
-    LRESULT OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
-    {
-        WCHAR szStartCaption[32];
+        /* In case the monitor sizes or saved sizes differ a bit (probably
+           not a lot, only so the tray window overlaps into another monitor
+           now), minimize the risk that we determine a wrong monitor by
+           using the center point of the tray window if we can't determine
+           it using the rectangle. */
+        hMon = MonitorFromRect(pRect, MONITOR_DEFAULTTONULL);
+        if (hMon == NULL)
+        {
+            POINT pt;
 
-        ((ITrayWindow*)this)->AddRef();
+            pt.x = pRect->left + ((pRect->right - pRect->left) / 2);
+            pt.y = pRect->top + ((pRect->bottom - pRect->top) / 2);
 
-        SetWindowTheme(m_hWnd, L"TaskBar", NULL);
-        OnThemeChanged();
+            /* be less error-prone, find the nearest monitor */
+            hMon = MonitorFromPoint(pt, MONITOR_DEFAULTTONEAREST);
+        }
 
-        InterlockedIncrement(&TrayWndCount);
+        return hMon;
+    }
 
-        if (!LoadString(hExplorerInstance,
-            IDS_START,
-            szStartCaption,
-            sizeof(szStartCaption) / sizeof(szStartCaption[0])))
-        {
-            szStartCaption[0] = TEXT('\0');
-        }
+    HMONITOR GetScreenRect(
+        IN HMONITOR hMonitor,
+        IN OUT RECT *pRect)
+    {
+        HMONITOR hMon = NULL;
 
-        if (m_CaptionFont == NULL)
+        if (hMonitor != NULL)
         {
-            NONCLIENTMETRICS ncm;
+            MONITORINFO mi;
 
-            /* Get the system fonts, we use the caption font,
-               always bold, though. */
-            ncm.cbSize = sizeof(ncm);
-            if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE))
+            mi.cbSize = sizeof(mi);
+            if (!GetMonitorInfo(hMonitor, &mi))
             {
-                if (m_CaptionFont == NULL)
+                /* Hm, the monitor is gone? Try to find a monitor where it
+                   could be located now */
+                hMon = GetMonitorFromRect(pRect);
+                if (hMon == NULL ||
+                    !GetMonitorInfo(hMon, &mi))
                 {
-                    ncm.lfCaptionFont.lfWeight = FW_NORMAL;
-                    m_CaptionFont = CreateFontIndirect(&ncm.lfCaptionFont);
+                    hMon = NULL;
+                    goto GetPrimaryRect;
                 }
             }
-        }
-        
-        /* Create the Start button */
-        m_StartButton.SubclassWindow(CreateWindowEx(
-            0,
-            WC_BUTTON,
-            szStartCaption,
-            WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS |
-            BS_PUSHBUTTON | BS_CENTER | BS_VCENTER | BS_BITMAP,
-            0,
-            0,
-            0,
-            0,
-            m_hWnd,
-            (HMENU) IDC_STARTBTN,
-            hExplorerInstance,
-            NULL));
 
-        if (m_StartButton.m_hWnd)
+            *pRect = mi.rcMonitor;
+        }
+        else
         {
-            m_StartButton.Initialize();
+GetPrimaryRect:
+            pRect->left = 0;
+            pRect->top = 0;
+            pRect->right = GetSystemMetrics(SM_CXSCREEN);
+            pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
         }
 
-        /* Load the saved tray window settings */
-        RegLoadSettings();
-
-        /* Create and initialize the start menu */
-        HBITMAP hbmBanner = LoadBitmap(hExplorerInstance, MAKEINTRESOURCE(IDB_STARTMENU));
-        m_StartMenuPopup = CreateStartMenu(this, &m_StartMenuBand, hbmBanner, 0);
+        return hMon;
+    }
 
-        /* Load the tray band site */
-        if (m_TrayBandSite != NULL)
+    VOID AdjustSizerRect(RECT *rc, DWORD pos)
+    {
+        switch (pos)
         {
-            m_TrayBandSite.Release();
+            case ABE_TOP:
+                rc->bottom -= GetSystemMetrics(SM_CXSIZEFRAME);
+                break;
+            case ABE_BOTTOM:
+                rc->top += GetSystemMetrics(SM_CXSIZEFRAME);
+                break;
+            case ABE_LEFT:
+                rc->right -= GetSystemMetrics(SM_CYSIZEFRAME);
+                break;
+            case ABE_RIGHT:
+                rc->left += GetSystemMetrics(SM_CYSIZEFRAME);
+                break;
         }
 
-        m_TrayBandSite = CreateTrayBandSite(this, &m_Rebar, &m_TaskSwitch);
-        SetWindowTheme(m_Rebar, L"TaskBar", NULL);
-
-        /* Create the tray notification window */
-        m_TrayNotify = CreateTrayNotifyWnd(this, HideClock, &m_TrayNotifyInstance);
+    }
 
-        if (UpdateNonClientMetrics())
+    VOID MakeTrayRectWithSize(IN DWORD Position,
+                              IN const SIZE *pTraySize,
+                              IN OUT RECT *pRect)
+    {
+        switch (Position)
         {
-            SetWindowsFont();
-        }
+        case ABE_LEFT:
+            pRect->right = pRect->left + pTraySize->cx;
+            break;
 
-        /* Move the tray window to the right position and resize it if neccessary */
-        CheckTrayWndPosition();
+        case ABE_TOP:
+            pRect->bottom = pRect->top + pTraySize->cy;
+            break;
 
-        /* Align all controls on the tray window */
-        AlignControls(NULL);
+        case ABE_RIGHT:
+            pRect->left = pRect->right - pTraySize->cx;
+            break;
+
+        case ABE_BOTTOM:
+        default:
+            pRect->top = pRect->bottom - pTraySize->cy;
+            break;
+        }
+    }
+
+    VOID GetTrayRectFromScreenRect(IN DWORD Position,
+                                   IN const RECT *pScreen,
+                                   IN const SIZE *pTraySize OPTIONAL,
+                                   OUT RECT *pRect)
+    {
+        if (pTraySize == NULL)
+            pTraySize = &m_TraySize;
 
-        InitShellServices(&(m_ShellServices));
+        *pRect = *pScreen;
 
-        if (AutoHide)
+        if(!IsThemeActive())
         {
-            m_AutoHideState = AUTOHIDE_HIDING;
-            SetTimer(TIMER_ID_AUTOHIDE, AUTOHIDE_DELAY_HIDE, NULL);
+            /* Move the border outside of the screen */
+            InflateRect(pRect,
+                        GetSystemMetrics(SM_CXEDGE),
+                        GetSystemMetrics(SM_CYEDGE));
         }
 
-        RegisterHotKey(m_hWnd, IDHK_RUN, MOD_WIN, 'R');
-        RegisterHotKey(m_hWnd, IDHK_MINIMIZE_ALL, MOD_WIN, 'M');
-        RegisterHotKey(m_hWnd, IDHK_RESTORE_ALL, MOD_WIN|MOD_SHIFT, 'M');
-        RegisterHotKey(m_hWnd, IDHK_HELP, MOD_WIN, VK_F1);
-        RegisterHotKey(m_hWnd, IDHK_EXPLORE, MOD_WIN, 'E');
-        RegisterHotKey(m_hWnd, IDHK_FIND, MOD_WIN, 'F');
-        RegisterHotKey(m_hWnd, IDHK_FIND_COMPUTER, MOD_WIN|MOD_CONTROL, 'F');
-        RegisterHotKey(m_hWnd, IDHK_NEXT_TASK, MOD_WIN, VK_TAB);
-        RegisterHotKey(m_hWnd, IDHK_PREV_TASK, MOD_WIN|MOD_SHIFT, VK_TAB);
-        RegisterHotKey(m_hWnd, IDHK_SYS_PROPERTIES, MOD_WIN, VK_PAUSE);
-        RegisterHotKey(m_hWnd, IDHK_DESKTOP, MOD_WIN, 'D');
-        RegisterHotKey(m_hWnd, IDHK_PAGER, MOD_WIN, 'B');
-
-        return TRUE;
+        MakeTrayRectWithSize(Position, pTraySize, pRect);
     }
 
-    HRESULT STDMETHODCALLTYPE Open()
+    BOOL IsPosHorizontal()
     {
-        RECT rcWnd;
+        return m_Position == ABE_TOP || m_Position == ABE_BOTTOM;
+    }
 
-        /* Check if there's already a window created and try to show it.
-           If it was somehow destroyed just create a new tray window. */
-        if (m_hWnd != NULL && IsWindow())
-        {
-            if (!IsWindowVisible(m_hWnd))
-            {
-                CheckTrayWndPosition();
+    HMONITOR CalculateValidSize(
+        IN DWORD Position,
+        IN OUT RECT *pRect)
+    {
+        RECT rcScreen;
+        //BOOL Horizontal;
+        HMONITOR hMon;
+        SIZE szMax, szWnd;
 
-                ShowWindow(SW_SHOW);
-            }
+        //Horizontal = IsPosHorizontal();
 
-            return S_OK;
-        }
+        szWnd.cx = pRect->right - pRect->left;
+        szWnd.cy = pRect->bottom - pRect->top;
 
-        DWORD dwExStyle = WS_EX_TOOLWINDOW | WS_EX_WINDOWEDGE;
-        if (AlwaysOnTop)
-            dwExStyle |= WS_EX_TOPMOST;
+        rcScreen = *pRect;
+        hMon = GetScreenRectFromRect(
+            &rcScreen,
+            MONITOR_DEFAULTTONEAREST);
 
-        DWORD dwStyle = WS_POPUP | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN |
-            WS_BORDER | WS_THICKFRAME;
+        /* Calculate the maximum size of the tray window and limit the window
+           size to half of the screen's size. */
+        szMax.cx = (rcScreen.right - rcScreen.left) / 2;
+        szMax.cy = (rcScreen.bottom - rcScreen.top) / 2;
+        if (szWnd.cx > szMax.cx)
+            szWnd.cx = szMax.cx;
+        if (szWnd.cy > szMax.cy)
+            szWnd.cy = szMax.cy;
 
-        ZeroMemory(&rcWnd, sizeof(rcWnd));
-        if (m_Position != (DWORD) -1)
-            rcWnd = m_TrayRects[m_Position];
+        /* FIXME - calculate */
 
-        if (!Create(NULL, rcWnd, NULL, dwStyle, dwExStyle))
-            return E_FAIL;
+        GetTrayRectFromScreenRect(Position,
+                                  &rcScreen,
+                                  &szWnd,
+                                  pRect);
 
-        return S_OK;
+        return hMon;
     }
 
-    HRESULT STDMETHODCALLTYPE Close()
+#if 0
+    VOID
+        GetMinimumWindowSize(
+        OUT RECT *pRect)
     {
-        if (m_hWnd != NULL)
-        {
-            SendMessage(m_hWnd,
-                        WM_APP_TRAYDESTROY,
-                        0,
-                        0);
-        }
+        RECT rcMin = {0};
 
-        return S_OK;
-    }
+        AdjustWindowRectEx(&rcMin,
+                           GetWindowLong(m_hWnd,
+                           GWL_STYLE),
+                           FALSE,
+                           GetWindowLong(m_hWnd,
+                           GWL_EXSTYLE));
 
-    HWND STDMETHODCALLTYPE GetHWND()
-    {
-        return m_hWnd;
+        *pRect = rcMin;
     }
+#endif
 
-    BOOL STDMETHODCALLTYPE IsSpecialHWND(IN HWND hWnd)
-    {
-        return (m_hWnd == hWnd ||
-                (m_DesktopWnd != NULL && m_hWnd == m_DesktopWnd));
-    }
 
-    BOOL STDMETHODCALLTYPE
-        IsHorizontal()
+    DWORD GetDraggingRectFromPt(
+        IN POINT pt,
+        OUT RECT *pRect,
+        OUT HMONITOR *phMonitor)
     {
-        return IsPosHorizontal();
-    }
+        HMONITOR hMon, hMonNew;
+        DWORD PosH, PosV, Pos;
+        SIZE DeltaPt, ScreenOffset;
+        RECT rcScreen;
 
-    HFONT STDMETHODCALLTYPE GetCaptionFonts(OUT HFONT *phBoldCaption OPTIONAL)
-    {
-        if (phBoldCaption != NULL)
-            *phBoldCaption = m_StartButton.GetFont();
+        rcScreen.left = 0;
+        rcScreen.top = 0;
 
-        return m_CaptionFont;
-    }
+        /* Determine the screen rectangle */
+        hMon = MonitorFromPoint(pt, MONITOR_DEFAULTTONULL);
+        if (hMon != NULL)
+        {
+            MONITORINFO mi;
 
-    DWORD WINAPI TrayPropertiesThread()
-    {
-        HWND hwnd;
-        RECT posRect;
+            mi.cbSize = sizeof(mi);
+            if (!GetMonitorInfo(hMon, &mi))
+            {
+                hMon = NULL;
+                goto GetPrimaryScreenRect;
+            }
 
-        GetWindowRect(m_StartButton.m_hWnd, &posRect);
-        hwnd = CreateWindowEx(0,
-                              WC_STATIC,
-                              NULL,
-                              WS_OVERLAPPED | WS_DISABLED | WS_CLIPSIBLINGS | WS_BORDER | SS_LEFT,
-                              posRect.left,
-                              posRect.top,
-                              posRect.right - posRect.left,
-                              posRect.bottom - posRect.top,
-                              NULL,
-                              NULL,
-                              NULL,
-                              NULL);
+            /* make left top corner of the screen zero based to
+               make calculations easier */
+            pt.x -= mi.rcMonitor.left;
+            pt.y -= mi.rcMonitor.top;
 
-        m_TrayPropertiesOwner = hwnd;
+            ScreenOffset.cx = mi.rcMonitor.left;
+            ScreenOffset.cy = mi.rcMonitor.top;
+            rcScreen.right = mi.rcMonitor.right - mi.rcMonitor.left;
+            rcScreen.bottom = mi.rcMonitor.bottom - mi.rcMonitor.top;
+        }
+        else
+        {
+GetPrimaryScreenRect:
+            ScreenOffset.cx = 0;
+            ScreenOffset.cy = 0;
+            rcScreen.right = GetSystemMetrics(SM_CXSCREEN);
+            rcScreen.bottom = GetSystemMetrics(SM_CYSCREEN);
+        }
+
+        /* Calculate the nearest screen border */
+        if (pt.x < rcScreen.right / 2)
+        {
+            DeltaPt.cx = pt.x;
+            PosH = ABE_LEFT;
+        }
+        else
+        {
+            DeltaPt.cx = rcScreen.right - pt.x;
+            PosH = ABE_RIGHT;
+        }
 
-        DisplayTrayProperties(hwnd);
+        if (pt.y < rcScreen.bottom / 2)
+        {
+            DeltaPt.cy = pt.y;
+            PosV = ABE_TOP;
+        }
+        else
+        {
+            DeltaPt.cy = rcScreen.bottom - pt.y;
+            PosV = ABE_BOTTOM;
+        }
 
-        m_TrayPropertiesOwner = NULL;
-        ::DestroyWindow(hwnd);
+        Pos = (DeltaPt.cx * rcScreen.bottom < DeltaPt.cy * rcScreen.right) ? PosH : PosV;
 
-        return 0;
-    }
+        /* Fix the screen origin to be relative to the primary monitor again */
+        OffsetRect(&rcScreen,
+                   ScreenOffset.cx,
+                   ScreenOffset.cy);
 
-    static DWORD WINAPI s_TrayPropertiesThread(IN OUT PVOID pParam)
-    {
-        CTrayWindow *This = (CTrayWindow*) pParam;
+        RECT rcPos = m_TrayRects[Pos];
 
-        return This->TrayPropertiesThread();
-    }
+        hMonNew = GetMonitorFromRect(&rcPos);
+        if (hMon != hMonNew)
+        {
+            SIZE szTray;
 
-    HWND STDMETHODCALLTYPE DisplayProperties()
-    {
-        HWND hTrayProp;
+            /* Recalculate the rectangle, we're dragging to another monitor.
+               We don't need to recalculate the rect on single monitor systems. */
+            szTray.cx = rcPos.right - rcPos.left;
+            szTray.cy = rcPos.bottom - rcPos.top;
 
-        if (m_TrayPropertiesOwner)
+            GetTrayRectFromScreenRect(Pos, &rcScreen, &szTray, pRect);
+            if (AutoHide)
+            {
+                pRect->left += m_AutoHideOffset.cx;
+                pRect->right += m_AutoHideOffset.cx;
+                pRect->top += m_AutoHideOffset.cy;
+                pRect->bottom += m_AutoHideOffset.cy;
+            }
+            hMon = hMonNew;
+        }
+        else
         {
-            hTrayProp = GetLastActivePopup(m_TrayPropertiesOwner);
-            if (hTrayProp != NULL &&
-                hTrayProp != m_TrayPropertiesOwner)
+            /* The user is dragging the tray window on the same monitor. We don't need
+               to recalculate the rectangle */
+            *pRect = rcPos;
+            if (AutoHide)
             {
-                SetForegroundWindow(hTrayProp);
-                return NULL;
+                pRect->left += m_AutoHideOffset.cx;
+                pRect->right += m_AutoHideOffset.cx;
+                pRect->top += m_AutoHideOffset.cy;
+                pRect->bottom += m_AutoHideOffset.cy;
             }
         }
 
-        CloseHandle(CreateThread(NULL, 0, s_TrayPropertiesThread, this, 0, NULL));
-        return NULL;
+        *phMonitor = hMon;
+
+        return Pos;
     }
 
-    VOID OpenCommonStartMenuDirectory(IN HWND hWndOwner, IN LPCTSTR lpOperation)
+    DWORD GetDraggingRectFromRect(
+        IN OUT RECT *pRect,
+        OUT HMONITOR *phMonitor)
     {
-        WCHAR szDir[MAX_PATH];
+        POINT pt;
 
-        if (SHGetSpecialFolderPath(hWndOwner,
-            szDir,
-            CSIDL_COMMON_STARTMENU,
-            FALSE))
-        {
-            ShellExecute(hWndOwner,
-                         lpOperation,
-                         NULL,
-                         NULL,
-                         szDir,
-                         SW_SHOWNORMAL);
-        }
-    }
+        /* Calculate the center of the rectangle. We call
+           GetDraggingRectFromPt to calculate a valid
+           dragging rectangle */
+        pt.x = pRect->left + ((pRect->right - pRect->left) / 2);
+        pt.y = pRect->top + ((pRect->bottom - pRect->top) / 2);
 
-    VOID OpenTaskManager(IN HWND hWndOwner)
-    {
-        ShellExecute(hWndOwner,
-                     TEXT("open"),
-                     TEXT("taskmgr.exe"),
-                     NULL,
-                     NULL,
-                     SW_SHOWNORMAL);
+        return GetDraggingRectFromPt(
+            pt,
+            pRect,
+            phMonitor);
     }
 
-    BOOL STDMETHODCALLTYPE ExecContextMenuCmd(IN UINT uiCmd)
+    VOID ChangingWinPos(IN OUT LPWINDOWPOS pwp)
     {
-        BOOL bHandled = TRUE;
+        RECT rcTray;
 
-        switch (uiCmd)
+        if (IsDragging)
         {
-        case ID_SHELL_CMD_PROPERTIES:
-            DisplayProperties();
-            break;
+            rcTray.left = pwp->x;
+            rcTray.top = pwp->y;
+            rcTray.right = rcTray.left + pwp->cx;
+            rcTray.bottom = rcTray.top + pwp->cy;
+            if (AutoHide)
+            {
+                rcTray.left -= m_AutoHideOffset.cx;
+                rcTray.right -= m_AutoHideOffset.cx;
+                rcTray.top -= m_AutoHideOffset.cy;
+                rcTray.bottom -= m_AutoHideOffset.cy;
+            }
 
-        case ID_SHELL_CMD_OPEN_ALL_USERS:
-            OpenCommonStartMenuDirectory(m_hWnd,
-                                         TEXT("open"));
-            break;
+            if (!EqualRect(&rcTray,
+                &m_TrayRects[m_DraggingPosition]))
+            {
+                /* Recalculate the rectangle, the user dragged the tray
+                   window to another monitor or the window was somehow else
+                   moved or resized */
+                m_DraggingPosition = GetDraggingRectFromRect(
+                    &rcTray,
+                    &m_DraggingMonitor);
+                //m_TrayRects[DraggingPosition] = rcTray;
+            }
 
-        case ID_SHELL_CMD_EXPLORE_ALL_USERS:
-            OpenCommonStartMenuDirectory(m_hWnd,
-                                         TEXT("explore"));
-            break;
+            //Monitor = CalculateValidSize(DraggingPosition,
+            //                             &rcTray);
 
-        case ID_LOCKTASKBAR:
-            if (SHRestricted(REST_CLASSICSHELL) == 0)
+            m_Monitor = m_DraggingMonitor;
+            m_Position = m_DraggingPosition;
+            IsDragging = FALSE;
+
+            m_TrayRects[m_Position] = rcTray;
+            goto ChangePos;
+        }
+        else if (GetWindowRect(&rcTray))
+        {
+            if (InSizeMove)
             {
-                Lock(!Locked);
-            }
-            break;
+                if (!(pwp->flags & SWP_NOMOVE))
+                {
+                    rcTray.left = pwp->x;
+                    rcTray.top = pwp->y;
+                }
 
-        case ID_SHELL_CMD_OPEN_TASKMGR:
-            OpenTaskManager(m_hWnd);
-            break;
+                if (!(pwp->flags & SWP_NOSIZE))
+                {
+                    rcTray.right = rcTray.left + pwp->cx;
+                    rcTray.bottom = rcTray.top + pwp->cy;
+                }
 
-        case ID_SHELL_CMD_UNDO_ACTION:
-            break;
+                m_Position = GetDraggingRectFromRect(&rcTray, &m_Monitor);
 
-        case ID_SHELL_CMD_SHOW_DESKTOP:
-            break;
+                if (!(pwp->flags & (SWP_NOMOVE | SWP_NOSIZE)))
+                {
+                    SIZE szWnd;
 
-        case ID_SHELL_CMD_TILE_WND_H:
-            TileWindows(NULL, MDITILE_HORIZONTAL, NULL, 0, NULL);
-            break;
+                    szWnd.cx = pwp->cx;
+                    szWnd.cy = pwp->cy;
 
-        case ID_SHELL_CMD_TILE_WND_V:
-            TileWindows(NULL, MDITILE_VERTICAL, NULL, 0, NULL);
-            break;
+                    MakeTrayRectWithSize(m_Position, &szWnd, &rcTray);
+                }
 
-        case ID_SHELL_CMD_CASCADE_WND:
-            CascadeWindows(NULL, MDITILE_SKIPDISABLED, NULL, 0, NULL);
-            break;
+                if (AutoHide)
+                {
+                    rcTray.left -= m_AutoHideOffset.cx;
+                    rcTray.right -= m_AutoHideOffset.cx;
+                    rcTray.top -= m_AutoHideOffset.cy;
+                    rcTray.bottom -= m_AutoHideOffset.cy;
+                }
+                m_TrayRects[m_Position] = rcTray;
+            }
+            else
+            {
+                /* If the user isn't resizing the tray window we need to make sure the
+                   new size or position is valid. this is to prevent changes to the window
+                   without user interaction. */
+                rcTray = m_TrayRects[m_Position];
+            }
 
-        case ID_SHELL_CMD_CUST_NOTIF:
-            break;
+ChangePos:
+            m_TraySize.cx = rcTray.right - rcTray.left;
+            m_TraySize.cy = rcTray.bottom - rcTray.top;
 
-        case ID_SHELL_CMD_ADJUST_DAT:
-            //FIXME: Use SHRunControlPanel
-            ShellExecuteW(m_hWnd, NULL, L"timedate.cpl", NULL, NULL, SW_NORMAL);
-            break;
+            if (AutoHide)
+            {
+                rcTray.left += m_AutoHideOffset.cx;
+                rcTray.right += m_AutoHideOffset.cx;
+                rcTray.top += m_AutoHideOffset.cy;
+                rcTray.bottom += m_AutoHideOffset.cy;
+            }
 
-        default:
-            TRACE("ITrayWindow::ExecContextMenuCmd(%u): Unhandled Command ID!\n", uiCmd);
-            bHandled = FALSE;
-            break;
+            pwp->flags &= ~(SWP_NOMOVE | SWP_NOSIZE);
+            pwp->x = rcTray.left;
+            pwp->y = rcTray.top;
+            pwp->cx = m_TraySize.cx;
+            pwp->cy = m_TraySize.cy;
         }
-
-        return bHandled;
     }
 
-    BOOL STDMETHODCALLTYPE Lock(IN BOOL bLock)
+    VOID ApplyClipping(IN BOOL Clip)
     {
-        BOOL bPrevLock = Locked;
+        RECT rcClip, rcWindow;
+        HRGN hClipRgn;
 
-        if (Locked != bLock)
+        if (GetWindowRect(&rcWindow))
         {
-            Locked = bLock;
+            /* Disable clipping on systems with only one monitor */
+            if (GetSystemMetrics(SM_CMONITORS) <= 1)
+                Clip = FALSE;
 
-            if (m_TrayBandSite != NULL)
+            if (Clip)
             {
-                if (!SUCCEEDED(m_TrayBandSite->Lock(bLock)))
+                rcClip = rcWindow;
+
+                GetScreenRect(m_Monitor, &rcClip);
+
+                if (!IntersectRect(&rcClip, &rcClip, &rcWindow))
                 {
-                    /* Reset?? */
-                    Locked = bPrevLock;
-                    return bPrevLock;
+                    rcClip = rcWindow;
                 }
-            }
 
-            if (Locked && m_Theme)
-            {
-                SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, 0);
+                OffsetRect(&rcClip,
+                           -rcWindow.left,
+                           -rcWindow.top);
+
+                hClipRgn = CreateRectRgnIndirect(&rcClip);
             }
             else
-            {
-                SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, WS_THICKFRAME | WS_BORDER);
-            }
-            SetWindowPos(NULL, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER);
+                hClipRgn = NULL;
 
+            /* Set the clipping region or make sure the window isn't clipped
+               by disabling it explicitly. */
+            SetWindowRgn(hClipRgn, TRUE);
         }
-
-        return bPrevLock;
     }
 
-
-    LRESULT DrawBackground(HDC hdc)
+    VOID ResizeWorkArea()
     {
-        RECT rect;
-        int partId;
+#if !WIN7_DEBUG_MODE
+        RECT rcTray, rcWorkArea;
 
-        GetClientRect(&rect);
+        /* If monitor has changed then fix the previous monitors work area */
+        if (m_PreviousMonitor != m_Monitor)
+        {
+            GetScreenRect(m_PreviousMonitor, &rcWorkArea);
+            SystemParametersInfoW(SPI_SETWORKAREA,
+                                  1,
+                                  &rcWorkArea,
+                                  SPIF_SENDCHANGE);
+        }
 
-        if (m_Theme)
+        rcTray = m_TrayRects[m_Position];
+
+        GetScreenRect(m_Monitor, &rcWorkArea);
+        m_PreviousMonitor = m_Monitor;
+
+        /* If AutoHide is false then change the workarea to exclude
+           the area that the taskbar covers. */
+        if (!AutoHide)
         {
-            GetClientRect(&rect);
             switch (m_Position)
             {
-            case ABE_LEFT:
-                partId = TBP_BACKGROUNDLEFT;
-                break;
             case ABE_TOP:
-                partId = TBP_BACKGROUNDTOP;
+                rcWorkArea.top = rcTray.bottom;
+                break;
+            case ABE_LEFT:
+                rcWorkArea.left = rcTray.right;
                 break;
             case ABE_RIGHT:
-                partId = TBP_BACKGROUNDRIGHT;
+                rcWorkArea.right = rcTray.left;
                 break;
             case ABE_BOTTOM:
-            default:
-                partId = TBP_BACKGROUNDBOTTOM;
+                rcWorkArea.bottom = rcTray.top;
                 break;
             }
-
-            DrawThemeBackground(m_Theme, hdc, partId, 0, &rect, 0);
         }
 
-        return TRUE;
+        /*
+         * Resize the current monitor work area. Win32k will also send
+         * a WM_SIZE message to automatically resize the desktop.
+         */
+        SystemParametersInfoW(SPI_SETWORKAREA,
+                              1,
+                              &rcWorkArea,
+                              SPIF_SENDCHANGE);
+#endif
     }
 
-    LRESULT OnEraseBackground(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    VOID CheckTrayWndPosition()
     {
-        HDC hdc = (HDC) wParam;
+        RECT rcTray;
 
-        if (!m_Theme)
+        rcTray = m_TrayRects[m_Position];
+
+        if (AutoHide)
         {
-            bHandled = FALSE;
-            return 0;
+            rcTray.left += m_AutoHideOffset.cx;
+            rcTray.right += m_AutoHideOffset.cx;
+            rcTray.top += m_AutoHideOffset.cy;
+            rcTray.bottom += m_AutoHideOffset.cy;
         }
 
-        return DrawBackground(hdc);
+        /* Move the tray window */
+        SetWindowPos(NULL,
+                     rcTray.left,
+                     rcTray.top,
+                     rcTray.right - rcTray.left,
+                     rcTray.bottom - rcTray.top,
+                     SWP_NOZORDER | SWP_NOACTIVATE);
+
+        ResizeWorkArea();
+
+        ApplyClipping(TRUE);
     }
 
-    int DrawSizer(IN HRGN hRgn)
+    typedef struct _TW_STUCKRECTS2
     {
-        HDC hdc;
-        RECT rect;
-        int backoundPart;
+        DWORD cbSize;
+        LONG Unknown;
+        DWORD dwFlags;
+        DWORD Position;
+        SIZE Size;
+        RECT Rect;
+    } TW_STRUCKRECTS2, *PTW_STUCKRECTS2;
 
-        GetWindowRect(m_hWnd, &rect);
-        OffsetRect(&rect, -rect.left, -rect.top);
+    VOID RegLoadSettings()
+    {
+        DWORD Pos;
+        TW_STRUCKRECTS2 sr;
+        RECT rcScreen;
+        SIZE WndSize, EdgeSize, DlgFrameSize;
+        DWORD cbSize = sizeof(sr);
+        SIZE StartBtnSize = m_StartButton.GetSize();
 
-        hdc = GetDCEx(m_hWnd, hRgn, DCX_WINDOW | DCX_INTERSECTRGN | DCX_PARENTCLIP);
+        EdgeSize.cx = GetSystemMetrics(SM_CXEDGE);
+        EdgeSize.cy = GetSystemMetrics(SM_CYEDGE);
+        DlgFrameSize.cx = GetSystemMetrics(SM_CXDLGFRAME);
+        DlgFrameSize.cy = GetSystemMetrics(SM_CYDLGFRAME);
 
-        switch (m_Position)
+        if (SHGetValue(hkExplorer,
+            TEXT("StuckRects2"),
+            TEXT("Settings"),
+            NULL,
+            &sr,
+            &cbSize) == ERROR_SUCCESS &&
+            sr.cbSize == sizeof(sr))
         {
-        case ABE_LEFT:
-            backoundPart = TBP_SIZINGBARLEFT;
-            rect.left = rect.right - GetSystemMetrics(SM_CXSIZEFRAME);
-            break;
-        case ABE_TOP:
-            backoundPart = TBP_SIZINGBARTOP;
-            rect.top = rect.bottom - GetSystemMetrics(SM_CYSIZEFRAME);
-            break;
-        case ABE_RIGHT:
-            backoundPart = TBP_SIZINGBARRIGHT;
-            rect.right = rect.left + GetSystemMetrics(SM_CXSIZEFRAME);
-            break;
-        case ABE_BOTTOM:
-        default:
-            backoundPart = TBP_SIZINGBARBOTTOM;
-            rect.bottom = rect.top + GetSystemMetrics(SM_CYSIZEFRAME);
-            break;
+            AutoHide = (sr.dwFlags & ABS_AUTOHIDE) != 0;
+            AlwaysOnTop = (sr.dwFlags & ABS_ALWAYSONTOP) != 0;
+            SmSmallIcons = (sr.dwFlags & 0x4) != 0;
+            HideClock = (sr.dwFlags & 0x8) != 0;
+
+            /* FIXME: Are there more flags? */
+
+#if WIN7_DEBUG_MODE
+            m_Position = ABE_LEFT;
+#else
+            if (sr.Position > ABE_BOTTOM)
+                m_Position = ABE_BOTTOM;
+            else
+                m_Position = sr.Position;
+#endif
+
+            /* Try to find out which monitor the tray window was located on last.
+               Here we're only interested in the monitor screen that we think
+               is the last one used. We're going to determine on which monitor
+               we really are after calculating the docked position. */
+            rcScreen = sr.Rect;
+            GetScreenRectFromRect(
+                &rcScreen,
+                MONITOR_DEFAULTTONEAREST);
         }
+        else
+        {
+            m_Position = ABE_BOTTOM;
+            AlwaysOnTop = TRUE;
 
-        DrawThemeBackground(m_Theme, hdc, backoundPart, 0, &rect, 0);
+            /* Use the minimum size of the taskbar, we'll use the start
+               button as a minimum for now. Make sure we calculate the
+               entire window size, not just the client size. However, we
+               use a thinner border than a standard thick border, so that
+               the start button and bands are not stuck to the screen border. */
+            if(!IsThemeActive())
+            {
+                sr.Size.cx = StartBtnSize.cx + (2 * (EdgeSize.cx + DlgFrameSize.cx));
+                sr.Size.cy = StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
+            }
+            else
+            {
+                sr.Size.cx = StartBtnSize.cx - EdgeSize.cx;
+                sr.Size.cy = StartBtnSize.cy - EdgeSize.cy;
+                if(!Locked)
+                    sr.Size.cy += GetSystemMetrics(SM_CYSIZEFRAME);
+            }
 
-        ReleaseDC(m_hWnd, hdc);
-        return 0;
+            /* Use the primary screen by default */
+            rcScreen.left = 0;
+            rcScreen.top = 0;
+            rcScreen.right = GetSystemMetrics(SM_CXSCREEN);
+            rcScreen.bottom = GetSystemMetrics(SM_CYSCREEN);
+            GetScreenRectFromRect(
+                &rcScreen,
+                MONITOR_DEFAULTTOPRIMARY);
+        }
+
+        if (m_hWnd != NULL)
+            SetWindowPos(
+            AlwaysOnTop ? HWND_TOPMOST : HWND_NOTOPMOST,
+            0,
+            0,
+            0,
+            0,
+            SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
+
+        /* Determine a minimum tray window rectangle. The "client" height is
+           zero here since we cannot determine an optimal minimum width when
+           loaded as a vertical tray window. We just need to make sure the values
+           loaded from the registry are at least. The windows explorer behaves
+           the same way, it allows the user to save a zero width vertical tray
+           window, but not a zero height horizontal tray window. */
+        if(!IsThemeActive())
+        {
+            WndSize.cx = 2 * (EdgeSize.cx + DlgFrameSize.cx);
+            WndSize.cy = StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
+        }
+        else
+        {
+            WndSize.cx = StartBtnSize.cx;
+            WndSize.cy = StartBtnSize.cy - EdgeSize.cx;
+        }
+
+        if (WndSize.cx < sr.Size.cx)
+            WndSize.cx = sr.Size.cx;
+        if (WndSize.cy < sr.Size.cy)
+            WndSize.cy = sr.Size.cy;
+
+        /* Save the calculated size */
+        m_TraySize = WndSize;
+
+        /* Calculate all docking rectangles. We need to do this here so they're
+           initialized and dragging the tray window to another position gives
+           usable results */
+        for (Pos = ABE_LEFT; Pos <= ABE_BOTTOM; Pos++)
+        {
+            GetTrayRectFromScreenRect(Pos,
+                                      &rcScreen,
+                                      &m_TraySize,
+                                      &m_TrayRects[Pos]);
+            // TRACE("m_TrayRects[%d(%d)]: %d,%d,%d,%d\n", Pos, Position, m_TrayRects[Pos].left, m_TrayRects[Pos].top, m_TrayRects[Pos].right, m_TrayRects[Pos].bottom);
+        }
+
+        /* Determine which monitor we are on. It shouldn't matter which docked
+           position rectangle we use */
+        m_Monitor = GetMonitorFromRect(&m_TrayRects[ABE_LEFT]);
     }
 
-    DWORD WINAPI RunFileDlgThread()
+    VOID AlignControls(IN PRECT prcClient OPTIONAL)
     {
-        HWND hwnd;
-        RECT posRect;
+        RECT rcClient;
+        SIZE TraySize, StartSize;
+        POINT ptTrayNotify = { 0, 0 };
+        BOOL Horizontal;
+        HDWP dwp;
+
+        m_StartButton.UpdateSize();
+        if (prcClient != NULL)
+        {
+            rcClient = *prcClient;
+        }
+        else
+        {
+            if (!GetClientRect(&rcClient))
+            {
+                ERR("Could not get client rect lastErr=%d\n", GetLastError());
+                return;
+            }
+        }
 
-        GetWindowRect(m_StartButton.m_hWnd, &posRect);
+        Horizontal = IsPosHorizontal();
 
-        hwnd = CreateWindowEx(0,
-                              WC_STATIC,
-                              NULL,
-                              WS_OVERLAPPED | WS_DISABLED | WS_CLIPSIBLINGS | WS_BORDER | SS_LEFT,
-                              posRect.left,
-                              posRect.top,
-                              posRect.right - posRect.left,
-                              posRect.bottom - posRect.top,
-                              NULL,
-                              NULL,
-                              NULL,
-                              NULL);
+        /* We're about to resize/move the start button, the rebar control and
+           the tray notification control */
+        dwp = BeginDeferWindowPos(3);
+        if (dwp == NULL)
+        {
+            ERR("BeginDeferWindowPos failed. lastErr=%d\n", GetLastError());
+            return;
+        }
+
+        /* Limit the Start button width to the client width, if necessary */
+        StartSize = m_StartButton.GetSize();
+        if (StartSize.cx > rcClient.right)
+            StartSize.cx = rcClient.right;
+
+        if (m_StartButton.m_hWnd != NULL)
+        {
+            /* Resize and reposition the button */
+            dwp = m_StartButton.DeferWindowPos(dwp,
+                                               NULL,
+                                               0,
+                                               0,
+                                               StartSize.cx,
+                                               StartSize.cy,
+                                               SWP_NOZORDER | SWP_NOACTIVATE);
+            if (dwp == NULL)
+            {
+                ERR("DeferWindowPos for start button failed. lastErr=%d\n", GetLastError());
+                return;
+            }
+        }
 
-        m_RunFileDlgOwner = hwnd;
+        /* Determine the size that the tray notification window needs */
+        if (Horizontal)
+        {
+            TraySize.cx = 0;
+            TraySize.cy = rcClient.bottom;
+        }
+        else
+        {
+            TraySize.cx = rcClient.right;
+            TraySize.cy = 0;
+        }
 
-        RunFileDlg(hwnd, NULL, NULL, NULL, NULL, RFF_CALCDIRECTORY);
+        if (m_TrayNotify != NULL &&
+            SendMessage(m_TrayNotify,
+                        TNWM_GETMINIMUMSIZE,
+                        (WPARAM)Horizontal,
+                        (LPARAM)&TraySize))
+        {
+            /* Move the tray notification window to the desired location */
+            if (Horizontal)
+                ptTrayNotify.x = rcClient.right - TraySize.cx;
+            else
+                ptTrayNotify.y = rcClient.bottom - TraySize.cy;
 
-        m_RunFileDlgOwner = NULL;
-        ::DestroyWindow(hwnd);
+            dwp = ::DeferWindowPos(dwp,
+                                   m_TrayNotify,
+                                   NULL,
+                                   ptTrayNotify.x,
+                                   ptTrayNotify.y,
+                                   TraySize.cx,
+                                   TraySize.cy,
+                                   SWP_NOZORDER | SWP_NOACTIVATE);
+            if (dwp == NULL)
+            {
+                ERR("DeferWindowPos for notification area failed. lastErr=%d\n", GetLastError());
+                return;
+            }
+        }
 
-        return 0;
-    }
+        /* Resize/Move the rebar control */
+        if (m_Rebar != NULL)
+        {
+            POINT ptRebar = { 0, 0 };
+            SIZE szRebar;
 
-    static DWORD WINAPI s_RunFileDlgThread(IN OUT PVOID pParam)
-    {
-        CTrayWindow * This = (CTrayWindow*) pParam;
-        return This->RunFileDlgThread();
-    }
+            SetWindowStyle(m_Rebar, CCS_VERT, Horizontal ? 0 : CCS_VERT);
 
-    void DisplayRunFileDlg()
-    {
-        HWND hRunDlg;
-        if (m_RunFileDlgOwner)
-        {
-            hRunDlg = GetLastActivePopup(m_RunFileDlgOwner);
-            if (hRunDlg != NULL &&
-                hRunDlg != m_RunFileDlgOwner)
+            if (Horizontal)
             {
-                SetForegroundWindow(hRunDlg);
-                return;
+                ptRebar.x = StartSize.cx + GetSystemMetrics(SM_CXSIZEFRAME);
+                szRebar.cx = ptTrayNotify.x - ptRebar.x;
+                szRebar.cy = rcClient.bottom;
+            }
+            else
+            {
+                ptRebar.y = StartSize.cy + GetSystemMetrics(SM_CYSIZEFRAME);
+                szRebar.cx = rcClient.right;
+                szRebar.cy = ptTrayNotify.y - ptRebar.y;
             }
+
+            dwp = ::DeferWindowPos(dwp,
+                                   m_Rebar,
+                                   NULL,
+                                   ptRebar.x,
+                                   ptRebar.y,
+                                   szRebar.cx,
+                                   szRebar.cy,
+                                   SWP_NOZORDER | SWP_NOACTIVATE);
         }
 
-        CloseHandle(CreateThread(NULL, 0, s_RunFileDlgThread, this, 0, NULL));
+        if (dwp != NULL)
+            EndDeferWindowPos(dwp);
+
+        if (m_TaskSwitch != NULL)
+        {
+            /* Update the task switch window configuration */
+            SendMessage(m_TaskSwitch, TSWM_UPDATETASKBARPOS, 0, 0);
+        }
     }
 
     void PopupStartMenu()
@@ -1979,7 +1615,7 @@ ChangePos:
             RECTL rcExclude;
             DWORD dwFlags = 0;
 
-            if (GetWindowRect(m_StartButton.m_hWnd, (RECT*) &rcExclude))
+            if (m_StartButton.GetWindowRect((RECT*) &rcExclude))
             {
                 switch (m_Position)
                 {
@@ -2020,7 +1656,7 @@ ChangePos:
         UINT state = m_AutoHideState;
 
         GetCursorPos(&pt);
-        GetWindowRect(m_hWnd, &rcCurrent);
+        GetWindowRect(&rcCurrent);
         over = PtInRect(&rcCurrent, pt);
 
         if (m_StartButton.SendMessage( BM_GETSTATE, 0, 0) != BST_UNCHECKED)
@@ -2148,40 +1784,374 @@ ChangePos:
                 m_AutoHideOffset.cx = 0;
             }
 
-            if (m_AutoHideOffset.cy >= AUTOHIDE_SPEED_SHOW)
-            {
-                m_AutoHideOffset.cy -= AUTOHIDE_SPEED_SHOW;
-            }
-            else if (m_AutoHideOffset.cy <= -AUTOHIDE_SPEED_SHOW)
+            if (m_AutoHideOffset.cy >= AUTOHIDE_SPEED_SHOW)
+            {
+                m_AutoHideOffset.cy -= AUTOHIDE_SPEED_SHOW;
+            }
+            else if (m_AutoHideOffset.cy <= -AUTOHIDE_SPEED_SHOW)
+            {
+                m_AutoHideOffset.cy += AUTOHIDE_SPEED_SHOW;
+            }
+            else
+            {
+                m_AutoHideOffset.cy = 0;
+            }
+
+            if (m_AutoHideOffset.cx != 0 || m_AutoHideOffset.cy != 0)
+            {
+                SetTimer(TIMER_ID_AUTOHIDE, AUTOHIDE_INTERVAL_ANIMATING, NULL);
+                break;
+            }
+
+            /* fallthrough */
+        case AUTOHIDE_SHOWN:
+
+            KillTimer(TIMER_ID_AUTOHIDE);
+            m_AutoHideState = AUTOHIDE_SHOWN;
+            break;
+        }
+
+        rc.left += m_AutoHideOffset.cx;
+        rc.right += m_AutoHideOffset.cx;
+        rc.top += m_AutoHideOffset.cy;
+        rc.bottom += m_AutoHideOffset.cy;
+
+        TRACE("AutoHide Changing position to (%d, %d, %d, %d) and state=%u.\n", rc.left, rc.top, rc.right, rc.bottom, m_AutoHideState);
+        SetWindowPos(NULL, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, SWP_NOACTIVATE | SWP_NOZORDER);
+    }
+
+
+
+
+
+    /**********************************************************
+     *    ##### taskbar drawing #####
+     */
+
+    LRESULT EraseBackgroundWithTheme(HDC hdc)
+    {
+        RECT rect;
+        int partId;
+        HRESULT res;
+
+        GetClientRect(&rect);
+
+        if (m_Theme)
+        {
+            GetClientRect(&rect);
+            switch (m_Position)
+            {
+            case ABE_LEFT:
+                partId = TBP_BACKGROUNDLEFT;
+                break;
+            case ABE_TOP:
+                partId = TBP_BACKGROUNDTOP;
+                break;
+            case ABE_RIGHT:
+                partId = TBP_BACKGROUNDRIGHT;
+                break;
+            case ABE_BOTTOM:
+            default:
+                partId = TBP_BACKGROUNDBOTTOM;
+                break;
+            }
+            res = DrawThemeBackground(m_Theme, hdc, partId, 0, &rect, 0);
+        }
+
+        return res;
+    }
+
+    int DrawSizerWithTheme(IN HRGN hRgn)
+    {
+        HDC hdc;
+        RECT rect;
+        int backgroundPart;
+
+        GetWindowRect(&rect);
+        OffsetRect(&rect, -rect.left, -rect.top);
+
+        hdc = GetWindowDC();
+
+        switch (m_Position)
+        {
+        case ABE_LEFT:
+            backgroundPart = TBP_SIZINGBARLEFT;
+            rect.left = rect.right - GetSystemMetrics(SM_CXSIZEFRAME);
+            break;
+        case ABE_TOP:
+            backgroundPart = TBP_SIZINGBARTOP;
+            rect.top = rect.bottom - GetSystemMetrics(SM_CYSIZEFRAME);
+            break;
+        case ABE_RIGHT:
+            backgroundPart = TBP_SIZINGBARRIGHT;
+            rect.right = rect.left + GetSystemMetrics(SM_CXSIZEFRAME);
+            break;
+        case ABE_BOTTOM:
+        default:
+            backgroundPart = TBP_SIZINGBARBOTTOM;
+            rect.bottom = rect.top + GetSystemMetrics(SM_CYSIZEFRAME);
+            break;
+        }
+        if (IsThemeBackgroundPartiallyTransparent(m_Theme, backgroundPart, 0))
+        {
+            DrawThemeParentBackground(m_hWnd, hdc, &rect);
+        }
+        DrawThemeBackground(m_Theme, hdc, backgroundPart, 0, &rect, 0);
+
+        ReleaseDC(hdc);
+        return 0;
+    }
+
+
+
+
+
+    /*
+     * ITrayWindow
+     */
+    HRESULT STDMETHODCALLTYPE Open()
+    {
+        RECT rcWnd;
+
+        /* Check if there's already a window created and try to show it.
+           If it was somehow destroyed just create a new tray window. */
+        if (m_hWnd != NULL && IsWindow())
+        {
+            if (!IsWindowVisible())
+            {
+                CheckTrayWndPosition();
+
+                ShowWindow(SW_SHOW);
+            }
+
+            return S_OK;
+        }
+
+        DWORD dwExStyle = WS_EX_TOOLWINDOW | WS_EX_WINDOWEDGE;
+        if (AlwaysOnTop)
+            dwExStyle |= WS_EX_TOPMOST;
+
+        DWORD dwStyle = WS_POPUP | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN;
+        if(!IsThemeActive())
+        {
+            dwStyle |= WS_THICKFRAME | WS_BORDER;
+        }
+
+        ZeroMemory(&rcWnd, sizeof(rcWnd));
+        if (m_Position != (DWORD) -1)
+            rcWnd = m_TrayRects[m_Position];
+
+        if (!Create(NULL, rcWnd, NULL, dwStyle, dwExStyle))
+            return E_FAIL;
+
+        return S_OK;
+    }
+
+    HRESULT STDMETHODCALLTYPE Close()
+    {
+        if (m_hWnd != NULL)
+        {
+            SendMessage(m_hWnd,
+                        WM_APP_TRAYDESTROY,
+                        0,
+                        0);
+        }
+
+        return S_OK;
+    }
+
+    HWND STDMETHODCALLTYPE GetHWND()
+    {
+        return m_hWnd;
+    }
+
+    BOOL STDMETHODCALLTYPE IsSpecialHWND(IN HWND hWnd)
+    {
+        return (m_hWnd == hWnd ||
+                (m_DesktopWnd != NULL && m_hWnd == m_DesktopWnd));
+    }
+
+    BOOL STDMETHODCALLTYPE IsHorizontal()
+    {
+        return IsPosHorizontal();
+    }
+
+    HFONT STDMETHODCALLTYPE GetCaptionFonts(OUT HFONT *phBoldCaption OPTIONAL)
+    {
+        if (phBoldCaption != NULL)
+            *phBoldCaption = m_StartButton.GetFont();
+
+        return m_CaptionFont;
+    }
+
+    BOOL STDMETHODCALLTYPE Lock(IN BOOL bLock)
+    {
+        BOOL bPrevLock = Locked;
+
+        if (Locked != bLock)
+        {
+            Locked = bLock;
+
+            if (m_TrayBandSite != NULL)
             {
-                m_AutoHideOffset.cy += AUTOHIDE_SPEED_SHOW;
+                if (!SUCCEEDED(m_TrayBandSite->Lock(bLock)))
+                {
+                    /* Reset?? */
+                    Locked = bPrevLock;
+                    return bPrevLock;
+                }
             }
-            else
+
+            if (m_Theme)
             {
-                m_AutoHideOffset.cy = 0;
+                /* Update cached tray sizes */
+                for(DWORD Pos = ABE_LEFT; Pos <= ABE_BOTTOM; Pos++)
+                {
+                    RECT rcGripper = {0};
+                    AdjustSizerRect(&rcGripper, Pos);
+
+                    if(Locked)
+                    {
+                        m_TrayRects[Pos].top += rcGripper.top;
+                        m_TrayRects[Pos].left += rcGripper.left;
+                        m_TrayRects[Pos].bottom += rcGripper.bottom;
+                        m_TrayRects[Pos].right += rcGripper.right;
+                    }
+                    else
+                    {
+                        m_TrayRects[Pos].top -= rcGripper.top;
+                        m_TrayRects[Pos].left -= rcGripper.left;
+                        m_TrayRects[Pos].bottom -= rcGripper.bottom;
+                        m_TrayRects[Pos].right -= rcGripper.right;
+                    }
+                }
             }
+            SetWindowPos(NULL, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOZORDER | SWP_NOOWNERZORDER);
+        }
 
-            if (m_AutoHideOffset.cx != 0 || m_AutoHideOffset.cy != 0)
+        return bPrevLock;
+    }
+
+
+
+
+
+
+    /**********************************************************
+     *    ##### message handling #####
+     */
+
+    LRESULT OnCreate(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    {
+        ((ITrayWindow*)this)->AddRef();
+
+        SetWindowTheme(m_hWnd, L"TaskBar", NULL);
+
+        InterlockedIncrement(&TrayWndCount);
+
+        if (m_CaptionFont == NULL)
+        {
+            NONCLIENTMETRICS ncm;
+
+            /* Get the system fonts, we use the caption font,
+               always bold, though. */
+            ncm.cbSize = sizeof(ncm);
+            if (SystemParametersInfoW(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE))
             {
-                SetTimer(TIMER_ID_AUTOHIDE, AUTOHIDE_INTERVAL_ANIMATING, NULL);
-                break;
+                if (m_CaptionFont == NULL)
+                {
+                    ncm.lfCaptionFont.lfWeight = FW_NORMAL;
+                    m_CaptionFont = CreateFontIndirect(&ncm.lfCaptionFont);
+                }
             }
+        }
 
-            /* fallthrough */
-        case AUTOHIDE_SHOWN:
+        /* Create the Start button */
+        m_StartButton.Create(m_hWnd);
 
-            KillTimer(TIMER_ID_AUTOHIDE);
-            m_AutoHideState = AUTOHIDE_SHOWN;
-            break;
+        /* Load the saved tray window settings */
+        RegLoadSettings();
+
+        /* Create and initialize the start menu */
+        HBITMAP hbmBanner = LoadBitmapW(hExplorerInstance, MAKEINTRESOURCEW(IDB_STARTMENU));
+        m_StartMenuPopup = CreateStartMenu(this, &m_StartMenuBand, hbmBanner, 0);
+
+        /* Load the tray band site */
+        m_TrayBandSite = CreateTrayBandSite(this, &m_Rebar, &m_TaskSwitch);
+        SetWindowTheme(m_Rebar, L"TaskBar", NULL);
+
+        /* Create the tray notification window */
+        m_TrayNotify = CreateTrayNotifyWnd(this, HideClock, &m_TrayNotifyInstance);
+
+        if (UpdateNonClientMetrics())
+        {
+            SetWindowsFont();
         }
 
-        rc.left += m_AutoHideOffset.cx;
-        rc.right += m_AutoHideOffset.cx;
-        rc.top += m_AutoHideOffset.cy;
-        rc.bottom += m_AutoHideOffset.cy;
+        /* Move the tray window to the right position and resize it if necessary */
+        CheckTrayWndPosition();
 
-        TRACE("AutoHide Changing position to (%d, %d, %d, %d) and state=%u.\n", rc.left, rc.top, rc.right, rc.bottom, m_AutoHideState);
-        SetWindowPos(NULL, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, SWP_NOACTIVATE | SWP_NOZORDER);
+        /* Align all controls on the tray window */
+        AlignControls(NULL);
+
+        InitShellServices(&m_ShellServices);
+
+        if (AutoHide)
+        {
+            m_AutoHideState = AUTOHIDE_HIDING;
+            SetTimer(TIMER_ID_AUTOHIDE, AUTOHIDE_DELAY_HIDE, NULL);
+        }
+
+        RegisterHotKey(m_hWnd, IDHK_RUN, MOD_WIN, 'R');
+        RegisterHotKey(m_hWnd, IDHK_MINIMIZE_ALL, MOD_WIN, 'M');
+        RegisterHotKey(m_hWnd, IDHK_RESTORE_ALL, MOD_WIN|MOD_SHIFT, 'M');
+        RegisterHotKey(m_hWnd, IDHK_HELP, MOD_WIN, VK_F1);
+        RegisterHotKey(m_hWnd, IDHK_EXPLORE, MOD_WIN, 'E');
+        RegisterHotKey(m_hWnd, IDHK_FIND, MOD_WIN, 'F');
+        RegisterHotKey(m_hWnd, IDHK_FIND_COMPUTER, MOD_WIN|MOD_CONTROL, 'F');
+        RegisterHotKey(m_hWnd, IDHK_NEXT_TASK, MOD_WIN, VK_TAB);
+        RegisterHotKey(m_hWnd, IDHK_PREV_TASK, MOD_WIN|MOD_SHIFT, VK_TAB);
+        RegisterHotKey(m_hWnd, IDHK_SYS_PROPERTIES, MOD_WIN, VK_PAUSE);
+        RegisterHotKey(m_hWnd, IDHK_DESKTOP, MOD_WIN, 'D');
+        RegisterHotKey(m_hWnd, IDHK_PAGER, MOD_WIN, 'B');
+
+        return TRUE;
+    }
+
+    LRESULT OnThemeChanged(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    {
+        if (m_Theme)
+            CloseThemeData(m_Theme);
+
+        if (IsThemeActive())
+            m_Theme = OpenThemeData(m_hWnd, L"TaskBar");
+        else
+            m_Theme = NULL;
+
+        if (m_Theme)
+        {
+            SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, 0);
+        }
+        else
+        {
+            SetWindowStyle(m_hWnd, WS_THICKFRAME | WS_BORDER, WS_THICKFRAME | WS_BORDER);
+        }
+        SetWindowPos(NULL, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOACTIVATE | SWP_NOMOVE | SWP_NOSIZE | SWP_NOZORDER | SWP_NOOWNERZORDER);
+
+        return TRUE;
+    }
+
+    LRESULT OnEraseBackground(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    {
+        HDC hdc = (HDC) wParam;
+
+        if (!m_Theme)
+        {
+            bHandled = FALSE;
+            return 0;
+        }
+
+        return EraseBackgroundWithTheme(hdc);
     }
 
     LRESULT OnDisplayChange(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
@@ -2189,7 +2159,7 @@ ChangePos:
         /* Load the saved tray window settings */
         RegLoadSettings();
 
-        /* Move the tray window to the right position and resize it if neccessary */
+        /* Move the tray window to the right position and resize it if necessary */
         CheckTrayWndPosition();
 
         /* Align all controls on the tray window */
@@ -2216,7 +2186,7 @@ ChangePos:
             return 0;
         }
 
-        return DrawSizer((HRGN) wParam);
+        return DrawSizerWithTheme((HRGN) wParam);
     }
 
     LRESULT OnCtlColorBtn(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
@@ -2240,7 +2210,7 @@ ChangePos:
 
         SetLastError(ERROR_SUCCESS);
         if (GetClientRect(&rcClient) &&
-            (MapWindowPoints(m_hWnd, NULL, (LPPOINT) &rcClient, 2) != 0 || GetLastError() == ERROR_SUCCESS))
+            (MapWindowPoints(NULL, (LPPOINT) &rcClient, 2) != 0 || GetLastError() == ERROR_SUCCESS))
         {
             pt.x = (SHORT) LOWORD(lParam);
             pt.y = (SHORT) HIWORD(lParam);
@@ -2315,6 +2285,36 @@ ChangePos:
 
         if (!Locked)
         {
+            /* Get the rect of the rebar */
+            RECT rebarRect, taskbarRect;
+            ::GetWindowRect(m_Rebar, &rebarRect);
+            ::GetWindowRect(m_hWnd, &taskbarRect);
+            OffsetRect(&rebarRect, -taskbarRect.left, -taskbarRect.top);
+
+            /* Calculate the difference of size of the taskbar and the rebar */
+            SIZE margins;
+            margins.cx = taskbarRect.right - taskbarRect.left - rebarRect.right + rebarRect.left;
+            margins.cy = taskbarRect.bottom - taskbarRect.top - rebarRect.bottom + rebarRect.top;
+
+            /* Calculate the new size of the rebar and make it resize, then change the new taskbar size */
+            switch (m_Position)
+            {
+            case ABE_TOP:
+                rebarRect.bottom = rebarRect.top + pRect->bottom - pRect->top - margins.cy;
+                ::SendMessageW(m_Rebar, RB_SIZETORECT, RBSTR_CHANGERECT,  (LPARAM)&rebarRect);
+                pRect->bottom = pRect->top + rebarRect.bottom - rebarRect.top + margins.cy;
+                break;
+            case ABE_BOTTOM:
+                rebarRect.top = rebarRect.bottom - (pRect->bottom - pRect->top - margins.cy);
+                ::SendMessageW(m_Rebar, RB_SIZETORECT, RBSTR_CHANGERECT,  (LPARAM)&rebarRect);
+                pRect->top = pRect->bottom - (rebarRect.bottom - rebarRect.top + margins.cy);
+                break;
+            case ABE_LEFT:
+            case ABE_RIGHT:
+                /* FIXME: what to do here? */
+                break;
+            }
+
             CalculateValidSize(m_Position, pRect);
         }
         else
@@ -2383,7 +2383,7 @@ ChangePos:
         if (!Locked)
         {
             /* Apply clipping */
-            PostMessage(m_hWnd, WM_SIZE, SIZE_RESTORED, 0);
+            PostMessage(WM_SIZE, SIZE_RESTORED, 0);
         }
         return TRUE;
     }
@@ -2406,17 +2406,16 @@ ChangePos:
                 SC_MINIMIZE,
             };
             HMENU hSysMenu;
-            INT i;
-            UINT uId;
+            UINT i, uId;
 
             /* temporarily enable the system menu */
             SetWindowStyle(m_hWnd, WS_SYSMENU, WS_SYSMENU);
 
-            hSysMenu = GetSystemMenu(m_hWnd, FALSE);
+            hSysMenu = GetSystemMenu(FALSE);
             if (hSysMenu != NULL)
             {
                 /* Disable all items that are not relevant */
-                for (i = 0; i != sizeof(uidDisableItem) / sizeof(uidDisableItem[0]); i++)
+                for (i = 0; i < _countof(uidDisableItem); i++)
                 {
                     EnableMenuItem(hSysMenu,
                                    uidDisableItem[i],
@@ -2500,17 +2499,17 @@ ChangePos:
                 POINT ptClient = *ppt;
 
                 /* Convert the coordinates to client-coordinates */
-                MapWindowPoints(NULL, m_hWnd, &ptClient, 1);
+                ::MapWindowPoints(NULL, m_hWnd, &ptClient, 1);
 
-                hWndAtPt = ChildWindowFromPoint(m_hWnd, ptClient);
+                hWndAtPt = ChildWindowFromPoint(ptClient);
                 if (hWndAtPt != NULL &&
-                    (hWndAtPt == m_Rebar || IsChild(m_Rebar, hWndAtPt)))
+                    (hWndAtPt == m_Rebar || ::IsChild(m_Rebar, hWndAtPt)))
                 {
                     /* Check if the user clicked on the task switch window */
                     ptClient = *ppt;
-                    MapWindowPoints(NULL, m_Rebar, &ptClient, 1);
+                    ::MapWindowPoints(NULL, m_Rebar, &ptClient, 1);
 
-                    hWndAtPt = ChildWindowFromPointEx(m_Rebar, ptClient, CWP_SKIPINVISIBLE | CWP_SKIPDISABLED);
+                    hWndAtPt = ::ChildWindowFromPointEx(m_Rebar, ptClient, CWP_SKIPINVISIBLE | CWP_SKIPDISABLED);
                     if (hWndAtPt == m_TaskSwitch)
                         goto HandleTrayContextMenu;
 
@@ -2558,7 +2557,7 @@ HandleTrayContextMenu:
                 {
                 case NTNWM_REALIGN:
                     /* Cause all controls to be aligned */
-                    PostMessage(m_hWnd, WM_SIZE, SIZE_RESTORED, 0);
+                    PostMessage(WM_SIZE, SIZE_RESTORED, 0);
                     break;
                 }
             }
@@ -2601,7 +2600,7 @@ HandleTrayContextMenu:
         if (FAILED_UNEXPECTEDLY(hr))
             return FALSE;
 
-        if (IsWindowVisible(hwndStartMenu))
+        if (::IsWindowVisible(hwndStartMenu))
         {
             m_StartMenuPopup->OnSelect(MPOS_CANCELLEVEL);
         }
@@ -2613,17 +2612,12 @@ HandleTrayContextMenu:
         return TRUE;
     }
 
-    LRESULT DoExitWindows()
-    {
-        ExitWindowsDialog(m_hWnd);
-        return 0;
-    }
-
     LRESULT OnDoExitWindows(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
     {
-        /* 
+        /*
          * TWM_DOEXITWINDOWS is send by the CDesktopBrowser to us
-         * to show the shutdown dialog.
+         * to show the shutdown dialog. Also a WM_CLOSE message sent
+         * by apps should show the dialog.
          */
         return DoExitWindows();
     }
@@ -2639,70 +2633,9 @@ HandleTrayContextMenu:
         return TRUE;
     }
 
-    HRESULT ExecResourceCmd(int id)
-    {
-        WCHAR szCommand[256];
-        WCHAR *pszParameters;
-
-        if (!LoadString(hExplorerInstance,
-                        id,
-                        szCommand,
-                        sizeof(szCommand) / sizeof(szCommand[0])))
-        {
-            return E_FAIL;
-        }
-
-        pszParameters = wcschr(szCommand, L'>');
-        if (pszParameters)
-        {
-            *pszParameters = 0;
-            pszParameters++;
-        }
-
-        ShellExecuteW(m_hWnd, NULL, szCommand, pszParameters, NULL, SW_SHOWNORMAL);
-        return S_OK;
-    }
-
     LRESULT OnHotkey(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
     {
-        switch (wParam)
-        {
-        case IDHK_RUN:
-            DisplayRunFileDlg();
-            break;
-        case IDHK_HELP:
-            ExecResourceCmd(IDS_HELP_COMMAND);
-            break;
-        case IDHK_EXPLORE:
-            //FIXME: We don't support this yet:
-            //ShellExecuteW(0, L"explore", NULL, NULL, NULL, 1);
-            ShellExecuteW(0, NULL, L"explorer.exe", NULL, NULL, 1); 
-            break;
-        case IDHK_FIND:
-            SHFindFiles(NULL, NULL);
-            break;
-        case IDHK_FIND_COMPUTER:
-            SHFindComputer(NULL, NULL);
-            break;
-        case IDHK_SYS_PROPERTIES:
-            //FIXME: Use SHRunControlPanel
-            ShellExecuteW(m_hWnd, NULL, L"sysdm.cpl", NULL, NULL, SW_NORMAL);
-            break;
-        case IDHK_NEXT_TASK:
-            break;
-        case IDHK_PREV_TASK:
-            break;
-        case IDHK_MINIMIZE_ALL:
-            break;
-        case IDHK_RESTORE_ALL:
-            break;
-        case IDHK_DESKTOP:
-            break;
-        case IDHK_PAGER:
-            break;
-        }
-
-        return 0;
+        return HandleHotKey(wParam);
     }
 
     LRESULT OnCommand(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
@@ -2717,38 +2650,7 @@ HandleTrayContextMenu:
 
         if (m_TrayBandSite == NULL || FAILED_UNEXPECTEDLY(m_TrayBandSite->ProcessMessage(m_hWnd, uMsg, wParam, lParam, &Ret)))
         {
-            switch (LOWORD(wParam))
-            {
-            case IDM_TASKBARANDSTARTMENU:
-                DisplayProperties();
-                break;
-
-            case IDM_SEARCH:
-                SHFindFiles(NULL, NULL);
-                break;
-
-            case IDM_HELPANDSUPPORT:
-                ExecResourceCmd(IDS_HELP_COMMAND);
-                break;
-
-            case IDM_RUN:
-                DisplayRunFileDlg();
-                break;
-
-            /* FIXME: Handle these commands as well */
-            case IDM_SYNCHRONIZE:
-            case IDM_DISCONNECT:
-            case IDM_UNDOCKCOMPUTER:
-                break;
-
-            case IDM_LOGOFF:
-                LogoffWindowsDialog(m_hWnd); // FIXME: Maybe handle it in a similar way as DoExitWindows?
-                break;
-
-            case IDM_SHUTDOWN:
-                DoExitWindows();
-                break;
-            }
+            return HandleCommand(LOWORD(wParam));
         }
         return Ret;
     }
@@ -2778,6 +2680,35 @@ HandleTrayContextMenu:
         return TRUE;
     }
 
+    LRESULT OnNcCalcSize(INT code, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
+    {
+        RECT *rc = NULL;
+        /* Ignore WM_NCCALCSIZE if we are not themed or locked */
+        if(!IsThemeActive() || Locked)
+        {
+            bHandled = FALSE;
+            return 0;
+        }
+        if(!wParam)
+        {
+            rc = (RECT*)wParam;
+        }
+        else
+        {
+            NCCALCSIZE_PARAMS *prms = (NCCALCSIZE_PARAMS*)lParam;
+            if(prms->lppos->flags & SWP_NOSENDCHANGING)
+            {
+                bHandled = FALSE;
+                return 0;
+            }
+            rc = &prms->rgrc[0];
+        }
+
+        AdjustSizerRect(rc, m_Position);
+
+        return 0;
+    }
+
     LRESULT OnRebarAutoSize(INT code, LPNMHDR nmhdr, BOOL& bHandled)
     {
 #if 0
@@ -2879,7 +2810,9 @@ HandleTrayContextMenu:
         MESSAGE_HANDLER(WM_APP_TRAYDESTROY, OnAppTrayDestroy)
         MESSAGE_HANDLER(TWM_OPENSTARTMENU, OnOpenStartMenu)
         MESSAGE_HANDLER(TWM_DOEXITWINDOWS, OnDoExitWindows)
+        MESSAGE_HANDLER(WM_CLOSE, OnDoExitWindows)
         MESSAGE_HANDLER(WM_HOTKEY, OnHotkey)
+        MESSAGE_HANDLER(WM_NCCALCSIZE, OnNcCalcSize)
     ALT_MSG_MAP(1)
     END_MSG_MAP()
 
@@ -3010,7 +2943,7 @@ public:
                          UINT idCmdLast,
                          UINT uFlags)
     {
-        HMENU menubase = LoadPopupMenu(hExplorerInstance, MAKEINTRESOURCE(IDM_TRAYWND));
+        HMENU menubase = LoadPopupMenu(hExplorerInstance, MAKEINTRESOURCEW(IDM_TRAYWND));
 
         if (!menubase)
             return HRESULT_FROM_WIN32(GetLastError());
@@ -3031,8 +2964,6 @@ public:
 
             TRACE("Adding item %d label %S type %d\n", mii.wID, mii.dwTypeData, mii.fType);
 
-            mii.fType |= MFT_RADIOCHECK;
-
             ::InsertMenuItemW(hPopup, i + 1, TRUE, &mii);
         }
 
@@ -3082,7 +3013,7 @@ public:
                     /* Setup and invoke the shell command */
                     cmici.cbSize = sizeof(cmici);
                     cmici.hwnd = hWndOwner;
-                    cmici.lpVerb = (LPCSTR) MAKEINTRESOURCE(uiCmdId - ID_SHELL_CMD_FIRST);
+                    cmici.lpVerb = (LPCSTR) MAKEINTRESOURCEW(uiCmdId - ID_SHELL_CMD_FIRST);
                     cmici.nShow = SW_NORMAL;
 
                     pcm->InvokeCommand(&cmici);