mirror of
https://github.com/reactos/reactos.git
synced 2024-11-10 16:48:16 +00:00
c424146e2c
svn path=/branches/cmake-bringup/; revision=48236
2652 lines
80 KiB
C
2652 lines
80 KiB
C
/*
|
|
* ReactOS Explorer
|
|
*
|
|
* Copyright 2006 - 2007 Thomas Weidenmueller <w3seek@reactos.org>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#include <precomp.h>
|
|
|
|
static const TRAYWINDOW_CTXMENU TrayWindowCtxMenu;
|
|
|
|
#define WM_APP_TRAYDESTROY (WM_APP + 0x100)
|
|
|
|
static LONG TrayWndCount = 0;
|
|
|
|
static const TCHAR szTrayWndClass[] = TEXT("Shell_TrayWnd");
|
|
|
|
static const ITrayWindowVtbl ITrayWindowImpl_Vtbl;
|
|
static const IShellDesktopTrayVtbl IShellDesktopTrayImpl_Vtbl;
|
|
|
|
/*
|
|
* ITrayWindow
|
|
*/
|
|
|
|
const GUID IID_IShellDesktopTray = {0x213e2df9, 0x9a14, 0x4328, {0x99, 0xb1, 0x69, 0x61, 0xf9, 0x14, 0x3c, 0xe9}};
|
|
|
|
typedef struct
|
|
{
|
|
const ITrayWindowVtbl *lpVtbl;
|
|
const IShellDesktopTrayVtbl *lpVtblShellDesktopTray;
|
|
LONG Ref;
|
|
|
|
HWND hWnd;
|
|
HWND hWndDesktop;
|
|
|
|
HWND hwndStart;
|
|
HIMAGELIST himlStartBtn;
|
|
SIZE StartBtnSize;
|
|
HFONT hStartBtnFont;
|
|
HFONT hCaptionFont;
|
|
|
|
ITrayBandSite *TrayBandSite;
|
|
HWND hwndRebar;
|
|
HWND hwndTaskSwitch;
|
|
HWND hwndTrayNotify;
|
|
|
|
DWORD Position;
|
|
HMONITOR Monitor;
|
|
DWORD DraggingPosition;
|
|
HMONITOR DraggingMonitor;
|
|
|
|
RECT rcTrayWnd[4];
|
|
RECT rcNewPosSize;
|
|
SIZE TraySize;
|
|
union
|
|
{
|
|
DWORD Flags;
|
|
struct
|
|
{
|
|
DWORD AutoHide : 1;
|
|
DWORD AlwaysOnTop : 1;
|
|
DWORD SmSmallIcons : 1;
|
|
DWORD HideClock : 1;
|
|
DWORD Locked : 1;
|
|
|
|
/* UI Status */
|
|
DWORD InSizeMove : 1;
|
|
DWORD IsDragging : 1;
|
|
DWORD NewPosSize : 1;
|
|
};
|
|
};
|
|
|
|
NONCLIENTMETRICS ncm;
|
|
HFONT hFont;
|
|
|
|
IMenuBand *StartMenuBand;
|
|
IMenuPopup *StartMenuPopup;
|
|
HBITMAP hbmStartMenu;
|
|
|
|
HWND hWndTrayProperties;
|
|
} ITrayWindowImpl;
|
|
|
|
static IUnknown *
|
|
IUnknown_from_impl(ITrayWindowImpl *This)
|
|
{
|
|
return (IUnknown *)&This->lpVtbl;
|
|
}
|
|
|
|
static ITrayWindow *
|
|
ITrayWindow_from_impl(ITrayWindowImpl *This)
|
|
{
|
|
return (ITrayWindow *)&This->lpVtbl;
|
|
}
|
|
|
|
static IShellDesktopTray *
|
|
IShellDesktopTray_from_impl(ITrayWindowImpl *This)
|
|
{
|
|
return (IShellDesktopTray *)&This->lpVtblShellDesktopTray;
|
|
}
|
|
|
|
static ITrayWindowImpl *
|
|
impl_from_ITrayWindow(ITrayWindow *iface)
|
|
{
|
|
return (ITrayWindowImpl *)((ULONG_PTR)iface - FIELD_OFFSET(ITrayWindowImpl,
|
|
lpVtbl));
|
|
}
|
|
|
|
static ITrayWindowImpl *
|
|
impl_from_IShellDesktopTray(IShellDesktopTray *iface)
|
|
{
|
|
return (ITrayWindowImpl *)((ULONG_PTR)iface - FIELD_OFFSET(ITrayWindowImpl,
|
|
lpVtblShellDesktopTray));
|
|
}
|
|
|
|
/*
|
|
* ITrayWindow
|
|
*/
|
|
|
|
static BOOL
|
|
ITrayWindowImpl_UpdateNonClientMetrics(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
This->ncm.cbSize = sizeof(This->ncm);
|
|
if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS,
|
|
sizeof(This->ncm),
|
|
&This->ncm,
|
|
0))
|
|
{
|
|
if (This->hFont != NULL)
|
|
DeleteObject(This->hFont);
|
|
|
|
This->hFont = CreateFontIndirect(&This->ncm.lfMessageFont);
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_SetWindowsFont(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
if (This->hwndTrayNotify != NULL)
|
|
{
|
|
SendMessage(This->hwndTrayNotify,
|
|
WM_SETFONT,
|
|
(WPARAM)This->hFont,
|
|
TRUE);
|
|
}
|
|
}
|
|
|
|
static HMONITOR
|
|
ITrayWindowImpl_GetScreenRectFromRect(IN OUT ITrayWindowImpl *This,
|
|
IN OUT RECT *pRect,
|
|
IN DWORD dwFlags)
|
|
{
|
|
MONITORINFO mi;
|
|
HMONITOR hMon;
|
|
|
|
mi.cbSize = sizeof(mi);
|
|
hMon = MonitorFromRect(pRect,
|
|
dwFlags);
|
|
if (hMon != NULL &&
|
|
GetMonitorInfo(hMon,
|
|
&mi))
|
|
{
|
|
*pRect = mi.rcMonitor;
|
|
}
|
|
else
|
|
{
|
|
pRect->left = 0;
|
|
pRect->top = 0;
|
|
pRect->right = GetSystemMetrics(SM_CXSCREEN);
|
|
pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
|
|
|
|
hMon = NULL;
|
|
}
|
|
|
|
return hMon;
|
|
}
|
|
|
|
static HMONITOR
|
|
ITrayWindowImpl_GetMonitorFromRect(IN OUT ITrayWindowImpl *This,
|
|
IN const RECT *pRect)
|
|
{
|
|
HMONITOR hMon;
|
|
|
|
/* 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;
|
|
|
|
pt.x = pRect->left + ((pRect->right - pRect->left) / 2);
|
|
pt.y = pRect->top + ((pRect->bottom - pRect->top) / 2);
|
|
|
|
/* be less error-prone, find the nearest monitor */
|
|
hMon = MonitorFromPoint(pt,
|
|
MONITOR_DEFAULTTONEAREST);
|
|
}
|
|
|
|
return hMon;
|
|
}
|
|
|
|
static HMONITOR
|
|
ITrayWindowImpl_GetScreenRect(IN OUT ITrayWindowImpl *This,
|
|
IN HMONITOR hMonitor,
|
|
IN OUT RECT *pRect)
|
|
{
|
|
HMONITOR hMon = NULL;
|
|
|
|
if (hMonitor != NULL)
|
|
{
|
|
MONITORINFO mi;
|
|
|
|
mi.cbSize = sizeof(mi);
|
|
if (!GetMonitorInfo(This->Monitor,
|
|
&mi))
|
|
{
|
|
/* Hm, the monitor is gone? Try to find a monitor where it
|
|
could be located now */
|
|
hMon = ITrayWindowImpl_GetMonitorFromRect(This,
|
|
pRect);
|
|
if (hMon == NULL ||
|
|
!GetMonitorInfo(hMon,
|
|
&mi))
|
|
{
|
|
hMon = NULL;
|
|
goto GetPrimaryRect;
|
|
}
|
|
}
|
|
|
|
*pRect = mi.rcMonitor;
|
|
}
|
|
else
|
|
{
|
|
GetPrimaryRect:
|
|
pRect->left = 0;
|
|
pRect->top = 0;
|
|
pRect->right = GetSystemMetrics(SM_CXSCREEN);
|
|
pRect->bottom = GetSystemMetrics(SM_CYSCREEN);
|
|
}
|
|
|
|
return hMon;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_MakeTrayRectWithSize(IN DWORD Position,
|
|
IN const SIZE *pTraySize,
|
|
IN OUT RECT *pRect)
|
|
{
|
|
switch (Position)
|
|
{
|
|
case ABE_LEFT:
|
|
pRect->right = pRect->left + pTraySize->cx;
|
|
break;
|
|
|
|
case ABE_TOP:
|
|
pRect->bottom = pRect->top + pTraySize->cy;
|
|
break;
|
|
|
|
case ABE_RIGHT:
|
|
pRect->left = pRect->right - pTraySize->cx;
|
|
break;
|
|
|
|
case ABE_BOTTOM:
|
|
default:
|
|
pRect->top = pRect->bottom - pTraySize->cy;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_GetTrayRectFromScreenRect(IN OUT ITrayWindowImpl *This,
|
|
IN DWORD Position,
|
|
IN const RECT *pScreen,
|
|
IN const SIZE *pTraySize OPTIONAL,
|
|
OUT RECT *pRect)
|
|
{
|
|
if (pTraySize == NULL)
|
|
pTraySize = &This->TraySize;
|
|
|
|
*pRect = *pScreen;
|
|
|
|
/* Move the border outside of the screen */
|
|
InflateRect(pRect,
|
|
GetSystemMetrics(SM_CXEDGE),
|
|
GetSystemMetrics(SM_CYEDGE));
|
|
|
|
ITrayWindowImpl_MakeTrayRectWithSize(Position,
|
|
pTraySize,
|
|
pRect);
|
|
}
|
|
|
|
BOOL
|
|
ITrayWindowImpl_IsPosHorizontal(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
return This->Position == ABE_TOP || This->Position == ABE_BOTTOM;
|
|
}
|
|
|
|
static HMONITOR
|
|
ITrayWindowImpl_CalculateValidSize(IN OUT ITrayWindowImpl *This,
|
|
IN DWORD Position,
|
|
IN OUT RECT *pRect)
|
|
{
|
|
RECT rcScreen;
|
|
BOOL Horizontal;
|
|
HMONITOR hMon;
|
|
SIZE szMax, szWnd;
|
|
|
|
Horizontal = ITrayWindowImpl_IsPosHorizontal(This);
|
|
|
|
szWnd.cx = pRect->right - pRect->left;
|
|
szWnd.cy = pRect->bottom - pRect->top;
|
|
|
|
rcScreen = *pRect;
|
|
hMon = ITrayWindowImpl_GetScreenRectFromRect(This,
|
|
&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 */
|
|
|
|
ITrayWindowImpl_GetTrayRectFromScreenRect(This,
|
|
Position,
|
|
&rcScreen,
|
|
&szWnd,
|
|
pRect);
|
|
|
|
return hMon;
|
|
}
|
|
|
|
#if 0
|
|
static VOID
|
|
ITrayWindowImpl_GetMinimumWindowSize(IN OUT ITrayWindowImpl *This,
|
|
OUT RECT *pRect)
|
|
{
|
|
RECT rcMin = {0};
|
|
|
|
AdjustWindowRectEx(&rcMin,
|
|
GetWindowLongPtr(This->hWnd,
|
|
GWL_STYLE),
|
|
FALSE,
|
|
GetWindowLongPtr(This->hWnd,
|
|
GWL_EXSTYLE));
|
|
|
|
*pRect = rcMin;
|
|
}
|
|
#endif
|
|
|
|
|
|
static DWORD
|
|
ITrayWindowImpl_GetDraggingRectFromPt(IN OUT ITrayWindowImpl *This,
|
|
IN POINT pt,
|
|
OUT RECT *pRect,
|
|
OUT HMONITOR *phMonitor)
|
|
{
|
|
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);
|
|
|
|
if (hMon != NULL)
|
|
{
|
|
MONITORINFO mi;
|
|
|
|
mi.cbSize = sizeof(mi);
|
|
if (!GetMonitorInfo(hMon,
|
|
&mi))
|
|
{
|
|
hMon = NULL;
|
|
goto GetPrimaryScreenRect;
|
|
}
|
|
|
|
/* 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);
|
|
|
|
hMonNew = ITrayWindowImpl_GetMonitorFromRect(This,
|
|
&This->rcTrayWnd[Pos]);
|
|
if (hMon != hMonNew)
|
|
{
|
|
SIZE szTray;
|
|
|
|
/* Recalculate the rectangle, we're dragging to another monitor.
|
|
We don't need to recalculate the rect on single monitor systems. */
|
|
szTray.cx = This->rcTrayWnd[Pos].right - This->rcTrayWnd[Pos].left;
|
|
szTray.cy = This->rcTrayWnd[Pos].bottom - This->rcTrayWnd[Pos].top;
|
|
|
|
ITrayWindowImpl_GetTrayRectFromScreenRect(This,
|
|
Pos,
|
|
&rcScreen,
|
|
&szTray,
|
|
pRect);
|
|
|
|
hMon = hMonNew;
|
|
}
|
|
else
|
|
{
|
|
/* The user is dragging the tray window on the same monitor. We don't need
|
|
to recalculate the rectangle */
|
|
*pRect = This->rcTrayWnd[Pos];
|
|
}
|
|
|
|
*phMonitor = hMon;
|
|
|
|
return Pos;
|
|
}
|
|
|
|
static DWORD
|
|
ITrayWindowImpl_GetDraggingRectFromRect(IN OUT ITrayWindowImpl *This,
|
|
IN OUT RECT *pRect,
|
|
OUT HMONITOR *phMonitor)
|
|
{
|
|
POINT pt;
|
|
|
|
/* Calculate the center of the rectangle. We call
|
|
ITrayWindowImpl_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 ITrayWindowImpl_GetDraggingRectFromPt(This,
|
|
pt,
|
|
pRect,
|
|
phMonitor);
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_ChangingWinPos(IN OUT ITrayWindowImpl *This,
|
|
IN OUT LPWINDOWPOS pwp)
|
|
{
|
|
RECT rcTray;
|
|
|
|
if (This->IsDragging)
|
|
{
|
|
rcTray.left = pwp->x;
|
|
rcTray.top = pwp->y;
|
|
rcTray.right = rcTray.left + pwp->cx;
|
|
rcTray.bottom = rcTray.top + pwp->cy;
|
|
|
|
if (!EqualRect(&rcTray,
|
|
&This->rcTrayWnd[This->DraggingPosition]))
|
|
{
|
|
/* Recalculate the rectangle, the user dragged the tray
|
|
window to another monitor or the window was somehow else
|
|
moved or resized */
|
|
This->DraggingPosition = ITrayWindowImpl_GetDraggingRectFromRect(This,
|
|
&rcTray,
|
|
&This->DraggingMonitor);
|
|
//This->rcTrayWnd[This->DraggingPosition] = rcTray;
|
|
}
|
|
|
|
//This->Monitor = ITrayWindowImpl_CalculateValidSize(This,
|
|
// This->DraggingPosition,
|
|
// &rcTray);
|
|
|
|
This->Monitor = This->DraggingMonitor;
|
|
This->Position = This->DraggingPosition;
|
|
This->IsDragging = FALSE;
|
|
|
|
This->rcTrayWnd[This->Position] = rcTray;
|
|
goto ChangePos;
|
|
}
|
|
else if (GetWindowRect(This->hWnd,
|
|
&rcTray))
|
|
{
|
|
if (This->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;
|
|
}
|
|
|
|
This->Position = ITrayWindowImpl_GetDraggingRectFromRect(This,
|
|
&rcTray,
|
|
&This->Monitor);
|
|
|
|
if (!(pwp->flags & (SWP_NOMOVE | SWP_NOSIZE)))
|
|
{
|
|
SIZE szWnd;
|
|
|
|
szWnd.cx = pwp->cx;
|
|
szWnd.cy = pwp->cy;
|
|
|
|
ITrayWindowImpl_MakeTrayRectWithSize(This->Position,
|
|
&szWnd,
|
|
&rcTray);
|
|
}
|
|
|
|
This->rcTrayWnd[This->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 = This->rcTrayWnd[This->Position];
|
|
}
|
|
|
|
ChangePos:
|
|
This->TraySize.cx = rcTray.right - rcTray.left;
|
|
This->TraySize.cy = rcTray.bottom - rcTray.top;
|
|
|
|
pwp->flags &= ~(SWP_NOMOVE | SWP_NOSIZE);
|
|
pwp->x = rcTray.left;
|
|
pwp->y = rcTray.top;
|
|
pwp->cx = This->TraySize.cx;
|
|
pwp->cy = This->TraySize.cy;
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_ApplyClipping(IN OUT ITrayWindowImpl *This,
|
|
IN BOOL Clip)
|
|
{
|
|
RECT rcClip, rcWindow;
|
|
HRGN hClipRgn;
|
|
|
|
if (GetWindowRect(This->hWnd,
|
|
&rcWindow))
|
|
{
|
|
/* Disable clipping on systems with only one monitor */
|
|
if (GetSystemMetrics(SM_CMONITORS) <= 1)
|
|
Clip = FALSE;
|
|
|
|
if (Clip)
|
|
{
|
|
rcClip = rcWindow;
|
|
|
|
ITrayWindowImpl_GetScreenRect(This,
|
|
This->Monitor,
|
|
&rcClip);
|
|
|
|
if (!IntersectRect(&rcClip,
|
|
&rcClip,
|
|
&rcWindow))
|
|
{
|
|
rcClip = rcWindow;
|
|
}
|
|
|
|
OffsetRect(&rcClip,
|
|
-rcWindow.left,
|
|
-rcWindow.top);
|
|
|
|
hClipRgn = CreateRectRgnIndirect(&rcClip);
|
|
}
|
|
else
|
|
hClipRgn = NULL;
|
|
|
|
/* Set the clipping region or make sure the window isn't clipped
|
|
by disabling it explicitly. */
|
|
SetWindowRgn(This->hWnd,
|
|
hClipRgn,
|
|
TRUE);
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_CheckTrayWndPosition(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
RECT rcTray;
|
|
|
|
rcTray = This->rcTrayWnd[This->Position];
|
|
// DbgPrint("CheckTray: %d: %d,%d,%d,%d\n", This->Position, rcTray.left, rcTray.top, rcTray.right, rcTray.bottom);
|
|
|
|
/* Move the tray window */
|
|
SetWindowPos(This->hWnd,
|
|
NULL,
|
|
rcTray.left,
|
|
rcTray.top,
|
|
rcTray.right - rcTray.left,
|
|
rcTray.bottom - rcTray.top,
|
|
SWP_NOZORDER);
|
|
|
|
ITrayWindowImpl_ApplyClipping(This,
|
|
TRUE);
|
|
}
|
|
|
|
typedef struct _TW_STUCKRECTS2
|
|
{
|
|
DWORD cbSize;
|
|
LONG Unknown;
|
|
DWORD dwFlags;
|
|
DWORD Position;
|
|
SIZE Size;
|
|
RECT Rect;
|
|
} TW_STRUCKRECTS2, *PTW_STUCKRECTS2;
|
|
|
|
static VOID
|
|
ITrayWindowImpl_RegLoadSettings(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
DWORD Pos;
|
|
TW_STRUCKRECTS2 sr;
|
|
RECT rcScreen;
|
|
SIZE WndSize, EdgeSize, DlgFrameSize;
|
|
DWORD cbSize = sizeof(sr);
|
|
|
|
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))
|
|
{
|
|
This->AutoHide = (sr.dwFlags & ABS_AUTOHIDE) != 0;
|
|
This->AlwaysOnTop = (sr.dwFlags & ABS_ALWAYSONTOP) != 0;
|
|
This->SmSmallIcons = (sr.dwFlags & 0x4) != 0;
|
|
This->HideClock = (sr.dwFlags & 0x8) != 0;
|
|
|
|
/* FIXME: Are there more flags? */
|
|
|
|
if (This->hWnd != NULL)
|
|
SetWindowPos (This->hWnd,
|
|
This->AlwaysOnTop ? HWND_TOPMOST : HWND_NOTOPMOST,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
SWP_NOMOVE | SWP_NOSIZE);
|
|
|
|
if (sr.Position > ABE_BOTTOM)
|
|
This->Position = ABE_BOTTOM;
|
|
else
|
|
This->Position = sr.Position;
|
|
|
|
/* 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;
|
|
ITrayWindowImpl_GetScreenRectFromRect(This,
|
|
&rcScreen,
|
|
MONITOR_DEFAULTTONEAREST);
|
|
}
|
|
else
|
|
{
|
|
This->Position = ABE_BOTTOM;
|
|
|
|
/* 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 = This->StartBtnSize.cx + (2 * (EdgeSize.cx + DlgFrameSize.cx));
|
|
sr.Size.cy = This->StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
|
|
|
|
/* Use the primary screen by default */
|
|
rcScreen.left = 0;
|
|
rcScreen.top = 0;
|
|
rcScreen.right = GetSystemMetrics(SM_CXSCREEN);
|
|
rcScreen.right = GetSystemMetrics(SM_CYSCREEN);
|
|
ITrayWindowImpl_GetScreenRectFromRect(This,
|
|
&rcScreen,
|
|
MONITOR_DEFAULTTOPRIMARY);
|
|
}
|
|
|
|
/* 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 = This->StartBtnSize.cy + (2 * (EdgeSize.cy + DlgFrameSize.cy));
|
|
|
|
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 */
|
|
This->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++)
|
|
{
|
|
ITrayWindowImpl_GetTrayRectFromScreenRect(This,
|
|
Pos,
|
|
&rcScreen,
|
|
&This->TraySize,
|
|
&This->rcTrayWnd[Pos]);
|
|
// DbgPrint("rcTrayWnd[%d(%d)]: %d,%d,%d,%d\n", Pos, This->Position, This->rcTrayWnd[Pos].left, This->rcTrayWnd[Pos].top, This->rcTrayWnd[Pos].right, This->rcTrayWnd[Pos].bottom);
|
|
}
|
|
|
|
/* Determine which monitor we are on. It shouldn't matter which docked
|
|
position rectangle we use */
|
|
This->Monitor = ITrayWindowImpl_GetMonitorFromRect(This,
|
|
&This->rcTrayWnd[ABE_LEFT]);
|
|
}
|
|
|
|
static UINT
|
|
ITrayWindowImpl_TrackMenu(IN OUT ITrayWindowImpl *This,
|
|
IN HMENU hMenu,
|
|
IN POINT *ppt OPTIONAL,
|
|
IN HWND hwndExclude OPTIONAL,
|
|
IN BOOL TrackUp,
|
|
IN BOOL IsContextMenu)
|
|
{
|
|
TPMPARAMS tmp, *ptmp = NULL;
|
|
POINT pt;
|
|
UINT cmdId;
|
|
UINT fuFlags;
|
|
|
|
if (hwndExclude != NULL)
|
|
{
|
|
/* Get the client rectangle and map it to screen coordinates */
|
|
if (GetClientRect(hwndExclude,
|
|
&tmp.rcExclude) &&
|
|
MapWindowPoints(hwndExclude,
|
|
NULL,
|
|
(LPPOINT)&tmp.rcExclude,
|
|
2) != 0)
|
|
{
|
|
ptmp = &tmp;
|
|
}
|
|
}
|
|
|
|
if (ppt == NULL)
|
|
{
|
|
if (ptmp == NULL &&
|
|
GetClientRect(This->hWnd,
|
|
&tmp.rcExclude) &&
|
|
MapWindowPoints(This->hWnd,
|
|
NULL,
|
|
(LPPOINT)&tmp.rcExclude,
|
|
2) != 0)
|
|
{
|
|
ptmp = &tmp;
|
|
}
|
|
|
|
if (ptmp != NULL)
|
|
{
|
|
/* NOTE: TrackPopupMenuEx will eventually align the track position
|
|
for us, no need to take care of it here as long as the
|
|
coordinates are somewhere within the exclusion rectangle */
|
|
pt.x = ptmp->rcExclude.left;
|
|
pt.y = ptmp->rcExclude.top;
|
|
}
|
|
else
|
|
pt.x = pt.y = 0;
|
|
}
|
|
else
|
|
pt = *ppt;
|
|
|
|
tmp.cbSize = sizeof(tmp);
|
|
|
|
fuFlags = TPM_RETURNCMD | TPM_VERTICAL;
|
|
fuFlags |= (TrackUp ? TPM_BOTTOMALIGN : TPM_TOPALIGN);
|
|
if (IsContextMenu)
|
|
fuFlags |= TPM_RIGHTBUTTON;
|
|
else
|
|
fuFlags |= (TrackUp ? TPM_VERNEGANIMATION : TPM_VERPOSANIMATION);
|
|
|
|
cmdId = TrackPopupMenuEx(hMenu,
|
|
fuFlags,
|
|
pt.x,
|
|
pt.y,
|
|
This->hWnd,
|
|
ptmp);
|
|
|
|
return cmdId;
|
|
}
|
|
|
|
static UINT
|
|
ITrayWindowImpl_TrackCtxMenu(IN OUT ITrayWindowImpl *This,
|
|
IN const TRAYWINDOW_CTXMENU *pMenu,
|
|
IN POINT *ppt OPTIONAL,
|
|
IN HWND hwndExclude OPTIONAL,
|
|
IN BOOL TrackUp,
|
|
IN PVOID Context OPTIONAL)
|
|
{
|
|
HMENU hPopup;
|
|
UINT cmdId = 0;
|
|
PVOID pcmContext = NULL;
|
|
|
|
hPopup = pMenu->CreateCtxMenu(This->hWnd,
|
|
&pcmContext,
|
|
Context);
|
|
if (hPopup != NULL)
|
|
{
|
|
cmdId = ITrayWindowImpl_TrackMenu(This,
|
|
hPopup,
|
|
ppt,
|
|
hwndExclude,
|
|
TrackUp,
|
|
TRUE);
|
|
|
|
pMenu->CtxMenuCommand(This->hWnd,
|
|
cmdId,
|
|
pcmContext,
|
|
Context);
|
|
}
|
|
|
|
return cmdId;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_Free(ITrayWindowImpl *This)
|
|
{
|
|
HeapFree(hProcessHeap,
|
|
0,
|
|
This);
|
|
}
|
|
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayWindowImpl_Release(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
ULONG Ret;
|
|
|
|
Ret = InterlockedDecrement(&This->Ref);
|
|
if (Ret == 0)
|
|
ITrayWindowImpl_Free(This);
|
|
|
|
return Ret;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_Destroy(ITrayWindowImpl *This)
|
|
{
|
|
(void)InterlockedExchangePointer((PVOID*)&This->hWnd,
|
|
NULL);
|
|
|
|
if (This->himlStartBtn != NULL)
|
|
{
|
|
ImageList_Destroy(This->himlStartBtn);
|
|
This->himlStartBtn = NULL;
|
|
}
|
|
|
|
if (This->hCaptionFont != NULL)
|
|
{
|
|
DeleteObject(This->hCaptionFont);
|
|
This->hCaptionFont = NULL;
|
|
}
|
|
|
|
if (This->hStartBtnFont != NULL)
|
|
{
|
|
DeleteObject(This->hStartBtnFont);
|
|
This->hStartBtnFont = NULL;
|
|
}
|
|
|
|
if (This->hFont != NULL)
|
|
{
|
|
DeleteObject(This->hFont);
|
|
This->hFont = NULL;
|
|
}
|
|
|
|
if (This->StartMenuPopup != NULL)
|
|
{
|
|
IMenuPopup_Release(This->StartMenuPopup);
|
|
This->StartMenuPopup = NULL;
|
|
}
|
|
|
|
if (This->hbmStartMenu != NULL)
|
|
{
|
|
DeleteObject(This->hbmStartMenu);
|
|
This->hbmStartMenu = NULL;
|
|
}
|
|
|
|
if (This->StartMenuBand != NULL)
|
|
{
|
|
IMenuBand_Release(This->StartMenuBand);
|
|
This->StartMenuBand = NULL;
|
|
}
|
|
|
|
if (This->TrayBandSite != NULL)
|
|
{
|
|
/* FIXME: Unload bands */
|
|
ITrayBandSite_Release(This->TrayBandSite);
|
|
This->TrayBandSite = NULL;
|
|
}
|
|
|
|
ITrayWindowImpl_Release(ITrayWindow_from_impl(This));
|
|
|
|
if (InterlockedDecrement(&TrayWndCount) == 0)
|
|
PostQuitMessage(0);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayWindowImpl_AddRef(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
return InterlockedIncrement(&This->Ref);
|
|
}
|
|
|
|
|
|
static BOOL
|
|
ITrayWindowImpl_NCCreate(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
ITrayWindowImpl_AddRef(ITrayWindow_from_impl(This));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_UpdateStartButton(IN OUT ITrayWindowImpl *This,
|
|
IN HBITMAP hbmStart OPTIONAL)
|
|
{
|
|
SIZE Size = { 0, 0 };
|
|
|
|
if (This->himlStartBtn == NULL ||
|
|
!SendMessage(This->hwndStart,
|
|
BCM_GETIDEALSIZE,
|
|
0,
|
|
(LPARAM)&Size))
|
|
{
|
|
Size.cx = GetSystemMetrics(SM_CXEDGE);
|
|
Size.cy = GetSystemMetrics(SM_CYEDGE);
|
|
|
|
if (hbmStart == NULL)
|
|
{
|
|
hbmStart = (HBITMAP)SendMessage(This->hwndStart,
|
|
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);
|
|
}
|
|
}
|
|
|
|
/* Save the size of the start button */
|
|
This->StartBtnSize = Size;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_AlignControls(IN OUT ITrayWindowImpl *This,
|
|
IN PRECT prcClient OPTIONAL)
|
|
{
|
|
RECT rcClient;
|
|
SIZE TraySize, StartSize;
|
|
POINT ptTrayNotify = { 0, 0 };
|
|
BOOL Horizontal;
|
|
HDWP dwp;
|
|
|
|
if (prcClient != NULL)
|
|
{
|
|
rcClient = *prcClient;
|
|
}
|
|
else
|
|
{
|
|
if (!GetClientRect(This->hWnd,
|
|
&rcClient))
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
Horizontal = ITrayWindowImpl_IsPosHorizontal(This);
|
|
|
|
/* We're about to resize/move the start button, the rebar control and
|
|
the tray notification control */
|
|
dwp = BeginDeferWindowPos(3);
|
|
if (dwp == NULL)
|
|
return;
|
|
|
|
/* Limit the Start button width to the client width, if neccessary */
|
|
StartSize = This->StartBtnSize;
|
|
if (StartSize.cx > rcClient.right)
|
|
StartSize.cx = rcClient.right;
|
|
|
|
if (This->hwndStart != NULL)
|
|
{
|
|
/* Resize and reposition the button */
|
|
dwp = DeferWindowPos(dwp,
|
|
This->hwndStart,
|
|
NULL,
|
|
0,
|
|
0,
|
|
StartSize.cx,
|
|
StartSize.cy,
|
|
SWP_NOZORDER | SWP_NOACTIVATE);
|
|
if (dwp == NULL)
|
|
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 (This->hwndTrayNotify != NULL &&
|
|
SendMessage(This->hwndTrayNotify,
|
|
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,
|
|
This->hwndTrayNotify,
|
|
NULL,
|
|
ptTrayNotify.x,
|
|
ptTrayNotify.y,
|
|
TraySize.cx,
|
|
TraySize.cy,
|
|
SWP_NOZORDER | SWP_NOACTIVATE);
|
|
if (dwp == NULL)
|
|
return;
|
|
}
|
|
|
|
/* Resize/Move the rebar control */
|
|
if (This->hwndRebar != NULL)
|
|
{
|
|
POINT ptRebar = { 0, 0 };
|
|
SIZE szRebar;
|
|
|
|
SetWindowStyle(This->hwndRebar,
|
|
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;
|
|
}
|
|
|
|
dwp = DeferWindowPos(dwp,
|
|
This->hwndRebar,
|
|
NULL,
|
|
ptRebar.x,
|
|
ptRebar.y,
|
|
szRebar.cx,
|
|
szRebar.cy,
|
|
SWP_NOZORDER | SWP_NOACTIVATE);
|
|
}
|
|
|
|
if (dwp != NULL)
|
|
EndDeferWindowPos(dwp);
|
|
|
|
if (This->hwndTaskSwitch != NULL)
|
|
{
|
|
/* Update the task switch window configuration */
|
|
SendMessage(This->hwndTaskSwitch,
|
|
TSWM_UPDATETASKBARPOS,
|
|
0,
|
|
0);
|
|
}
|
|
}
|
|
|
|
static BOOL
|
|
ITrayWindowImpl_CreateStartBtnImageList(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
HICON hIconStart;
|
|
SIZE IconSize;
|
|
|
|
if (This->himlStartBtn != 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 = LoadImage(hExplorerInstance,
|
|
MAKEINTRESOURCE(IDI_START),
|
|
IMAGE_ICON,
|
|
IconSize.cx,
|
|
IconSize.cy,
|
|
LR_SHARED | LR_DEFAULTCOLOR);
|
|
|
|
if (hIconStart != NULL)
|
|
{
|
|
This->himlStartBtn = ImageList_Create(IconSize.cx,
|
|
IconSize.cy,
|
|
ILC_COLOR32 | ILC_MASK,
|
|
1,
|
|
1);
|
|
if (This->himlStartBtn != NULL)
|
|
{
|
|
if (ImageList_AddIcon(This->himlStartBtn,
|
|
hIconStart) >= 0)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
/* Failed to add the icon! */
|
|
ImageList_Destroy(This->himlStartBtn);
|
|
This->himlStartBtn = NULL;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static HBITMAP
|
|
ITrayWindowImpl_CreateStartButtonBitmap(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
TCHAR 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 = 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 = SelectObject(hDC,
|
|
This->hStartBtnFont);
|
|
|
|
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 = SelectObject(hDC,
|
|
hbmp);
|
|
|
|
Flags = DC_TEXT | DC_INBUTTON;
|
|
if (hIconStart != NULL)
|
|
Flags |= DC_ICON;
|
|
|
|
if (DrawCapTemp != NULL)
|
|
{
|
|
Ret = DrawCapTemp(NULL,
|
|
hDC,
|
|
&rcButton,
|
|
This->hStartBtnFont,
|
|
hIconStart,
|
|
szStartCaption,
|
|
Flags);
|
|
}
|
|
|
|
SelectObject(hDC,
|
|
hbmpOld);
|
|
|
|
if (!Ret)
|
|
goto Cleanup;
|
|
|
|
/* We successfully created the bitmap! */
|
|
hBitmap = hbmp;
|
|
hbmp = NULL;
|
|
|
|
Cleanup:
|
|
if (hDCScreen != NULL)
|
|
{
|
|
ReleaseDC(NULL,
|
|
hDCScreen);
|
|
}
|
|
|
|
if (hbmp != NULL)
|
|
DeleteObject(hbmp);
|
|
|
|
if (hDC != NULL)
|
|
DeleteDC(hDC);
|
|
|
|
return hBitmap;
|
|
}
|
|
|
|
static VOID
|
|
ITrayWindowImpl_Create(IN OUT ITrayWindowImpl *This)
|
|
{
|
|
TCHAR szStartCaption[32];
|
|
|
|
InterlockedIncrement(&TrayWndCount);
|
|
|
|
if (!LoadString(hExplorerInstance,
|
|
IDS_START,
|
|
szStartCaption,
|
|
sizeof(szStartCaption) / sizeof(szStartCaption[0])))
|
|
{
|
|
szStartCaption[0] = TEXT('\0');
|
|
}
|
|
|
|
if (This->hStartBtnFont == NULL || This->hCaptionFont == NULL)
|
|
{
|
|
NONCLIENTMETRICS ncm;
|
|
|
|
/* Get the system fonts, we use the caption font,
|
|
always bold, though. */
|
|
ncm.cbSize = sizeof(ncm);
|
|
if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS,
|
|
sizeof(ncm),
|
|
&ncm,
|
|
FALSE))
|
|
{
|
|
if (This->hCaptionFont == NULL)
|
|
{
|
|
ncm.lfCaptionFont.lfWeight = FW_NORMAL;
|
|
This->hCaptionFont = CreateFontIndirect(&ncm.lfCaptionFont);
|
|
}
|
|
|
|
if (This->hStartBtnFont == NULL)
|
|
{
|
|
ncm.lfCaptionFont.lfWeight = FW_BOLD;
|
|
This->hStartBtnFont = CreateFontIndirect(&ncm.lfCaptionFont);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Create the Start button */
|
|
This->hwndStart = CreateWindowEx(0,
|
|
WC_BUTTON,
|
|
szStartCaption,
|
|
WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS |
|
|
BS_PUSHBUTTON | BS_CENTER | BS_VCENTER | BS_BITMAP,
|
|
0,
|
|
0,
|
|
0,
|
|
0,
|
|
This->hWnd,
|
|
(HMENU)IDC_STARTBTN,
|
|
hExplorerInstance,
|
|
NULL);
|
|
if (This->hwndStart)
|
|
{
|
|
SendMessage(This->hwndStart,
|
|
WM_SETFONT,
|
|
(WPARAM)This->hStartBtnFont,
|
|
FALSE);
|
|
|
|
if (ITrayWindowImpl_CreateStartBtnImageList(This))
|
|
{
|
|
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 = This->himlStartBtn;
|
|
bil.margin.left = bil.margin.right = 1;
|
|
bil.margin.top = bil.margin.bottom = 1;
|
|
bil.uAlign = BUTTON_IMAGELIST_ALIGN_LEFT;
|
|
|
|
if (!SendMessage(This->hwndStart,
|
|
BCM_SETIMAGELIST,
|
|
0,
|
|
(LPARAM)&bil))
|
|
{
|
|
/* Fall back to the deprecated method on older systems that don't
|
|
support Common Controls 6.0 */
|
|
ImageList_Destroy(This->himlStartBtn);
|
|
This->himlStartBtn = NULL;
|
|
|
|
goto SetStartBtnImage;
|
|
}
|
|
|
|
/* We're using the image list, remove the BS_BITMAP style and
|
|
don't center it horizontally */
|
|
SetWindowStyle(This->hwndStart,
|
|
BS_BITMAP | BS_RIGHT,
|
|
0);
|
|
|
|
ITrayWindowImpl_UpdateStartButton(This,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
HBITMAP hbmStart, hbmOld;
|
|
|
|
SetStartBtnImage:
|
|
hbmStart = ITrayWindowImpl_CreateStartButtonBitmap(This);
|
|
if (hbmStart != NULL)
|
|
{
|
|
ITrayWindowImpl_UpdateStartButton(This,
|
|
hbmStart);
|
|
|
|
hbmOld = (HBITMAP)SendMessage(This->hwndStart,
|
|
BM_SETIMAGE,
|
|
IMAGE_BITMAP,
|
|
(LPARAM)hbmStart);
|
|
|
|
if (hbmOld != NULL)
|
|
DeleteObject(hbmOld);
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Load the saved tray window settings */
|
|
ITrayWindowImpl_RegLoadSettings(This);
|
|
|
|
/* Create and initialize the start menu */
|
|
This->hbmStartMenu = LoadBitmap(hExplorerInstance,
|
|
MAKEINTRESOURCE(IDB_STARTMENU));
|
|
This->StartMenuPopup = CreateStartMenu(ITrayWindow_from_impl(This),
|
|
&This->StartMenuBand,
|
|
This->hbmStartMenu,
|
|
0);
|
|
|
|
/* Load the tray band site */
|
|
if (This->TrayBandSite != NULL)
|
|
{
|
|
ITrayBandSite_Release(This->TrayBandSite);
|
|
}
|
|
|
|
This->TrayBandSite = CreateTrayBandSite(ITrayWindow_from_impl(This),
|
|
&This->hwndRebar,
|
|
&This->hwndTaskSwitch);
|
|
|
|
/* Create the tray notification window */
|
|
This->hwndTrayNotify = CreateTrayNotifyWnd(ITrayWindow_from_impl(This),
|
|
This->HideClock);
|
|
|
|
if (ITrayWindowImpl_UpdateNonClientMetrics(This))
|
|
{
|
|
ITrayWindowImpl_SetWindowsFont(This);
|
|
}
|
|
|
|
/* Move the tray window to the right position and resize it if neccessary */
|
|
ITrayWindowImpl_CheckTrayWndPosition(This);
|
|
|
|
/* Align all controls on the tray window */
|
|
ITrayWindowImpl_AlignControls(This,
|
|
NULL);
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_QueryInterface(IN OUT ITrayWindow *iface,
|
|
IN REFIID riid,
|
|
OUT LPVOID *ppvObj)
|
|
{
|
|
ITrayWindowImpl *This;
|
|
|
|
if (ppvObj == NULL)
|
|
return E_POINTER;
|
|
|
|
This = impl_from_ITrayWindow(iface);
|
|
|
|
if (IsEqualIID(riid,
|
|
&IID_IUnknown))
|
|
{
|
|
*ppvObj = IUnknown_from_impl(This);
|
|
}
|
|
else if (IsEqualIID(riid,
|
|
&IID_IShellDesktopTray))
|
|
{
|
|
*ppvObj = IShellDesktopTray_from_impl(This);
|
|
}
|
|
else
|
|
{
|
|
*ppvObj = NULL;
|
|
return E_NOINTERFACE;
|
|
}
|
|
|
|
ITrayWindowImpl_AddRef(iface);
|
|
return S_OK;
|
|
}
|
|
|
|
static ITrayWindowImpl *
|
|
ITrayWindowImpl_Construct(VOID)
|
|
{
|
|
ITrayWindowImpl *This;
|
|
|
|
This = HeapAlloc(hProcessHeap,
|
|
0,
|
|
sizeof(*This));
|
|
if (This == NULL)
|
|
return NULL;
|
|
|
|
ZeroMemory(This,
|
|
sizeof(*This));
|
|
This->lpVtbl = &ITrayWindowImpl_Vtbl;
|
|
This->lpVtblShellDesktopTray = &IShellDesktopTrayImpl_Vtbl;
|
|
This->Ref = 1;
|
|
This->Position = (DWORD)-1;
|
|
|
|
return This;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_Open(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
HRESULT Ret = S_OK;
|
|
HWND hWnd;
|
|
DWORD dwExStyle;
|
|
|
|
/* 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 (This->hWnd != NULL)
|
|
{
|
|
if (IsWindow(This->hWnd))
|
|
{
|
|
if (!IsWindowVisible(This->hWnd))
|
|
{
|
|
ITrayWindowImpl_CheckTrayWndPosition(This);
|
|
|
|
ShowWindow(This->hWnd,
|
|
SW_SHOW);
|
|
}
|
|
}
|
|
else
|
|
goto TryCreateTrayWnd;
|
|
}
|
|
else
|
|
{
|
|
RECT rcWnd;
|
|
|
|
TryCreateTrayWnd:
|
|
dwExStyle = WS_EX_TOOLWINDOW | WS_EX_WINDOWEDGE;
|
|
if (This->AlwaysOnTop)
|
|
dwExStyle |= WS_EX_TOPMOST;
|
|
|
|
if (This->Position != (DWORD)-1)
|
|
rcWnd = This->rcTrayWnd[This->Position];
|
|
else
|
|
{
|
|
ZeroMemory(&rcWnd,
|
|
sizeof(rcWnd));
|
|
}
|
|
|
|
hWnd = CreateWindowEx(dwExStyle,
|
|
szTrayWndClass,
|
|
NULL,
|
|
WS_POPUP | WS_VISIBLE | WS_CLIPSIBLINGS | WS_CLIPCHILDREN |
|
|
WS_BORDER | WS_THICKFRAME,
|
|
rcWnd.left,
|
|
rcWnd.top,
|
|
rcWnd.right - rcWnd.left,
|
|
rcWnd.bottom - rcWnd.top,
|
|
NULL,
|
|
NULL,
|
|
hExplorerInstance,
|
|
This);
|
|
if (hWnd == NULL)
|
|
Ret = E_FAIL;
|
|
}
|
|
|
|
return Ret;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_Close(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
if (This->hWnd != NULL)
|
|
{
|
|
SendMessage(This->hWnd,
|
|
WM_APP_TRAYDESTROY,
|
|
0,
|
|
0);
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
static HWND STDMETHODCALLTYPE
|
|
ITrayWindowImpl_GetHWND(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
return This->hWnd;
|
|
}
|
|
|
|
static BOOL STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IsSpecialHWND(IN OUT ITrayWindow *iface,
|
|
IN HWND hWnd)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
return (hWnd == This->hWnd ||
|
|
(This->hWndDesktop != NULL && hWnd == This->hWndDesktop));
|
|
}
|
|
|
|
static BOOL STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IsHorizontal(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
return ITrayWindowImpl_IsPosHorizontal(This);
|
|
}
|
|
|
|
static HFONT STDMETHODCALLTYPE
|
|
ITrayWIndowImpl_GetCaptionFonts(IN OUT ITrayWindow *iface,
|
|
OUT HFONT *phBoldCaption OPTIONAL)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
if (phBoldCaption != NULL)
|
|
*phBoldCaption = This->hStartBtnFont;
|
|
|
|
return This->hCaptionFont;
|
|
}
|
|
|
|
static HWND STDMETHODCALLTYPE
|
|
ITrayWindowImpl_DisplayProperties(IN OUT ITrayWindow *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
if (This->hWndTrayProperties != NULL)
|
|
{
|
|
BringWindowToTop(This->hWndTrayProperties);
|
|
return This->hWndTrayProperties;
|
|
}
|
|
|
|
This->hWndTrayProperties = DisplayTrayProperties(ITrayWindow_from_impl(This));
|
|
return This->hWndTrayProperties;
|
|
}
|
|
|
|
static VOID
|
|
OpenCommonStartMenuDirectory(IN HWND hWndOwner,
|
|
IN LPCTSTR lpOperation)
|
|
{
|
|
TCHAR szDir[MAX_PATH];
|
|
|
|
if (SHGetSpecialFolderPath(hWndOwner,
|
|
szDir,
|
|
CSIDL_COMMON_STARTMENU,
|
|
FALSE))
|
|
{
|
|
ShellExecute(hWndOwner,
|
|
lpOperation,
|
|
NULL,
|
|
NULL,
|
|
szDir,
|
|
SW_SHOWNORMAL);
|
|
}
|
|
}
|
|
|
|
static VOID
|
|
OpenTaskManager(IN HWND hWndOwner)
|
|
{
|
|
ShellExecute(hWndOwner,
|
|
TEXT("open"),
|
|
TEXT("taskmgr.exe"),
|
|
NULL,
|
|
NULL,
|
|
SW_SHOWNORMAL);
|
|
}
|
|
|
|
static BOOL STDMETHODCALLTYPE
|
|
ITrayWindowImpl_ExecContextMenuCmd(IN OUT ITrayWindow *iface,
|
|
IN UINT uiCmd)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
BOOL bHandled = TRUE;
|
|
|
|
switch (uiCmd)
|
|
{
|
|
case ID_SHELL_CMD_PROPERTIES:
|
|
ITrayWindow_DisplayProperties(iface);
|
|
break;
|
|
|
|
case ID_SHELL_CMD_OPEN_ALL_USERS:
|
|
OpenCommonStartMenuDirectory(This->hWnd,
|
|
TEXT("open"));
|
|
break;
|
|
|
|
case ID_SHELL_CMD_EXPLORE_ALL_USERS:
|
|
OpenCommonStartMenuDirectory(This->hWnd,
|
|
TEXT("explore"));
|
|
break;
|
|
|
|
case ID_LOCKTASKBAR:
|
|
if (SHRestricted(REST_CLASSICSHELL) == 0)
|
|
{
|
|
ITrayWindow_Lock(iface,
|
|
!This->Locked);
|
|
}
|
|
break;
|
|
|
|
case ID_SHELL_CMD_OPEN_TASKMGR:
|
|
OpenTaskManager(This->hWnd);
|
|
break;
|
|
|
|
|
|
default:
|
|
DbgPrint("ITrayWindow::ExecContextMenuCmd(%u): Unhandled Command ID!\n", uiCmd);
|
|
bHandled = FALSE;
|
|
break;
|
|
}
|
|
|
|
return bHandled;
|
|
}
|
|
|
|
static BOOL STDMETHODCALLTYPE
|
|
ITrayWindowImpl_Lock(IN OUT ITrayWindow *iface,
|
|
IN BOOL bLock)
|
|
{
|
|
BOOL bPrevLock;
|
|
ITrayWindowImpl *This = impl_from_ITrayWindow(iface);
|
|
|
|
bPrevLock = This->Locked;
|
|
if (This->Locked != bLock)
|
|
{
|
|
This->Locked = bLock;
|
|
|
|
if (This->TrayBandSite != NULL)
|
|
{
|
|
if (!SUCCEEDED(ITrayBandSite_Lock(This->TrayBandSite,
|
|
bLock)))
|
|
{
|
|
/* Reset?? */
|
|
This->Locked = bPrevLock;
|
|
}
|
|
}
|
|
}
|
|
|
|
return bPrevLock;
|
|
}
|
|
|
|
static const ITrayWindowVtbl ITrayWindowImpl_Vtbl =
|
|
{
|
|
/* IUnknown */
|
|
ITrayWindowImpl_QueryInterface,
|
|
ITrayWindowImpl_AddRef,
|
|
ITrayWindowImpl_Release,
|
|
/* ITrayWindow */
|
|
ITrayWindowImpl_Open,
|
|
ITrayWindowImpl_Close,
|
|
ITrayWindowImpl_GetHWND,
|
|
ITrayWindowImpl_IsSpecialHWND,
|
|
ITrayWindowImpl_IsHorizontal,
|
|
ITrayWIndowImpl_GetCaptionFonts,
|
|
ITrayWindowImpl_DisplayProperties,
|
|
ITrayWindowImpl_ExecContextMenuCmd,
|
|
ITrayWindowImpl_Lock
|
|
};
|
|
|
|
static LRESULT CALLBACK
|
|
TrayWndProc(IN HWND hwnd,
|
|
IN UINT uMsg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam)
|
|
{
|
|
ITrayWindowImpl *This = NULL;
|
|
LRESULT Ret = FALSE;
|
|
|
|
if (uMsg != WM_NCCREATE)
|
|
{
|
|
This = (ITrayWindowImpl*)GetWindowLongPtr(hwnd,
|
|
0);
|
|
}
|
|
|
|
if (This != NULL || uMsg == WM_NCCREATE)
|
|
{
|
|
if (This != NULL && This->StartMenuBand != NULL)
|
|
{
|
|
MSG Msg;
|
|
LRESULT lRet;
|
|
|
|
Msg.hwnd = hwnd;
|
|
Msg.message = uMsg;
|
|
Msg.wParam = wParam;
|
|
Msg.lParam = lParam;
|
|
|
|
if (IMenuBand_TranslateMenuMessage(This->StartMenuBand,
|
|
&Msg,
|
|
&lRet) == S_OK)
|
|
{
|
|
return lRet;
|
|
}
|
|
|
|
wParam = Msg.wParam;
|
|
lParam = Msg.lParam;
|
|
}
|
|
|
|
switch (uMsg)
|
|
{
|
|
case WM_NCHITTEST:
|
|
{
|
|
RECT rcClient;
|
|
POINT pt;
|
|
|
|
if (This->Locked)
|
|
{
|
|
/* The user may not be able to resize the tray window.
|
|
Pretend like the window is not sizeable when the user
|
|
clicks on the border. */
|
|
return HTBORDER;
|
|
}
|
|
|
|
if (GetClientRect(hwnd,
|
|
&rcClient) &&
|
|
MapWindowPoints(hwnd,
|
|
NULL,
|
|
(LPPOINT)&rcClient,
|
|
2) != 0)
|
|
{
|
|
pt.x = (SHORT)LOWORD(lParam);
|
|
pt.y = (SHORT)HIWORD(lParam);
|
|
|
|
if (PtInRect(&rcClient,
|
|
pt))
|
|
{
|
|
/* The user is trying to drag the tray window */
|
|
return HTCAPTION;
|
|
}
|
|
|
|
/* Depending on the position of the tray window, allow only
|
|
changing the border next to the monitor working area */
|
|
switch (This->Position)
|
|
{
|
|
case ABE_TOP:
|
|
if (pt.y > rcClient.bottom)
|
|
return HTBOTTOM;
|
|
break;
|
|
|
|
case ABE_BOTTOM:
|
|
if (pt.y < rcClient.top)
|
|
return HTTOP;
|
|
break;
|
|
|
|
case ABE_LEFT:
|
|
if (pt.x > rcClient.right)
|
|
return HTRIGHT;
|
|
break;
|
|
|
|
case ABE_RIGHT:
|
|
if (pt.x < rcClient.left)
|
|
return HTLEFT;
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
return HTBORDER;
|
|
}
|
|
|
|
case WM_MOVING:
|
|
{
|
|
POINT ptCursor;
|
|
PRECT pRect = (PRECT)lParam;
|
|
|
|
/* We need to ensure that an application can not accidently
|
|
move the tray window (using SetWindowPos). However, we still
|
|
need to be able to move the window in case the user wants to
|
|
drag the tray window to another position or in case the user
|
|
wants to resize the tray window. */
|
|
if (!This->Locked && GetCursorPos(&ptCursor))
|
|
{
|
|
This->IsDragging = TRUE;
|
|
This->DraggingPosition = ITrayWindowImpl_GetDraggingRectFromPt(This,
|
|
ptCursor,
|
|
pRect,
|
|
&This->DraggingMonitor);
|
|
}
|
|
else
|
|
{
|
|
*pRect = This->rcTrayWnd[This->Position];
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
case WM_SIZING:
|
|
{
|
|
PRECT pRect = (PRECT)lParam;
|
|
|
|
if (!This->Locked)
|
|
{
|
|
ITrayWindowImpl_CalculateValidSize(This,
|
|
This->Position,
|
|
pRect);
|
|
}
|
|
else
|
|
{
|
|
*pRect = This->rcTrayWnd[This->Position];
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
case WM_WINDOWPOSCHANGING:
|
|
{
|
|
ITrayWindowImpl_ChangingWinPos(This,
|
|
(LPWINDOWPOS)lParam);
|
|
break;
|
|
}
|
|
|
|
case WM_SIZE:
|
|
{
|
|
RECT rcClient;
|
|
|
|
if (wParam == SIZE_RESTORED && lParam == 0)
|
|
{
|
|
/* Clip the tray window on multi monitor systems so the edges can't
|
|
overlap into another monitor */
|
|
ITrayWindowImpl_ApplyClipping(This,
|
|
TRUE);
|
|
|
|
if (!GetClientRect(This->hWnd,
|
|
&rcClient))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rcClient.left = rcClient.top = 0;
|
|
rcClient.right = LOWORD(lParam);
|
|
rcClient.bottom = HIWORD(lParam);
|
|
}
|
|
|
|
ITrayWindowImpl_AlignControls(This,
|
|
&rcClient);
|
|
break;
|
|
}
|
|
|
|
case WM_ENTERSIZEMOVE:
|
|
This->InSizeMove = TRUE;
|
|
This->IsDragging = FALSE;
|
|
if (!This->Locked)
|
|
{
|
|
/* Remove the clipping on multi monitor systems while dragging around */
|
|
ITrayWindowImpl_ApplyClipping(This,
|
|
FALSE);
|
|
}
|
|
break;
|
|
|
|
case WM_EXITSIZEMOVE:
|
|
This->InSizeMove = FALSE;
|
|
if (!This->Locked)
|
|
{
|
|
/* Apply clipping */
|
|
PostMessage(This->hWnd,
|
|
WM_SIZE,
|
|
SIZE_RESTORED,
|
|
0);
|
|
}
|
|
break;
|
|
|
|
case WM_SYSCHAR:
|
|
switch (wParam)
|
|
{
|
|
case TEXT(' '):
|
|
{
|
|
/* The user pressed Alt+Space, this usually brings up the system menu of a window.
|
|
The tray window needs to handle this specially, since it normally doesn't have
|
|
a system menu. */
|
|
|
|
static const UINT uidDisableItem[] = {
|
|
SC_RESTORE,
|
|
SC_MOVE,
|
|
SC_SIZE,
|
|
SC_MAXIMIZE,
|
|
SC_MINIMIZE,
|
|
};
|
|
HMENU hSysMenu;
|
|
INT i;
|
|
UINT uId;
|
|
|
|
/* temporarily enable the system menu */
|
|
SetWindowStyle(hwnd,
|
|
WS_SYSMENU,
|
|
WS_SYSMENU);
|
|
|
|
hSysMenu = GetSystemMenu(hwnd,
|
|
FALSE);
|
|
if (hSysMenu != NULL)
|
|
{
|
|
/* Disable all items that are not relevant */
|
|
for (i = 0; i != sizeof(uidDisableItem) / sizeof(uidDisableItem[0]); i++)
|
|
{
|
|
EnableMenuItem(hSysMenu,
|
|
uidDisableItem[i],
|
|
MF_BYCOMMAND | MF_GRAYED);
|
|
}
|
|
|
|
EnableMenuItem(hSysMenu,
|
|
SC_CLOSE,
|
|
MF_BYCOMMAND |
|
|
(SHRestricted(REST_NOCLOSE) ? MF_GRAYED : MF_ENABLED));
|
|
|
|
/* Display the system menu */
|
|
uId = ITrayWindowImpl_TrackMenu(This,
|
|
hSysMenu,
|
|
NULL,
|
|
This->hwndStart,
|
|
This->Position != ABE_TOP,
|
|
FALSE);
|
|
if (uId != 0)
|
|
{
|
|
SendMessage(This->hWnd,
|
|
WM_SYSCOMMAND,
|
|
(WPARAM)uId,
|
|
0);
|
|
}
|
|
}
|
|
|
|
/* revert the system menu window style */
|
|
SetWindowStyle(hwnd,
|
|
WS_SYSMENU,
|
|
0);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
goto DefHandler;
|
|
}
|
|
break;
|
|
|
|
case WM_NCRBUTTONUP:
|
|
/* We want the user to be able to get a context menu even on the nonclient
|
|
area (including the sizing border)! */
|
|
uMsg = WM_CONTEXTMENU;
|
|
wParam = (WPARAM)hwnd;
|
|
/* fall through */
|
|
|
|
case WM_CONTEXTMENU:
|
|
{
|
|
POINT pt, *ppt = NULL;
|
|
HWND hWndExclude = NULL;
|
|
|
|
/* Check if the administrator has forbidden access to context menus */
|
|
if (SHRestricted(REST_NOTRAYCONTEXTMENU))
|
|
break;
|
|
|
|
pt.x = (SHORT)LOWORD(lParam);
|
|
pt.y = (SHORT)HIWORD(lParam);
|
|
|
|
if (pt.x != -1 || pt.y != -1)
|
|
ppt = &pt;
|
|
else
|
|
hWndExclude = This->hwndStart;
|
|
|
|
if ((HWND)wParam == This->hwndStart)
|
|
{
|
|
/* Make sure we can't track the context menu if the start
|
|
menu is currently being shown */
|
|
if (!(SendMessage(This->hwndStart,
|
|
BM_GETSTATE,
|
|
0,
|
|
0) & BST_PUSHED))
|
|
{
|
|
ITrayWindowImpl_TrackCtxMenu(This,
|
|
&StartMenuBtnCtxMenu,
|
|
ppt,
|
|
hWndExclude,
|
|
This->Position == ABE_BOTTOM,
|
|
This);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* See if the context menu should be handled by the task band site */
|
|
if (ppt != NULL && This->TrayBandSite != NULL)
|
|
{
|
|
HWND hWndAtPt;
|
|
POINT ptClient = *ppt;
|
|
|
|
/* Convert the coordinates to client-coordinates */
|
|
MapWindowPoints(NULL,
|
|
This->hWnd,
|
|
&ptClient,
|
|
1);
|
|
|
|
hWndAtPt = ChildWindowFromPoint(This->hWnd,
|
|
ptClient);
|
|
if (hWndAtPt != NULL &&
|
|
(hWndAtPt == This->hwndRebar || IsChild(This->hwndRebar,
|
|
hWndAtPt)))
|
|
{
|
|
/* Check if the user clicked on the task switch window */
|
|
ptClient = *ppt;
|
|
MapWindowPoints(NULL,
|
|
This->hwndRebar,
|
|
&ptClient,
|
|
1);
|
|
|
|
hWndAtPt = ChildWindowFromPointEx(This->hwndRebar,
|
|
ptClient,
|
|
CWP_SKIPINVISIBLE | CWP_SKIPDISABLED);
|
|
if (hWndAtPt == This->hwndTaskSwitch)
|
|
goto HandleTrayContextMenu;
|
|
|
|
/* Forward the message to the task band site */
|
|
ITrayBandSite_ProcessMessage(This->TrayBandSite,
|
|
hwnd,
|
|
uMsg,
|
|
wParam,
|
|
lParam,
|
|
&Ret);
|
|
}
|
|
else
|
|
goto HandleTrayContextMenu;
|
|
}
|
|
else
|
|
{
|
|
HandleTrayContextMenu:
|
|
/* Tray the default tray window context menu */
|
|
ITrayWindowImpl_TrackCtxMenu(This,
|
|
&TrayWindowCtxMenu,
|
|
ppt,
|
|
NULL,
|
|
FALSE,
|
|
This);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_NOTIFY:
|
|
{
|
|
/* FIXME: We can't check with IsChild whether the hwnd is somewhere inside
|
|
the rebar control! But we shouldn't forward messages that the band
|
|
site doesn't handle, such as other controls (start button, tray window */
|
|
if (This->TrayBandSite == NULL ||
|
|
!SUCCEEDED(ITrayBandSite_ProcessMessage(This->TrayBandSite,
|
|
hwnd,
|
|
uMsg,
|
|
wParam,
|
|
lParam,
|
|
&Ret)))
|
|
{
|
|
const NMHDR *nmh = (const NMHDR *)lParam;
|
|
|
|
if (nmh->hwndFrom == This->hwndTrayNotify)
|
|
{
|
|
switch (nmh->code)
|
|
{
|
|
case NTNWM_REALIGN:
|
|
/* Cause all controls to be aligned */
|
|
PostMessage(This->hWnd,
|
|
WM_SIZE,
|
|
SIZE_RESTORED,
|
|
0);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case WM_NCLBUTTONDBLCLK:
|
|
/* We "handle" this message so users can't cause a weird maximize/restore
|
|
window animation when double-clicking the tray window! */
|
|
break;
|
|
|
|
case WM_NCCREATE:
|
|
{
|
|
LPCREATESTRUCT CreateStruct = (LPCREATESTRUCT)lParam;
|
|
This = (ITrayWindowImpl*)CreateStruct->lpCreateParams;
|
|
|
|
if (InterlockedCompareExchangePointer((PVOID*)&This->hWnd,
|
|
(PVOID)hwnd,
|
|
NULL) != NULL)
|
|
{
|
|
/* Somebody else was faster... */
|
|
return FALSE;
|
|
}
|
|
|
|
SetWindowLongPtr(hwnd,
|
|
0,
|
|
(LONG_PTR)This);
|
|
|
|
return ITrayWindowImpl_NCCreate(This);
|
|
}
|
|
|
|
case WM_CREATE:
|
|
ITrayWindowImpl_Create(This);
|
|
break;
|
|
|
|
case WM_NCDESTROY:
|
|
ITrayWindowImpl_Destroy(This);
|
|
break;
|
|
|
|
case WM_APP_TRAYDESTROY:
|
|
DestroyWindow(hwnd);
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
if ((HWND)lParam == This->hwndStart)
|
|
{
|
|
if (This->StartMenuPopup != NULL)
|
|
{
|
|
POINTL pt;
|
|
RECTL rcExclude;
|
|
DWORD dwFlags = 0;
|
|
|
|
if (GetWindowRect(This->hwndStart,
|
|
(RECT*)&rcExclude))
|
|
{
|
|
if (ITrayWindowImpl_IsPosHorizontal(This))
|
|
{
|
|
pt.x = rcExclude.left;
|
|
pt.y = rcExclude.top;
|
|
dwFlags |= MPPF_BOTTOM;
|
|
}
|
|
else
|
|
{
|
|
if (This->Position == ABE_LEFT)
|
|
pt.x = rcExclude.left;
|
|
else
|
|
pt.x = rcExclude.right;
|
|
|
|
pt.y = rcExclude.bottom;
|
|
dwFlags |= MPPF_BOTTOM;
|
|
}
|
|
|
|
IMenuPopup_Popup(This->StartMenuPopup,
|
|
&pt,
|
|
&rcExclude,
|
|
dwFlags);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (This->TrayBandSite == NULL ||
|
|
!SUCCEEDED(ITrayBandSite_ProcessMessage(This->TrayBandSite,
|
|
hwnd,
|
|
uMsg,
|
|
wParam,
|
|
lParam,
|
|
&Ret)))
|
|
{
|
|
switch(LOWORD(wParam))
|
|
{
|
|
/* FIXME: Handle these commands as well */
|
|
case IDM_TASKBARANDSTARTMENU:
|
|
case IDM_SEARCH:
|
|
case IDM_HELPANDSUPPORT:
|
|
break;
|
|
|
|
case IDM_RUN:
|
|
{
|
|
HANDLE hShell32;
|
|
RUNFILEDLG RunFileDlg;
|
|
|
|
hShell32 = GetModuleHandle(TEXT("SHELL32.DLL"));
|
|
RunFileDlg = (RUNFILEDLG)GetProcAddress(hShell32, (LPCSTR)61);
|
|
|
|
RunFileDlg(hwnd, NULL, NULL, NULL, NULL, RFF_CALCDIRECTORY);
|
|
break;
|
|
}
|
|
|
|
/* FIXME: Handle these commands as well */
|
|
case IDM_SYNCHRONIZE:
|
|
case IDM_LOGOFF:
|
|
case IDM_DISCONNECT:
|
|
case IDM_UNDOCKCOMPUTER:
|
|
break;
|
|
|
|
case IDM_SHUTDOWN:
|
|
{
|
|
HANDLE hShell32;
|
|
EXITWINDLG ExitWinDlg;
|
|
|
|
hShell32 = GetModuleHandle(TEXT("SHELL32.DLL"));
|
|
ExitWinDlg = (EXITWINDLG)GetProcAddress(hShell32, (LPCSTR)60);
|
|
|
|
ExitWinDlg(hwnd);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
goto DefHandler;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DefHandler:
|
|
Ret = DefWindowProc(hwnd,
|
|
uMsg,
|
|
wParam,
|
|
lParam);
|
|
}
|
|
|
|
return Ret;
|
|
}
|
|
|
|
/*
|
|
* Tray Window Context Menu
|
|
*/
|
|
|
|
static HMENU
|
|
CreateTrayWindowContextMenu(IN HWND hWndOwner,
|
|
IN PVOID *ppcmContext,
|
|
IN PVOID Context OPTIONAL)
|
|
{
|
|
ITrayWindowImpl *This = (ITrayWindowImpl *)Context;
|
|
IContextMenu *pcm = NULL;
|
|
HMENU hPopup;
|
|
|
|
hPopup = LoadPopupMenu(hExplorerInstance,
|
|
MAKEINTRESOURCE(IDM_TRAYWND));
|
|
|
|
if (hPopup != NULL)
|
|
{
|
|
if (SHRestricted(REST_CLASSICSHELL) != 0)
|
|
{
|
|
DeleteMenu(hPopup,
|
|
ID_LOCKTASKBAR,
|
|
MF_BYCOMMAND);
|
|
}
|
|
|
|
CheckMenuItem(hPopup,
|
|
ID_LOCKTASKBAR,
|
|
MF_BYCOMMAND | (This->Locked ? MF_CHECKED : MF_UNCHECKED));
|
|
|
|
if (This->TrayBandSite != NULL)
|
|
{
|
|
if (SUCCEEDED(ITrayBandSite_AddContextMenus(This->TrayBandSite,
|
|
hPopup,
|
|
0,
|
|
ID_SHELL_CMD_FIRST,
|
|
ID_SHELL_CMD_LAST,
|
|
CMF_NORMAL,
|
|
&pcm)))
|
|
{
|
|
DbgPrint("ITrayBandSite::AddContextMenus succeeded!\n");
|
|
*(IContextMenu **)ppcmContext = pcm;
|
|
}
|
|
}
|
|
}
|
|
|
|
return hPopup;
|
|
}
|
|
|
|
static VOID
|
|
OnTrayWindowContextMenuCommand(IN HWND hWndOwner,
|
|
IN UINT uiCmdId,
|
|
IN PVOID pcmContext OPTIONAL,
|
|
IN PVOID Context OPTIONAL)
|
|
{
|
|
ITrayWindowImpl *This = (ITrayWindowImpl *)Context;
|
|
IContextMenu *pcm = (IContextMenu *)pcmContext;
|
|
|
|
if (uiCmdId != 0)
|
|
{
|
|
if (uiCmdId >= ID_SHELL_CMD_FIRST && uiCmdId <= ID_SHELL_CMD_LAST)
|
|
{
|
|
CMINVOKECOMMANDINFO cmici = {0};
|
|
|
|
if (pcm != NULL)
|
|
{
|
|
/* Setup and invoke the shell command */
|
|
cmici.cbSize = sizeof(cmici);
|
|
cmici.hwnd = hWndOwner;
|
|
cmici.lpVerb = (LPCSTR)MAKEINTRESOURCE(uiCmdId - ID_SHELL_CMD_FIRST);
|
|
cmici.nShow = SW_NORMAL;
|
|
|
|
IContextMenu_InvokeCommand(pcm,
|
|
&cmici);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ITrayWindow_ExecContextMenuCmd(ITrayWindow_from_impl(This),
|
|
uiCmdId);
|
|
}
|
|
}
|
|
|
|
if (pcm != NULL)
|
|
IContextMenu_Release(pcm);
|
|
}
|
|
|
|
static const TRAYWINDOW_CTXMENU TrayWindowCtxMenu = {
|
|
CreateTrayWindowContextMenu,
|
|
OnTrayWindowContextMenuCommand
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
|
|
BOOL
|
|
RegisterTrayWindowClass(VOID)
|
|
{
|
|
WNDCLASS wcTrayWnd;
|
|
BOOL Ret;
|
|
|
|
if (!RegisterTrayNotifyWndClass())
|
|
return FALSE;
|
|
|
|
wcTrayWnd.style = CS_DBLCLKS;
|
|
wcTrayWnd.lpfnWndProc = TrayWndProc;
|
|
wcTrayWnd.cbClsExtra = 0;
|
|
wcTrayWnd.cbWndExtra = sizeof(ITrayWindowImpl *);
|
|
wcTrayWnd.hInstance = hExplorerInstance;
|
|
wcTrayWnd.hIcon = NULL;
|
|
wcTrayWnd.hCursor = LoadCursor(NULL,
|
|
IDC_ARROW);
|
|
wcTrayWnd.hbrBackground = (HBRUSH)(COLOR_3DFACE + 1);
|
|
wcTrayWnd.lpszMenuName = NULL;
|
|
wcTrayWnd.lpszClassName = szTrayWndClass;
|
|
|
|
Ret = RegisterClass(&wcTrayWnd) != 0;
|
|
|
|
if (!Ret)
|
|
UnregisterTrayNotifyWndClass();
|
|
|
|
return Ret;
|
|
}
|
|
|
|
VOID
|
|
UnregisterTrayWindowClass(VOID)
|
|
{
|
|
UnregisterTrayNotifyWndClass();
|
|
|
|
UnregisterClass(szTrayWndClass,
|
|
hExplorerInstance);
|
|
}
|
|
|
|
ITrayWindow *
|
|
CreateTrayWindow(VOID)
|
|
{
|
|
ITrayWindowImpl *This;
|
|
ITrayWindow *TrayWindow;
|
|
|
|
This = ITrayWindowImpl_Construct();
|
|
if (This != NULL)
|
|
{
|
|
TrayWindow = ITrayWindow_from_impl(This);
|
|
|
|
ITrayWindowImpl_Open(TrayWindow);
|
|
|
|
return TrayWindow;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
VOID
|
|
TrayProcessMessages(IN OUT ITrayWindow *Tray)
|
|
{
|
|
ITrayWindowImpl *This;
|
|
MSG Msg;
|
|
|
|
This = impl_from_ITrayWindow(Tray);
|
|
|
|
/* FIXME: We should keep a reference here... */
|
|
|
|
while (PeekMessage(&Msg,
|
|
NULL,
|
|
0,
|
|
0,
|
|
PM_REMOVE))
|
|
{
|
|
if (Msg.message == WM_QUIT)
|
|
break;
|
|
|
|
if (This->StartMenuBand == NULL ||
|
|
IMenuBand_IsMenuMessage(This->StartMenuBand,
|
|
&Msg) != S_OK)
|
|
{
|
|
TranslateMessage(&Msg);
|
|
DispatchMessage(&Msg);
|
|
}
|
|
}
|
|
}
|
|
|
|
VOID
|
|
TrayMessageLoop(IN OUT ITrayWindow *Tray)
|
|
{
|
|
ITrayWindowImpl *This;
|
|
MSG Msg;
|
|
BOOL Ret;
|
|
|
|
This = impl_from_ITrayWindow(Tray);
|
|
|
|
/* FIXME: We should keep a reference here... */
|
|
|
|
while (1)
|
|
{
|
|
Ret = (GetMessage(&Msg,
|
|
NULL,
|
|
0,
|
|
0) != 0);
|
|
|
|
if (Ret != -1)
|
|
{
|
|
if (!Ret)
|
|
break;
|
|
|
|
if (This->StartMenuBand == NULL ||
|
|
IMenuBand_IsMenuMessage(This->StartMenuBand,
|
|
&Msg) != S_OK)
|
|
{
|
|
TranslateMessage(&Msg);
|
|
DispatchMessage(&Msg);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* IShellDesktopTray
|
|
*
|
|
* NOTE: This is a very windows-specific COM interface used by SHCreateDesktop()!
|
|
* These are the calls I observed, it may be wrong/incomplete/buggy!!!
|
|
* The reason we implement it is because we have to use SHCreateDesktop() so
|
|
* that the shell provides the desktop window and all the features that come
|
|
* with it (especially positioning of desktop icons)
|
|
*/
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_QueryInterface(IN OUT IShellDesktopTray *iface,
|
|
IN REFIID riid,
|
|
OUT LPVOID *ppvObj)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_IShellDesktopTray(iface);
|
|
ITrayWindow *tray = ITrayWindow_from_impl(This);
|
|
|
|
DbgPrint("IShellDesktopTray::QueryInterface(0x%p, 0x%p)\n", riid, ppvObj);
|
|
return ITrayWindowImpl_QueryInterface(tray,
|
|
riid,
|
|
ppvObj);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_Release(IN OUT IShellDesktopTray *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_IShellDesktopTray(iface);
|
|
ITrayWindow *tray = ITrayWindow_from_impl(This);
|
|
|
|
DbgPrint("IShellDesktopTray::Release()\n");
|
|
return ITrayWindowImpl_Release(tray);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_AddRef(IN OUT IShellDesktopTray *iface)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_IShellDesktopTray(iface);
|
|
ITrayWindow *tray = ITrayWindow_from_impl(This);
|
|
|
|
DbgPrint("IShellDesktopTray::AddRef()\n");
|
|
return ITrayWindowImpl_AddRef(tray);
|
|
}
|
|
|
|
static ULONG STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_GetState(IN OUT IShellDesktopTray *iface)
|
|
{
|
|
/* FIXME: Return ABS_ flags? */
|
|
DbgPrint("IShellDesktopTray::GetState() unimplemented!\n");
|
|
return 0;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_GetTrayWindow(IN OUT IShellDesktopTray *iface,
|
|
OUT HWND *phWndTray)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_IShellDesktopTray(iface);
|
|
DbgPrint("IShellDesktopTray::GetTrayWindow(0x%p)\n", phWndTray);
|
|
*phWndTray = This->hWnd;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_RegisterDesktopWindow(IN OUT IShellDesktopTray *iface,
|
|
IN HWND hWndDesktop)
|
|
{
|
|
ITrayWindowImpl *This = impl_from_IShellDesktopTray(iface);
|
|
DbgPrint("IShellDesktopTray::RegisterDesktopWindow(0x%p)\n", hWndDesktop);
|
|
|
|
This->hWndDesktop = hWndDesktop;
|
|
return S_OK;
|
|
}
|
|
|
|
static HRESULT STDMETHODCALLTYPE
|
|
ITrayWindowImpl_IShellDesktopTray_Unknown(IN OUT IShellDesktopTray *iface,
|
|
IN DWORD dwUnknown1,
|
|
IN DWORD dwUnknown2)
|
|
{
|
|
DbgPrint("IShellDesktopTray::Unknown(%u,%u) unimplemented!\n", dwUnknown1, dwUnknown2);
|
|
return S_OK;
|
|
}
|
|
|
|
static const IShellDesktopTrayVtbl IShellDesktopTrayImpl_Vtbl =
|
|
{
|
|
/*** IUnknown ***/
|
|
ITrayWindowImpl_IShellDesktopTray_QueryInterface,
|
|
ITrayWindowImpl_IShellDesktopTray_AddRef,
|
|
ITrayWindowImpl_IShellDesktopTray_Release,
|
|
/*** IShellDesktopTray ***/
|
|
ITrayWindowImpl_IShellDesktopTray_GetState,
|
|
ITrayWindowImpl_IShellDesktopTray_GetTrayWindow,
|
|
ITrayWindowImpl_IShellDesktopTray_RegisterDesktopWindow,
|
|
ITrayWindowImpl_IShellDesktopTray_Unknown
|
|
};
|