windows-nt/Source/XPSP1/NT/shell/cpls/plus/store.cpp
2020-09-26 16:20:57 +08:00

630 lines
19 KiB
C++

//---------------------------------------------------------------------------
//
// File: plustab.cpp
//
// Main Implementation of the Effects page
//
//---------------------------------------------------------------------------
#include "precomp.hxx"
#include "shlwapip.h"
#include "shlguidp.h"
#include "EffectsAdvPg.h"
#include "store.h"
#pragma hdrstop
// OLE-Registry magic number
GUID g_CLSID_CplExt = { 0x41e300e0, 0x78b6, 0x11ce,
{ 0x84, 0x9b, 0x44, 0x45, 0x53, 0x54, 0x00, 0x00}
};
#define SPI_GETKEYBOARDINDICATORS SPI_GETMENUUNDERLINES//0x100A
#define SPI_SETKEYBOARDINDICATORS SPI_SETMENUUNDERLINES//0x100B
#define SPI_GETFONTCLEARTYPE 116
#define SPI_SETFONTCLEARTYPE 117
//Defines for getting registry settings
const TCHAR c_szHICKey[] = TEXT("Control Panel\\Desktop\\WindowMetrics"); // show icons using highest possible colors
const TCHAR c_szHICVal[] = TEXT("Shell Icon BPP"); // (4 if the checkbox is false, otherwise 16, don't set it to anything else)
const TCHAR c_szSSKey[] = TEXT("Control Panel\\Desktop");
const TCHAR c_szSSVal[] = TEXT("SmoothScroll");
const TCHAR c_szWMSISVal[] = TEXT("Shell Icon Size"); // Normal Icon Size (default 32 x 32)
#ifdef INET_EXP_ICON
const TCHAR c_szIEXP[] = TEXT("\\Program Files\\Microsoft Internet Explorer 4");
#endif
// Handle to the DLL
extern HINSTANCE g_hInst;
BOOL g_bMirroredOS = FALSE;
// vars needed for new shell api
#define SZ_SHELL32 TEXT("shell32.dll")
#define SZ_SHUPDATERECYCLEBINICON "SHUpdateRecycleBinIcon" // Parameter for GetProcAddr()... DO NOT TEXT("") IT!
HINSTANCE g_hmodShell32 = NULL;
typedef void (* PFNSHUPDATERECYCLEBINICON)( void );
// Function prototype
BOOL CALLBACK PlusPackDlgProc( HWND hDlg, UINT uMessage, WPARAM wParam, LPARAM lParam );
// Icon Stuff
int GetIconState (void);
BOOL ChangeIconSizes(int iOldState, int iNewState);
// animation stuff
WPARAM GetAnimations(DWORD *pdwEffect);
void SetAnimations(WPARAM wVal, DWORD dwEffect);
BOOL DisplayFontSmoothingDetails(DWORD *pdwSetting)
{
return FALSE;
}
int GetBitsPerPixel(void)
{
int iBitsPerPixel = 8;
HDC hDC = GetDC(NULL);
if (hDC)
{
iBitsPerPixel = GetDeviceCaps(hDC, BITSPIXEL);
ReleaseDC(NULL, hDC);
}
return iBitsPerPixel;
}
int GetIconState(void)
{
BOOL bRet;
int iSize;
bRet = GetRegValueInt(HKEY_CURRENT_USER, c_szHICKey, c_szWMSISVal, &iSize);
if (bRet == FALSE)
return ICON_DEFAULT;
if (iSize == ICON_DEFAULT_NORMAL)
return ICON_DEFAULT;
else if (iSize == ICON_DEFAULT_LARGE)
return ICON_LARGE;
return ICON_INDETERMINATE;
}
BOOL ChangeIconSizes(int iOldState, int iNewState)
{
BOOL bRet;
int iOldSize, iNewSize;
int iHorz;
int iVert;
// Don't bother if nothing changed
if (iOldState == iNewState)
return FALSE;
// Get New Size
switch (iNewState)
{
case ICON_DEFAULT:
iNewSize = ICON_DEFAULT_NORMAL;
break;
case ICON_LARGE:
iNewSize = ICON_DEFAULT_LARGE;
break;
case ICON_INDETERMINATE:
// Don't bother to change anything
return FALSE;
default:
return FALSE;
}
// Get Original Size
bRet = GetRegValueInt (HKEY_CURRENT_USER, c_szHICKey, c_szWMSISVal, &iOldSize);
if (!bRet)
{
// Try geting system default instead
iOldSize = GetSystemMetrics (SM_CXICON);
}
// Don't need to change size if nothing has really changed
if (iNewSize == iOldSize)
return FALSE;
// Get new horizontal spacing
iHorz = GetSystemMetrics (SM_CXICONSPACING);
iHorz -= iOldSize;
if (iHorz < 0)
{
iHorz = 0;
}
iHorz += iNewSize;
// Get new vertical spacing
iVert = GetSystemMetrics (SM_CYICONSPACING);
iVert -= iOldSize;
if (iVert < 0)
{
iVert = 0;
}
iVert += iNewSize;
// Set New sizes and spacing
bRet = SetRegValueInt( HKEY_CURRENT_USER, c_szHICKey, c_szWMSISVal, iNewSize );
if (!bRet)
return FALSE;
SystemParametersInfo( SPI_ICONHORIZONTALSPACING, iHorz, NULL, SPIF_UPDATEINIFILE );
SystemParametersInfo( SPI_ICONVERTICALSPACING, iVert, NULL, SPIF_UPDATEINIFILE );
// We did change the sizes
return TRUE;
}
//
// GetAnimations
//
// Get current state of animations (windows / menus / etc.).
//
WPARAM GetAnimations(DWORD *pdwEffect)
{
BOOL fMenu = FALSE, fWindow = FALSE, fCombo = FALSE, fSmooth = FALSE, fList = FALSE, fFade = FALSE;
ANIMATIONINFO ai;
ai.cbSize = sizeof(ai);
if (SystemParametersInfo( SPI_GETANIMATION, sizeof(ai), (PVOID)&ai, 0 ))
{
fWindow = (ai.iMinAnimate) ? TRUE : FALSE;
}
SystemParametersInfo( SPI_GETCOMBOBOXANIMATION, 0, (PVOID)&fCombo, 0 );
SystemParametersInfo( SPI_GETLISTBOXSMOOTHSCROLLING, 0, (PVOID)&fList, 0 );
SystemParametersInfo( SPI_GETMENUANIMATION, 0, (PVOID)&fMenu, 0 );
fSmooth = (BOOL)GetRegValueDword( HKEY_CURRENT_USER,
(LPTSTR)c_szSSKey,
(LPTSTR)c_szSSVal
);
if (fSmooth == REG_BAD_DWORD)
{
fSmooth = 1;
}
SystemParametersInfo( SPI_GETMENUFADE, 0, (PVOID)&fFade, 0 );
*pdwEffect = (fFade ? MENU_EFFECT_FADE : MENU_EFFECT_SCROLL);
if (fMenu && fWindow && fCombo && fSmooth && fList)
return BST_CHECKED;
if ((!fMenu) && (!fWindow) && (!fCombo) && (!fSmooth) && (!fList))
return BST_UNCHECKED;
return BST_INDETERMINATE;
}
//
// SetAnimations
//
// Set animations according (windows / menus / etc.) according to flag.
//
void SetAnimations(WPARAM wVal, DWORD dwEffect)
{
ANIMATIONINFO ai;
// Note, if the checkbox is indeterminate, we treat it like it was checked.
// We should never get this far if the user didn't change something so this should be okay.
BOOL bVal = (wVal == BST_UNCHECKED) ? 0 : 1;
BOOL bEfx = (dwEffect == MENU_EFFECT_FADE) ? 1 : 0;
ai.cbSize = sizeof(ai);
ai.iMinAnimate = bVal;
SystemParametersInfo( SPI_SETANIMATION, sizeof(ai), (PVOID)&ai, SPIF_UPDATEINIFILE );
SystemParametersInfo( SPI_SETCOMBOBOXANIMATION, 0, IntToPtr(bVal), SPIF_UPDATEINIFILE );
SystemParametersInfo( SPI_SETLISTBOXSMOOTHSCROLLING, 0, IntToPtr(bVal), SPIF_UPDATEINIFILE );
SystemParametersInfo( SPI_SETMENUANIMATION, 0, IntToPtr(bVal), SPIF_UPDATEINIFILE );
SystemParametersInfo( SPI_SETTOOLTIPANIMATION, 0, IntToPtr(bVal), SPIF_UPDATEINIFILE );
SetRegValueDword(HKEY_CURRENT_USER, (LPTSTR)c_szSSKey, (LPTSTR)c_szSSVal, bVal);
SystemParametersInfo( SPI_SETMENUFADE, 0, IntToPtr(bEfx), SPIF_UPDATEINIFILE);
SystemParametersInfo( SPI_SETTOOLTIPFADE, 0, IntToPtr(bEfx), SPIF_UPDATEINIFILE);
SystemParametersInfo( SPI_SETSELECTIONFADE, 0, bVal ? IntToPtr(bEfx) : 0, SPIF_UPDATEINIFILE);
}
CEffectState::CEffectState()
{
}
CEffectState::~CEffectState()
{
}
HRESULT CEffectState::Load(void)
{
HRESULT hr = S_OK;
// Get the values for the settings from the registry and set the checkboxes
// Large Icons
_nLargeIcon = GetIconState();
// Full Color Icons
if(FALSE == GetRegValueInt(HKEY_CURRENT_USER, c_szHICKey, c_szHICVal, &_nHighIconColor)) // Key not in registry yet
{
_nHighIconColor = 4;
}
// Use animations
_wpMenuAnimation = GetAnimations(&_dwAnimationEffect);
// Smooth edges of screen fonts
_fFontSmoothing = FALSE;
SystemParametersInfo(SPI_GETFONTSMOOTHING, 0, (PVOID)&_fFontSmoothing, 0);
_dwFontSmoothingType = FONT_SMOOTHING_AA;
#ifdef CLEARTYPECOMBO
if (SystemParametersInfo(SPI_GETFONTSMOOTHINGTYPE, 0, (PVOID)&_dwFontSmoothingType, 0))
{
}
#endif //CLEARTYPECOMBO
// Show contents while dragging
_fDragWindow = FALSE;
SystemParametersInfo(SPI_GETDRAGFULLWINDOWS, 0, (PVOID)&_fDragWindow, 0);
_fKeyboardIndicators = FALSE;
SystemParametersInfo(SPI_GETKEYBOARDINDICATORS, 0, (PVOID)&_fKeyboardIndicators, 0);
_fMenuShadows = TRUE;
SystemParametersInfo(SPI_GETDROPSHADOW, 0, (PVOID)&_fMenuShadows, 0);
// Set the old values so we know when they changed.
_nOldLargeIcon = _nLargeIcon;
_nOldHighIconColor = _nHighIconColor;
_wpOldMenuAnimation = _wpMenuAnimation;
_fOldFontSmoothing = _fFontSmoothing;
_dwOldFontSmoothingType = _dwFontSmoothingType;
_fOldDragWindow = _fDragWindow;
_fOldKeyboardIndicators = _fKeyboardIndicators;
_dwOldAnimationEffect = _dwAnimationEffect;
_fOldMenuShadows = _fMenuShadows;
// load the icons and add them to the image lists
// get the icon files and indexes from the registry, including for the Default recycle bin
for (int nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
TCHAR szTemp[MAX_PATH];
szTemp[0] = 0;
IconGetRegValueString(c_aIconRegKeys[nIndex].pclsid, TEXT("DefaultIcon"), c_aIconRegKeys[nIndex].szIconValue, szTemp, ARRAYSIZE(szTemp));
int iIndex = PathParseIconLocation(szTemp);
// store the icon information
lstrcpy(_IconData[nIndex].szOldFile, szTemp);
lstrcpy(_IconData[nIndex].szNewFile, szTemp);
_IconData[nIndex].iOldIndex = iIndex;
_IconData[nIndex].iNewIndex = iIndex;
}
return hr;
}
BOOL CEffectState::HasIconsChanged(IN CEffectState * pCurrent)
{
BOOL fHasIconsChanged = FALSE;
int nIndex;
for (nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
if (StrCmpI(_IconData[nIndex].szNewFile, pCurrent->_IconData[nIndex].szNewFile) ||
(_IconData[nIndex].iNewIndex != pCurrent->_IconData[nIndex].iNewIndex))
{
fHasIconsChanged = TRUE;
break;
}
}
return fHasIconsChanged;
}
HRESULT CEffectState::Save(void)
{
HRESULT hr = S_OK;
BOOL bDorked = FALSE;
int iX;
// Large Icons
BOOL bSendSettingsChange = ChangeIconSizes(_nOldLargeIcon, _nLargeIcon);
if (bSendSettingsChange)
{
_nOldLargeIcon = _nLargeIcon;
bDorked = TRUE;
}
// Full Color Icons
if(_nOldHighIconColor != _nHighIconColor)
{
SetRegValueInt(HKEY_CURRENT_USER, c_szHICKey, c_szHICVal, _nHighIconColor);
if ((GetBitsPerPixel() < 16) && (_nHighIconColor == 16)) // Display mode won't support icon high colors
{
}
else
{
_nOldHighIconColor = _nHighIconColor;
bSendSettingsChange = TRUE;
}
}
// Full window drag
if (_fOldDragWindow != _fDragWindow)
{
_fOldDragWindow = _fDragWindow;
SystemParametersInfo(SPI_SETDRAGFULLWINDOWS, _fDragWindow, 0, SPIF_UPDATEINIFILE);
// we need to send this because the tray's autohide switches off this
bSendSettingsChange = TRUE;
}
// Show Menu Shadows
if (_fOldMenuShadows != _fMenuShadows)
{
_fOldMenuShadows = _fMenuShadows;
SystemParametersInfo(SPI_SETDROPSHADOW, _fMenuShadows, 0, SPIF_UPDATEINIFILE);
// we need to send this because the tray's autohide switches off this
bSendSettingsChange = TRUE;
}
// Font smoothing
if ((_fOldFontSmoothing != _fFontSmoothing) || (_dwOldFontSmoothingType != _dwFontSmoothingType))
{
#ifdef CLEARTYPECOMBO
SystemParametersInfo(SPI_SETFONTSMOOTHINGTYPE, 0, (PVOID)UlongToPtr(_dwFontSmoothingType), SPIF_UPDATEINIFILE);
#endif
_dwOldFontSmoothingType = _dwFontSmoothingType;
_fOldFontSmoothing = _fFontSmoothing;
SystemParametersInfo(SPI_SETFONTSMOOTHING, _fFontSmoothing, 0, SPIF_UPDATEINIFILE);
}
// Menu animations
if ((_wpOldMenuAnimation != _wpMenuAnimation) || (_dwOldAnimationEffect != _dwAnimationEffect))
{
_wpOldMenuAnimation = _wpMenuAnimation;
SetAnimations(_wpMenuAnimation, _dwAnimationEffect);
_dwOldAnimationEffect = _dwAnimationEffect;
}
// Change the system icons
for( iX = 0; iX < NUM_ICONS; iX++)
{
if ((lstrcmpi(_IconData[iX].szNewFile, _IconData[iX].szOldFile) != 0) ||
(_IconData[iX].iNewIndex != _IconData[iX].iOldIndex))
{
TCHAR szTemp[MAX_PATH];
wnsprintf(szTemp, ARRAYSIZE(szTemp), TEXT("%s,%d"), _IconData[iX].szNewFile, _IconData[iX].iNewIndex);
IconSetRegValueString(c_aIconRegKeys[iX].pclsid, TEXT("DefaultIcon"), c_aIconRegKeys[iX].szIconValue, szTemp);
// Next two lines necessary if the user does an Apply as opposed to OK
lstrcpy(_IconData[iX].szOldFile, _IconData[iX].szNewFile);
_IconData[iX].iOldIndex = _IconData[iX].iNewIndex;
bDorked = TRUE;
}
}
// Keyboard indicators
if (_fOldKeyboardIndicators != _fKeyboardIndicators)
{
_fOldKeyboardIndicators = _fKeyboardIndicators;
DWORD_PTR dwResult;
// Are we turning this on? (!_fKeyboardIndicators means "don't show" -> hide)
if (!_fKeyboardIndicators)
{
// Yes, on: hide the key cues, turn on the mechanism
SystemParametersInfo(SPI_SETKEYBOARDINDICATORS, 0, IntToPtr(_fKeyboardIndicators), SPIF_UPDATEINIFILE);
SendMessageTimeout(HWND_BROADCAST, WM_CHANGEUISTATE, MAKEWPARAM(UIS_SET, UISF_HIDEFOCUS | UISF_HIDEACCEL), 0, SMTO_ABORTIFHUNG, 10*1000, &dwResult);
}
else
{
// No, off: means show the keycues, turn off the mechanism
SendMessageTimeout(HWND_BROADCAST, WM_CHANGEUISTATE,
MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS | UISF_HIDEACCEL),
0, SMTO_ABORTIFHUNG, 10*1000, &dwResult);
SystemParametersInfo(SPI_SETKEYBOARDINDICATORS, 0, IntToPtr(_fKeyboardIndicators), SPIF_UPDATEINIFILE);
}
}
// Make the system notice we changed the system icons
if (bDorked)
{
PFNSHUPDATERECYCLEBINICON pfnSHUpdateRecycleBinIcon = NULL;
SHChangeNotify(SHCNE_ASSOCCHANGED, 0, NULL, NULL); // should do the trick!
if (!pfnSHUpdateRecycleBinIcon)
{
// Load SHUpdateRecycleBinIcon() if it exists
g_hmodShell32 = LoadLibrary(SZ_SHELL32);
pfnSHUpdateRecycleBinIcon = (PFNSHUPDATERECYCLEBINICON)GetProcAddress(g_hmodShell32, SZ_SHUPDATERECYCLEBINICON);
}
if (pfnSHUpdateRecycleBinIcon != NULL)
{
pfnSHUpdateRecycleBinIcon();
}
}
if (bSendSettingsChange)
{
DWORD_PTR dwResult = 0;
SendMessageTimeout(HWND_BROADCAST, WM_SETTINGCHANGE, 0, 0, SMTO_ABORTIFHUNG, 10*1000, &dwResult);
}
return hr;
}
#define SZ_ICON_DEFAULTICON L"DefaultValue"
HRESULT CEffectState::GetIconPath(IN CLSID clsid, IN LPCWSTR pszName, IN LPWSTR pszPath, IN DWORD cchSize)
{
HRESULT hr = E_FAIL;
int nIndex;
if (!StrCmpIW(SZ_ICON_DEFAULTICON, pszName))
{
pszName = L"";
}
for (nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
if (IsEqualCLSID(*(c_aIconRegKeys[nIndex].pclsid), clsid) &&
!StrCmpIW(pszName, c_aIconRegKeys[nIndex].szIconValue))
{
// We found it.
wnsprintfW(pszPath, cchSize, L"%s,%d", _IconData[nIndex].szNewFile, _IconData[nIndex].iNewIndex);
hr = S_OK;
break;
}
}
return hr;
}
HRESULT CEffectState::SetIconPath(IN CLSID clsid, IN LPCWSTR pszName, IN LPCWSTR pszPath, IN int nResourceID)
{
HRESULT hr = E_FAIL;
int nIndex;
if (!StrCmpIW(SZ_ICON_DEFAULTICON, pszName))
{
pszName = L"";
}
for (nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
if (IsEqualCLSID(*(c_aIconRegKeys[nIndex].pclsid), clsid) &&
!StrCmpIW(pszName, c_aIconRegKeys[nIndex].szIconValue))
{
TCHAR szTemp[MAX_PATH];
if (!pszPath || !pszPath[0])
{
// The caller didn't specify an icon so use the default values.
if (!SHExpandEnvironmentStrings(c_aIconRegKeys[nIndex].szDefault, szTemp, ARRAYSIZE(szTemp)))
{
StrCpyN(szTemp, c_aIconRegKeys[nIndex].szDefault, ARRAYSIZE(szTemp));
}
pszPath = szTemp;
nResourceID = c_aIconRegKeys[nIndex].nDefaultIndex;
}
// We found it.
StrCpyNW(_IconData[nIndex].szNewFile, pszPath, ARRAYSIZE(_IconData[nIndex].szNewFile));
_IconData[nIndex].iNewIndex = nResourceID;
hr = S_OK;
break;
}
}
return hr;
}
HRESULT CEffectState::Clone(OUT CEffectState ** ppEffectClone)
{
HRESULT hr = E_OUTOFMEMORY;
*ppEffectClone = new CEffectState();
if (*ppEffectClone)
{
hr = S_OK;
(*ppEffectClone)->_nLargeIcon = _nLargeIcon;
(*ppEffectClone)->_nHighIconColor = _nHighIconColor;
(*ppEffectClone)->_wpMenuAnimation = _wpMenuAnimation;
(*ppEffectClone)->_fFontSmoothing = _fFontSmoothing;
(*ppEffectClone)->_dwFontSmoothingType = _dwFontSmoothingType;
(*ppEffectClone)->_fDragWindow = _fDragWindow;
(*ppEffectClone)->_fKeyboardIndicators = _fKeyboardIndicators;
(*ppEffectClone)->_dwAnimationEffect = _dwAnimationEffect;
(*ppEffectClone)->_nOldLargeIcon = _nOldLargeIcon;
(*ppEffectClone)->_nOldHighIconColor = _nOldHighIconColor;
(*ppEffectClone)->_wpOldMenuAnimation = _wpOldMenuAnimation;
(*ppEffectClone)->_fOldFontSmoothing = _fOldFontSmoothing;
(*ppEffectClone)->_dwOldFontSmoothingType = _dwOldFontSmoothingType;
(*ppEffectClone)->_fOldDragWindow = _fOldDragWindow;
(*ppEffectClone)->_fOldKeyboardIndicators = _fOldKeyboardIndicators;
(*ppEffectClone)->_dwOldAnimationEffect = _dwOldAnimationEffect;
(*ppEffectClone)->_fOldMenuShadows = _fOldMenuShadows;
for (int nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
(*ppEffectClone)->_IconData[nIndex].iOldIndex = _IconData[nIndex].iOldIndex;
(*ppEffectClone)->_IconData[nIndex].iNewIndex = _IconData[nIndex].iNewIndex;
StrCpyN((*ppEffectClone)->_IconData[nIndex].szOldFile, _IconData[nIndex].szOldFile, ARRAYSIZE((*ppEffectClone)->_IconData[nIndex].szOldFile));
StrCpyN((*ppEffectClone)->_IconData[nIndex].szNewFile, _IconData[nIndex].szNewFile, ARRAYSIZE((*ppEffectClone)->_IconData[nIndex].szNewFile));
}
}
return hr;
}
BOOL CEffectState::IsDirty(void)
{
BOOL fDirty = FALSE;
if ((_nLargeIcon != _nOldLargeIcon) ||
(_nHighIconColor != _nOldHighIconColor) ||
(_wpMenuAnimation != _wpOldMenuAnimation) ||
(_fFontSmoothing != _fOldFontSmoothing) ||
(_dwFontSmoothingType != _dwOldFontSmoothingType) ||
(_fDragWindow != _fOldDragWindow) ||
(_fMenuShadows != _fOldMenuShadows) ||
(_fKeyboardIndicators != _fOldKeyboardIndicators) ||
(_dwAnimationEffect != _dwOldAnimationEffect))
{
fDirty = TRUE;
}
else
{
for (int nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
{
if ((_IconData[nIndex].iNewIndex != _IconData[nIndex].iOldIndex) ||
(StrCmpI(_IconData[nIndex].szNewFile, _IconData[nIndex].szOldFile)))
{
fDirty = TRUE;
break;
}
}
}
return fDirty;
}