|
|
#include "cabinet.h"
#include <wtsapi32.h> // for NOTIFY_FOR_THIS_SESSION
#include <winsta.h> // for disconnect and reconnect messages from terminal server
#include "mmsysp.h"
#include "rcids.h"
#include "dlg.h"
#include <atlstuff.h>
#include <shlapip.h>
#include "trayclok.h"
#include <help.h> // help ids
#include <desktray.h>
#include "util.h"
#include "tray.h"
#if defined(FE_IME)
#include <immp.h>
#endif
#include <regstr.h>
#include "bandsite.h"
#include "startmnu.h"
#include "uemapp.h"
#include <uxthemep.h>
#define NO_NOTIFYSUBCLASSWNDPROC
#include "cwndproc.cpp"
#include "desktop2.h"
#include "mixer.h"
#include "winbrand.h"
#define DM_FOCUS 0 // focus
#define DM_SHUTDOWN TF_TRAY // shutdown
#define DM_UEMTRACE TF_TRAY // timer service, other UEM stuff
#define DM_MISC 0 // miscellany
const GUID CLSID_MSUTBDeskBand = {0x540d8a8b, 0x1c3f, 0x4e32, 0x81, 0x32, 0x53, 0x0f, 0x6a, 0x50, 0x20, 0x90};
// From Desktop2\proglist.cpp
HRESULT AddMenuItemsCacheTask(IShellTaskScheduler* pSystemScheduler, BOOL fKeepCacheWhenFinished);
// import the WIN31 Compatibility HACKs from the shell32.dll
STDAPI_(void) CheckWinIniForAssocs(void);
// hooks to Shell32.dll
STDAPI CheckDiskSpace(); STDAPI CheckStagingArea();
// startmnu.cpp
void HandleFirstTime();
HWND v_hwndDesktop = NULL; HWND v_hwndTray = NULL; HWND v_hwndStartPane = NULL;
BOOL g_fDesktopRaised = FALSE; BOOL g_fInSizeMove = FALSE;
UINT _uMsgEnableUserTrackedBalloonTips = 0;
void ClearRecentDocumentsAndMRUStuff(BOOL fBroadcastChange); void DoTaskBarProperties(HWND hwnd, DWORD dwFlags);
void ClassFactory_Start(); void ClassFactory_Stop();
//
// Settings UI entry point types.
//
typedef void (WINAPI *PTRAYPROPSHEETCALLBACK)(DWORD nStartPage); typedef void (WINAPI *PSETTINGSUIENTRY)(PTRAYPROPSHEETCALLBACK);
// Shell perf automation
extern DWORD g_dwShellStartTime; extern DWORD g_dwShellStopTime; extern DWORD g_dwStopWatchMode;
CTray c_tray;
// from explorer\desktop2
STDAPI DesktopV2_Create( IMenuPopup **ppmp, IMenuBand **ppmb, void **ppvStartPane); STDAPI DesktopV2_Build(void *pvStartPane);
// dyna-res change for multi-config hot/warm-doc
void HandleDisplayChange(int x, int y, BOOL fCritical); DWORD GetMinDisplayRes(void);
// timer IDs
#define IDT_AUTOHIDE 2
#define IDT_AUTOUNHIDE 3
#ifdef DELAYWININICHANGE
#define IDT_DELAYWININICHANGE 5
#endif
#define IDT_DESKTOP 6
#define IDT_PROGRAMS IDM_PROGRAMS
#define IDT_RECENT IDM_RECENT
#define IDT_REBUILDMENU 7
#define IDT_HANDLEDELAYBOOTSTUFF 8
#define IDT_REVERTPROGRAMS 9
#define IDT_REVERTRECENT 10
#define IDT_REVERTFAVORITES 11
#define IDT_STARTMENU 12
#define IDT_ENDUNHIDEONTRAYNOTIFY 13
#define IDT_SERVICE0 14
#define IDT_SERVICE1 15
#define IDT_SERVICELAST IDT_SERVICE1
#define IDT_SAVESETTINGS 17
#define IDT_ENABLEUNDO 18
#define IDT_STARTUPFAILED 19
#define IDT_CHECKDISKSPACE 21
#define IDT_STARTBUTTONBALLOON 22
#define IDT_CHANGENOTIFY 23
#define IDT_COFREEUNUSED 24
#define IDT_DESKTOPCLEANUP 25
#define FADEINDELAY 100
#define BALLOONTIPDELAY 10000 // default balloon time copied from traynot.cpp
// INSTRUMENTATION WARNING: If you change anything here, make sure to update instrument.c
// we need to start at 500 because we're now sharing the hotkey handler
// with shortcuts.. they use an index array so they need to be 0 based
// NOTE, this constant is also in desktop.cpp, so that we can forward hotkeys from the desktop for
// NOTE, app compatibility.
#define GHID_FIRST 500
enum { GHID_RUN = GHID_FIRST, GHID_MINIMIZEALL, GHID_UNMINIMIZEALL, GHID_HELP, GHID_EXPLORER, GHID_FINDFILES, GHID_FINDCOMPUTER, GHID_TASKTAB, GHID_TASKSHIFTTAB, GHID_SYSPROPERTIES, GHID_DESKTOP, GHID_TRAYNOTIFY, GHID_MAX };
const DWORD GlobalKeylist[] = { MAKELONG(TEXT('R'), MOD_WIN), MAKELONG(TEXT('M'), MOD_WIN), MAKELONG(TEXT('M'), MOD_SHIFT|MOD_WIN), MAKELONG(VK_F1,MOD_WIN), MAKELONG(TEXT('E'),MOD_WIN), MAKELONG(TEXT('F'),MOD_WIN), MAKELONG(TEXT('F'), MOD_CONTROL|MOD_WIN), MAKELONG(VK_TAB, MOD_WIN), MAKELONG(VK_TAB, MOD_WIN|MOD_SHIFT), MAKELONG(VK_PAUSE,MOD_WIN), MAKELONG(TEXT('D'),MOD_WIN), MAKELONG(TEXT('B'),MOD_WIN), };
CTray::CTray() : _fCanSizeMove(TRUE), _fIsLogoff(FALSE), _fIsDesktopConnected(TRUE), _fBandSiteInitialized(FALSE) { }
void CTray::ClosePopupMenus() { if (_pmpStartMenu) _pmpStartMenu->OnSelect(MPOS_FULLCANCEL); if (_pmpStartPane) _pmpStartPane->OnSelect(MPOS_FULLCANCEL); }
BOOL Tray_StartPanelEnabled() { SHELLSTATE ss = {0}; SHGetSetSettings(&ss, SSF_STARTPANELON, FALSE); return ss.fStartPanelOn; }
#ifdef FEATURE_STARTPAGE
BOOL Tray_ShowStartPageEnabled() { #if 0
SHELLSTATE ss = {0}; SHGetSetSettings(&ss, SSF_SHOWSTARTPAGE, FALSE); return ss.fShowStartPage; #endif
static BOOL s_fEnabled = -1;
if (s_fEnabled == -1) { DWORD dwType = REG_DWORD; DWORD dwData; DWORD cbData = sizeof(DWORD); DWORD dwDefault = 0; // FaultID is a decoy so that hackers won't think of trying to enable this feature.
LONG lRet = SHRegGetUSValue(REGSTR_PATH_EXPLORER, TEXT("FaultID"), &dwType, &dwData, &cbData, FALSE, &dwDefault, sizeof(dwDefault)); if (lRet == ERROR_SUCCESS && dwData == 101) // Some random value not easily guessed
s_fEnabled = TRUE; else s_fEnabled = FALSE; } return s_fEnabled; } #endif
//
// The StartButtonBalloonTip registry value can have one of these values:
//
// 0 (or nonexistent): User has never clicked the Start Button.
// 1: User has clicked the Start Button on a pre-Whistler system.
// 2: User has clicked the Start Button on a Whistler system.
//
// In case 0, we always want to show the balloon tip regardless of whether
// the user is running Classic or Personal.
//
// In case 1, we want to show the balloon tip if the user is using the
// Personal Start Menu, but not if using Classic (since he's already
// seen the Classic Start Menu). In the Classic case, upgrade the counter
// to 2 so the user won't be annoyed when they switch from Classic to
// Personal.
//
// In case 2, we don't want to show the balloon tip at all since the
// user has seen all we have to offer.
//
BOOL CTray::_ShouldWeShowTheStartButtonBalloon() { DWORD dwType; DWORD dwData = 0; DWORD cbSize = sizeof(DWORD); SHGetValue(HKEY_CURRENT_USER, REGSTR_EXPLORER_ADVANCED, TEXT("StartButtonBalloonTip"), &dwType, (BYTE*)&dwData, &cbSize);
if (Tray_StartPanelEnabled()) { // Personal Start Menu is enabled, so show the balloon if the
// user has never logged on to a Whistler machine before.
return dwData < 2; } else { // Classic Start Menu is enabled.
switch (dwData) { case 0: // User has never seen the Start Menu before, not even the
// classic one. So show the tip.
return TRUE;
case 1: // User has already seen the Classic Start Menu, so don't
// prompt them again. Note that this means that they aren't
// prompted when they turn on the Personal Start Menu, but
// that's okay, because by the time they switch to Personal,
// they clearly have demonstrated that they know how the
// Start Button works and don't need a tip.
_DontShowTheStartButtonBalloonAnyMore(); return FALSE;
default: // User has seen Whistler Start menu before, so don't show tip.
return FALSE; } } }
//
// Set the value to 2 to indicate that the user has seen a Whistler
// Start Menu (either Classic or Personal).
//
void CTray::_DontShowTheStartButtonBalloonAnyMore() { DWORD dwData = 2; SHSetValue(HKEY_CURRENT_USER, REGSTR_EXPLORER_ADVANCED, TEXT("StartButtonBalloonTip"), REG_DWORD, (BYTE*)&dwData, sizeof(dwData)); }
void CTray::_DestroyStartButtonBalloon() { if (_hwndStartBalloon) { DestroyWindow(_hwndStartBalloon); _hwndStartBalloon = NULL; } KillTimer(_hwnd, IDT_STARTBUTTONBALLOON); }
void CTray::CreateStartButtonBalloon(UINT idsTitle, UINT idsMessage) { if (!_hwndStartBalloon) {
_hwndStartBalloon = CreateWindow(TOOLTIPS_CLASS, NULL, WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP | TTS_BALLOON, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, _hwnd, NULL, hinstCabinet, NULL);
if (_hwndStartBalloon) { // set the version so we can have non buggy mouse event forwarding
SendMessage(_hwndStartBalloon, CCM_SETVERSION, COMCTL32_VERSION, 0); SendMessage(_hwndStartBalloon, TTM_SETMAXTIPWIDTH, 0, (LPARAM)300);
// taskbar windows are themed under Taskbar subapp name
SendMessage(_hwndStartBalloon, TTM_SETWINDOWTHEME, 0, (LPARAM)c_wzTaskbarTheme);
// Tell the Start Menu that this is a special balloon tip
SetProp(_hwndStartBalloon, PROP_DV2_BALLOONTIP, DV2_BALLOONTIP_STARTBUTTON); } }
if (_hwndStartBalloon) { TCHAR szTip[MAX_PATH]; szTip[0] = TEXT('\0'); LoadString(hinstCabinet, idsMessage, szTip, ARRAYSIZE(szTip)); if (szTip[0]) { RECT rc; TOOLINFO ti = {0};
ti.cbSize = sizeof(ti); ti.uFlags = TTF_IDISHWND | TTF_TRACK | TTF_TRANSPARENT; ti.hwnd = _hwnd; ti.uId = (UINT_PTR)_hwndStart; //ti.lpszText = NULL;
SendMessage(_hwndStartBalloon, TTM_ADDTOOL, 0, (LPARAM)(LPTOOLINFO)&ti); SendMessage(_hwndStartBalloon, TTM_TRACKACTIVATE, (WPARAM)FALSE, (LPARAM)0);
ti.lpszText = szTip; SendMessage(_hwndStartBalloon, TTM_UPDATETIPTEXT, 0, (LPARAM)&ti);
LoadString(hinstCabinet, idsTitle, szTip, ARRAYSIZE(szTip)); if (szTip[0]) { SendMessage(_hwndStartBalloon, TTM_SETTITLE, TTI_INFO, (LPARAM)szTip); }
GetWindowRect(_hwndStart, &rc);
SendMessage(_hwndStartBalloon, TTM_TRACKPOSITION, 0, MAKELONG((rc.left + rc.right)/2, rc.top));
SetWindowZorder(_hwndStartBalloon, HWND_TOPMOST);
SendMessage(_hwndStartBalloon, TTM_TRACKACTIVATE, (WPARAM)TRUE, (LPARAM)&ti);
SetTimer(_hwnd, IDT_STARTBUTTONBALLOON, BALLOONTIPDELAY, NULL); } } }
void CTray::_ShowStartButtonToolTip() { if (!_ShouldWeShowTheStartButtonBalloon() || SHRestricted(REST_NOSMBALLOONTIP)) { PostMessage(_hwnd, TM_SHOWTRAYBALLOON, TRUE, 0); return; }
if (Tray_StartPanelEnabled()) { // In order to display the Start Menu, we need foreground activation
// so keyboard focus will work properly.
if (SetForegroundWindow(_hwnd)) { LockSetForegroundWindow(LSFW_LOCK); _fForegroundLocked = TRUE;
// Inform the tray that start button is auto-popping, so the tray
// can hold off on showing balloons.
PostMessage(_hwnd, TM_SHOWTRAYBALLOON, FALSE, 0);
// This pushes the start button and causes the start menu to popup.
SendMessage(GetDlgItem(_hwnd, IDC_START), BM_SETSTATE, TRUE, 0);
// Once successfully done once, don't do it again.
_DontShowTheStartButtonBalloonAnyMore(); } } else { PostMessage(_hwnd, TM_SHOWTRAYBALLOON, TRUE, 0); CreateStartButtonBalloon(IDS_STARTMENUBALLOON_TITLE, IDS_STARTMENUBALLOON_TIP); } }
BOOL CTray::_CreateClockWindow() { _hwndNotify = _trayNotify.TrayNotifyCreate(_hwnd, IDC_CLOCK, hinstCabinet); SendMessage(_hwndNotify, TNM_UPDATEVERTICAL, 0, !STUCK_HORIZONTAL(_uStuckPlace));
return BOOLFROMPTR(_hwndNotify); }
BOOL CTray::_InitTrayClass() { WNDCLASS wc = {0};
wc.lpszClassName = TEXT(WNDCLASS_TRAYNOTIFY); wc.style = CS_DBLCLKS; wc.lpfnWndProc = s_WndProc; wc.hInstance = hinstCabinet; wc.hCursor = LoadCursor(NULL, IDC_ARROW); wc.hbrBackground = (HBRUSH)(COLOR_3DFACE+1); wc.cbWndExtra = sizeof(LONG_PTR);
return RegisterClass(&wc); }
HFONT CTray::_CreateStartFont(HWND hwndTray) { HFONT hfontStart = NULL; NONCLIENTMETRICS ncm;
ncm.cbSize = sizeof(ncm); if (SystemParametersInfo(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE)) { WORD wLang = GetUserDefaultLangID();
// Select normal weight font for chinese language.
if (PRIMARYLANGID(wLang) == LANG_CHINESE && ((SUBLANGID(wLang) == SUBLANG_CHINESE_TRADITIONAL) || (SUBLANGID(wLang) == SUBLANG_CHINESE_SIMPLIFIED))) ncm.lfCaptionFont.lfWeight = FW_NORMAL; else ncm.lfCaptionFont.lfWeight = FW_BOLD;
hfontStart = CreateFontIndirect(&ncm.lfCaptionFont); }
return hfontStart; }
// Set the stuck monitor for the tray window
void CTray::_SetStuckMonitor() { // use STICK_LEFT because most of the multi-monitors systems are set up
// side by side. use DEFAULTTONULL because we don't want to get the wrong one
// use the center point to call again in case we failed the first time.
_hmonStuck = MonitorFromRect(&_arStuckRects[STICK_LEFT], MONITOR_DEFAULTTONULL); if (!_hmonStuck) { POINT pt; pt.x = (_arStuckRects[STICK_LEFT].left + _arStuckRects[STICK_LEFT].right)/2; pt.y = (_arStuckRects[STICK_LEFT].top + _arStuckRects[STICK_LEFT].bottom)/2; _hmonStuck = MonitorFromPoint(pt, MONITOR_DEFAULTTONEAREST); }
_hmonOld = _hmonStuck; }
DWORD _GetDefaultTVSDFlags() { DWORD dwFlags = TVSD_TOPMOST;
// if we are on a remote hydra session and if there is no previous saved value,
// do not display the clock.
if (SHGetMachineInfo(GMI_TSCLIENT)) { dwFlags |= TVSD_HIDECLOCK; } return dwFlags; }
void CTray::_GetSaveStateAndInitRects() { TVSDCOMPAT tvsd; RECT rcDisplay; DWORD dwTrayFlags; UINT uStick; SIZE size;
// first fill in the defaults
SetRect(&rcDisplay, 0, 0, g_cxPrimaryDisplay, g_cyPrimaryDisplay);
// size gets defaults
size.cx = _sizeStart.cx + 2 * (g_cxDlgFrame + g_cxBorder); size.cy = _sizeStart.cy + 2 * (g_cyDlgFrame + g_cyBorder);
// sStuckWidths gets minimum
_sStuckWidths.cx = 2 * (g_cxDlgFrame + g_cxBorder); _sStuckWidths.cy = _sizeStart.cy + 2 * (g_cyDlgFrame + g_cyBorder);
_uStuckPlace = STICK_BOTTOM; dwTrayFlags = _GetDefaultTVSDFlags();
_uAutoHide = 0;
// now try to load saved vaules
// BUG : 231077
// Since Tasbar properties don't roam from NT5 to NT4, (NT4 -> NT5 yes)
// Allow roaming from NT4 to NT5 only for the first time the User logs
// on to NT5, so that future changes to NT5 are not lost when the user
// logs on to NT4 after customizing the taskbar properties on NT5.
DWORD cbData1 = sizeof(tvsd); DWORD cbData2 = sizeof(tvsd); if (Reg_GetStruct(g_hkeyExplorer, TEXT("StuckRects2"), TEXT("Settings"), &tvsd, &cbData1) || Reg_GetStruct(g_hkeyExplorer, TEXT("StuckRects"), TEXT("Settings"), &tvsd, &cbData2)) { if (IS_CURRENT_TVSD(tvsd.t) && IsValidSTUCKPLACE(tvsd.t.uStuckPlace)) { _GetDisplayRectFromRect(&rcDisplay, &tvsd.t.rcLastStuck, MONITOR_DEFAULTTONEAREST);
size = tvsd.t.sStuckWidths; _uStuckPlace = tvsd.t.uStuckPlace;
dwTrayFlags = tvsd.t.dwFlags; } else if (MAYBE_WIN95_TVSD(tvsd.w95) && IsValidSTUCKPLACE(tvsd.w95.uStuckPlace)) { _uStuckPlace = tvsd.w95.uStuckPlace; dwTrayFlags = tvsd.w95.dwFlags; if (tvsd.w95.uAutoHide & AH_ON) dwTrayFlags |= TVSD_AUTOHIDE;
switch (_uStuckPlace) { case STICK_LEFT: size.cx = tvsd.w95.dxLeft; break;
case STICK_RIGHT: size.cx = tvsd.w95.dxRight; break;
case STICK_BOTTOM: size.cy = tvsd.w95.dyBottom; break;
case STICK_TOP: size.cy = tvsd.w95.dyTop; break; } } }
ASSERT(IsValidSTUCKPLACE(_uStuckPlace));
//
// use the size only if it is not bogus
//
if (_sStuckWidths.cx < size.cx) _sStuckWidths.cx = size.cx;
if (_sStuckWidths.cy < size.cy) _sStuckWidths.cy = size.cy;
//
// set the tray flags
//
_fAlwaysOnTop = BOOLIFY(dwTrayFlags & TVSD_TOPMOST); _fSMSmallIcons = BOOLIFY(dwTrayFlags & TVSD_SMSMALLICONS); _fHideClock = SHRestricted(REST_HIDECLOCK) || BOOLIFY(dwTrayFlags & TVSD_HIDECLOCK); _uAutoHide = (dwTrayFlags & TVSD_AUTOHIDE) ? AH_ON | AH_HIDING : 0; _RefreshSettings();
//
// initialize stuck rects
//
for (uStick = STICK_LEFT; uStick <= STICK_BOTTOM; uStick++) _MakeStuckRect(&_arStuckRects[uStick], &rcDisplay, _sStuckWidths, uStick);
_UpdateVertical(_uStuckPlace); // Determine which monitor the tray is on using its stuck rectangles
_SetStuckMonitor(); }
IBandSite * BandSite_CreateView(); HRESULT BandSite_SaveView(IUnknown *pbs); LRESULT BandSite_OnMarshallBS(WPARAM wParam, LPARAM lParam);
void CTray::_SaveTrayStuff(void) { TVSD tvsd;
tvsd.dwSize = sizeof(tvsd); tvsd.lSignature = TVSDSIG_CURRENT;
// position
CopyRect(&tvsd.rcLastStuck, &_arStuckRects[_uStuckPlace]); tvsd.sStuckWidths = _sStuckWidths; tvsd.uStuckPlace = _uStuckPlace;
tvsd.dwFlags = 0; if (_fAlwaysOnTop) tvsd.dwFlags |= TVSD_TOPMOST; if (_fSMSmallIcons) tvsd.dwFlags |= TVSD_SMSMALLICONS; if (_fHideClock && !SHRestricted(REST_HIDECLOCK)) tvsd.dwFlags |= TVSD_HIDECLOCK; if (_uAutoHide & AH_ON) tvsd.dwFlags |= TVSD_AUTOHIDE;
// Save in Stuck rects.
Reg_SetStruct(g_hkeyExplorer, TEXT("StuckRects2"), TEXT("Settings"), &tvsd, sizeof(tvsd));
BandSite_SaveView(_ptbs);
return; }
// align toolbar so that buttons are flush with client area
// and make toolbar's buttons to be MENU style
void CTray::_AlignStartButton() { HWND hwndStart = _hwndStart; if (hwndStart) { TCHAR szStart[50]; LoadString(hinstCabinet, _hTheme ? IDS_START : IDS_STARTCLASSIC, szStart, ARRAYSIZE(szStart)); SetWindowText(_hwndStart, szStart);
RECT rcClient; if (!_sizeStart.cx) { Button_GetIdealSize(hwndStart, &_sizeStart); } GetClientRect(_hwnd, &rcClient);
if (rcClient.right < _sizeStart.cx) { SetWindowText(_hwndStart, L""); }
int cyStart = _sizeStart.cy;
if (_hwndTasks) { if (_hTheme) { cyStart = max(cyStart, SendMessage(_hwndTasks, TBC_BUTTONHEIGHT, 0, 0)); } else { cyStart = SendMessage(_hwndTasks, TBC_BUTTONHEIGHT, 0, 0); } }
SetWindowPos(hwndStart, NULL, 0, 0, min(rcClient.right, _sizeStart.cx), cyStart, SWP_NOZORDER | SWP_NOACTIVATE); } }
// Helper function for CDesktopHost so clicking twice on the Start Button
// treats the second click as a dismiss rather than a redisplay.
//
// The crazy state machine goes like this:
//
// SBSM_NORMAL - normal state, nothing exciting
//
// When user opens Start Pane, we become
//
// SBSM_SPACTIVE - start pane is active
//
// If user clicks Start Button while SBSM_SPACTIVE, then we become
//
// SBSM_EATING - eat mouse clicks
//
// Until we receive a WM_MOUSEFIRST/WM_MOUSELAST message, and then
// we return to SBSM_NORMAL.
//
// If user dismisses Start Pane, we go straight to SBSM_NORMAL.
//
//
// We eat the mouse clicks so that the click that the user made
// to "unclick" the start button doesn't cause it to get pushed down
// again (and cause the Start Menu to reopen).
//
#define SBSM_NORMAL 0
#define SBSM_SPACTIVE 1
#define SBSM_EATING 2
void Tray_UnlockStartPane() { // If we had locked out foreground window changes (first-boot),
// then release the lock.
if (c_tray._fForegroundLocked) { c_tray._fForegroundLocked = FALSE; LockSetForegroundWindow(LSFW_UNLOCK); } }
void Tray_SetStartPaneActive(BOOL fActive) { if (fActive) { // Start Pane appearing
c_tray._uStartButtonState = SBSM_SPACTIVE; } else if (c_tray._uStartButtonState != SBSM_EATING) { // Start Pane dismissing, not eating messages -> return to normal
c_tray._uStartButtonState = SBSM_NORMAL; Tray_UnlockStartPane(); } }
// Allow us to do stuff on a "button-down".
LRESULT WINAPI CTray::StartButtonSubclassWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { return c_tray._StartButtonSubclassWndProc(hwnd, uMsg, wParam, lParam); }
LRESULT CTray::_StartButtonSubclassWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { LRESULT lRet;
ASSERT(_pfnButtonProc)
// Is the button going down?
if (uMsg == BM_SETSTATE) { // Is it going Down?
if (wParam) { // DebugMsg(DM_TRACE, "c.stswp: Set state %d", wParam);
// Yes - proceed if it's currently up and it's allowed to be down
if (!_uDown) { // Nope.
INSTRUMENT_STATECHANGE(SHCNFI_STATE_START_DOWN); _uDown = 1;
// If we are going down, then we do not want to popup again until the Start Menu is collapsed
_fAllowUp = FALSE;
SendMessage(_hwndTrayTips, TTM_ACTIVATE, FALSE, 0L);
// Show the button down.
lRet = CallWindowProc(_pfnButtonProc, hwnd, uMsg, wParam, lParam); // Notify the parent.
SendMessage(GetParent(hwnd), WM_COMMAND, (WPARAM)LOWORD(GetDlgCtrlID(hwnd)), (LPARAM)hwnd); _tmOpen = GetTickCount(); return lRet; } else { // Yep. Do nothing.
// fDown = FALSE;
return DefWindowProc(hwnd, uMsg, wParam, lParam); } } else { // DebugMsg(DM_TRACE, "c.stswp: Set state %d", wParam);
// Nope, buttons coming up.
// Is it supposed to be down? Is it not allowed to be up?
if (_uDown == 1 || !_fAllowUp) { INSTRUMENT_STATECHANGE(SHCNFI_STATE_START_UP);
// Yep, do nothing.
_uDown = 2; return DefWindowProc(hwnd, uMsg, wParam, lParam); } else { SendMessage(_hwndTrayTips, TTM_ACTIVATE, TRUE, 0L); // Nope, Forward it on.
_uDown = 0; return CallWindowProc(_pfnButtonProc, hwnd, uMsg, wParam, lParam); } } } else { if (_uStartButtonState == SBSM_EATING && uMsg >= WM_MOUSEFIRST && uMsg <= WM_MOUSELAST) { _uStartButtonState = SBSM_NORMAL;
// Explicitly dismiss the Start Panel because it might be
// stuck in this limbo state where it is open but not the
// foreground window (_ShowStartButtonToolTip does this)
// so it doesn't know that it needs to go away.
ClosePopupMenus(); }
switch (uMsg) { case WM_LBUTTONDOWN: // The button was clicked on, then we don't need no stink'n focus rect.
SendMessage(GetParent(hwnd), WM_UPDATEUISTATE, MAKEWPARAM(UIS_SET, UISF_HIDEFOCUS), 0);
goto ProcessCapture; break;
case WM_KEYDOWN: // The user pressed enter or return or some other bogus key combination when
// the start button had keyboard focus, so show the rect....
SendMessage(GetParent(hwnd), WM_UPDATEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);
if (wParam == VK_RETURN) PostMessage(_hwnd, WM_COMMAND, IDC_KBSTART, 0);
// We do not need the capture, because we do all of our button processing
// on the button down. In fact taking capture for no good reason screws with
// drag and drop into the menus. We're overriding user.
ProcessCapture: lRet = CallWindowProc(_pfnButtonProc, hwnd, uMsg, wParam, lParam); SetCapture(NULL); return lRet; break;
case WM_MOUSEMOVE: { MSG msg;
msg.lParam = lParam; msg.wParam = wParam; msg.message = uMsg; msg.hwnd = hwnd; SendMessage(_hwndTrayTips, TTM_RELAYEVENT, 0, (LPARAM)(LPMSG)& msg);
break; }
case WM_MOUSEACTIVATE: if (_uStartButtonState != SBSM_NORMAL) { _uStartButtonState = SBSM_EATING; return MA_ACTIVATEANDEAT; } break;
//
// Debounce the Start Button. Usability shows that lots of people
// double-click the Start Button, resulting in the menu opening
// and then immediately closing...
//
case WM_NCHITTEST: if (GetTickCount() - _tmOpen < GetDoubleClickTime()) { return HTNOWHERE; } break;
case WM_NULL: break; }
return CallWindowProc(_pfnButtonProc, hwnd, uMsg, wParam, lParam); } }
EXTERN_C const WCHAR c_wzTaskbarTheme[] = L"Taskbar"; EXTERN_C const WCHAR c_wzTaskbarVertTheme[] = L"TaskbarVert";
// create the toolbar with the three buttons and align windows
HWND CTray::_CreateStartButton() { DWORD dwStyle = 0;//BS_BITMAP;
_uStartButtonBalloonTip = RegisterWindowMessage(TEXT("Welcome Finished"));
_uLogoffUser = RegisterWindowMessage(TEXT("Logoff User"));
// Register for MM device changes
_uWinMM_DeviceChange = RegisterWindowMessage(WINMMDEVICECHANGEMSGSTRING);
HWND hwnd = CreateWindowEx(0, WC_BUTTON, TEXT("Start"), WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | BS_PUSHBUTTON | BS_LEFT | BS_VCENTER | dwStyle, 0, 0, 0, 0, _hwnd, (HMENU)IDC_START, hinstCabinet, NULL); if (hwnd) { // taskbar windows are themed under Taskbar subapp name
SetWindowTheme(hwnd, L"Start", NULL);
SendMessage(hwnd, CCM_DPISCALE, TRUE, 0);
// Subclass it.
_hwndStart = hwnd; _pfnButtonProc = SubclassWindow(hwnd, StartButtonSubclassWndProc);
_StartButtonReset(); } return hwnd; }
void CTray::_GetWindowSizes(UINT uStuckPlace, PRECT prcClient, PRECT prcView, PRECT prcNotify) { prcView->top = 0; prcView->left = 0; prcView->bottom = prcClient->bottom; prcView->right = prcClient->right;
if (STUCK_HORIZONTAL(uStuckPlace)) { DWORD_PTR dwNotifySize = SendMessage(_hwndNotify, WM_CALCMINSIZE, prcClient->right / 2, prcClient->bottom); prcNotify->top = 0; prcNotify->left = prcClient->right - LOWORD(dwNotifySize); prcNotify->bottom = HIWORD(dwNotifySize); prcNotify->right = prcClient->right;
prcView->left = _sizeStart.cx + g_cxFrame + 1; prcView->right = prcNotify->left; } else { DWORD_PTR dwNotifySize = SendMessage(_hwndNotify, WM_CALCMINSIZE, prcClient->right, prcClient->bottom / 2); prcNotify->top = prcClient->bottom - HIWORD(dwNotifySize); prcNotify->left = 0; prcNotify->bottom = prcClient->bottom; prcNotify->right = LOWORD(dwNotifySize);
prcView->top = _sizeStart.cy + g_cyTabSpace; prcView->bottom = prcNotify->top; } }
void CTray::_RestoreWindowPos() { WINDOWPLACEMENT wp;
//first restore the stuck postitions
_GetSaveStateAndInitRects();
wp.length = sizeof(wp); wp.showCmd = SW_HIDE;
_uMoveStuckPlace = (UINT)-1; _GetDockedRect(&wp.rcNormalPosition, FALSE);
SendMessage(_hwndNotify, TNM_TRAYHIDE, 0, _fHideClock); SetWindowPlacement(_hwnd, &wp); }
// Get the display (monitor) rectangle from the given arbitrary point
HMONITOR CTray::_GetDisplayRectFromPoint(LPRECT prcDisplay, POINT pt, UINT uFlags) { RECT rcEmpty = {0}; HMONITOR hmon = MonitorFromPoint(pt, uFlags); if (hmon && prcDisplay) GetMonitorRect(hmon, prcDisplay); else if (prcDisplay) *prcDisplay = rcEmpty;
return hmon; }
// Get the display (monitor) rectangle from the given arbitrary rectangle
HMONITOR CTray::_GetDisplayRectFromRect(LPRECT prcDisplay, LPCRECT prcIn, UINT uFlags) { RECT rcEmpty = {0}; HMONITOR hmon = MonitorFromRect(prcIn, uFlags); if (hmon && prcDisplay) GetMonitorRect(hmon, prcDisplay); else if (prcDisplay) *prcDisplay = rcEmpty;
return hmon; }
// Get the display (monitor) rectangle where the taskbar is currently on,
// if that monitor is invalid, get the nearest one.
void CTray::_GetStuckDisplayRect(UINT uStuckPlace, LPRECT prcDisplay) { ASSERT(prcDisplay); BOOL fValid = GetMonitorRect(_hmonStuck, prcDisplay);
if (!fValid) _GetDisplayRectFromRect(prcDisplay, &_arStuckRects[uStuckPlace], MONITOR_DEFAULTTONEAREST); }
void CTray::_AdjustRectForSizingBar(UINT uStuckPlace, LPRECT prc, int iIncrement) { if (iIncrement != 0) { switch (uStuckPlace) { case STICK_BOTTOM: prc->top -= iIncrement * _sizeSizingBar.cy; break; case STICK_TOP: prc->bottom += iIncrement * _sizeSizingBar.cy; break; case STICK_LEFT: prc->right += iIncrement * _sizeSizingBar.cx; break; case STICK_RIGHT: prc->left -= iIncrement * _sizeSizingBar.cx; break; } } else { if (IS_BIDI_LOCALIZED_SYSTEM()) { switch (uStuckPlace) { case STICK_BOTTOM: prc->bottom = prc->top + _sizeSizingBar.cy; break; case STICK_TOP: prc->top = prc->bottom - _sizeSizingBar.cy; break; case STICK_LEFT: prc->right = prc->left + _sizeSizingBar.cx; break; case STICK_RIGHT: prc->left = prc->right - _sizeSizingBar.cx; break; } } else { switch (uStuckPlace) { case STICK_BOTTOM: prc->bottom = prc->top + _sizeSizingBar.cy; break; case STICK_TOP: prc->top = prc->bottom - _sizeSizingBar.cy; break; case STICK_LEFT: prc->left = prc->right - _sizeSizingBar.cx; break; case STICK_RIGHT: prc->right = prc->left + _sizeSizingBar.cx; break; } } } }
// Snap a StuckRect to the edge of a containing rectangle
// fClip determines whether to clip the rectangle if it's off the display or move it onto the screen
void CTray::_MakeStuckRect(LPRECT prcStick, LPCRECT prcBound, SIZE size, UINT uStick) { CopyRect(prcStick, prcBound);
if (_hTheme && (_fCanSizeMove || _fShowSizingBarAlways)) { _AdjustRectForSizingBar(uStick, prcStick, 1); }
if (!_hTheme) { InflateRect(prcStick, g_cxEdge, g_cyEdge); }
if (size.cx < 0) size.cx *= -1; if (size.cy < 0) size.cy *= -1;
switch (uStick) { case STICK_LEFT: prcStick->right = (prcStick->left + size.cx); break; case STICK_TOP: prcStick->bottom = (prcStick->top + size.cy); break; case STICK_RIGHT: prcStick->left = (prcStick->right - size.cx); break; case STICK_BOTTOM: prcStick->top = (prcStick->bottom - size.cy); break; } }
// the screen size has changed, so the docked rectangles need to be
// adjusted to the new screen.
void CTray::_ResizeStuckRects(RECT *arStuckRects) { RECT rcDisplay; _GetStuckDisplayRect(_uStuckPlace, &rcDisplay); for (UINT uStick = STICK_LEFT; uStick <= STICK_BOTTOM; uStick++) { _MakeStuckRect(&arStuckRects[uStick], &rcDisplay, _sStuckWidths, uStick); } }
//*** CTray::InvisibleUnhide -- temporary 'invisible' un-autohide
// DESCRIPTION
// various tray resize routines need the tray to be un-autohide'd for
// stuff to be calculated correctly. so we un-autohide it (invisibly...)
// here. note the WM_SETREDRAW to prevent flicker (nt5:182340).
// note that this is kind of a hack -- ideally the tray code would do
// stuff correctly even if hidden.
//
void CTray::InvisibleUnhide(BOOL fShowWindow) { if (fShowWindow == FALSE) { if (_cHided++ == 0) { SendMessage(_hwnd, WM_SETREDRAW, FALSE, 0); ShowWindow(_hwnd, SW_HIDE); Unhide(); } } else { ASSERT(_cHided > 0); // must be push/pop
if (--_cHided == 0) { _Hide(); ShowWindow(_hwnd, SW_SHOWNA); SendMessage(_hwnd, WM_SETREDRAW, TRUE, 0); } } }
void CTray::VerifySize(BOOL fWinIni, BOOL fRoundUp /* = FALSE */) { RECT rc; BOOL fHiding;
fHiding = (_uAutoHide & AH_HIDING); if (fHiding) { // force it visible so various calculations will happen relative
// to unhidden size/position.
//
// fixes (e.g.) ie5:154536, where dropping a large-icon ISFBand
// onto hidden tray didn't do size negotiation.
//
InvisibleUnhide(FALSE); }
rc = _arStuckRects[_uStuckPlace]; _HandleSizing(0, NULL, _uStuckPlace);
if (!EqualRect(&rc, &_arStuckRects[_uStuckPlace])) { if (fWinIni) { // if we're changing size or position, we need to be unhidden
Unhide(); SizeWindows(); } rc = _arStuckRects[_uStuckPlace];
if (EVAL((_uAutoHide & (AH_ON | AH_HIDING)) != (AH_ON | AH_HIDING))) { _fSelfSizing = TRUE; SetWindowPos(_hwnd, NULL, rc.left, rc.top, RECTWIDTH(rc),RECTHEIGHT(rc), SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOCOPYBITS);
_fSelfSizing = FALSE; }
_StuckTrayChange(); }
if (fWinIni) SizeWindows();
if (fHiding) { InvisibleUnhide(TRUE); } }
HWND CTray::_GetClockWindow(void) { return (HWND)SendMessage(_hwndNotify, TNM_GETCLOCK, 0, 0L); }
UINT _GetStartIDB() { UINT id;
#ifdef _WIN64
if (IsOS(OS_ANYSERVER)) { id = IDB_WIN64ADVSERSTART; } else { id = IDB_WIN64PROSTART; }
#else
if (IsOS(OS_TERMINALCLIENT)) { id = IDB_TERMINALSERVICESBKG; } else if (IsOS(OS_DATACENTER)) { id = IDB_DCSERVERSTARTBKG; } else if (IsOS(OS_ADVSERVER)) { id = IDB_ADVSERVERSTARTBKG; } else if (IsOS(OS_SERVER)) { id = IDB_SERVERSTARTBKG; } else if (IsOS(OS_EMBEDDED)) { id = IDB_EMBEDDED; } else if (IsOS(OS_PERSONAL)) { id = IDB_PERSONALSTARTBKG; } else if (IsOS(OS_MEDIACENTER)) { id = IDB_MEDIACENTER_STARTBKG; } else if (IsOS(OS_TABLETPC)) { id = IDB_TABLETPC_STARTBKG; } else { id = IDB_PROFESSIONALSTARTBKG; }
#endif // _WIN64
return id; }
void CTray::_CreateTrayTips() { _hwndTrayTips = CreateWindowEx(WS_EX_TRANSPARENT, TOOLTIPS_CLASS, NULL, WS_POPUP | TTS_NOPREFIX | TTS_ALWAYSTIP, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, _hwnd, NULL, hinstCabinet, NULL);
if (_hwndTrayTips) { // taskbar windows are themed under Taskbar subapp name
SendMessage(_hwndTrayTips, TTM_SETWINDOWTHEME, 0, (LPARAM)c_wzTaskbarTheme);
SetWindowZorder(_hwndTrayTips, HWND_TOPMOST);
TOOLINFO ti;
ti.cbSize = sizeof(ti); ti.uFlags = TTF_IDISHWND | TTF_EXCLUDETOOLAREA; ti.hwnd = _hwnd; ti.uId = (UINT_PTR)_hwndStart; ti.lpszText = (LPTSTR)MAKEINTRESOURCE(IDS_STARTBUTTONTIP); ti.hinst = hinstCabinet; SendMessage(_hwndTrayTips, TTM_ADDTOOL, 0, (LPARAM)(LPTOOLINFO)&ti);
HWND hwndClock = _GetClockWindow(); if (hwndClock) { ti.uFlags = TTF_EXCLUDETOOLAREA; ti.uId = (UINT_PTR)hwndClock; ti.lpszText = LPSTR_TEXTCALLBACK; ti.rect.left = ti.rect.top = ti.rect.bottom = ti.rect.right = 0; SendMessage(_hwndTrayTips, TTM_ADDTOOL, 0, (LPARAM)(LPTOOLINFO)&ti); } } }
#define SHCNE_STAGINGAREANOTIFICATIONS (SHCNE_CREATE | SHCNE_MKDIR | SHCNE_UPDATEDIR | SHCNE_UPDATEITEM)
LRESULT CTray::_CreateWindows() { if (_CreateStartButton() && _CreateClockWindow()) { //
// We need to set the tray position, before creating
// the view window, because it will call back our
// GetWindowRect member functions.
//
_RestoreWindowPos();
_CreateTrayTips();
SendMessage(_hwndNotify, TNM_HIDECLOCK, 0, _fHideClock);
_ptbs = BandSite_CreateView(); if (_ptbs) { IUnknown_GetWindow(_ptbs, &_hwndRebar); SetWindowStyle(_hwndRebar, RBS_BANDBORDERS, FALSE);
// No need to check the disk space thing for non-privileged users, this reduces activity in the TS case
// and only admins can properly free disk space anyways.
if (IsUserAnAdmin() && !SHRestricted(REST_NOLOWDISKSPACECHECKS)) { SetTimer(_hwnd, IDT_CHECKDISKSPACE, 60 * 1000, NULL); // 60 seconds poll
}
if (IsOS(OS_PERSONAL) || IsOS(OS_PROFESSIONAL)) { SetTimer(_hwnd, IDT_DESKTOPCLEANUP, 24 * 60 * 60 * 1000, NULL); // 24 hour poll
}
if (!SHRestricted(REST_NOCDBURNING)) { LPITEMIDLIST pidlStaging; if (SUCCEEDED(SHGetFolderLocation(NULL, CSIDL_CDBURN_AREA | CSIDL_FLAG_CREATE, NULL, 0, &pidlStaging))) { SHChangeNotifyEntry fsne; fsne.fRecursive = FALSE; fsne.pidl = pidlStaging; _uNotify = SHChangeNotifyRegister(_hwnd, SHCNRF_NewDelivery | SHCNRF_ShellLevel | SHCNRF_InterruptLevel, SHCNE_STAGINGAREANOTIFICATIONS, TM_CHANGENOTIFY, 1, &fsne);
// start off by checking the first time.
_CheckStagingAreaOnTimer();
ILFree(pidlStaging); } } return 1; } }
return -1; }
LRESULT CTray::_InitStartButtonEtc() { UINT uID = _GetStartIDB(); // NOTE: This bitmap is used as a flag in CTaskBar::OnPosRectChangeDB to
// tell when we are done initializing, so we don't resize prematurely
#ifndef _WIN64
if (uID == IDB_MEDIACENTER_STARTBKG || uID == IDB_TABLETPC_STARTBKG) // SP1 Only!
{ HMODULE hmod = LoadLibraryEx(TEXT("winbrand.dll"), NULL, LOAD_LIBRARY_AS_DATAFILE); if (hmod) { _hbmpStartBkg = LoadBitmap(hmod, MAKEINTRESOURCE(uID)); FreeLibrary(hmod); } else { _hbmpStartBkg = LoadBitmap(hinstCabinet, MAKEINTRESOURCE(IDB_PROFESSIONALSTARTBKG)); } } else #endif
{ _hbmpStartBkg = LoadBitmap(hinstCabinet, MAKEINTRESOURCE(uID)); }
if (_hbmpStartBkg) { UpdateWindow(_hwnd); _BuildStartMenu(); _RegisterDropTargets();
if (_CheckAssociations()) CheckWinIniForAssocs();
SendNotifyMessage(HWND_BROADCAST, RegisterWindowMessage(TEXT("TaskbarCreated")), 0, 0);
return 1; }
return -1; }
void CTray::_AdjustMinimizedMetrics() { MINIMIZEDMETRICS mm;
mm.cbSize = sizeof(mm); SystemParametersInfo(SPI_GETMINIMIZEDMETRICS, sizeof(mm), &mm, FALSE); mm.iArrange |= ARW_HIDE; SystemParametersInfo(SPI_SETMINIMIZEDMETRICS, sizeof(mm), &mm, FALSE); }
void CTray::_UpdateBandSiteStyle() { if (_ptbs) { BANDSITEINFO bsi; bsi.dwMask = BSIM_STYLE; _ptbs->GetBandSiteInfo(&bsi);
BOOL fCanMoveBands = _fCanSizeMove && !SHRestricted(REST_NOMOVINGBAND);
DWORD dwStyleNew; if (fCanMoveBands) { dwStyleNew = (bsi.dwStyle & ~(BSIS_NOGRIPPER | BSIS_LOCKED)) | BSIS_AUTOGRIPPER | BSIS_PREFERNOLINEBREAK; } else { dwStyleNew = (bsi.dwStyle & ~BSIS_AUTOGRIPPER) | BSIS_NOGRIPPER | BSIS_LOCKED | BSIS_PREFERNOLINEBREAK; }
// only bother with refresh if something's changed
if (bsi.dwStyle ^ dwStyleNew) { bsi.dwStyle = dwStyleNew; _ptbs->SetBandSiteInfo(&bsi); IUnknown_Exec(_ptbs, &CGID_DeskBand, DBID_BANDINFOCHANGED, 0, NULL, NULL); } } }
BOOL _IsSizeMoveRestricted() { return SHRegGetBoolUSValue(REGSTR_POLICIES_EXPLORER, TEXT("LockTaskbar"), FALSE, FALSE); }
BOOL _IsSizeMoveEnabled() { BOOL fCanSizeMove;
if (_IsSizeMoveRestricted()) { fCanSizeMove = FALSE; } else { fCanSizeMove = SHRegGetBoolUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("TaskbarSizeMove"), FALSE, TRUE); }
return fCanSizeMove; }
void CTray::_RefreshSettings() { BOOL fOldCanSizeMove = _fCanSizeMove; _fCanSizeMove = _IsSizeMoveEnabled(); BOOL fOldShowSizingBarAlways = _fShowSizingBarAlways; _fShowSizingBarAlways = (_uAutoHide & AH_ON) ? TRUE : FALSE;
if ((fOldCanSizeMove != _fCanSizeMove) || (_fShowSizingBarAlways != fOldShowSizingBarAlways)) { BOOL fHiding = (_uAutoHide & AH_HIDING); if (fHiding) { InvisibleUnhide(FALSE); }
RECT rc; GetWindowRect(_hwnd, &rc);
if (_hTheme && !_fShowSizingBarAlways) { if (_fCanSizeMove) { _AdjustRectForSizingBar(_uStuckPlace, &rc, 1); } else { _AdjustRectForSizingBar(_uStuckPlace, &rc, -1); } }
_ClipWindow(FALSE); _fSelfSizing = TRUE; SetWindowPos(_hwnd, NULL, rc.left, rc.top, RECTWIDTH(rc), RECTHEIGHT(rc), SWP_NOZORDER | SWP_FRAMECHANGED); _fSelfSizing = FALSE; _ClipWindow(TRUE);
_arStuckRects[_uStuckPlace] = rc; _StuckTrayChange();
if (fHiding) { InvisibleUnhide(TRUE); }
if (!_fCanSizeMove) { SetWindowPos(_hwnd, NULL, rc.left, rc.top, RECTWIDTH(rc), RECTHEIGHT(rc), SWP_NOZORDER); } } }
LRESULT CTray::_OnCreateAsync() { LRESULT lres;
if (g_dwProfileCAP & 0x00000004) { StartCAP(); }
lres = _InitStartButtonEtc();
if (g_dwProfileCAP & 0x00000004) { StopCAP(); }
_hMainAccel = LoadAccelerators(hinstCabinet, MAKEINTRESOURCE(ACCEL_TRAY));
_RegisterGlobalHotkeys();
// We spin a thread that will process "Load=", "Run=", CU\Run, and CU\RunOnce
RunStartupApps();
// If there were any startup failures that occurred before we were
// ready to handle them, re-raise the failure now that we're ready.
if (_fEarlyStartupFailure) LogFailedStartupApp();
// we run the tray thread that handles Ctrl-Esc with a high priority
// class so that it can respond even on a stressed system.
SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL);
return lres; }
LRESULT CTray::_OnCreate(HWND hwnd) { LRESULT lres = -1; v_hwndTray = hwnd;
Mixer_SetCallbackWindow(hwnd); SendMessage(_hwnd, WM_CHANGEUISTATE, MAKEWPARAM(UIS_INITIALIZE, 0), 0);
_AdjustMinimizedMetrics();
_hTheme = OpenThemeData(hwnd, c_wzTaskbarTheme);
_fShowSizingBarAlways = (_uAutoHide & AH_ON) ? TRUE : FALSE; if (_hTheme) { GetThemeBool(_hTheme, 0, 0, TMT_ALWAYSSHOWSIZINGBAR, &_fShowSizingBarAlways); }
SetWindowStyle(_hwnd, WS_BORDER | WS_THICKFRAME, !_hTheme);
// Force Refresh of frame
SetWindowPos(_hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER | SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE);
if (_HotkeyCreate()) { lres = _CreateWindows(); } return lres; }
typedef struct tagFSEPDATA { LPRECT prc; HMONITOR hmon; CTray* ptray; } FSEPDATA, *PFSEPDATA;
BOOL WINAPI CTray::FullScreenEnumProc(HMONITOR hmon, HDC hdc, LPRECT prc, LPARAM dwData) { BOOL fFullScreen; // Is there a rude app on this monitor?
PFSEPDATA pd = (PFSEPDATA)dwData; if (pd->hmon == hmon) { fFullScreen = TRUE; } else if (pd->prc) { RECT rc, rcMon; GetMonitorRect(hmon, &rcMon); IntersectRect(&rc, &rcMon, pd->prc); fFullScreen = EqualRect(&rc, &rcMon); } else { fFullScreen = FALSE; }
if (hmon == pd->ptray->_hmonStuck) { pd->ptray->_fStuckRudeApp = fFullScreen; }
//
// Tell all the appbars on the same display to get out of the way too
//
pd->ptray->_AppBarNotifyAll(hmon, ABN_FULLSCREENAPP, NULL, fFullScreen);
return TRUE; }
void CTray::HandleFullScreenApp(HWND hwnd) { //
// First check to see if something has actually changed
//
_hwndRude = hwnd;
//
// Enumerate all the monitors, see if the app is rude on each, adjust
// app bars and _fStuckRudeApp as necessary. (Some rude apps, such
// as the NT Logon Screen Saver, span multiple monitors.)
//
FSEPDATA d = {0}; RECT rc; if (hwnd && GetWindowRect(hwnd, &rc)) { d.prc = &rc; d.hmon = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONULL); } d.ptray = this;
EnumDisplayMonitors(NULL, NULL, FullScreenEnumProc, (LPARAM)&d);
//
// Now that we've set _fStuckRudeApp, update the tray's z-order position
//
_ResetZorder();
//
// stop the clock so we don't eat cycles and keep tons of code paged in
//
SendMessage(_hwndNotify, TNM_TRAYHIDE, 0, _fStuckRudeApp);
//
// Finally, let traynot know about whether the tray is hiding
//
SendMessage(_hwndNotify, TNM_RUDEAPP, _fStuckRudeApp, 0); }
BOOL CTray::_IsTopmost() { return BOOLIFY(GetWindowLong(_hwnd, GWL_EXSTYLE) & WS_EX_TOPMOST); }
BOOL CTray::_IsPopupMenuVisible() { HWND hwnd; return ((SUCCEEDED(IUnknown_GetWindow(_pmpStartMenu, &hwnd)) && IsWindowVisible(hwnd)) || (SUCCEEDED(IUnknown_GetWindow(_pmpStartPane, &hwnd)) && IsWindowVisible(hwnd)) || (SUCCEEDED(IUnknown_GetWindow(_pmpTasks, &hwnd)) && IsWindowVisible(hwnd))); }
BOOL CTray::_IsActive() { //
// We say the tray is "active" iff:
//
// (a) the foreground window is the tray or a window owned by the tray, or
// (b) the start menu is showing
//
BOOL fActive = FALSE; HWND hwnd = GetForegroundWindow();
if (hwnd != NULL && (hwnd == _hwnd || (GetWindowOwner(hwnd) == _hwnd))) { fActive = TRUE; } else if (_IsPopupMenuVisible()) { fActive = TRUE; }
return fActive; }
void CTray::_ResetZorder() { HWND hwndZorder, hwndZorderCurrent;
if (g_fDesktopRaised || _fProcessingDesktopRaise || (_fAlwaysOnTop && !_fStuckRudeApp)) { hwndZorder = HWND_TOPMOST; } else if (_IsActive()) { hwndZorder = HWND_TOP; } else if (_fStuckRudeApp) { hwndZorder = HWND_BOTTOM; } else { hwndZorder = HWND_NOTOPMOST; }
//
// We don't have to worry about the HWND_BOTTOM current case -- it's ok
// to keep moving ourselves down to the bottom when there's a rude app.
//
// Nor do we have to worry about the HWND_TOP current case -- it's ok
// to keep moving ourselves up to the top when we're active.
//
hwndZorderCurrent = _IsTopmost() ? HWND_TOPMOST : HWND_NOTOPMOST;
if (hwndZorder != hwndZorderCurrent) { // only do this if somehting has changed.
// this keeps us from popping up over menus as desktop async
// notifies us of it's state
SHForceWindowZorder(_hwnd, hwndZorder); } }
void CTray::_MessageLoop() { for (;;) { MSG msg; if (PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { if (msg.message == WM_QUIT) { if (_hwnd && IsWindow(_hwnd)) { // Tell the tray to save everything off if we got here
// without it being destroyed.
SendMessage(_hwnd, WM_ENDSESSION, 1, 0); } return; // break all the way out of the main loop
}
if (_pmbTasks) { HRESULT hr = _pmbTasks->IsMenuMessage(&msg); if (hr == E_FAIL) { if (_hwndTasks) SendMessage(_hwndTasks, TBC_FREEPOPUPMENUS, 0, 0); } else if (hr == S_OK) { continue; } }
// Note that this needs to come before _pmbStartMenu since
// the start pane sometimes hosts the start menu and it needs
// to handle the start menu messages in that case.
if (_pmbStartPane && _pmbStartPane->IsMenuMessage(&msg) == S_OK) { continue; }
if (_pmbStartMenu && _pmbStartMenu->IsMenuMessage(&msg) == S_OK) { continue; }
if (_hMainAccel && TranslateAccelerator(_hwnd, _hMainAccel, &msg)) { continue; }
TranslateMessage(&msg); DispatchMessage(&msg); } else { WaitMessage(); } } }
BOOL CTray::Init() { // use _COINIT to make sure COM is inited disabling the OLE1 support
return SHCreateThread(MainThreadProc, this, CTF_COINIT, SyncThreadProc) && (_hwnd != NULL); }
int CTray::_GetPart(BOOL fSizingBar, UINT uStuckPlace) { if (fSizingBar) { switch (uStuckPlace) { case STICK_BOTTOM: return TBP_SIZINGBARBOTTOM; case STICK_LEFT: return TBP_SIZINGBARLEFT; case STICK_TOP: return TBP_SIZINGBARTOP; case STICK_RIGHT: return TBP_SIZINGBARRIGHT; } } else { switch (uStuckPlace) { case STICK_BOTTOM: return TBP_BACKGROUNDBOTTOM; case STICK_LEFT: return TBP_BACKGROUNDLEFT; case STICK_TOP: return TBP_BACKGROUNDTOP; case STICK_RIGHT: return TBP_BACKGROUNDRIGHT; } }
return 0; }
void CTray::_UpdateVertical(UINT uStuckPlace, BOOL fForce) { static UINT _uOldStuckPlace = STICK_MAX + 1;
if ((_uOldStuckPlace != uStuckPlace) || fForce) { _uOldStuckPlace = uStuckPlace;
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_uv tray is now %s"), STUCK_HORIZONTAL(uStuckPlace) ? TEXT("HORIZONTAL") : TEXT("VERTICAL"));
if (_ptbs) { // This following function will cause a WINDOWPOSCHANGING which will call DoneMoving
// DoneMoving will then go a screw up all of the window sizing
_fIgnoreDoneMoving = TRUE; BandSite_SetMode(_ptbs, STUCK_HORIZONTAL(uStuckPlace) ? 0 : DBIF_VIEWMODE_VERTICAL); BandSite_SetWindowTheme(_ptbs, (LPWSTR)(STUCK_HORIZONTAL(uStuckPlace) ? c_wzTaskbarTheme : c_wzTaskbarVertTheme)); _fIgnoreDoneMoving = FALSE; }
SendMessage(_hwndNotify, TNM_UPDATEVERTICAL, 0, !STUCK_HORIZONTAL(uStuckPlace));
if (_hTheme) { HDC hdc = GetDC(_hwnd); GetThemePartSize(_hTheme, hdc, _GetPart(TRUE, uStuckPlace), 0, NULL, TS_TRUE, &_sizeSizingBar); ReleaseDC(_hwnd, hdc); } } }
void CTray::_InitBandsite() { ASSERT(_hwnd);
// we initilize the contents after all the infrastructure is created and sized properly
// need to notify which side we're on.
// nt5:211881: set mode *before* load, o.w. Update->RBAutoSize messed up
_UpdateBandSiteStyle();
BandSite_Load(); // now that the mode is set, we need to force an update because we
// explicitly avoided the update during BandSite_Load
_UpdateVertical(_uStuckPlace, TRUE); BandSite_Update(_ptbs); BandSite_UIActivateDBC(_ptbs, DBC_SHOW);
BandSite_FindBand(_ptbs, CLSID_TaskBand, IID_PPV_ARG(IDeskBand, &_pdbTasks), NULL, NULL); IUnknown_GetWindow(_pdbTasks, &_hwndTasks);
// Now that bandsite is ready, set the correct size
// BUG: 606284 was caused by the tray do _HandleSizing before the BandSite's band were ready to go
// they all had their default band size of 0,0 so the tray became 0 px height
_fBandSiteInitialized = TRUE; VerifySize(FALSE, TRUE); }
void CTray::_KickStartAutohide() { if (_uAutoHide & AH_ON) { // tray always starts out hidden on autohide
_uAutoHide = AH_ON | AH_HIDING;
// we and many apps rely upon us having calculated the size correctly
Unhide();
// register it
if (!_AppBarSetAutoHideBar2(_hwnd, TRUE, _uStuckPlace)) { // don't bother putting up UI in this case
// if someone is there just silently convert to normal
// (the shell is booting who would be there anyway?)
_SetAutoHideState(FALSE); } } }
void CTray::_InitNonzeroGlobals() { // initalize globals that need to be non-zero
if (GetSystemMetrics(SM_SLOWMACHINE)) { _dtSlideHide = 0; // dont slide the tray out
_dtSlideShow = 0; } else { _dtSlideHide = 400; _dtSlideShow = 200; }
_RefreshSettings(); }
void CTray::_CreateTrayWindow() { _InitTrayClass();
_uMsgEnableUserTrackedBalloonTips = RegisterWindowMessage(ENABLE_BALLOONTIP_MESSAGE);
_fNoToolbarsOnTaskbarPolicyEnabled = (SHRestricted(REST_NOTOOLBARSONTASKBAR) != 0);
DWORD dwExStyle = WS_EX_WINDOWEDGE | WS_EX_TOOLWINDOW; // Don't fadein because layered windows suck
// If you create a layered window on a non-active desktop then the window goes black
dwExStyle |= IS_BIDI_LOCALIZED_SYSTEM() ? dwExStyleRTLMirrorWnd : 0L;
CreateWindowEx(dwExStyle, TEXT(WNDCLASS_TRAYNOTIFY), NULL, WS_CLIPCHILDREN | WS_POPUP, 0, 0, 0, 0, NULL, NULL, hinstCabinet, (void*)this);
}
DWORD WINAPI CTray::SyncThreadProc(void *pv) { CTray* ptray = (CTray*)pv; return ptray->_SyncThreadProc(); }
DWORD CTray::_SyncThreadProc() { if (g_dwStopWatchMode) StopWatch_StartTimed(SWID_STARTUP, TEXT("_SyncThreadProc"), SPMODE_SHELL | SPMODE_DEBUGOUT, GetPerfTime());
if (g_dwProfileCAP & 0x00000002) StartCAP();
InitializeCriticalSection(&_csHotkey);
OleInitialize(NULL); // matched in MainThreadProc()
ClassFactory_Start();
_InitNonzeroGlobals(); _ssomgr.Init();
//
// Watch the registry key that tells us which app is the default
// web browser, so we can track it in
// HKLM\Software\Clients\StartMenuInternet. We have to track it
// ourselves because downlevel browsers won't know about it.
//
// We need to do this only if we have write access to the key.
// (If we don't have write access, then we can't change it,
// so there's no point watching for it to change...)
//
// Well, okay, even if we only have read access, we have to do
// it once in case it changed while we were logged off.
//
// The order of these operations is important...
//
// 1. Migrate browser settings.
// 2. Build default MFU. (Depends on browser settings.)
// 3. Create tray window. (Relies on value MFU.)
//
_hHTTPEvent = CreateEvent(NULL, FALSE, TRUE, NULL); if (_hHTTPEvent) { // Make one migration pass immediately so HandleFirstTime
// sees good information. This also kick-starts the
// registry change notification process if the current user
// has write permission.
_MigrateOldBrowserSettings();
if (RegisterWaitForSingleObject(&_hHTTPWait, _hHTTPEvent, _MigrateOldBrowserSettingsCB, this, INFINITE, WT_EXECUTEDEFAULT)) { // Yay, everything is fine.
} }
// Build the default MFU if necessary
HandleFirstTime();
_CreateTrayWindow();
if (_hwnd && _ptbs) { _ResetZorder(); // obey the "always on top" flag
_KickStartAutohide();
_InitBandsite();
_ClipWindow(TRUE); // make sure we clip the taskbar to the current monitor before showing it
// it looks really strange for the tray to pop up and rehide at logon
// if we are autohide don't activate the tray when we show it
// if we aren't autohide do what Win95 did (tray is active by default)
ShowWindow(_hwnd, (_uAutoHide & AH_HIDING) ? SW_SHOWNA : SW_SHOW);
UpdateWindow(_hwnd); _StuckTrayChange();
// get the system background scheduler thread
IShellTaskScheduler* pScheduler; if (SUCCEEDED(CoCreateInstance(CLSID_SharedTaskScheduler, NULL, CLSCTX_INPROC, IID_PPV_ARG(IShellTaskScheduler, &pScheduler)))) { AddMenuItemsCacheTask(pScheduler, Tray_StartPanelEnabled()); pScheduler->Release(); }
SetTimer(_hwnd, IDT_HANDLEDELAYBOOTSTUFF, 5 * 1000, NULL); }
if (g_dwProfileCAP & 0x00020000) StopCAP();
if (g_dwStopWatchMode) StopWatch_StopTimed(SWID_STARTUP, TEXT("_SyncThreadProc"), SPMODE_SHELL | SPMODE_DEBUGOUT, GetPerfTime());
return FALSE; }
// the rest of the thread proc that includes the message loop
DWORD WINAPI CTray::MainThreadProc(void *pv) { CTray* ptray = (CTray*)pv;
if (!ptray->_hwnd) return FALSE;
ptray->_OnCreateAsync();
PERFSETMARK("ExplorerStartMenuReady");
ptray->_MessageLoop();
ClassFactory_Stop(); OleUninitialize(); // matched in _SyncThreadProc()
return FALSE; }
#define DM_IANELHK 0
#define HKIF_NULL 0
#define HKIF_CACHED 1
#define HKIF_FREEPIDLS 2
typedef struct { LPITEMIDLIST pidlFolder; LPITEMIDLIST pidlItem; WORD wGHotkey; WORD wFlags; } HOTKEYITEM;
UINT CTray::_HotkeyGetFreeItemIndex(void) { int i, cItems; HOTKEYITEM *phki;
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
cItems = DSA_GetItemCount(_hdsaHKI); for (i=0; i<cItems; i++) { phki = (HOTKEYITEM *)DSA_GetItemPtr(_hdsaHKI, i); if (!phki->wGHotkey) { ASSERT(!phki->pidlFolder); ASSERT(!phki->pidlItem); break; } } return i; }
// Weird, Global hotkeys use different flags for modifiers than window hotkeys
// (and hotkeys returned by the hotkey control)
WORD _MapHotkeyToGlobalHotkey(WORD wHotkey) { UINT nMod = 0;
// Map the modifiers.
if (HIBYTE(wHotkey) & HOTKEYF_SHIFT) nMod |= MOD_SHIFT; if (HIBYTE(wHotkey) & HOTKEYF_CONTROL) nMod |= MOD_CONTROL; if (HIBYTE(wHotkey) & HOTKEYF_ALT) nMod |= MOD_ALT; UINT nVirtKey = LOBYTE(wHotkey); return (WORD)((nMod*256) + nVirtKey); }
// NB This takes a regular window hotkey not a global hotkey (it does
// the convertion for you).
int CTray::HotkeyAdd(WORD wHotkey, LPCITEMIDLIST pidlFolder, LPCITEMIDLIST pidlItem, BOOL fClone) { if (wHotkey) { LPCITEMIDLIST pidl1, pidl2;
HOTKEYITEM hki;
EnterCriticalSection(&_csHotkey);
int i = _HotkeyGetFreeItemIndex();
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
// DebugMsg(DM_IANELHK, "c.hl_a: Hotkey %x with id %d.", wHotkey, i);
if (fClone) { pidl1 = ILClone(pidlFolder); pidl2 = ILClone(pidlItem); hki.wFlags = HKIF_FREEPIDLS; } else { pidl1 = pidlFolder; pidl2 = pidlItem; hki.wFlags = HKIF_NULL; }
hki.pidlFolder = (LPITEMIDLIST)pidl1; hki.pidlItem = (LPITEMIDLIST)pidl2; hki.wGHotkey = _MapHotkeyToGlobalHotkey(wHotkey); DSA_SetItem(_hdsaHKI, i, &hki);
LeaveCriticalSection(&_csHotkey); return i; }
return -1; }
// NB Cached hotkeys have their own pidls that need to be free but
// regular hotkeys just keep a pointer to pidls used by the startmenu and
// so don't.
int CTray::_HotkeyAddCached(WORD wGHotkey, LPITEMIDLIST pidl) { int i = -1;
if (wGHotkey) { LPITEMIDLIST pidlItem = ILClone(ILFindLastID(pidl));
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
if (pidlItem) { if (ILRemoveLastID(pidl)) { HOTKEYITEM hki;
EnterCriticalSection(&_csHotkey);
i = _HotkeyGetFreeItemIndex();
// DebugMsg(DM_IANELHK, "c.hl_ac: Hotkey %x with id %d.", wGHotkey, i);
hki.pidlFolder = pidl; hki.pidlItem = pidlItem; hki.wGHotkey = wGHotkey; hki.wFlags = HKIF_CACHED | HKIF_FREEPIDLS; DSA_SetItem(_hdsaHKI, i, &hki);
LeaveCriticalSection(&_csHotkey); } } }
return i; }
// NB Again, this takes window hotkey not a Global one.
// NB This doesn't delete cached hotkeys.
int CTray::_HotkeyRemove(WORD wHotkey) { int iRet = -1; if (EVAL(_hdsaHKI)) { int i, cItems; HOTKEYITEM *phki; WORD wGHotkey;
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
// DebugMsg(DM_IANELHK, "c.hl_r: Remove hotkey for %x" , wHotkey);
// Unmap the modifiers.
wGHotkey = _MapHotkeyToGlobalHotkey(wHotkey); EnterCriticalSection(&_csHotkey);
cItems = DSA_GetItemCount(_hdsaHKI); for (i=0; i<cItems; i++) { phki = (HOTKEYITEM *)DSA_GetItemPtr(_hdsaHKI, i); if (phki && !(phki->wFlags & HKIF_CACHED) && (phki->wGHotkey == wGHotkey)) { // DebugMsg(DM_IANELHK, "c.hl_r: Invalidating %d", i);
if (phki->wFlags & HKIF_FREEPIDLS) { if (phki->pidlFolder) ILFree(phki->pidlFolder); if (phki->pidlItem) ILFree(phki->pidlItem); } phki->wGHotkey = 0; phki->pidlFolder = NULL; phki->pidlItem = NULL; phki->wFlags &= ~HKIF_FREEPIDLS; iRet = i; break; } } LeaveCriticalSection(&_csHotkey); }
return iRet; }
// NB This takes a global hotkey.
int CTray::_HotkeyRemoveCached(WORD wGHotkey) { int iRet = -1; int i, cItems; HOTKEYITEM *phki;
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
// DebugMsg(DM_IANELHK, "c.hl_rc: Remove hotkey for %x" , wGHotkey);
EnterCriticalSection(&_csHotkey);
cItems = DSA_GetItemCount(_hdsaHKI); for (i=0; i<cItems; i++) { phki = (HOTKEYITEM *)DSA_GetItemPtr(_hdsaHKI, i); if (phki && (phki->wFlags & HKIF_CACHED) && (phki->wGHotkey == wGHotkey)) { // DebugMsg(DM_IANELHK, "c.hl_r: Invalidating %d", i);
if (phki->wFlags & HKIF_FREEPIDLS) { if (phki->pidlFolder) ILFree(phki->pidlFolder); if (phki->pidlItem) ILFree(phki->pidlItem); } phki->pidlFolder = NULL; phki->pidlItem = NULL; phki->wGHotkey = 0; phki->wFlags &= ~(HKIF_CACHED | HKIF_FREEPIDLS); iRet = i; break; } } LeaveCriticalSection(&_csHotkey);
return iRet; }
// NB Some (the ones not marked HKIF_FREEPIDLS) of the items in the list of hotkeys
// have pointers to idlists used by the filemenu so they are only valid for
// the lifetime of the filemenu.
BOOL CTray::_HotkeyCreate(void) { if (!_hdsaHKI) { // DebugMsg(DM_TRACE, "c.hkl_c: Creating global hotkey list.");
_hdsaHKI = DSA_Create(sizeof(HOTKEYITEM), 0); }
if (_hdsaHKI) return TRUE;
return FALSE; }
void CTray::_BuildStartMenu() { HRESULT hr;
ClosePopupMenus();
//
// Avoid redundant rebuilds: Peek out any pending SBM_REBUILDMENU messages
// since the rebuild we're about to do will take care of it. Do this
// before destroying the Start Menu so we never yield while there isn't
// a Start Menu.
//
MSG msg; while (PeekMessage(&msg, _hwnd, SBM_REBUILDMENU, SBM_REBUILDMENU, PM_REMOVE | PM_NOYIELD)) { // Keep sucking them out
}
_DestroyStartMenu();
if (Tray_StartPanelEnabled()) { hr = DesktopV2_Create(&_pmpStartPane, &_pmbStartPane, &_pvStartPane); DesktopV2_Build(_pvStartPane); } else { hr = StartMenuHost_Create(&_pmpStartMenu, &_pmbStartMenu); if (SUCCEEDED(hr)) { IBanneredBar* pbb;
hr = _pmpStartMenu->QueryInterface(IID_PPV_ARG(IBanneredBar, &pbb)); if (SUCCEEDED(hr)) { pbb->SetBitmap(_hbmpStartBkg); if (_fSMSmallIcons) pbb->SetIconSize(BMICON_SMALL); else pbb->SetIconSize(BMICON_LARGE);
pbb->Release(); } } }
if (FAILED(hr)) { TraceMsg(TF_ERROR, "Could not create StartMenu"); } }
void CTray::_DestroyStartMenu() { IUnknown_SetSite(_pmpStartMenu, NULL); ATOMICRELEASET(_pmpStartMenu, IMenuPopup); ATOMICRELEASET(_pmbStartMenu, IMenuBand); IUnknown_SetSite(_pmpStartPane, NULL); ATOMICRELEASET(_pmpStartPane, IMenuPopup); ATOMICRELEASET(_pmbStartPane, IMenuBand); ATOMICRELEASET(_pmpTasks, IMenuPopup); ATOMICRELEASET(_pmbTasks, IMenuBand); }
void CTray::ForceStartButtonUp() { MSG msg; // don't do that check message pos because it gets screwy with
// keyboard cancel. and besides, we always want it cleared after
// track menu popup is done.
// do it twice to be sure it's up due to the _uDown cycling twice in
// the subclassing stuff
// pull off any button downs
PeekMessage(&msg, _hwndStart, WM_LBUTTONDOWN, WM_LBUTTONDOWN, PM_REMOVE); SendMessage(_hwndStart, BM_SETSTATE, FALSE, 0); SendMessage(_hwndStart, BM_SETSTATE, FALSE, 0);
if (_hwndTasks) SendMessage(_hwndTasks, TBC_SETPREVFOCUS, 0, NULL);
PostMessage(_hwnd, TM_STARTMENUDISMISSED, 0, 0); }
void Tray_OnStartMenuDismissed() { c_tray._bMainMenuInit = FALSE; // Tell the Start Button that it's allowed to be in the up position now. This
// prevents the problem where the start menu is displayed but the button is
// in the up position... This happens when dialog boxes are displayed
c_tray._fAllowUp = TRUE;
// Now tell it to be in the up position
c_tray.ForceStartButtonUp();
PostMessage(v_hwndTray, TM_SHOWTRAYBALLOON, TRUE, 0); }
#ifdef FEATURE_STARTPAGE
void Tray_OnStartPageDismissed() { if (g_fDesktopRaised) c_tray._RaiseDesktop(!g_fDesktopRaised, TRUE);
Tray_OnStartMenuDismissed(); } void Tray_MenuInvoke(int idCmd) { c_tray.ContextMenuInvoke(idCmd); } #endif
int CTray::_TrackMenu(HMENU hmenu) { TPMPARAMS tpm; int iret;
tpm.cbSize = sizeof(tpm); GetClientRect(_hwndStart, &tpm.rcExclude);
RECT rcClient; GetClientRect(_hwnd, &rcClient); tpm.rcExclude.bottom = min(tpm.rcExclude.bottom, rcClient.bottom);
MapWindowPoints(_hwndStart, NULL, (LPPOINT)&tpm.rcExclude, 2);
SendMessage(_hwndTrayTips, TTM_ACTIVATE, FALSE, 0L); iret = TrackPopupMenuEx(hmenu, TPM_VERTICAL | TPM_BOTTOMALIGN | TPM_RETURNCMD, tpm.rcExclude.left, tpm.rcExclude.bottom, _hwnd, &tpm);
SendMessage(_hwndTrayTips, TTM_ACTIVATE, TRUE, 0L); return iret; }
/*------------------------------------------------------------------
** Respond to a button's pressing by bringing up the appropriate menu. ** Clean up the button depression when the menu is dismissed. **------------------------------------------------------------------*/ void CTray::_ToolbarMenu() { RECTL rcExclude; POINTL ptPop; DWORD dwFlags = MPPF_KEYBOARD; // Assume that we're popuping
// up because of the keyboard
// This is for the underlines on NT5
if (_hwndTasks) SendMessage(_hwndTasks, TBC_FREEPOPUPMENUS, 0, 0);
if (_hwndStartBalloon) { _DontShowTheStartButtonBalloonAnyMore(); ShowWindow(_hwndStartBalloon, SW_HIDE); _DestroyStartButtonBalloon(); }
SetActiveWindow(_hwnd); _bMainMenuInit = TRUE;
// Exclude rect is the VISIBLE portion of the Start Button.
{ RECT rcParent; GetClientRect(_hwndStart, (RECT *)&rcExclude); MapWindowRect(_hwndStart, HWND_DESKTOP, &rcExclude);
GetClientRect(_hwnd, &rcParent); MapWindowRect(_hwnd, HWND_DESKTOP, &rcParent);
IntersectRect((RECT*)&rcExclude, (RECT*)&rcExclude, &rcParent); } ptPop.x = rcExclude.left; ptPop.y = rcExclude.top;
// Close any Context Menus
SendMessage(_hwnd, WM_CANCELMODE, 0, 0);
// Is the "Activate" button down (If the buttons are swapped, then it's the
// right button, otherwise the left button)
if (GetKeyState(GetSystemMetrics(SM_SWAPBUTTON)?VK_RBUTTON:VK_LBUTTON) < 0) { dwFlags = 0; // Then set to the default
}
#ifdef FEATURE_STARTPAGE
if(!(GetAsyncKeyState(VK_SHIFT) < 0) && Tray_ShowStartPageEnabled()) //If the start page is on, then just raise the desktop!
{ _hwndFocusBeforeRaise = GetForegroundWindow(); _RaiseDesktop(TRUE, TRUE); Tray_SetStartPaneActive(TRUE); } else #endif //FEATURE_STARTPAGE
{ IMenuPopup **ppmpToDisplay = &_pmpStartMenu;
if (_pmpStartPane) { ppmpToDisplay = &_pmpStartPane; }
// Close race window: The user can click on the Start Button
// before we get a chance to rebuild the Start Menu in its new
// form. In such case, rebuild it now.
if (!*ppmpToDisplay) { TraceMsg(TF_WARNING, "e.tbm: Rebuilding Start Menu"); _BuildStartMenu(); }
if (*ppmpToDisplay && SUCCEEDED((*ppmpToDisplay)->Popup(&ptPop, &rcExclude, dwFlags))) { // All is well - the menu is up
TraceMsg(DM_MISC, "e.tbm: dwFlags=%x (0=mouse 1=key)", dwFlags); } else { TraceMsg(TF_WARNING, "e.tbm: %08x->Popup failed", *ppmpToDisplay); // Start Menu failed to display -- reset the Start Button
// so the user can click it again to try again
Tray_OnStartMenuDismissed(); }
if (dwFlags == MPPF_KEYBOARD) { // Since the user has launched the start button by Ctrl-Esc, or some other worldly
// means, then turn the rect on.
SendMessage(_hwndStart, WM_UPDATEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0); } } }
HRESULT CTray::_AppBarSetState(UINT uFlags) { if (uFlags & ~(ABS_AUTOHIDE | ABS_ALWAYSONTOP)) { return E_INVALIDARG; } else { _SetAutoHideState(uFlags & ABS_AUTOHIDE); _UpdateAlwaysOnTop(uFlags & ABS_ALWAYSONTOP); return S_OK; } }
//
// can't use SubtractRect sometimes because of inclusion limitations
//
void CTray::_AppBarSubtractRect(PAPPBAR pab, LPRECT lprc) { switch (pab->uEdge) { case ABE_TOP: if (pab->rc.bottom > lprc->top) lprc->top = pab->rc.bottom; break;
case ABE_LEFT: if (pab->rc.right > lprc->left) lprc->left = pab->rc.right; break;
case ABE_BOTTOM: if (pab->rc.top < lprc->bottom) lprc->bottom = pab->rc.top; break;
case ABE_RIGHT: if (pab->rc.left < lprc->right) lprc->right = pab->rc.left; break; } }
void CTray::_AppBarSubtractRects(HMONITOR hmon, LPRECT lprc) { int i;
if (!_hdpaAppBars) return;
i = DPA_GetPtrCount(_hdpaAppBars);
while (i--) { PAPPBAR pab = (PAPPBAR)DPA_GetPtr(_hdpaAppBars, i);
//
// autohide bars are not in our DPA or live on the edge
// don't subtract the appbar if it's on a different display
// don't subtract the appbar if we are in a locked desktop
//
// if (hmon == MonitorFromRect(&pab->rc, MONITOR_DEFAULTTONULL))
if (hmon == MonitorFromRect(&pab->rc, MONITOR_DEFAULTTONULL) && !_fIsDesktopLocked) _AppBarSubtractRect(pab, lprc); } }
#define RWA_NOCHANGE 0
#define RWA_CHANGED 1
#define RWA_BOTTOMMOSTTRAY 2
// (dli) This is a hack put in because bottommost tray is wierd, once
// it becomes a toolbar, this code should go away.
// In the bottommost tray case, even though the work area has not changed,
// we should notify the desktop.
int CTray::_RecomputeWorkArea(HWND hwndCause, HMONITOR hmon, LPRECT prcWork) { int iRet = RWA_NOCHANGE; MONITORINFO mi; mi.cbSize = sizeof(mi);
if (_fIsLogoff) { if (GetMonitorInfo(hmon, &mi)) { *prcWork = mi.rcMonitor; iRet = RWA_CHANGED; } return iRet; }
ASSERT(!_fIsLogoff);
//
// tell everybody that this window changed positions _on_this_monitor_
// note that this notify happens even if we don't change the work area
// since it may cause another app to change the work area...
//
PostMessage(_hwnd, TM_RELAYPOSCHANGED, (WPARAM)hwndCause, (LPARAM)hmon); //
// get the current info for this monitor
// we subtract down from the display rectangle to build the work area
//
if (GetMonitorInfo(hmon, &mi)) { //
// don't subtract the tray if it is autohide
// don't subtract the tray if it is not always on top
// don't subtract the tray if it's on a different display
// don't subtract the tray if it is on a different desktop
//
if (!(_uAutoHide & AH_ON) && _fAlwaysOnTop && (hmon == _hmonStuck) && !_fIsDesktopLocked) { SubtractRect(prcWork, &mi.rcMonitor, &_arStuckRects[_uStuckPlace]); } else *prcWork = mi.rcMonitor;
//
// now subtract off all the appbars on this display
//
_AppBarSubtractRects(hmon, prcWork);
//
// return whether we changed anything
//
if (!EqualRect(prcWork, &mi.rcWork)) iRet = RWA_CHANGED; else if (!(_uAutoHide & AH_ON) && (!_fAlwaysOnTop) && (!IsRectEmpty(&_arStuckRects[_uStuckPlace]))) // NOTE: This is the bottommost case, it only applies for the tray.
// this should be taken out when bottommost tray becomes toolbar
iRet = RWA_BOTTOMMOSTTRAY; } else { iRet = RWA_NOCHANGE; } return iRet; }
void RedrawDesktop(RECT *prcWork) { // This rect point should always be valid (dli)
RIP(prcWork); if (v_hwndDesktop && g_fCleanBoot) { MapWindowRect(NULL, v_hwndDesktop, prcWork);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac invalidating desktop rect {%d,%d,%d,%d}"), prcWork->left, prcWork->top, prcWork->right, prcWork->bottom); RedrawWindow(v_hwndDesktop, prcWork, NULL, RDW_INVALIDATE | RDW_ERASE | RDW_ALLCHILDREN); } }
void CTray::_StuckAppChange(HWND hwndCause, LPCRECT prcOld, LPCRECT prcNew, BOOL bTray) { RECT rcWork1, rcWork2; HMONITOR hmon1, hmon2 = 0; int iChange = 0;
//
// PERF FEATURE:
// there are cases where we end up setting the work area multiple times
// we need to keep a static array of displays that have changed and a
// reenter count so we can avoid pain of sending notifies to the whole
// planet...
//
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac from_AppBar %08X"), hwndCause);
//
// see if the work area changed on the display containing prcOld
//
if (prcOld) { if (bTray) hmon1 = _hmonOld; else hmon1 = MonitorFromRect(prcOld, MONITOR_DEFAULTTONEAREST);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac old pos {%d,%d,%d,%d} on monitor %08X"), prcOld->left, prcOld->top, prcOld->right, prcOld->bottom, hmon1);
if (hmon1) { int iret = _RecomputeWorkArea(hwndCause, hmon1, &rcWork1); if (iret == RWA_CHANGED) iChange = 1; if (iret == RWA_BOTTOMMOSTTRAY) iChange = 4; } } else hmon1 = NULL;
//
// see if the work area changed on the display containing prcNew
//
if (prcNew) { hmon2 = MonitorFromRect(prcNew, MONITOR_DEFAULTTONULL);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac new pos {%d,%d,%d,%d} on monitor %08X"), prcNew->left, prcNew->top, prcNew->right, prcNew->bottom, hmon2);
if (hmon2 && (hmon2 != hmon1)) { int iret = _RecomputeWorkArea(hwndCause, hmon2, &rcWork2); if (iret == RWA_CHANGED) iChange |= 2; else if (iret == RWA_BOTTOMMOSTTRAY && (!iChange)) iChange = 4; } }
//
// did the prcOld's display's work area change?
//
if (iChange & 1) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac changing work area for monitor %08X"), hmon1);
// only send SENDWININICHANGE if the desktop has been created (otherwise
// we will hang the explorer because the main thread is currently blocked)
SystemParametersInfo(SPI_SETWORKAREA, TRUE, &rcWork1, (iChange == 1 && v_hwndDesktop)? SPIF_SENDWININICHANGE : 0);
RedrawDesktop(&rcWork1); }
//
// did the prcOld's display's work area change?
//
if (iChange & 2) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sac changing work area for monitor %08X"), hmon2);
// only send SENDWININICHANGE if the desktop has been created (otherwise
// we will hang the explorer because the main thread is currently blocked)
SystemParametersInfo(SPI_SETWORKAREA, TRUE, &rcWork2, v_hwndDesktop ? SPIF_SENDWININICHANGE : 0);
RedrawDesktop(&rcWork2); }
// only send if the desktop has been created...
// need to send if it's from the tray or any outside app that causes size change
// from the tray because autohideness will affect desktop size even if it's not always on top
if ((bTray || iChange == 4) && v_hwndDesktop) SendMessage(v_hwndDesktop, WM_SIZE, 0, 0); }
void CTray::_StuckTrayChange() { // We used to blow off the _StuckAppChange when the tray was in autohide
// mode, since moving or resizing an autohid tray doesn't change the
// work area. Now we go ahead with the _StuckAppChange in this case
// too. The reason is that we can get into a state where the work area
// size is incorrect, and we want the taskbar to always be self-repairing
// in this case (so that resizing or moving the taskbar will correct the
// work area size).
//
// pass a NULL window here since we don't want to hand out our window and
// the tray doesn't get these anyway (nobody cares as long as its not them)
//
_StuckAppChange(NULL, &_rcOldTray, &_arStuckRects[_uStuckPlace], TRUE);
//
// save off the new tray position...
//
_rcOldTray = _arStuckRects[_uStuckPlace]; }
UINT CTray::_RecalcStuckPos(LPRECT prc) { RECT rcDummy; POINT pt;
if (!prc) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_rsp no rect supplied, using window rect"));
prc = &rcDummy; GetWindowRect(_hwnd, prc); }
// use the center of the original drag rect as a staring point
pt.x = prc->left + RECTWIDTH(*prc) / 2; pt.y = prc->top + RECTHEIGHT(*prc) / 2;
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_rsp rect is {%d, %d, %d, %d} point is {%d, %d}"), prc->left, prc->top, prc->right, prc->bottom, pt.x, pt.y);
// reset this so the drag code won't give it preference
_uMoveStuckPlace = (UINT)-1;
// simulate a drag back to figure out where we originated from
// you may be tempted to remove this. before you do think about dragging
// the tray across monitors and then hitting ESC...
return _CalcDragPlace(pt); }
/*------------------------------------------------------------------
** the position is changing in response to a move operation. ** ** if the docking status changed, we need to get a new size and ** maybe a new frame style. change the WINDOWPOS to reflect ** these changes accordingly. **------------------------------------------------------------------*/ void CTray::_DoneMoving(LPWINDOWPOS lpwp) { RECT rc, *prc;
if ((_uMoveStuckPlace == (UINT)-1) || (_fIgnoreDoneMoving)) return;
if (_fSysSizing) _fDeferedPosRectChange = TRUE;
rc.left = lpwp->x; rc.top = lpwp->y; rc.right = lpwp->x + lpwp->cx; rc.bottom = lpwp->y + lpwp->cy;
prc = &_arStuckRects[_uMoveStuckPlace];
if (!EqualRect(prc, &rc)) { _uMoveStuckPlace = _RecalcStuckPos(&rc); prc = &_arStuckRects[_uMoveStuckPlace]; }
// Get the new hmonitor
_hmonStuck = MonitorFromRect(prc, MONITOR_DEFAULTTONEAREST);
lpwp->x = prc->left; lpwp->y = prc->top; lpwp->cx = RECTWIDTH(*prc); lpwp->cy = RECTHEIGHT(*prc);
lpwp->flags &= ~(SWP_NOMOVE | SWP_NOSIZE);
// if we were autohiding, we need to update our appbar autohide rect
if (_uAutoHide & AH_ON) { // unregister us from the old side
_AppBarSetAutoHideBar2(_hwnd, FALSE, _uStuckPlace); }
// All that work might've changed _uMoveStuckPlace (since there
// was a lot of message traffic), so check one more time.
// Somehow, NT Stress manages to get us in here with an invalid
// uMoveStuckPlace.
if (IsValidSTUCKPLACE(_uMoveStuckPlace)) { // remember the new state
_uStuckPlace = _uMoveStuckPlace; } _uMoveStuckPlace = (UINT)-1; _UpdateVertical(_uStuckPlace);
_HandleSizing(0, prc, _uStuckPlace); if ((_uAutoHide & AH_ON) && !_AppBarSetAutoHideBar2(_hwnd, TRUE, _uStuckPlace)) { _AutoHideCollision(); } }
UINT CTray::_CalcDragPlace(POINT pt) { UINT uPlace = _uMoveStuckPlace;
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_cdp starting point is {%d, %d}"), pt.x, pt.y);
//
// if the mouse is currently over the tray position leave it alone
//
if ((uPlace == (UINT)-1) || !PtInRect(&_arStuckRects[uPlace], pt)) { HMONITOR hmonDrag; SIZE screen, error; UINT uHorzEdge, uVertEdge; RECT rcDisplay, *prcStick;
//
// which display is the mouse on?
//
hmonDrag = _GetDisplayRectFromPoint(&rcDisplay, pt, MONITOR_DEFAULTTOPRIMARY);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_cdp monitor is %08X"), hmonDrag);
//
// re-origin at zero to make calculations simpler
//
screen.cx = RECTWIDTH(rcDisplay); screen.cy = RECTHEIGHT(rcDisplay); pt.x -= rcDisplay.left; pt.y -= rcDisplay.top;
//
// are we closer to the left or right side of this display?
//
if (pt.x < (screen.cx / 2)) { uVertEdge = STICK_LEFT; error.cx = pt.x; } else { uVertEdge = STICK_RIGHT; error.cx = screen.cx - pt.x; }
//
// are we closer to the top or bottom side of this display?
//
if (pt.y < (screen.cy / 2)) { uHorzEdge = STICK_TOP; error.cy = pt.y; } else { uHorzEdge = STICK_BOTTOM; error.cy = screen.cy - pt.y; }
//
// closer to a horizontal or vertical edge?
//
uPlace = ((error.cy * screen.cx) > (error.cx * screen.cy))? uVertEdge : uHorzEdge;
// which StuckRect should we use?
prcStick = &_arStuckRects[uPlace];
//
// need to recalc stuck rect for new monitor?
//
if ((hmonDrag != _GetDisplayRectFromRect(NULL, prcStick, MONITOR_DEFAULTTONULL))) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_cdp re-snapping rect for new display")); _MakeStuckRect(prcStick, &rcDisplay, _sStuckWidths, uPlace); } }
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_cdp edge is %d, rect is {%d, %d, %d, %d}"), uPlace, _arStuckRects[uPlace].left, _arStuckRects[uPlace].top, _arStuckRects[uPlace].right, _arStuckRects[uPlace].bottom); ASSERT(IsValidSTUCKPLACE(uPlace)); return uPlace; }
void CTray::_HandleMoving(WPARAM wParam, LPRECT lprc) { POINT ptCursor; GetCursorPos(&ptCursor);
// If the cursor is not far from its starting point, then ignore it.
// A very common problem is the user clicks near the corner of the clock,
// twitches the mouse 5 pixels, and BLAM, their taskbar is now vertical
// and they don't know what they did or how to get it back.
if (g_fInSizeMove && PtInRect(&_rcSizeMoveIgnore, ptCursor)) { // Ignore -- user is merely twitching
_uMoveStuckPlace = _uStuckPlace; } else { _uMoveStuckPlace = _CalcDragPlace(ptCursor); }
*lprc = _arStuckRects[_uMoveStuckPlace]; _HandleSizing(wParam, lprc, _uMoveStuckPlace); }
// store the tray size when dragging is finished
void CTray::_SnapshotStuckRectSize(UINT uPlace) { RECT rcDisplay, *prc = &_arStuckRects[uPlace];
//
// record the width of this stuck rect
//
if (STUCK_HORIZONTAL(uPlace)) _sStuckWidths.cy = RECTHEIGHT(*prc); else _sStuckWidths.cx = RECTWIDTH(*prc);
//
// we only present a horizontal or vertical size to the end user
// so update the StuckRect on the other side of the screen to match
//
_GetStuckDisplayRect(uPlace, &rcDisplay);
uPlace += 2; uPlace %= 4; prc = &_arStuckRects[uPlace];
_MakeStuckRect(prc, &rcDisplay, _sStuckWidths, uPlace); }
// Size the icon area to fill as much of the tray window as it can.
void CTray::SizeWindows() { RECT rcView, rcNotify, rcClient; int fHiding;
if (!_hwndRebar || !_hwnd || !_hwndNotify) return;
fHiding = (_uAutoHide & AH_HIDING); if (fHiding) { InvisibleUnhide(FALSE); }
// remember our current size
_SnapshotStuckRectSize(_uStuckPlace);
GetClientRect(_hwnd, &rcClient); _AlignStartButton();
_GetWindowSizes(_uStuckPlace, &rcClient, &rcView, &rcNotify);
InvalidateRect(_hwndStart, NULL, TRUE); InvalidateRect(_hwnd, NULL, TRUE);
// position the view
SetWindowPos(_hwndRebar, NULL, rcView.left, rcView.top, RECTWIDTH(rcView), RECTHEIGHT(rcView), SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOCOPYBITS); UpdateWindow(_hwndRebar);
// And the clock
SetWindowPos(_hwndNotify, NULL, rcNotify.left, rcNotify.top, RECTWIDTH(rcNotify), RECTHEIGHT(rcNotify), SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOCOPYBITS);
{ TOOLINFO ti; HWND hwndClock = _GetClockWindow();
ti.cbSize = sizeof(ti); ti.uFlags = 0; ti.hwnd = _hwnd; ti.lpszText = LPSTR_TEXTCALLBACK; ti.uId = (UINT_PTR)hwndClock; GetWindowRect(hwndClock, &ti.rect); MapWindowPoints(HWND_DESKTOP, _hwnd, (LPPOINT)&ti.rect, 2); SendMessage(_hwndTrayTips, TTM_NEWTOOLRECT, 0, (LPARAM)((LPTOOLINFO)&ti)); }
if (fHiding) { InvisibleUnhide(TRUE); } }
void CTray::_HandleSize() { //
// if somehow we got minimized go ahead and un-minimize
//
if (((GetWindowLong(_hwnd, GWL_STYLE)) & WS_MINIMIZE)) { ASSERT(FALSE); ShowWindow(_hwnd, SW_RESTORE); }
//
// if we are in the move/size loop and are visible then
// re-snap the current stuck rect to the new window size
//
#ifdef DEBUG
if (_fSysSizing && (_uAutoHide & AH_HIDING)) { TraceMsg(DM_TRACE, "fSysSize && hiding"); ASSERT(0); } #endif
if (_fSysSizing && ((_uAutoHide & (AH_ON | AH_HIDING)) != (AH_ON | AH_HIDING))) { _uStuckPlace = _RecalcStuckPos(NULL); _UpdateVertical(_uStuckPlace); }
//
// if we are in fulldrag or we are not in the middle of a move/size then
// we should resize all our child windows to reflect our new size
//
if (g_fDragFullWindows || !_fSysSizing) SizeWindows();
//
// if we are just plain resized and we are visible we may need re-dock
//
if (!_fSysSizing && !_fSelfSizing && IsWindowVisible(_hwnd)) { if (_uAutoHide & AH_ON) { UINT uPlace = _uStuckPlace; HWND hwndOther =_AppBarGetAutoHideBar(uPlace);
//
// we sometimes defer checking for this until after a move
// so as to avoid interrupting a full-window-drag in progress
// if there is a different autohide window in our slot then whimper
//
if (hwndOther? (hwndOther != _hwnd) : !_AppBarSetAutoHideBar2(_hwnd, TRUE, uPlace)) { _AutoHideCollision(); } }
_StuckTrayChange();
//
// make sure we clip to tray to the current monitor (if necessary)
//
_ClipWindow(TRUE); }
if (_hwndStartBalloon) { RECT rc; GetWindowRect(_hwndStart, &rc); SendMessage(_hwndStartBalloon, TTM_TRACKPOSITION, 0, MAKELONG((rc.left + rc.right)/2, rc.top)); SetWindowZorder(_hwndStartBalloon, HWND_TOPMOST); } }
BOOL _IsSliverHeight(int cy) { //
// Is this height clearly bigger than the pure-border height that you
// get when you resize the taskbar as small as it will go?
//
return (cy < (3 * (g_cyDlgFrame + g_cyBorder))); }
BOOL CTray::_HandleSizing(WPARAM code, LPRECT lprc, UINT uStuckPlace) { BOOL fChangedSize = FALSE; RECT rcDisplay; SIZE sNewWidths; RECT rcTemp; BOOL fHiding;
if (!lprc) { rcTemp = _arStuckRects[uStuckPlace]; lprc = &rcTemp; }
fHiding = (_uAutoHide & AH_HIDING); if (fHiding) { InvisibleUnhide(FALSE); }
//
// get the a bunch of relevant dimensions
//
// (dli) need to change this funciton or get rid of it
_GetDisplayRectFromRect(&rcDisplay, lprc, MONITOR_DEFAULTTONEAREST);
if (code) { // if code != 0, this is the user sizing.
// make sure they clip it to the screen.
RECT rcMax = rcDisplay; if (!_hTheme) { InflateRect(&rcMax, g_cxEdge, g_cyEdge); } // don't do intersect rect because of sizing up from the bottom
// (when taskbar docked on bottom) confuses people
switch (uStuckPlace) { case STICK_LEFT: lprc->left = rcMax.left; break;
case STICK_TOP: lprc->top = rcMax.top; break;
case STICK_RIGHT: lprc->right = rcMax.right; break;
case STICK_BOTTOM: lprc->top += (rcMax.bottom-lprc->bottom); lprc->bottom = rcMax.bottom; break; } }
//
// compute the new widths
// don't let either be more than half the screen
//
sNewWidths.cx = min(RECTWIDTH(*lprc), RECTWIDTH(rcDisplay) / 2); sNewWidths.cy = min(RECTHEIGHT(*lprc), RECTHEIGHT(rcDisplay) / 2);
if (_hTheme && (_fCanSizeMove || _fShowSizingBarAlways)) { sNewWidths.cy = max(_sizeSizingBar.cy, sNewWidths.cy); }
//
// compute an initial size
//
_MakeStuckRect(lprc, &rcDisplay, sNewWidths, uStuckPlace); DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hs starting rect is {%d, %d, %d, %d}"), lprc->left, lprc->top, lprc->right, lprc->bottom);
//
// negotiate the exact size with our children
//
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hs tray is being calculated for %s"), STUCK_HORIZONTAL(uStuckPlace) ? TEXT("HORIZONTAL") : TEXT("VERTICAL"));
_UpdateVertical(uStuckPlace); if (_ptbs && _fBandSiteInitialized) { IDeskBarClient* pdbc; if (SUCCEEDED(_ptbs->QueryInterface(IID_PPV_ARG(IDeskBarClient, &pdbc)))) { RECT rcClient = *lprc; RECT rcOldClient = _arStuckRects[uStuckPlace];
// Go from a Window Rect to Client Rect
if (_hTheme && (_fCanSizeMove || _fShowSizingBarAlways)) { _AdjustRectForSizingBar(uStuckPlace, &rcClient, -1); _AdjustRectForSizingBar(uStuckPlace, &rcOldClient, -1); } else if (!_hTheme) { InflateRect(&rcClient, -g_cxFrame, -g_cyFrame); InflateRect(&rcOldClient, -g_cxFrame, -g_cyFrame); } // Make rcClient start at 0,0, Rebar only used the right and bottom values of this rect
OffsetRect(&rcClient, -rcClient.left, -rcClient.top); OffsetRect(&rcOldClient, -rcOldClient.left, -rcOldClient.top); DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hs starting client rect is {%d, %d, %d, %d}"), rcClient.left, rcClient.top, rcClient.right, rcClient.bottom);
RECT rcNotify; RECT rcView; RECT rcOldView; // Go from the taskbar's client rect to the rebar's client rect
_GetWindowSizes(uStuckPlace, &rcClient, &rcView, &rcNotify); _GetWindowSizes(uStuckPlace, &rcOldClient, &rcOldView, &rcNotify); // Make rcView start at 0,0, Rebar only used the right and bottom values of this rect
OffsetRect(&rcView, -rcView.left, -rcView.top); OffsetRect(&rcOldView, -rcOldView.left, -rcOldView.top); if (!_fCanSizeMove || (RECTHEIGHT(rcView) && RECTWIDTH(rcView))) { // This following function will cause a WINDOWPOSCHAGING which will call DoneMoving
// DoneMoving will then go a screw up all of the window sizing
_fIgnoreDoneMoving = TRUE; pdbc->GetSize(DBC_GS_SIZEDOWN, &rcView); _fIgnoreDoneMoving = FALSE; }
// Go from a Client Rect to Window Rect
if (STUCK_HORIZONTAL(uStuckPlace)) { rcClient.top = rcView.top; rcClient.bottom = rcView.bottom; } else { rcClient.left = rcView.left; rcClient.right = rcView.right; }
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hs ending client rect is {%d, %d, %d, %d}"), rcClient.left, rcClient.top, rcClient.right, rcClient.bottom); if (_hTheme && (_fCanSizeMove || _fShowSizingBarAlways)) { _AdjustRectForSizingBar(uStuckPlace, &rcClient, 1); _AdjustRectForSizingBar(uStuckPlace, &rcOldClient, 1); } else if (!_hTheme) { InflateRect(&rcClient, g_cxFrame, g_cyFrame); InflateRect(&rcOldClient, g_cxFrame, g_cyFrame); }
// Prevent huge growth of taskbar, caused by bugs in the rebar sizing code
if (RECTHEIGHT(rcView) && RECTHEIGHT(rcOldView) && (RECTHEIGHT(rcClient) > (3 * RECTHEIGHT(rcOldClient)))) { rcClient = rcOldClient; }
if (STUCK_HORIZONTAL(uStuckPlace) && sNewWidths.cy != RECTHEIGHT(rcClient)) { sNewWidths.cy = RECTHEIGHT(rcClient); fChangedSize = TRUE; } if (!STUCK_HORIZONTAL(uStuckPlace) && sNewWidths.cx != RECTWIDTH(rcClient)) { sNewWidths.cx = RECTWIDTH(rcClient); fChangedSize = TRUE; }
pdbc->Release(); } }
//
// was there a change?
//
if (fChangedSize) { //
// yes, update the final rectangle
//
_MakeStuckRect(lprc, &rcDisplay, sNewWidths, uStuckPlace); }
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hs final rect is {%d, %d, %d, %d}"), lprc->left, lprc->top, lprc->right, lprc->bottom);
//
// store the new size in the appropriate StuckRect
//
_arStuckRects[uStuckPlace] = *lprc;
if (fHiding) { InvisibleUnhide(TRUE); }
if (_hwndStartBalloon) { RECT rc; GetWindowRect(_hwndStart, &rc); SendMessage(_hwndStartBalloon, TTM_TRACKPOSITION, 0, MAKELONG((rc.left + rc.right)/2, rc.top)); SetWindowZorder(_hwndStartBalloon, HWND_TOPMOST); }
return fChangedSize; }
/*-------------------------------------------------------------------
** the screen size changed, and we need to adjust some stuff, mostly ** globals. if the tray was docked, it needs to be resized, too. ** ** TRICKINESS: the handling of WM_WINDOWPOSCHANGING is used to ** actually do all the real sizing work. this saves a bit of ** extra code here. **-------------------------------------------------------------------*/
BOOL WINAPI CTray::MonitorEnumProc(HMONITOR hMonitor, HDC hdc, LPRECT lprc, LPARAM lData) { CTray* ptray = (CTray*)lData;
RECT rcWork; int iRet = ptray->_RecomputeWorkArea(NULL, hMonitor, &rcWork);
if (iRet == RWA_CHANGED) { // only send SENDWININICHANGE if the desktop has been created (otherwise
// we will hang the explorer because the main thread is currently blocked)
// PERF FEATURE: it will be nice to send WININICHANGE only once, but we can't
// because each time the rcWork is different, and there is no way to do it all
SystemParametersInfo(SPI_SETWORKAREA, TRUE, &rcWork, v_hwndDesktop ? SPIF_SENDWININICHANGE : 0); RedrawDesktop(&rcWork); }
return TRUE; }
void CTray::_RecomputeAllWorkareas() { EnumDisplayMonitors(NULL, NULL, MonitorEnumProc, (LPARAM)this); }
void CTray::_ScreenSizeChange(HWND hwnd) { // Set our new HMONITOR in case there is some change
{ MONITORINFO mi = {0}; mi.cbSize = sizeof(mi);
// Is our old HMONITOR still valid?
// NOTE: This test is used to tell whether somethings happened to the
// HMONITOR's or just the screen size changed
if (!GetMonitorInfo(_hmonStuck, &mi)) { // No, this means the HMONITORS changed, our monitor might have gone away
_SetStuckMonitor(); _fIsLogoff = FALSE; _RecomputeAllWorkareas(); } }
// screen size changed, so we need to adjust globals
g_cxPrimaryDisplay = GetSystemMetrics(SM_CXSCREEN); g_cyPrimaryDisplay = GetSystemMetrics(SM_CYSCREEN);
_ResizeStuckRects(_arStuckRects);
if (hwnd) { //
// set a bogus windowpos and actually repaint with the right
// shape/size in handling the WINDOWPOSCHANGING message
//
SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER | SWP_NOACTIVATE); }
SizeWindows(); RECT rc = _arStuckRects[_uStuckPlace]; _HandleSizing(0, &rc, _uStuckPlace); // In the multi-monitor case, we need to turn on clipping for the dynamic
// monitor changes i.e. when the user add monitors or remove them from the
// control panel.
_ClipWindow(TRUE); }
BOOL CTray::_UpdateAlwaysOnTop(BOOL fAlwaysOnTop) { BOOL fChanged = ((_fAlwaysOnTop == 0) != (fAlwaysOnTop == 0)); //
// The user clicked on the AlwaysOnTop menu item, we should now toggle
// the state and update the window accordingly...
//
_fAlwaysOnTop = fAlwaysOnTop; _ResetZorder();
// Make sure the screen limits are update to the new state.
_StuckTrayChange(); return fChanged; }
void CTray::_SaveTrayAndDesktop(void) { _SaveTray();
if (v_hwndDesktop) SendMessage(v_hwndDesktop, DTM_SAVESTATE, 0, 0);
if (_hwndNotify) SendMessage(_hwndNotify, TNM_SAVESTATE, 0, 0); }
void CTray::_SlideStep(HWND hwnd, const RECT *prcMonitor, const RECT *prcOld, const RECT *prcNew) { SIZE sizeOld = {prcOld->right - prcOld->left, prcOld->bottom - prcOld->top}; SIZE sizeNew = {prcNew->right - prcNew->left, prcNew->bottom - prcNew->top}; BOOL fClipFirst = FALSE; UINT flags;
DAD_ShowDragImage(FALSE); // Make sure this is off - client function must turn back on!!!
if (prcMonitor) { RECT rcClip, rcClipSafe, rcClipTest;
_CalcClipCoords(&rcClip, prcMonitor, prcNew);
rcClipTest = rcClip;
OffsetRect(&rcClipTest, prcOld->left, prcOld->top); IntersectRect(&rcClipSafe, &rcClipTest, prcMonitor);
fClipFirst = EqualRect(&rcClipTest, &rcClipSafe);
if (fClipFirst) _ClipInternal(&rcClip); }
flags = SWP_NOZORDER|SWP_NOACTIVATE; if ((sizeOld.cx == sizeNew.cx) && (sizeOld.cy == sizeNew.cy)) flags |= SWP_NOSIZE;
SetWindowPos(hwnd, NULL, prcNew->left, prcNew->top, sizeNew.cx, sizeNew.cy, flags);
if (prcMonitor && !fClipFirst) { RECT rcClip;
_CalcClipCoords(&rcClip, prcMonitor, prcNew); _ClipInternal(&rcClip); } }
void CTray::_SlideWindow(HWND hwnd, RECT *prc, BOOL fShow) { RECT rcLast; RECT rcMonitor; const RECT *prcMonitor; DWORD dt; BOOL fAnimate;
if (!IsWindowVisible(hwnd)) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sw window is hidden, just moving")); MoveWindow(_hwnd, prc->left, prc->top, RECTWIDTH(*prc), RECTHEIGHT(*prc), FALSE); return; }
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sw -----------------BEGIN-----------------"));
if (GetSystemMetrics(SM_CMONITORS) > 1) { _GetStuckDisplayRect(_uStuckPlace, &rcMonitor); prcMonitor = &rcMonitor; } else prcMonitor = NULL;
GetWindowRect(hwnd, &rcLast);
dt = fShow? _dtSlideShow : _dtSlideHide;
// See if we can use animation effects.
SystemParametersInfo(SPI_GETMENUANIMATION, 0, &fAnimate, 0);
if (g_fDragFullWindows && fAnimate && (dt > 0)) { RECT rcOld, rcNew, rcMove; int dx, dy, priority; DWORD t, t2, t0; HANDLE me;
rcOld = rcLast; rcNew = *prc;
dx = ((rcNew.left + rcNew.right) - (rcOld.left + rcOld.right)) / 2; dy = ((rcNew.top + rcNew.bottom) - (rcOld.top + rcOld.bottom)) / 2; ASSERT(dx == 0 || dy == 0);
me = GetCurrentThread(); priority = GetThreadPriority(me); SetThreadPriority(me, THREAD_PRIORITY_HIGHEST);
t2 = t0 = GetTickCount();
rcMove = rcOld; while ((t = GetTickCount()) - t0 < dt) { int dtdiff; if (t != t2) { rcMove.right -= rcMove.left; rcMove.left = rcOld.left + (dx) * (t - t0) / dt; rcMove.right += rcMove.left;
rcMove.bottom -= rcMove.top; rcMove.top = rcOld.top + (dy) * (t - t0) / dt; rcMove.bottom += rcMove.top;
_SlideStep(hwnd, prcMonitor, &rcLast, &rcMove);
if (fShow) UpdateWindow(hwnd);
rcLast = rcMove; t2 = t; }
// don't draw frames faster than user can see, e.g. 20ms
#define ONEFRAME 20
dtdiff = GetTickCount(); if ((dtdiff - t) < ONEFRAME) Sleep(ONEFRAME - (dtdiff - t));
// try to give desktop a chance to update
// only do it on hide because desktop doesn't need to paint on taskbar show
if (!fShow) { DWORD_PTR lres; SendMessageTimeout(v_hwndDesktop, DTM_UPDATENOW, 0, 0, SMTO_ABORTIFHUNG, 50, &lres); } }
SetThreadPriority(me, priority); }
_SlideStep(hwnd, prcMonitor, &rcLast, prc);
if (fShow) UpdateWindow(hwnd);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sw ------------------END------------------")); }
void CTray::_UnhideNow() { if (_uModalMode == MM_SHUTDOWN) return;
_fSelfSizing = TRUE; DAD_ShowDragImage(FALSE); // unlock the drag sink if we are dragging.
_SlideWindow(_hwnd, &_arStuckRects[_uStuckPlace], _dtSlideShow); DAD_ShowDragImage(TRUE); // restore the lock state.
_fSelfSizing = FALSE;
SendMessage(_hwndNotify, TNM_TRAYHIDE, 0, FALSE); }
void CTray::_ComputeHiddenRect(LPRECT prc, UINT uStuck) { int dwh; HMONITOR hMon; RECT rcMon; hMon = MonitorFromRect(prc, MONITOR_DEFAULTTONULL); if (!hMon) return; GetMonitorRect(hMon, &rcMon); if (STUCK_HORIZONTAL(uStuck)) dwh = prc->bottom - prc->top; else dwh = prc->right - prc->left;
switch (uStuck) { case STICK_LEFT: prc->right = rcMon.left + (g_cxFrame / 2); prc->left = prc->right - dwh; break;
case STICK_RIGHT: prc->left = rcMon.right - (g_cxFrame / 2); prc->right = prc->left + dwh; break;
case STICK_TOP: prc->bottom = rcMon.top + (g_cyFrame / 2); prc->top = prc->bottom - dwh; break;
case STICK_BOTTOM: prc->top = rcMon.bottom - (g_cyFrame / 2); prc->bottom = prc->top + dwh; break; } }
UINT CTray::_GetDockedRect(LPRECT prc, BOOL fMoving) { UINT uPos;
if (fMoving && (_uMoveStuckPlace != (UINT)-1)) uPos = _uMoveStuckPlace; else uPos = _uStuckPlace;
*prc = _arStuckRects[uPos];
if ((_uAutoHide & (AH_ON | AH_HIDING)) == (AH_ON | AH_HIDING)) { _ComputeHiddenRect(prc, uPos); }
return uPos; }
void CTray::_CalcClipCoords(RECT *prcClip, const RECT *prcMonitor, const RECT *prcNew) { RECT rcMonitor; RECT rcWindow;
if (!prcMonitor) { _GetStuckDisplayRect(_uStuckPlace, &rcMonitor); prcMonitor = &rcMonitor; }
if (!prcNew) { GetWindowRect(_hwnd, &rcWindow); prcNew = &rcWindow; }
IntersectRect(prcClip, prcMonitor, prcNew); OffsetRect(prcClip, -prcNew->left, -prcNew->top); }
void CTray::_ClipInternal(const RECT *prcClip) { HRGN hrgnClip;
// don't worry about clipping if there's only one monitor
if (GetSystemMetrics(SM_CMONITORS) <= 1) prcClip = NULL;
if (prcClip) { _fMonitorClipped = TRUE; hrgnClip = CreateRectRgnIndirect(prcClip); } else { // SetWindowRgn is expensive, skip ones that are NOPs
if (!_fMonitorClipped) return;
_fMonitorClipped = FALSE; hrgnClip = NULL; }
SetWindowRgn(_hwnd, hrgnClip, TRUE); }
void CTray::_ClipWindow(BOOL fClipState) { RECT rcClip; RECT *prcClip;
if (_fSelfSizing || _fSysSizing) { TraceMsg(TF_WARNING, "_ClipWindow: _fSelfSizing %x, _fSysSizing %x", _fSelfSizing, _fSysSizing); return; }
if ((GetSystemMetrics(SM_CMONITORS) <= 1) || _hTheme) fClipState = FALSE;
if (fClipState) { prcClip = &rcClip; _CalcClipCoords(prcClip, NULL, NULL); } else prcClip = NULL;
_ClipInternal(prcClip); }
void CTray::_Hide() { RECT rcNew;
// if we're in shutdown or if we're on boot up
// don't hide
if (_uModalMode == MM_SHUTDOWN) { TraceMsg(TF_TRAY, "e.th: suppress hide (shutdown || Notify)"); return; }
KillTimer(_hwnd, IDT_AUTOHIDE);
_fSelfSizing = TRUE;
//
// update the flags here to prevent race conditions
//
_uAutoHide = AH_ON | AH_HIDING; _GetDockedRect(&rcNew, FALSE);
DAD_ShowDragImage(FALSE); // unlock the drag sink if we are dragging.
_SlideWindow(_hwnd, &rcNew, _dtSlideHide); DAD_ShowDragImage(FALSE); // Another thread could have locked while we were gone
DAD_ShowDragImage(TRUE); // restore the lock state.
SendMessage(_hwndNotify, TNM_TRAYHIDE, 0, TRUE);
_fSelfSizing = FALSE; }
void CTray::_AutoHideCollision() { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_ahc COLLISION! (posting UI request)"));
PostMessage(_hwnd, TM_WARNNOAUTOHIDE, ((_uAutoHide & AH_ON) != 0), 0L); }
LONG CTray::_SetAutoHideState(BOOL fAutoHide) { //
// make sure we have something to do
//
if ((fAutoHide != 0) == ((_uAutoHide & AH_ON) != 0)) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.sahs nothing to do")); return MAKELONG(FALSE, TRUE); }
//
// make sure we can do it
//
if (!_AppBarSetAutoHideBar2(_hwnd, fAutoHide, _uStuckPlace)) { _AutoHideCollision(); return MAKELONG(FALSE, FALSE); }
//
// do it
//
if (fAutoHide) { _uAutoHide = AH_ON; _RefreshSettings(); _Hide(); #ifdef DEBUG
// _Hide updates the flags for us (sanity)
if (!(_uAutoHide & AH_ON)) { TraceMsg(DM_WARNING, "e.sahs: !AH_ON"); // ok to fail on boot/shutdown
} #endif
} else { _uAutoHide = 0; KillTimer(_hwnd, IDT_AUTOHIDE); _UnhideNow(); _RefreshSettings(); }
//
// brag about it
//
_StuckTrayChange(); return MAKELONG(TRUE, TRUE); }
void CTray::_HandleEnterMenuLoop() { // kill the timer when we're in the menu loop so that we don't
// pop done while browsing the menus.
if (_uAutoHide & AH_ON) { KillTimer(_hwnd, IDT_AUTOHIDE); } }
void CTray::_SetAutoHideTimer() { if (_uAutoHide & AH_ON) { SetTimer(_hwnd, IDT_AUTOHIDE, 500, NULL); } }
void CTray::_HandleExitMenuLoop() { // when we leave the menu stuff, start checking again.
_SetAutoHideTimer(); }
void CTray::Unhide() { // handle autohide
if ((_uAutoHide & AH_ON) && (_uAutoHide & AH_HIDING)) { _UnhideNow(); _uAutoHide &= ~AH_HIDING; _SetAutoHideTimer();
if (_fShouldResize) { ASSERT(0); ASSERT(!(_uAutoHide & AH_HIDING)); SizeWindows(); _fShouldResize = FALSE; } } }
void CTray::_SetUnhideTimer(LONG x, LONG y) { // handle autohide
if ((_uAutoHide & AH_ON) && (_uAutoHide & AH_HIDING)) { LONG dx = x-_ptLastHittest.x; LONG dy = y-_ptLastHittest.y; LONG rr = dx*dx + dy*dy; LONG dd = GetSystemMetrics(SM_CXDOUBLECLK) * GetSystemMetrics(SM_CYDOUBLECLK);
if (rr > dd) { SetTimer(_hwnd, IDT_AUTOUNHIDE, 50, NULL); _ptLastHittest.x = x; _ptLastHittest.y = y; } } }
void CTray::_StartButtonReset() { // Get an idea about how big we need everyhting to be.
TCHAR szStart[50]; LoadString(hinstCabinet, _hTheme ? IDS_START : IDS_STARTCLASSIC, szStart, ARRAYSIZE(szStart)); SetWindowText(_hwndStart, szStart);
if (_hFontStart) DeleteObject(_hFontStart);
_hFontStart = _CreateStartFont(_hwndStart);
int idbStart = IDB_START16; HDC hdc = GetDC(NULL); if (hdc) { int bpp = GetDeviceCaps(hdc, BITSPIXEL) * GetDeviceCaps(hdc, PLANES); if (bpp > 8) { idbStart = _hTheme ? IDB_START : IDB_STARTCLASSIC; }
ReleaseDC(NULL, hdc); }
HBITMAP hbmFlag = (HBITMAP)LoadImage(hinstCabinet, MAKEINTRESOURCE(idbStart), IMAGE_BITMAP, 0, 0, LR_CREATEDIBSECTION); if (hbmFlag) { BITMAP bm; if (GetObject(hbmFlag, sizeof(BITMAP), &bm)) { BUTTON_IMAGELIST biml = {0}; if (_himlStartFlag) ImageList_Destroy(_himlStartFlag);
DWORD dwFlags = ILC_COLOR32; HBITMAP hbmFlagMask = NULL; if (idbStart == IDB_START16) { dwFlags = ILC_COLOR8 | ILC_MASK; hbmFlagMask = (HBITMAP)LoadImage(hinstCabinet, MAKEINTRESOURCE(IDB_START16MASK), IMAGE_BITMAP, 0, 0, LR_MONOCHROME); }
if (IS_WINDOW_RTL_MIRRORED(_hwndStart)) { dwFlags |= ILC_MIRROR; } biml.himl = _himlStartFlag = ImageList_Create(bm.bmWidth, bm.bmHeight, dwFlags, 1, 1); ImageList_Add(_himlStartFlag, hbmFlag, hbmFlagMask); biml.uAlign = BUTTON_IMAGELIST_ALIGN_LEFT;
Button_SetImageList(_hwndStart, &biml); } DeleteObject(hbmFlag); }
if (_hFontStart) { SendMessage(_hwndStart, WM_SETFONT, (WPARAM)_hFontStart, TRUE); _sizeStart.cx = 0; }
_AlignStartButton();
}
void CTray::_OnNewSystemSizes() { TraceMsg(TF_TRAY, "Handling win ini change."); _StartButtonReset(); VerifySize(TRUE); }
//*** CheckWindowPositions -- flag which windows actually changed
// ENTRY/EXIT
// _pPositions->hdsaWP[*]->fRestore modified
// NOTES
// in order to correctly implement 'Undo Minimize-all(Cascade/Tile)',
// we need to tell which windows were changed by the 'Do' operation.
// (nt5:183421: we used to restore *every* top-level window).
int WINAPI CTray::CheckWndPosEnumProc(void *pItem, void *pData) { HWNDANDPLACEMENT *pI2 = (HWNDANDPLACEMENT *)pItem; WINDOWPLACEMENT wp;
wp.length = sizeof(wp); pI2->fRestore = TRUE; if (GetWindowPlacement(pI2->hwnd, &wp)) { if (memcmp(&pI2->wp, &wp, sizeof(wp)) == 0) pI2->fRestore = FALSE; }
TraceMsg(TF_TRAY, "cwp: (hwnd=0x%x) fRestore=%d", pI2->hwnd, pI2->fRestore);
return 1; // 1:continue enum
}
void CTray::CheckWindowPositions() { ENTERCRITICAL; // i think this is needed...
if (_pPositions) { if (_pPositions->hdsaWP) { DSA_EnumCallback(_pPositions->hdsaWP, CheckWndPosEnumProc, NULL); } } LEAVECRITICAL;
return; }
BOOL BandSite_PermitAutoHide(IUnknown* punk) { OLECMD cmd = { DBID_PERMITAUTOHIDE, 0 }; if (SUCCEEDED(IUnknown_QueryStatus(punk, &IID_IDockingWindow, 1, &cmd, NULL))) { return !(cmd.cmdf & OLECMDF_SUPPORTED) || (cmd.cmdf & OLECMDF_ENABLED); } return TRUE; }
void CTray::_HandleTimer(WPARAM wTimerID) { switch (wTimerID) { case IDT_CHECKDISKSPACE: CheckDiskSpace(); break;
case IDT_DESKTOPCLEANUP: _CheckDesktopCleanup(); break;
case IDT_CHANGENOTIFY: // did somebody send another notify since we last handled one?
if (_fUseChangeNotifyTimer) { // yep.
// all we do is check the staging area.
CheckStagingArea();
// kill it off the next time.
_fUseChangeNotifyTimer = FALSE; } else { // nope.
KillTimer(_hwnd, IDT_CHANGENOTIFY); _fChangeNotifyTimerRunning = FALSE; } break;
case IDT_HANDLEDELAYBOOTSTUFF: KillTimer(_hwnd, IDT_HANDLEDELAYBOOTSTUFF); PostMessage(_hwnd, TM_HANDLEDELAYBOOTSTUFF, 0, 0); break;
case IDT_STARTMENU: SetForegroundWindow(_hwnd); KillTimer(_hwnd, wTimerID); DAD_ShowDragImage(FALSE); // unlock the drag sink if we are dragging.
SendMessage(_hwndStart, BM_SETSTATE, TRUE, 0); UpdateWindow(_hwndStart); DAD_ShowDragImage(TRUE); // restore the lock state.
break;
case IDT_SAVESETTINGS: KillTimer(_hwnd, IDT_SAVESETTINGS); _SaveTray(); break; case IDT_ENABLEUNDO: KillTimer(_hwnd, IDT_ENABLEUNDO); CheckWindowPositions(); _fUndoEnabled = TRUE; break;
case IDT_AUTOHIDE: if (!_fSysSizing && (_uAutoHide & AH_ON)) { POINT pt; RECT rc;
// Don't hide if we're already hiding, a balloon tip is showing, or
// (on NT5) if any apps are flashing.
//
if (!(_uAutoHide & AH_HIDING) && BandSite_PermitAutoHide(_ptbs) && !_fBalloonUp) { // Get the cursor position.
GetCursorPos(&pt);
// Get the tray rect and inflate it a bit.
rc = _arStuckRects[_uStuckPlace]; InflateRect(&rc, g_cxEdge * 4, g_cyEdge*4);
// Don't hide if cursor is within inflated tray rect.
if (!PtInRect(&rc, pt)) { // Don't hide if the tray is active
if (!_IsActive() && _uStartButtonState != SBSM_SPACTIVE) { // Don't hide if the view has a system menu up.
if (!SendMessage(_hwndTasks, TBC_SYSMENUCOUNT, 0, 0L)) { // Phew! We made it. Hide the tray.
_Hide(); } } } } } break;
case IDT_AUTOUNHIDE: if (!_fSysSizing && (_uAutoHide & AH_ON)) { POINT pt; RECT rc;
KillTimer(_hwnd, wTimerID); _ptLastHittest.x = -0x0fff; _ptLastHittest.y = -0x0fff; GetWindowRect(_hwnd, &rc); if (_uAutoHide & AH_HIDING) { GetCursorPos(&pt); if (PtInRect(&rc, pt)) Unhide(); } } break;
case IDT_STARTBUTTONBALLOON: _DestroyStartButtonBalloon(); break;
case IDT_COFREEUNUSED: CoFreeUnusedLibraries(); KillTimer(_hwnd, IDT_COFREEUNUSED); break; } }
void CTray::_CheckStagingAreaOnTimer() { if (_fChangeNotifyTimerRunning) { // we're already running the timer, so force a check the next time it comes up
_fUseChangeNotifyTimer = TRUE; } else { _fChangeNotifyTimerRunning = TRUE;
// check once immediately
CheckStagingArea();
// check again in half a minute, but only if notifies have been happening in the meantime.
SetTimer(_hwnd, IDT_CHANGENOTIFY, 30 * 1000, NULL); } }
void CTray::_HandleChangeNotify(WPARAM wParam, LPARAM lParam) { LPITEMIDLIST *ppidl; LONG lEvent; LPSHChangeNotificationLock pshcnl = SHChangeNotification_Lock((HANDLE)wParam, (DWORD)lParam, &ppidl, &lEvent); if (pshcnl) { if (lEvent & SHCNE_STAGINGAREANOTIFICATIONS) { // something has changed within the staging area.
_CheckStagingAreaOnTimer(); } SHChangeNotification_Unlock(pshcnl); } }
BOOL _ExecItemByPidls(HWND hwnd, LPITEMIDLIST pidlFolder, LPITEMIDLIST pidlItem) { BOOL fRes = FALSE;
if (pidlFolder && pidlItem) { IShellFolder *psf = BindToFolder(pidlFolder); if (psf) { fRes = SUCCEEDED(SHInvokeDefaultCommand(hwnd, psf, pidlItem)); } else { TCHAR szPath[MAX_PATH]; SHGetPathFromIDList(pidlFolder, szPath); ShellMessageBox(hinstCabinet, hwnd, MAKEINTRESOURCE(IDS_CANTFINDSPECIALDIR), NULL, MB_ICONEXCLAMATION, szPath); } } return fRes; }
void _DestroySavedWindowPositions(LPWINDOWPOSITIONS pPositions);
LRESULT CTray::_HandleDestroy() { MINIMIZEDMETRICS mm;
TraceMsg(DM_SHUTDOWN, "_HD: enter");
mm.cbSize = sizeof(mm); SystemParametersInfo(SPI_GETMINIMIZEDMETRICS, sizeof(mm), &mm, FALSE); mm.iArrange &= ~ARW_HIDE; SystemParametersInfo(SPI_SETMINIMIZEDMETRICS, sizeof(mm), &mm, FALSE);
_RevokeDropTargets(); _DestroyStartMenu(); Mixer_Shutdown();
// Tell the start menu to free all its cached darwin links
SHRegisterDarwinLink(NULL, NULL, TRUE);
_DestroySavedWindowPositions(_pPositions); _pPositions = NULL;
if (_hTheme) { CloseThemeData(_hTheme); _hTheme = NULL; }
_UnregisterGlobalHotkeys();
if (_uNotify) { SHChangeNotifyDeregister(_uNotify); _uNotify = 0; }
ATOMICRELEASE(_ptbs); ATOMICRELEASE(_pdbTasks); _hwndTasks = NULL;
if (_hwndTrayTips) { DestroyWindow(_hwndTrayTips); _hwndTrayTips = NULL; }
_DestroyStartButtonBalloon();
// REVIEW
PostQuitMessage(0);
if (_hbmpStartBkg) { DeleteBitmap(_hbmpStartBkg); }
if (_hFontStart) { DeleteObject(_hFontStart); }
if (_himlStartFlag) { ImageList_Destroy(_himlStartFlag); }
// clean up service objects
_ssomgr.Destroy();
if (_hShellReadyEvent) { ResetEvent(_hShellReadyEvent); CloseHandle(_hShellReadyEvent); _hShellReadyEvent = NULL; }
if (_fHandledDelayBootStuff) { TBOOL(WinStationUnRegisterConsoleNotification(SERVERNAME_CURRENT, v_hwndTray)); }
DeleteCriticalSection(&_csHotkey);
// The order in which we shut down the HTTP key monitoring is important.
//
// We must close the key before closing the event handle because
// closing the key causes the event to be signalled and we don't
// want ADVAPI32 to try to signal an event after we closed its handle...
//
// To avoid a spurious trigger when the event fires, we unregister
// the wait before closing the key.
//
if (_hHTTPWait) { UnregisterWait(_hHTTPWait); _hHTTPWait = NULL; }
if (_hkHTTP) { RegCloseKey(_hkHTTP); _hkHTTP = NULL; }
if (_hHTTPEvent) { CloseHandle(_hHTTPEvent); _hHTTPEvent = NULL; }
// End of order-sensitive operations ----------------------------------
v_hwndTray = NULL; _hwndStart = NULL;
TraceMsg(DM_SHUTDOWN, "_HD: leave"); return 0; }
void CTray::_SetFocus(HWND hwnd) { IUnknown_UIActivateIO(_ptbs, FALSE, NULL); SetFocus(hwnd); }
#define TRIEDTOOMANYTIMES 100
void CTray::_ActAsSwitcher() { if (_uModalMode) { if (_uModalMode != MM_SHUTDOWN) { SwitchToThisWindow(GetLastActivePopup(_hwnd), TRUE); } MessageBeep(0); } else { HWND hwndForeground; HWND hwndActive;
static int s_iRecurse = 0; s_iRecurse++;
ASSERT(s_iRecurse < TRIEDTOOMANYTIMES); TraceMsg(TF_TRAY, "s_iRecurse = %d", s_iRecurse);
hwndForeground = GetForegroundWindow(); hwndActive = GetActiveWindow(); BOOL fIsTrayActive = (hwndForeground == _hwnd) && (hwndActive == _hwnd); if (v_hwndStartPane && hwndForeground == v_hwndStartPane && hwndActive == v_hwndStartPane) { fIsTrayActive = TRUE; } // only do the button once we're the foreground dude.
if (fIsTrayActive) { // This code path causes the start button to do something because
// of the keyboard. So reflect that with the focus rect.
SendMessage(_hwndStart, WM_UPDATEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);
if (SendMessage(_hwndStart, BM_GETSTATE, 0, 0) & BST_PUSHED) { ClosePopupMenus(); ForceStartButtonUp(); } else { // This pushes the start button and causes the start menu to popup.
SendMessage(GetDlgItem(_hwnd, IDC_START), BM_SETSTATE, TRUE, 0); } s_iRecurse = 0; } else { // we don't want to loop endlessly trying to become
// foreground. With NT's new SetForegroundWindow rules, it would
// be pointless to try and hopefully we won't need to anyhow.
// Randomly, I picked a quarter as many times as the debug squirty would indicate
// as the number of times to try on NT.
// Hopefully that is enough on most machines.
if (s_iRecurse > (TRIEDTOOMANYTIMES / 4)) { s_iRecurse = 0; return; } // until then, try to come forward.
HandleFullScreenApp(NULL); if (hwndForeground == v_hwndDesktop) { _SetFocus(_hwndStart); if (GetFocus() != _hwndStart) return; }
SwitchToThisWindow(_hwnd, TRUE); SetForegroundWindow(_hwnd); Sleep(20); // give some time for other async activation messages to get posted
PostMessage(_hwnd, TM_ACTASTASKSW, 0, 0); } } }
void CTray::_OnWinIniChange(HWND hwnd, WPARAM wParam, LPARAM lParam) { Cabinet_InitGlobalMetrics(wParam, (LPTSTR)lParam);
// Reset the programs menu.
// REVIEW IANEL - We should only need to listen to the SPI_SETNONCLIENT stuff
// but deskcpl doesn't send one.
if (wParam == SPI_SETNONCLIENTMETRICS || (!wParam && (!lParam || (lstrcmpi((LPTSTR)lParam, TEXT("WindowMetrics")) == 0)))) { #ifdef DEBUG
if (wParam == SPI_SETNONCLIENTMETRICS) TraceMsg(TF_TRAY, "c.t_owic: Non-client metrics (probably) changed."); else TraceMsg(TF_TRAY, "c.t_owic: Window metrics changed."); #endif
_OnNewSystemSizes(); }
// Handle old extensions.
if (!lParam || (lParam && (lstrcmpi((LPTSTR)lParam, TEXT("Extensions")) == 0))) { TraceMsg(TF_TRAY, "t_owic: Extensions section change."); CheckWinIniForAssocs(); }
if (lParam && (0 == lstrcmpi((LPCTSTR)lParam, TEXT("TraySettings")))) { _Command(FCIDM_REFRESH); }
// Tell shell32 to refresh its cache
SHSettingsChanged(wParam, lParam); }
HWND CTray::_HotkeyInUse(WORD wHK) { HWND hwnd; DWORD_PTR lrHKInUse = 0; int nMod; WORD wHKNew; #ifdef DEBUG
TCHAR sz[MAX_PATH]; #endif
// Map the modifiers back.
nMod = 0; if (HIBYTE(wHK) & MOD_SHIFT) nMod |= HOTKEYF_SHIFT; if (HIBYTE(wHK) & MOD_CONTROL) nMod |= HOTKEYF_CONTROL; if (HIBYTE(wHK) & MOD_ALT) nMod |= HOTKEYF_ALT;
wHKNew = (WORD)((nMod*256)+LOBYTE(wHK));
DebugMsg(DM_IANELHK, TEXT("c.hkl_hiu: Checking for %x"), wHKNew); hwnd = GetWindow(GetDesktopWindow(), GW_CHILD); while (hwnd) { SendMessageTimeout(hwnd, WM_GETHOTKEY, 0, 0, SMTO_ABORTIFHUNG| SMTO_BLOCK, 3000, &lrHKInUse); if (wHKNew == (WORD)lrHKInUse) { #ifdef DEBUG
GetWindowText(hwnd, sz, ARRAYSIZE(sz)); DebugMsg(DM_IANELHK, TEXT("c.hkl_hiu: %s (%x) is using %x"), sz, hwnd, lrHKInUse); #endif
return hwnd; } #ifdef DEBUG
else if (lrHKInUse) { GetWindowText(hwnd, sz, ARRAYSIZE(sz)); DebugMsg(DM_IANELHK, TEXT("c.hkl_hiu: %s (%x) is using %x"), sz, hwnd, lrHKInUse); } #endif
hwnd = GetWindow(hwnd, GW_HWNDNEXT); } return NULL; }
void CTray::_HandleHotKey(int nID) { TraceMsg(TF_TRAY, "c.hkl_hh: Handling hotkey (%d).", nID);
// Find it in the list.
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
EnterCriticalSection(&_csHotkey);
HOTKEYITEM *phki = (HOTKEYITEM *)DSA_GetItemPtr(_hdsaHKI, nID); if (phki && phki->wGHotkey) { TraceMsg(TF_TRAY, "c.hkl_hh: Hotkey listed.");
// Are global hotkeys enabled?
if (!_fGlobalHotkeyDisable) { // Yep.
HWND hwnd = _HotkeyInUse(phki->wGHotkey); // Make sure this hotkey isn't already in use by someone.
if (hwnd) { TraceMsg(TF_TRAY, "c.hkl_hh: Hotkey is already in use."); // Activate it.
SwitchToThisWindow(GetLastActivePopup(hwnd), TRUE); } else { DECLAREWAITCURSOR; // Exec the item.
SetWaitCursor(); TraceMsg(TF_TRAY, "c.hkl_hh: Hotkey is not in use, execing item."); ASSERT(phki->pidlFolder && phki->pidlItem); BOOL fRes = _ExecItemByPidls(_hwnd, phki->pidlFolder, phki->pidlItem); ResetWaitCursor(); #ifdef DEBUG
if (!fRes) { DebugMsg(DM_ERROR, TEXT("c.hkl_hh: Can't exec command .")); } #endif
} } else { DebugMsg(DM_ERROR, TEXT("c.hkl_hh: Global hotkeys have been disabled.")); } } else { DebugMsg(DM_ERROR, TEXT("c.hkl_hh: Hotkey not listed.")); } LeaveCriticalSection(&_csHotkey); }
LRESULT CTray::_UnregisterHotkey(HWND hwnd, int i) { TraceMsg(TF_TRAY, "c.t_uh: Unregistering hotkey (%d).", i);
if (!UnregisterHotKey(hwnd, i)) { DebugMsg(DM_ERROR, TEXT("c.t_rh: Unable to unregister hotkey %d."), i); } return TRUE; }
// Add hotkey to the shell's list of global hotkeys.
LRESULT CTray::_ShortcutRegisterHotkey(HWND hwnd, WORD wHotkey, ATOM atom) { int i; LPITEMIDLIST pidl; TCHAR szPath[MAX_PATH]; ASSERT(atom);
if (GlobalGetAtomName(atom, szPath, MAX_PATH)) { TraceMsg(TF_TRAY, "c.t_srh: Hotkey %d for %s", wHotkey, szPath);
pidl = ILCreateFromPath(szPath); if (pidl) { i = _HotkeyAddCached(_MapHotkeyToGlobalHotkey(wHotkey), pidl); if (i != -1) { _RegisterHotkey(_hwnd, i); } } return TRUE; } else { return FALSE; } }
// Remove hotkey from shell's list.
LRESULT CTray::_ShortcutUnregisterHotkey(HWND hwnd, WORD wHotkey) { // DebugMsg(DM_TRACE, "c.t_suh: Hotkey %d", wHotkey);
int i = _HotkeyRemove(wHotkey); if (i == -1) i = _HotkeyRemoveCached(_MapHotkeyToGlobalHotkey(wHotkey));
if (i != -1) _UnregisterHotkey(hwnd, i);
return TRUE; }
LRESULT CTray::_RegisterHotkey(HWND hwnd, int i) { HOTKEYITEM *phki; WORD wGHotkey = 0;
ASSERT(IS_VALID_HANDLE(_hdsaHKI, DSA));
TraceMsg(TF_TRAY, "c.t_rh: Registering hotkey (%d).", i);
EnterCriticalSection(&_csHotkey);
phki = (HOTKEYITEM *)DSA_GetItemPtr(_hdsaHKI, i); ASSERT(phki); if (phki) { wGHotkey = phki->wGHotkey; }
LeaveCriticalSection(&_csHotkey); if (wGHotkey) { // Is the hotkey available?
if (RegisterHotKey(hwnd, i, HIBYTE(wGHotkey), LOBYTE(wGHotkey))) { // Yes.
return TRUE; } else { // Delete any cached items that might be using this
// hotkey.
int iCached = _HotkeyRemoveCached(wGHotkey); ASSERT(iCached != i); if (iCached != -1) { // Free up the hotkey for us.
_UnregisterHotkey(hwnd, iCached); // Yep, nuked the cached item. Try again.
if (RegisterHotKey(hwnd, i, HIBYTE(wGHotkey), LOBYTE(wGHotkey))) { return TRUE; } } }
// Can't set hotkey for this item.
DebugMsg(DM_ERROR, TEXT("c.t_rh: Unable to register hotkey %d."), i); // Null out this item.
phki->wGHotkey = 0; phki->pidlFolder = NULL; phki->pidlItem = NULL; } else { DebugMsg(DM_ERROR, TEXT("c.t_rh: Hotkey item is invalid.")); } return FALSE; }
#define GetABDHWnd(pabd) ((HWND)ULongToPtr((pabd)->dwWnd))
void CTray::_AppBarGetTaskBarPos(PTRAYAPPBARDATA ptabd) { APPBARDATA3264 *pabd;
pabd = (APPBARDATA3264*)SHLockShared(UlongToPtr(ptabd->hSharedABD), ptabd->dwProcId); if (pabd) { pabd->rc = _arStuckRects[_uStuckPlace]; pabd->uEdge = _uStuckPlace; // compat: new to ie4
SHUnlockShared(pabd); } }
void CTray::_NukeAppBar(int i) { LocalFree(DPA_GetPtr(_hdpaAppBars, i)); DPA_DeletePtr(_hdpaAppBars, i); }
void CTray::_AppBarRemove(PTRAYAPPBARDATA ptabd) { int i;
if (!_hdpaAppBars) return;
i = DPA_GetPtrCount(_hdpaAppBars);
while (i--) { PAPPBAR pab = (PAPPBAR)DPA_GetPtr(_hdpaAppBars, i);
if (GetABDHWnd(&ptabd->abd) == pab->hwnd) { RECT rcNuke = pab->rc;
_NukeAppBar(i); _StuckAppChange(GetABDHWnd(&ptabd->abd), &rcNuke, NULL, FALSE); } } }
PAPPBAR CTray::_FindAppBar(HWND hwnd) { if (_hdpaAppBars) { int i = DPA_GetPtrCount(_hdpaAppBars);
while (i--) { PAPPBAR pab = (PAPPBAR)DPA_GetPtr(_hdpaAppBars, i);
if (hwnd == pab->hwnd) return pab; } }
return NULL; }
void CTray::_AppBarNotifyAll(HMONITOR hmon, UINT uMsg, HWND hwndExclude, LPARAM lParam) { if (!_hdpaAppBars) return;
int i = DPA_GetPtrCount(_hdpaAppBars);
while (i--) { PAPPBAR pab = (PAPPBAR)DPA_GetPtr(_hdpaAppBars, i);
// We need to check pab here as an appbar can delete other
// appbars on the callback.
if (pab && (hwndExclude != pab->hwnd)) { if (!IsWindow(pab->hwnd)) { _NukeAppBar(i); continue; }
//
// if a monitor was specified only tell appbars on that display
//
if (hmon && (hmon != MonitorFromWindow(pab->hwnd, MONITOR_DEFAULTTONULL))) { continue; }
PostMessage(pab->hwnd, pab->uCallbackMessage, uMsg, lParam); } } }
BOOL CTray::_AppBarNew(PTRAYAPPBARDATA ptabd) { PAPPBAR pab; if (!_hdpaAppBars) { _hdpaAppBars = DPA_Create(4); if (!_hdpaAppBars) return FALSE;
} else if (_FindAppBar(GetABDHWnd(&ptabd->abd))) { // already have this hwnd
return FALSE; }
pab = (PAPPBAR)LocalAlloc(LPTR, sizeof(APPBAR)); if (!pab) return FALSE;
pab->hwnd = GetABDHWnd(&ptabd->abd); pab->uCallbackMessage = ptabd->abd.uCallbackMessage; pab->uEdge = (UINT)-1;
if (DPA_AppendPtr(_hdpaAppBars, pab) == -1) { // insertion failed
LocalFree(pab); return FALSE; }
return TRUE; }
BOOL CTray::_AppBarOutsideOf(PAPPBAR pabReq, PAPPBAR pab) { if (pabReq->uEdge == pab->uEdge) { switch (pab->uEdge) { case ABE_RIGHT: return (pab->rc.right >= pabReq->rc.right);
case ABE_BOTTOM: return (pab->rc.bottom >= pabReq->rc.bottom);
case ABE_TOP: return (pab->rc.top <= pabReq->rc.top);
case ABE_LEFT: return (pab->rc.left <= pabReq->rc.left); } } return FALSE; }
void CTray::_AppBarQueryPos(PTRAYAPPBARDATA ptabd) { int i; PAPPBAR pabReq = _FindAppBar(GetABDHWnd(&ptabd->abd));
if (pabReq) { APPBARDATA3264 *pabd;
pabd = (APPBARDATA3264*)SHLockShared(UlongToPtr(ptabd->hSharedABD), ptabd->dwProcId); if (pabd) { HMONITOR hmon;
pabd->rc = ptabd->abd.rc;
//
// default to the primary display for this call because old appbars
// sometimes pass a huge rect and let us pare it down. if they do
// something like that they don't support multiple displays anyway
// so just put them on the primary display...
//
hmon = MonitorFromRect(&pabd->rc, MONITOR_DEFAULTTOPRIMARY);
//
// always subtract off the tray if it's on the same display
//
if (!_uAutoHide && (hmon == _hmonStuck)) { APPBAR ab;
ab.uEdge = _GetDockedRect(&ab.rc, FALSE); _AppBarSubtractRect(&ab, &pabd->rc); }
i = DPA_GetPtrCount(_hdpaAppBars);
while (i--) { PAPPBAR pab = (PAPPBAR)DPA_GetPtr(_hdpaAppBars, i);
//
// give top and bottom preference
// ||
// if we're not changing edges,
// subtract anything currently on the outside of us
// ||
// if we are changing sides,
// subtract off everything on the new side.
//
// of course ignore appbars which are not on the same display...
//
if ((((pabReq->hwnd != pab->hwnd) && STUCK_HORIZONTAL(pab->uEdge) && !STUCK_HORIZONTAL(ptabd->abd.uEdge)) || ((pabReq->hwnd != pab->hwnd) && (pabReq->uEdge == ptabd->abd.uEdge) && _AppBarOutsideOf(pabReq, pab)) || ((pabReq->hwnd != pab->hwnd) && (pabReq->uEdge != ptabd->abd.uEdge) && (pab->uEdge == ptabd->abd.uEdge))) && (hmon == MonitorFromRect(&pab->rc, MONITOR_DEFAULTTONULL))) { _AppBarSubtractRect(pab, &pabd->rc); } } SHUnlockShared(pabd); } } }
void CTray::_AppBarSetPos(PTRAYAPPBARDATA ptabd) { PAPPBAR pab = _FindAppBar(GetABDHWnd(&ptabd->abd));
if (pab) { RECT rcOld; APPBARDATA3264 *pabd; BOOL fChanged = FALSE;
_AppBarQueryPos(ptabd);
pabd = (APPBARDATA3264*)SHLockShared(UlongToPtr(ptabd->hSharedABD), ptabd->dwProcId); if (pabd) { if (!EqualRect(&pab->rc, &pabd->rc)) { rcOld = pab->rc; pab->rc = pabd->rc; pab->uEdge = ptabd->abd.uEdge; fChanged = TRUE; } SHUnlockShared(pabd); }
if (fChanged) _StuckAppChange(GetABDHWnd(&ptabd->abd), &rcOld, &pab->rc, FALSE); } }
//
// FEATURE: need to get rid of this array-based implementation to allow autohide
// appbars on secondary display (or a/h tray on 2nd with a/h appbar on primary)
// change it to an _AppBarFindAutoHideBar that keeps flags on the appbardata...
//
HWND CTray::_AppBarGetAutoHideBar(UINT uEdge) { if (uEdge >= ABE_MAX) return FALSE; else { HWND hwndAutoHide = _aHwndAutoHide[uEdge]; if (!IsWindow(hwndAutoHide)) { _aHwndAutoHide[uEdge] = NULL; } return _aHwndAutoHide[uEdge]; } }
BOOL CTray::_AppBarSetAutoHideBar2(HWND hwnd, BOOL fAutoHide, UINT uEdge) { HWND hwndAutoHide = _aHwndAutoHide[uEdge]; if (!IsWindow(hwndAutoHide)) { _aHwndAutoHide[uEdge] = NULL; }
if (fAutoHide) { // register
if (!_aHwndAutoHide[uEdge]) { _aHwndAutoHide[uEdge] = hwnd; }
return _aHwndAutoHide[uEdge] == hwnd; } else { // unregister
if (_aHwndAutoHide[uEdge] == hwnd) { _aHwndAutoHide[uEdge] = NULL; }
return TRUE; } }
BOOL CTray::_AppBarSetAutoHideBar(PTRAYAPPBARDATA ptabd) { UINT uEdge = ptabd->abd.uEdge; if (uEdge >= ABE_MAX) return FALSE; else { return _AppBarSetAutoHideBar2(GetABDHWnd(&ptabd->abd), BOOLFROMPTR(ptabd->abd.lParam), uEdge); } }
void CTray::_AppBarActivationChange2(HWND hwnd, UINT uEdge) { //
// FEATURE: make this multi-monitor cool
//
HWND hwndAutoHide = _AppBarGetAutoHideBar(uEdge);
if (hwndAutoHide && (hwndAutoHide != hwnd)) { //
// the _AppBar got this notification inside a SendMessage from USER
// and is now in a SendMessage to us. don't try to do a SetWindowPos
// right now...
//
PostMessage(_hwnd, TM_BRINGTOTOP, (WPARAM)hwndAutoHide, uEdge); } }
void CTray::_AppBarActivationChange(PTRAYAPPBARDATA ptabd) { PAPPBAR pab = _FindAppBar(GetABDHWnd(&ptabd->abd)); if (pab) { // if this is an autohide bar and they're claiming to be on an edge not the same as their autohide edge,
// we don't do any activation of other autohides
for (UINT i = 0; i < ABE_MAX; i++) { if (_aHwndAutoHide[i] == GetABDHWnd(&ptabd->abd) && i != pab->uEdge) return; } _AppBarActivationChange2(GetABDHWnd(&ptabd->abd), pab->uEdge); } }
LRESULT CTray::_OnAppBarMessage(PCOPYDATASTRUCT pcds) { PTRAYAPPBARDATA ptabd = (PTRAYAPPBARDATA)pcds->lpData;
ASSERT(pcds->cbData == sizeof(TRAYAPPBARDATA)); ASSERT(ptabd->abd.cbSize == sizeof(APPBARDATA3264));
switch (ptabd->dwMessage) { case ABM_NEW: return _AppBarNew(ptabd);
case ABM_REMOVE: _AppBarRemove(ptabd); break;
case ABM_QUERYPOS: _AppBarQueryPos(ptabd); break;
case ABM_SETPOS: _AppBarSetPos(ptabd); break;
case ABM_GETSTATE: return _desktray.AppBarGetState();
case ABM_SETSTATE: _AppBarSetState((UINT)ptabd->abd.lParam); break;
case ABM_GETTASKBARPOS: _AppBarGetTaskBarPos(ptabd); break;
case ABM_WINDOWPOSCHANGED: case ABM_ACTIVATE: _AppBarActivationChange(ptabd); break;
case ABM_GETAUTOHIDEBAR: return (LRESULT)_AppBarGetAutoHideBar(ptabd->abd.uEdge);
case ABM_SETAUTOHIDEBAR: return _AppBarSetAutoHideBar(ptabd);
default: return FALSE; }
return TRUE;
}
// EA486701-7F92-11cf-9E05-444553540000
const GUID CLSID_HIJACKINPROC = {0xEA486701, 0x7F92, 0x11cf, 0x9E, 0x05, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00};
HRESULT CTray::_LoadInProc(PCOPYDATASTRUCT pcds) { ASSERT(pcds->cbData == sizeof(LOADINPROCDATA));
PLOADINPROCDATA plipd = (PLOADINPROCDATA)pcds->lpData;
// Hack to allow us to kill W95 shell extensions that do reall hacky things that
// we can not support. In this case Hijack pro
if (IsEqualIID(plipd->clsid, CLSID_HIJACKINPROC)) { return E_FAIL; }
return _ssomgr.EnableObject(&plipd->clsid, plipd->dwFlags); }
// Allow the trays global hotkeys to be disabled for a while.
LRESULT CTray::_SetHotkeyEnable(HWND hwnd, BOOL fEnable) { _fGlobalHotkeyDisable = !fEnable; return TRUE; }
BOOL IsPosInHwnd(LPARAM lParam, HWND hwnd) { RECT r1; POINT pt;
pt.x = GET_X_LPARAM(lParam); pt.y = GET_Y_LPARAM(lParam); GetWindowRect(hwnd, &r1); return PtInRect(&r1, pt); }
void CTray::_HandleWindowPosChanging(LPWINDOWPOS lpwp) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hwpc"));
if (_uMoveStuckPlace != (UINT)-1) { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hwpc handling pending move")); _DoneMoving(lpwp); } else if (_fSysSizing || !_fSelfSizing) { RECT rc;
if (_fSysSizing) { GetWindowRect(_hwnd, &rc); if (!(lpwp->flags & SWP_NOMOVE)) { rc.left = lpwp->x; rc.top = lpwp->y; } if (!(lpwp->flags & SWP_NOSIZE)) { rc.right = rc.left + lpwp->cx; rc.bottom = rc.top + lpwp->cy; }
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hwpc sys sizing to rect {%d, %d, %d, %d}"), rc.left, rc.top, rc.right, rc.bottom);
_uStuckPlace = _RecalcStuckPos(&rc); _UpdateVertical(_uStuckPlace); }
_GetDockedRect(&rc, _fSysSizing);
DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.t_hwpc using rect {%d, %d, %d, %d}"), rc.left, rc.top, rc.right, rc.bottom);
lpwp->x = rc.left; lpwp->y = rc.top; lpwp->cx = RECTWIDTH(rc); lpwp->cy = RECTHEIGHT(rc);
lpwp->flags &= ~(SWP_NOMOVE | SWP_NOSIZE); }
lpwp->flags |= SWP_FRAMECHANGED; }
void CTray::_HandlePowerStatus(UINT uMsg, WPARAM wParam, LPARAM lParam) { BOOL fResetDisplay = FALSE;
//
// always reset the display when the machine wakes up from a
// suspend. NOTE: we don't need this for a standby suspend.
//
// a critical resume does not generate a WM_POWERBROADCAST
// to windows for some reason, but it does generate an old
// WM_POWER message.
//
switch (uMsg) { case WM_POWER: fResetDisplay = (wParam == PWR_CRITICALRESUME); break;
case WM_POWERBROADCAST: switch (wParam) { case PBT_APMRESUMECRITICAL: fResetDisplay = TRUE; break; } break; }
if (fResetDisplay) ChangeDisplaySettings(NULL, CDS_RESET); }
//////////////////////////////////////////////////////
//
// This function checks whether we need to run the cleaner
// We will not run if user is guest, user has forced us not to, or if the requisite
// number of days have not yet elapsed
//
// We execute a great deal of code to decide whether to run or not that logically should be
// in fldrclnr.dll, but we execute it here so that we don't have to load fldrclnr.dll unless
// we absolutely have to, since we execute this path on every logon of explorer.exe
//
#define REGSTR_PATH_CLEANUPWIZ REGSTR_PATH_EXPLORER TEXT("\\Desktop\\CleanupWiz")
#define REGSTR_OEM_PATH REGSTR_PATH_SETUP TEXT("\\OemStartMenuData")
#define REGSTR_VAL_TIME TEXT("Last used time")
#define REGSTR_VAL_DELTA_DAYS TEXT("Days between clean up")
#define REGSTR_VAL_DONTRUN TEXT("NoRun")
#define REGSTR_OEM_OPTIN TEXT("DoDesktopCleanup")
//
// iDays can be negative or positive, indicating time in the past or future
//
//
#define FTsPerDayOver1000 (10000*60*60*24) // we've got (1000 x 10,000) 100ns intervals per second
void CTray::_DesktopCleanup_GetFileTimeNDaysFromGivenTime(const FILETIME *pftGiven, FILETIME * pftReturn, int iDays) { __int64 i64 = *((__int64 *) pftGiven); i64 += Int32x32To64(iDays*1000,FTsPerDayOver1000);
*pftReturn = *((FILETIME *) &i64); }
//////////////////////////////////////////////////////
BOOL CTray::_DesktopCleanup_ShouldRun() { BOOL fRetVal = FALSE;
if (!IsOS(OS_ANYSERVER) && _fIsDesktopConnected && !SHTestTokenMembership(NULL, DOMAIN_ALIAS_RID_GUESTS) && !SHRestricted(REST_NODESKTOPCLEANUP)) { fRetVal = TRUE;
FILETIME ftNow, ftLast; SYSTEMTIME st; GetLocalTime(&st); SystemTimeToFileTime(&st, &ftNow); DWORD cb = sizeof(ftLast); DWORD dwData; if (ERROR_SUCCESS != SHRegGetUSValue(REGSTR_PATH_CLEANUPWIZ, REGSTR_VAL_TIME, NULL, &ftLast, &cb, FALSE, NULL, 0)) { cb = sizeof(dwData); if ((ERROR_SUCCESS == SHGetValue(HKEY_LOCAL_MACHINE, REGSTR_OEM_PATH, REGSTR_OEM_OPTIN, NULL, &dwData, &cb)) && (dwData != 0)) { // to get the timer to kick in 60 days from now, set last to be now
ftLast = ftNow; } else { // to get the timer to kick in 14 days from now, set last to be 46 days ago
_DesktopCleanup_GetFileTimeNDaysFromGivenTime(&ftNow, &ftLast, -60 + 14); }
SHRegSetUSValue(REGSTR_PATH_CLEANUPWIZ, REGSTR_VAL_TIME, NULL, &ftLast, sizeof(ftLast), SHREGSET_FORCE_HKCU); }
HUSKEY hkey = NULL; if (ERROR_SUCCESS == SHRegOpenUSKey(REGSTR_PATH_CLEANUPWIZ, KEY_READ, NULL, &hkey, FALSE)) { //
// if we're in normal mode and the DONT RUN flag is set, we return immediately
// (the user checked the "don't run automatically" box)
//
cb = sizeof (DWORD); if ((ERROR_SUCCESS == SHRegQueryUSValue(hkey, REGSTR_VAL_DONTRUN, NULL, &dwData, &cb, FALSE, NULL, 0)) && (dwData != 0)) { fRetVal = FALSE; } else { //
// we need to figure out if if we are within the (last run time + delta days)
// time period
//
int iDays = 60; if (ERROR_SUCCESS == (SHRegGetUSValue(REGSTR_PATH_CLEANUPWIZ, REGSTR_VAL_DELTA_DAYS, NULL, &dwData, &cb,FALSE, NULL, 0))) { iDays = dwData; }
// if (iDays == 0), run every time!
if (iDays > 0) { FILETIME ftRange; _DesktopCleanup_GetFileTimeNDaysFromGivenTime(&ftLast, &ftRange, iDays); if (!(CompareFileTime(&ftNow, &ftRange) > 0)) { fRetVal = FALSE; } } }
SHRegCloseUSKey(hkey); } }
return fRetVal; }
void CTray::_CheckDesktopCleanup() { if (_DesktopCleanup_ShouldRun()) { PROCESS_INFORMATION pi = {0}; TCHAR szRunDll32[MAX_PATH];
GetSystemDirectory(szRunDll32, ARRAYSIZE(szRunDll32)); PathAppend(szRunDll32, TEXT("rundll32.exe"));
if (CreateProcessWithArgs(szRunDll32, TEXT("fldrclnr.dll,Wizard_RunDLL"), NULL, &pi)) { CloseHandle(pi.hProcess); CloseHandle(pi.hThread); } } }
//////////////////////////////////////////////////////
//
// Try tacking a 1, 2, 3 or whatever on to a file or
// directory name until it is unique. When on a file,
// stick it before the extension.
//
void _MakeBetterUniqueName(LPTSTR pszPathName) { TCHAR szNewPath[MAX_PATH]; int i = 1;
if (PathIsDirectory(pszPathName)) { do { wsprintf(szNewPath, TEXT("%s%d"), pszPathName, i++); } while (-1 != GetFileAttributes(szNewPath));
lstrcpy(pszPathName, szNewPath); } else { TCHAR szExt[MAX_PATH]; LPTSTR pszExt;
pszExt = PathFindExtension(pszPathName); if (pszExt) { lstrcpy(szExt, pszExt); *pszExt = 0;
do { wsprintf(szNewPath, TEXT("%s%d%s"), pszPathName, i++,szExt); } while (-1 != GetFileAttributes(szNewPath));
lstrcpy(pszPathName, szNewPath); } } }
BOOL_PTR WINAPI CTray::RogueProgramFileDlgProc(HWND hWnd, UINT iMsg, WPARAM wParam, LPARAM lParam) { TCHAR szBuffer[MAX_PATH*2]; TCHAR szBuffer2[MAX_PATH*2]; static TCHAR szBetterPath[MAX_PATH]; static TCHAR *pszPath = NULL; switch (iMsg) { case WM_INITDIALOG: pszPath = (TCHAR *)lParam; lstrcpy(szBetterPath, pszPath); _MakeBetterUniqueName(szBetterPath); SendDlgItemMessage(hWnd, IDC_MSG, WM_GETTEXT, (WPARAM)(MAX_PATH*2), (LPARAM)szBuffer); wsprintf(szBuffer2, szBuffer, pszPath, szBetterPath); SendDlgItemMessage(hWnd, IDC_MSG, WM_SETTEXT, (WPARAM)0, (LPARAM)szBuffer2);
return TRUE;
case WM_COMMAND: switch(LOWORD(wParam)) { case IDC_RENAME: //rename and fall through
if (pszPath) { MoveFile(pszPath, szBetterPath); } EndDialog(hWnd, IDC_RENAME); return TRUE;
case IDIGNORE: EndDialog(hWnd, IDIGNORE); return TRUE; } break; } return FALSE; }
//
// Check to see if there are any files or folders that could interfere
// with the fact that Program Files has a space in it.
//
// An example would be a directory called: "C:\Program" or a file called"C:\Program.exe".
//
// This can prevent apps that dont quote strings in the registry or call CreateProcess with
// unquoted strings from working properly since CreateProcess wont know what the real exe is.
//
void CTray::_CheckForRogueProgramFile() { TCHAR szProgramFilesPath[MAX_PATH]; TCHAR szProgramFilesShortName[MAX_PATH];
if (S_OK == SHGetFolderPath(NULL, CSIDL_PROGRAM_FILES, NULL, SHGFP_TYPE_CURRENT, szProgramFilesPath)) { LPTSTR pszRoguePattern;
pszRoguePattern = StrChr(szProgramFilesPath, TEXT(' ')); if (pszRoguePattern) { HANDLE hFind; WIN32_FIND_DATA wfd;
// Remember short name for folder name comparison below
*pszRoguePattern = TEXT('\0'); lstrcpy(szProgramFilesShortName, szProgramFilesPath);
// turn "C:\program files" into "C:\program.*"
lstrcpy(pszRoguePattern, TEXT(".*")); pszRoguePattern = szProgramFilesPath;
hFind = FindFirstFile(pszRoguePattern, &wfd);
while (hFind != INVALID_HANDLE_VALUE) { int iRet = 0; TCHAR szRogueFileName[MAX_PATH];
// we found a file (eg "c:\Program.txt")
lstrcpyn(szRogueFileName, pszRoguePattern, ARRAYSIZE(szRogueFileName)); PathRemoveFileSpec(szRogueFileName); lstrcatn(szRogueFileName, wfd.cFileName, ARRAYSIZE(szRogueFileName));
// don't worry about folders unless they are called "Program"
if (!((wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) && lstrcmpi(szProgramFilesShortName, szRogueFileName) != 0)) { iRet = SHMessageBoxCheckEx(GetDesktopWindow(), hinstCabinet, MAKEINTRESOURCE(DLG_PROGRAMFILECONFLICT), RogueProgramFileDlgProc, (void *)szRogueFileName, IDIGNORE, TEXT("RogueProgramName")); } if ((iRet == IDIGNORE) || !FindNextFile(hFind, &wfd)) { // user hit ignore or we are done, so don't keep going
break; } }
if (hFind != INVALID_HANDLE_VALUE) { FindClose(hFind); } } } }
void CTray::_OnWaitCursorNotify(LPNMHDR pnm) { _iWaitCount += (pnm->code == NM_STARTWAIT ? 1 :-1); ASSERT(_iWaitCount >= 0); // Don't let it go negative or we'll never get rid of it.
if (_iWaitCount < 0) _iWaitCount = 0; // what we really want is for user to simulate a mouse move/setcursor
SetCursor(LoadCursor(NULL, _iWaitCount ? IDC_APPSTARTING : IDC_ARROW)); }
void CTray::_HandlePrivateCommand(LPARAM lParam) { LPSTR psz = (LPSTR) lParam; // lParam always ansi.
if (!lstrcmpiA(psz, "ToggleDesktop")) { _RaiseDesktop(!g_fDesktopRaised, TRUE); } else if (!lstrcmpiA(psz, "Explorer")) { // Fast way to bring up explorer window on root of
// windows dir.
SHELLEXECUTEINFO shei = {0}; TCHAR szPath[MAX_PATH];
if (GetWindowsDirectory(szPath, ARRAYSIZE(szPath)) != 0) { PathStripToRoot(szPath);
shei.lpIDList = ILCreateFromPath(szPath); if (shei.lpIDList) { shei.cbSize = sizeof(shei); shei.fMask = SEE_MASK_IDLIST; shei.nShow = SW_SHOWNORMAL; shei.lpVerb = TEXT("explore"); ShellExecuteEx(&shei); ILFree((LPITEMIDLIST)shei.lpIDList); } } }
LocalFree(psz); }
//***
//
void CTray::_OnFocusMsg(UINT uMsg, WPARAM wParam, LPARAM lParam) { BOOL fActivate = (BOOL) wParam;
switch (uMsg) { case TM_UIACTIVATEIO: { #ifdef DEBUG
{ int dtb = (int) lParam;
TraceMsg(DM_FOCUS, "tiois: TM_UIActIO fAct=%d dtb=%d", fActivate, dtb);
ASSERT(dtb == 1 || dtb == -1); } #endif
if (fActivate) { // Since we are tabbing into the tray, turn the focus rect on.
SendMessage(_hwnd, WM_UPDATEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);
SendMessage(v_hwndDesktop, DTM_ONFOCUSCHANGEIS, TRUE, (LPARAM) _hwnd); SetForegroundWindow(_hwnd);
// fake an IUnknown_UIActivateIO(_ptbs, TRUE, &msg);
if (GetAsyncKeyState(VK_SHIFT) < 0) { _SetFocus(_hwndNotify); } else { _SetFocus(_hwndStart); } } else { Ldeact: IUnknown_UIActivateIO(_ptbs, FALSE, NULL); SetForegroundWindow(v_hwndDesktop); }
break; }
case TM_ONFOCUSCHANGEIS: { HWND hwnd = (HWND) lParam;
TraceMsg(DM_FOCUS, "tiois: TM_OnFocChgIS hwnd=%x fAct=%d", hwnd, fActivate);
if (fActivate) { // someone else is activating, so we need to deactivate
goto Ldeact; }
break; }
default: ASSERT(0); break; }
return; }
#define TSVC_NTIMER (IDT_SERVICELAST - IDT_SERVICE0 + 1)
struct { #ifdef DEBUG
UINT_PTR idtWin; #endif
TIMERPROC pfnSvc; } g_timerService[TSVC_NTIMER];
#define TSVC_IDToIndex(id) ((id) - IDT_SERVICE0)
#define TSVC_IndexToID(i) ((i) + IDT_SERVICE0)
int CTray::_OnTimerService(UINT uMsg, WPARAM wParam, LPARAM lParam) { int i; UINT_PTR idt; TIMERPROC pfn; BOOL b;
switch (uMsg) { case TM_SETTIMER: TraceMsg(DM_UEMTRACE, "e.TM_SETTIMER: wP=0x%x lP=%x", wParam, lParam); ASSERT(IS_VALID_CODE_PTR(lParam, TIMERPROC)); for (i = 0; i < TSVC_NTIMER; i++) { if (g_timerService[i].pfnSvc == 0) { g_timerService[i].pfnSvc = (TIMERPROC)lParam; idt = SetTimer(_hwnd, TSVC_IndexToID(i), (UINT)wParam, 0); if (idt == 0) { TraceMsg(DM_UEMTRACE, "e.TM_SETTIMER: ST()=%d (!)", idt); break; } ASSERT(idt == (UINT_PTR)TSVC_IndexToID(i)); DBEXEC(TRUE, (g_timerService[i].idtWin = idt)); TraceMsg(DM_UEMTRACE, "e.TM_SETTIMER: ret=0x%x", TSVC_IndexToID(i)); return TSVC_IndexToID(i); // idtWin
} } TraceMsg(DM_UEMTRACE, "e.TM_SETTIMER: ret=0 (!)"); return 0;
case TM_KILLTIMER: // lP=idtWin
TraceMsg(DM_UEMTRACE, "e.TM_KILLTIMER: wP=0x%x lP=%x", wParam, lParam); if (EVAL(IDT_SERVICE0 <= lParam && lParam <= IDT_SERVICE0 + TSVC_NTIMER - 1)) { i = (int)TSVC_IDToIndex(lParam); if (g_timerService[i].pfnSvc) { ASSERT(g_timerService[i].idtWin == (UINT)lParam); b = KillTimer(_hwnd, lParam); ASSERT(b); g_timerService[i].pfnSvc = 0; DBEXEC(TRUE, (g_timerService[i].idtWin = 0)); return TRUE; } } return 0;
case WM_TIMER: // wP=idtWin lP=0
TraceMsg(DM_UEMTRACE, "e.TM_TIMER: wP=0x%x lP=%x", wParam, lParam); if (EVAL(IDT_SERVICE0 <= wParam && wParam <= IDT_SERVICE0 + TSVC_NTIMER - 1)) { i = (int)TSVC_IDToIndex(wParam); pfn = g_timerService[i].pfnSvc; if (EVAL(IS_VALID_CODE_PTR(pfn, TIMERPROC))) (*pfn)(_hwnd, WM_TIMER, wParam, GetTickCount()); } return 0; }
ASSERT(0); /*NOTREACHED*/ return 0; }
void CTray::RealityCheck() { //
// Make sure that the tray's actual z-order position agrees with what we think
// it is. We need to do this because there's a recurring bug where the tray
// gets bumped out of TOPMOST position. (Lots of things, like a tray-owned
// window moving itself to non-TOPMOST or a random app messing with the tray
// window position, can cause this.)
//
_ResetZorder(); }
#define DELAY_STARTUPTROUBLESHOOT (15 * 1000)
void CTray::LogFailedStartupApp() { if (_hwnd) { PostMessage(_hwnd, TM_HANDLESTARTUPFAILED, 0, 0); } else { _fEarlyStartupFailure = TRUE; } }
void WINAPI CTray::TroubleShootStartupCB(HWND hwnd, UINT uMsg, UINT_PTR idTimer, DWORD dwTime) { KillTimer(hwnd, idTimer);
if (!c_tray._fStartupTroubleshooterLaunched) { TCHAR szCmdLine[MAX_PATH]; DWORD cb;
c_tray._fStartupTroubleshooterLaunched = TRUE; cb = sizeof(szCmdLine); if (SHGetValue(HKEY_LOCAL_MACHINE, REGSTR_PATH_EXPLORER, TEXT("StartupTroubleshoot"), NULL, szCmdLine, &cb) == ERROR_SUCCESS) { ShellExecuteRegApp(szCmdLine, RRA_NOUI | RRA_DELETE); } } }
void CTray::_OnHandleStartupFailed() { /*
* Don't launch the troubleshooter until we have gone * DELAY_STARTUPTROUBLESHOOT milliseconds without a startup problem. * This gives time for the system to settle down before starting to * annoy the user all over again. * * (And, of course, don't launch it more than once.) */ if (!_fStartupTroubleshooterLaunched) { SetTimer(_hwnd, IDT_STARTUPFAILED, DELAY_STARTUPTROUBLESHOOT, TroubleShootStartupCB); } }
void CTray::_HandleDelayBootStuff() { // This posted message is the last one processed by the primary
// thread (tray thread) when we boot. At this point we will
// want to load the shell services (which usually create threads)
// and resume both the background start menu thread and the fs_notfiy
// thread.
if (!_fHandledDelayBootStuff) { if (GetShellWindow() == NULL) { // The desktop browser hasn't finished navigating yet.
SetTimer(_hwnd, IDT_HANDLEDELAYBOOTSTUFF, 3 * 1000, NULL); return; }
_fHandledDelayBootStuff = TRUE;
if (g_dwStopWatchMode) { StopWatch_StartTimed(SWID_STARTUP, TEXT("_DelayedBootStuff"), SPMODE_SHELL | SPMODE_DEBUGOUT, GetPerfTime()); }
PostMessage(_hwnd, TM_SHELLSERVICEOBJECTS, 0, 0);
BandSite_HandleDelayBootStuff(_ptbs);
//check to see if there are any files or folders that could interfere
//with the fact that Program Files has a space in it. An example would
//be a folder called "C:\Program" or a file called "C:\Program.exe"
_CheckForRogueProgramFile(); // Create a named event and fire it so that the services can
// go to work, reducing contention during boot.
_hShellReadyEvent = CreateEvent(0, TRUE, TRUE, TEXT("ShellReadyEvent")); if (_hShellReadyEvent) { // Set the event in case it was already created and our "create
// signaled" parameter to CreateEvent got ignored.
SetEvent(_hShellReadyEvent); }
// Check whether we should launch Desktop Cleanup Wizard
_CheckDesktopCleanup();
TBOOL(WinStationRegisterConsoleNotification(SERVERNAME_CURRENT, _hwnd, NOTIFY_FOR_THIS_SESSION));
if (g_dwStopWatchMode) { StopWatch_StopTimed(SWID_STARTUP, TEXT("_DelayedBootStuff"), SPMODE_SHELL | SPMODE_DEBUGOUT, GetPerfTime()); } } }
LRESULT CTray::_OnDeviceChange(HWND hwnd, WPARAM wParam, LPARAM lParam) { switch (wParam) { case DBT_CONFIGCHANGED: // We got an update. Refresh.
_RefreshStartMenu(); break;
case DBT_QUERYCHANGECONFIG: //
// change to registry settings
//
ChangeDisplaySettings(NULL, 0); break;
case DBT_MONITORCHANGE: //
// handle monitor change
//
HandleDisplayChange(LOWORD(lParam), HIWORD(lParam), TRUE); break;
case DBT_CONFIGCHANGECANCELED: //
// if the config change was canceled go back
//
HandleDisplayChange(0, 0, FALSE); break; }
Mixer_DeviceChange(wParam, lParam); return 0; }
//
// The "resizable" edge of the taskbar is the edge adjacent to
// the desktop, i.e. opposite the stuck place.
//
// returns HTXXX if on a resizable edge, else HTBORDER
//
DWORD CTray::_PtOnResizableEdge(POINT pt, LPRECT prcClient) { RECT rc; GetWindowRect(_hwnd, &rc);
DWORD dwHit = HTBORDER;
switch (_uStuckPlace) { case STICK_LEFT: rc.left = prcClient->right; dwHit = HTRIGHT; break; case STICK_TOP: rc.top = prcClient->bottom; dwHit = HTBOTTOM; break; case STICK_RIGHT: rc.right = prcClient->left; dwHit = HTLEFT; break; case STICK_BOTTOM: rc.bottom = prcClient->top; dwHit = HTTOP; break; }
return PtInRect(&rc, pt) ? dwHit : HTBORDER; }
//
// _OnFactoryMessage
//
// The OPK "factory.exe" tool sends us this message to tell us that
// it has dorked some setting or other and we should refresh so the
// OEM can see the effect immediately and feel confident that it
// actually worked. This is not technically necessary but it cuts
// down on OEM support calls when they ask us why their setting didn't
// work. (It did work, they just have to log off and back on to see
// it.)
//
int CTray::_OnFactoryMessage(WPARAM wParam, LPARAM lParam) { switch (wParam) { case 0: // FACTORY_OEMLINK: factory.exe has dorked the OEM link
ClosePopupMenus(); _BuildStartMenu(); // Force a rebuild
return 1;
case 1: // FACTORY_MFU: factory.exe has written a new MFU
HandleFirstTime(); // Rebuild the default MFU
ClosePopupMenus(); _BuildStartMenu(); // Force a rebuild
return 1; }
return 0; }
#define CX_OFFSET g_cxEdge
#define CY_OFFSET g_cyEdge
//
// _MapNCToClient
//
// see comments in _TryForwardNCToClient
//
BOOL CTray::_MapNCToClient(LPARAM* plParam) { POINT pt = { GET_X_LPARAM(*plParam), GET_Y_LPARAM(*plParam) };
RECT rcClient; GetClientRect(_hwnd, &rcClient); MapWindowPoints(_hwnd, NULL, (LPPOINT)&rcClient, 2);
//
// point must be outside the client area and not on the
// resizable edge of the taskbar
//
if (!PtInRect(&rcClient, pt) && _PtOnResizableEdge(pt, &rcClient) == HTBORDER) { //
// fudge it over onto the client edge and return TRUE
//
if (pt.x < rcClient.left) pt.x = rcClient.left + CX_OFFSET; else if (pt.x > rcClient.right) pt.x = rcClient.right - CX_OFFSET;
if (pt.y < rcClient.top) pt.y = rcClient.top + CY_OFFSET; else if (pt.y > rcClient.bottom) pt.y = rcClient.bottom - CY_OFFSET;
*plParam = MAKELONG(pt.x, pt.y); return TRUE; }
//
// didn't pass the test. leave the point alone and return FALSE.
//
return FALSE; }
HWND _TopChildWindowFromPoint(HWND hwnd, POINT pt) { HWND hwndLast = NULL; hwnd = ChildWindowFromPoint(hwnd, pt); while (hwnd && hwnd != hwndLast) { hwndLast = hwnd; hwnd = ChildWindowFromPoint(hwnd, pt); } return hwndLast; }
//
// _TryForwardNCToClient
//
// Hack! This exists to solve a usability problem. When you slam your
// mouse into the bottom corner of the screen and click, we want that to
// activate the start button. Similarly, when you slam your mouse below
// a Quick Launch button or task button and click, we want that to
// activate the button.
//
// We hack this by remapping the coordinate of NC mouse messages and
// manually forwarding to the appropriate window. We only do this for
// clicks on the edges non-resizable edge of the taskbar.
//
// We also warp the mouse cursor over to the new position. This is needed
// because e.g. if toolbar is the client window we're forwarding
// to, it will set capture and receive subsequent mouse messages. (And
// once it gets a mouse message outside the window, it will deselect the
// button and so the button won't get activated.)
//
// _MapNCToClient has the rules for figuring out if the click is on
// one of the edges we want and for remapping the coordinate into the
// client area.
//
BOOL CTray::_TryForwardNCToClient(UINT uMsg, LPARAM lParam) { if (_MapNCToClient(&lParam)) { // see if this is over one of our windows
POINT pt = { GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam) }; MapWindowPoints(NULL, _hwnd, &pt, 1); HWND hwnd = _TopChildWindowFromPoint(_hwnd, pt);
if (hwnd) { // warp the mouse cursor to this screen coord
SetCursorPos(GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam));
// map to window coords
MapWindowPoints(_hwnd, hwnd, &pt, 1);
// set lparam to window coords
lParam = MAKELONG(pt.x, pt.y);
// map to client message
ASSERT(InRange(uMsg, WM_NCMOUSEFIRST, WM_NCMOUSELAST)); uMsg += (WM_LBUTTONDOWN - WM_NCLBUTTONDOWN);
// forward it
SendMessage(hwnd, uMsg, 0, lParam); return TRUE; } } return FALSE; }
// --------------------------------------------------------------------------
// CTray::CountOfRunningPrograms
//
// Arguments: <none>
//
// Returns: DWORD
//
// Purpose: Iterates the window list. Looks for windows that are visible
// with a non-zero length window title. Gets that window process
// ID and keeps the IDs in a list. For each window iterated it
// checks against the list to see if the process is already
// tagged and if so doesn't add it again. Finally it returns the
// count of the unique processes handling open visible windows
// in the user's desktop.
//
// The list is fixed at 1000 processes (using stack space).
//
// History: 2000-06-29 vtan created
// --------------------------------------------------------------------------
static const int MAXIMUM_PROCESS_COUNT = 1000;
typedef struct { DWORD dwCount; DWORD dwProcessIDs[MAXIMUM_PROCESS_COUNT]; } tProcessIDList;
bool FoundProcessID (tProcessIDList *pProcessIDList, DWORD dwProcessID)
{ bool fFound; DWORD dwIndex;
for (fFound = false, dwIndex = 0; !fFound && (dwIndex < pProcessIDList->dwCount); ++dwIndex) { fFound = (pProcessIDList->dwProcessIDs[dwIndex] == dwProcessID); } return(fFound); }
void AddProcessID (tProcessIDList *pProcessIDList, DWORD dwProcessID)
{ if (pProcessIDList->dwCount < MAXIMUM_PROCESS_COUNT) { pProcessIDList->dwProcessIDs[pProcessIDList->dwCount++] = dwProcessID; } }
BOOL CALLBACK CountOfRunningProgramsEnumWindowsProc (HWND hwnd, LPARAM lParam)
{ if ((GetShellWindow() != hwnd) && IsWindowVisible(hwnd)) { DWORD dwThreadID, dwProcessID; TCHAR szWindowTitle[256];
dwThreadID = GetWindowThreadProcessId(hwnd, &dwProcessID); if ((InternalGetWindowText(hwnd, szWindowTitle, ARRAYSIZE(szWindowTitle)) > 0) && (szWindowTitle[0] != TEXT('\0'))) { if (!FoundProcessID(reinterpret_cast<tProcessIDList*>(lParam), dwProcessID)) { AddProcessID(reinterpret_cast<tProcessIDList*>(lParam), dwProcessID); } } } return(TRUE); }
DWORD CTray::CountOfRunningPrograms() { tProcessIDList processIDList = {0}; TBOOL(EnumWindows(CountOfRunningProgramsEnumWindowsProc, reinterpret_cast<LPARAM>(&processIDList))); return processIDList.dwCount; }
// --------------------------------------------------------------------------
// CTray::_OnSessionChange
//
// Arguments: wParam = WTS_xxx notification.
// lParam = WTS_SESSION_NOTIFICATION struct pointer.
//
// Returns: LRESULT
//
// Purpose: Handles console/remote dis/reconnects.
//
// History: 2000-07-12 vtan created
// --------------------------------------------------------------------------
LRESULT CTray::_OnSessionChange(WPARAM wParam, LPARAM lParam) { ASSERTMSG(DWORD(lParam) == NtCurrentPeb()->SessionId, "Session ID mismatch in CTray::_OnSessionChange");
if ((WTS_CONSOLE_CONNECT == wParam) || (WTS_REMOTE_CONNECT == wParam) || (WTS_SESSION_UNLOCK == wParam)) { _fIsDesktopConnected = TRUE; } else if ((WTS_CONSOLE_DISCONNECT == wParam) || (WTS_REMOTE_DISCONNECT == wParam) || (WTS_SESSION_LOCK == wParam)) { _fIsDesktopConnected = FALSE; }
if ((WTS_CONSOLE_CONNECT == wParam) || (WTS_REMOTE_CONNECT == wParam)) { _RefreshStartMenu(); SHUpdateRecycleBinIcon(); } else if ((WTS_SESSION_LOCK == wParam) || (WTS_SESSION_UNLOCK == wParam)) { if (IsOS(OS_FASTUSERSWITCHING)) { if (wParam == WTS_SESSION_LOCK) { ExplorerPlaySound(TEXT("WindowsLogoff")); } else if (wParam == WTS_SESSION_UNLOCK) { ExplorerPlaySound(TEXT("WindowsLogon")); } }
PostMessage(_hwnd, TM_WORKSTATIONLOCKED, (WTS_SESSION_LOCK == wParam), 0); }
return 1; }
LRESULT CTray::_NCPaint(HRGN hrgn) { ASSERT(_hTheme);
if (_fCanSizeMove || _fShowSizingBarAlways) { if ((INT_PTR)hrgn == 1) hrgn = NULL;
HDC hdc = GetDCEx( _hwnd, hrgn, DCX_USESTYLE|DCX_WINDOW|DCX_LOCKWINDOWUPDATE| ((hrgn != NULL) ? DCX_INTERSECTRGN|DCX_NODELETERGN : 0));
if (hdc) { RECT rc; GetWindowRect(_hwnd, &rc); OffsetRect(&rc, -rc.left, -rc.top);
_AdjustRectForSizingBar(_uStuckPlace, &rc, 0); DrawThemeBackground(_hTheme, hdc, _GetPart(TRUE, _uStuckPlace), 0, &rc, 0); ReleaseDC(_hwnd, hdc); } }
return 0; }
LRESULT CTray::v_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { static UINT uDDEExec = 0; LRESULT lres = 0; MSG msg;
msg.hwnd = hwnd; msg.message = uMsg; msg.wParam = wParam; msg.lParam = lParam;
if (_pmbStartMenu && _pmbStartMenu->TranslateMenuMessage(&msg, &lres) == S_OK) return lres;
if (_pmbStartPane && _pmbStartPane->TranslateMenuMessage(&msg, &lres) == S_OK) return lres;
if (_pmbTasks && _pmbTasks->TranslateMenuMessage(&msg, &lres) == S_OK) return lres;
wParam = msg.wParam; lParam = msg.lParam;
INSTRUMENT_WNDPROC(SHCNFI_TRAY_WNDPROC, hwnd, uMsg, wParam, lParam);
switch (uMsg) { case WMTRAY_REGISTERHOTKEY: return _RegisterHotkey(hwnd, (int)wParam);
case WMTRAY_UNREGISTERHOTKEY: return _UnregisterHotkey(hwnd, (int)wParam);
case WMTRAY_SCREGISTERHOTKEY: return _ShortcutRegisterHotkey(hwnd, (WORD)wParam, (ATOM)lParam);
case WMTRAY_SCUNREGISTERHOTKEY: return _ShortcutUnregisterHotkey(hwnd, (WORD)wParam);
case WMTRAY_SETHOTKEYENABLE: return _SetHotkeyEnable(hwnd, (BOOL)wParam);
case WMTRAY_QUERY_MENU: return (LRESULT)_hmenuStart;
case WMTRAY_QUERY_VIEW: return (LRESULT)_hwndTasks;
case WMTRAY_TOGGLEQL: return _ToggleQL((int)lParam);
case WM_COPYDATA: // Check for NULL it can happen if user runs out of selectors or memory...
if (lParam) { switch (((PCOPYDATASTRUCT)lParam)->dwData) { case TCDM_NOTIFY: { BOOL bRefresh = FALSE;
lres = _trayNotify.TrayNotify(_hwndNotify, (HWND)wParam, (PCOPYDATASTRUCT)lParam, &bRefresh); if (bRefresh) { SizeWindows(); } return(lres); } case TCDM_APPBAR: return _OnAppBarMessage((PCOPYDATASTRUCT)lParam);
case TCDM_LOADINPROC: return (UINT)_LoadInProc((PCOPYDATASTRUCT)lParam); } } return FALSE;
case WM_NCCALCSIZE: if (_hTheme) { if ((_fCanSizeMove || _fShowSizingBarAlways) && lParam) { _AdjustRectForSizingBar(_uStuckPlace, (LPRECT)lParam, -1); }
return 0; } else { goto L_default; } break;
case WM_NCLBUTTONDBLCLK: if (!_TryForwardNCToClient(uMsg, lParam)) { if (IsPosInHwnd(lParam, _hwndNotify)) { _Command(IDM_SETTIME);
// Hack! If you click on the tray clock, this tells the tooltip
// "Hey, I'm using this thing; stop putting up a tip for me."
// You can get the tooltip to lose track of when it needs to
// reset the "stop it!" flag and you get stuck in "stop it!" mode.
// It's particularly easy to make happen on Terminal Server.
//
// So let's assume that the only reason people click on the
// tray clock is to change the time. when they change the time,
// kick the tooltip in the head to reset the "stop it!" flag.
SendMessage(_hwndTrayTips, TTM_POP, 0, 0); } } break;
case WM_NCLBUTTONDOWN: case WM_NCLBUTTONUP: if (!_TryForwardNCToClient(uMsg, lParam)) { goto L_WM_NCMOUSEMOVE; } break;
case WM_NCMOUSEMOVE: L_WM_NCMOUSEMOVE: if (IsPosInHwnd(lParam, _hwndNotify)) { MSG msgInner; msgInner.lParam = lParam; msgInner.wParam = wParam; msgInner.message = uMsg; msgInner.hwnd = hwnd; SendMessage(_hwndTrayTips, TTM_RELAYEVENT, 0, (LPARAM)(LPMSG)&msgInner); if (uMsg == WM_NCLBUTTONDOWN) _SetFocus(_hwndNotify); } goto DoDefault;
case WM_CREATE: return _OnCreate(hwnd);
case WM_DESTROY: return _HandleDestroy();
#ifdef DEBUG
case WM_QUERYENDSESSION: TraceMsg(DM_SHUTDOWN, "Tray.wp WM_QUERYENDSESSION"); goto DoDefault; #endif
case WM_ENDSESSION: // save our settings if we are shutting down
if (wParam) { if (lParam | ENDSESSION_LOGOFF) { _fIsLogoff = TRUE; _RecomputeAllWorkareas(); }
_SaveTrayAndDesktop();
ShowWindow(_hwnd, SW_HIDE); ShowWindow(v_hwndDesktop, SW_HIDE);
DestroyWindow(_hwnd); } break;
case WM_PRINTCLIENT: case WM_PAINT: { RECT rc; PAINTSTRUCT ps; HDC hdc = (HDC)wParam; if (hdc == 0) hdc = BeginPaint(hwnd, &ps);
GetClientRect(hwnd, &rc);
if (_hTheme) { RECT rcClip; if (GetClipBox(hdc, &rcClip) == NULLREGION) rcClip = rc;
DrawThemeBackground(_hTheme, hdc, _GetPart(FALSE, _uStuckPlace), 0, &rc, &rcClip); } else { FillRect(hdc, &rc, (HBRUSH)(COLOR_3DFACE + 1)); // draw etched line around on either side of the bandsite
MapWindowPoints(HWND_DESKTOP, hwnd, (LPPOINT)&rc, 2); InflateRect(&rc, g_cxEdge, g_cyEdge); DrawEdge(hdc, &rc, EDGE_ETCHED, BF_TOPLEFT); }
if (wParam == 0) EndPaint(hwnd, &ps); } break;
case WM_ERASEBKGND: if (_hTheme) { if (!_fSkipErase) { RECT rc; GetClientRect(hwnd, &rc); DrawThemeBackground(_hTheme, (HDC)wParam, _GetPart(FALSE, _uStuckPlace), 0, &rc, NULL);
// Only draw the first time to prevent piece-mail taskbar painting
_fSkipErase = TRUE; } return 1; } else { goto DoDefault; } break;
case WM_NCPAINT: if (_hTheme) { return _NCPaint((HRGN)wParam); } else { goto DoDefault; } break;
case WM_POWER: case WM_POWERBROADCAST: _PropagateMessage(hwnd, uMsg, wParam, lParam); _HandlePowerStatus(uMsg, wParam, lParam); goto DoDefault;
case WM_DEVICECHANGE: lres = _OnDeviceChange(hwnd, wParam, lParam); if (lres == 0) { goto DoDefault; } break;
case WM_NOTIFY: { NMHDR *pnm = (NMHDR*)lParam; if (!BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, &lres)) { switch (pnm->code) { case SEN_DDEEXECUTE: if (((LPNMHDR)lParam)->idFrom == 0) { LPNMVIEWFOLDER pnmPost = DDECreatePostNotify((LPNMVIEWFOLDER)pnm);
if (pnmPost) { PostMessage(hwnd, GetDDEExecMsg(), 0, (LPARAM)pnmPost); return TRUE; } } break;
case NM_STARTWAIT: case NM_ENDWAIT: _OnWaitCursorNotify((NMHDR *)lParam); PostMessage(v_hwndDesktop, ((NMHDR*)lParam)->code == NM_STARTWAIT ? DTM_STARTWAIT : DTM_ENDWAIT, 0, 0); // forward it along
break;
case NM_THEMECHANGED: // Force the start button to recalc its size
_sizeStart.cx = 0; SizeWindows(); break;
case TTN_NEEDTEXT: //
// Make the clock manage its own tooltip.
//
return SendMessage(_GetClockWindow(), WM_NOTIFY, wParam, lParam);
case TTN_SHOW: SetWindowZorder(_hwndTrayTips, HWND_TOP); break;
} } break; }
case WM_CLOSE: _DoExitWindows(v_hwndDesktop); break;
case WM_NCHITTEST: { RECT r1; POINT pt;
GetClientRect(hwnd, &r1); MapWindowPoints(hwnd, NULL, (LPPOINT)&r1, 2);
pt.x = GET_X_LPARAM(lParam); pt.y = GET_Y_LPARAM(lParam);
_SetUnhideTimer(pt.x, pt.y);
// If the user can't size or move the taskbar, then just say
// they hit something useless
if (!_fCanSizeMove) { return HTBORDER; } else if (PtInRect(&r1, pt)) { // allow dragging if mouse is in client area of _hwnd
return HTCAPTION; } else { return _PtOnResizableEdge(pt, &r1); } } break;
case WM_WINDOWPOSCHANGING: _HandleWindowPosChanging((LPWINDOWPOS)lParam); break;
case WM_ENTERSIZEMOVE: DebugMsg(DM_TRAYDOCK, TEXT("Tray -- WM_ENTERSIZEMOVE")); g_fInSizeMove = TRUE; GetCursorPos((LPPOINT)&_rcSizeMoveIgnore); _rcSizeMoveIgnore.right = _rcSizeMoveIgnore.left; _rcSizeMoveIgnore.bottom = _rcSizeMoveIgnore.top; InflateRect(&_rcSizeMoveIgnore, GetSystemMetrics(SM_CXICON), GetSystemMetrics(SM_CYICON));
//
// unclip the tray from the current monitor.
// keeping the tray properly clipped in the MoveSize loop is extremely
// hairy and provides almost no benefit. we'll re-clip when it lands.
//
_ClipWindow(FALSE);
// Remember the old monitor we were on
_hmonOld = _hmonStuck;
// set up for WM_MOVING/WM_SIZING messages
_uMoveStuckPlace = (UINT)-1; _fSysSizing = TRUE; if (!g_fDragFullWindows) { SendMessage(_hwndRebar, WM_SETREDRAW, FALSE, 0); } break;
case WM_EXITSIZEMOVE: DebugMsg(DM_TRAYDOCK, TEXT("Tray -- WM_EXITSIZEMOVE"));
// done sizing
_fSysSizing = FALSE; _fDeferedPosRectChange = FALSE;
if (!g_fDragFullWindows) { SendMessage(_hwndRebar, WM_SETREDRAW, TRUE, 0); }
//
// kick the size code one last time after the loop is done.
// NOTE: we rely on the WM_SIZE code re-clipping the tray.
//
PostMessage(hwnd, WM_SIZE, 0, 0L); g_fInSizeMove = FALSE; break;
case WM_MOVING: _HandleMoving(wParam, (LPRECT)lParam); break;
case WM_ENTERMENULOOP: // DebugMsg(DM_TRACE, "c.twp: Enter menu loop.");
_HandleEnterMenuLoop(); break;
case WM_EXITMENULOOP: // DebugMsg(DM_TRACE, "c.twp: Exit menu loop.");
_HandleExitMenuLoop(); break;
case WM_TIMER: if (IDT_SERVICE0 <= wParam && wParam <= IDT_SERVICELAST) return _OnTimerService(uMsg, wParam, lParam); _HandleTimer(wParam); break;
case WM_SIZING: _HandleSizing(wParam, (LPRECT)lParam, _uStuckPlace); break;
case WM_SIZE: _HandleSize(); break;
case WM_DISPLAYCHANGE: // NOTE: we get WM_DISPLAYCHANGE in the below two situations
// 1. a display size changes (HMON will not change in USER)
// 2. a display goes away or gets added (HMON will change even if
// the monitor that went away has nothing to do with our hmonStuck)
// In the above two situations we actually need to do different things
// because in 1, we do not want to update our hmonStuck because we might
// end up on another monitor, but in 2 we do want to update hmonStuck because
// our hmon is invalid
// The way we handle this is to call GetMonitorInfo on our old HMONITOR
// and see if it's still valid, if not, we update it by calling _SetStuckMonitor
// all these code is in _ScreenSizeChange;
_ScreenSizeChange(hwnd);
// Force the Start Pane to rebuild because a change in color depth
// causes themes to run around destroying fonts (ruining the OOBE
// text) and we need to reload our bitmaps for the new color depth
// anyway.
::PostMessage(_hwnd, SBM_REBUILDMENU, 0, 0);
break; // Don't go to default wnd proc for this one...
case WM_INPUTLANGCHANGEREQUEST: return(LRESULT)0L;
case WM_GETMINMAXINFO: ((MINMAXINFO *)lParam)->ptMinTrackSize.x = g_cxFrame; ((MINMAXINFO *)lParam)->ptMinTrackSize.y = g_cyFrame; break;
case WM_WININICHANGE: if (lParam && (0 == lstrcmpi((LPCTSTR)lParam, TEXT("SaveTaskbar")))) { _SaveTrayAndDesktop(); } else { BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, NULL); _PropagateMessage(hwnd, uMsg, wParam, lParam); _OnWinIniChange(hwnd, wParam, lParam); }
if (lParam) TraceMsg(TF_TRAY, "Tray Got: lParam=%s", (LPCSTR)lParam);
break;
case WM_TIMECHANGE: _PropagateMessage(hwnd, uMsg, wParam, lParam); break;
case WM_SYSCOLORCHANGE: _OnNewSystemSizes(); BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, NULL); _PropagateMessage(hwnd, uMsg, wParam, lParam); break;
case WM_SETCURSOR: if (_iWaitCount) { SetCursor(LoadCursor(NULL, IDC_APPSTARTING)); return TRUE; } else goto DoDefault;
case WM_SETFOCUS: IUnknown_UIActivateIO(_ptbs, TRUE, NULL); break;
case WM_SYSCHAR: if (wParam == TEXT(' ')) { HMENU hmenu; int idCmd;
SHSetWindowBits(hwnd, GWL_STYLE, WS_SYSMENU, WS_SYSMENU); hmenu = GetSystemMenu(hwnd, FALSE); if (hmenu) { EnableMenuItem(hmenu, SC_RESTORE, MFS_GRAYED | MF_BYCOMMAND); EnableMenuItem(hmenu, SC_MAXIMIZE, MFS_GRAYED | MF_BYCOMMAND); EnableMenuItem(hmenu, SC_MINIMIZE, MFS_GRAYED | MF_BYCOMMAND); EnableMenuItem(hmenu, SC_MOVE, (_fCanSizeMove ? MFS_ENABLED : MFS_GRAYED) | MF_BYCOMMAND); EnableMenuItem(hmenu, SC_SIZE, (_fCanSizeMove ? MFS_ENABLED : MFS_GRAYED) | MF_BYCOMMAND);
idCmd = _TrackMenu(hmenu); if (idCmd) SendMessage(_hwnd, WM_SYSCOMMAND, idCmd, 0L); } SHSetWindowBits(hwnd, GWL_STYLE, WS_SYSMENU, 0L); } break;
case WM_SYSCOMMAND:
// if we are sizing, make the full screen accessible
switch (wParam & 0xFFF0) { case SC_CLOSE: _DoExitWindows(v_hwndDesktop); break;
default: goto DoDefault; } break;
case TM_DESKTOPSTATE: _OnDesktopState(lParam); break;
case TM_RAISEDESKTOP: _RaiseDesktop((BOOL)wParam, FALSE); break;
#ifdef DEBUG
case TM_NEXTCTL: #endif
case TM_UIACTIVATEIO: case TM_ONFOCUSCHANGEIS: _OnFocusMsg(uMsg, wParam, lParam); break;
case TM_MARSHALBS: // wParam=IID lRes=pstm
return BandSite_OnMarshallBS(wParam, lParam);
case TM_SETTIMER: case TM_KILLTIMER: return _OnTimerService(uMsg, wParam, lParam); break;
case TM_FACTORY: return _OnFactoryMessage(wParam, lParam);
case TM_ACTASTASKSW: _ActAsSwitcher(); break;
case TM_RELAYPOSCHANGED: _AppBarNotifyAll((HMONITOR)lParam, ABN_POSCHANGED, (HWND)wParam, 0); break;
case TM_BRINGTOTOP: SetWindowZorder((HWND)wParam, HWND_TOP); break;
case TM_WARNNOAUTOHIDE: DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.twp collision UI request"));
//
// this may look a little funny but what we do is post this message all
// over the place and ignore it when we think it is a bad time to put
// up a message (like the middle of a fulldrag...)
//
// wParam tells us if we need to try to clear the state
// the lowword of _SetAutoHideState's return tells if anything changed
//
if ((!_fSysSizing || !g_fDragFullWindows) && (!wParam || LOWORD(_SetAutoHideState(FALSE)))) { ShellMessageBox(hinstCabinet, hwnd, MAKEINTRESOURCE(IDS_ALREADYAUTOHIDEBAR), MAKEINTRESOURCE(IDS_TASKBAR), MB_OK | MB_ICONINFORMATION); } else { DebugMsg(DM_TRAYDOCK, TEXT("TRAYDOCK.twp blowing off extraneous collision UI request")); } break;
case TM_PRIVATECOMMAND: _HandlePrivateCommand(lParam); break;
case TM_HANDLEDELAYBOOTSTUFF: _HandleDelayBootStuff(); break;
case TM_SHELLSERVICEOBJECTS: _ssomgr.LoadRegObjects(); break;
case TM_CHANGENOTIFY: _HandleChangeNotify(wParam, lParam); break;
case TM_GETHMONITOR: *((HMONITOR *)lParam) = _hmonStuck; break;
case TM_DOTRAYPROPERTIES: DoProperties(TPF_TASKBARPAGE); break;
case TM_STARTUPAPPSLAUNCHED: PostMessage(_hwndNotify, TNM_STARTUPAPPSLAUNCHED, 0, 0); break;
case TM_LANGUAGEBAND: return _ToggleLanguageBand(lParam); case WM_NCRBUTTONUP: uMsg = WM_CONTEXTMENU; wParam = (WPARAM)_hwndTasks; goto L_WM_CONTEXTMENU;
case WM_CONTEXTMENU: L_WM_CONTEXTMENU:
if (!SHRestricted(REST_NOTRAYCONTEXTMENU)) { if (((HWND)wParam) == _hwndStart) { // Don't display of the Start Menu is up.
if (SendMessage(_hwndStart, BM_GETSTATE, 0, 0) & BST_PUSHED) break;
_fFromStart = TRUE;
StartMenuContextMenu(_hwnd, (DWORD)lParam);
_fFromStart = FALSE; } else if (IsPosInHwnd(lParam, _hwndNotify) || SHIsChildOrSelf(_hwndNotify, GetFocus()) == S_OK) { // if click was inthe clock, include
// the time
_ContextMenu((DWORD)lParam, TRUE); } else { BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, &lres); } } break;
case WM_INITMENUPOPUP: case WM_MEASUREITEM: case WM_DRAWITEM: case WM_MENUCHAR: // Don't call bandsite message handler when code path started via the start button context menu
if (!_fFromStart) { BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, &lres); } break;
case TM_DOEXITWINDOWS: _DoExitWindows(v_hwndDesktop); break;
case TM_HANDLESTARTUPFAILED: _OnHandleStartupFailed(); break;
case WM_HOTKEY: if (wParam < GHID_FIRST) { _HandleHotKey((WORD)wParam); } else { _HandleGlobalHotkey(wParam); } break;
case WM_COMMAND: if (!BandSite_HandleMessage(_ptbs, hwnd, uMsg, wParam, lParam, &lres)) _Command(GET_WM_COMMAND_ID(wParam, lParam)); break;
case SBM_CANCELMENU: ClosePopupMenus(); break;
case SBM_REBUILDMENU: _BuildStartMenu(); break;
case WM_WINDOWPOSCHANGED: _AppBarActivationChange2(hwnd, _uStuckPlace); SendMessage(_hwndNotify, TNM_TRAYPOSCHANGED, 0, 0); goto DoDefault;
case WM_LBUTTONDOWN: case WM_RBUTTONDOWN: case WM_MBUTTONDOWN: if (_hwndStartBalloon) { RECT rc; POINT pt = {GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)}; GetWindowRect(_hwndStartBalloon, &rc); MapWindowRect(HWND_DESKTOP, _hwnd, &rc);
if (PtInRect(&rc, pt)) { ShowWindow(_hwndStartBalloon, SW_HIDE); _DontShowTheStartButtonBalloonAnyMore(); _DestroyStartButtonBalloon(); } } break;
case TM_SETPUMPHOOK: ATOMICRELEASE(_pmbTasks); ATOMICRELEASE(_pmpTasks); if (wParam && lParam) { _pmbTasks = (IMenuBand*)wParam; _pmbTasks->AddRef(); _pmpTasks = (IMenuPopup*)lParam; _pmpTasks->AddRef(); } break; case WM_ACTIVATE: _AppBarActivationChange2(hwnd, _uStuckPlace); if (wParam != WA_INACTIVE) { Unhide(); } else { // When tray is deactivated, remove our keyboard cues:
//
SendMessage(hwnd, WM_CHANGEUISTATE, MAKEWPARAM(UIS_SET, UISF_HIDEFOCUS | UISF_HIDEACCEL), 0);
IUnknown_UIActivateIO(_ptbs, FALSE, NULL); } //
// Tray activation is a good time to do a reality check
// (make sure "always-on-top" agrees with actual window
// position, make sure there are no ghost buttons, etc).
//
RealityCheck(); goto L_default;
case WM_WTSSESSION_CHANGE: { lres = _OnSessionChange(wParam, lParam); break; }
case WM_THEMECHANGED: { if (_hTheme) { CloseThemeData(_hTheme); _hTheme = NULL; } if (wParam) { _hTheme = OpenThemeData(_hwnd, c_wzTaskbarTheme); _fShowSizingBarAlways = (_uAutoHide & AH_ON) ? TRUE : FALSE; if (_hTheme) { GetThemeBool(_hTheme, 0, 0, TMT_ALWAYSSHOWSIZINGBAR, &_fShowSizingBarAlways); } _UpdateVertical(_uStuckPlace, TRUE); // Force Refresh of frame
SetWindowPos(_hwnd, NULL, 0, 0, 0, 0, SWP_NOZORDER | SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOMOVE | SWP_NOACTIVATE); }
// Force the start button to recalc its size
_sizeStart.cx = 0; _StartButtonReset(); InvalidateRect(_hwnd, NULL, TRUE);
// Force the Start Pane to rebuild with new theme
::PostMessage(_hwnd, SBM_REBUILDMENU, 0, 0);
SetWindowStyle(_hwnd, WS_BORDER | WS_THICKFRAME, !_hTheme); } break;
case TM_WORKSTATIONLOCKED: { // Desktop locked status changed...
BOOL fIsDesktopLocked = (BOOL) wParam; if (_fIsDesktopLocked != fIsDesktopLocked) { _fIsDesktopLocked = fIsDesktopLocked; _fIsLogoff = FALSE; _RecomputeAllWorkareas(); PostMessage(_hwndNotify, TNM_WORKSTATIONLOCKED, wParam, 0); } } break;
case TM_SHOWTRAYBALLOON: PostMessage(_hwndNotify, TNM_SHOWTRAYBALLOON, wParam, 0); break;
case TM_STARTMENUDISMISSED: // 107561 - call CoFreeUnusedLibraries() peridically to free up dlls - ZekeL - 4-MAY-2001
// specifically to support MSONSEXT (webfolders) being used in RecentDocs
// after a file has been opened via webfolders. we get the icon via
// their namespace but then COM holds on to the DLL for a while (forever?)
// calling CoFreeUnusedLibraries() does the trick
SetTimer(_hwnd, IDT_COFREEUNUSED, 3 * 60 * 1000, NULL); break;
case MM_MIXM_CONTROL_CHANGE: Mixer_ControlChange(wParam, lParam); break;
default: L_default: if (uMsg == GetDDEExecMsg()) { ASSERT(lParam && 0 == ((LPNMHDR)lParam)->idFrom); DDEHandleViewFolderNotify(NULL, _hwnd, (LPNMVIEWFOLDER)lParam); LocalFree((LPNMVIEWFOLDER)lParam); return TRUE; } else if (uMsg == _uStartButtonBalloonTip) { _ShowStartButtonToolTip(); } else if (uMsg == _uLogoffUser) { // Log off the current user (message from U&P control panel)
ExitWindowsEx(EWX_LOGOFF, 0); } else if (uMsg == _uMsgEnableUserTrackedBalloonTips) { PostMessage(_hwndNotify, TNM_ENABLEUSERTRACKINGINFOTIPS, wParam, 0); } else if (uMsg == _uWinMM_DeviceChange) { Mixer_MMDeviceChange(); }
DoDefault: return DefWindowProc(hwnd, uMsg, wParam, lParam); } return lres; }
void CTray::_DoExitWindows(HWND hwnd) { static BOOL s_fShellShutdown = FALSE;
if (!s_fShellShutdown) {
if (_Restricted(hwnd, REST_NOCLOSE)) return;
{ UEMFireEvent(&UEMIID_SHELL, UEME_CTLSESSION, UEMF_XEVENT, FALSE, -1); // really #ifdef DEBUG, but want for testing
// however can't do unconditionally due to perf
if (ERROR_SUCCESS == SHGetValue(HKEY_CURRENT_USER, REGSTR_EXPLORER_ADVANCED, TEXT("StartMenuForceRefresh"), NULL, NULL, NULL) || GetAsyncKeyState(VK_SHIFT) < 0) { _RefreshStartMenu(); } }
_SaveTrayAndDesktop();
_uModalMode = MM_SHUTDOWN; ExitWindowsDialog(hwnd);
// NB User can have problems if the focus is forcebly changed to the desktop while
// shutting down since it tries to serialize the whole process by making windows sys-modal.
// If we hit this code at just the wrong moment (ie just after the sharing dialog appears)
// the desktop will become sys-modal so you can't switch back to the sharing dialog
// and you won't be able to shutdown.
// SetForegroundWindow(hwnd);
// SetFocus(hwnd);
_uModalMode = 0; if ((GetKeyState(VK_SHIFT) < 0) && (GetKeyState(VK_CONTROL) < 0) && (GetKeyState(VK_MENU) < 0)) { // User cancelled...
// The shift key means exit the tray...
// ??? - Used to destroy all cabinets...
// PostQuitMessage(0);
g_fFakeShutdown = TRUE; // Don't blow away session state; the session will survive
TraceMsg(TF_TRAY, "c.dew: Posting quit message for tid=%#08x hwndDesk=%x(IsWnd=%d) hwndTray=%x(IsWnd=%d)", GetCurrentThreadId(), v_hwndDesktop,IsWindow(v_hwndDesktop), _hwnd,IsWindow(_hwnd)); // 1 means close all the shell windows too
PostMessage(v_hwndDesktop, WM_QUIT, 0, 1); PostMessage(_hwnd, WM_QUIT, 0, 0);
s_fShellShutdown = TRUE; } } }
void CTray::_SaveTray(void) { if (SHRestricted(REST_NOSAVESET)) return;
if (SHRestricted(REST_CLEARRECENTDOCSONEXIT)) ClearRecentDocumentsAndMRUStuff(FALSE);
//
// Don't persist tray stuff if in safe mode. We want this
// to be a temporary mode where the UI settings don't stick.
//
if (GetSystemMetrics(SM_CLEANBOOT) == 0) { _SaveTrayStuff(); } }
DWORD WINAPI CTray::PropertiesThreadProc(void* pv) { return c_tray._PropertiesThreadProc(PtrToUlong(pv)); }
DWORD CTray::_PropertiesThreadProc(DWORD dwFlags) { HWND hwnd; RECT rc; DWORD dwExStyle = WS_EX_TOOLWINDOW;
GetWindowRect(_hwndStart, &rc); dwExStyle |= IS_BIDI_LOCALIZED_SYSTEM() ? dwExStyleRTLMirrorWnd : 0L;
_hwndProp = hwnd = CreateWindowEx(dwExStyle, TEXT("static"), NULL, 0 , rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, NULL, NULL, hinstCabinet, NULL);
#define IDI_STTASKBR 40 // stolen from shell32\ids.h
if (_hwndProp) { // Get the Alt+Tab icon right
HICON hicoStub = LoadIcon(GetModuleHandle(TEXT("SHELL32")), MAKEINTRESOURCE(IDI_STTASKBR)); SendMessage(_hwndProp, WM_SETICON, ICON_BIG, (LPARAM)hicoStub);
// SwitchToThisWindow(hwnd, TRUE);
// SetForegroundWindow(hwnd);
DoTaskBarProperties(hwnd, dwFlags);
_hwndProp = NULL; DestroyWindow(hwnd); if (hicoStub) DestroyIcon(hicoStub); }
return TRUE; }
#define RUNWAITSECS 5
void CTray::DoProperties(DWORD dwFlags) { if (!_Restricted(_hwnd, REST_NOSETTASKBAR)) { int i = RUNWAITSECS; while (_hwndProp == ((HWND)-1) &&i--) { // we're in the process of coming up. wait
Sleep(1000); }
// failed! blow it off.
if (_hwndProp == (HWND)-1) { _hwndProp = NULL; }
if (_hwndProp) { // there's a window out there... activate it
SwitchToThisWindow(GetLastActivePopup(_hwndProp), TRUE); } else { _hwndProp = (HWND)-1; if (!SHCreateThread(PropertiesThreadProc, IntToPtr(dwFlags), CTF_COINIT, NULL)) { _hwndProp = NULL; } } } }
BOOL CTray::TileEnumProc(HWND hwnd, LPARAM lParam) { CTray* ptray = (CTray*)lParam;
if (IsWindowVisible(hwnd) && !IsIconic(hwnd) && ((GetWindowLong(hwnd, GWL_STYLE) & WS_CAPTION) == WS_CAPTION) && (hwnd != ptray->_hwnd) && hwnd != v_hwndDesktop) { return FALSE; // we *can* tile this guy
} return TRUE; // we *cannot* tile this guy
}
HMENU CTray::BuildContextMenu(BOOL fIncludeTime) { HMENU hmContext = LoadMenuPopup(MAKEINTRESOURCE(MENU_TRAYCONTEXT)); if (!hmContext) return NULL;
if (fIncludeTime) { if (_trayNotify.GetIsNoTrayItemsDisplayPolicyEnabled()) { // We know the position of IDM_NOTIFYCUST from the menu resource...
DeleteMenu(hmContext, 1, MF_BYPOSITION); } else { UINT uEnable = MF_BYCOMMAND; if (_trayNotify.GetIsNoAutoTrayPolicyEnabled() || !_trayNotify.GetIsAutoTrayEnabledByUser()) { uEnable |= MFS_DISABLED; } else { uEnable |= MFS_ENABLED; } EnableMenuItem(hmContext, IDM_NOTIFYCUST, uEnable); } } else { INSTRUMENT_STATECHANGE(SHCNFI_STATE_TRAY_CONTEXT); for (int i = 2; i >= 0; i--) // separator, IDM_SETTIME, IDM_NOTIFYCUST,
{ DeleteMenu(hmContext, i, MF_BYPOSITION); } }
CheckMenuItem(hmContext, IDM_LOCKTASKBAR, MF_BYCOMMAND | (_fCanSizeMove ? MF_UNCHECKED : MF_CHECKED));
// Don't let users accidentally check lock the taskbar when the taskbar is zero height
RECT rc; GetClientRect(_hwnd, &rc); EnableMenuItem(hmContext, IDM_LOCKTASKBAR, MF_BYCOMMAND | ((_IsSizeMoveRestricted() || (RECTHEIGHT(rc) == 0)) ? MFS_DISABLED : MFS_ENABLED));
if (!_fUndoEnabled || !_pPositions) { DeleteMenu(hmContext, IDM_UNDO, MF_BYCOMMAND); } else { TCHAR szTemplate[30]; TCHAR szCommand[30]; TCHAR szMenu[64]; LoadString(hinstCabinet, IDS_UNDOTEMPLATE, szTemplate, ARRAYSIZE(szTemplate)); LoadString(hinstCabinet, _pPositions->idRes, szCommand, ARRAYSIZE(szCommand)); wsprintf(szMenu, szTemplate, szCommand); ModifyMenu(hmContext, IDM_UNDO, MF_BYCOMMAND | MF_STRING, IDM_UNDO, szMenu); }
if (g_fDesktopRaised) { TCHAR szHideDesktop[64]; LoadString(hinstCabinet, IDS_HIDEDESKTOP, szHideDesktop, ARRAYSIZE(szHideDesktop)); ModifyMenu(hmContext, IDM_TOGGLEDESKTOP, MF_BYCOMMAND | MF_STRING, IDM_TOGGLEDESKTOP, szHideDesktop); } if (!_CanTileAnyWindows()) { EnableMenuItem(hmContext, IDM_CASCADE, MFS_GRAYED | MF_BYCOMMAND); EnableMenuItem(hmContext, IDM_HORIZTILE, MFS_GRAYED | MF_BYCOMMAND); EnableMenuItem(hmContext, IDM_VERTTILE, MFS_GRAYED | MF_BYCOMMAND); }
HKEY hKeyPolicy; if (RegOpenKeyEx(HKEY_CURRENT_USER, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Policies\\System"), 0, KEY_READ, &hKeyPolicy) == ERROR_SUCCESS) { DWORD dwType, dwData = 0, dwSize = sizeof(dwData); RegQueryValueEx(hKeyPolicy, TEXT("DisableTaskMgr"), NULL, &dwType, (LPBYTE) &dwData, &dwSize); RegCloseKey(hKeyPolicy); if (dwData) EnableMenuItem(hmContext, IDM_SHOWTASKMAN, MFS_GRAYED | MF_BYCOMMAND); }
return hmContext; }
void CTray::ContextMenuInvoke(int idCmd) { if (idCmd) { if (idCmd < IDM_TRAYCONTEXTFIRST) { BandSite_HandleMenuCommand(_ptbs, idCmd); } else { _Command(idCmd); } } }
//
// CTray::AsyncSaveSettings
//
// We need to save our tray settings, but there may be a bunch
// of these calls coming, (at startup time or when dragging
// items in the task bar) so gather them up into one save that
// will happen in at least 2 seconds.
//
void CTray::AsyncSaveSettings() { if (!_fHandledDelayBootStuff) // no point in saving if we're not done booting
return;
KillTimer(_hwnd, IDT_SAVESETTINGS); SetTimer(_hwnd, IDT_SAVESETTINGS, 2000, NULL); }
void CTray::_ContextMenu(DWORD dwPos, BOOL fIncludeTime) { POINT pt = {LOWORD(dwPos), HIWORD(dwPos)};
SwitchToThisWindow(_hwnd, TRUE); SetForegroundWindow(_hwnd); SendMessage(_hwndTrayTips, TTM_ACTIVATE, FALSE, 0L);
if (dwPos != (DWORD)-1 && IsChildOrHWND(_hwndRebar, WindowFromPoint(pt))) { // if the context menu came from below us, reflect down
BandSite_HandleMessage(_ptbs, _hwnd, WM_CONTEXTMENU, 0, dwPos, NULL);
} else { HMENU hmenu;
if (dwPos == (DWORD)-1) { HWND hwnd = GetFocus(); pt.x = pt.y = 0; ClientToScreen(hwnd, &pt); dwPos = MAKELONG(pt.x, pt.y); }
hmenu = BuildContextMenu(fIncludeTime); if (hmenu) { int idCmd;
BandSite_AddMenus(_ptbs, hmenu, 0, 0, IDM_TRAYCONTEXTFIRST);
idCmd = TrackPopupMenu(hmenu, TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTALIGN, GET_X_LPARAM(dwPos), GET_Y_LPARAM(dwPos), 0, _hwnd, NULL);
DestroyMenu(hmenu);
ContextMenuInvoke(idCmd); } }
SendMessage(_hwndTrayTips, TTM_ACTIVATE, TRUE, 0L); }
void _RunFileDlg(HWND hwnd, UINT idIcon, LPCITEMIDLIST pidlWorkingDir, UINT idTitle, UINT idPrompt, DWORD dwFlags) { HICON hIcon; LPCTSTR lpszTitle; LPCTSTR lpszPrompt; TCHAR szTitle[256]; TCHAR szPrompt[256]; TCHAR szWorkingDir[MAX_PATH];
dwFlags |= RFD_USEFULLPATHDIR; szWorkingDir[0] = 0;
hIcon = idIcon ? LoadIcon(hinstCabinet, MAKEINTRESOURCE(idIcon)) : NULL;
if (!pidlWorkingDir || !SHGetPathFromIDList(pidlWorkingDir, szWorkingDir)) { // This is either the Tray, or some non-file system folder, so
// we will "suggest" the Desktop as a working dir, but if the
// user types a full path, we will use that instead. This is
// what WIN31 Progman did (except they started in the Windows
// dir instead of the Desktop).
goto UseDesktop; }
// if it's a removable dir, make sure it's still there
if (szWorkingDir[0]) { int idDrive = PathGetDriveNumber(szWorkingDir); if ((idDrive != -1)) { UINT dtype = DriveType(idDrive); if (((dtype == DRIVE_REMOVABLE) || (dtype == DRIVE_CDROM)) && !PathFileExists(szWorkingDir)) { goto UseDesktop; } } }
//
// Check if this is a directory. Notice that it could be a in-place
// navigated document.
//
if (PathIsDirectory(szWorkingDir)) { goto UseWorkingDir; }
UseDesktop: SHGetSpecialFolderPath(hwnd, szWorkingDir, CSIDL_DESKTOPDIRECTORY, FALSE);
UseWorkingDir:
if (idTitle) { LoadString(hinstCabinet, idTitle, szTitle, ARRAYSIZE(szTitle)); lpszTitle = szTitle; } else lpszTitle = NULL; if (idPrompt) { LoadString(hinstCabinet, idPrompt, szPrompt, ARRAYSIZE(szPrompt)); lpszPrompt = szPrompt; } else lpszPrompt = NULL;
RunFileDlg(hwnd, hIcon, szWorkingDir, lpszTitle, lpszPrompt, dwFlags); }
BOOL CTray::SavePosEnumProc(HWND hwnd, LPARAM lParam) { // dont need to entercritical here since we are only ever
// called from SaveWindowPositions, which as already entered the critical secion
// for _pPositions
ASSERTCRITICAL;
CTray* ptray = (CTray*)lParam;
ASSERT(ptray->_pPositions);
if (IsWindowVisible(hwnd) && (hwnd != ptray->_hwnd) && (hwnd != v_hwndDesktop)) { HWNDANDPLACEMENT hap;
hap.wp.length = sizeof(WINDOWPLACEMENT); GetWindowPlacement(hwnd, &hap.wp);
if (hap.wp.showCmd != SW_SHOWMINIMIZED) { hap.hwnd = hwnd; hap.fRestore = TRUE;
DSA_AppendItem(ptray->_pPositions->hdsaWP, &hap); } } return TRUE; }
void CTray::SaveWindowPositions(UINT idRes) { ENTERCRITICAL; if (_pPositions) { if (_pPositions->hdsaWP) DSA_DeleteAllItems(_pPositions->hdsaWP); } else { _pPositions = (LPWINDOWPOSITIONS)LocalAlloc(LPTR, sizeof(WINDOWPOSITIONS)); if (_pPositions) { _pPositions->hdsaWP = DSA_Create(sizeof(HWNDANDPLACEMENT), 4); } }
if (_pPositions) { _pPositions->idRes = idRes;
// CheckWindowPositions tested for these...
ASSERT(idRes == IDS_MINIMIZEALL || idRes == IDS_CASCADE || idRes == IDS_TILE); EnumWindows(SavePosEnumProc, (LPARAM)this); } LEAVECRITICAL; }
typedef struct { LPWINDOWPOSITIONS pPositions; HWND hwndDesktop; HWND hwndTray; BOOL fPostLowerDesktop; } RESTOREWNDDATA, *PRESTOREWNDDATA;
DWORD WINAPI RestoreWndPosThreadProc(void* pv) { PRESTOREWNDDATA pWndData = (PRESTOREWNDDATA)pv; if (pWndData && pWndData->pPositions) { LPHWNDANDPLACEMENT phap; LONG iAnimate; ANIMATIONINFO ami;
ami.cbSize = sizeof(ANIMATIONINFO); SystemParametersInfo(SPI_GETANIMATION, sizeof(ami), &ami, FALSE); iAnimate = ami.iMinAnimate; ami.iMinAnimate = FALSE; SystemParametersInfo(SPI_SETANIMATION, sizeof(ami), &ami, FALSE);
if (pWndData->pPositions->hdsaWP) { for (int i = DSA_GetItemCount(pWndData->pPositions->hdsaWP) - 1 ; i >= 0; i--) { phap = (LPHWNDANDPLACEMENT)DSA_GetItemPtr(pWndData->pPositions->hdsaWP, i); if (IsWindow(phap->hwnd)) {
#ifndef WPF_ASYNCWINDOWPLACEMENT
#define WPF_ASYNCWINDOWPLACEMENT 0x0004
#endif
// pass this async.
if (!IsHungAppWindow(phap->hwnd)) { phap->wp.length = sizeof(WINDOWPLACEMENT); phap->wp.flags |= WPF_ASYNCWINDOWPLACEMENT; if (phap->fRestore) { // only restore those guys we've actually munged.
SetWindowPlacement(phap->hwnd, &phap->wp); } } } } }
ami.iMinAnimate = iAnimate; SystemParametersInfo(SPI_SETANIMATION, sizeof(ami), &ami, FALSE);
_DestroySavedWindowPositions(pWndData->pPositions);
if (pWndData->fPostLowerDesktop) { PostMessage(pWndData->hwndDesktop, DTM_RAISE, (WPARAM)pWndData->hwndTray, DTRF_LOWER); }
delete pWndData; }
return 1; }
BOOL CTray::_RestoreWindowPositions(BOOL fPostLowerDesktop) { BOOL fRet = FALSE;
ENTERCRITICAL; if (_pPositions) { PRESTOREWNDDATA pWndData = new RESTOREWNDDATA; if (pWndData) { pWndData->pPositions = _pPositions; pWndData->fPostLowerDesktop = fPostLowerDesktop; pWndData->hwndDesktop = v_hwndDesktop; pWndData->hwndTray = _hwnd;
if (SHCreateThread(RestoreWndPosThreadProc, pWndData, 0, NULL)) { fRet = TRUE; _pPositions = NULL; } else { delete pWndData; } } } LEAVECRITICAL;
return fRet; }
void _DestroySavedWindowPositions(LPWINDOWPOSITIONS pPositions) { ENTERCRITICAL; if (pPositions) { // free the global struct
DSA_Destroy(pPositions->hdsaWP); LocalFree(pPositions); } LEAVECRITICAL; }
void CTray::HandleWindowDestroyed(HWND hwnd) { // enter critical section so we dont corrupt the hdsaWP
ENTERCRITICAL; if (_pPositions) { int i = DSA_GetItemCount(_pPositions->hdsaWP) - 1; for (; i >= 0; i--) { LPHWNDANDPLACEMENT phap = (LPHWNDANDPLACEMENT)DSA_GetItemPtr(_pPositions->hdsaWP, i); if (phap->hwnd == hwnd || !IsWindow(phap->hwnd)) { DSA_DeleteItem(_pPositions->hdsaWP, i); } }
if (!DSA_GetItemCount(_pPositions->hdsaWP)) { _DestroySavedWindowPositions(_pPositions); _pPositions = NULL; } } LEAVECRITICAL; }
// Allow us to bump the activation of the run dlg hidden window.
// Certain apps (Norton Desktop setup) use the active window at RunDlg time
// as the parent for their dialogs. If that window disappears then they fault.
// We don't want the tray to get the activation coz it will cause it to appeare
// if you're in auto-hide mode.
LRESULT WINAPI RunDlgStaticSubclassWndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { switch (uMsg) { case WM_ACTIVATE: if (wParam == WA_ACTIVE) { // Bump the activation to the desktop.
if (v_hwndDesktop) { SetForegroundWindow(v_hwndDesktop); return 0; } } break;
case WM_NOTIFY: // relay it to the tray
return SendMessage(v_hwndTray, uMsg, wParam, lParam); }
return DefWindowProc(hwnd, uMsg, wParam, lParam); }
DWORD WINAPI CTray::RunDlgThreadProc(void *pv) { return c_tray._RunDlgThreadProc((HANDLE)pv); }
BOOL _IsBrowserWindow(HWND hwnd) { static const TCHAR* c_szClasses[] = { TEXT("ExploreWClass"), TEXT("CabinetWClass"), TEXT("IEFrame"), };
TCHAR szClass[32]; GetClassName(hwnd, szClass, ARRAYSIZE(szClass));
for (int i = 0; i < ARRAYSIZE(c_szClasses); i++) { if (lstrcmpi(szClass, c_szClasses[i]) == 0) { return TRUE; } } return FALSE; }
DWORD CTray::_RunDlgThreadProc(HANDLE hdata) { RECT rc, rcTemp; HRESULT hrInit = SHCoInitialize();
// 99/04/12 #316424 vtan: Get the rectangle for the "Start" button.
// If this is off the screen the tray is probably in auto hide mode.
// In this case offset the rectangle into the monitor where it should
// belong. This may be up, down, left or right depending on the
// position of the tray.
// First thing to do is to establish the dimensions of the monitor on
// which the tray resides. If no monitor can be found then use the
// primary monitor.
MONITORINFO monitorInfo; monitorInfo.cbSize = sizeof(monitorInfo); if (GetMonitorInfo(_hmonStuck, &monitorInfo) == 0) { TBOOL(SystemParametersInfo(SPI_GETWORKAREA, 0, &monitorInfo.rcMonitor, 0)); }
// Get the co-ordinates of the "Start" button.
GetWindowRect(_hwndStart, &rc);
// Look for an intersection in the monitor.
if (IntersectRect(&rcTemp, &rc, &monitorInfo.rcMonitor) == 0) { LONG lDeltaX, lDeltaY;
// Does not exist in the monitor. Move the co-ordinates by the
// width or height of the tray so that it does.
// This bizarre arithmetic is used because _ComputeHiddenRect()
// takes into account the frame and that right/bottom of RECT
// is exclusive in GDI.
lDeltaX = _sStuckWidths.cx - g_cxFrame; lDeltaY = _sStuckWidths.cy - g_cyFrame; if (rc.left < monitorInfo.rcMonitor.left) { --lDeltaX; lDeltaY = 0; } else if (rc.top < monitorInfo.rcMonitor.top) { lDeltaX = 0; --lDeltaY; } else if (rc.right > monitorInfo.rcMonitor.right) { lDeltaX = -lDeltaX; lDeltaY = 0; } else if (rc.bottom > monitorInfo.rcMonitor.bottom) { lDeltaX = 0; lDeltaY = -lDeltaY; } TBOOL(OffsetRect(&rc, lDeltaX, lDeltaY)); }
HWND hwnd = CreateWindowEx(WS_EX_TOOLWINDOW, TEXT("static"), NULL, 0 , rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, NULL, NULL, hinstCabinet, NULL); if (hwnd) { BOOL fSimple = FALSE; HANDLE hMemWorkDir = NULL; LPITEMIDLIST pidlWorkingDir = NULL;
// Subclass it.
SubclassWindow(hwnd, RunDlgStaticSubclassWndProc);
if (hdata) SetProp(hwnd, TEXT("WaitingThreadID"), hdata);
if (!SHRestricted(REST_STARTRUNNOHOMEPATH)) { // On NT, we like to start apps in the HOMEPATH directory. This
// should be the current directory for the current process.
TCHAR szDir[MAX_PATH]; TCHAR szPath[MAX_PATH];
GetEnvironmentVariable(TEXT("HOMEDRIVE"), szDir, ARRAYSIZE(szDir)); GetEnvironmentVariable(TEXT("HOMEPATH"), szPath, ARRAYSIZE(szPath));
if (PathAppend(szDir, szPath) && PathIsDirectory(szDir)) { pidlWorkingDir = SHSimpleIDListFromPath(szDir); if (pidlWorkingDir) { // free it the "simple" way...
fSimple = TRUE; } } } if (!pidlWorkingDir) { // If the last active window was a folder/explorer window with the
// desktop as root, use its as the current dir
if (_hwndLastActive) { ENTERCRITICAL; if (_hwndLastActive && !IsMinimized(_hwndLastActive) && _IsBrowserWindow(_hwndLastActive)) { SendMessageTimeout(_hwndLastActive, CWM_CLONEPIDL, GetCurrentProcessId(), 0, SMTO_ABORTIFHUNG | SMTO_BLOCK, 500, (DWORD_PTR*)&hMemWorkDir); pidlWorkingDir = (LPITEMIDLIST)SHLockShared(hMemWorkDir, GetCurrentProcessId()); } LEAVECRITICAL; } }
_RunFileDlg(hwnd, 0, pidlWorkingDir, 0, 0, 0); if (pidlWorkingDir) { if (fSimple) { ILFree(pidlWorkingDir); } else { SHUnlockShared(pidlWorkingDir); } }
if (hMemWorkDir) { ASSERT(fSimple == FALSE); SHFreeShared(hMemWorkDir, GetCurrentProcessId()); }
if (hdata) { RemoveProp(hwnd, TEXT("WaitingThreadID")); }
DestroyWindow(hwnd); } SHCoUninitialize(hrInit); return TRUE; }
void CTray::_RunDlg() { HANDLE hEvent; void *pvThreadParam;
if (!_Restricted(_hwnd, REST_NORUN)) { TCHAR szRunDlgTitle[MAX_PATH]; HWND hwndOldRun; LoadString(hinstCabinet, IDS_RUNDLGTITLE, szRunDlgTitle, ARRAYSIZE(szRunDlgTitle));
// See if there is already a run dialog up, and if so, try to activate it
hwndOldRun = FindWindow(WC_DIALOG, szRunDlgTitle); if (hwndOldRun) { DWORD dwPID;
GetWindowThreadProcessId(hwndOldRun, &dwPID); if (dwPID == GetCurrentProcessId()) { if (IsWindowVisible(hwndOldRun)) { SetForegroundWindow(hwndOldRun); return; } } }
// Create an event so we can wait for the run dlg to appear before
// continue - this allows it to capture any type-ahead.
hEvent = CreateEvent(NULL, TRUE, FALSE, TEXT("MSShellRunDlgReady")); if (hEvent) pvThreadParam = IntToPtr(GetCurrentThreadId()); else pvThreadParam = NULL;
if (SHQueueUserWorkItem(RunDlgThreadProc, pvThreadParam, 0, 0, NULL, NULL, TPS_LONGEXECTIME | TPS_DEMANDTHREAD)) { if (hEvent) { SHProcessMessagesUntilEvent(NULL, hEvent, 10 * 1000); DebugMsg(DM_TRACE, TEXT("c.t_rd: Done waiting.")); } }
if (hEvent) CloseHandle(hEvent); } }
void CTray::_ExploreCommonStartMenu(BOOL bExplore) { TCHAR szPath[MAX_PATH]; TCHAR szCmdLine[MAX_PATH + 50]; //
// Get the common start menu path.
//
// we want to force the directory to exist, but not on W95 machines
if (!SHGetSpecialFolderPath(NULL, szPath, CSIDL_COMMON_STARTMENU, FALSE)) { return; } //
// If we are starting in explorer view, then the command line
// has a "/e, " before the quoted diretory.
//
if (bExplore) { lstrcpy(szCmdLine, TEXT("explorer.exe /e, \"")); } else { lstrcpy(szCmdLine, TEXT("explorer.exe \"")); } lstrcat(szCmdLine, szPath); lstrcat(szCmdLine, TEXT("\"")); // Initialize process startup info
STARTUPINFO si = {0}; si.cb = sizeof(si); si.dwFlags = STARTF_USESHOWWINDOW; si.wShowWindow = SW_SHOWNORMAL; // Start explorer
PROCESS_INFORMATION pi = {0}; if (CreateProcess(NULL, szCmdLine, NULL, NULL, FALSE, NORMAL_PRIORITY_CLASS, NULL, NULL, &si, &pi)) { // Close the process and thread handles
CloseHandle(pi.hProcess); CloseHandle(pi.hThread); } }
int CTray::_GetQuickLaunchID() { int iQLBandID = -1; DWORD dwBandID; for (int i = 0; (iQLBandID == -1) && SUCCEEDED(_ptbs->EnumBands(i, &dwBandID)); i++) { if (BandSite_TestBandCLSID(_ptbs, dwBandID, CLSID_ISFBand) == S_OK) { IUnknown* punk; if (SUCCEEDED(_ptbs->GetBandObject(dwBandID, IID_PPV_ARG(IUnknown, &punk)))) { VARIANTARG v = {0}; v.vt = VT_I4; if (SUCCEEDED(IUnknown_Exec(punk, &CLSID_ISFBand, 1, 0, NULL, &v))) { if ((v.vt == VT_I4) && (CSIDL_APPDATA == (DWORD)v.lVal)) { iQLBandID = (int)dwBandID; } } punk->Release(); } } }
return iQLBandID; }
int CTray::_ToggleQL(int iVisible) { int iQLBandID = _GetQuickLaunchID();
bool fOldVisible = (-1 != iQLBandID); bool fNewVisible = (0 != iVisible);
if ((iVisible != -1) && (fNewVisible != fOldVisible)) { if (fNewVisible) { LPITEMIDLIST pidl; if (SUCCEEDED(SHGetSpecialFolderLocation(NULL, CSIDL_APPDATA, &pidl))) { TCHAR szPath[MAX_PATH]; SHGetPathFromIDList(pidl, szPath); PathCombine(szPath, szPath, L"Microsoft\\Internet Explorer\\Quick Launch"); ILFree(pidl); pidl = ILCreateFromPath(szPath); if (pidl) { IFolderBandPriv *pfbp; // create an ISF band to show folders as hotlinks
if (SUCCEEDED(CoCreateInstance(CLSID_ISFBand, NULL, CLSCTX_INPROC, IID_PPV_ARG(IFolderBandPriv, &pfbp)))) { IShellFolderBand* psfb; if (SUCCEEDED(pfbp->QueryInterface(IID_PPV_ARG(IShellFolderBand, &psfb)))) { if (SUCCEEDED(psfb->InitializeSFB(NULL, pidl))) { pfbp->SetNoText(TRUE);
VARIANTARG v; v.vt = VT_I4; v.lVal = CSIDL_APPDATA; IUnknown_Exec(psfb, &CLSID_ISFBand, 1, 0, &v, NULL);
v.lVal = UEMIND_SHELL; // UEMIND_SHELL/BROWSER
IUnknown_Exec(psfb, &CGID_ShellDocView, SHDVID_UEMLOG, 0, &v, NULL);
IDeskBand* ptb; if (SUCCEEDED(pfbp->QueryInterface(IID_PPV_ARG(IDeskBand, &ptb)))) { HRESULT hr = _ptbs->AddBand(ptb); if (SUCCEEDED(hr)) { _ptbs->SetBandState(ShortFromResult(hr), BSSF_NOTITLE, BSSF_NOTITLE); } ptb->Release(); } } psfb->Release(); } pfbp->Release(); }
ILFree(pidl); } } } else { int iBandID; do { iBandID = _GetQuickLaunchID(); if (iBandID != -1) { _ptbs->RemoveBand(iBandID); } } while (iBandID != -1); } }
return iQLBandID; }
void CTray::StartMenuContextMenu(HWND hwnd, DWORD dwPos) { LPITEMIDLIST pidlStart = SHCloneSpecialIDList(hwnd, CSIDL_STARTMENU, TRUE); INSTRUMENT_STATECHANGE(SHCNFI_STATE_TRAY_CONTEXT_START); HandleFullScreenApp(NULL);
SetForegroundWindow(hwnd);
if (pidlStart) { LPITEMIDLIST pidlLast = ILClone(ILFindLastID(pidlStart)); ILRemoveLastID(pidlStart);
if (pidlLast) { IShellFolder *psf = BindToFolder(pidlStart); if (psf) { HMENU hmenu = CreatePopupMenu(); if (hmenu) { IContextMenu *pcm; HRESULT hr = psf->GetUIObjectOf(hwnd, 1, (LPCITEMIDLIST*)&pidlLast, IID_X_PPV_ARG(IContextMenu, NULL, &pcm)); if (SUCCEEDED(hr)) { hr = pcm->QueryContextMenu(hmenu, 0, IDSYSPOPUP_FIRST, IDSYSPOPUP_LAST, CMF_VERBSONLY); if (SUCCEEDED(hr)) { int idCmd; TCHAR szCommon[MAX_PATH];
//Add the menu to invoke the "Start Menu Properties"
LoadString (hinstCabinet, IDS_STARTMENUPROP, szCommon, ARRAYSIZE(szCommon)); AppendMenu (hmenu, MF_STRING, IDSYSPOPUP_STARTMENUPROP, szCommon); if (!SHRestricted(REST_NOCOMMONGROUPS)) { // If the user has access to the Common Start Menu, then we can add those items. If not,
// then we should not.
BOOL fAddCommon = (S_OK == SHGetFolderPath(NULL, CSIDL_COMMON_STARTMENU, NULL, 0, szCommon));
if (fAddCommon) fAddCommon = IsUserAnAdmin();
// Since we don't show this on the start button when the user is not an admin, don't show it here... I guess...
if (fAddCommon) { AppendMenu (hmenu, MF_SEPARATOR, 0, NULL); LoadString (hinstCabinet, IDS_OPENCOMMON, szCommon, ARRAYSIZE(szCommon)); AppendMenu (hmenu, MF_STRING, IDSYSPOPUP_OPENCOMMON, szCommon); LoadString (hinstCabinet, IDS_EXPLORECOMMON, szCommon, ARRAYSIZE(szCommon)); AppendMenu (hmenu, MF_STRING, IDSYSPOPUP_EXPLORECOMMON, szCommon); } }
if (dwPos == (DWORD)-1) { idCmd = _TrackMenu(hmenu); } else { SendMessage(_hwndTrayTips, TTM_ACTIVATE, FALSE, 0L); idCmd = TrackPopupMenu(hmenu, TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTALIGN, GET_X_LPARAM(dwPos), GET_Y_LPARAM(dwPos), 0, hwnd, NULL); SendMessage(_hwndTrayTips, TTM_ACTIVATE, TRUE, 0L); }
switch(idCmd) { case 0: //User did not select a menu item; so, nothing to do!
break; case IDSYSPOPUP_OPENCOMMON: _ExploreCommonStartMenu(FALSE); break;
case IDSYSPOPUP_EXPLORECOMMON: _ExploreCommonStartMenu(TRUE); break;
case IDSYSPOPUP_STARTMENUPROP: DoProperties(TPF_STARTMENUPAGE); break; default: TCHAR szPath[MAX_PATH]; CMINVOKECOMMANDINFOEX ici = {0}; #ifdef UNICODE
CHAR szPathAnsi[MAX_PATH]; #endif
ici.cbSize = sizeof(CMINVOKECOMMANDINFOEX); ici.hwnd = hwnd; ici.lpVerb = (LPSTR)MAKEINTRESOURCE(idCmd - IDSYSPOPUP_FIRST); ici.nShow = SW_NORMAL; #ifdef UNICODE
SHGetPathFromIDListA(pidlStart, szPathAnsi); SHGetPathFromIDList(pidlStart, szPath); ici.lpDirectory = szPathAnsi; ici.lpDirectoryW = szPath; ici.fMask |= CMIC_MASK_UNICODE; #else
SHGetPathFromIDList(pidlStart, szPath); ici.lpDirectory = szPath; #endif
pcm->InvokeCommand((LPCMINVOKECOMMANDINFO)&ici);
break; } // Switch(idCmd)
} pcm->Release(); } DestroyMenu(hmenu); } psf->Release(); } ILFree(pidlLast); } ILFree(pidlStart); } }
void GiveDesktopFocus() { SetForegroundWindow(v_hwndDesktop); SendMessage(v_hwndDesktop, DTM_UIACTIVATEIO, (WPARAM) TRUE, /*dtb*/0); }
/*----------------------------------------------------------
Purpose: loads the given resource string and executes it. The resource string should follow this format:
"program.exe>parameters"
If there are no parameters, the format should simply be:
"program.exe"
*/ void _ExecResourceCmd(UINT ids) { TCHAR szCmd[2*MAX_PATH];
if (LoadString(hinstCabinet, ids, szCmd, SIZECHARS(szCmd))) { SHELLEXECUTEINFO sei = {0};
// Find list of parameters (if any)
LPTSTR pszParam = StrChr(szCmd, TEXT('>'));
if (pszParam) { // Replace the '>' with a null terminator
*pszParam = 0; pszParam++; }
sei.cbSize = sizeof(sei); sei.nShow = SW_SHOWNORMAL; sei.lpFile = szCmd; sei.lpParameters = pszParam; ShellExecuteEx(&sei); } }
void CTray::_RefreshStartMenu() { if (_pmbStartMenu) { IUnknown_Exec(_pmbStartMenu, &CLSID_MenuBand, MBANDCID_REFRESH, 0, NULL, NULL); } else if (_pmpStartPane) { IUnknown_Exec(_pmpStartPane, &CLSID_MenuBand, MBANDCID_REFRESH, 0, NULL, NULL); } _RefreshSettings(); _UpdateBandSiteStyle(); }
BOOL CTray::_CanMinimizeAll() { #ifdef FEATURE_STARTPAGE
if (Tray_ShowStartPageEnabled()) return FALSE; #endif
return (_hwndTasks && SendMessage(_hwndTasks, TBC_CANMINIMIZEALL, 0, 0)); }
BOOL CTray::_MinimizeAll(BOOL fPostRaiseDesktop) { BOOL fRet = FALSE;
if (_hwndTasks) { fRet = (BOOL)SendMessage(_hwndTasks, TBC_MINIMIZEALL, (WPARAM)_hwnd, (LPARAM)fPostRaiseDesktop); }
return fRet; }
extern void _UpdateNotifySetting(BOOL fNotifySetting);
//
// Due to the weirdness of PnP, if the eject request occurs on a thread
// that contains windows, the eject stalls for 15 seconds. So do it
// on its own thread.
//
DWORD CALLBACK _EjectThreadProc(LPVOID lpThreadParameter) { CM_Request_Eject_PC(); return 0; }
void CTray::_Command(UINT idCmd) { INSTRUMENT_ONCOMMAND(SHCNFI_TRAYCOMMAND, _hwnd, idCmd);
switch (idCmd) {
case IDM_CONTROLS: case IDM_PRINTERS: _ShowFolder(_hwnd, idCmd == IDM_CONTROLS ? CSIDL_CONTROLS : CSIDL_PRINTERS, COF_USEOPENSETTINGS); break;
case IDM_EJECTPC: // Must use SHCreateThread and not a queued workitem because
// a workitem might inherit a thread that has windows on it.
// CTF_INSIST: In emergency, eject synchronously. This stalls
// for 15 seconds but it's better than nothing.
SHCreateThread(_EjectThreadProc, NULL, CTF_INSIST, NULL); break;
case IDM_LOGOFF: // Let the desktop get a chance to repaint to get rid of the
// start menu bits before bringing up the logoff dialog box.
UpdateWindow(_hwnd); Sleep(100);
_SaveTrayAndDesktop(); LogoffWindowsDialog(v_hwndDesktop); break;
case IDM_MU_DISCONNECT: // Do the same sleep as above for the same reason.
UpdateWindow(_hwnd); Sleep(100); DisconnectWindowsDialog(v_hwndDesktop); break;
case IDM_EXITWIN: // Do the same sleep as above for the same reason.
UpdateWindow(_hwnd); Sleep(100);
_DoExitWindows(v_hwndDesktop); break;
case IDM_TOGGLEDESKTOP: _RaiseDesktop(!g_fDesktopRaised, TRUE); break;
case IDM_FILERUN: _RunDlg(); break;
case IDM_MINIMIZEALLHOTKEY: _HandleGlobalHotkey(GHID_MINIMIZEALL); break;
#ifdef DEBUG
case IDM_SIZEUP: { RECT rcView; GetWindowRect(_hwndRebar, &rcView); MapWindowPoints(HWND_DESKTOP, _hwnd, (LPPOINT)&rcView, 2); rcView.bottom -= 18; SetWindowPos(_hwndRebar, NULL, 0, 0, RECTWIDTH(rcView), RECTHEIGHT(rcView), SWP_NOMOVE | SWP_NOZORDER); } break;
case IDM_SIZEDOWN: { RECT rcView; GetWindowRect(_hwndRebar, &rcView); MapWindowPoints(HWND_DESKTOP, _hwnd, (LPPOINT)&rcView, 2); rcView.bottom += 18; SetWindowPos(_hwndRebar, NULL, 0, 0, RECTWIDTH(rcView), RECTHEIGHT(rcView), SWP_NOMOVE | SWP_NOZORDER); } break; #endif
case IDM_MINIMIZEALL: // minimize all window
_MinimizeAll(FALSE); _fUndoEnabled = TRUE; break;
case IDM_UNDO: _RestoreWindowPositions(FALSE); break;
case IDM_SETTIME: // run the default applet in timedate.cpl
SHRunControlPanel(TEXT("timedate.cpl"), _hwnd); break;
case IDM_NOTIFYCUST: DoProperties(TPF_TASKBARPAGE | TPF_INVOKECUSTOMIZE); break;
case IDM_LOCKTASKBAR: { BOOL fCanSizeMove = !_fCanSizeMove; // toggle
SHRegSetUSValue(REGSTR_EXPLORER_ADVANCED, TEXT("TaskbarSizeMove"), REG_DWORD, &fCanSizeMove , sizeof(DWORD), SHREGSET_FORCE_HKCU); _RefreshSettings(); _UpdateBandSiteStyle(); } break;
case IDM_SHOWTASKMAN: RunSystemMonitor(); break;
case IDM_CASCADE: case IDM_VERTTILE: case IDM_HORIZTILE: if (_CanTileAnyWindows()) { SaveWindowPositions((idCmd == IDM_CASCADE) ? IDS_CASCADE : IDS_TILE);
_AppBarNotifyAll(NULL, ABN_WINDOWARRANGE, NULL, TRUE);
if (idCmd == IDM_CASCADE) { CascadeWindows(GetDesktopWindow(), 0, NULL, 0, NULL); } else { TileWindows(GetDesktopWindow(), ((idCmd == IDM_VERTTILE)? MDITILE_VERTICAL : MDITILE_HORIZONTAL), NULL, 0, NULL); }
// do it *before* ABN_xxx so don't get 'indirect' moves
// REVIEW or should it be after?
// CheckWindowPositions();
_fUndoEnabled = FALSE; SetTimer(_hwnd, IDT_ENABLEUNDO, 500, NULL);
_AppBarNotifyAll(NULL, ABN_WINDOWARRANGE, NULL, FALSE); } break;
case IDM_TRAYPROPERTIES: DoProperties(TPF_TASKBARPAGE); break;
case IDM_SETTINGSASSIST: SHCreateThread(SettingsUIThreadProc, NULL, 0, NULL); break;
case IDM_HELPSEARCH: _ExecResourceCmd(IDS_HELP_CMD); break;
// NB The Alt-s comes in here.
case IDC_KBSTART: SetForegroundWindow(_hwnd); // This pushes the start button and causes the start menu to popup.
SendMessage(_hwndStart, BM_SETSTATE, TRUE, 0); // This forces the button back up.
SendMessage(_hwndStart, BM_SETSTATE, FALSE, 0); break;
case IDC_ASYNCSTART: #if 0 // (for testing UAssist locking code)
UEMFireEvent(&UEMIID_SHELL, UEME_DBSLEEP, UEMF_XEVENT, -1, (LPARAM)10000); #endif
#ifdef DEBUG
if (GetAsyncKeyState(VK_SHIFT) < 0) { UEMFireEvent(&UEMIID_SHELL, UEME_CTLSESSION, UEMF_XEVENT, TRUE, -1); _RefreshStartMenu(); } #endif
// Make sure the button is down.
// DebugMsg(DM_TRACE, "c.twp: IDC_START.");
// Make sure the Start button is down.
if (!_bMainMenuInit && SendMessage(_hwndStart, BM_GETSTATE, 0, 0) & BST_PUSHED) { // DebugMsg(DM_TRACE, "c.twp: Start button down.");
// Set the focus.
_SetFocus(_hwndStart); _ToolbarMenu(); } break;
// NB LButtonDown on the Start button come in here.
// Space-bar stuff also comes in here.
case IDC_START: // User gets a bit confused with space-bar tuff (the popup ends up
// getting the key-up and beeps).
PostMessage(_hwnd, WM_COMMAND, IDC_ASYNCSTART, 0); break;
case FCIDM_FINDFILES: SHFindFiles(NULL, NULL); break;
case FCIDM_FINDCOMPUTER: SHFindComputer(NULL, NULL); break;
case FCIDM_REFRESH: _RefreshStartMenu(); break;
case FCIDM_NEXTCTL: { MSG msg = { 0, WM_KEYDOWN, VK_TAB }; HWND hwndFocus = GetFocus();
// Since we are Tab or Shift Tab we should turn the focus rect on.
//
// Note: we don't need to do this in the GiveDesktopFocus cases below,
// but in those cases we're probably already in the UIS_CLEAR UISF_HIDEFOCUS
// state so this message is cheap to send.
//
SendMessage(_hwnd, WM_UPDATEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);
BOOL fShift = GetAsyncKeyState(VK_SHIFT) < 0;
if (hwndFocus && (IsChildOrHWND(_hwndStart, hwndFocus))) { if (fShift) { // gotta deactivate manually
GiveDesktopFocus(); } else { IUnknown_UIActivateIO(_ptbs, TRUE, &msg); } } else if (hwndFocus && (IsChildOrHWND(_hwndNotify, hwndFocus))) { if (fShift) { IUnknown_UIActivateIO(_ptbs, TRUE, &msg); } else { GiveDesktopFocus(); } } else { if (IUnknown_TranslateAcceleratorIO(_ptbs, &msg) != S_OK) { if (fShift) { _SetFocus(_hwndStart); } else { // if you tab forward out of the bands, the next focus guy is the tray notify set
_SetFocus(_hwndNotify); } } } } break;
case IDM_MU_SECURITY: MuSecurity(); break; } }
//// Start menu/Tray tab as a drop target
HRESULT CStartDropTarget::_GetStartMenuDropTarget(IDropTarget** pptgt) { HRESULT hr = E_FAIL; *pptgt = NULL;
LPITEMIDLIST pidlStart = SHCloneSpecialIDList(NULL, CSIDL_STARTMENU, TRUE);
if (pidlStart) { IShellFolder *psf = BindToFolder(pidlStart); if (psf) { hr = psf->CreateViewObject(_ptray->_hwnd, IID_PPV_ARG(IDropTarget, pptgt)); psf->Release(); }
ILFree(pidlStart); } return hr; }
STDMETHODIMP CDropTargetBase::QueryInterface(REFIID riid, void ** ppvObj) { static const QITAB qit[] = { QITABENT(CDropTargetBase, IDropTarget), { 0 }, };
return QISearch(this, qit, riid, ppvObj); }
STDMETHODIMP_(ULONG) CDropTargetBase::AddRef() { return 2; }
STDMETHODIMP_(ULONG) CDropTargetBase::Release() { return 1; }
STDMETHODIMP CDropTargetBase::DragEnter(IDataObject *pdtobj, DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { _ptray->_SetUnhideTimer(ptl.x, ptl.y);
HWND hwndLock = _ptray->_hwnd; // no clippy
_DragEnter(hwndLock, ptl, pdtobj);
return S_OK; }
STDMETHODIMP CDropTargetBase::DragOver(DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { _ptray->_SetUnhideTimer(ptl.x, ptl.y); _DragMove(_ptray->_hwndStart, ptl);
return S_OK; }
STDMETHODIMP CDropTargetBase::DragLeave() { DAD_DragLeave(); return S_OK; }
STDMETHODIMP CDropTargetBase::Drop(IDataObject *pdtobj, DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { DAD_DragLeave(); return S_OK; }
//
// There are two different policies for the Start Button depending on
// whether we are in Classic mode or Personal (New Start Pane) mode.
//
// Classic mode: Drops onto the Start Button are treated as if they
// were drops into the CSIDL_STARTMENU folder.
//
// Personal mode: Drops onto the Start Button are treated as if they
// were drops into the pin list.
//
CStartDropTarget::CStartDropTarget() : CDropTargetBase(IToClass(CTray, _dtStart, this)) { }
CTrayDropTarget::CTrayDropTarget() : CDropTargetBase(IToClass(CTray, _dtTray, this)) { }
STDMETHODIMP CTrayDropTarget::DragEnter(IDataObject *pdtobj, DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { *pdwEffect = DROPEFFECT_NONE; return CDropTargetBase::DragEnter(pdtobj, grfKeyState, ptl, pdwEffect); }
STDMETHODIMP CTrayDropTarget::DragOver(DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { *pdwEffect = DROPEFFECT_NONE; return CDropTargetBase::DragOver(grfKeyState, ptl, pdwEffect); }
void CStartDropTarget::_StartAutoOpenTimer(POINTL *pptl) { POINT pt = { pptl->x, pptl->y }; RECT rc; //Make sure it really is in the start menu..
GetWindowRect(_ptray->_hwndStart, &rc); if (PtInRect(&rc,pt)) { SetTimer(_ptray->_hwnd, IDT_STARTMENU, 1000, NULL); } }
STDMETHODIMP CStartDropTarget::DragEnter(IDataObject *pdtobj, DWORD grfKeyState, POINTL ptl, DWORD *pdwEffect) { HRESULT hr; if (Tray_StartPanelEnabled()) { // Personal mode: Treat it as an add to the pin list.
if (_ptray->_psmpin && _ptray->_psmpin->IsPinnable(pdtobj, SMPINNABLE_REJECTSLOWMEDIA, NULL) == S_OK) { _dwEffectsAllowed = DROPEFFECT_LINK; } else { _dwEffectsAllowed = DROPEFFECT_NONE; }
*pdwEffect &= _dwEffectsAllowed;
// Always start the AutoOpen timer because once we open, the user
// can drop onto other things which may have different drop policies
// from the pin list.
_StartAutoOpenTimer(&ptl); hr = S_OK; } else { // Classic mode: Treat it as a drop on the Start Menu folder.
IDropTarget* ptgt; _dwEffectsAllowed = DROPEFFECT_LINK;
hr = _GetStartMenuDropTarget(&ptgt); if (SUCCEEDED(hr)) { // Check to make sure that we're going to accept the drop before we expand the start menu.
ptgt->DragEnter(pdtobj, grfKeyState, ptl, pdwEffect);
// DROPEFFECT_NONE means it ain't gonna work, so don't popup the Start Menu.
if (*pdwEffect != DROPEFFECT_NONE) { _StartAutoOpenTimer(&ptl); }
ptgt->DragLeave(); ptgt->Release(); } }
CDropTargetBase::DragEnter(pdtobj, grfKeyState, ptl, pdwEffect); return hr; }
STDMETHODIMP CStartDropTarget::DragOver(DWORD grfKeyState, POINTL pt, DWORD *pdwEffect) { *pdwEffect = (_dwEffectsAllowed & DROPEFFECT_LINK); return CDropTargetBase::DragOver(grfKeyState, pt, pdwEffect); }
STDMETHODIMP CStartDropTarget::DragLeave() { KillTimer(_ptray->_hwnd, IDT_STARTMENU); return CDropTargetBase::DragLeave(); }
STDMETHODIMP CStartDropTarget::Drop(IDataObject *pdtobj, DWORD grfKeyState, POINTL pt, DWORD *pdwEffect) { KillTimer(_ptray->_hwnd, IDT_STARTMENU);
HRESULT hr;
if (Tray_StartPanelEnabled()) { // Personal mode: Treat it as an add to the pin list.
LPITEMIDLIST pidl; if (_ptray->_psmpin && _ptray->_psmpin->IsPinnable(pdtobj, SMPINNABLE_REJECTSLOWMEDIA, &pidl) == S_OK) { // Delete it from the pin list if it's already there because
// we want to move it to the bottom.
_ptray->_psmpin->Modify(pidl, NULL); // Now add it to the bottom.
_ptray->_psmpin->Modify(NULL, pidl); ILFree(pidl); hr = S_OK; } else { hr = E_FAIL; } } else { IDropTarget* pdrop; hr = _GetStartMenuDropTarget(&pdrop); if (SUCCEEDED(hr)) { if (!Tray_StartPanelEnabled()) { POINTL ptDrop = { 0, 0 }; DWORD grfKeyStateDrop = 0;
*pdwEffect &= DROPEFFECT_LINK;
pdrop->DragEnter(pdtobj, grfKeyStateDrop, ptDrop, pdwEffect); hr = pdrop->Drop(pdtobj, grfKeyStateDrop, ptDrop, pdwEffect);
pdrop->DragLeave(); } pdrop->Release(); } }
DAD_DragLeave();
return hr; }
void CTray::_RegisterDropTargets() { THR(RegisterDragDrop(_hwndStart, &_dtStart)); THR(RegisterDragDrop(_hwnd, &_dtTray));
// It is not a serious error if this fails; it just means that
// drag/drop to the Start Button will not add to the pin list
CoCreateInstance(CLSID_StartMenuPin, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IStartMenuPin, &_psmpin)); }
void CTray::_RevokeDropTargets() { RevokeDragDrop(_hwndStart); RevokeDragDrop(_hwnd); ATOMICRELEASET(_psmpin, IStartMenuPin); }
void CTray::_HandleGlobalHotkey(WPARAM wParam) { INSTRUMENT_HOTKEY(SHCNFI_GLOBALHOTKEY, wParam);
switch(wParam) { case GHID_RUN: _RunDlg(); break;
case GHID_MINIMIZEALL: if (_CanMinimizeAll()) _MinimizeAll(FALSE); SetForegroundWindow(v_hwndDesktop); break;
case GHID_UNMINIMIZEALL: _RestoreWindowPositions(FALSE); break;
case GHID_HELP: _Command(IDM_HELPSEARCH); break;
case GHID_DESKTOP: _RaiseDesktop(!g_fDesktopRaised, TRUE); break;
case GHID_TRAYNOTIFY: SwitchToThisWindow(_hwnd, TRUE); SetForegroundWindow(_hwnd); _SetFocus(_hwndNotify); break;
case GHID_EXPLORER: _ShowFolder(_hwnd, CSIDL_DRIVES, COF_CREATENEWWINDOW | COF_EXPLORE); break;
case GHID_FINDFILES: if (!SHRestricted(REST_NOFIND)) _Command(FCIDM_FINDFILES); break;
case GHID_FINDCOMPUTER: if (!SHRestricted(REST_NOFIND)) _Command(FCIDM_FINDCOMPUTER); break;
case GHID_TASKTAB: case GHID_TASKSHIFTTAB: if (GetForegroundWindow() != _hwnd) SetForegroundWindow(_hwnd); SendMessage(_hwndTasks, TBC_TASKTAB, wParam == GHID_TASKTAB ? 1 : -1, 0L); break;
case GHID_SYSPROPERTIES: #define IDS_SYSDMCPL 0x2334 // from shelldll
SHRunControlPanel(MAKEINTRESOURCE(IDS_SYSDMCPL), _hwnd); break; } }
void CTray::_UnregisterGlobalHotkeys() { for (int i = GHID_FIRST; i < GHID_MAX; i++) { UnregisterHotKey(_hwnd, i); } }
void CTray::_RegisterGlobalHotkeys() { int i; // Are the Windows keys restricted?
DWORD dwRestricted = SHRestricted(REST_NOWINKEYS);
for (i = GHID_FIRST ; i < GHID_MAX; i++) { // If the Windows Keys are Not restricted or it's not a Windows key
if (!((HIWORD(GlobalKeylist[i - GHID_FIRST]) & MOD_WIN) && dwRestricted)) { // Then register it.
RegisterHotKey(_hwnd, i, HIWORD(GlobalKeylist[i - GHID_FIRST]), LOWORD(GlobalKeylist[i - GHID_FIRST])); } } }
void CTray::_RaiseDesktop(BOOL fRaise, BOOL fRestoreWindows) { if (v_hwndDesktop && (fRaise == !g_fDesktopRaised) && !_fProcessingDesktopRaise) { _fProcessingDesktopRaise = TRUE; BOOL fPostMessage = TRUE;
if (fRaise) { HWND hwndFG = GetForegroundWindow(); // If no window has focus then set focus to the tray
if (hwndFG) { hwndFG = _hwnd; }
if (!_hwndFocusBeforeRaise) { // See if the Foreground Window had a popup window
_hwndFocusBeforeRaise = GetLastActivePopup(hwndFG); } if (!IsWindowVisible(_hwndFocusBeforeRaise)) { _hwndFocusBeforeRaise = hwndFG; }
// _MinimizeAll will save the windows positions synchronously, and will minimize the
// the windows on a background thread
_fMinimizedAllBeforeRaise = _CanMinimizeAll(); if (_fMinimizedAllBeforeRaise) { fPostMessage = !_MinimizeAll(TRUE); } } else { if (fRestoreWindows) { HWND hwnd = _hwndFocusBeforeRaise; if (_fMinimizedAllBeforeRaise) { // Since the windows are restored on a seperate thread, I want the make that the
// desktop is not raised until they are done, so the window restore thread will
// actually post the message for raising the desktop
fPostMessage = !_RestoreWindowPositions(TRUE); }
SetForegroundWindow(hwnd); if (hwnd == _hwnd) { _SetFocus(_hwndStart); } }
_hwndFocusBeforeRaise = NULL; }
#ifdef FEATURE_STARTPAGE
if (Tray_ShowStartPageEnabled()) { if (fPostMessage) SendMessage(v_hwndDesktop, DTM_RAISE, (WPARAM)_hwnd, fRaise ? DTRF_RAISE : DTRF_LOWER); } else #endif
{ if (fPostMessage) PostMessage(v_hwndDesktop, DTM_RAISE, (WPARAM)_hwnd, fRaise ? DTRF_RAISE : DTRF_LOWER); } } }
void CTray::_OnDesktopState(LPARAM lParam) { g_fDesktopRaised = (!(lParam & DTRF_LOWER));
DAD_ShowDragImage(FALSE); // unlock the drag sink if we are dragging.
if (!g_fDesktopRaised) { HandleFullScreenApp(NULL); } else { // if the desktop is raised, we need to force the tray to be always on top
// until it's lowered again
_ResetZorder(); }
DAD_ShowDragImage(TRUE); // unlock the drag sink if we are dragging.
_fProcessingDesktopRaise = FALSE; }
BOOL CTray::_ToggleLanguageBand(BOOL fShowIt) { HRESULT hr = E_FAIL;
DWORD dwBandID; BOOL fFound = FALSE; for (int i = 0; !fFound && SUCCEEDED(_ptbs->EnumBands(i, &dwBandID)); i++) { if (BandSite_TestBandCLSID(_ptbs, dwBandID, CLSID_MSUTBDeskBand) == S_OK) { fFound = TRUE; } }
BOOL fShow = fFound;
if (fShowIt && !fFound) { IDeskBand* pdb; HRESULT hr = CoCreateInstance(CLSID_MSUTBDeskBand, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IDeskBand, &pdb)); if (SUCCEEDED(hr)) { hr = _ptbs->AddBand(pdb); fShow = TRUE; pdb->Release(); } } else if (!fShowIt && fFound) { hr = _ptbs->RemoveBand(dwBandID); if (SUCCEEDED(hr)) { fShow = FALSE; } }
return fShow; }
// Process the message by propagating it to all of our child windows
typedef struct { UINT uMsg; WPARAM wP; LPARAM lP; CTray* ptray; } CABPM;
BOOL CTray::PropagateEnumProc(HWND hwnd, LPARAM lParam) { CABPM *ppm = (CABPM *)lParam; if (SHIsChildOrSelf(ppm->ptray->_hwndRebar, hwnd) == S_OK) { return TRUE; } SendMessage(hwnd, ppm->uMsg, ppm->wP, ppm->lP); return TRUE; }
void CTray::_PropagateMessage(HWND hwnd, UINT uMessage, WPARAM wParam, LPARAM lParam) { CABPM pm = {uMessage, wParam, lParam, this};
ASSERT(hwnd != _hwndRebar); EnumChildWindows(hwnd, PropagateEnumProc, (LPARAM)&pm); }
//
// Called from SETTINGS.DLL when the tray property sheet needs
// to be activated. See SettingsUIThreadProc below.
//
// Also used by desktop2\deskhost.cpp to get to the tray properties.
//
void WINAPI Tray_DoProperties(DWORD dwFlags) { c_tray.DoProperties(dwFlags); }
DWORD WINAPI CTray::SettingsUIThreadProc(void *pv) { //
// Open up the "Settings Wizards" UI.
//
HMODULE hmodSettings = LoadLibrary(TEXT("settings.dll")); if (NULL != hmodSettings) { //
// Entry point in SETTINGS.DLL is ordinal 1.
// Don't want to export this entry point by name.
//
PSETTINGSUIENTRY pfDllEntry = (PSETTINGSUIENTRY)GetProcAddress(hmodSettings, (LPCSTR)1); if (NULL != pfDllEntry) { //
// This call will open and run the UI.
// The thread's message loop is inside settings.dll.
// This call will not return until the settings UI has been closed.
//
(*pfDllEntry)(Tray_DoProperties); }
FreeLibrary(hmodSettings); } return 0; }
void _MigrateIEXPLORE(HKEY hkBrowser) { TCHAR szAppName[MAX_PATH]; LONG cb = sizeof(szAppName);
if (ERROR_SUCCESS == RegQueryValue(hkBrowser, NULL, szAppName, &cb)) { if (0 == lstrcmp(szAppName, TEXT("Internet Explorer"))) { TraceMsg(TF_WARNING, "Migrating old StartMenuInternet setting (you just upgraded from build 2465+)");
// copy in the good value
lstrcpy(szAppName, TEXT("iexplore.exe"));
// this will fail if the user is not an admin, oh well.
if (ERROR_SUCCESS == RegSetValueEx(hkBrowser, NULL, 0, REG_SZ, (BYTE *)szAppName, sizeof(TCHAR) * (lstrlen(szAppName)+1))) { // Now tell everybody about the change
SHSendMessageBroadcast(WM_SETTINGCHANGE, 0, (LPARAM)TEXT("Software\\Clients\\StartMenuInternet")); } } } }
//
// This function is called whenever we detect a change to the default
// browser registration in HKCR\http\shell\open\command.
//
// For compatibility with old browsers, if the default browser (URL handler)
// is not XP-aware, then auto-generate a StartMenuInternet client
// registration and set it as the default.
//
void CTray::_MigrateOldBrowserSettings() { // We want only one person to do this work per machine (though it doesn't
// hurt to have more than one person do it; it's just pointless), so try
// to filter out people who clearly didn't instigate the key change.
//
if (!_fIsDesktopLocked && _fIsDesktopConnected) { // If the user does not have write access then we can't migrate the
// setting... (In which case there was nothing to migrate anyway
// since you need to be administrator to change the default browser...)
HKEY hkBrowser; DWORD dwDisposition;
if (RegCreateKeyEx(HKEY_LOCAL_MACHINE, TEXT("Software\\Clients\\StartMenuInternet"), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE, NULL, &hkBrowser, &dwDisposition) == ERROR_SUCCESS) { TCHAR szCommand[MAX_PATH]; DWORD cch = ARRAYSIZE(szCommand);
// It is important that we use AssocQueryString to parse the
// executable, because Netscape has a habit of registering
// their path incorrectly (they forget to quote the space in
// "Program Files") but AssocQueryString has special recovery
// code to detect and repair that case...
if (SUCCEEDED(AssocQueryString(ASSOCF_NOUSERSETTINGS, ASSOCSTR_EXECUTABLE, L"http", L"open", szCommand, &cch)) && szCommand[0]) { TCHAR szAppName[MAX_PATH]; lstrcpyn(szAppName, PathFindFileName(szCommand), ARRAYSIZE(szAppName));
// You might think that we need to special-case MSN Explorer,
// since they shipped before XP RTM'd, and convert MSN6.EXE
// to "MSN Explorer", but that's not true because
// they never take over as the default http handler, so we
// will never see them here!
// TODO - Remove _MigrateIXPLORE post-whistler. (saml 010622)
// we do need to special-case the string "Internet Explorer"
// which was written out starting with build 2465 (post-B2!)
// We want ie.inx to continue to specify FLG_ADDREG_NOCLOBBER (=2) on software\client\startmenuinternet,
// since we really don't want want to clobber this value
// Therefore, just special-case migrate the known bad value to the new good value ("iexplore.exe")
_MigrateIEXPLORE(hkBrowser);
// Create a registration for the new default browser if necessary.
// We keep our hands off once we see a DefaultIcon key, since that
// proves that the application is XP-aware.
// When IE Access is turned off, StartMenuInternet\IExplore.exe is
// also removed so that IE will not appear in "Customize Start Menu"
// dialog box. Do not migrate IE.
HKEY hkClient;
if ( 0 != lstrcmpi(szAppName, TEXT("IEXPLORE.EXE")) && RegCreateKeyEx(hkBrowser, szAppName, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hkClient, &dwDisposition) == ERROR_SUCCESS) { if (dwDisposition == REG_CREATED_NEW_KEY) { TCHAR szFriendly[MAX_PATH]; cch = ARRAYSIZE(szFriendly); if (SUCCEEDED(AssocQueryString(ASSOCF_NOUSERSETTINGS | ASSOCF_INIT_BYEXENAME | ASSOCF_VERIFY, ASSOCSTR_FRIENDLYAPPNAME, szCommand, NULL, szFriendly, &cch))) { // Set the friendly name
RegSetValueEx(hkClient, TEXT("LocalizedString"), 0, REG_SZ, (BYTE*)szFriendly, sizeof(TCHAR) * (cch + 1));
// Set the command string (properly quoted)
PathQuoteSpaces(szCommand); SHSetValue(hkClient, TEXT("shell\\open\\command"), NULL, REG_SZ, szCommand, sizeof(TCHAR) * (1 + lstrlen(szCommand))); } }
LONG l = 0; if (RegQueryValue(hkClient, TEXT("DefaultIcon"), NULL, &l) == ERROR_FILE_NOT_FOUND) { // Set it as the system default
RegSetValueEx(hkBrowser, NULL, 0, REG_SZ, (BYTE*)szAppName, sizeof(TCHAR) * (lstrlen(szAppName) + 1));
// Now tell everybody about the change
SHSendMessageBroadcast(WM_SETTINGCHANGE, 0, (LPARAM)TEXT("Software\\Clients\\StartMenuInternet")); } RegCloseKey(hkClient); }
} RegCloseKey(hkBrowser); } }
// Restart the monitoring of the registry...
// (RegNotifyChangeKeyValue is good for only one shot.)
// Some apps (like Opera) delete the key as part of their registration,
// which causes our HKEY to go bad, so close it and make a new one.
if (_hkHTTP) { RegCloseKey(_hkHTTP); _hkHTTP = NULL; }
//
// Note! We have to register on HKCR\http\shell recursively
// even though we only care about HKCR\http\shell\open\command.
// The reason is that shell\open\command might not exist (IE
// deletes it as part of its uninstall) and you can't register
// a wait on a key that doesn't exist. We don't want to create
// a blank key on our own, because that means "To launch a web
// browser, run the null string as a command," which doesn't work
// too great.
//
if (RegCreateKeyEx(HKEY_CLASSES_ROOT, TEXT("http\\shell"), 0, NULL, REG_OPTION_NON_VOLATILE, KEY_ENUMERATE_SUB_KEYS | KEY_QUERY_VALUE | KEY_SET_VALUE | KEY_NOTIFY, NULL, &_hkHTTP, NULL) == ERROR_SUCCESS) { RegNotifyChangeKeyValue(_hkHTTP, TRUE, REG_NOTIFY_CHANGE_NAME | REG_NOTIFY_CHANGE_LAST_SET, _hHTTPEvent, TRUE); } }
void CTray::_MigrateOldBrowserSettingsCB(PVOID lpParameter, BOOLEAN) { //
// Sleep a little while so the app can finish installing all the
// registry keys it wants before we start cleaning up behind it.
//
Sleep(1000);
CTray *self = (CTray *)lpParameter; self->_MigrateOldBrowserSettings(); }
//
// *** WARNING ***
//
// This is a private interface EXPLORER.EXE exposes to SHDOCVW, which
// allows SHDOCVW (mostly desktop) to access tray. All member must be
// thread safe!
//
CDeskTray::CDeskTray() { _ptray = IToClass(CTray, _desktray, this); }
HRESULT CDeskTray::QueryInterface(REFIID riid, void ** ppvObj) { #if 0 // no IID_IDeskTray yet defined
static const QITAB qit[] = { QITABENT(CDeskTray, IDeskTray), { 0 }, }; return QISearch(this, qit, riid, ppvObj); #else
return E_NOTIMPL; #endif
}
ULONG CDeskTray::AddRef() { return 2; }
ULONG CDeskTray::Release() { return 1; }
HRESULT CDeskTray::GetTrayWindow(HWND* phwndTray) { ASSERT(_ptray->_hwnd); *phwndTray = _ptray->_hwnd; return S_OK; }
HRESULT CDeskTray::SetDesktopWindow(HWND hwndDesktop) { ASSERT(v_hwndDesktop == NULL); v_hwndDesktop = hwndDesktop; return S_OK; }
UINT CDeskTray::AppBarGetState() { return (_ptray->_uAutoHide ? ABS_AUTOHIDE : 0) | (_ptray->_fAlwaysOnTop ? ABS_ALWAYSONTOP : 0); }
//*** CDeskTray::SetVar -- set an explorer variable (var#i := value)
// ENTRY/EXIT
// var id# of variable to be changed
// value value to be assigned
// NOTES
// WARNING: thread safety is up to caller!
// notes: currently only called in 1 place, but extra generality is cheap
// minimal cost
HRESULT CDeskTray::SetVar(int var, DWORD value) { extern BOOL g_fExitExplorer;
TraceMsg(DM_TRACE, "c.cdt_sv: set var(%d):=%d", var, value); switch (var) { case SVTRAY_EXITEXPLORER: TraceMsg(DM_TRACE, "c.cdt_sv: set g_fExitExplorer:=%d", value); g_fExitExplorer = value; WriteCleanShutdown(1); break;
default: ASSERT(0); return S_FALSE; } return S_OK; }
|