Leaked source code of windows server 2003
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

1612 lines
54 KiB

//
// CleanupWiz.cpp
//
#include "CleanupWiz.h"
#include "resource.h"
#include "dblnul.h"
#include <windowsx.h> // for SetWindowFont
#include <varutil.h>
#include <commctrl.h>
#include <shlwapi.h>
#include <shguidp.h>
#include <ieguidp.h>
// UEM stuff: including this source file is the
// recommended way of using it in your project
// (see comments in the file itself for the reason)
#include "..\inc\uassist.cpp"
////////////////////////////////////////////
//
// Globals, constants, externs etc...
//
////////////////////////////////////////////
// none of these strings are ever localized, so it's safe to use them
const LPTSTR c_szRegStrSHELLFOLDERS = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Shell Folders");
const LPTSTR c_szRegStrDESKTOPNAMESPACE = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Desktop\\NameSpace");
const LPTSTR c_szRegStrPROFILELIST = TEXT("Software\\Microsoft\\Windows NT\\CurrentVersion\\ProfileList");
const LPTSTR c_szRegStrMSNCODES = TEXT("Software\\Microsoft\\MSN6\\Setup\\MSN\\Codes");
const LPTSTR c_szRegStrPATH_OCMANAGER = TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Setup\\OC Manager\\Subcomponents");
const LPTSTR c_szRegStrWMP_PATH_SETUP = TEXT("Software\\Microsoft\\MediaPlayer\\Setup");
const LPTSTR c_szRegStrPROFILESDIR = TEXT("ProfilesDirectory");
const LPTSTR c_szRegStrALLUSERS = TEXT("AllUsersProfile");
const LPTSTR c_szRegStrDEFAULTUSER = TEXT("DefaultUserProfile");
const LPTSTR c_szRegStrDESKTOP = TEXT("Desktop");
const LPTSTR c_szRegStrMSN_IAONLY = TEXT("IAOnly");
const LPTSTR c_szDESKTOP_DIR = TEXT("Desktop"); // backup in case we can't get the localized version
const LPTSTR c_szRegStrIEACCESS = TEXT("IEAccess");
const LPTSTR c_szRegStrYES = TEXT("yes");
const LPTSTR c_szRegStrWMP_REGVALUE = TEXT("DesktopShortcut");
const LPTSTR c_szDEFAULT_USER = TEXT("Default User");
const LPTSTR c_szVAL_TIME = TEXT("Last used time");
const LPTSTR c_szVAL_DELTA_DAYS = TEXT("Days between clean up");
const LPTSTR c_szVAL_DONTRUN = TEXT("NoRun");
const LPTSTR c_szVALUE_STARTPANEL = TEXT("NewStartPanel");
const LPTSTR c_szVALUE_CLASSICMENU = TEXT("ClassicStartMenu");
const LPTSTR c_szOEM_TITLEVAL = TEXT("DesktopShortcutsFolderName");
const LPTSTR c_szOEM_DISABLE = TEXT("DesktopShortcutsCleanupDisable");
const LPTSTR c_szOEM_SEVENDAY_DISABLE = TEXT("OemDesktopCleanupDisable");
extern HINSTANCE g_hInst;
STDMETHODIMP GetItemCLSID(IShellFolder2 *psf, LPCITEMIDLIST pidlLast, CLSID *pclsid);
//
// number of items to grow dsa by
//
const int c_GROWBYSIZE = 4;
//
// number of pages in the wizard
//
const int c_NUM_PAGES = 3;
//
// dialog prompt text length
//
const int c_MAX_PROMPT_TEXT = 1024;
const int c_MAX_HEADER_LEN = 64;
const int c_MAX_DATE_LEN = 40;
//
// file modified more than 60 days back is candidate for cleanup
// this value can be overriden by policy
//
const int c_NUMDAYSTODECAY = 60;
//
// Needed for hiding regitems
//
#define DEFINE_SCID(name, fmtid, pid) const SHCOLUMNID name = { fmtid, pid }
DEFINE_SCID(SCID_DESCRIPTIONID, PSGUID_SHELLDETAILS, PID_DESCRIPTIONID);
//
// pointer to member function typedef
//
typedef INT_PTR (STDMETHODCALLTYPE CCleanupWiz::* PCFC_DlgProcFn)(HWND, UINT, WPARAM, LPARAM);
//
// struct for helping us manage our dialog procs
//
typedef struct
{
CCleanupWiz * pcfc;
PCFC_DlgProcFn pfnDlgProc;
} DLGPROCINFO, *PDLGPROCINFO;
//
// enum for columns
//
typedef enum eColIndex
{
FC_COL_SHORTCUT,
FC_COL_DATE
};
//////////////////////////////////////////////////////
//
// 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
STDAPI_(void) GetFileTimeNDaysFromGivenTime(const FILETIME *pftGiven, FILETIME * pftReturn, int iDays)
{
__int64 i64 = *((__int64 *) pftGiven);
i64 += Int32x32To64(iDays*1000,FTsPerDayOver1000);
*pftReturn = *((FILETIME *) &i64);
}
STDAPI_(void) GetFileTimeNDaysFromCurrentTime(FILETIME *pf, int iDays)
{
SYSTEMTIME st;
FILETIME ftNow;
GetLocalTime(&st);
SystemTimeToFileTime(&st, &ftNow);
GetFileTimeNDaysFromGivenTime(&ftNow, pf, iDays);
}
/////////////////////////////////////////////////
//
//
// The CCleanupWiz class implementation
//
//
/////////////////////////////////////////////////
CCleanupWiz::CCleanupWiz(): _psf(NULL),
_hdsaItems(NULL),
_hTitleFont(NULL),
_iDeltaDays(0),
_dwCleanMode(CLEANUP_MODE_NORMAL)
{
INITCOMMONCONTROLSEX icce;
icce.dwSize = sizeof(icce);
icce.dwICC = ICC_LISTVIEW_CLASSES;
_bInited = InitCommonControlsEx(&icce) && SUCCEEDED(SHGetDesktopFolder(&_psf));
};
CCleanupWiz::~CCleanupWiz()
{
_CleanUpDSA();
ATOMICRELEASE(_psf);
};
STDMETHODIMP CCleanupWiz::Run(DWORD dwCleanMode, HWND hwndParent)
{
HRESULT hr;
if (!_bInited)
{
hr = E_FAIL;
}
else
{
_dwCleanMode = dwCleanMode;
if (CLEANUP_MODE_SILENT == _dwCleanMode)
{
hr = _RunSilent();
}
else
{
hr = _RunInteractive(hwndParent);
}
}
return hr;
}
STDMETHODIMP CCleanupWiz::_RunInteractive(HWND hwndParent)
{
HRESULT hr;
_iDeltaDays = GetNumDaysBetweenCleanup();
if (_iDeltaDays < 0)
{
_iDeltaDays = c_NUMDAYSTODECAY; //initial default value
}
LoadString(g_hInst, IDS_ARCHIVEFOLDER, _szFolderName, MAX_PATH);
// init the common control classes we need
hr = _LoadDesktopContents();
if (SUCCEEDED(hr))
{
UINT cItems = DSA_GetItemCount(_hdsaItems);
if (CLEANUP_MODE_NORMAL == _dwCleanMode)
{
if (cItems > 0) // if there are items, we want to notify and proceed only if the user wants us to.
{
hr = _ShowBalloonNotification();
}
else
{
hr = S_FALSE;
}
}
else
{
ASSERT(CLEANUP_MODE_ALL == _dwCleanMode);
hr = S_OK; // run manually, we show everything
}
if (S_OK == hr)
{
_cItemsOnDesktop = cItems;
hr = _InitializeAndLaunchWizard(hwndParent);
}
_LogUsage(); // set registry values to indicate the last run time
}
return hr;
}
//
// Creates the property pages for the wizard and launches the wizard
//
//
STDMETHODIMP CCleanupWiz::_InitializeAndLaunchWizard(HWND hwndParent)
{
HRESULT hr = S_OK;
DLGPROCINFO adpi[c_NUM_PAGES];
HPROPSHEETPAGE ahpsp[c_NUM_PAGES];
PROPSHEETPAGE psp = {0};
if (!_hTitleFont)
{
NONCLIENTMETRICS ncm = {0};
ncm.cbSize = sizeof(ncm);
SystemParametersInfo(SPI_GETNONCLIENTMETRICS, 0, &ncm, 0);
LOGFONT TitleLogFont = ncm.lfMessageFont;
TitleLogFont.lfWeight = FW_BOLD;
TCHAR szFont[128];
LoadString(g_hInst, IDS_TITLELOGFONT, szFont, ARRAYSIZE(szFont));
if (SUCCEEDED(StringCchCopy(TitleLogFont.lfFaceName, ARRAYSIZE(TitleLogFont.lfFaceName), szFont)))
{
HDC hdc = GetDC(NULL);
INT FontSize = 12;
TitleLogFont.lfHeight = 0 - GetDeviceCaps(hdc, LOGPIXELSY) * FontSize / 72;
_hTitleFont = CreateFontIndirect(&TitleLogFont);
ReleaseDC(NULL, hdc);
}
}
//
// Intro Page
//
adpi[0].pcfc = this;
adpi[0].pfnDlgProc = &CCleanupWiz::_IntroPageDlgProc;
psp.dwSize = sizeof(psp);
psp.dwFlags = PSP_DEFAULT|PSP_HIDEHEADER;
psp.hInstance = g_hInst;
psp.lParam = (LPARAM) &adpi[0];
psp.pfnDlgProc = s_StubDlgProc;
psp.pszTemplate = MAKEINTRESOURCE(IDD_INTRO);
ahpsp[0] = CreatePropertySheetPage(&psp);
//
// Choose files page
//
adpi[1].pcfc = this;
adpi[1].pfnDlgProc = &CCleanupWiz::_ChooseFilesPageDlgProc;
psp.hInstance = g_hInst;
psp.dwFlags = PSP_DEFAULT|PSP_USEHEADERTITLE| PSP_USEHEADERSUBTITLE;
psp.lParam = (LPARAM) &adpi[1];
psp.pszHeaderTitle = MAKEINTRESOURCE(IDS_CHOOSEFILES);
psp.pszHeaderSubTitle = MAKEINTRESOURCE(IDS_CHOOSEFILES_INFO);
psp.pszTemplate = MAKEINTRESOURCE(IDD_CHOOSEFILES);
psp.pfnDlgProc = s_StubDlgProc;
ahpsp[1] = CreatePropertySheetPage(&psp);
//
// Completion Page
//
adpi[2].pcfc = this;
adpi[2].pfnDlgProc = &CCleanupWiz::_FinishPageDlgProc;
psp.dwFlags = PSP_DEFAULT|PSP_HIDEHEADER;
psp.hInstance = g_hInst;
psp.lParam = (LPARAM) &adpi[2];
psp.pfnDlgProc = s_StubDlgProc;
psp.pszTemplate = MAKEINTRESOURCE(IDD_FINISH);
ahpsp[2] = CreatePropertySheetPage(&psp);
//
// The wizard property sheet
//
PROPSHEETHEADER psh = {0};
psh.dwSize = sizeof(psh);
psh.hInstance = g_hInst;
psh.hwndParent = hwndParent;
psh.phpage = ahpsp;
psh.dwFlags = PSH_WIZARD97|PSH_WATERMARK|PSH_HEADER;
psh.pszbmWatermark = MAKEINTRESOURCE(IDB_WATERMARK);
psh.pszbmHeader = MAKEINTRESOURCE(IDB_LOGO);
psh.nStartPage = _cItemsOnDesktop ? 0 : c_NUM_PAGES - 1; // if there are no pages on desktop, start on final page
psh.nPages = c_NUM_PAGES;
PropertySheet(&psh);
return hr;
}
//
// Pops up a balloon notification tip which asks the user
// if he wants to clean up the desktop.
//
// returns S_OK if user wants us to cleanup.
//
STDMETHODIMP CCleanupWiz::_ShowBalloonNotification()
{
IUserNotification *pun;
HRESULT hr = CoCreateInstance(CLSID_UserNotification, NULL,
CLSCTX_INPROC_SERVER, IID_PPV_ARG(IUserNotification, &pun));
if (SUCCEEDED(hr))
{
TCHAR szTitle[64], szMsg[256]; // we leave enough room for localization bloat
LoadString(g_hInst, IDS_NOTIFICATION_TITLE, szTitle, ARRAYSIZE(szTitle));
LoadString(g_hInst, IDS_NOTIFICATION_TEXT, szMsg, ARRAYSIZE(szMsg));
// these pun->Set functions can't fail...
pun->SetIconInfo(LoadIcon(g_hInst, MAKEINTRESOURCE(IDI_WIZ_ICON)), szTitle);
pun->SetBalloonInfo(szTitle, szMsg, NIIF_WARNING);
pun->SetBalloonRetry(20 * 1000, -1, 1); // try once, for 20 seconds
// returns S_OK if user wants to continue, ERROR_CANCELLED if timedout
// or canncelled otherwise.
hr = pun->Show(NULL, 0); // we don't support iquerycontinue, we will just wait
pun->Release();
}
return hr;
}
//
// Gets the list of items on the desktop that should be cleaned
//
// if dwCleanMode == CLEANUP_MODE_NORMAL, it only loads items which have not been used recently
// if dwCleanMode == CLEANUP_MODE_ALL, it loads all items on the desktop, marking those which have not been used recently
//
//
STDMETHODIMP CCleanupWiz::_LoadDesktopContents()
{
ASSERT(_psf);
ASSERT((CLEANUP_MODE_ALL == _dwCleanMode) || (CLEANUP_MODE_NORMAL == _dwCleanMode));
IEnumIDList * ppenum;
DWORD grfFlags = SHCONTF_NONFOLDERS;
HRESULT hr = _psf->EnumObjects(NULL, grfFlags, &ppenum);
if (SUCCEEDED(hr))
{
_CleanUpDSA();
_hdsaItems = DSA_Create(sizeof(FOLDERITEMDATA), c_GROWBYSIZE);
if (_hdsaItems)
{
ULONG celtFetched;
FOLDERITEMDATA fid = {0};
hr = S_OK;
while(SUCCEEDED(hr) && (S_OK == ppenum->Next(1,&fid.pidl, &celtFetched)))
{
if (_IsSupportedType(fid.pidl)) // only support links and regitems
{
// note, the call to _IsCandidateForRemoval also obtains the last
// used timestamp for the item
BOOL bShouldRemove = _IsCandidateForRemoval(fid.pidl, &fid.ftLastUsed);
if ( (CLEANUP_MODE_ALL == _dwCleanMode) || bShouldRemove)
{
SHFILEINFO sfi = {0};
if (SHGetFileInfo((LPCTSTR) fid.pidl,
0,
&sfi,
sizeof(sfi),
SHGFI_PIDL | SHGFI_DISPLAYNAME | SHGFI_ICON | SHGFI_SMALLICON ))
{
if (Str_SetPtr(&(fid.pszName), sfi.szDisplayName))
{
fid.hIcon = sfi.hIcon;
fid.bSelected = bShouldRemove;
if (-1 != DSA_AppendItem(_hdsaItems, &fid))
{
// All is well, the item has succesfully been added
// to the dsa, we zero out the fields so as not to
// free those resources now, they will be freed when the
// dsa is destroyed.
ZeroMemory(&fid, sizeof(fid));
continue;
}
else
{
hr = E_OUTOFMEMORY;
}
}
}
}
}
// Common cleanup path for various failure cases above,
// we did not add this item to the dsa, so cleanup now.
_CleanUpDSAItem(&fid);
}
//
// If we did not load any items into the dsa, we return S_FALSE
//
if (SUCCEEDED(hr))
{
hr = DSA_GetItemCount(_hdsaItems) > 0 ? S_OK : S_FALSE;
}
}
else
{
// we failed to allocate memory for the DSA
hr = E_OUTOFMEMORY;
}
ppenum->Release();
}
return hr;
}
//
// Gets the list of items on the desktop that should be cleaned
//
// if dwCleanMode == CLEANUP_MODE_SILENT, it loads all items on all desktops, marking them all
//
//
STDMETHODIMP CCleanupWiz::_LoadMergedDesktopContents()
{
ASSERT(_psf);
IEnumIDList * ppenum;
DWORD grfFlags = _dwCleanMode == CLEANUP_MODE_SILENT ? SHCONTF_FOLDERS | SHCONTF_NONFOLDERS: SHCONTF_NONFOLDERS;
HRESULT hr = _psf->EnumObjects(NULL, grfFlags, &ppenum);
if (SUCCEEDED(hr))
{
_CleanUpDSA();
_hdsaItems = DSA_Create(sizeof(FOLDERITEMDATA), c_GROWBYSIZE);
if (_hdsaItems)
{
ULONG celtFetched;
FOLDERITEMDATA fid = {0};
hr = S_OK;
while(SUCCEEDED(hr) && (S_OK == ppenum->Next(1,&fid.pidl, &celtFetched)))
{
if (_IsSupportedType(fid.pidl)) // only support links and regitems
{
// note, the call to _IsCandidateForRemoval also obtains the last
// used timestamp for the item
BOOL bShouldRemove = ((CLEANUP_MODE_SILENT == _dwCleanMode) ||
(_IsCandidateForRemoval(fid.pidl, &fid.ftLastUsed)));
if ( (CLEANUP_MODE_ALL == _dwCleanMode) || bShouldRemove)
{
SHFILEINFO sfi = {0};
if (SHGetFileInfo((LPCTSTR) fid.pidl,
0,
&sfi,
sizeof(sfi),
SHGFI_PIDL | SHGFI_DISPLAYNAME | SHGFI_ICON | SHGFI_SMALLICON ))
{
if (Str_SetPtr(&(fid.pszName), sfi.szDisplayName))
{
fid.hIcon = sfi.hIcon;
fid.bSelected = bShouldRemove;
if (-1 != DSA_AppendItem(_hdsaItems, &fid))
{
// All is well, the item has succesfully been added
// to the dsa, we zero out the fields so as not to
// free those resources now, they will be freed when the
// dsa is destroyed.
ZeroMemory(&fid, sizeof(fid));
continue;
}
else
{
hr = E_OUTOFMEMORY;
}
}
}
}
}
// Common cleanup path for various failure cases above,
// we did not add this item to the dsa, so cleanup now.
_CleanUpDSAItem(&fid);
}
//
// If we did not load any items into the dsa, we return S_FALSE
//
if (SUCCEEDED(hr))
{
hr = DSA_GetItemCount(_hdsaItems) > 0 ? S_OK : S_FALSE;
}
}
else
{
// we failed to allocate memory for the DSA
hr = E_OUTOFMEMORY;
}
ppenum->Release();
}
return hr;
}
//
// Expects the given pidl to be a link or regitem. Determines if it is a candidate for removal based
// on when was the last time it was used.
//
STDMETHODIMP_(BOOL) CCleanupWiz::_IsCandidateForRemoval(LPCITEMIDLIST pidl, FILETIME * pftLastUsed)
{
BOOL bRet = FALSE; // be conservative, if we do not know anything about the
// object we will not volunteer to remove it
int cHit = 0;
TCHAR szName[MAX_PATH];
ASSERT(_psf);
//
// we store UEM usage info for the regitems and links on the desktop
//
if (SUCCEEDED(DisplayNameOf(_psf,
pidl,
SHGDN_INFOLDER | SHGDN_FORPARSING,
szName,
ARRAYSIZE(szName))))
{
if (SUCCEEDED(_GetUEMInfo(-1, (LPARAM) szName, &cHit, pftLastUsed)))
{
FILETIME ft;
GetFileTimeNDaysFromCurrentTime(&ft, -_iDeltaDays);
#ifdef DEBUG
SYSTEMTIME st;
FileTimeToSystemTime(&ft, &st);
#endif
bRet = (CompareFileTime(pftLastUsed, &ft) < 0);
}
}
return bRet;
}
STDMETHODIMP CCleanupWiz::_ShouldShow(IShellFolder* psf, LPCITEMIDLIST pidlFolder, LPCITEMIDLIST pidlItem)
{
HRESULT hr;
IShellFolder2 *psf2;
hr = psf->QueryInterface(IID_PPV_ARG(IShellFolder2, &psf2));
if (SUCCEEDED(hr))
{
// Get the GUID in the pidl, which requires IShellFolder2.
CLSID guidItem;
hr = GetItemCLSID(psf2, pidlItem, &guidItem);
if (SUCCEEDED(hr))
{
SHELLSTATE ss = {0};
SHGetSetSettings(&ss, SSF_STARTPANELON, FALSE); //See if the StartPanel is on!
TCHAR szRegPath[MAX_PATH];
//Get the proper registry path based on if StartPanel is ON/OFF
hr = StringCchPrintf(szRegPath, ARRAYSIZE(szRegPath), REGSTR_PATH_HIDDEN_DESKTOP_ICONS, (ss.fStartPanelOn ? c_szVALUE_STARTPANEL : c_szVALUE_CLASSICMENU));
if (SUCCEEDED(hr))
{
//Convert the guid to a string
TCHAR szGuidValue[MAX_GUID_STRING_LEN];
SHStringFromGUID(guidItem, szGuidValue, ARRAYSIZE(szGuidValue));
//See if this item is turned off in the registry.
if (SHRegGetBoolUSValue(szRegPath, szGuidValue, FALSE, /* default */FALSE))
hr = S_FALSE; //They want to hide it; So, return S_FALSE.
if (SHRestricted(REST_NOMYCOMPUTERICON) && IsEqualCLSID(CLSID_MyComputer, guidItem))
hr = S_FALSE;
}
}
psf2->Release();
}
// if we fail for some reason, be generous and say that we should offer to clean this up
if (FAILED(hr))
{
hr = S_OK;
}
return hr;
}
//
// Normal, All: We only support removing regitems and links from the desktop
// Silent: WE NEVER GET HERE FROM SILENT
//
STDMETHODIMP_(BOOL) CCleanupWiz::_IsSupportedType(LPCITEMIDLIST pidl)
{
ASSERT(_dwCleanMode != CLEANUP_MODE_SILENT);
BOOL fRetVal = FALSE;
eFILETYPE eType = _GetItemType(pidl);
if (FC_TYPE_REGITEM == eType) // handle regitems
{
fRetVal = TRUE;
if (S_OK != _ShouldShow(_psf, NULL, pidl)) // regitems must succeed _ShouldShow
{
fRetVal = FALSE;
}
else
{
IShellFolder2 *psf2;
CLSID guidItem;
if (SUCCEEDED(_psf->QueryInterface(IID_PPV_ARG(IShellFolder2, &psf2)))) // must not be one of the shell desktop items
{
if (SUCCEEDED(GetItemCLSID(psf2, pidl, &guidItem)) &&
(IsEqualCLSID(CLSID_MyComputer, guidItem) ||
IsEqualCLSID(CLSID_MyDocuments, guidItem) ||
IsEqualCLSID(CLSID_NetworkPlaces, guidItem) ||
IsEqualCLSID(CLSID_RecycleBin, guidItem)))
{
fRetVal = FALSE;
}
psf2->Release();
}
}
}
else if (FC_TYPE_LINK == eType) // handle links
{
fRetVal = TRUE;
TCHAR szName[MAX_PATH];
if (SUCCEEDED(SHGetNameAndFlags(pidl, SHGDN_INFOLDER | SHGDN_FORPARSING, szName, ARRAYSIZE(szName), NULL)))
{
if (!lstrcmp(szName, _szFolderName)) // must not be the folder we're cleaning things up to
{
fRetVal = FALSE;
}
}
}
return fRetVal;
}
//
// Returns the type of the pidl.
// We are only interested in Links and Regitems, so we return FC_TYPE_OTHER for
// all other items.
//
STDMETHODIMP_(eFILETYPE) CCleanupWiz::_GetItemType(LPCITEMIDLIST pidl)
{
eFILETYPE eftVal = FC_TYPE_OTHER;
TCHAR szName[MAX_PATH];
TCHAR szPath[MAX_PATH];
IShellLink *psl;
ASSERT(_psf);
if (FAILED(SHGetNameAndFlags(pidl, SHGDN_INFOLDER | SHGDN_FORPARSING, szName, ARRAYSIZE(szName), NULL)))
{
szName[0] = 0;
}
if (FAILED(SHGetNameAndFlags(pidl, SHGDN_FORPARSING, szPath, ARRAYSIZE(szPath), NULL)))
{
szPath[0] = 0;
}
if(_IsRegItemName(szName))
{
eftVal = FC_TYPE_REGITEM;
}
else if (SUCCEEDED( _psf->GetUIObjectOf(NULL,
1,
&pidl,
IID_PPV_ARG_NULL(IShellLink, &psl))))
{
eftVal = FC_TYPE_LINK;
psl->Release();
}
else if (PathIsDirectory(szPath))
{
eftVal = FC_TYPE_FOLDER;
}
else
{
//
// Maybe this item is a kind of .{GUID} object we created to restore
// regitems. In that case we want to actually restore the regitem
// at this point by marking it as unhidden.
//
LPTSTR pszExt = PathFindExtension(szName);
if (TEXT('.') == *pszExt // it is a file extension
&& lstrlen(++pszExt) == (GUIDSTR_MAX - 1) // AND the extension is of the right length
// note: GUIDSTR_MAX includes the terminating NULL
// while lstrlen does not, hence the expression
&& TEXT('{') == *pszExt) // AND looks like it is a guid...
{
// we most prob have a bonafide guid string
// pszExt now points to the beginning of the GUID string
TCHAR szGUIDName[ARRAYSIZE(TEXT("::")) + GUIDSTR_MAX];
// put it in the regitem SHGDN_FORPARSING name format, which is like
//
// "::{16B280C6-EE70-11D1-9066-00C04FD9189D}"
//
if (SUCCEEDED(StringCchPrintf(szGUIDName, ARRAYSIZE(szGUIDName), TEXT("::%s"), pszExt)))
{
LPITEMIDLIST pidlGUID;
DWORD dwAttrib = SFGAO_NONENUMERATED;
//
// get the pidl of the regitem, if this call succeeds, it means we do have
// a corresponding regitem in the desktop's namespace
//
if (SUCCEEDED(_psf->ParseDisplayName(NULL,
NULL,
szGUIDName,
NULL,
&pidlGUID,
&dwAttrib)))
{
//
// check if the regitem is marked as hidden
//
if (dwAttrib & SFGAO_NONENUMERATED)
{
//
// One last check before we enable the regitem:
// Does the regitem have the same display name as the .CLSID file.
// In case the user has restored this .CLSID file and renamed it we will
// not attempt to restore the regitem as it may confuse the user.
//
TCHAR szNameRegItem[MAX_PATH];
if (SUCCEEDED((DisplayNameOf(_psf,
pidl,
SHGDN_NORMAL,
szName,
ARRAYSIZE(szName)))) &&
SUCCEEDED((DisplayNameOf(_psf,
pidlGUID,
SHGDN_NORMAL,
szNameRegItem,
ARRAYSIZE(szNameRegItem)))) &&
lstrcmp(szName, szNameRegItem) == 0)
{
if (SUCCEEDED(_HideRegPidl(pidlGUID, FALSE)))
{
// delete the file corresponding to the regitem
if (SUCCEEDED(DisplayNameOf(_psf,
pidl,
SHGDN_NORMAL | SHGDN_FORPARSING,
szName,
ARRAYSIZE(szName))))
{
DeleteFile(szName); // too bad if we fail, we will just
// have two identical icons on the desktop
}
//
// Log the current time as the last used time of the regitem.
// We just re-enabled this regitem but we do not have the
// usage info for the corresponding .{CLSID} which the user had
// been using so far. So we will be conservative and say that
// it was used right now, so that it does not become a candidate
// for removal soon. As this is a regitem that the user restored
// after the wizard removed it, so it is a fair assumption that
// the user has used it after restoring it and is not a candidate
// for cleanup right now.
//
UEMFireEvent(&UEMIID_SHELL, UEME_RUNPATH, UEMF_XEVENT, -1, (LPARAM)szGUIDName);
}
}
}
ILFree(pidlGUID);
}
}
}
}
return eftVal;
}
//
// Determines if a filename is that of a regitem
//
// a regitem's SHGDN_INFOLDER | SHGDN_FORPARSING name is always "::{someguid}"
//
// CDefview::_LogDesktopLinksAndRegitems() uses the same test to determine
// if a given pidl is a regitem. This case can lead to false positives if
// you have other items on the desktop which have infoder parsing names
// beginning with "::{", but as ':' is not presently allowed in filenames
// it should not be a problem.
//
STDMETHODIMP_(BOOL) CCleanupWiz::_IsRegItemName(LPTSTR pszName)
{
return (pszName[0] == TEXT(':') && pszName[1] == TEXT(':') && pszName[2] == TEXT('{'));
}
STDMETHODIMP_(BOOL) CCleanupWiz::_CreateFakeRegItem(LPCTSTR pszDestPath, LPCTSTR pszName, LPCTSTR pszGUID)
{
BOOL fRetVal = FALSE;
TCHAR szLinkName[MAX_PATH];
if (SUCCEEDED(StringCchCopy(szLinkName, ARRAYSIZE(szLinkName), pszDestPath)) &&
PathAppend(szLinkName, pszName) &&
SUCCEEDED(StringCchCat(szLinkName, ARRAYSIZE(szLinkName), TEXT("."))) &&
SUCCEEDED(StringCchCat(szLinkName, ARRAYSIZE(szLinkName), pszGUID)))
{
//
// We use the CREATE_ALWAYS flag so that if the file already exists
// in the Unused Desktop Files folder, we will go ahead and hide the
// regitem.
//
HANDLE hFile = CreateFile(szLinkName, 0, FILE_SHARE_READ, NULL,
CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
if (hFile != INVALID_HANDLE_VALUE)
{
// we created/opened the shortcut, now hide the regitem and close
// the shortcut file
fRetVal = TRUE;
CloseHandle(hFile);
}
}
return fRetVal;
}
//
// Given path to an exe, returns the UEM hit count and last used date for it
//
STDMETHODIMP CCleanupWiz::_GetUEMInfo(WPARAM wParam, LPARAM lParam, int * pcHit, FILETIME * pftLastUsed)
{
UEMINFO uei;
uei.cbSize = sizeof(uei);
uei.dwMask = UEIM_HIT | UEIM_FILETIME;
HRESULT hr = UEMQueryEvent(&UEMIID_SHELL, UEME_RUNPATH, wParam, lParam, &uei);
if (SUCCEEDED(hr))
{
*pcHit = uei.cHit;
*pftLastUsed = uei.ftExecute;
}
return hr;
}
STDMETHODIMP_(BOOL) CCleanupWiz::_ShouldProcess()
{
BOOL fRetVal = FALSE;
if (_dwCleanMode == CLEANUP_MODE_SILENT)
{
fRetVal = TRUE;
}
else
{
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
if (pfid && pfid->bSelected)
{
fRetVal = TRUE;
break;
}
}
}
return fRetVal;
}
//
// Process the list of items. At this point _hdsaItems only contains the
// items that the user wants to delete
//
STDMETHODIMP CCleanupWiz::_ProcessItems()
{
TCHAR szFolderLocation[MAX_PATH]; // desktop folder
HRESULT hr = S_OK;
if (_ShouldProcess())
{
LPITEMIDLIST pidlCommonDesktop = NULL;
ASSERT(_psf);
// use the archive folder on the desktop
if (CLEANUP_MODE_SILENT != _dwCleanMode)
{
hr = DisplayNameOf(_psf, NULL, SHGDN_FORPARSING, szFolderLocation, ARRAYSIZE(szFolderLocation));
}
else // use the archive folder in Program Files
{
hr = SHGetFolderLocation(NULL, CSIDL_PROGRAM_FILES , NULL, 0, &pidlCommonDesktop);
if (SUCCEEDED(hr))
{
hr = DisplayNameOf(_psf, pidlCommonDesktop, SHGDN_FORPARSING, szFolderLocation, ARRAYSIZE(szFolderLocation));
}
}
if (SUCCEEDED(hr))
{
ASSERTMSG(*_szFolderName, "Desktop Cleaner: Archive Folder Name not present");
// create the full path of the archive folder
TCHAR szFolderPath[MAX_PATH];
hr = StringCchCopy(szFolderPath, ARRAYSIZE(szFolderPath), szFolderLocation);
if (SUCCEEDED(hr))
{
if (!PathAppend(szFolderPath, _szFolderName))
{
hr = E_FAIL;
}
else
{
//
// We have to make sure that this folder exists, as otherwise, if we try to move
// a single shortcut using SHFileOperation, that file will be renamed to the target
// name instead of being put in a folder with that name.
//
SECURITY_ATTRIBUTES sa;
sa.nLength = sizeof (sa);
sa.lpSecurityDescriptor = NULL; // we get the default attributes for this process
sa.bInheritHandle = FALSE;
int iRetVal = SHCreateDirectoryEx(NULL, szFolderPath, &sa);
if (ERROR_SUCCESS == iRetVal || ERROR_FILE_EXISTS == iRetVal || ERROR_ALREADY_EXISTS == iRetVal)
{
DblNulTermList dnSourceFiles;
TCHAR szFileName[MAX_PATH + 1]; // to pad an extra null char for SHFileOpStruct
//
//
// now we can start on the files we need to move
//
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
if (pfid && (pfid->bSelected || _dwCleanMode == CLEANUP_MODE_SILENT) &&
SUCCEEDED(DisplayNameOf(_psf,pfid->pidl,
SHGDN_FORPARSING,
szFileName,
ARRAYSIZE(szFileName) - 1)))
{
if (_IsRegItemName(szFileName))
{
// if its a regitem, we create a "Item Name.{GUID}" file
// and mark the regitem as hidden.
//
if (_CreateFakeRegItem(szFolderPath, pfid->pszName, szFileName+2))
{
_HideRegPidl(pfid->pidl, TRUE);
}
}
else // not a regitem, will move it
{
dnSourceFiles.AddString(szFileName);
}
}
}
if (dnSourceFiles.Count() > 0)
{
DblNulTermList dnTargetFolder;
dnTargetFolder.AddString(szFolderPath);
SHFILEOPSTRUCT sfo;
sfo.hwnd = NULL;
sfo.wFunc = FO_MOVE;
sfo.pFrom = (LPCTSTR) dnSourceFiles;
sfo.pTo = (LPCTSTR) dnTargetFolder;
sfo.fFlags = FOF_NORECURSION | FOF_NOCONFIRMMKDIR | FOF_ALLOWUNDO ;
hr = SHFileOperation(&sfo);
}
SHChangeNotify(SHCNE_UPDATEDIR, SHCNF_PATH, (LPCVOID) szFolderPath, 0);
SHChangeNotify(SHCNE_UPDATEDIR, SHCNF_PATH, (LPCVOID) szFolderLocation, 0);
}
else
{
// we failed to create the Unused Desktop Files folder
hr = E_FAIL;
}
}
}
}
}
return hr;
}
////////////////////////////////////////////////////////
//
// DialogProcs
//
// TODO: test for accessibilty issues
//
////////////////////////////////////////////////////////
INT_PTR STDMETHODCALLTYPE CCleanupWiz::_IntroPageDlgProc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam)
{
INT_PTR ipRet = FALSE;
switch (wMsg)
{
case WM_INITDIALOG:
{
HWND hWnd = GetDlgItem(hDlg, IDC_TEXT_TITLE_WELCOME);
if (_hTitleFont)
{
SetWindowFont(hWnd, _hTitleFont, TRUE);
}
}
break;
case WM_NOTIFY :
{
LPNMHDR lpnm = (LPNMHDR) lParam;
switch (lpnm->code)
{
case PSN_SETACTIVE:
PropSheet_SetWizButtons(GetParent(hDlg), PSWIZB_NEXT);
break;
}
break;
}
}
return ipRet;
}
INT_PTR STDMETHODCALLTYPE CCleanupWiz::_ChooseFilesPageDlgProc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam)
{
INT_PTR ipRet = FALSE;
HWND hwLV = NULL;
switch (wMsg)
{
case WM_INITDIALOG:
_InitChoosePage(hDlg);
ipRet = TRUE;
break;
case WM_NOTIFY :
LPNMHDR lpnm = (LPNMHDR) lParam;
switch (lpnm->code)
{
case PSN_SETACTIVE:
PropSheet_SetWizButtons(GetParent(hDlg), PSWIZB_NEXT | PSWIZB_BACK);
hwLV = GetDlgItem(hDlg, IDC_LV_PROMPT);
_SetCheckedState(hwLV);
break;
case PSN_WIZNEXT:
// remember the items the user selected
hwLV = GetDlgItem(hDlg, IDC_LV_PROMPT);
_MarkSelectedItems(hwLV);
break;
case PSN_WIZBACK:
// remember the items the user selected
hwLV = GetDlgItem(hDlg, IDC_LV_PROMPT);
_MarkSelectedItems(hwLV);
break;
}
break;
}
return ipRet;
}
INT_PTR STDMETHODCALLTYPE CCleanupWiz::_FinishPageDlgProc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam)
{
INT_PTR ipRet = FALSE;
switch (wMsg)
{
case WM_INITDIALOG:
_InitFinishPage(hDlg);
ipRet = TRUE;
break;
case WM_NOTIFY :
{
LPNMHDR lpnm = (LPNMHDR) lParam;
switch (lpnm->code)
{
case PSN_SETACTIVE:
PropSheet_SetWizButtons(GetParent(hDlg), _cItemsOnDesktop ? PSWIZB_BACK | PSWIZB_FINISH : PSWIZB_FINISH);
// selection can change so need to do this everytime you come to this page
_RefreshFinishPage(hDlg);
break;
case PSN_WIZFINISH:
// process the items now
_ProcessItems();
break;
}
break;
}
}
return ipRet;
}
//
// stub dialog proc which redirects calls to the right dialog procs
//
INT_PTR CALLBACK CCleanupWiz::s_StubDlgProc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam)
{
PDLGPROCINFO pInfo = (PDLGPROCINFO) GetWindowLongPtr(hDlg, DWLP_USER);
if (WM_INITDIALOG == wMsg)
{
pInfo = (PDLGPROCINFO) ((LPPROPSHEETPAGE) lParam) -> lParam;
SetWindowLongPtr(hDlg, DWLP_USER, (LPARAM) pInfo);
}
if (pInfo)
{
CCleanupWiz * pThis = pInfo->pcfc;
PCFC_DlgProcFn pfn = pInfo->pfnDlgProc;
return (pThis->*pfn)(hDlg, wMsg, wParam, lParam);
}
return FALSE;
}
STDMETHODIMP CCleanupWiz::_InitListBox(HWND hWndListView)
{
ListView_SetExtendedListViewStyle(hWndListView, LVS_EX_SUBITEMIMAGES);
//
// add the columns
//
LVCOLUMN lvcDate;
TCHAR szDateHeader[c_MAX_HEADER_LEN];
lvcDate.mask = LVCF_SUBITEM | LVCF_WIDTH | LVCF_TEXT ;
lvcDate.iSubItem = FC_COL_SHORTCUT;
lvcDate.cx = 200;
LoadString(g_hInst, IDS_HEADER_ITEM, szDateHeader, ARRAYSIZE(szDateHeader));
lvcDate.pszText = szDateHeader;
ListView_InsertColumn(hWndListView, FC_COL_SHORTCUT, &lvcDate);
lvcDate.mask = LVCF_SUBITEM | LVCF_FMT | LVCF_WIDTH | LVCF_TEXT ;
lvcDate.iSubItem = FC_COL_DATE;
lvcDate.fmt = LVCFMT_LEFT;
lvcDate.cx = 1;
LoadString(g_hInst, IDS_HEADER_DATE, szDateHeader, ARRAYSIZE(szDateHeader));
lvcDate.pszText = szDateHeader;
ListView_InsertColumn(hWndListView, FC_COL_DATE, &lvcDate);
ListView_SetColumnWidth(hWndListView, FC_COL_DATE, LVSCW_AUTOSIZE_USEHEADER);
return S_OK;
}
STDMETHODIMP CCleanupWiz::_InitChoosePage(HWND hDlg)
{
HWND hWndListView = GetDlgItem(hDlg, IDC_LV_PROMPT);
_InitListBox(hWndListView);
//
// add the images
//
HIMAGELIST hSmall = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
GetSystemMetrics(SM_CYSMICON),
ILC_MASK | ILC_COLOR32 , c_GROWBYSIZE, c_GROWBYSIZE);
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
if (pfid)
{
ImageList_AddIcon(hSmall, pfid->hIcon);
}
}
ListView_SetImageList(hWndListView, hSmall, LVSIL_SMALL);
//
// set the checkboxes style
//
ListView_SetExtendedListViewStyleEx(hWndListView, LVS_EX_CHECKBOXES, LVS_EX_CHECKBOXES);
_PopulateListView(hWndListView);
return S_OK;
}
STDMETHODIMP CCleanupWiz::_InitFinishPage(HWND hDlg)
{
HWND hWnd = GetDlgItem(hDlg, IDC_TEXT_TITLE_WELCOME);
if (_hTitleFont)
{
SetWindowFont(hWnd, _hTitleFont, TRUE);
}
HIMAGELIST hSmall = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
GetSystemMetrics(SM_CYSMICON),
ILC_MASK | ILC_COLOR32, c_GROWBYSIZE, c_GROWBYSIZE);
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
if (pfid)
{
ImageList_AddIcon(hSmall, pfid->hIcon);
}
}
ListView_SetImageList(GetDlgItem(hDlg, IDC_LV_INFORM), hSmall, LVSIL_SMALL);
return S_OK;
}
STDMETHODIMP CCleanupWiz::_RefreshFinishPage(HWND hDlg)
{
HRESULT hr;
HWND hWndListView = GetDlgItem(hDlg, IDC_LV_INFORM);
ListView_DeleteAllItems(hWndListView);
int cMovedItems = _PopulateListViewFinish(hWndListView);
// set the informative text to reflect how many items were moved
HWND hWnd = GetDlgItem(hDlg, IDC_TEXT_INFORM);
TCHAR szDisplayText[c_MAX_PROMPT_TEXT];
ShowWindow(GetDlgItem(hDlg, IDC_LV_INFORM), BOOLIFY(cMovedItems));
ShowWindow(GetDlgItem(hDlg, IDC_TEXT_SHORTCUTS), BOOLIFY(cMovedItems));
ShowWindow(GetDlgItem(hDlg, IDC_TEXT_CHANGE), BOOLIFY(cMovedItems));
if ( 0 == cMovedItems)
{
LoadString(g_hInst, _cItemsOnDesktop ? IDS_INFORM_NONE : IDS_INFORM_NONEFOUND,
szDisplayText, ARRAYSIZE(szDisplayText));
hr = S_OK;
}
else if (1 == cMovedItems)
{
LoadString(g_hInst, IDS_INFORM_SINGLE,
szDisplayText, ARRAYSIZE(szDisplayText));
hr = S_OK;
}
else
{
TCHAR szRawText[c_MAX_PROMPT_TEXT];
LoadString(g_hInst, IDS_INFORM, szRawText, ARRAYSIZE(szRawText));
hr = StringCchPrintf(szDisplayText, ARRAYSIZE(szDisplayText), szRawText, cMovedItems);
}
SetWindowText(hWnd, szDisplayText);
return hr;
}
STDMETHODIMP_(int) CCleanupWiz::_PopulateListView(HWND hWndListView)
{
LVITEM lvi = {0};
int cRet = 0;
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
lvi.mask = LVIF_TEXT | LVIF_IMAGE;
lvi.pszText = pfid->pszName;
lvi.iImage = i;
lvi.iItem = i;
lvi.iSubItem = FC_COL_SHORTCUT;
ListView_InsertItem(hWndListView, &lvi);
cRet++;
// set the last used date
TCHAR szDate[c_MAX_DATE_LEN];
if (SUCCEEDED(_GetDateFromFileTime(pfid->ftLastUsed, szDate, ARRAYSIZE(szDate))))
{
ListView_SetItemText(hWndListView, i, FC_COL_DATE, szDate);
}
}
return cRet;
}
STDMETHODIMP_(int) CCleanupWiz::_PopulateListViewFinish(HWND hWndListView)
{
LVITEM lvi = {0};
lvi.mask = LVIF_TEXT | LVIF_IMAGE ;
int cRet = 0;
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
//
// it's the Finish Page, we only show the items we were asked to move
//
if (pfid && pfid->bSelected)
{
lvi.pszText = pfid->pszName;
lvi.iImage = i;
lvi.iItem = i;
ListView_InsertItem(hWndListView, &lvi);
cRet++;
}
}
return cRet;
}
//
// Converts a given FILETIME date into s displayable string
//
STDMETHODIMP CCleanupWiz::_GetDateFromFileTime(FILETIME ftLastUsed, LPTSTR pszDate, int cch )
{
HRESULT hr;
if (0 == ftLastUsed.dwHighDateTime && 0 == ftLastUsed.dwLowDateTime)
{
LoadString(g_hInst, IDS_NEVER, pszDate, cch);
hr = S_OK;
{
hr = E_FAIL;
}
}
else
{
DWORD dwFlags = FDTF_SHORTDATE;
if (0 == SHFormatDateTime(&ftLastUsed, &dwFlags, pszDate, cch))
{
hr = E_FAIL;
}
else
{
hr = S_OK;
}
}
return hr;
}
//
// Marks listview items as checked or unchecked
//
STDMETHODIMP CCleanupWiz::_SetCheckedState(HWND hWndListView)
{
int cItems = DSA_GetItemCount(_hdsaItems);
for (int i = 0; i < cItems; i++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, i);
if (pfid)
{
ListView_SetCheckState(hWndListView, i, pfid->bSelected);
}
}
return S_OK;
}
//
// Reverse of above, updates our list based on user selection.
//
STDMETHODIMP CCleanupWiz::_MarkSelectedItems(HWND hWndListView)
{
int cItems = ListView_GetItemCount(hWndListView);
for (int iLV = 0; iLV < cItems; iLV++)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems, iLV);
if (pfid)
{
pfid->bSelected = ListView_GetCheckState(hWndListView, iLV);
}
}
return S_OK;
}
//
// These methods clean up _hdsaItems and free the allocated memory
//
STDMETHODIMP_(void) CCleanupWiz::_CleanUpDSA()
{
if (_hdsaItems != NULL)
{
for (int i = DSA_GetItemCount(_hdsaItems)-1; i >= 0; i--)
{
FOLDERITEMDATA * pfid = (FOLDERITEMDATA *) DSA_GetItemPtr(_hdsaItems,i);
if (pfid)
{
_CleanUpDSAItem(pfid);
}
}
DSA_Destroy(_hdsaItems);
_hdsaItems = NULL;
}
}
STDMETHODIMP CCleanupWiz::_CleanUpDSAItem(FOLDERITEMDATA * pfid)
{
if (pfid->pidl)
{
ILFree(pfid->pidl);
}
if (pfid->pszName)
{
Str_SetPtr(&(pfid->pszName), NULL);
}
if (pfid->hIcon)
{
DestroyIcon(pfid->hIcon);
}
ZeroMemory(pfid, sizeof(*pfid));
return S_OK;
}
//////////////////////
//
// Hide regitems
//
//////////////////////
//
// Helper routines used below.
// Cloned from shell32/util.cpp
//
STDMETHODIMP GetItemCLSID(IShellFolder2 *psf, LPCITEMIDLIST pidlLast, CLSID *pclsid)
{
VARIANT var;
HRESULT hr = psf->GetDetailsEx(pidlLast, &SCID_DESCRIPTIONID, &var);
if (SUCCEEDED(hr))
{
SHDESCRIPTIONID did;
hr = VariantToBuffer(&var, (void *)&did, sizeof(did));
if (SUCCEEDED(hr))
*pclsid = did.clsid;
VariantClear(&var);
}
return hr;
}
//
// Given a regitem, it sets the SFGAO_NONENUMERATED bit on it so that
// that it no longer shows up in the folder.
//
// Since we are primarily only interested in cleaning up desktop clutter,
// that means we don't have to worry about all possible kinds of regitems.
// Our main target is apps like Outlook which create regitems instead of
// .lnk shortcuts. So our code does not have to be as complex as the
// regfldr.cpp version for deleting regitems, which has to account for
// everything, from legacy regitems to delegate folders.
//
//
STDMETHODIMP CCleanupWiz::_HideRegPidl(LPCITEMIDLIST pidlr, BOOL fHide)
{
IShellFolder2 *psf2;
HRESULT hr = _psf->QueryInterface(IID_PPV_ARG(IShellFolder2, &psf2));
if (SUCCEEDED(hr))
{
CLSID clsid;
hr = GetItemCLSID(psf2, pidlr, &clsid);
if (SUCCEEDED(hr))
{
hr = _HideRegItem(&clsid, fHide, NULL);
}
psf2->Release();
}
return hr;
}
STDMETHODIMP CCleanupWiz::_HideRegItem(CLSID* pclsid, BOOL fHide, BOOL* pfWasVisible)
{
HKEY hkey;
if (pfWasVisible)
{
*pfWasVisible = FALSE;
}
HRESULT hr = SHRegGetCLSIDKey(*pclsid, TEXT("ShellFolder"), FALSE, TRUE, &hkey);
if(SUCCEEDED(hr))
{
DWORD dwAttr, dwErr;
DWORD dwType = 0;
DWORD cbSize = sizeof(dwAttr);
if (ERROR_SUCCESS == RegQueryValueEx(hkey, TEXT("Attributes"), NULL, &dwType, (BYTE *) &dwAttr, &cbSize))
{
if (pfWasVisible)
{
*pfWasVisible = !(dwAttr & SFGAO_NONENUMERATED);
}
fHide ? dwAttr |= SFGAO_NONENUMERATED : dwAttr &= ~SFGAO_NONENUMERATED;
}
else
{
// attributes do not exist, so we will try creating them
fHide ? dwAttr = SFGAO_NONENUMERATED : dwAttr = 0;
}
dwErr = RegSetValueEx(hkey, TEXT("Attributes"), NULL, dwType, (BYTE *) &dwAttr, cbSize);
hr = HRESULT_FROM_WIN32(dwErr);
RegCloseKey(hkey);
}
return hr;
}
//
// Method writes out the last used time in the registry and the
// number of days it was checkin for
//
STDMETHODIMP CCleanupWiz::_LogUsage()
{
FILETIME ft;
SYSTEMTIME st;
GetLocalTime(&st);
SystemTimeToFileTime(&st, &ft);
//
// we ignore if any of these calls fail, as we cannot really do anything
// in that case. the next time we run, we will run maybe sooner that expected.
//
SHRegSetUSValue(REGSTR_PATH_CLEANUPWIZ, c_szVAL_TIME,
REG_BINARY, &ft, sizeof(ft),
SHREGSET_FORCE_HKCU);
SHRegSetUSValue(REGSTR_PATH_CLEANUPWIZ, c_szVAL_DELTA_DAYS,
REG_DWORD,(DWORD *) &_iDeltaDays, sizeof(_iDeltaDays),
SHREGSET_FORCE_HKCU);
//
// TODO: also write out to log file here
//
return S_OK;
}
//
// returns the current value from the policy key or the user settings
//
STDMETHODIMP_(int) CCleanupWiz::GetNumDaysBetweenCleanup()
{
DWORD dwData;
DWORD dwType;
DWORD cch = sizeof (DWORD);
int iDays = -1; // if the value does not exist
//
// ISSUE-2000/12/01-AIDANL Removed REGSTR_POLICY_CLEANUP, don't think we need both, but want to
// leave this note in case issues come up later.
//
if (ERROR_SUCCESS == (SHRegGetUSValue(REGSTR_PATH_CLEANUPWIZ, c_szVAL_DELTA_DAYS,
&dwType, &dwData, &cch,FALSE, NULL, 0)))
{
iDays = dwData;
}
return iDays;
}
// helper functions
STDAPI_(BOOL) IsUserAGuest()
{
return SHTestTokenMembership(NULL, DOMAIN_ALIAS_RID_GUESTS);
}