Windows2000/private/shell/shell32/unicpp/deskstat.cpp

5317 lines
178 KiB
C++
Raw Normal View History

2001-01-01 00:00:00 +01:00
#include "stdafx.h"
#pragma hdrstop
#ifdef _WIN64
extern "C" char * __cdecl StrTokEx(char ** pstring, const char * control);
#else
#include <iert.h>
#endif
#define DXA_GROWTH_CONST 10
#define ZINDEX_START 1000
#define MAXID_LENGTH 10 //Maximum number of digits in ID string plus 1.
#if 0
#define TF_DESKSTAT TF_CUSTOM2
#define TF_DYNAMICHTML TF_CUSTOM1
#else
#define TF_DESKSTAT 0
#define TF_DYNAMICHTML 0
#endif
IActiveDesktop *g_pActiveDesk = NULL;
#define c_szRegStrDesktop REGSTR_PATH_DESKTOP
#define c_szWallpaper REG_VAL_GENERAL_WALLPAPER
#define c_szBackupWallpaper REG_VAL_GENERAL_BACKUPWALLPAPER
#define c_szPattern TEXT("Pattern")
#define c_szTileWall REG_VAL_GENERAL_TILEWALLPAPER
#define c_szWallpaperStyle REG_VAL_GENERAL_WALLPAPERSTYLE
#define c_szWallpaperTime REG_VAL_GENERAL_WALLPAPERTIME
#define c_szRefreshDesktop TEXT("RefreshDesktop")
#define c_szBufferedRefresh TEXT("BufferedRefresh")
#define COMP_TYPE 0x00000003
#define COMP_SELECTED 0x00002000
#define COMP_NOSCROLL 0x00004000
#ifdef DEBUG
#define ENTERPROC EnterProcDS
#define EXITPROC ExitProcDS
void EnterProcDS(DWORD dwTraceLevel, LPSTR szFmt, ...);
void ExitProcDS(DWORD dwTraceLevel, LPSTR szFmt, ...);
extern DWORD g_dwDeskStatTrace;
#else
#ifndef CCOVER
#pragma warning(disable:4002)
#define ENTERPROC()
#define EXITPROC()
#else // ccover buildi
// these are needed because of a bug in cl.exe that results in improper processing
// of #pragma when run with cl -P, and then compiling
#define ENTERPROC 1 ? (void) 0 : (void)
#define EXITPROC 1 ? (void) 0 : (void)
#endif //end of ccover
#endif
MAKE_CONST_BSTR(s_sstrBeforeEnd, L"BeforeEnd");
MAKE_CONST_BSTR(s_sstrDeskMovr, L"DeskMovr");
MAKE_CONST_BSTR(s_sstrDeskMovrW, L"DeskMovrW");
MAKE_CONST_BSTR(s_sstrclassid, L"classid");
MAKE_CONST_BSTR(s_sstrEmpty, L"");
STDAPI ParseDesktopComponent(HWND hwndOwner, LPWSTR wszURL, COMPONENT *pInfo);
WCHAR wUnicodeBOM = 0xfeff; // Little endian unicode Byte Order Mark.First byte:0xff, Second byte: 0xfe.
CReadFileObj::CReadFileObj(LPCTSTR lpszFileName)
{
//Open the file
if ((_hFile = CreateFile(lpszFileName,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
FILE_ATTRIBUTE_NORMAL,
0)) != INVALID_HANDLE_VALUE)
{
WCHAR wBOM;
DWORD dwBytesRead = 0;
if ((ReadFile(_hFile, (LPVOID)&wBOM, sizeof(WCHAR), &dwBytesRead, NULL)) && (dwBytesRead == sizeof(WCHAR)))
{
if (wBOM == wUnicodeBOM)
_iCharset = UNICODE_HTML_CHARSET;
else
{
//Note: Anything other than the little endian unicode file is treated as ansi.
_iCharset = ANSI_HTML_CHARSET;
SetFilePointer(_hFile, 0L, NULL, FILE_BEGIN); //Seek to the begining of the file
}
}
}
}
CReadFileObj::~CReadFileObj()
{
if (_hFile != INVALID_HANDLE_VALUE)
{
CloseHandle(_hFile);
_hFile = NULL;
}
}
// This will read and if necessary convert between ANSI and UNICODE
// NOTE: The uiCharsToRead must be atleast one less than the size of the buffer (lpwszBuff)
// because one null may be written at the end of the buffer by SHAnsiToUnicode().
HRESULT CReadFileObj::FileReadAndConvertChars(int iDestCharset, LPWSTR lpwszBuff, UINT uiCharsToRead, UINT *puiCharsActuallyRead, UINT *puiCharsConverted)
{
HRESULT hres = S_OK;
DWORD dwCharsRead = 0;
DWORD dwTotalCharsConverted = 0;
if (_hFile != INVALID_HANDLE_VALUE)
{
if (_iCharset == UNICODE_HTML_CHARSET)
{
if (iDestCharset == UNICODE_HTML_CHARSET)
{
hres = FileReadCharsW(lpwszBuff, uiCharsToRead, (UINT *)&dwCharsRead);
dwTotalCharsConverted = dwCharsRead;
}
else
{
//Destination is ansi; Read the UNICODE source and convert to ANSI.
WCHAR wszBuf[INTERNET_MAX_URL_LENGTH + 1]; //Temp buffer to read the UNICODE chars into.
LPSTR lpszBuff = (LPSTR)lpwszBuff;
DWORD dwTotalCharsToRead = (DWORD)uiCharsToRead;
while (dwTotalCharsToRead)
{
DWORD dwCount;
DWORD dwActuallyRead;
// - 1 to give room for a null character at the end.
dwCount = min(dwTotalCharsToRead, (ARRAYSIZE(wszBuf) - 1));
if (ReadFile(_hFile, (LPSTR)wszBuf, dwCount * sizeof(WCHAR), &dwActuallyRead, NULL))
{
DWORD dwConverted;
dwActuallyRead = dwActuallyRead / sizeof(WCHAR);
//Null terminate the source buffer.
wszBuf[dwActuallyRead] = L'\0'; //UNICODE null terminate the source.
//Convert what we just read.
dwConverted = SHUnicodeToAnsi(wszBuf, lpszBuff, dwActuallyRead + 1); //+1 for null termination
//Update the count & stuff.
lpszBuff += dwConverted - 1; //Subtract the null.
dwTotalCharsToRead -= dwActuallyRead;
dwCharsRead += dwActuallyRead;
dwTotalCharsConverted += dwConverted - 1; //Subtract the null.
if (dwActuallyRead < dwCount)
break; //We have reached the end of file.
}
else
{
hres = E_FAIL;
break;
}
}
}
}
else
{
//Source file is ANSI. Check the Destination.
if (iDestCharset == ANSI_HTML_CHARSET)
{
//Destination is ANSI too! Cool! No need for conversion!
hres = FileReadCharsA((LPSTR)lpwszBuff, uiCharsToRead, (UINT *)&dwCharsRead);
dwTotalCharsConverted = dwCharsRead;
}
else
{
//Destination is UNICODE! Read the ansi and convert it to UNICODE!
char szBuf[INTERNET_MAX_URL_LENGTH + 1]; //Temp buffer to read the ansi chars into.
DWORD dwTotalCharsToRead = (DWORD)uiCharsToRead;
while (dwTotalCharsToRead)
{
DWORD dwCount;
DWORD dwActuallyRead;
// - 1 to give room for a null character at the end.
dwCount = min(dwTotalCharsToRead, (ARRAYSIZE(szBuf) - 1));
if (ReadFile(_hFile, (LPSTR)szBuf, dwCount, &dwActuallyRead, NULL))
{
DWORD dwConverted;
//Null terminate the source buffer.
szBuf[dwActuallyRead] = '\0'; //ANSI null terminate the source.
//Convert what we just read.
dwConverted = SHAnsiToUnicode(szBuf, lpwszBuff, dwActuallyRead + 1); //+1 for null termination
//Update the count & stuff.
lpwszBuff += dwConverted - 1; //Subtract the null.
dwTotalCharsToRead -= dwActuallyRead;
dwCharsRead += dwActuallyRead;
dwTotalCharsConverted += dwConverted - 1; //Subtract the null.
if (dwActuallyRead < dwCount)
break; //We have reached the end of file.
} else {
hres = E_FAIL;
break;
}
} //while
}
}
}
else
hres = E_FAIL; //The file handle is bad.
*puiCharsActuallyRead = (UINT)dwCharsRead;
*puiCharsConverted = (UINT)dwTotalCharsConverted;
return hres;
}
HRESULT CReadFileObj::FileReadCharsA(LPSTR lpszBuff, UINT uiCharsToRead, UINT *puiCharsActuallyRead)
{
HRESULT hres = E_FAIL;
DWORD dwCharsRead = 0;
if ((_hFile != INVALID_HANDLE_VALUE) &&
(_iCharset == ANSI_HTML_CHARSET) &&
ReadFile(_hFile, (LPVOID)lpszBuff, (DWORD)(uiCharsToRead), &dwCharsRead, NULL))
{
dwCharsRead = dwCharsRead; //get the number of wchars read.
hres = S_OK;
}
*puiCharsActuallyRead = (UINT)dwCharsRead;
return hres;
}
HRESULT CReadFileObj::FileReadCharsW(LPWSTR lpwszBuff, UINT uiCharsToRead, UINT *puiCharsActuallyRead)
{
HRESULT hres = E_FAIL;
DWORD dwCharsRead = 0;
if ((_hFile != INVALID_HANDLE_VALUE) &&
(_iCharset == UNICODE_HTML_CHARSET) &&
ReadFile(_hFile, (LPVOID)lpwszBuff, (DWORD)(uiCharsToRead * sizeof(WCHAR)), &dwCharsRead, NULL))
{
dwCharsRead = dwCharsRead / sizeof(WCHAR); //get the number of wchars read.
hres = S_OK;
}
*puiCharsActuallyRead = (UINT)dwCharsRead;
return hres;
}
HRESULT CReadFileObj::FileSeekChars(LONG lCharOffset, DWORD dwOrigin)
{
HRESULT hres = E_FAIL;
if (_hFile != INVALID_HANDLE_VALUE)
{
if (SetFilePointer(_hFile,
lCharOffset*((_iCharset == UNICODE_HTML_CHARSET) ? sizeof(WCHAR) : sizeof(char)),
NULL,
dwOrigin) != INVALID_SET_FILE_POINTER)
hres = S_OK;
}
return hres;
}
HRESULT CReadFileObj::FileGetCurCharOffset(LONG *plCharOffset)
{
HRESULT hres = E_FAIL;
DWORD dwByteOffset = 0;
*plCharOffset = 0;
if (_hFile != INVALID_HANDLE_VALUE)
{
if ((dwByteOffset = SetFilePointer(_hFile, 0L, NULL, FILE_CURRENT)) != INVALID_SET_FILE_POINTER)
{
*plCharOffset = dwByteOffset / ((_iCharset == UNICODE_HTML_CHARSET) ? sizeof(WCHAR) : sizeof(char));
hres = S_OK;
}
}
return hres;
}
int GetIntFromReg(HKEY hKey,
LPCTSTR lpszSubkey,
LPCTSTR lpszNameValue,
int iDefault)
{
TCHAR szValue[20];
DWORD dwSizeofValueBuff = SIZEOF(szValue);
int iRetValue = iDefault;
DWORD dwType;
if ((SHGetValue(hKey, lpszSubkey, lpszNameValue, &dwType, (LPBYTE)szValue, &dwSizeofValueBuff) == ERROR_SUCCESS) && dwSizeofValueBuff)
{
if (dwType == REG_SZ)
{
iRetValue = (int)StrToInt(szValue);
}
}
return iRetValue;
}
BOOL GetStringFromReg(HKEY hkey,
LPCTSTR lpszSubkey,
LPCTSTR lpszValueName,
LPCTSTR lpszDefault,
LPTSTR lpszValue,
DWORD cchSizeofValueBuff)
{
BOOL fRet = FALSE;
DWORD dwType;
cchSizeofValueBuff *= sizeof(TCHAR);
if (SHGetValue(hkey, lpszSubkey, lpszValueName, &dwType, lpszValue, &cchSizeofValueBuff) == ERROR_SUCCESS)
{
fRet = TRUE;
}
// On failure use the default string.
if (!fRet && lpszDefault)
{
lstrcpy(lpszValue, lpszDefault);
}
return fRet;
}
void GetWallpaperFileTime(LPCTSTR pszWallpaper, LPFILETIME lpftFileTime)
{
HANDLE hFile;
BOOL fRet = FALSE;
if ((hFile = CreateFile(pszWallpaper, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0)) != INVALID_HANDLE_VALUE)
{
fRet = GetFileTime(hFile, NULL, NULL, lpftFileTime);
CloseHandle(hFile);
}
if (!fRet)
ZeroMemory(lpftFileTime, SIZEOF(FILETIME));
// no return value
}
BOOL HasWallpaperReallyChanged(LPCTSTR pszRegKey, LPTSTR pszOldWallpaper, LPTSTR pszBackupWallpaper, DWORD dwOldWallpaperStyle, DWORD dwNewWallpaperStyle)
{
// we default to TRUE here.
if ((dwOldWallpaperStyle == dwNewWallpaperStyle)
&& (0 == lstrcmpi(pszOldWallpaper, pszBackupWallpaper)))
{
// The wallpaper filename and style hasn't changed.
// But, the content of this file could have changed.
// See if the content has changed by looking at the
// last-written date and time stamp on this file.
FILETIME ftOld, ftBack;
DWORD dwType, cbBack = SIZEOF(ftBack);
// if either of these fail, then they will
// remain Zero so the compare will
// be successful.
GetWallpaperFileTime(pszOldWallpaper, &ftOld);
if (ERROR_SUCCESS != SHGetValue(HKEY_CURRENT_USER, pszRegKey, c_szWallpaperTime, &dwType, &ftBack, &cbBack))
ZeroMemory(&ftBack, SIZEOF(ftBack));
//Get the last written time of the backup wallpaper from registry
if (ftOld.dwLowDateTime == ftBack.dwLowDateTime && (ftOld.dwHighDateTime == ftBack.dwHighDateTime))
// everything is the same!
return FALSE;
}
return TRUE;
}
// Function: ReadWallpaperStyleFromReg()
// This function reads the "TileWallpaper" and the "WallpaperStyle" from the given location
// in the registry.
void ReadWallpaperStyleFromReg(LPCTSTR pszRegKey, DWORD *pdwWallpaperStyle, BOOL fIgnorePlatforms)
{
//Do not read "Stretch" bits for older platforms.
if (fIgnorePlatforms || g_bRunOnMemphis || g_bRunOnNT5)
*pdwWallpaperStyle = GetIntFromReg(HKEY_CURRENT_USER, pszRegKey, c_szWallpaperStyle, WPSTYLE_CENTER);
else
*pdwWallpaperStyle = WPSTYLE_CENTER;
if (GetIntFromReg(HKEY_CURRENT_USER, pszRegKey, c_szTileWall, WPSTYLE_TILE))
{
// "Tile" overrides the "Stretch" style.
*pdwWallpaperStyle = WPSTYLE_TILE;
}
// else, STRETCH or CENTER.
}
BOOL CActiveDesktop::_IsDisplayInSafeMode(void)
{
WCHAR wszDisplay[MAX_PATH];
DWORD dwcch = MAX_PATH;
return (SUCCEEDED(GetScheme(wszDisplay, &dwcch, SCHEME_GLOBAL | SCHEME_DISPLAY))
&& (0 == StrCmpW(wszDisplay, REG_DESKCOMP_SAFEMODE_SUFFIX_L)));
}
BOOL ReadPolicyForWallpaper(LPTSTR lpszPolicyForWallpaper, DWORD dwSizeofBuff)
{
TCHAR szWallpaperName[MAX_PATH];
DWORD dwType;
if (!lpszPolicyForWallpaper)
{
lpszPolicyForWallpaper = szWallpaperName;
dwSizeofBuff = ARRAYSIZE(szWallpaperName);
}
if ((SHGetValue(HKEY_CURRENT_USER, REGSTR_PATH_WP_POLICY, c_szWallpaper, &dwType, (LPBYTE)lpszPolicyForWallpaper,
&dwSizeofBuff) == ERROR_SUCCESS) && dwSizeofBuff)
return TRUE; //Policy is there!
else
return FALSE; //No policy is set!
}
BOOL ReadPolicyForWPStyle(LPDWORD lpdwStyle)
{
DWORD dwStyle;
DWORD dwType;
TCHAR szValue[20];
DWORD dwSizeofValueBuff = ARRAYSIZE(szValue);
BOOL fRet = FALSE;
// The caller can passin a NULL, if they are not interested in the actual value and they just
// want to know if this policy is set or not.
if (!lpdwStyle)
lpdwStyle = &dwStyle;
if ((SHGetValue(HKEY_CURRENT_USER, REGSTR_PATH_WP_POLICY, c_szWallpaperStyle, &dwType, (LPBYTE)szValue,
&dwSizeofValueBuff) == ERROR_SUCCESS) && dwSizeofValueBuff)
{
if (dwType == REG_SZ)
{
*lpdwStyle = (DWORD)StrToInt(szValue);
fRet = TRUE;
}
}
return fRet;
}
void CActiveDesktop::_ReadWallpaper(BOOL fActiveDesktop)
{
ENTERPROC(2, "DS ReadWallpaper()");
TCHAR lpszDeskcomp[MAX_PATH];
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_GENERAL, _pszScheme);
_fPolicyForWPName = ReadPolicyForWallpaper(_szSelectedWallpaper, ARRAYSIZE(_szSelectedWallpaper));
_fPolicyForWPStyle = ReadPolicyForWPStyle(&_wpo.dwStyle);
// Read in the wallpaper and style from the appropriate registry location.
LPCTSTR pszRegKey;
if (fActiveDesktop)
{
pszRegKey = (LPCTSTR)lpszDeskcomp;
TCHAR szOldWallpaper[MAX_PATH];
DWORD dwOldWallpaperStyle;
// Read the Wallpaper from the Old location.
GetStringFromReg(HKEY_CURRENT_USER, c_szRegStrDesktop, c_szWallpaper, c_szNULL, szOldWallpaper, ARRAYSIZE(szOldWallpaper));
// Read wallpaper style from the old location.
ReadWallpaperStyleFromReg((LPCTSTR)c_szRegStrDesktop, &dwOldWallpaperStyle, FALSE);
// Read the wallpaper from the new location too!
if ((!_fPolicyForWPName) || (_IsDisplayInSafeMode()))
{
if (!GetStringFromReg(HKEY_CURRENT_USER, pszRegKey, c_szWallpaper, szOldWallpaper, _szSelectedWallpaper, ARRAYSIZE(_szSelectedWallpaper)))
{
pszRegKey = c_szRegStrDesktop;
}
}
//Read wallpaper style from the new location too!
if (!_fPolicyForWPStyle)
ReadWallpaperStyleFromReg(pszRegKey, &_wpo.dwStyle, FALSE);
//If there is a Safe mode scheme here do NOT attempt to change wallpaper
if ((!_IsDisplayInSafeMode()) && (!_fPolicyForWPName))
{
//Read what is stored as "Backup" wallpaper.
GetStringFromReg(HKEY_CURRENT_USER, pszRegKey, c_szBackupWallpaper, szOldWallpaper, _szBackupWallpaper, ARRAYSIZE(_szBackupWallpaper));
//See if the Old wallpaper is differnet from the backed up wallpaper
if (HasWallpaperReallyChanged(pszRegKey, szOldWallpaper, _szBackupWallpaper, dwOldWallpaperStyle, _wpo.dwStyle))
{
//They are different. This means that some other app has changed the "Old" wallpaper
//after the last time we backed it up in the registry.
// Make this wallpaper as the Selected wallpaper!
#ifdef WE_WANT_DEFAULT_WALLPAPER
// The following hack is needed only if we want default wallpapers. For example, when
// we port shell code to Win2000 millenium, we may want a default wallpaper in which case,
// we need to enable the following hack and also in the deskcls.cpp file.
// This is a kinda hack, but the best possible solution right now. The scenario is as follows.
// The Memphis setup guys replace what the user specifies as the wallpaper in the old location
// and restore it after setup is complete. But, SetDefaultWallpaper() gets called bet. these
// two times and we are supposed to take a decision on whether to set the default htm wallpaper or not,
// depending on what the user had set before the installation. The solution is to delay making
// this decision until after the setup guys have restored the user's wallpaper. We do this in
// CActiveDesktop::_ReadWallpaper(). We specify that SetDefaultWallpaper() was called by setting
// the backup wallpaper in the new location to the default wallpaper.
TCHAR szDefaultWallpaper[MAX_PATH];
GetDefaultWallpaper(szDefaultWallpaper, SIZECHARS(szDefaultWallpaper));
if (lstrcmp(_szBackupWallpaper, szDefaultWallpaper) == 0
&& (!szOldWallpaper[0] || lstrcmp(szOldWallpaper, g_szNone) == 0))
{
lstrcpy(_szSelectedWallpaper, szDefaultWallpaper);
}
else
{
lstrcpy(_szSelectedWallpaper, szOldWallpaper);
}
#else // WE_WANT_DEFAULT_WALLPAPER
lstrcpy(_szSelectedWallpaper, szOldWallpaper);
#endif // WE_WANT_DEFAULT_WALLPAPER
_wpo.dwStyle = dwOldWallpaperStyle;
_fWallpaperDirty = TRUE;
_fWallpaperChangedDuringInit = TRUE;
}
}
//Make a backup of the "Old" wallpaper
lstrcpy(_szBackupWallpaper, szOldWallpaper);
} else {
pszRegKey = c_szRegStrDesktop; //Get it from the old location!
//Since active desktop is not available, read wallpaper from old location.
if (!_fPolicyForWPName)
GetStringFromReg(HKEY_CURRENT_USER, pszRegKey, c_szWallpaper, c_szNULL, _szSelectedWallpaper, ARRAYSIZE(_szSelectedWallpaper));
//Make a backup of the "Old" wallpaper
lstrcpy(_szBackupWallpaper, _szSelectedWallpaper);
//Read the wallpaper style
if (!_fPolicyForWPStyle)
ReadWallpaperStyleFromReg(pszRegKey, &_wpo.dwStyle, TRUE);
}
EXITPROC(2, "DS ReadWallpaper! (_szSelectedWP=>%s<)", _szSelectedWallpaper);
}
void CActiveDesktop::_ReadPattern(void)
{
ENTERPROC(2, "DS ReadPattern()");
GetStringFromReg(HKEY_CURRENT_USER, c_szRegStrDesktop, c_szPattern, c_szNULL, _szSelectedPattern, ARRAYSIZE(_szSelectedPattern));
EXITPROC(2, "DS ReadPattern! (_szSelectedPattern=>%s<)", _szSelectedPattern);
}
void CActiveDesktop::_ReadComponent(HKEY hkey, LPCTSTR pszComp)
{
ENTERPROC(2, "DS ReadComponent(hk=%08X,pszComp=>%s<)", hkey, pszComp);
HKEY hkeyComp;
if (RegOpenKeyEx(hkey, pszComp, 0, KEY_READ, &hkeyComp) == ERROR_SUCCESS)
{
DWORD cbSize, dwType;
COMPONENTA comp;
comp.dwSize = sizeof(COMPONENTA);
// Read in the source string.
cbSize = SIZEOF(comp.szSource);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_SOURCE, NULL, &dwType, (LPBYTE)&comp.szSource, &cbSize) != ERROR_SUCCESS)
{
comp.szSource[0] = TEXT('\0');
}
// Read in the SubscribedURL string.
cbSize = SIZEOF(comp.szSubscribedURL);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_SUBSCRIBED_URL, NULL, &dwType, (LPBYTE)&comp.szSubscribedURL, &cbSize) != ERROR_SUCCESS)
{
comp.szSubscribedURL[0] = TEXT('\0');
}
// Read in the Friendly name string.
cbSize = SIZEOF(comp.szFriendlyName);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_NAME, NULL, &dwType, (LPBYTE)&comp.szFriendlyName, &cbSize) != ERROR_SUCCESS)
{
comp.szFriendlyName[0] = TEXT('\0');
}
// Read in and parse the flags.
DWORD dwFlags;
cbSize = SIZEOF(dwFlags);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_FLAGS, NULL, &dwType, (LPBYTE)&dwFlags, &cbSize) != ERROR_SUCCESS)
{
dwFlags = 0;
}
comp.iComponentType = dwFlags & COMP_TYPE;
comp.fChecked = (dwFlags & COMP_SELECTED) != 0;
comp.fNoScroll = (dwFlags & COMP_NOSCROLL) != 0;
comp.fDirty = FALSE; //Reading it fresh from registry; Can't be dirty!
// Read in the location.
cbSize = SIZEOF(comp.cpPos);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_POSITION, NULL, &dwType, (LPBYTE)&comp.cpPos, &cbSize) != ERROR_SUCCESS)
{
ZeroMemory(&comp.cpPos, SIZEOF(comp.cpPos));
}
// In IE4.x, we have a very huge positive number (0x7fffffff) as the COMPONENT_TOP;
// As a result some component's z-index overflowed into the negative range (0x80000003)
// To fix this, we halved the COMPONENT_TOP (0x3fffffff) and also check for negative z-index
// values and covert them to postive values.
if (comp.cpPos.izIndex < 0)
comp.cpPos.izIndex = COMPONENT_TOP;
// Make sure the cpPos.dwSize is set to correct value
comp.cpPos.dwSize = sizeof(COMPPOS);
// Read in the current ItemState
cbSize = SIZEOF(comp.dwCurItemState);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_CURSTATE, NULL, &dwType, (LPBYTE)&comp.dwCurItemState, &cbSize) != ERROR_SUCCESS)
{
//If the item state is missing, we must be reading from IE4 machine.
comp.dwCurItemState = IS_NORMAL;
}
// Read in the Original state info.
cbSize = SIZEOF(comp.csiOriginal);
if ((SHQueryValueEx(hkeyComp, REG_VAL_COMP_ORIGINALSTATEINFO, NULL, &dwType, (LPBYTE)&comp.csiOriginal, &cbSize) != ERROR_SUCCESS) ||
(comp.csiOriginal.dwSize != SIZEOF(comp.csiOriginal)))
{
//If the item state is missing, we must be reading from IE4 machine.
// Set the OriginalState to the default info.
SetStateInfo(&comp.csiOriginal, &comp.cpPos, IS_NORMAL);
comp.csiOriginal.dwHeight = comp.csiOriginal.dwWidth = COMPONENT_DEFAULT_WIDTH;
}
// Read in the Restored state info.
cbSize = SIZEOF(comp.csiRestored);
if (SHQueryValueEx(hkeyComp, REG_VAL_COMP_RESTOREDSTATEINFO, NULL, &dwType, (LPBYTE)&comp.csiRestored, &cbSize) != ERROR_SUCCESS)
{
//If the item state is missing, we must be reading from IE4 machine.
// Set the restored State to the default info.
SetStateInfo(&comp.csiRestored, &comp.cpPos, IS_NORMAL);
}
// Add the component to the component list.
AddComponentPrivate(&comp, StrToInt(pszComp));
// Increment our counter so we know where to add any new
// components after we're done.
_dwNextID++;
RegCloseKey(hkeyComp);
}
EXITPROC(2, "DS ReadComponent!");
}
typedef struct _tagSortStruct {
int ihdsaIndex;
int izIndex;
} SORTSTRUCT;
int CALLBACK pfnComponentSort(LPVOID p1, LPVOID p2, LPARAM lParam)
{
SORTSTRUCT * pss1 = (SORTSTRUCT *)p1;
SORTSTRUCT * pss2 = (SORTSTRUCT *)p2;
if (pss1->izIndex > pss2->izIndex)
return 1;
if (pss1->izIndex < pss2->izIndex)
return -1;
return(0);
}
// ModifyZIndex
// Little helper function to put the zindex of the windowed and windowless components
// into correct buckets so that zorting will produce a correct order by zindex.
// If we don't do this then windowless components may end up zordering above windowed ones.
void ModifyZIndex(COMPONENTA * pcomp)
{
if (pcomp->cpPos.izIndex != COMPONENT_TOP) {
if (!IsWindowLessComponent(pcomp))
pcomp->cpPos.izIndex += COMPONENT_TOP_WINDOWLESS;
}
else
{
if (IsWindowLessComponent(pcomp))
pcomp->cpPos.izIndex = COMPONENT_TOP_WINDOWLESS;
}
}
// SortAndRationalize
// SortAndRationalize will take an unsorted component list and sort it such that the components
// come out in the correct z-index indicated order. It will also rebase the z-index values at
// a known constant so that the z-index values will not grow endlessly. SortAndRationalize also
// imposes windowed vs. windowless criteria to the zindex values such that windowless components
// always zorder under windowed ones.
void CActiveDesktop::_SortAndRationalize(void)
{
int icComponents;
HDPA hdpa;
if (_hdsaComponent && ((icComponents = DSA_GetItemCount(_hdsaComponent)) > 1) && (hdpa = DPA_Create(0))) {
COMPONENTA * pcomp;
SORTSTRUCT * pss;
int i, iCur = ZINDEX_START;
BOOL fInsertFailed = FALSE;
HDSA hdsaOld;
// Go through each component and insert it's hdsa-index and zindex into the hdpa
for (i = 0; i < icComponents; i++)
{
if (!(pss = (SORTSTRUCT *)LocalAlloc(LPTR, sizeof(SORTSTRUCT))))
break;
pcomp = (COMPONENTA *)DSA_GetItemPtr(_hdsaComponent, i);
ModifyZIndex(pcomp);
pss->ihdsaIndex = i;
pss->izIndex = pcomp->cpPos.izIndex;
if (DPA_AppendPtr(hdpa, (void FAR *)pss) == -1) {
LocalFree((HANDLE)pss);
break;
}
}
// Sort the hdpa by zindex
DPA_Sort(hdpa, pfnComponentSort, 0);
// Save old values
hdsaOld = _hdsaComponent;
// Null out the old hdsa, so AddComponentPrivate will create a new one
_hdsaComponent = NULL;
// Now go through the sorted hdpa and update the component zindex with a ZINDEX_START based zindex, then
// add the component to the new hdsa in sorted order.
for (i = 0; i < icComponents; i++) {
if (!(pss = (SORTSTRUCT *)DPA_GetPtr(hdpa, i)))
break;
// Get component and update it's zIndex and id
pcomp = (COMPONENTA *)DSA_GetItemPtr(hdsaOld, pss->ihdsaIndex);
pcomp->cpPos.izIndex = iCur;
iCur += 2;
// Free ptr
LocalFree((HANDLE)pss);
// Add to new hdsa in sorted order
if (!fInsertFailed) {
fInsertFailed = !AddComponentPrivate(pcomp, pcomp->dwID);
}
}
// If we're completely successfull then destroy the old hdsa. Otherwise we need
// to destroy the new one and restore the old one.
if ((i == icComponents) && !fInsertFailed) {
DSA_Destroy(hdsaOld);
}
else {
if (_hdsaComponent)
DSA_Destroy(_hdsaComponent);
_hdsaComponent = hdsaOld;
}
DPA_Destroy(hdpa);
}
}
void CActiveDesktop::_ReadComponents(BOOL fActiveDesktop)
{
ENTERPROC(2, "DS ReadComponents()");
HKEY hkey;
TCHAR lpszDeskcomp[MAX_PATH];
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_COMPONENTS, _pszScheme);
if (RegOpenKeyEx(HKEY_CURRENT_USER, lpszDeskcomp, 0, KEY_READ, &hkey) == ERROR_SUCCESS)
{
DWORD cbSize, dwType;
int i = 0;
TCHAR lpszSubkey[MAX_PATH];
// Read in the general settings.
DWORD dwSettings;
cbSize = SIZEOF(dwSettings);
if (SHQueryValueEx(hkey, REG_VAL_COMP_SETTINGS, NULL, &dwType, (LPBYTE)&dwSettings, &cbSize) == ERROR_SUCCESS)
{
_co.fEnableComponents = (dwSettings & COMPSETTING_ENABLE) != 0;
}
_co.fActiveDesktop = fActiveDesktop;
// Read in all the desktop components
while (RegEnumKey(hkey, i, lpszSubkey, ARRAYSIZE(lpszSubkey)) == ERROR_SUCCESS)
{
_ReadComponent(hkey, lpszSubkey);
i++;
}
_SortAndRationalize();
RegCloseKey(hkey);
}
EXITPROC(2, "DS ReadComponents!");
}
void CActiveDesktop::_Initialize(void)
{
ENTERPROC(2, "DS Initialize()");
if (!_fInitialized)
{
_fInitialized = TRUE;
InitDeskHtmlGlobals();
SHELLSTATE ss = { 0 };
SHGetSetSettings(&ss, SSF_DESKTOPHTML, FALSE);
BOOL fActiveDesktop = BOOLIFY(ss.fDesktopHTML);
_co.dwSize = SIZEOF(_co);
_wpo.dwSize = SIZEOF(_wpo);
// This per-user registry branch may not exist for this user. Or, even if
// it does exist, it may have some stale info. So ensure that atlreast the
// default components are there and that the html version is current for this
// branch of the registry!
// If everything is current, the following function does nothing!
CDeskHtmlProp_RegUnReg(TRUE); //TRUE => install.
_ReadWallpaper(fActiveDesktop);
_ReadPattern();
_ReadComponents(fActiveDesktop);
// If we are in safemode, the we can not use Dynamic Html to make updates because
// updates involve complete change of background Html.
if (_IsDisplayInSafeMode())
_fUseDynamicHtml = FALSE;
else
_fUseDynamicHtml = TRUE; //Any component added after the initialization must go through dynamic html.
_fDirty = FALSE;
_fNeedBodyEnd = FALSE;
}
EXITPROC(2, "DS Initialize!");
}
void CActiveDesktop::_SaveWallpaper(void)
{
ENTERPROC(2, "DS SaveWallpaper");
TCHAR lpszDeskcomp[MAX_PATH];
BOOL fNormalWallpaper;
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_GENERAL, _pszScheme);
// Compute tiling string.
TCHAR szTiled[2];
lstrcpy(szTiled, TEXT("0"));
szTiled[0] = szTiled[0] + (TCHAR)(_wpo.dwStyle & WPSTYLE_TILE);
// Compute the Wallpaper styling string
TCHAR szWPStyle[2];
lstrcpy(szWPStyle, TEXT("0"));
// NOTE: If WPSTYLE_TILE is set, we still want to say WallpaperStyle="0"; This won't hurt
// because TileWallpaper="1" will over-ride this anyway.
// The reason for this hack is that during memphis setup, they put a tiled wallpaper. Then we
// write WallpaperStyle=1 and TileWallpaper=1 in new and old locations. Then, then change
// the wallpaper and set TileWallpaper=0. Since the WallpaperStyle continues to be 1, they
// get a tiled wallpaper finally. The following is to avoid this problem!
szWPStyle[0] = szWPStyle[0] + (TCHAR)(_wpo.dwStyle & WPSTYLE_STRETCH);
// Write out wallpaper settings in new active desktop area.
if (_fWallpaperDirty || _fWallpaperChangedDuringInit)
{
if (!_fPolicyForWPStyle)
{
SHSetValue(HKEY_CURRENT_USER, lpszDeskcomp, c_szTileWall, REG_SZ, (LPBYTE)szTiled, SIZEOF(TCHAR)*(lstrlen(szTiled) + 1));
}
// Note: We do not write the Wallpaper Style string for older systems because we do not
// want to over-write what PlusPack writes. However, for newer Operating systems, we
// want to write the WallpaperStyle also.
if ((g_bRunOnMemphis || g_bRunOnNT5) && (!_fPolicyForWPStyle))
{
SHSetValue(HKEY_CURRENT_USER, lpszDeskcomp,
c_szWallpaperStyle, REG_SZ, (LPBYTE)szWPStyle, SIZEOF(TCHAR)*(lstrlen(szWPStyle) + 1));
}
if (!_fPolicyForWPName)
SHRegSetPath(HKEY_CURRENT_USER, lpszDeskcomp, c_szWallpaper, _szSelectedWallpaper, 0);
}
if (fNormalWallpaper = IsNormalWallpaper(_szSelectedWallpaper))
{
lstrcpyn(_szBackupWallpaper, _szSelectedWallpaper, ARRAYSIZE(_szBackupWallpaper));
}
if (!_fPolicyForWPName)
{
FILETIME ft;
GetWallpaperFileTime(_szBackupWallpaper, &ft);
// Backup the "Old type" wallpaper's name here in the new location
// sothat we can detect when this gets changed by some other app.
SHRegSetPath(HKEY_CURRENT_USER, lpszDeskcomp, c_szBackupWallpaper, _szBackupWallpaper, 0);
SHSetValue(HKEY_CURRENT_USER, lpszDeskcomp, c_szWallpaperTime, REG_BINARY, (LPBYTE)&ft, SIZEOF(ft));
}
// Even if this wallpaper is not valid in normal desktop (i.e., even if it is not a .BMP),
// write it out in normal desktop registry area.
if (_fWallpaperDirty)
{
if (!_fPolicyForWPStyle)
{
SHSetValue(HKEY_CURRENT_USER, c_szRegStrDesktop,
c_szTileWall, REG_SZ, (LPBYTE)szTiled, SIZEOF(TCHAR)*(lstrlen(szTiled) + 1));
}
// Note: We do not write the Wallpaper Style string for older systems because we do not
// want to over-write what PlusPack writes. However, for newer Operating systems, we
// want to write the WallpaperStyle also.
if ((g_bRunOnMemphis || g_bRunOnNT5) && (!_fPolicyForWPStyle))
{
SHSetValue(HKEY_CURRENT_USER, c_szRegStrDesktop, c_szWallpaperStyle, REG_SZ, (LPBYTE)szWPStyle, SIZEOF(TCHAR)*(lstrlen(szWPStyle) + 1));
}
if (!_fPolicyForWPName)
{
SystemParametersInfo(SPI_SETDESKWALLPAPER, 0, (fNormalWallpaper ? _szSelectedWallpaper : _szBackupWallpaper), SPIF_UPDATEINIFILE);
}
}
EXITPROC(2, "DS SaveWallpaper");
}
void CActiveDesktop::_SaveComponent(HKEY hkey, int iIndex, COMPONENTA *pcomp)
{
ENTERPROC(2, "DS SaveComponent(hkey=%08X,iIndex=%d,pcomp=%08X)", hkey, iIndex, pcomp);
TCHAR szSubKey[8];
HKEY hkeySub;
wsprintf(szSubKey, TEXT("%d"), iIndex);
if (RegCreateKey(hkey, szSubKey, &hkeySub) == ERROR_SUCCESS)
{
pcomp->fDirty = FALSE; //Since we are saving in the registry, reset this!
// Write out the source string and Friendly name string.
RegSetValueEx(hkeySub, REG_VAL_COMP_SOURCE, 0, REG_SZ, (LPBYTE)pcomp->szSource, (lstrlen(pcomp->szSource) + 1)*SIZEOF(TCHAR));
RegSetValueEx(hkeySub, REG_VAL_COMP_SUBSCRIBED_URL, 0, REG_SZ, (LPBYTE)pcomp->szSubscribedURL, (lstrlen(pcomp->szSubscribedURL) + 1)*SIZEOF(TCHAR));
RegSetValueEx(hkeySub, REG_VAL_COMP_NAME, 0, REG_SZ, (LPBYTE)pcomp->szFriendlyName, (lstrlen(pcomp->szFriendlyName) + 1)*SIZEOF(TCHAR));
// Compute and write out flags.
DWORD dwFlags = 0;
dwFlags |= pcomp->iComponentType;
if (pcomp->fChecked)
{
dwFlags |= COMP_SELECTED;
}
if (pcomp->fNoScroll)
{
dwFlags |= COMP_NOSCROLL;
}
RegSetValueEx(hkeySub, REG_VAL_COMP_FLAGS, 0, REG_DWORD, (LPBYTE)&dwFlags, SIZEOF(dwFlags));
// Write out the position.
RegSetValueEx(hkeySub, REG_VAL_COMP_POSITION, 0, REG_BINARY, (LPBYTE)&pcomp->cpPos, SIZEOF(pcomp->cpPos));
// Write out the Current state
RegSetValueEx(hkeySub, REG_VAL_COMP_CURSTATE, 0, REG_DWORD, (LPBYTE)&pcomp->dwCurItemState, SIZEOF(pcomp->dwCurItemState));
// Write out the Original State Info
RegSetValueEx(hkeySub, REG_VAL_COMP_ORIGINALSTATEINFO, 0, REG_BINARY, (LPBYTE)&pcomp->csiOriginal, SIZEOF(pcomp->csiOriginal));
// Write out the Restored State Info
RegSetValueEx(hkeySub, REG_VAL_COMP_RESTOREDSTATEINFO, 0, REG_BINARY, (LPBYTE)&pcomp->csiRestored, SIZEOF(pcomp->csiRestored));
RegCloseKey(hkeySub);
}
EXITPROC(2, "DS SaveComponent!");
}
void CActiveDesktop::_SaveComponents(void)
{
ENTERPROC(2, "DS SaveComponents");
DWORD dwType, dwFlags = 0, dwDataLength = SIZEOF(dwFlags);
int i;
TCHAR lpszDeskcomp[MAX_PATH];
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_COMPONENTS, _pszScheme);
// We need to preserve the old GENFLAGS, so read them now before we roach them.
SHGetValue(HKEY_CURRENT_USER, (LPCTSTR)lpszDeskcomp, REG_VAL_COMP_GENFLAGS, &dwType, (LPBYTE)(&dwFlags), &dwDataLength);
// Delete the entire registry key.
SHDeleteKey(HKEY_CURRENT_USER, lpszDeskcomp);
// Recreate the registry key.
HKEY hkey;
if (RegCreateKey(HKEY_CURRENT_USER, lpszDeskcomp, &hkey) == ERROR_SUCCESS)
{
// Write out the version number.
DWORD dw = CUR_DESKHTML_VERSION;
RegSetValueEx(hkey, REG_VAL_COMP_VERSION, 0, REG_DWORD, (LPBYTE)(&dw), SIZEOF(dw));
dw = CUR_DESKHTML_MINOR_VERSION;
RegSetValueEx(hkey, REG_VAL_COMP_MINOR_VERSION, 0, REG_DWORD, (LPBYTE)(&dw), SIZEOF(dw));
// Write out the general settings.
DWORD dwSettings = 0;
if (_co.fEnableComponents)
{
dwSettings |= COMPSETTING_ENABLE;
}
RegSetValueEx(hkey, REG_VAL_COMP_SETTINGS, 0, REG_DWORD, (LPBYTE)&dwSettings, SIZEOF(dwSettings));
// Write out the general flags
RegSetValueEx(hkey, REG_VAL_COMP_GENFLAGS, 0, REG_DWORD, (LPBYTE)&dwFlags, SIZEOF(dwFlags));
if (_hdsaComponent)
{
// Write out the settings for each component
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA * pcomp;
if (pcomp = (COMPONENTA *)DSA_GetItemPtr(_hdsaComponent, i))
{
pcomp->dwID = i;
_SaveComponent(hkey, i, pcomp);
}
}
}
RegCloseKey(hkey);
}
EXITPROC(2, "DS SaveComponents");
}
void CActiveDesktop::_SavePattern(DWORD dwFlags)
{
ENTERPROC(2, "DS SavePattern()");
if (_fPatternDirty && (dwFlags & SAVE_PATTERN_NAME))
{
// Write out the pattern to the registry and INI files.
SystemParametersInfo(SPI_SETDESKPATTERN, 0, _szSelectedPattern, SPIF_UPDATEINIFILE);
}
if (IsValidPattern(_szSelectedPattern) && (dwFlags & GENERATE_PATTERN_FILE))
{
// Write out the pattern as a BMP file for use in HTML.
TCHAR szBitmapFile[MAX_PATH];
GetPerUserFileName(szBitmapFile, ARRAYSIZE(szBitmapFile), PATTERN_FILENAME);
HANDLE hFileBitmap;
hFileBitmap = CreateFile(szBitmapFile, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM, NULL);
if (hFileBitmap != INVALID_HANDLE_VALUE)
{
DWORD cbWritten;
BITMAPFILEHEADER bmfh = { 0 };
bmfh.bfType = 0x4D42; // 'BM'
bmfh.bfSize = SIZEOF(BITMAPFILEHEADER) + SIZEOF(BITMAPINFOHEADER) + 2 * SIZEOF(RGBQUAD) + 8 * SIZEOF(DWORD);
bmfh.bfOffBits = SIZEOF(BITMAPFILEHEADER) + SIZEOF(BITMAPINFOHEADER) + 2 * SIZEOF(RGBQUAD);
WriteFile(hFileBitmap, &bmfh, SIZEOF(bmfh), &cbWritten, NULL);
BITMAPINFOHEADER bmih = { 0 };
bmih.biSize = SIZEOF(BITMAPINFOHEADER);
bmih.biWidth = 8;
bmih.biHeight = 8;
bmih.biPlanes = 1;
bmih.biBitCount = 1;
bmih.biCompression = BI_RGB;
WriteFile(hFileBitmap, &bmih, SIZEOF(bmih), &cbWritten, NULL);
RGBQUAD argbTable[2] = { 0 };
DWORD rgb;
rgb = GetSysColor(COLOR_BACKGROUND);
argbTable[0].rgbBlue = GetBValue(rgb);
argbTable[0].rgbGreen = GetGValue(rgb);
argbTable[0].rgbRed = GetRValue(rgb);
rgb = GetSysColor(COLOR_WINDOWTEXT);
argbTable[1].rgbBlue = GetBValue(rgb);
argbTable[1].rgbGreen = GetGValue(rgb);
argbTable[1].rgbRed = GetRValue(rgb);
WriteFile(hFileBitmap, argbTable, SIZEOF(argbTable), &cbWritten, NULL);
DWORD adwBits[8];
PatternToDwords(_szSelectedPattern, adwBits);
WriteFile(hFileBitmap, adwBits, SIZEOF(adwBits), &cbWritten, NULL);
CloseHandle(hFileBitmap);
}
}
EXITPROC(2, "DS SavePattern!");
}
void CActiveDesktop::_WriteHtmlFromString(LPCTSTR psz)
{
ENTERPROC(3, "DS WriteHtmlFromString(psz=>%s<)", psz);
LPCWSTR pwsz;
WCHAR szBuf[INTERNET_MAX_URL_LENGTH];
UINT uiLen;
int cch;
#ifdef UNICODE
if ((_pStream == NULL) && (_iDestFileCharset == ANSI_HTML_CHARSET))
{
cch = SHUnicodeToAnsi(psz, (LPSTR)szBuf, ARRAYSIZE(szBuf));
ASSERT(cch == lstrlenW((LPWSTR)psz) + 1);
pwsz = (LPCWSTR)szBuf;
uiLen = lstrlenA((LPSTR)szBuf);
}
else
{
pwsz = psz;
uiLen = lstrlenW(pwsz);
}
#else
if ((_pStream == NULL) && (_iDestFileCharset == ANSI_HTML_CHARSET))
{
pwsz = (LPWSTR)psz;
uiLen = lstrlenA(psz);
}
else
{
cch = SHAnsiToUnicode(psz, szBuf, ARRAYSIZE(szBuf));
ASSERT(cch == lstrlenA(psz) + 1);
pwsz = (LPCWSTR)szBuf;
uiLen = lstrlenW(pwsz);
}
#endif
UINT cbWritten;
_WriteHtmlW(pwsz, uiLen, &cbWritten);
EXITPROC(3, "DS WriteHtmlFromString!");
}
void CActiveDesktop::_WriteHtmlFromId(UINT uid)
{
ENTERPROC(3, "DS WriteHtmlFromId(uid=%d)", uid);
TCHAR szBuf[INTERNET_MAX_URL_LENGTH];
LoadString(HINST_THISDLL, uid, szBuf, ARRAYSIZE(szBuf));
_WriteHtmlFromString(szBuf);
EXITPROC(3, "DS WriteHtmlFromId!");
}
void CActiveDesktop::_WriteHtmlFromIdF(UINT uid, ...)
{
ENTERPROC(3, "DS WriteHtmlFromIdF(uid=%d,...)", uid);
TCHAR szBufFmt[INTERNET_MAX_URL_LENGTH];
TCHAR szBuf[INTERNET_MAX_URL_LENGTH];
LoadString(HINST_THISDLL, uid, szBufFmt, ARRAYSIZE(szBufFmt));
va_list arglist;
va_start(arglist, uid);
wvsprintf(szBuf, szBufFmt, arglist);
va_end(arglist);
_WriteHtmlFromString(szBuf);
EXITPROC(3, "DS WriteHtmlFromIdF!");
}
void CActiveDesktop::_WriteHtmlFromFile(LPCTSTR pszContents)
{
ENTERPROC(3, "DS WriteHtmlFromFile(pszContents=>%s<)", pszContents);
CReadFileObj *pReadFileObj = new CReadFileObj(pszContents);
if (pReadFileObj)
{
if (pReadFileObj->_hFile != INVALID_HANDLE_VALUE)
{
WCHAR wcBuf[INTERNET_MAX_URL_LENGTH + 1];
UINT uiCharCount = ARRAYSIZE(wcBuf) - 1; //Leave room for null termination.
UINT uiCharsRead;
UINT uiCharsConverted;
int iDestCharset = (_pStream ? UNICODE_HTML_CHARSET : _iDestFileCharset);
while (SUCCEEDED(pReadFileObj->FileReadAndConvertChars(iDestCharset, wcBuf, uiCharCount, &uiCharsRead, &uiCharsConverted)) && uiCharsRead)
{
UINT cbWritten;
_WriteHtmlW(wcBuf, uiCharsConverted, &cbWritten);
if (uiCharsRead < uiCharCount)
{
break;
}
}
}
delete pReadFileObj;
}
EXITPROC(3, "DS WriteHtmlFromFile!");
}
void CActiveDesktop::_WriteHtmlFromReadFileObj(CReadFileObj *pFileObj, int iOffsetStart, int iOffsetEnd)
{
ENTERPROC(3, "DS WriteHtmlFromReadFileObj(pFileObj=%08X,iOffsetStart=%d,iOffsetEnd=%d)", pFileObj, iOffsetStart, iOffsetEnd);
if (iOffsetStart != -1)
{
pFileObj->FileSeekChars(iOffsetStart, FILE_BEGIN);
}
else
{
ASSERT(iOffsetEnd == -1);
iOffsetEnd = -1;
}
//Get the number of WIDECHARs to be written
UINT cchWrite = (iOffsetEnd == -1) ? 0xFFFFFFFF : (iOffsetEnd - iOffsetStart);
while (cchWrite)
{
WCHAR wcBuf[INTERNET_MAX_URL_LENGTH + 1];
// Read a chunk.
UINT cchTryRead = min(cchWrite, (ARRAYSIZE(wcBuf) - 1));
UINT cchActualRead;
HRESULT hres;
//Note: if we are reading ANSI, we still use the unicode buff; but cast it!
if (_iDestFileCharset == ANSI_HTML_CHARSET)
hres = pFileObj->FileReadCharsA((LPSTR)wcBuf, cchTryRead, &cchActualRead);
else
hres = pFileObj->FileReadCharsW(wcBuf, cchTryRead, &cchActualRead);
if (SUCCEEDED(hres) && cchActualRead)
{
// Write a chunk.
UINT cchWritten;
_WriteHtmlW(wcBuf, cchActualRead, &cchWritten);
if (cchActualRead < cchTryRead)
{
// End of file, all done.
break;
}
cchWrite -= cchActualRead;
}
else
{
// Error reading from file, all done.
break;
}
}
EXITPROC(3, "DS WriteHtmlFromHfile!");
}
int CActiveDesktop::_ScanForTagA(CReadFileObj *pFileObj, int iOffsetStart, LPCSTR pszTag)
{
ENTERPROC(2, "DS ScanForTagA(pFileObj=%08X,iOffsetStart=%d,pszTagA=>%s<)",
pFileObj, iOffsetStart, pszTag);
int iRet = -1;
BOOL fDoneReading = FALSE;
int iOffset;
DWORD cchTag = lstrlenA(pszTag);
pFileObj->FileSeekChars(iOffsetStart, FILE_BEGIN);
iOffset = iOffsetStart;
DWORD cchBuf = 0;
while (!fDoneReading)
{
char szBuf[INTERNET_MAX_URL_LENGTH + 1];
// Fill in the buffer.
UINT cchTryRead = ARRAYSIZE(szBuf) - cchBuf - 1;
UINT cchRead;
if (SUCCEEDED(pFileObj->FileReadCharsA(&szBuf[cchBuf], cchTryRead, &cchRead)) && cchRead)
{
cchBuf += cchRead;
// Terminate the string.
szBuf[cchBuf] = '\0';
// Scan for the tag.
LPSTR pszTagInBuf = StrStrIA(szBuf, pszTag);
if (pszTagInBuf)
{
// Found the tag, compute the offset.
iRet = (int)(iOffset + pszTagInBuf - szBuf);
fDoneReading = TRUE;
}
else if (cchRead < cchTryRead)
{
// Ran out of file without finding tag.
fDoneReading = TRUE;
}
else
{
// Compute how many bytes we want to throw away
// from this buffer so we can read in more data.
// We don't want to throw away all the bytes because
// the tag we want may span two buffers.
DWORD cchSkip = cchBuf - cchTag;
// Advance the file offset.
iOffset += cchSkip;
// Reduce the buffer size.
cchBuf -= cchSkip;
// Move the kept bytes to the beginning of the buffer.
MoveMemory(szBuf, szBuf + cchSkip, cchBuf);
}
}
else
{
fDoneReading = TRUE;
}
}
EXITPROC(2, "DS ScanForTagA=%d", iRet);
return iRet;
}
int CActiveDesktop::_ScanForTagW(CReadFileObj *pFileObj, int iOffsetStart, LPCWSTR pwszTag)
{
ENTERPROC(2, "DS ScanForTag(pFileObj=%08X,iOffsetStart=%d,pszTagA=>%s<)", pFileObj, iOffsetStart, pwszTag);
int iRet = -1;
BOOL fDoneReading = FALSE;
int iOffset;
DWORD cchTag = lstrlenW(pwszTag);
pFileObj->FileSeekChars(iOffsetStart, FILE_BEGIN);
iOffset = iOffsetStart;
DWORD cchBuf = 0;
while (!fDoneReading)
{
WCHAR wszBuf[INTERNET_MAX_URL_LENGTH + 1];
// Fill in the buffer.
UINT cchTryRead = ARRAYSIZE(wszBuf) - cchBuf - 1;
UINT cchRead;
if (SUCCEEDED(pFileObj->FileReadCharsW(&wszBuf[cchBuf], cchTryRead, &cchRead)) && cchRead)
{
cchBuf += cchRead;
// Terminate the string.
wszBuf[cchBuf] = L'\0';
// Scan for the tag.
LPWSTR pwszTagInBuf = StrStrIW(wszBuf, pwszTag);
if (pwszTagInBuf)
{
// Found the tag, compute the offset.
iRet = (int)(iOffset + pwszTagInBuf - wszBuf);
fDoneReading = TRUE;
}
else if (cchRead < cchTryRead)
{
// Ran out of file without finding tag.
fDoneReading = TRUE;
}
else
{
// Compute how many bytes we want to throw away
// from this buffer so we can read in more data.
// We don't want to throw away all the bytes because
// the tag we want may span two buffers.
DWORD cchSkip = cchBuf - cchTag;
iOffset += cchSkip;// Advance the file offset.
cchBuf -= cchSkip;// Reduce the buffer size.
// Move the kept bytes to the beginning of the buffer.
MoveMemory(wszBuf, wszBuf + cchSkip, cchBuf * sizeof(WCHAR));
}
}
else
{
fDoneReading = TRUE;
}
}
EXITPROC(2, "DS ScanForTag=%d", iRet);
return iRet;
}
int CActiveDesktop::_ScanTagEntriesA(CReadFileObj *pReadFileObj, int iOffsetStart, TAGENTRYA *pte, int cte)
{
ENTERPROC(2, "DS ScanTagEntriesA(pReadFileObj=%08X,iOffsetStart=%d,pte=%08X,cte=%d)",
pReadFileObj, iOffsetStart, pte, cte);
int iRet = -1;
int i;
for (i = 0; i < cte; i++, pte++)
{
iRet = _ScanForTagA(pReadFileObj, iOffsetStart, pte->pszTag);
if (iRet != -1)
{
if (pte->fSkipPast)
{
iRet += lstrlenA(pte->pszTag);
}
break;
}
}
EXITPROC(2, "DS ScanTagEntriesA=%d", iRet);
return iRet;
}
int CActiveDesktop::_ScanTagEntriesW(CReadFileObj *pReadFileObj, int iOffsetStart, TAGENTRYW *pte, int cte)
{
ENTERPROC(2, "DS ScanTagEntriesW(pReadFileObj=%08X,iOffsetStart=%d,pte=%08X,cte=%d)",
pReadFileObj, iOffsetStart, pte, cte);
int iRet = -1;
int i;
for (i = 0; i < cte; i++, pte++)
{
iRet = _ScanForTagW(pReadFileObj, iOffsetStart, pte->pwszTag);
if (iRet != -1)
{
if (pte->fSkipPast)
{
iRet += lstrlenW(pte->pwszTag);
}
break;
}
}
EXITPROC(2, "DS ScanTagEntriesW=%d", iRet);
return iRet;
}
void CActiveDesktop::_ParseAnsiInputHtmlFile(LPTSTR szSelectedWallpaper, int *piOffsetBase, int *piOffsetComp)
{
// Figure out where to insert the base href tag.
int iOffsetBase = 0, iBaseTagStart;
BOOL fUseBaseHref;
LONG lOffsetDueToBOM = 0; //Character Offset due to the Byte Order Mark.
//1 for UNICODE and 0 for ANSI files.
// 98/11/11 #248047 vtan: This code looks for a <BASE HREF=...> tag.
// It used to use a scan for "<BASE" and assume that this was the
// desired tag. HTML allows a "<BASEFONT>" tag which was being
// mistaken for a "<BASE HREF=...>" tag. The code now looks for the
// same string but looks at the character following the "<BASE" to
// see if it's a white-space character.
fUseBaseHref = TRUE;
_pReadFileObjHtmlBkgd->FileGetCurCharOffset(&lOffsetDueToBOM);
iOffsetBase = (int)lOffsetDueToBOM;
iBaseTagStart = _ScanForTagA(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, "<BASE");
if (iBaseTagStart != -1)
{
UINT uiCountChars, uiTryToRead;
char szBaseTagBuffer[6 + 1]; // allow for "<BASEx" plus a NULL.
_pReadFileObjHtmlBkgd->FileSeekChars(iBaseTagStart, FILE_BEGIN);
uiTryToRead = ARRAYSIZE(szBaseTagBuffer) - 1;
if (SUCCEEDED(_pReadFileObjHtmlBkgd->FileReadCharsA(szBaseTagBuffer, uiTryToRead, &uiCountChars)) && uiCountChars)
{
char ch;
ch = szBaseTagBuffer[5];
fUseBaseHref = ((ch != ' ') &&
(ch != '\r') &&
(ch != '\n') && // this covers the UNIX line break scheme
(ch != '\t'));
}
}
if (fUseBaseHref)
{
TAGENTRYA rgteBase[] = {
{ "<HEAD>", TRUE, },
{ "<BODY", FALSE, },
{ "<HTML>", TRUE, },
};
iOffsetBase = _ScanTagEntriesA(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, rgteBase, ARRAYSIZE(rgteBase));
if (iOffsetBase == -1)
{
iOffsetBase = (int)lOffsetDueToBOM;
}
}
// Figure out where to insert the components.
TAGENTRYA rgteComponents[] = {
{ "</BODY>", FALSE, },
{ "</HTML>", FALSE, },
};
int iOffsetComponents = _ScanTagEntriesA(_pReadFileObjHtmlBkgd, iOffsetBase, rgteComponents, ARRAYSIZE(rgteComponents));
// Write out the initial HTML up to the <HEAD> tag.
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, iOffsetBase);
// Write out the base tag.
if (fUseBaseHref)
{
//BASE tag must point to the base "URL". So, don't strip out the filename.
_WriteHtmlFromIdF(IDS_BASE_TAG, szSelectedWallpaper);
}
// Figure out where to insert the DIV clause
TAGENTRYA rgteBodyStart[] = {
{ "<BODY", FALSE, },
};
int iOffsetBodyStart = _ScanTagEntriesA(_pReadFileObjHtmlBkgd, iOffsetBase, rgteBodyStart, ARRAYSIZE(rgteBodyStart));
// Write out HTML until after the <BODY ......>
if (iOffsetBodyStart == -1)
{ // the <BODY> tag is not found, so we need to insert it.
// Copy over stuff until </HEAD>
TAGENTRYA rgteHeadEnd[] = {
{ "</HEAD>", TRUE, },
};
int iOffsetHeadEnd = _ScanTagEntriesA(_pReadFileObjHtmlBkgd, iOffsetBase, rgteHeadEnd, ARRAYSIZE(rgteHeadEnd));
if (iOffsetHeadEnd != -1)
{
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, iOffsetBase, iOffsetHeadEnd);
iOffsetBase = iOffsetHeadEnd;
}
_WriteHtmlFromIdF(IDS_BODY_CENTER_WP2); // "<BODY>"
_fNeedBodyEnd = TRUE;
}
else
{
TAGENTRYA rgteBodyEnd[] = {
{ ">", TRUE, },
};
int iOffsetBodyEnd = _ScanTagEntriesA(_pReadFileObjHtmlBkgd, iOffsetBodyStart, rgteBodyEnd, ARRAYSIZE(rgteBodyEnd));
if (iOffsetBodyEnd == -1)
{ // An error in the HTML.
iOffsetBodyEnd = iOffsetBodyStart; // BUGBUG: We need a better recovery idea.
}
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, iOffsetBase, iOffsetBodyEnd);
iOffsetBase = iOffsetBodyEnd;
}
*piOffsetBase = iOffsetBase;
*piOffsetComp = iOffsetComponents;
}
void CActiveDesktop::_GenerateHtmlHeader(void)
{
ENTERPROC(2, "DS GenerateHtmlHeader()");
EnumMonitorsArea ema;
GetMonitorSettings(&ema);
RECT rcViewAreas[LV_MAX_WORKAREAS]; // WorkArea minus toolbar/tray areas
int nViewAreas = ARRAYSIZE(rcViewAreas);
// Get the ViewAreas
if (!GetViewAreas(rcViewAreas, &nViewAreas))
{
nViewAreas = 0;
}
//Assume that the final Deskstat.htt that we generate is going to be in UNICODE.
//This will change to ANSI only if the background html wallpaper is ANSI (determined later)
_iDestFileCharset = UNICODE_HTML_CHARSET;
// Write out the background and color.
TCHAR szSelectedWallpaper[INTERNET_MAX_URL_LENGTH];
// If the wallpaper does not have a directory specified (this may happen if other apps. change this value),
// we have to figure it out.
GetWallpaperWithPath(_szSelectedWallpaper, szSelectedWallpaper, ARRAYSIZE(szSelectedWallpaper));
BOOL fValidWallpaper = GetFileAttributes(szSelectedWallpaper) != 0xFFFFFFFF;
if (_fSingleItem || IsWallpaperPicture(szSelectedWallpaper) || !fValidWallpaper)
{
// Write the BOM for UNICODE
if (_hFileHtml)
{
DWORD cbWritten;
WriteFile(_hFileHtml, (LPCSTR)&wUnicodeBOM, sizeof(wUnicodeBOM), &cbWritten, NULL);
}
// To account for the vagaries of the desktop browser (it's TopLeft starts from the TopLeft
// of the Desktop ViewArea instead of the TopLeft of the monitor, as might be expected)
// which happens only in the case of one active monitor systems, we add the width of the
// tray/toolbars to the co-ordinates of the DIV section of each monitor's wallpaper.
int iLeft, iTop;
if (nViewAreas == 1)
{
iLeft = rcViewAreas[0].left - ema.rcVirtualMonitor.left;
iTop = rcViewAreas[0].top - ema.rcVirtualMonitor.top;
}
else
{
iLeft = 0;
iTop = 0;
}
// Write out the standard header.
UINT i;
for (i = IDS_COMMENT_BEGIN; i < IDS_BODY_BEGIN; i++)
{
_WriteHtmlFromIdF(i);
}
// Write out the body tag, with background bitmap.
DWORD rgbDesk;
rgbDesk = GetSysColor(COLOR_DESKTOP);
TCHAR szBitmapFile[MAX_PATH];
GetPerUserFileName(szBitmapFile, ARRAYSIZE(szBitmapFile), PATTERN_FILENAME);
if (!_fSingleItem && _szSelectedWallpaper[0] && fValidWallpaper)
{
TCHAR szWallpaperUrl[INTERNET_MAX_URL_LENGTH];
DWORD cch = ARRAYSIZE(szWallpaperUrl);
UrlCreateFromPath(szSelectedWallpaper, szWallpaperUrl, &cch, URL_INTERNAL_PATH);
switch (_wpo.dwStyle)
{
case WPSTYLE_TILE:
// Ignore the pattern, tile the wallpaper as background.
_WriteHtmlFromIdF(IDS_BODY_BEGIN2, szWallpaperUrl, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
break;
case WPSTYLE_CENTER:
if (IsValidPattern(_szSelectedPattern))
{
// Tile the pattern as the main background.
_WriteHtmlFromIdF(IDS_BODY_BEGIN2, szBitmapFile, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
if (_fBackgroundHtml) // We are generating the HTML for preview
{
_WriteHtmlFromIdF(IDS_BODY_PATTERN_AND_WP, szWallpaperUrl);
}
else
{
// Write out a DIV section for a centered, untiled wallpaper.
// write it out for each monitor.
for (int i = 0; i < ema.iMonitors; i++)
{
_WriteHtmlFromIdF(IDS_BODY_PATTERN_AND_WP2,
ema.rcMonitor[i].left - ema.rcVirtualMonitor.left - iLeft,
ema.rcMonitor[i].top - ema.rcVirtualMonitor.top - iTop,
ema.rcMonitor[i].right - ema.rcMonitor[i].left,
ema.rcMonitor[i].bottom - ema.rcMonitor[i].top,
szWallpaperUrl);
}
}
}
else
{
// Write out a non-tiled, centered wallpaper as background.
if (_fBackgroundHtml) // We are generating the HTML for preview
{
_WriteHtmlFromIdF(IDS_BODY_CENTER_WP, szWallpaperUrl, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
}
else
{
_WriteHtmlFromIdF(IDS_BODY_CENTER_WP2, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
// write it out for each monitor.
for (int i = 0; i < ema.iMonitors; i++)
{
_WriteHtmlFromIdF(IDS_BODY_PATTERN_AND_WP2,
ema.rcMonitor[i].left - ema.rcVirtualMonitor.left - iLeft,
ema.rcMonitor[i].top - ema.rcVirtualMonitor.top - iTop,
ema.rcMonitor[i].right - ema.rcMonitor[i].left,
ema.rcMonitor[i].bottom - ema.rcMonitor[i].top,
szWallpaperUrl);
}
}
}
break;
case WPSTYLE_STRETCH:
// Ignore the pattern, create a DIV section of the wallpaper
// stretched to 100% of the screen.
_WriteHtmlFromIdF(IDS_BODY_BEGIN2, c_szNULL, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
if (_fBackgroundHtml) // We are generating the HTML for preview
{
_WriteHtmlFromIdF(IDS_STRETCH_WALLPAPER, szWallpaperUrl);
}
else
{
// stretch it for each monitor.
for (int i = 0; i < ema.iMonitors; i++)
{
_WriteHtmlFromIdF(IDS_DIV_START3, ema.rcMonitor[i].left - ema.rcVirtualMonitor.left - iLeft,
ema.rcMonitor[i].top - ema.rcVirtualMonitor.top - iTop,
ema.rcMonitor[i].right - ema.rcMonitor[i].left,
ema.rcMonitor[i].bottom - ema.rcMonitor[i].top);
_WriteHtmlFromIdF(IDS_STRETCH_WALLPAPER, szWallpaperUrl);
_WriteHtmlFromId(IDS_DIV_END);
}
}
break;
}
}
else
{
// Ignore the wallpaper, generate either a tiled pattern
// or solid color background.
_WriteHtmlFromIdF(IDS_BODY_BEGIN2, !_fSingleItem && IsValidPattern(_szSelectedPattern) ? szBitmapFile : c_szNULL, GetRValue(rgbDesk), GetGValue(rgbDesk), GetBValue(rgbDesk));
}
}
else
{
if ((_pReadFileObjHtmlBkgd = new CReadFileObj(szSelectedWallpaper)) &&
(_pReadFileObjHtmlBkgd->_hFile != INVALID_HANDLE_VALUE))
{
//The final Desktop.htt will be in ANSI only if the source html file is also in ansi.
//So, get the type from the selected wallpaper object.
_iDestFileCharset = _pReadFileObjHtmlBkgd->_iCharset;
// Write the BOM for UNICODE
if (_hFileHtml && (_iDestFileCharset == UNICODE_HTML_CHARSET))
{
DWORD cbWritten;
WriteFile(_hFileHtml, (LPCSTR)&wUnicodeBOM, sizeof(wUnicodeBOM), &cbWritten, NULL);
}
// Figure out where to insert the base href tag.
int iOffsetBase = 0;
int iOffsetComponents;
// 98/11/11 #248047 vtan: This code looks for a <BASE HREF=...> tag.
// It used to use a scan for "<BASE" and assume that this was the
// desired tag. HTML allows a "<BASEFONT>" tag which was being
// mistaken for a "<BASE HREF=...>" tag. The code now looks for the
// same string but looks at the character following the "<BASE" to
// see if it's a white-space character.
if (_iDestFileCharset == ANSI_HTML_CHARSET)
{
//The following function parses the ANSI input html file and finds various offsets
_ParseAnsiInputHtmlFile(szSelectedWallpaper, &iOffsetBase, &iOffsetComponents);
}
else
{
//The following code parses the UNICODE input html wallpaper file.
int iBaseTagStart;
BOOL fUseBaseHref;
LONG lOffsetDueToBOM = 0; //Character Offset due to the Byte Order Mark.
//1 for UNICODE and 0 for ANSI files.
fUseBaseHref = TRUE;
_pReadFileObjHtmlBkgd->FileGetCurCharOffset(&lOffsetDueToBOM);
iOffsetBase = (int)lOffsetDueToBOM;
iBaseTagStart = _ScanForTagW(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, L"<BASE");
if (iBaseTagStart != -1)
{
UINT uiCountChars, uiTryToRead;
WCHAR wszBaseTagBuffer[6 + 1]; // allow for "<BASEx" plus a NULL.
_pReadFileObjHtmlBkgd->FileSeekChars(iBaseTagStart, FILE_BEGIN);
uiTryToRead = ARRAYSIZE(wszBaseTagBuffer) - 1;
if (SUCCEEDED(_pReadFileObjHtmlBkgd->FileReadCharsW(wszBaseTagBuffer, uiTryToRead, &uiCountChars)) && uiCountChars)
{
WCHAR wc;
wc = wszBaseTagBuffer[5];
fUseBaseHref = ((wc != L' ') &&
(wc != L'\r') &&
(wc != L'\n') && // this covers the UNIX line break scheme
(wc != L'\t'));
}
}
if (fUseBaseHref)
{
TAGENTRYW rgteBase[] = {
{ L"<HEAD>", TRUE, },
{ L"<BODY", FALSE, },
{ L"<HTML>", TRUE, },
};
iOffsetBase = _ScanTagEntriesW(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, rgteBase, ARRAYSIZE(rgteBase));
if (iOffsetBase == -1)
{
iOffsetBase = (int)lOffsetDueToBOM;
}
}
// Figure out where to insert the components.
TAGENTRYW rgteComponents[] = {
{ L"</BODY>", FALSE, },
{ L"</HTML>", FALSE, },
};
iOffsetComponents = _ScanTagEntriesW(_pReadFileObjHtmlBkgd, iOffsetBase, rgteComponents, ARRAYSIZE(rgteComponents));
// Write out the initial HTML up to the <HEAD> tag.
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, (int)lOffsetDueToBOM, iOffsetBase);
// Write out the base tag.
if (fUseBaseHref)
{
//BASE tag must point to the base "URL". So, don't strip out the filename.
_WriteHtmlFromIdF(IDS_BASE_TAG, szSelectedWallpaper);
}
// Figure out where to insert the DIV clause
TAGENTRYW rgteBodyStart[] = {
{ L"<BODY", FALSE, },
};
int iOffsetBodyStart = _ScanTagEntriesW(_pReadFileObjHtmlBkgd, iOffsetBase, rgteBodyStart, ARRAYSIZE(rgteBodyStart));
// Write out HTML until after the <BODY ......>
if (iOffsetBodyStart == -1)
{ // the <BODY> tag is not found, so we need to insert it.
// Copy over stuff until </HEAD>
TAGENTRYW rgteHeadEnd[] = {
{ L"</HEAD>", TRUE, },
};
int iOffsetHeadEnd = _ScanTagEntriesW(_pReadFileObjHtmlBkgd, iOffsetBase, rgteHeadEnd, ARRAYSIZE(rgteHeadEnd));
if (iOffsetHeadEnd != -1)
{
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, iOffsetBase, iOffsetHeadEnd);
iOffsetBase = iOffsetHeadEnd;
}
_WriteHtmlFromIdF(IDS_BODY_CENTER_WP2); // "<BODY>"
_fNeedBodyEnd = TRUE;
}
else
{
TAGENTRYW rgteBodyEnd[] = {
{ L">", TRUE, },
};
int iOffsetBodyEnd = _ScanTagEntriesW(_pReadFileObjHtmlBkgd, iOffsetBodyStart, rgteBodyEnd, ARRAYSIZE(rgteBodyEnd));
if (iOffsetBodyEnd == -1)
{ // An error in the HTML.
iOffsetBodyEnd = iOffsetBodyStart; // BUGBUG: We need a better recovery idea.
}
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, iOffsetBase, iOffsetBodyEnd);
iOffsetBase = iOffsetBodyEnd;
}
}
// Insert the DIV clause
if (ema.iMonitors > 1)
{
int iIndexPrimaryMonitor;
HMONITOR hMonitorPrimary;
MONITORINFO monitorInfo;
// 99/03/23 #275429 vtan: We used GetViewAreas() to fill in rcViewAreas above.
// The code here used to assume that [0] ALWAYS referred to the primary monitor.
// This isn't the case if the monitor settings are changed without a restart.
// In order to compensate for this and always render the wallpaper into the
// primary monitor, a search is performed to find a (left, top) that matches
// one of the work areas and this is used as the primary monitor. If none can
// be found then default to the old algorithm.
hMonitorPrimary = GetPrimaryMonitor();
monitorInfo.cbSize = sizeof(monitorInfo);
TBOOL(GetMonitorInfo(hMonitorPrimary, &monitorInfo));
iIndexPrimaryMonitor = -1;
for (int i = 0; (iIndexPrimaryMonitor < 0) && (i < nViewAreas); ++i)
{
if ((monitorInfo.rcWork.left == rcViewAreas[i].left) && (monitorInfo.rcWork.top == rcViewAreas[i].top))
{
iIndexPrimaryMonitor = i;
}
}
if (iIndexPrimaryMonitor < 0)
iIndexPrimaryMonitor = 0;
if ((nViewAreas <= 0) || (rcViewAreas[iIndexPrimaryMonitor].right == rcViewAreas[iIndexPrimaryMonitor].left))
// The second case could occur on bootup
{
// Some error occured when getting the ViewAreas. Recover from the error by using the workarea.
// Get the workarea of the primary monitor, since HTML wallpapers are displayed only there.
GetMonitorWorkArea(hMonitorPrimary, &rcViewAreas[iIndexPrimaryMonitor]);
}
_WriteHtmlFromIdF(IDS_DIV_START3,
rcViewAreas[iIndexPrimaryMonitor].left - ema.rcVirtualMonitor.left,
rcViewAreas[iIndexPrimaryMonitor].top - ema.rcVirtualMonitor.top,
rcViewAreas[iIndexPrimaryMonitor].right - rcViewAreas[iIndexPrimaryMonitor].left,
rcViewAreas[iIndexPrimaryMonitor].bottom - rcViewAreas[iIndexPrimaryMonitor].top);
}
// Write out HTML from after <HEAD> tag to just before </BODY> tag.
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, iOffsetBase, iOffsetComponents);
if (ema.iMonitors > 1)
{
_WriteHtmlFromId(IDS_DIV_END);
}
}
else
{
if (_pReadFileObjHtmlBkgd)
delete _pReadFileObjHtmlBkgd;
_pReadFileObjHtmlBkgd = NULL;
}
}
EXITPROC(2, "DS GenerateHtmlHeader!");
}
void CActiveDesktop::_WriteResizeable(COMPONENTA *pcomp)
{
TCHAR szResizeable[3];
szResizeable[0] = TEXT('\0');
//If Resize is set, then the comp is resizeable in both X and Y directions!
if (pcomp->cpPos.fCanResize)
lstrcat(szResizeable, TEXT("XY"));
else
{
if (pcomp->cpPos.fCanResizeX)
lstrcat(szResizeable, TEXT("X"));
if (pcomp->cpPos.fCanResizeY)
lstrcat(szResizeable, TEXT("Y"));
}
_WriteHtmlFromIdF(IDS_RESIZEABLE, szResizeable);
}
void CActiveDesktop::_WriteHtmlW(LPCWSTR wcBuf, UINT cchToWrite, UINT *pcchWritten)
{
ULONG cchWritten = 0;
UINT uiSize;
if (_pStream)
{
uiSize = sizeof(WCHAR);
_pStream->Write((LPVOID)wcBuf, cchToWrite * uiSize, &cchWritten);
}
else
{
ASSERT(_hFileHtml);
uiSize = (_iDestFileCharset == ANSI_HTML_CHARSET) ? sizeof(char) : sizeof(WCHAR);
WriteFile(_hFileHtml, (LPCVOID)wcBuf, cchToWrite * uiSize, &cchWritten, NULL);
}
*pcchWritten = (UINT)(cchWritten / uiSize); //Convert to number of chars.
}
void CActiveDesktop::_GenerateHtmlPicture(COMPONENTA *pcomp)
{
ENTERPROC(2, "DS GenerateHtmlPicture(pcomp=%08X)");
// Write out the image src HTML.
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
DWORD cch = ARRAYSIZE(szUrl);
if (FAILED(UrlCreateFromPath(pcomp->szSource, szUrl, &cch, 0)))
{
lstrcpy(szUrl, pcomp->szSource);
}
_WriteHtmlFromIdF(IDS_IMAGE_BEGIN2, pcomp->dwID, szUrl);
// Write out whether this image is resizeable or not!
_WriteResizeable(pcomp);
// Write out the URL that must be used for subscription purposes.
_WriteHtmlFromIdF(IDS_SUBSCRIBEDURL, pcomp->szSubscribedURL);
// Write out the image location HTML.
if ((pcomp->cpPos.dwWidth == COMPONENT_DEFAULT_WIDTH) &&
(pcomp->cpPos.dwHeight == COMPONENT_DEFAULT_HEIGHT))
{
_WriteHtmlFromIdF(IDS_IMAGE_LOCATION, _fSingleItem ? 0 : pcomp->cpPos.iLeft, _fSingleItem ? 0 : pcomp->cpPos.iTop, pcomp->cpPos.izIndex);
}
else
{
_WriteHtmlFromIdF(IDS_IMAGE_SIZE, _fSingleItem ? 0 : pcomp->cpPos.iLeft, _fSingleItem ? 0 : pcomp->cpPos.iTop,
pcomp->cpPos.dwWidth, pcomp->cpPos.dwHeight, pcomp->cpPos.izIndex);
}
EXITPROC(2, "DS GenerateHtmlPicture!");
}
void CActiveDesktop::_GenerateHtmlDoc(COMPONENTA *pcomp)
{
ENTERPROC(2, "DS GenerateHtmlDoc(pcomp=%08X)");
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
DWORD dwSize = ARRAYSIZE(szUrl);
LPTSTR lpszUrl = szUrl;
if (FAILED(UrlCreateFromPath(pcomp->szSource, szUrl, &dwSize, 0)))
lpszUrl = pcomp->szSource;
// Write out the DIV header HTML.
_WriteHtmlFromIdF(IDS_DIV_START2, pcomp->dwID, lpszUrl);
// Write out whether this component is resizeable or not!
_WriteResizeable(pcomp);
// Write out the DIV location HTML.
_WriteHtmlFromIdF(IDS_DIV_SIZE, pcomp->cpPos.dwHeight, _fSingleItem ? 0 : pcomp->cpPos.iLeft,
_fSingleItem ? 0 : pcomp->cpPos.iTop, pcomp->cpPos.dwWidth, pcomp->cpPos.izIndex);
// Extract the doc contents directly into the HTML.
_WriteHtmlFromFile(pcomp->szSource);
// Close the DIV section.
_WriteHtmlFromId(IDS_DIV_END);
EXITPROC(2, "DS GenerateHtmlDoc!");
}
void CActiveDesktop::_GenerateHtmlSite(COMPONENTA *pcomp)
{
ENTERPROC(2, "DS GenerateHtmlSite(pcomp=%08X)");
// Write out the frame src HTML.
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
DWORD cch = ARRAYSIZE(szUrl);
if (FAILED(UrlCreateFromPath(pcomp->szSource, szUrl, &cch, 0)))
{
lstrcpy(szUrl, pcomp->szSource);
}
DWORD currentURLLength, maximumURLLength;
TCHAR *pURL, formatBuffer[0x0100];
// 98/09/29 #211384 vtan: There is a limitation in wvsprintf.
// It only allows 2048 bytes in its buffer. If the URL is
// longer than 1024 characters less the IDS_IFRAME_BEGIN2
// string length less the component ID less "scrolling=no"
// if the component cannot be scrolled then the URL string
// will not be correctly inserted into the IDS_IFRAME_BEGIN2
// string and there will be a missing end-quote and trident
// will fail to render desktop.htt correctly.
// To correct against this the followING limits the length of
// the URL to this maximum and truncates any characters
// beyond the limit so that the IDS_IFRAME_BEGIN2 string
// contains its end-quote and trident does not barf.
// The above condition is a boundary condition and this
// check is quick so that the calculations that follow do
// not have to be executed repeatedly.
currentURLLength = lstrlen(szUrl);
if (currentURLLength > 768) // a hard-coded limit
{
maximumURLLength = 1024; // wvsprintf limit
LoadString(HINST_THISDLL, IDS_IFRAME_BEGIN2, formatBuffer, ARRAYSIZE(formatBuffer));
maximumURLLength -= lstrlen(formatBuffer); // IDS_IFRAME_BEGIN2
maximumURLLength -= 16; // pcomp->dwID
maximumURLLength -= lstrlen(TEXT("scrolling=no")); // pcomp->fNoScroll
if (currentURLLength > maximumURLLength)
szUrl[maximumURLLength] = static_cast<TCHAR>('\0');
}
_WriteHtmlFromIdF(IDS_IFRAME_BEGIN2, pcomp->dwID, szUrl, pcomp->fNoScroll ? TEXT("scrolling=no") : c_szNULL);
// Write out whether this Component is resizeable or not!
_WriteResizeable(pcomp);
// 98/09/29 #211384 vtan: See above.
currentURLLength = lstrlen(pcomp->szSubscribedURL);
if (currentURLLength > 768)
{
lstrcpy(szUrl, pcomp->szSubscribedURL);
maximumURLLength = 1024;
LoadString(HINST_THISDLL, IDS_SUBSCRIBEDURL, formatBuffer, ARRAYSIZE(formatBuffer));
maximumURLLength -= lstrlen(formatBuffer); // IDS_SUBSCRIBEDURL
if (currentURLLength > maximumURLLength)
szUrl[maximumURLLength] = static_cast<TCHAR>('\0');
pURL = szUrl;
}
else
pURL = pcomp->szSubscribedURL;
// Write out the URL that must be used for subscription purposes.
_WriteHtmlFromIdF(IDS_SUBSCRIBEDURL, pURL);
// Write out the frame location HTML.
_WriteHtmlFromIdF(IDS_IFRAME_SIZE, _fSingleItem ? 0 : pcomp->cpPos.iLeft, _fSingleItem ? 0 : pcomp->cpPos.iTop,
pcomp->cpPos.dwWidth, pcomp->cpPos.dwHeight, pcomp->cpPos.izIndex);
EXITPROC(2, "DS GenerateHtmlSite!");
}
void CActiveDesktop::_GenerateHtmlControl(COMPONENTA *pcomp)
{
ENTERPROC(2, "DS GenerateHtmlControl(pcomp=%08X)");
ASSERT(pcomp);
// Did the Administrator restrict "Channel UI"?
if (SHRestricted2W(REST_NoChannelUI, NULL, 0))
{
TCHAR szChannelOCGUID[GUIDSTR_MAX];
SHStringFromGUID(CLSID_ChannelOC, szChannelOCGUID, ARRAYSIZE(szChannelOCGUID));
if (!StrCmpNI(pcomp->szSource, &(szChannelOCGUID[1]), lstrlen(pcomp->szSource) - 3))
{
// Yes, so we need to hide the Channel Desktop Component.
// Return here before we generate it.
return;
}
}
// Write out the control HTML.
// First the control header
_WriteHtmlFromIdF(IDS_CONTROL_1, pcomp->dwID);
// then the size
_WriteHtmlFromIdF(IDS_CONTROL_2, pcomp->cpPos.dwHeight, _fSingleItem ? 0 : pcomp->cpPos.iLeft,
_fSingleItem ? 0 : pcomp->cpPos.iTop, pcomp->cpPos.dwWidth, pcomp->cpPos.izIndex);
// Write out whether this Control is resizeable or not!
_WriteResizeable(pcomp);
// Finally the rest of the control
_WriteHtmlFromIdF(IDS_CONTROL_3, pcomp->szSource);
EXITPROC(2, "DS GenerateHtmlControl!");
}
void CActiveDesktop::_GenerateHtmlComponent(COMPONENTA *pcomp)
{
ENTERPROC(2, "DS GenerateHtmlComponent(pcomp=%08X)");
switch (pcomp->iComponentType)
{
case COMP_TYPE_PICTURE:
_GenerateHtmlPicture(pcomp);
break;
case COMP_TYPE_HTMLDOC:
_GenerateHtmlDoc(pcomp);
break;
case COMP_TYPE_WEBSITE:
_GenerateHtmlSite(pcomp);
break;
case COMP_TYPE_CONTROL:
_GenerateHtmlControl(pcomp);
break;
}
EXITPROC(2, "DS GenerateHtmlComponent!");
}
void CActiveDesktop::_GenerateHtmlFooter(void)
{
ENTERPROC(2, "DS GenerateHtmlFooter()");
// Write out the deskmovr object.
if (!_fNoDeskMovr)
{
TCHAR szDeskMovrFile[MAX_PATH];
GetWindowsDirectory(szDeskMovrFile, ARRAYSIZE(szDeskMovrFile));
lstrcat(szDeskMovrFile, DESKMOVR_FILENAME);
_WriteHtmlFromFile(szDeskMovrFile);
}
// Write out the concluding HTML tags.
if (_pReadFileObjHtmlBkgd)
{
if (_fNeedBodyEnd)
{ // We had introduced the <BODY> tag by ourselves.
_WriteHtmlFromId(IDS_BODY_END2);
_fNeedBodyEnd = FALSE;
}
_WriteHtmlFromReadFileObj(_pReadFileObjHtmlBkgd, -1, -1);
delete _pReadFileObjHtmlBkgd; //Close the file and cleanup!
_pReadFileObjHtmlBkgd = NULL;
}
else
{
_WriteHtmlFromId(IDS_BODY_END);
}
EXITPROC(2, "DS GenerateHtmlFooter!");
}
void CActiveDesktop::_GenerateHtml(void)
{
ENTERPROC(2, "DS GenerateHtml()");
TCHAR szHtmlFile[MAX_PATH];
// Compute the filename.
szHtmlFile[0] = TEXT('\0');
GetPerUserFileName(szHtmlFile, ARRAYSIZE(szHtmlFile), DESKTOPHTML_FILENAME);
// Recreate the file.
_hFileHtml = CreateFile(szHtmlFile, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM, NULL);
if (_hFileHtml != INVALID_HANDLE_VALUE)
{
_GenerateHtmlHeader();
if (_co.fEnableComponents && _hdsaComponent && DSA_GetItemCount(_hdsaComponent) && !SHRestricted(REST_NODESKCOMP))
{
int i;
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA comp;
comp.dwSize = sizeof(COMPONENTA);
if ((DSA_GetItem(_hdsaComponent, i, &comp) != -1) && (comp.fChecked))
{
_GenerateHtmlComponent(&comp);
}
}
}
_GenerateHtmlFooter();
CloseHandle(_hFileHtml);
SetDesktopFlags(COMPONENTS_DIRTY, 0);
}
else
{
// 99/05/19 #340772 vtan: If unable to open desktop.htt it's probably
// in use by another process or task (perhaps trident is trying to
// render it). In this case mark it dirty so that it will get recreated
// - yet again but this time with more current data.
SetDesktopFlags(COMPONENTS_DIRTY, COMPONENTS_DIRTY);
}
EXITPROC(2, "DS GenerateHtml!");
}
HRESULT CActiveDesktop::GenerateDesktopItemHtml(LPCWSTR pwszFileName, COMPONENT *pcomp, DWORD dwReserved)
{
HRESULT hres = E_FAIL;
ENTERPROC(2, "DS GenerateComponentHtml(pcomp=%08X)", pcomp);
LPTSTR pszFileName;
//Check for the input parameters
if (!pwszFileName || (pcomp && (pcomp->dwSize != SIZEOF(*pcomp)) && (pcomp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
ASSERT(!dwReserved); // These should be 0
#ifndef UNICODE
CHAR szFileName[MAX_PATH];
SHUnicodeToAnsi(pwszFileName, szFileName, ARRAYSIZE(szFileName));
pszFileName = szFileName;
#else
pszFileName = (LPTSTR)pwszFileName;
#endif
// Create the file.
_hFileHtml = CreateFile(pszFileName, GENERIC_WRITE, 0, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL, NULL);
if (_hFileHtml != INVALID_HANDLE_VALUE)
{
_fNoDeskMovr = TRUE;
_fBackgroundHtml = TRUE;
//Check if we need to add a component
if (pcomp)
{
COMPONENTA CompA;
CompA.dwSize = sizeof(CompA);
WideCompToMultiComp(pcomp, &CompA);
_fSingleItem = TRUE;
_GenerateHtmlHeader();
_GenerateHtmlComponent(&CompA);
_GenerateHtmlFooter();
_fSingleItem = FALSE;
}
else
{
//generate just the header and the footer with proper
// wallpaper and pattern info!
_GenerateHtmlHeader();
_GenerateHtmlFooter();
}
_fBackgroundHtml = FALSE;
_fNoDeskMovr = FALSE;
CloseHandle(_hFileHtml);
hres = S_OK;
}
_hFileHtml = NULL;
EXITPROC(2, "DS GenerateComponentHtml=%d", hres);
return hres;
}
// AddUrl
HRESULT CActiveDesktop::AddUrl(HWND hwnd, LPCWSTR pszSourceW, LPCOMPONENT pcomp, DWORD dwFlags)
{
LPTSTR pszExt;
HRESULT fOkay = TRUE;
BOOL fExtIsCdf, fPathIsUrl;
BOOL fSubscribed = FALSE;
COMPONENT compLocal;
COMPONENTA compA;
TCHAR szSource[INTERNET_MAX_URL_LENGTH];
// 98/08/28 vtan #202777: The following if statement sanitizes parameters
// passed to AddUrl(). The statements following the "||" are executed
// despite the for pcomp against NULL. This causes an access violation
// and an exception to be thrown.
#if 0
//Check for the input parameters.
if (!pszSourceW || (pcomp &&
((pcomp->dwSize != SIZEOF(*pcomp)) && (pcomp->dwSize != SIZEOF(IE4COMPONENT))) ||
((pcomp->dwSize == SIZEOF(*pcomp)) && !VALIDATESTATE(pcomp->dwCurItemState))))
return E_INVALIDARG;
#else
// The following performs the same comparison but is spread into three
// separate comparisons. As performance is not a critical issue here
// but correctness is this makes the tests clear and understandable.
// The invalid conditions are described.
// Validate input parameters. Invalid parameters are:
// 1) NULL pszSourceW
// 2) pcomp->dwSize for a COMPONENT struct but invalid pcomp->dwCurItemState
// 3) pcomp->dwSize is not for a COMPONENT struct nor for a IE4COMPONENT struct
if (pszSourceW == NULL)
return(E_INVALIDARG);
if (pcomp != NULL)
{
if ((pcomp->dwSize == sizeof(*pcomp)) && !VALIDATESTATE(pcomp->dwCurItemState))
return(E_INVALIDARG);
if ((pcomp->dwSize != sizeof(*pcomp)) && (pcomp->dwSize != sizeof(IE4COMPONENT)))
return(E_INVALIDARG);
}
#endif
// Catch folks that call our API's to add components and prevent them from doing
// so if the restriction is in place.
if (SHIsRestricted(NULL, REST_NOADDDESKCOMP))
return E_ACCESSDENIED;
if (!pcomp)
{
pcomp = &compLocal;
pcomp->dwSize = sizeof(compLocal);
pcomp->dwCurItemState = IS_NORMAL;
}
// Attempt to come up with a reasonable window handle if none is passed in. ParseDesktopComponent
// will fail to attempt to create a subscription if a NULL window handle is passed in.
if (!hwnd)
hwnd = GetLastActivePopup(GetActiveWindow());
compA.dwSize = sizeof(compA);
compA.dwCurItemState = (pcomp->dwSize != SIZEOF(IE4COMPONENT)) ? pcomp->dwCurItemState : IS_NORMAL;
g_pActiveDesk = this;
SHUnicodeToTChar(pszSourceW, szSource, ARRAYSIZE(szSource));
pszExt = PathFindExtension(szSource);
fExtIsCdf = lstrcmpi(pszExt, TEXT(".CDF")) == 0;
fPathIsUrl = PathIsURL(szSource) && !UrlIsFileUrl(szSource);
if (FindComponent(szSource))
{
if (dwFlags & ADDURL_SILENT)
{
lstrcpy(compA.szSource, szSource);
MultiCompToWideComp(&compA, pcomp);
RemoveDesktopItem(pcomp, 0);
}
else
{
// This is a long string. So,...
TCHAR szMsg[512];
TCHAR szMsg2[256];
TCHAR szTitle[128];
LoadString(HINST_THISDLL, IDS_COMP_EXISTS, szMsg, ARRAYSIZE(szMsg));
LoadString(HINST_THISDLL, IDS_COMP_EXISTS_2, szMsg2, ARRAYSIZE(szMsg2));
lstrcat(szMsg, szMsg2);
LoadString(HINST_THISDLL, IDS_COMP_TITLE, szTitle, ARRAYSIZE(szTitle));
MessageBox(hwnd, szMsg, szTitle, MB_OK);
fOkay = FALSE;
}
}
if (fOkay && CheckForExistingSubscription(szSource))
{
if ((dwFlags & ADDURL_SILENT) ||
(ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_COMP_SUBSCRIBED),
MAKEINTRESOURCE(IDS_COMP_TITLE), MB_YESNO) == IDYES))
{
DeleteFromSubscriptionList(szSource);
}
else
{
fOkay = FALSE;
}
}
if (fOkay)
{
if (fPathIsUrl || fExtIsCdf)
{
WCHAR szUrlW[INTERNET_MAX_URL_LENGTH];
SHTCharToUnicode(szSource, szUrlW, ARRAYSIZE(szUrlW));
HRESULT hr;
IProgressDialog * pProgressDlg = NULL;
DECLAREWAITCURSOR;
// 98/12/16 vtan #250938: Cannot add new components that are not
// local with ICW run to completion. Tell the user and launch ICW.
if (!IsICWCompleted())
{
if ((dwFlags & ADDURL_SILENT) == 0)
{
ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_COMP_ICW_ADD), MAKEINTRESOURCE(IDS_COMP_ICW_TITLE), MB_OK);
LaunchICW();
}
fOkay = FALSE;
}
else
{
SetWaitCursor();
// ParseDesktopComponent can hang for a long time, we need some sort of progress
// UI up before we call it.
if (!(dwFlags & ADDURL_SILENT) && !fExtIsCdf)
{
if (pProgressDlg = CProgressDialog_CreateInstance(IDS_COMP_TITLE, IDA_ISEARCH, g_hinst))
{
WCHAR szConnectingW[80];
TCHAR szConnecting[80];
LoadString(HINST_THISDLL, IDS_CONNECTING, szConnecting, ARRAYSIZE(szConnecting));
SHTCharToUnicode(szConnecting, szConnectingW, ARRAYSIZE(szConnectingW));
pProgressDlg->SetLine(1, szConnectingW, FALSE, NULL);
pProgressDlg->SetLine(2, szUrlW, TRUE, NULL);
pProgressDlg->StartProgressDialog(hwnd, NULL, PROGDLG_AUTOTIME | PROGDLG_NOPROGRESSBAR, NULL);
}
}
hr = ParseDesktopComponent(hwnd, szUrlW, pcomp);
if (pProgressDlg)
{
pProgressDlg->StopProgressDialog();
fOkay = !pProgressDlg->HasUserCancelled(); // User may have cancelled the progress dialog
pProgressDlg->Release();
}
ResetWaitCursor();
if (hr == S_FALSE) // User cancelled operation via subscription download dialog
fOkay = FALSE;
if (fOkay)
{
if (SUCCEEDED(hr))
{
// Convert ed's wide thinggy to multi.
WideCompToMultiComp(pcomp, &compA);
fSubscribed = TRUE;
}
else if (!fExtIsCdf)
{
// This is some non-CDF url.
CreateComponent(&compA, szSource);
}
else
{
// We barfed on a CDF, bring up an error message.
if (!(dwFlags & ADDURL_SILENT))
{
ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_COMP_BADURL),
MAKEINTRESOURCE(IDS_COMP_TITLE), MB_OK);
}
fOkay = FALSE;
}
}
}
}
else
{
// This is just some local file.
CreateComponent(&compA, szSource);
}
}
if (fOkay && fPathIsUrl && !fSubscribed)
{
// Run subscription code on URLs if CDF code hasn't already.
if (dwFlags & ADDURL_SILENT)
{
ISubscriptionMgr *psm;
if (SUCCEEDED(CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&psm)))
{
WCHAR wzURL[INTERNET_MAX_URL_LENGTH];
//We need to zero init this structure except the cbSize field.
SUBSCRIPTIONINFO siDefault = { sizeof(SUBSCRIPTIONINFO) };
SHTCharToUnicode(szSource, wzURL, ARRAYSIZE(wzURL));
//This field is already initialized above.
//siDefault.cbSize = sizeof(siDefault);
psm->CreateSubscription(hwnd, wzURL, wzURL, CREATESUBS_NOUI, SUBSTYPE_DESKTOPURL, &siDefault);
psm->UpdateSubscription(wzURL);
psm->Release();
}
}
else
{
HRESULT hres = CreateSubscriptionsWizard(SUBSTYPE_DESKTOPURL, szSource, NULL, hwnd);
if (!SUCCEEDED(hres)) //Some error, or the user chose Cancel - we should fail.
{
ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_COMP_BADSUBSCRIBE), MAKEINTRESOURCE(IDS_COMP_TITLE), MB_OK);
}
fOkay = (hres == S_OK); //could be S_FALSE, which means CreateSubscription was cancelled
//so we don't display the above error, but we don't create the DTI
}
}
MultiCompToWideComp(&compA, pcomp);
if (fOkay)
{
AddDesktopItem(pcomp, 0);
return S_OK;
}
else
{
return E_FAIL;
}
}
void CActiveDesktop::_SaveSettings(DWORD dwFlags)
{
ENTERPROC(2, "DS SaveSettings()");
if (dwFlags & AD_APPLY_SAVE)
{
// Don't ever modify the safemode settings
TCHAR lpszDeskcomp[MAX_PATH];
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_GENERAL, _pszScheme);
if (!StrStr(lpszDeskcomp, REG_DESKCOMP_SAFEMODE_SUFFIX))
{
// Write out registry settings.
_SaveWallpaper();
_SaveComponents();
_SavePattern(SAVE_PATTERN_NAME);
}
};
if (dwFlags & AD_APPLY_HTMLGEN)
{
//We need to generate the Patten.bmp file too!
_SavePattern(GENERATE_PATTERN_FILE);
// Write out HTML file.
_GenerateHtml();
}
if (dwFlags & AD_APPLY_REFRESH)
{
HWND hwndShell = GetShellWindow();
SHELLSTATE ss = { 0 };
SHGetSetSettings(&ss, SSF_DESKTOPHTML, FALSE);
BOOL fWasActiveDesktop = BOOLIFY(ss.fDesktopHTML);
BOOL fIsActiveDesktop = BOOLIFY(_co.fActiveDesktop);
if (fIsActiveDesktop && !IsICWCompleted())
TBOOL(DisableUndisplayableComponents(this));
if (fIsActiveDesktop != fWasActiveDesktop)
{
SendMessage(hwndShell, WM_WININICHANGE, SPI_SETDESKWALLPAPER, (LPARAM)TEXT("ToggleDesktop"));
//Force a SHRefresh with this dummy call
SHGetSetSettings(NULL, 0, TRUE);
}
else if (fIsActiveDesktop)
{
//See if we can simply make the changes dynamically instead of refreshing the whole page
// 98/09/22 #182982 vtan: Use dynamic HTML to refresh only if specifically told by a flag.
if (_fUseDynamicHtml && (dwFlags & AD_APPLY_DYNAMICREFRESH))
{
SendMessage(hwndShell, DTM_MAKEHTMLCHANGES, (WPARAM)0, (LPARAM)0L);
}
else
{
//Can't use dynamic html. We have to refresh the whole page.
SendMessage(hwndShell, WM_WININICHANGE, SPI_SETDESKWALLPAPER,
(LPARAM)((dwFlags & AD_APPLY_BUFFERED_REFRESH) ? c_szBufferedRefresh : c_szRefreshDesktop));
}
}
_fUseDynamicHtml = TRUE;
}
// Data is no longer dirty.
_fDirty = FALSE;
_fWallpaperDirty = FALSE;
_fWallpaperChangedDuringInit = FALSE;
_fPatternDirty = FALSE;
EXITPROC(2, "DS SaveSettings!");
}
ULONG CActiveDesktop::AddRef(void)
{
ENTERPROC(1, "DS AddRef()");
_cRef++;
EXITPROC(1, "DS AddRef=%d", _cRef);
return _cRef;
}
HRESULT CActiveDesktop::ApplyChanges(DWORD dwFlags)
{
HRESULT hres = E_FAIL;
ENTERPROC(1, "DS Apply(dwFlags=%08X)", dwFlags);
if (dwFlags & AD_APPLY_FORCE)
{
_fDirty = TRUE;
_fWallpaperDirty = TRUE;
_fPatternDirty = TRUE;
}
if (_fDirty || _fWallpaperChangedDuringInit)
{
_SaveSettings(dwFlags);
}
hres = S_OK;
EXITPROC(1, "DS ApplyChanges=%d", hres);
return hres;
}
ULONG CActiveDesktop::Release(void)
{
UINT nRet = --_cRef;
ENTERPROC(1, "DS Release()");
if (_cRef == 0)
{
delete this;
}
EXITPROC(1, "DS Release=%d", nRet);
return nRet;
}
CActiveDesktop::CActiveDesktop()
{
_cRef = 1;
_fNoDeskMovr = FALSE;
_fBackgroundHtml = FALSE;
_fUseDynamicHtml = TRUE;
DllAddRef();
}
CActiveDesktop::~CActiveDesktop()
{
if (_hdsaComponent)
{
DSA_Destroy(_hdsaComponent);
}
if (_pszScheme)
{
LocalFree((HANDLE)_pszScheme);
}
DllRelease();
}
HRESULT CActiveDesktop::GetWallpaper(LPWSTR pwszWallpaper, UINT cchWallpaper, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
ENTERPROC(1, "DS GetWallpaper(pszWallpaper=%08X,cchWallpaper=%d)", pwszWallpaper, cchWallpaper);
ASSERT(!dwReserved); // These should be 0
if (pwszWallpaper && cchWallpaper)
{
SHTCharToUnicode(_szSelectedWallpaper, pwszWallpaper, cchWallpaper);
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS GetWallpaper unable to return wallpaper");
}
EXITPROC(1, "DS GetWallpaper=%d", hres);
return hres;
}
HRESULT CActiveDesktop::SetWallpaper(LPCWSTR pwszWallpaper, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
LPCTSTR pszWallpaper;
ASSERT(!dwReserved); // These should be 0
if (_fPolicyForWPName) // If a policy exists, the caller can not change the wallpaper.
return S_FALSE;
if (pwszWallpaper)
{
#ifdef UNICODE
pszWallpaper = pwszWallpaper;
#else
TCHAR szWallpaper[INTERNET_MAX_URL_LENGTH];
SHUnicodeToTChar(pwszWallpaper, szWallpaper, ARRAYSIZE(szWallpaper));
pszWallpaper = szWallpaper;
#endif
if (lstrcmp(_szSelectedWallpaper, pszWallpaper) != 0)
{
lstrcpyn(_szSelectedWallpaper, pszWallpaper, ARRAYSIZE(_szSelectedWallpaper));
_fWallpaperDirty = TRUE;
_fDirty = TRUE;
_fUseDynamicHtml = FALSE; //Setting wallpaper causes a lot of change; So, can't use dynamic html
}
hres = S_OK;
}
ENTERPROC(1, "DS SetWallpaper(pszWallpaper=>%s<)", pwszWallpaper ? pszWallpaper : TEXT("(NULL)"));
EXITPROC(1, "DS SetWallpaper=%d", hres);
return hres;
}
HRESULT CActiveDesktop::GetWallpaperOptions(WALLPAPEROPT *pwpo, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
ENTERPROC(1, "DS GetWallpaperOptions(pwpo=%08X)");
ASSERT(!dwReserved); // These should be 0
if ((pwpo) && (pwpo->dwSize == SIZEOF(*pwpo)))
{
*pwpo = _wpo;
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS GetWallpaperOptions could not return options");
}
EXITPROC(1, "DS GetWallpaperOptions=%d", hres);
return hres;
}
HRESULT CActiveDesktop::SetWallpaperOptions(LPCWALLPAPEROPT pwpo, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
ENTERPROC(1, "DS SetWallpaperOptions(pwpo=%08X)", pwpo);
ASSERT(!dwReserved); // These should be 0
if (_fPolicyForWPStyle) //If a policy exists for wallpaper style, the caller can not change it.
return S_FALSE;
if ((pwpo) && (pwpo->dwSize == SIZEOF(*pwpo)))
{
_wpo = *pwpo;
_fWallpaperDirty = TRUE;
_fDirty = TRUE;
_fUseDynamicHtml = FALSE; //Changing wallpaper options causes us to regenerate the whole thing.
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS SetWallpaperOptions could not set options");
}
EXITPROC(1, "DS SetWallpaperOptions=%d", hres);
return hres;
}
HRESULT CActiveDesktop::GetPattern(LPWSTR pwszPattern, UINT cchPattern, DWORD dwReserved)
{
HRESULT hres = S_OK;
ENTERPROC(1, "DS GetPattern(psz=%08X,cch=%d)", pwszPattern, cchPattern);
ASSERT(!dwReserved); // These should be 0
if (!pwszPattern || (cchPattern == 0))
return (E_INVALIDARG);
SHTCharToUnicode(_szSelectedPattern, pwszPattern, cchPattern);
EXITPROC(1, "DS GetPattern=%d", hres);
return hres;
}
HRESULT CActiveDesktop::SetPattern(LPCWSTR pwszPattern, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
LPCTSTR pszPattern;
ASSERT(!dwReserved); // These should be 0
if (pwszPattern)
{
#ifndef UNICODE
TCHAR szPattern[INTERNET_MAX_URL_LENGTH];
SHUnicodeToAnsi(pwszPattern, szPattern, ARRAYSIZE(szPattern));
pszPattern = szPattern;
#else
pszPattern = pwszPattern;
#endif
if (lstrcmp(_szSelectedPattern, pszPattern) != 0)
{
lstrcpyn(_szSelectedPattern, pszPattern, ARRAYSIZE(_szSelectedPattern));
_fPatternDirty = TRUE;
_fDirty = TRUE;
_fUseDynamicHtml = FALSE; //Setting pattern causes us to regenerate the whole thing.
hres = S_OK;
}
else
hres = E_FAIL;
}
ENTERPROC(1, "DS SetPattern(psz=>%s<)", pwszPattern ? pszPattern : TEXT("(NULL)"));
EXITPROC(1, "DS SetPattern=%d", hres);
return hres;
}
HRESULT CActiveDesktop::GetDesktopItemOptions(COMPONENTSOPT *pco, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
ENTERPROC(1, "DS GetComponentsOptions(pco=%08X)", pco);
ASSERT(!dwReserved); // These should be 0
if (pco && (pco->dwSize == SIZEOF(*pco)))
{
*pco = _co;
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS GetComponentsOptions unable to return options");
}
EXITPROC(1, "DS GetComponentsOptions=%d", hres);
return hres;
}
HRESULT CActiveDesktop::SetDesktopItemOptions(LPCCOMPONENTSOPT pco, DWORD dwReserved)
{
HRESULT hres = E_INVALIDARG;
ENTERPROC(1, "DS SetComponentsOptions(pco=%08X)", pco);
ASSERT(!dwReserved); // These should be 0
if (pco && (pco->dwSize == SIZEOF(*pco)))
{
_co = *pco;
_fDirty = TRUE;
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS SetComponentsOptions unable to set options");
}
EXITPROC(1, "DS SetComponentsOptions=%d", hres);
return hres;
}
// SetStateInfo()
// This function simply sets up the COMPSTATEINFO structure passed using the current
// position and size from the COMPPOS structure and the itemState passed.
void SetStateInfo(COMPSTATEINFO *pCompStateInfo, COMPPOS *pCompPos, DWORD dwItemState)
{
pCompStateInfo->dwSize = sizeof(*pCompStateInfo);
pCompStateInfo->iLeft = pCompPos->iLeft;
pCompStateInfo->iTop = pCompPos->iTop;
pCompStateInfo->dwWidth = pCompPos->dwWidth;
pCompStateInfo->dwHeight = pCompPos->dwHeight;
pCompStateInfo->dwItemState = dwItemState;
}
void ConvertCompStruct(COMPONENTA *pCompDest, COMPONENTA *pCompSrc, BOOL fPubToPriv)
{
pCompDest->dwID = pCompSrc->dwID;
pCompDest->iComponentType = pCompSrc->iComponentType;
pCompDest->fChecked = pCompSrc->fChecked;
pCompDest->fDirty = pCompSrc->fDirty;
pCompDest->fNoScroll = pCompSrc->fNoScroll;
pCompDest->cpPos = pCompSrc->cpPos;
if (fPubToPriv)
{
COMPONENT *pComp = (COMPONENT *)pCompSrc;
pCompDest->dwSize = sizeof(COMPONENTA);
SHUnicodeToTChar(pComp->wszSource, pCompDest->szSource, ARRAYSIZE(pCompDest->szSource));
SHUnicodeToTChar(pComp->wszFriendlyName, pCompDest->szFriendlyName, ARRAYSIZE(pCompDest->szFriendlyName));
SHUnicodeToTChar(pComp->wszSubscribedURL, pCompDest->szSubscribedURL, ARRAYSIZE(pCompDest->szSubscribedURL));
//Check to see if the public component is from IE4 app (old size)
if (pCompSrc->dwSize == sizeof(COMPONENT))
{
// Since the dest component is the same size as the most current structure, all fields
// are valid.
// CAUTION: The following fields are at a different offset in public and private
// structures. So, you need to use pcomp instead of pCompSrc for example.
pCompDest->dwCurItemState = pComp->dwCurItemState;
pCompDest->csiOriginal = pComp->csiOriginal;
pCompDest->csiRestored = pComp->csiRestored;
}
else
{
// Since the size did not match, we assume that this is an older structure.
// Since the older struct does not have any Original and Restored sizes, let's copy
// the default values.
IE4COMPONENT *pIE4Comp = (IE4COMPONENT *)pCompSrc;
pCompDest->dwCurItemState = IS_NORMAL;
SetStateInfo(&pCompDest->csiOriginal, &pIE4Comp->cpPos, IS_NORMAL);
SetStateInfo(&pCompDest->csiRestored, &pIE4Comp->cpPos, IS_NORMAL);
}
}
else
{
COMPONENT *pComp = (COMPONENT *)pCompDest;
if (pCompDest->dwSize != sizeof(COMPONENT))
pCompDest->dwSize = sizeof(IE4COMPONENT);
SHTCharToUnicode(pCompSrc->szSource, pComp->wszSource, ARRAYSIZE(pComp->wszSource));
SHTCharToUnicode(pCompSrc->szFriendlyName, pComp->wszFriendlyName, ARRAYSIZE(pComp->wszFriendlyName));
SHTCharToUnicode(pCompSrc->szSubscribedURL, pComp->wszSubscribedURL, ARRAYSIZE(pComp->wszSubscribedURL));
//Check to see if the public component is from IE4 app (old size)
if (pComp->dwSize == sizeof(COMPONENT))
{
// Since the dest component is the same size as the most current structure, all fields
// are valid.
// CAUTION: The following fields are at a different offset in public and private
// structures. So, you need to use pcomp instead of pCompDest for example.
pComp->dwCurItemState = pCompSrc->dwCurItemState;
pComp->csiOriginal = pCompSrc->csiOriginal;
pComp->csiRestored = pCompSrc->csiRestored;
}
// else, the dest component is IE4COMPONENT and the additional fields are not there.
}
}
HRESULT CActiveDesktop::_AddDTIWithUIPrivateA(HWND hwnd, LPCCOMPONENT pComp, DWORD dwFlags)
{
HRESULT hres = E_FAIL;
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
int nScheme;
DWORD dwCurItemState;
SHUnicodeToTChar(pComp->wszSource, szUrl, ARRAYSIZE(szUrl));
nScheme = GetUrlScheme(szUrl);
if ((URL_SCHEME_INVALID == nScheme) || (URL_SCHEME_UNKNOWN == nScheme))
{
TCHAR szFullyQualified[INTERNET_MAX_URL_LENGTH];
DWORD cchSize = ARRAYSIZE(szFullyQualified);
if (SUCCEEDED(ParseURLFromOutsideSource(szUrl, szFullyQualified, &cchSize, NULL)))
nScheme = GetUrlScheme(szFullyQualified);
}
// Is this URL valid to subscribe to? Did the caller specify they want use
// to try to subscribe to it?
if ((URL_SCHEME_FILE != nScheme) && (URL_SCHEME_ABOUT != nScheme) &&
IsFlagSet(dwFlags, DTI_ADDUI_DISPSUBWIZARD) && hwnd)
{
//Create a subscription.
hres = CreateSubscriptionsWizard(SUBSTYPE_DESKTOPURL, szUrl, NULL, hwnd);
if (hres != S_OK)
{
return hres;
}
}
// Add the component to the registry.
// BUGBUG: This sucks because this function creates a second COM objects.
// We need to Inline the functionality.
if (pComp->dwSize == SIZEOF(IE4COMPONENT))
dwCurItemState = IS_NORMAL;
else
dwCurItemState = pComp->dwCurItemState;
hres = AddDesktopComponentNoUI(AD_APPLY_ALL | AD_APPLY_DYNAMICREFRESH, szUrl, NULL, pComp->iComponentType,
pComp->cpPos.iLeft, pComp->cpPos.iTop, pComp->cpPos.dwWidth, pComp->cpPos.dwHeight, TRUE, dwCurItemState) ? S_OK : E_FAIL;
return hres;
}
#define STC_DESKTOPCOMPONENT 0x00000002
EXTERN_C STDAPI_(HRESULT) SubscribeToCDF(HWND hwndParent, LPCWSTR pwzUrl, DWORD dwCDFTypes);
HRESULT CActiveDesktop::AddDesktopItemWithUI(HWND hwnd, LPCOMPONENT pComp, DWORD dwFlags)
{
HRESULT hres = E_FAIL;
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pComp ||
((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))) ||
((pComp->dwSize == SIZEOF(*pComp)) && !VALIDATESTATE(pComp->dwCurItemState)) ||
((pComp->iComponentType < 0) || (pComp->iComponentType > COMP_TYPE_MAX))) //Validate the component type
return E_INVALIDARG;
// Catch folks that call our API's to add components and prevent them from doing
// so if the restriction is in place.
if (SHIsRestricted(NULL, REST_NOADDDESKCOMP))
return E_ACCESSDENIED;
// Check if the component already exists.
BOOL fCompExists = FALSE;
int cComp;
GetDesktopItemCount(&cComp, 0);
int i;
COMPONENT comp;
comp.dwSize = sizeof(COMPONENT); //This needs to be initialized for ConvertCompStruc to work!
COMPONENTA compA;
TCHAR szSource[INTERNET_MAX_URL_LENGTH];
SHUnicodeToTChar(pComp->wszSource, szSource, ARRAYSIZE(szSource));
for (i = 0; i < cComp && !fCompExists; i++)
{
compA.dwSize = SIZEOF(compA);
if (GetComponentPrivate(i, &compA)
&& lstrcmpi(szSource, compA.szSource) == 0)
{
fCompExists = TRUE;
ConvertCompStruct((COMPONENTA *)&comp, &compA, FALSE);
break;
}
}
BOOL fAskToInstall;
if (ZoneCheckUrlW(pComp->wszSource, URLACTION_SHELL_INSTALL_DTITEMS, (PUAF_NOUI), NULL) == S_OK)
{
fAskToInstall = TRUE;
}
else
{
fAskToInstall = FALSE;
}
if (S_OK != ZoneCheckUrlW(pComp->wszSource, URLACTION_SHELL_INSTALL_DTITEMS, (hwnd ? (PUAF_FORCEUI_FOREGROUND | PUAF_WARN_IF_DENIED) : PUAF_NOUI), NULL))
return E_ACCESSDENIED;
BOOL fCompSubDeleted = FALSE;
SUBSCRIPTIONINFO si = { sizeof(SUBSCRIPTIONINFO) };
// si.bstrUserName = NULL;
// si.bstrPassword = NULL;
// si.bstrFriendlyName = NULL;
// Confirmation dialog.
if (hwnd)
{
if (fCompExists)
{
//Prompt the user to delete the existing ADI.
// This is a long string. So,...
TCHAR szMsg[512];
TCHAR szMsg2[256];
TCHAR szTitle[128];
LoadString(HINST_THISDLL, IDS_COMP_EXISTS, szMsg, ARRAYSIZE(szMsg));
LoadString(HINST_THISDLL, IDS_COMP_EXISTS_2, szMsg2, ARRAYSIZE(szMsg2));
lstrcat(szMsg, szMsg2);
LoadString(HINST_THISDLL, IDS_COMP_TITLE, szTitle, ARRAYSIZE(szTitle));
MessageBox(hwnd, szMsg, szTitle, MB_OK);
return E_FAIL;
#if 0
comp.dwSize = SIZEOF(comp);
//Prompt the user to reinstall the ADI.
if (ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_CONFIRM_ADI_REINSTALL),
MAKEINTRESOURCE(IDS_COMP_TITLE), MB_YESNO) != IDYES)
{
return E_FAIL; //User choses not to install this desktop component!
}
else
{
ISubscriptionMgr *psm;
if (SUCCEEDED(hres = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&psm)))
{
si.cbSize = sizeof(si);
si.fUpdateFlags = SUBSINFO_ALLFLAGS;
//Backup and remove the subscription also
hres = psm->GetSubscriptionInfo(comp.wszSubscribedURL, &si);
if (SUCCEEDED(hres))
{
hres = RemoveDesktopItem(&comp, 0);
if (SUCCEEDED(hres))
{
psm->DeleteSubscription(comp.wszSubscribedURL, NULL);
ApplyChanges(AD_APPLY_SAVE);
fCompSubDeleted = TRUE;
// Set the new component to be enabled
pComp->fChecked = TRUE;
}
}
psm->Release();
}
else
{
TraceMsg(TF_WARNING, "CActiveDesktop::AddDesktopItemWithUI : CoCreateInstance for CLSID_SubscriptionMgr failed.");
}
}
#endif
}
else if (fAskToInstall)
{
if (ShellMessageBox(HINST_THISDLL, hwnd, MAKEINTRESOURCE(IDS_CONFIRM_ADD),
MAKEINTRESOURCE(IDS_INTERNET_EXPLORER), MB_YESNO) != IDYES)
{
return E_FAIL; //User choses not to install this desktop component!
}
}
}
hres = SubscribeToCDF(hwnd, pComp->wszSubscribedURL, STC_DESKTOPCOMPONENT);
switch (hres)
{
case E_INVALIDARG:
{
// E_UNEXPECTED is returned from SubscribeToCDFUrlA() when the URL doesn't point to
// a CDF file, so we assume it's a web page.
hres = _AddDTIWithUIPrivateA(hwnd, pComp, dwFlags);
if (hres != S_OK && fCompSubDeleted) // Restore the old component
{
hres = AddDesktopItem(&comp, 0);
if (SUCCEEDED(hres))
{
ISubscriptionMgr *psm;
if (SUCCEEDED(CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&psm)))
{
si.cbSize = sizeof(si);
psm->CreateSubscription(hwnd, comp.wszSubscribedURL, si.bstrFriendlyName, CREATESUBS_NOUI, SUBSTYPE_DESKTOPURL, &si);
psm->Release();
}
else
{
TraceMsg(TF_WARNING, "CActiveDesktop::AddDesktopItemWithUI : CoCreateInstance for CLSID_SubscriptionMgr failed.");
}
}
}
ApplyChanges(AD_APPLY_ALL | AD_APPLY_DYNAMICREFRESH);
}
break;
case E_ACCESSDENIED:
// The file was a CDF but didn't contain Desktop Component Information
if (hwnd)
{
TCHAR szMsg[MAX_PATH];
TCHAR szTitle[MAX_PATH];
LoadString(HINST_THISDLL, IDS_ADDCOMP_ERROR_CDFNODTI, szMsg, ARRAYSIZE(szMsg));
LoadString(HINST_THISDLL, IDS_INTERNET_EXPLORER, szTitle, ARRAYSIZE(szTitle));
MessageBox(hwnd, szMsg, szTitle, MB_OK);
}
break;
case E_UNEXPECTED:
// This was a CDF but it was misauthored.
if (hwnd)
{
TCHAR szMsg[MAX_PATH];
TCHAR szTitle[MAX_PATH];
LoadString(HINST_THISDLL, IDS_ADDCOMP_ERROR_CDFINALID, szMsg, ARRAYSIZE(szMsg));
LoadString(HINST_THISDLL, IDS_INTERNET_EXPLORER, szTitle, ARRAYSIZE(szTitle));
MessageBox(hwnd, szMsg, szTitle, MB_OK);
}
break;
default:
break;
}
if (hwnd && SUCCEEDED(hres))
{
// If the active desktop is currently OFF, we need to turn it ON
SHELLSTATE ss = { 0 };
SHGetSetSettings(&ss, SSF_DESKTOPHTML, FALSE);
if (!ss.fDesktopHTML)
{
COMPONENTSOPT co;
co.dwSize = sizeof(COMPONENTSOPT);
GetDesktopItemOptions(&co, 0);
co.fActiveDesktop = TRUE;
SetDesktopItemOptions(&co, 0);
ApplyChanges(AD_APPLY_REFRESH | AD_APPLY_DYNAMICREFRESH);
}
}
if (fCompSubDeleted)
{
if (si.bstrUserName)
{
SysFreeString(si.bstrUserName);
}
if (si.bstrPassword)
{
SysFreeString(si.bstrPassword);
}
if (si.bstrFriendlyName)
{
SysFreeString(si.bstrFriendlyName);
}
}
return hres;
}
void RestoreComponent(HDSA hdsaComponents, COMPONENTA * pcomp)
{
int i;
// If we are split then set the bit saying that the listview needs to be adjusted. This is done
// when we check the state of desktop.htm in EnsureUpdateHtml.
if (pcomp->dwCurItemState & IS_SPLIT)
{
pcomp->dwCurItemState |= IS_ADJUSTLISTVIEW;
SetDesktopFlags(COMPONENTS_ZOOMDIRTY, COMPONENTS_ZOOMDIRTY);
}
for (i = 0; i < DSA_GetItemCount(hdsaComponents); i++)
{
COMPONENTA * pcompT;
if (pcompT = (COMPONENTA *)DSA_GetItemPtr(hdsaComponents, i))
{
// If this component is split/fullscreen and is different from the source component
// but it is at the same location then it must be on this monitor (work area) so restore it.
if (ISZOOMED(pcompT) &&
lstrcmpi(pcomp->szSource, pcompT->szSource) &&
(pcomp->cpPos.iTop == pcompT->cpPos.iTop) &&
((pcomp->cpPos.iLeft + pcomp->cpPos.dwWidth) == (pcompT->cpPos.iLeft + pcompT->cpPos.dwWidth)))
{
pcompT->dwCurItemState = pcompT->csiRestored.dwItemState;
pcompT->cpPos.iLeft = pcompT->csiRestored.iLeft;
pcompT->cpPos.iTop = pcompT->csiRestored.iTop;
pcompT->cpPos.dwWidth = pcompT->csiRestored.dwWidth;
pcompT->cpPos.dwHeight = pcompT->csiRestored.dwHeight;
pcompT->cpPos.izIndex = COMPONENT_TOP;
pcompT->fDirty = TRUE;
break;
}
}
}
}
HRESULT CActiveDesktop::AddDesktopItem(LPCCOMPONENT pComp, DWORD dwReserved)
{
HRESULT hres = E_FAIL;
COMPONENTA CompA;
CompA.dwSize = SIZEOF(CompA);
ASSERT(!dwReserved); // These should be 0
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pComp ||
((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))) ||
((pComp->dwSize == SIZEOF(*pComp)) && !VALIDATESTATE(pComp->dwCurItemState)))
return E_INVALIDARG;
// Catch folks that call our API's to add components and prevent them from doing
// so if the restriction is in place.
if (SHIsRestricted(NULL, REST_NOADDDESKCOMP))
return E_ACCESSDENIED;
// Convert the external structure to the internal format
ConvertCompStruct(&CompA, (COMPONENTA *)pComp, TRUE);
// If the component is already present, then fail the call!
if (_FindComponentBySource(CompA.szSource, &CompA) > -1)
return hres;
//Make sure that the COMPPOS size field is set before we add it!
CompA.cpPos.dwSize = sizeof(COMPPOS);
PositionComponent(&CompA, &CompA.cpPos, CompA.iComponentType, TRUE);
if (_hdsaComponent && ISZOOMED(&CompA))
RestoreComponent(_hdsaComponent, &CompA);
//Make sure the this component's fDirty flag is off.
CompA.fDirty = FALSE;
// Set the dummy bit here - this forces folks to do bitwise testing on the dwCurItemState field
// instead of testing for equality. This will allow us to expand use of the field down the
// road without compatibility problems.
CompA.dwCurItemState |= IS_INTERNALDUMMYBIT;
if (AddComponentPrivate(&CompA, _dwNextID++))
{
// It might be cheaper to attempt to insert the component in the
// correct z-order but it's less code to just call _SortAndRationalize
// after the insertion is done.
_SortAndRationalize();
hres = S_OK;
}
return(hres);
}
BOOL CActiveDesktop::AddComponentPrivate(COMPONENTA *pcomp, DWORD_PTR dwID)
{
BOOL fRet = FALSE;
ENTERPROC(1, "DS AddComponent(pcomp=%08X)", pcomp);
if (pcomp)
{
if (_hdsaComponent == NULL)
{
_hdsaComponent = DSA_Create(SIZEOF(COMPONENTA), DXA_GROWTH_CONST);
}
if (_hdsaComponent)
{
pcomp->dwID = dwID;
if (DSA_AppendItem(_hdsaComponent, pcomp) != -1)
{
_fDirty = TRUE;
fRet = TRUE;
}
else
{
TraceMsg(TF_WARNING, "DS AddComponent unable to append DSA");
}
}
else
{
TraceMsg(TF_WARNING, "DS AddComponent unable to create DSA");
}
}
else
{
TraceMsg(TF_WARNING, "DS AddComponent unable to add a component");
}
EXITPROC(1, "DS AddComponent=%d", fRet);
return fRet;
}
// This finds out if a given component already exists by comparing the szSource
// If so, it fills out the correct dwID and returns the index.
int CActiveDesktop::_FindComponentBySource(LPTSTR lpszSource, COMPONENTA *pComp)
{
int iRet = -1;
ENTERPROC(2, "DS FindComponentIdBySource(pComp=%8X)", pComp);
if (_hdsaComponent)
{
int i;
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA comp;
comp.dwSize = sizeof(COMPONENTA);
if (DSA_GetItem(_hdsaComponent, i, &comp) != -1)
{
if (!lstrcmpi(comp.szSource, lpszSource))
{
*pComp = comp;
iRet = i;
break;
}
}
else
{
TraceMsg(TF_WARNING, "DS FindComponentIndexByID unable to get a component");
}
}
}
EXITPROC(2, "DS FindComponentIdBySource=%d", iRet);
return iRet;
}
int CActiveDesktop::_FindComponentIndexByID(DWORD_PTR dwID)
{
int iRet = -1;
ENTERPROC(2, "DS FindComponentIndexByID(dwID=%d)", dwID);
if (_hdsaComponent)
{
int i;
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA comp;
comp.dwSize = sizeof(COMPONENTA);
if (DSA_GetItem(_hdsaComponent, i, &comp) != -1)
{
if (comp.dwID == dwID)
{
iRet = i;
break;
}
}
else
{
TraceMsg(TF_WARNING, "DS FindComponentIndexByID unable to get a component");
}
}
}
EXITPROC(2, "DS FindComponent=%d", iRet);
return iRet;
}
// This function is to be used only in special situations. Given a Url, it finds a component
// that has the src= pointed to that url. Note that what we have is szSource is something like
// "c:\foo\bar.bmp"; But, what is passed to this function is "file://c:/foo/bar.htm"
// Warning: This function does a conversion from Path to Url for every component before
// comparing with the given Url. This is in-efficient. We do it this way because converting
// the given Url ,which was converted to url from path, back to Path may not result in the
// original path. In other words a round-trip from path to Url and back to path may not result
// in the path that was originally entered by the end-user.
int CActiveDesktop::_FindComponentBySrcUrl(LPTSTR lpszSrcUrl, COMPONENTA *pComp)
{
int iRet = -1;
ENTERPROC(2, "DS FindComponentBySrcUrl(pComp=%8X)", pComp);
if (_hdsaComponent)
{
int i;
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA comp;
comp.dwSize = sizeof(COMPONENTA);
if (DSA_GetItem(_hdsaComponent, i, &comp) != -1)
{
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
LPTSTR lpszUrl = szUrl;
DWORD dwSize;
//Convert the szSource to Url
dwSize = ARRAYSIZE(szUrl);
if (FAILED(UrlCreateFromPath(comp.szSource, lpszUrl, &dwSize, 0)))
lpszUrl = comp.szSource;
if (!lstrcmpi(lpszUrl, lpszSrcUrl))
{
*pComp = comp;
iRet = i;
break;
}
}
else
{
TraceMsg(TF_WARNING, "DS FindComponentBySrcUrl unable to get a component");
}
}
}
EXITPROC(2, "DS FindComponentBySrcUrl=%d", iRet);
return iRet;
}
HRESULT CActiveDesktop::GetDesktopItemByID(ULONG_PTR dwID, COMPONENT *pcomp, DWORD dwReserved)
{
HRESULT hres = E_FAIL;
ENTERPROC(1, "DS GetComponentByID(dwID=%d,pcomp=%08X)", dwID, pcomp);
COMPONENTA CompA;
ASSERT(!dwReserved); // These should be 0
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pcomp || ((pcomp->dwSize != SIZEOF(*pcomp)) && (pcomp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
if (pcomp)
{
int index = _FindComponentIndexByID(dwID);
if (index != -1)
{
if (DSA_GetItem(_hdsaComponent, index, &CompA) != -1)
{
hres = S_OK;
}
else
{
TraceMsg(TF_WARNING, "DS GetComponentByID unable to get component");
}
}
else
{
TraceMsg(TF_WARNING, "DS GetComponentByID unable to find component");
}
}
else
{
TraceMsg(TF_WARNING, "DS GetComponentByID given NULL pcomp");
}
if (SUCCEEDED(hres))
{
MultiCompToWideComp(&CompA, pcomp);
}
EXITPROC(1, "DS GetComponentByID=%d", hres);
return hres;
}
HRESULT CActiveDesktop::RemoveDesktopItem(LPCCOMPONENT pComp, DWORD dwReserved)
{
COMPONENTA CompA, CompToDelete;
int iIndex;
HRESULT hres = E_FAIL;
ASSERT(!dwReserved); // These should be 0
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pComp || ((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
CompA.dwSize = SIZEOF(CompA);
CompToDelete.dwSize = SIZEOF(CompToDelete);
//Convert the struct to internal struct.
ConvertCompStruct(&CompA, (COMPONENTA *)pComp, TRUE);
// See if the component already exists.
iIndex = _FindComponentBySource(CompA.szSource, &CompToDelete);
if (iIndex > -1)
{
if (RemoveComponentPrivate(iIndex, &CompToDelete))
{
hres = S_OK;
}
}
return(hres);
}
BOOL CActiveDesktop::RemoveComponentPrivate(int iIndex, COMPONENTA *pcomp)
{
BOOL fRet = FALSE;
ENTERPROC(1, "DS RemoveComponent(pcomp=%08X)", pcomp);
if (_hdsaComponent)
{
if (iIndex == -1)
iIndex = _FindComponentIndexByID(pcomp->dwID);
if (iIndex != -1)
{
if (DSA_DeleteItem(_hdsaComponent, iIndex) != -1)
{
_fDirty = TRUE;
fRet = TRUE;
}
else
{
TraceMsg(TF_WARNING, "DS RemoveComponent could not remove an item");
}
}
else
{
TraceMsg(TF_WARNING, "DS RemoveComponent could not find item to remove");
}
}
else
{
TraceMsg(TF_WARNING, "DS RemoveComponent has no components to remove");
}
EXITPROC(1, "DS RemoveComponent=%d", fRet);
return fRet;
}
HRESULT CActiveDesktop::_CopyComponent(COMPONENTA *pCompDest, COMPONENTA *pCompSrc, DWORD dwFlags)
{
//Copy only those elements mentioned in the flag!
// if(dwFlags & COMP_ELEM_ID)
// pCompDest->dwID = pCompSrc->dwID;
if (dwFlags & COMP_ELEM_TYPE)
pCompDest->iComponentType = pCompSrc->iComponentType;
if (dwFlags & COMP_ELEM_CHECKED)
pCompDest->fChecked = pCompSrc->fChecked;
if (dwFlags & COMP_ELEM_DIRTY)
pCompDest->fDirty = pCompSrc->fDirty;
if (dwFlags & COMP_ELEM_NOSCROLL)
pCompDest->fNoScroll = pCompSrc->fNoScroll;
if (dwFlags & COMP_ELEM_POS_LEFT)
pCompDest->cpPos.iLeft = pCompSrc->cpPos.iLeft;
if (dwFlags & COMP_ELEM_POS_TOP)
pCompDest->cpPos.iTop = pCompSrc->cpPos.iTop;
if (dwFlags & COMP_ELEM_SIZE_WIDTH)
pCompDest->cpPos.dwWidth = pCompSrc->cpPos.dwWidth;
if (dwFlags & COMP_ELEM_SIZE_HEIGHT)
pCompDest->cpPos.dwHeight = pCompSrc->cpPos.dwHeight;
if (dwFlags & COMP_ELEM_POS_ZINDEX)
pCompDest->cpPos.izIndex = pCompSrc->cpPos.izIndex;
if (dwFlags & COMP_ELEM_SOURCE)
lstrcpy(pCompDest->szSource, pCompSrc->szSource);
if (dwFlags & COMP_ELEM_FRIENDLYNAME)
lstrcpy(pCompDest->szFriendlyName, pCompSrc->szFriendlyName);
if (dwFlags & COMP_ELEM_SUBSCRIBEDURL)
lstrcpy(pCompDest->szSubscribedURL, pCompSrc->szSubscribedURL);
if (dwFlags & COMP_ELEM_ORIGINAL_CSI)
pCompDest->csiOriginal = pCompSrc->csiOriginal;
if (dwFlags & COMP_ELEM_RESTORED_CSI)
{
pCompDest->csiRestored = pCompSrc->csiRestored;
// 98/08/21 vtan #174542: When changing csiRestored using the Active
// Desktop API and the component is zoomed the csiRestored information
// needs to be copied to the cpPos field as well as this is where the
// actual information is stored when the component is restored. This
// is only applicable to the case when the component is zoomed.
if (ISZOOMED(pCompDest))
{
pCompDest->cpPos.iLeft = pCompSrc->csiRestored.iLeft;
pCompDest->cpPos.iTop = pCompSrc->csiRestored.iTop;
pCompDest->cpPos.dwWidth = pCompSrc->csiRestored.dwWidth;
pCompDest->cpPos.dwHeight = pCompSrc->csiRestored.dwHeight;
}
}
if (dwFlags & COMP_ELEM_CURITEMSTATE) // Only allow the modification of the public bits - propagate the internal bits unchanged.
pCompDest->dwCurItemState = (pCompDest->dwCurItemState & IS_VALIDINTERNALBITS) | (pCompSrc->dwCurItemState & ~IS_VALIDINTERNALBITS);
return(S_OK);
}
HRESULT CActiveDesktop::GetDesktopItemBySource(LPCWSTR lpcwszSource, LPCOMPONENT pComp, DWORD dwFlags)
{
COMPONENTA CompNew;
HRESULT hres = E_FAIL;
int iIndex;
//Passing a NULL to SHUnicodeToTChar causes a fault. So, let's fail it.
if (lpcwszSource == NULL)
return E_INVALIDARG;
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pComp || ((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
CompNew.dwSize = sizeof(COMPONENTA);
SHUnicodeToTChar(lpcwszSource, CompNew.szSource, ARRAYSIZE(CompNew.szSource));
iIndex = _FindComponentBySource(CompNew.szSource, &CompNew);
if (iIndex > -1)
{
MultiCompToWideComp(&CompNew, pComp);
hres = S_OK;
}
return(hres);
}
HRESULT CActiveDesktop::ModifyDesktopItem(LPCCOMPONENT pComp, DWORD dwFlags)
{
COMPONENTA CompA, CompNew;
HRESULT hres = E_FAIL;
int iIndex = -1;
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if (!pComp || ((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
CompA.dwSize = sizeof(COMPONENTA);
CompNew.dwSize = sizeof(COMPONENTA);
//Convert public param structure to private param structure.
ConvertCompStruct(&CompA, (COMPONENTA *)pComp, TRUE);
//See if this component already exists.
iIndex = _FindComponentBySource(CompA.szSource, &CompNew);
if (iIndex > -1)
{
_CopyComponent(&CompNew, &CompA, dwFlags);
if (dwFlags & (COMP_ELEM_POS_LEFT | COMP_ELEM_POS_TOP | COMP_ELEM_SIZE_WIDTH | COMP_ELEM_SIZE_HEIGHT | COMP_ELEM_CHECKED | COMP_ELEM_CURITEMSTATE))
PositionComponent(&CompNew, &CompNew.cpPos, CompNew.iComponentType, FALSE);
if (ISZOOMED(&CompNew))
RestoreComponent(_hdsaComponent, &CompNew);
CompNew.fDirty = TRUE; //Since the component is modified, we set the dirty bit!
if (UpdateComponentPrivate(iIndex, &CompNew))
hres = S_OK;
}
return(hres);
}
BOOL CActiveDesktop::UpdateComponentPrivate(int iIndex, COMPONENTA *pcomp)
{
BOOL fRet = FALSE;
ENTERPROC(1, "DS UpdateComponentPrivate(pcomp=%08X)", pcomp);
if (_hdsaComponent)
{
if (iIndex == -1)
iIndex = _FindComponentIndexByID(pcomp->dwID);
if (iIndex != -1)
{
if (DSA_SetItem(_hdsaComponent, iIndex, pcomp) != -1)
{
_fDirty = TRUE;
fRet = TRUE;
}
else
{
TraceMsg(TF_WARNING, "DS UpdateComponent could not update an item");
}
}
else
{
TraceMsg(TF_WARNING, "DS UpdateComponent could not find item to update");
}
}
else
{
TraceMsg(TF_WARNING, "DS UpdateComponent has no components to update");
}
EXITPROC(1, "DS UpdateComponent=%d", fRet);
return fRet;
}
HRESULT CActiveDesktop::GetDesktopItemCount(LPINT lpiCount, DWORD dwReserved)
{
if (!lpiCount)
return (E_INVALIDARG);
*lpiCount = 0;
ENTERPROC(1, "DS GetComponentsCount()");
ASSERT(!dwReserved); // These should be 0
if (_hdsaComponent)
{
*lpiCount = DSA_GetItemCount(_hdsaComponent);
}
EXITPROC(1, "DS GetComponentsCount=%d", *lpiCount);
return S_OK;
}
HRESULT CActiveDesktop::GetDesktopItem(int nComponent, COMPONENT *pComp, DWORD dwReserved)
{
COMPONENTA CompA;
ASSERT(!dwReserved); // These should be 0
// We need to support IE4 apps calling with the old component structure too!
// We use the size field to detect IE4 v/s newer apps!
if ((nComponent < 0) || !pComp || ((pComp->dwSize != SIZEOF(*pComp)) && (pComp->dwSize != SIZEOF(IE4COMPONENT))))
return E_INVALIDARG;
CompA.dwSize = sizeof(COMPONENTA);
if (GetComponentPrivate(nComponent, &CompA))
{
//Convert the structure to the Public form.
ConvertCompStruct((COMPONENTA *)pComp, &CompA, FALSE);
return(S_OK);
}
else
return(E_FAIL);
}
BOOL CActiveDesktop::GetComponentPrivate(int nComponent, COMPONENTA *pcomp)
{
BOOL fRet = FALSE;
ENTERPROC(1, "DS GetComponent(nComponent=%d,pcomp=%08X)", nComponent, pcomp);
if (_hdsaComponent && pcomp && (nComponent < DSA_GetItemCount(_hdsaComponent)))
{
if (DSA_GetItem(_hdsaComponent, nComponent, pcomp) != -1)
{
fRet = TRUE;
}
else
{
TraceMsg(TF_WARNING, "DS GetComponent unable to get a component");
}
}
else
{
TraceMsg(TF_WARNING, "DS GetComponent does not have a DSA");
}
EXITPROC(1, "DS GetComponent=%d", fRet);
return fRet;
}
HRESULT CActiveDesktop::QueryInterface(REFIID riid, LPVOID *ppvObj)
{
if (IsEqualIID(riid, IID_IActiveDesktop))
{
*ppvObj = (IActiveDesktop *)this;
_Initialize();
}
else if (IsEqualIID(riid, IID_IUnknown))
{
*ppvObj = (IActiveDesktop *)this;
}
else if (IsEqualIID(riid, IID_IActiveDesktopP))
{
*ppvObj = (IActiveDesktopP *)this;
}
else if (IsEqualIID(riid, IID_IADesktopP2))
{
*ppvObj = (IADesktopP2 *)this;
}
else
{
*ppvObj = NULL;
return E_NOINTERFACE;
}
AddRef();
return S_OK;
}
// Helper function so that it's easy to create one internally
// Actually, it's not ver much help any more...
STDAPI CActiveDesktop_InternalCreateInstance(LPUNKNOWN * ppunk, REFIID riid)
{
return CActiveDesktop_CreateInstance(NULL, riid, (void **)ppunk);
}
// Our class factory create instance code
STDAPI CActiveDesktop_CreateInstance(LPUNKNOWN punkOuter, REFIID riid, void **ppvOut)
{
TraceMsg(TF_DESKSTAT, "CActiveDesktop- CreateInstance");
CActiveDesktop *pad = new CActiveDesktop();
if (pad)
{
HRESULT hres = pad->QueryInterface(riid, ppvOut);
pad->Release();
return hres;
}
*ppvOut = NULL;
return E_OUTOFMEMORY;
}
#ifdef DEBUG
// BUGBUG - Move g_dwDeskStatTrace into ccshell.ini to prevent recompiles.
DWORD g_dwDeskStatTrace = 2;
static DWORD g_dwIndent = 0;
static const TCHAR c_szDotDot[] = TEXT("..");
#define MAX_INDENTATION_VALUE 0x20
void EnterProcDS(DWORD dwTraceLevel, LPSTR pszFmt, ...)
{
TCHAR szFmt[1000];
TCHAR szOutput[1000];
va_list arglist;
SHAnsiToTChar(pszFmt, szFmt, ARRAYSIZE(szFmt));
if (dwTraceLevel <= g_dwDeskStatTrace)
{
szOutput[0] = TEXT('\0');
for (DWORD i = 0; i < g_dwIndent; i++)
{
lstrcat(szOutput, c_szDotDot);
}
va_start(arglist, pszFmt);
wvsprintf(szOutput + lstrlen(szOutput), szFmt, arglist);
va_end(arglist);
TraceMsg(TF_DESKSTAT, "%s", szOutput);
// We don't want this value to get out of hand because of
// unmatched Enter and Exit calls in functions (which will
// trash the stack).
if (g_dwIndent < MAX_INDENTATION_VALUE)
g_dwIndent++;
}
}
void ExitProcDS(DWORD dwTraceLevel, LPSTR pszFmt, ...)
{
TCHAR szFmt[1000];
TCHAR szOutput[1000];
va_list arglist;
SHAnsiToTChar(pszFmt, szFmt, ARRAYSIZE(szFmt));
if (dwTraceLevel <= g_dwDeskStatTrace)
{
//This can happen if the Enter and Exit procs are unmatched.
if (g_dwIndent > 0)
g_dwIndent--;
szOutput[0] = TEXT('\0');
for (DWORD i = 0; i < g_dwIndent; i++)
{
lstrcat(szOutput, c_szDotDot);
}
va_start(arglist, pszFmt);
wvsprintf(szOutput + lstrlen(szOutput), szFmt, arglist);
va_end(arglist);
TraceMsg(TF_DESKSTAT, "%s", szOutput);
}
}
#endif
/*
* IActiveDesktopP methods and helper functions
* IActiveDesktopP is a private interface used to implement helper
* functionality that is used internally by the various shell binaries.
* Notes:
* Getting an interface to IActiveDesktopP does not initialize the state
* of the object such that member functions are able to call IActiveDesktop
* member functions. This is so that it is a more lightweight implementation
* and also simplifies the implementation of SetScheme. If a subsequent QI for
* IActiveDesktop is performed then it will initialize properly and any member
* function can then be called.
*/
// SetScheme
// Used to set the current scheme that the object will read and write to
// when it is initialized. This method must be called before a subsequent
// QI to IActiveDesktop is made.
HRESULT CActiveDesktop::SetScheme(LPCWSTR pwszSchemeName, DWORD dwFlags)
{
LPTSTR pszSchemeName, pszAlloc;
int icch;
// Can't set the local scheme after we've been initialized...we can fix this
// later if necessary but for now it's simplest this way.
if (_fInitialized && (dwFlags & SCHEME_LOCAL))
return E_FAIL;
// Sanity checks
if (!pwszSchemeName || ((icch = lstrlenW(pwszSchemeName)) > MAX_PATH - 1))
return E_INVALIDARG;
#ifndef UNICODE
CHAR szName[MAX_PATH];
SHUnicodeToAnsi(pwszSchemeName, szName, ARRAYSIZE(szName));
pszSchemeName = szName;
#else
pszSchemeName = (LPTSTR)pwszSchemeName;
#endif
if (dwFlags & SCHEME_CREATE)
{
HRESULT hres;
HKEY hkey, hkey2;
if (ERROR_SUCCESS == (hres = RegCreateKey(HKEY_CURRENT_USER, REG_DESKCOMP_SCHEME_LOCATION, &hkey)))
{
if (ERROR_SUCCESS == (hres = RegCreateKey(hkey, pszSchemeName, &hkey2)))
RegCloseKey(hkey2);
RegCloseKey(hkey);
}
if (FAILED(hres))
return hres;
}
if (dwFlags & SCHEME_LOCAL)
{
// The local case is easy - just copy the string to our local variable,
// it will be used when IActiveDesktop is initialized.
if (!(pszAlloc = (LPTSTR)LocalAlloc(LPTR, (icch + 1) * sizeof(TCHAR))))
return E_OUTOFMEMORY;
if (_pszScheme)
LocalFree((HANDLE)_pszScheme);
_pszScheme = pszAlloc;
lstrcpy(_pszScheme, pszSchemeName);
}
if (dwFlags & SCHEME_GLOBAL)
{
// Update the registry with the new global scheme value
if (dwFlags & SCHEME_DISPLAY)
SHSetValue(HKEY_CURRENT_USER, REG_DESKCOMP_SCHEME, REG_VAL_SCHEME_DISPLAY,
REG_SZ, (LPBYTE)pszSchemeName, (lstrlen(pszSchemeName) + 1) * sizeof(TCHAR));
if (dwFlags & SCHEME_EDIT)
SHSetValue(HKEY_CURRENT_USER, REG_DESKCOMP_SCHEME, REG_VAL_SCHEME_EDIT,
REG_SZ, (LPBYTE)pszSchemeName, (lstrlen(pszSchemeName) + 1) * sizeof(TCHAR));
}
if (dwFlags & (SCHEME_REFRESH | SCHEME_UPDATE))
{
DWORD dwUpdateFlags = AD_APPLY_FORCE | AD_APPLY_HTMLGEN | AD_APPLY_SAVE;
if (dwFlags & SCHEME_REFRESH)
dwUpdateFlags |= (AD_APPLY_REFRESH | AD_APPLY_DYNAMICREFRESH);
_Initialize();
_fUseDynamicHtml = FALSE;
ApplyChanges(dwUpdateFlags);
}
return S_OK;
}
HRESULT GetGlobalScheme(LPWSTR pwszScheme, LPDWORD lpdwcchBuffer, DWORD dwFlags)
{
DWORD dwType, dwcbBuffer;
LONG lret;
TCHAR szScheme[MAX_PATH];
// Just pull it from the registry
dwcbBuffer = sizeof(szScheme);
if (ERROR_SUCCESS ==
(lret = SHGetValue(HKEY_CURRENT_USER, REG_DESKCOMP_SCHEME,
(dwFlags & SCHEME_EDIT) ? REG_VAL_SCHEME_EDIT : REG_VAL_SCHEME_DISPLAY, &dwType,
(LPBYTE)szScheme, &dwcbBuffer)))
{
SHTCharToUnicode(szScheme, pwszScheme, *lpdwcchBuffer);
// Fortunately, even Win95 implements lstrlenW
*lpdwcchBuffer = lstrlenW(pwszScheme);
}
return (lret == ERROR_SUCCESS ? S_OK : E_FAIL);
}
// GetScheme
HRESULT CActiveDesktop::GetScheme(LPWSTR pwszSchemeName, LPDWORD lpdwcchBuffer, DWORD dwFlags)
{
// Sanity checks
if (!pwszSchemeName || *lpdwcchBuffer == 0)
return E_INVALIDARG;
if (dwFlags & SCHEME_LOCAL)
{
if (!_pszScheme)
{
HRESULT hres;
// Special case if no local scheme has explicitly been selected yet.
// The default scheme is the global display scheme in this case.
if (SUCCEEDED(hres = GetGlobalScheme(pwszSchemeName, lpdwcchBuffer, SCHEME_DISPLAY)))
{
hres = SetScheme(pwszSchemeName, SCHEME_LOCAL);
}
return hres;
}
SHTCharToUnicode(_pszScheme, pwszSchemeName, *lpdwcchBuffer);
*lpdwcchBuffer = lstrlenW(pwszSchemeName);
return S_OK;
}
if (dwFlags & SCHEME_GLOBAL)
{
return GetGlobalScheme(pwszSchemeName, lpdwcchBuffer, dwFlags);
}
return E_INVALIDARG;
}
BOOL UpdateAllDesktopSubscriptions();
HRESULT CActiveDesktop::UpdateAllDesktopSubscriptions()
{
::UpdateAllDesktopSubscriptions();
return S_OK;
}
void CActiveDesktop::_GenerateHtmlBStrForComp(COMPONENTA *pComp, BSTR *pbstr)
{
ENTERPROC(2, "DS _GenerateHtmlBstrForComp");
if (_pStream = SHCreateMemStream(NULL, 0)) //Create a mem stream.
{
LARGE_INTEGER libMove = { 0 };
ULARGE_INTEGER libCurPos;
// Since _pStream is setup, the following call will generate the component HTML into
// that stream.
_GenerateHtmlComponent(pComp);
//Get the size of the stream generated.
_pStream->Seek(libMove, STREAM_SEEK_CUR, &libCurPos);
//Allocare a BSTR big enough to hold our component HTML code.
if (*pbstr = SysAllocStringLen(NULL, (libCurPos.LowPart) / sizeof(WCHAR)))
{
_pStream->Seek(libMove, STREAM_SEEK_SET, NULL);
_pStream->Read(*pbstr, libCurPos.LowPart, NULL);
}
//NOTE: The bStr is released by the caller.
ATOMICRELEASE(_pStream);
}
else
TraceMsg(TF_WARNING, "DS _GenerateHtmlBstrForComp unable to create a mem stream");
EXITPROC(2, "DS _GenerateHtmlBstrForComp");
}
void CActiveDesktop::_UpdateStyleOfElement(IHTMLElement *pElem, LPCOMPONENTA lpCompA)
{
IHTMLStyle *pHtmlStyle;
if (SUCCEEDED(pElem->get_style(&pHtmlStyle)))
{
long lPixelVal;
VARIANT vVal;
VARIANT vValNew;
if (SUCCEEDED(pHtmlStyle->get_pixelLeft(&lPixelVal)) && (lPixelVal != lpCompA->cpPos.iLeft))
{
TraceMsg(TF_DYNAMICHTML, "iLeft changes from %d to %d", lPixelVal, lpCompA->cpPos.iLeft);
pHtmlStyle->put_pixelLeft((long)(lpCompA->cpPos.iLeft));
}
if (SUCCEEDED(pHtmlStyle->get_pixelTop(&lPixelVal)) && (lPixelVal != lpCompA->cpPos.iTop))
{
TraceMsg(TF_DYNAMICHTML, "iTop changes from %d to %d", lPixelVal, lpCompA->cpPos.iTop);
pHtmlStyle->put_pixelTop((long)(lpCompA->cpPos.iTop));
}
VariantInit(&vVal);
if (SUCCEEDED(pHtmlStyle->get_width(&vVal))) //Get the width as BSTR to see if width attribute exists
{
//See if the width attribute exists now.
if ((vVal.vt == VT_BSTR) && (vVal.bstrVal == NULL))
{
// Width attribute does not exist for this element; This means that
// this element has the default width (may be a picture shown in it's original width).
if (lpCompA->cpPos.dwWidth != COMPONENT_DEFAULT_WIDTH)
{
//Component's new width is different from the default width. So, set the new width.
TraceMsg(TF_DYNAMICHTML, "dwWidth changes from default to %d", lpCompA->cpPos.dwWidth);
pHtmlStyle->put_pixelWidth((long)(lpCompA->cpPos.dwWidth));
}
//else, nothing to do! (the widths match exactly).
}
else
{
// Width attribute exists! That means that this element has a width other than the
// default width.
// See if the new width is the default width.
if (lpCompA->cpPos.dwWidth == COMPONENT_DEFAULT_WIDTH)
{
// The old width is NOT default; But, the new width is default. So, let's just
// remove the width attribute.
VariantInit(&vValNew);
vValNew.vt = VT_BSTR;
vValNew.bstrVal = NULL;
pHtmlStyle->put_width(vValNew);
VariantClear(&vValNew);
}
else
{
//Get the existing width in pixels.
if (SUCCEEDED(pHtmlStyle->get_pixelWidth(&lPixelVal)) && (((DWORD)lPixelVal) != lpCompA->cpPos.dwWidth))
{
TraceMsg(TF_DYNAMICHTML, "dwWidth changes from %d to %d", lPixelVal, lpCompA->cpPos.dwWidth);
pHtmlStyle->put_pixelWidth((long)(lpCompA->cpPos.dwWidth));
}
//else, nothing else to do because the widths match!
}
}
VariantClear(&vVal);
}
if (SUCCEEDED(pHtmlStyle->get_height(&vVal))) //Get the height as BSTR to see if height attribute exists
{
// See if the height attribute exists.
if ((vVal.vt == VT_BSTR) && (vVal.bstrVal == NULL))
{
// Height attribute does not exist for this element; This means that
// this element has the default height (may be a picture shown in it's original height).
if (lpCompA->cpPos.dwHeight != COMPONENT_DEFAULT_HEIGHT)
{
//Component's new height is different from the default height. So, set the new height.
TraceMsg(TF_DYNAMICHTML, "dwHeight changes from default to %d", lpCompA->cpPos.dwHeight);
pHtmlStyle->put_pixelHeight((long)(lpCompA->cpPos.dwHeight));
}
//else, nothing to do! (the heights match exactly).
}
else
{
// Height attribute exists! That means that this element has a height other than the
// default height.
// See if the new height is the default height.
if (lpCompA->cpPos.dwHeight == COMPONENT_DEFAULT_HEIGHT)
{
// The old height is NOT default; But, the new height is default. So, let's just
// remove the height attribute.
VariantInit(&vValNew);
vValNew.vt = VT_BSTR;
vValNew.bstrVal = NULL;
pHtmlStyle->put_height(vValNew); //remove the height attribute!
VariantClear(&vValNew);
}
else
{
//Get the existing height in pixels and see if it is different.
if (SUCCEEDED(pHtmlStyle->get_pixelHeight(&lPixelVal)) && (((DWORD)lPixelVal) != lpCompA->cpPos.dwHeight))
{
//Since the new height is different, let's use set the new height!
TraceMsg(TF_DYNAMICHTML, "dwHeight changes from %d to %d", lPixelVal, lpCompA->cpPos.dwHeight);
pHtmlStyle->put_pixelHeight((long)(lpCompA->cpPos.dwHeight));
}
//else, nothing else to do because the heights match!
}
}
VariantClear(&vVal);
}
if (SUCCEEDED(pHtmlStyle->get_zIndex(&vVal)) && (vVal.vt == VT_I4) && (vVal.lVal != lpCompA->cpPos.izIndex))
{
TraceMsg(TF_DYNAMICHTML, "ZIndex changes from %d to %d", vVal.lVal, lpCompA->cpPos.izIndex);
vVal.lVal = lpCompA->cpPos.izIndex;
pHtmlStyle->put_zIndex(vVal);
}
VariantClear(&vVal);
pHtmlStyle->Release();
}
//BUGBUG: Should we check for and set the other attributes like "resizeable" etc.,?
}
BOOL CActiveDesktop::_UpdateIdOfElement(IHTMLElement *pElem, LPCOMPONENTA lpCompA)
{
BSTR bstrId;
BOOL fWholeElementReplaced = FALSE; //Assume that the item id does not change.
//Check if the Id of the component and the element matches.
if (SUCCEEDED(pElem->get_id(&bstrId))) //get the old id
{
if (((DWORD)StrToIntW(bstrId)) != lpCompA->dwID)
{
// The following technic does not work in some versions of MSHTML.DLL
// because IHTMLElement->put_id() does not work unless the doc
// is in "design mode".
TCHAR szNewId[MAXID_LENGTH];
BSTR bstrNewId;
HRESULT hr = S_OK;
wsprintf(szNewId, TEXT("%d"), lpCompA->dwID);
#ifdef DEBUG
{
TCHAR szOldId[MAXID_LENGTH];
wsprintf(szOldId, TEXT("%d"), StrToIntW(bstrId));
TraceMsg(TF_DYNAMICHTML, "DHTML: Id changes from %s to %s", szOldId, szNewId);
}
#endif //DEBUG
//The Ids do not match. So, let's set the new ID.
if (bstrNewId = SysAllocStringT(szNewId))
{
hr = pElem->put_id(bstrNewId);
SysFreeString(bstrNewId);
}
if (FAILED(hr))
{
//Replace the whole element's HTML with the newly generated HTML
BSTR bstrComp = 0;
_GenerateHtmlBStrForComp(lpCompA, &bstrComp);
if (bstrComp)
{
if (FAILED(hr = pElem->put_outerHTML(bstrComp)))
TraceMsg(TF_DYNAMICHTML, "DHTML: put_outerHTML failed");
fWholeElementReplaced = TRUE;
SysFreeString(bstrComp);
}
else
{
AssertMsg(FALSE, TEXT("DHTML: Unable to create html for comp"));
}
}
}
//else the ids match; nothing to do!
SysFreeString(bstrId); //free the old id.
}
else
{
AssertMsg(FALSE, TEXT("DS Unable to get the id of the element"));
}
return fWholeElementReplaced;
}
HRESULT CActiveDesktop::_UpdateHtmlElement(IHTMLElement *pElem)
{
VARIANT vData;
TCHAR szUrl[INTERNET_MAX_URL_LENGTH];
TCHAR szSrcPath[INTERNET_MAX_URL_LENGTH];
LPTSTR lpszSrcPath;
COMPONENTA CompA;
int iIndex;
//If all components are disabled, then we nuke this component from HTML page.
if (!_co.fEnableComponents)
{
TraceMsg(TF_DYNAMICHTML, "DHTML: No item shown in this mode; so, deleting items");
pElem->put_outerHTML((BSTR)s_sstrEmpty.wsz);
return S_OK; //Nothing else to do!
}
VariantInit(&vData);
//First determine if the given element is currently a desktop item. (It could have been deleted)
//Get the element's "src" attribute.
if (FAILED(pElem->getAttribute((BSTR)s_sstrSRCMember.wsz, VARIANT_FALSE, &vData)) ||
(vData.vt == VT_NULL) ||
(vData.bstrVal == NULL))
{
//If the subscribed_url is not present, then it could be an object with a classid.
if (FAILED(pElem->getAttribute((BSTR)s_sstrclassid.wsz, VARIANT_FALSE, &vData)) ||
(vData.vt == VT_NULL))
{
//This element is does not have "src=" or "classid=" attributes. How did this ever
// become a desktop item with "name=deskmovr" or "name=deskmovrw"?? Hmmmmmm....!!!
#ifdef DEBUG
{
BSTR bstrHtmlForElem;
// Get the HTML corresponding to the element that does not have a subscribed URL
if (SUCCEEDED(pElem->get_outerHTML(&bstrHtmlForElem)))
{
TraceMsg(TF_DYNAMICHTML, "DHTML: Rogue element: %s", bstrHtmlForElem);
SysFreeString(bstrHtmlForElem);
}
}
TraceMsg(TF_WARNING, "DHTML: Unable to get the subscribed_url or classid");
#endif
//Since this element does not seem to be a valid desktop item, let's nuke it!
pElem->put_outerHTML((BSTR)s_sstrEmpty.wsz); //delete this element.
return (E_FAIL); //Nothing else to for this element! It's gone!!!
}
if ((vData.vt == VT_NULL) || (vData.bstrVal == NULL))
return E_FAIL;
ASSERT(vData.vt == VT_BSTR);
ASSERT(StrCmpNW(vData.bstrVal, L"clsid:", lstrlen(TEXT("clsid:"))) == 0);
SHUnicodeToTChar(vData.bstrVal + lstrlen(TEXT("clsid:")), szUrl, ARRAYSIZE(szUrl));
lpszSrcPath = szUrl; //For classid, the SrcPath and the Url are the same.
}
else
{
DWORD dwSize;
if (vData.bstrVal == NULL)
return (E_FAIL);
ASSERT(vData.vt == VT_BSTR);
SHUnicodeToTChar((LPWSTR)vData.bstrVal, szUrl, ARRAYSIZE(szUrl));
dwSize = ARRAYSIZE(szSrcPath);
lpszSrcPath = szSrcPath;
if (FAILED(PathCreateFromUrl(szUrl, lpszSrcPath, &dwSize, 0)))
{
lpszSrcPath = szUrl;
}
}
VariantClear(&vData); //We made a TCHAR copy above. So, ok to free this.
CompA.dwSize = sizeof(CompA);
// First use the Source path to Find the component; This is much more efficient because it
// involves no conversion from Path to Url and vice-versa.
if ((iIndex = _FindComponentBySource(lpszSrcPath, &CompA)) < 0)
{
// Could not find component using SrcPath!
// Let's try using the SrcUrl; This is less efficient.
iIndex = _FindComponentBySrcUrl(szUrl, &CompA);
}
if ((iIndex >= 0) && (CompA.fChecked))
{
//The component is found and it is enabled.
TraceMsg(TF_DYNAMICHTML, "DHTML:Updating desktop item with URL: %s", szUrl);
// If the id changes, we replace the whole HTML for that element, so, no need to check for
// the individual styles.
if (!_UpdateIdOfElement(pElem, &CompA))
_UpdateStyleOfElement(pElem, &CompA);
CompA.fDirty = TRUE; //Mark the component sothat we know that it had been updated.
UpdateComponentPrivate(iIndex, &CompA);
}
else
{
ASSERT((iIndex == -1) || (!CompA.fChecked)); //Component not found OR it is disabled!
TraceMsg(TF_DYNAMICHTML, "DHTML: Deleting desktop item with URL: %s, SrcPath:%s", szUrl, lpszSrcPath);
//The component is not present now. So, delete this element from the html page.
pElem->put_outerHTML((BSTR)s_sstrEmpty.wsz);
}
return S_OK;
}
// This code enumerates and then updates all the desktop item elements in the active desktop based
// on the current status of the active desktop items in the CActiveDesktop object (The current
// status is initialized by reading from the registry when ActiveDesktop object is initialized).
HRESULT CActiveDesktop::_UpdateDesktopItemHtmlElements(IHTMLDocument2 *pDoc)
{
HRESULT hres = S_OK;
IHTMLElementCollection *pAllElements;
TraceMsg(TF_DYNAMICHTML, "DHTML: Updating Desktop html elements dynamically");
if (!_fInitialized) //If not yet initialized, initialize now because we need _co.fEnableComponents.
_Initialize();
// We need to check for a change in the background color only if there is no wallpaper or
// the wallpaper is a picture.
if (IsWallpaperPicture(_szSelectedWallpaper))
{
COLORREF rgbDesktop;
TCHAR szRgbDesktop[10];
VARIANT vColor;
//Check to see if the background color has changed
rgbDesktop = GetSysColor(COLOR_DESKTOP);
wsprintf(szRgbDesktop, TEXT("#%02lx%02lx%02lx"), GetRValue(rgbDesktop),
GetGValue(rgbDesktop),
GetBValue(rgbDesktop));
if (SUCCEEDED(pDoc->get_bgColor(&vColor)) && (vColor.vt == VT_BSTR))
{
BSTR bstrNewBgColor = SysAllocStringT(szRgbDesktop);
//Compare the new and the old strings.
if (StrCmpW(vColor.bstrVal, bstrNewBgColor))
{
BSTR bstrOldBgColor = vColor.bstrVal; //Save the old bstr.
//So, the colors are different. Set the new color.
vColor.bstrVal = bstrNewBgColor;
bstrNewBgColor = bstrOldBgColor; //Set it here sothat it is freed later.
if (FAILED(pDoc->put_bgColor(vColor)))
{
TraceMsg(TF_DYNAMICHTML, "DHTML: Unable to change the background color");
}
}
if (bstrNewBgColor)
SysFreeString(bstrNewBgColor);
VariantClear(&vColor);
}
}
//Get a collection of All elements in the Document
if (SUCCEEDED(pDoc->get_all(&pAllElements)))
{
VARIANT vName, vIndex;
IDispatch *pDisp;
int i;
long lItemsEnumerated = 0;
long lLength = 0;
#ifdef DEBUG
pAllElements->get_length(&lLength);
TraceMsg(TF_DYNAMICHTML, "DHTML: Length of All elements:%d", lLength);
#endif
for (i = 0; i <= 1; i++)
{
//Collect all the elements that have the name="DeskMovr" and then name="DeskMovrW"
vName.vt = VT_BSTR;
vName.bstrVal = (BSTR)((i == 0) ? s_sstrDeskMovr.wsz : s_sstrDeskMovrW.wsz);
VariantInit(&vIndex); //We want to get all elements. So, vIndex is set to VT_EMPTY
if (SUCCEEDED(pAllElements->item(vName, vIndex, &pDisp)) && pDisp) //Collect all elements we want
{
IHTMLElementCollection *pDeskCollection;
if (SUCCEEDED(pDisp->QueryInterface(IID_IHTMLElementCollection, (void **)&pDeskCollection)))
{
IUnknown *pUnk;
IEnumVARIANT *pEnumVar;
if (SUCCEEDED(pDeskCollection->get_length(&lLength))) //Number of elements.
lItemsEnumerated += lLength; //Total number of items enumerated.
TraceMsg(TF_DYNAMICHTML, "DHTML: Enumerated %d number of elements", lLength);
//Get the enumerator
if (SUCCEEDED(pDeskCollection->get__newEnum(&pUnk)))
{
if (SUCCEEDED(pUnk->QueryInterface(IID_IEnumVARIANT, (void **)&pEnumVar)))
{
VARIANT vElem;
long lEnumCount = 0;
DWORD cElementsFetched;
while (SUCCEEDED(pEnumVar->Next(1, &vElem, &cElementsFetched)) && (cElementsFetched == 1))
{
IHTMLElement *pElem;
lEnumCount++;
// Access the element from the variant.....!
if ((vElem.vt == VT_DISPATCH) && SUCCEEDED(vElem.pdispVal->QueryInterface(IID_IHTMLElement, (void **)&pElem)))
{
_UpdateHtmlElement(pElem); //Update the desktop element's attributes.
pElem->Release();
}
VariantClear(&vElem);
}
//Number of items enumerated must be the same as the length
ASSERT(lEnumCount == lLength);
pEnumVar->Release();
}
pUnk->Release();
}
pDeskCollection->Release();
}
else
{
IHTMLElement *pElem;
// The QI(IID_IHTMLElementCollection) has failed. It may be because only one item
// was returned rather than a collection.
if (SUCCEEDED(pDisp->QueryInterface(IID_IHTMLElement, (void **)&pElem)))
{
_UpdateHtmlElement(pElem); //Update the desktop element's attributes.
pElem->Release();
}
else
TraceMsg(TF_WARNING, "DHTML: Unable to get a collection or a single element");
}
pDisp->Release();
}
} // for loop enumeating "DeskMovr" and "DeskMovrW" items.
pAllElements->Release();
}
// All the elements already present in the Doc have been updated. Now, let's add the
// new elements, if any.
if (_co.fEnableComponents)
_InsertNewDesktopItems(pDoc);
else
{
TraceMsg(TF_DYNAMICHTML, "DHTML: No components are to be shown in this mode;");
}
return hres;
}
HRESULT CActiveDesktop::_InsertNewDesktopItems(IHTMLDocument2 *pDoc)
{
IHTMLElement *pBody;
if (SUCCEEDED(pDoc->get_body(&pBody)))
{
if (_hdsaComponent)
{
int i, iCount;
iCount = DSA_GetItemCount(_hdsaComponent);
for (i = 0; i < iCount; i++)
{
COMPONENTA comp;
comp.dwSize = sizeof(comp);
if (DSA_GetItem(_hdsaComponent, i, &comp) != -1)
{
//Check if this is a newly added component AND it is enabled.
if ((!comp.fDirty) && comp.fChecked)
{
TraceMsg(TF_DYNAMICHTML, "DHTML: Inserted comp: %s", comp.szSource);
//Yup! This is a newly added component!
BSTR bstrComp = 0;
//This is a new component. Generate the HTML for the component.
_GenerateHtmlBStrForComp(&comp, &bstrComp);
//Insert the component.
pBody->insertAdjacentHTML((BSTR)s_sstrBeforeEnd.wsz, (BSTR)bstrComp);
//Free the string.
SysFreeString(bstrComp);
}
}
else
{
TraceMsg(TF_WARNING, "DHTML: InsertNewComp: Unable to get component %d.", i);
}
}
}
pBody->Release();
}
return S_OK;
}
// This function takes a pointer to the ActiveDesktop's ole obj, reads all the changes to be done
// from the registry and makes those changes to the various elements through dynamic HTML interfaces.
HRESULT CActiveDesktop::MakeDynamicChanges(IOleObject *pOleObj)
{
IHTMLDocument2 *pDoc;
HRESULT hres = E_FAIL;
ENTERPROC(2, "MakeDynamicChanges");
if (pOleObj && SUCCEEDED(pOleObj->QueryInterface(IID_IHTMLDocument2, (void **)&pDoc)))
{
// Enumerate all the active desktop components and ensure they are up to date.
_UpdateDesktopItemHtmlElements(pDoc);
pDoc->Release();
}
else
{
TraceMsg(TF_WARNING, "DHTML: MakeDynamicChanges: Unable to get IHTMLDocument2");
}
EXITPROC(2, "MakeDynamicChanges");
return(hres);
}
// SetSafeMode
// Either puts the active desktop in safemode or restores it to the previous
// scheme before safemode was entered.
HRESULT CActiveDesktop::SetSafeMode(DWORD dwFlags)
{
// Make sure we are in active desktop mode.
SHELLSTATE ss = { 0 };
BOOL fSetSafeMode = (dwFlags & SSM_SET) != 0;
SHGetSetSettings(&ss, SSF_DESKTOPHTML, FALSE);
if (ss.fDesktopHTML)
{
// All we need to do is switch the "display" scheme to "safemode" in order to
// go into safemode. To go out, we just switch the "display" scheme back to the
// previous "edit" scheme.
WCHAR wszEdit[MAX_PATH];
WCHAR wszDisplay[MAX_PATH];
DWORD dwcch = MAX_PATH;
if (SUCCEEDED(GetScheme(wszEdit, &dwcch, SCHEME_GLOBAL | SCHEME_EDIT)))
{
dwcch = MAX_PATH;
if (SUCCEEDED(GetScheme(wszDisplay, &dwcch, SCHEME_GLOBAL | SCHEME_DISPLAY)))
{
BOOL fInSafeMode = (StrCmpW(wszDisplay, REG_DESKCOMP_SAFEMODE_SUFFIX_L) == 0);
if (fSetSafeMode != fInSafeMode)
{
LPWSTR lpwstr;
DWORD dwSchemeFlags = SCHEME_GLOBAL | SCHEME_DISPLAY;
if (dwFlags & SSM_REFRESH)
dwSchemeFlags |= SCHEME_REFRESH;
if (dwFlags & SSM_UPDATE)
dwSchemeFlags |= SCHEME_UPDATE;
lpwstr = fSetSafeMode ? REG_DESKCOMP_SAFEMODE_SUFFIX_L : wszEdit;
SetScheme(lpwstr, dwSchemeFlags);
}
}
}
}
return S_OK;
}
// EnsureUpdateHTML
// Ensures that the current html file present on the disk is in sync
// with the registry information for the current active desktop scheme. If
// it is not in sync then a fresh copy of the file is generated from the
// registry for the current scheme.
HRESULT CActiveDesktop::EnsureUpdateHTML(void)
{
DWORD dwFlags = 0;
DWORD dwDataLength = sizeof(DWORD);
DWORD dwType;
LONG lRet;
TCHAR lpszDeskcomp[MAX_PATH];
TCHAR szDesktopFile[MAX_PATH];
DWORD dwRestrictUpdate;
DWORD dwRestrict = SHRestricted2W(REST_NoChannelUI, NULL, 0);
DWORD dwSize = SIZEOF(dwRestrictUpdate);
BOOL fComponentsDirty = FALSE; //Assume that the components are NOT dirty!
DWORD dwVersion;
DWORD dwMinorVersion;
BOOL fStaleInfoInReg = FALSE;
BOOL fComponentsZoomDirty = FALSE;
static BOOL s_fNoDeskComp = (BOOL)-1;
static BOOL s_fNoWallpaper = (BOOL)-1;
BOOL fNoDeskComp = SHRestricted(REST_NODESKCOMP);
BOOL fNoWallpaper = SHRestricted(REST_NOHTMLWALLPAPER);
BOOL fAdminComponent = FALSE;
HKEY hkey = NULL;
if (ERROR_SUCCESS != SHGetValue(HKEY_CURRENT_USER, REG_DESKCOMP_COMPONENTS_ROOT, REG_VAL_GENERAL_RESTRICTUPDATE, NULL, (LPVOID)&dwRestrictUpdate, &dwSize))
dwRestrictUpdate = 0;
GetRegLocation(lpszDeskcomp, SIZECHARS(lpszDeskcomp), REG_DESKCOMP_COMPONENTS, NULL);
//See if this branch of registry is old
if ((lRet = SHGetValue(HKEY_CURRENT_USER, (LPCTSTR)lpszDeskcomp, REG_VAL_COMP_VERSION, &dwType,
(LPBYTE)(&dwVersion), &dwDataLength)) == ERROR_SUCCESS)
{
if (dwVersion < CUR_DESKHTML_VERSION)
fStaleInfoInReg = TRUE;
else
{
//Major versions are equal. Check minor versions.
if ((lRet = SHGetValue(HKEY_CURRENT_USER, (LPCTSTR)lpszDeskcomp, REG_VAL_COMP_MINOR_VERSION, &dwType,
(LPBYTE)(&dwMinorVersion), &dwDataLength)) == ERROR_SUCCESS)
{
if (dwMinorVersion != CUR_DESKHTML_MINOR_VERSION)
fStaleInfoInReg = TRUE;
}
else
fStaleInfoInReg = TRUE;
}
}
else
fStaleInfoInReg = TRUE;
dwDataLength = SIZEOF(DWORD);
//Check the dirty bit to see if we need to re-generate the desktop html
if ((lRet = SHGetValue(HKEY_CURRENT_USER, (LPCTSTR)lpszDeskcomp, REG_VAL_COMP_GENFLAGS, &dwType,
(LPBYTE)(&dwFlags), &dwDataLength)) == ERROR_SUCCESS)
{
if (IsFlagSet(dwFlags, COMPONENTS_DIRTY))
fComponentsDirty = TRUE;
if (IsFlagSet(dwFlags, COMPONENTS_ZOOMDIRTY))
fComponentsZoomDirty = TRUE;
}
// See if we need to add/delete an administrator added desktop component now
if (ERROR_SUCCESS == RegOpenKeyEx(HKEY_CURRENT_USER, REG_DESKCOMP_ADMINCOMP_ROOT, 0, KEY_READ, &hkey))
{
WCHAR wszDisplay[MAX_PATH];
DWORD dwcch = MAX_PATH;
if (FAILED(GetScheme(wszDisplay, &dwcch, SCHEME_GLOBAL | SCHEME_DISPLAY)) || (StrCmpW(wszDisplay, REG_DESKCOMP_SAFEMODE_SUFFIX_L) != 0))
{
// We're not in safe mode, it's OK to add the components
fAdminComponent = TRUE;
}
}
// 99/03/23 #237632 vtan: If the monitor arrangement has been changed from underneath
// the user (perhaps by another user) then make sure that the components are in valid
// positions. If not then snap them back into the visible space and mark them as dirty
// so that the desktop.htt file is regenerated.
_Initialize();
if (_hdsaComponent)
{
int i, iComponentCount;;
iComponentCount = DSA_GetItemCount(_hdsaComponent);
for (i = 0; i < iComponentCount; ++i)
{
COMPONENTA *pComponent;
pComponent = reinterpret_cast<COMPONENTA*>(DSA_GetItemPtr(_hdsaComponent, i));
if (pComponent != NULL)
{
bool bChangedPosition, bChangedSize;
ValidateComponentPosition(&pComponent->cpPos, pComponent->dwCurItemState, pComponent->iComponentType, &bChangedPosition, &bChangedSize);
if (bChangedPosition || bChangedSize)
{
TBOOL(UpdateComponentPrivate(i, pComponent));
fComponentsDirty = true;
}
}
}
}
GetPerUserFileName(szDesktopFile, ARRAYSIZE(szDesktopFile), DESKTOPHTML_FILENAME);
if (fComponentsDirty ||
fComponentsZoomDirty ||
fStaleInfoInReg ||
fAdminComponent ||
fNoDeskComp != s_fNoDeskComp ||
fNoWallpaper != s_fNoWallpaper ||
(dwRestrictUpdate != dwRestrict) ||
(!PathFileExistsAndAttributes(szDesktopFile, NULL))) //See if the file exists!
{
// Clear out any html wallpaper if it exists and the restriction is set
if (fNoWallpaper != s_fNoWallpaper)
{
if (fNoWallpaper && !IsWallpaperPicture(_szSelectedWallpaper))
SetWallpaper(L"", 0);
s_fNoWallpaper = fNoWallpaper;
}
// Disable components if the restriction is set
if (fNoDeskComp != s_fNoDeskComp)
{
// We can't set fEnableComponents to FALSE because there is no way via the UI
// for the user to turn it back on again if the restriction is lifted. Instead we add
// special case code to _GenerateHtml that checks the restriction too.
// _co.fEnableComponents = !fNoDeskComp;
s_fNoDeskComp = fNoDeskComp;
}
if (fAdminComponent)
{
COMPONENT comp;
TCHAR pszAdminComp[INTERNET_MAX_URL_LENGTH];
CHAR szUrl[INTERNET_MAX_URL_LENGTH];
CHAR * pszUrl;
CHAR * pszUrlList;
TCHAR * aszAdminComp[] = { REG_VAL_ADMINCOMP_ADD, REG_VAL_ADMINCOMP_DELETE, NULL };
int i = 0;
comp.dwSize = SIZEOF(comp);
comp.dwCurItemState = IS_SPLIT | IS_ADJUSTLISTVIEW;
while (aszAdminComp[i])
{
dwDataLength = SIZEOF(pszAdminComp);
// The reg value contains an array of space separated urls - currently we support adding and deleting
// a desktop item via this mechanism.
if (SHQueryValueEx(hkey, aszAdminComp[i], NULL, &dwType, (LPBYTE)pszAdminComp, &dwDataLength) == ERROR_SUCCESS)
{
SHTCharToAnsi(pszAdminComp, szUrl, ARRAYSIZE(szUrl));
pszUrlList = szUrl;
while (pszUrl = StrTokEx(&pszUrlList, " ")) {
SHAnsiToUnicode(pszUrl, comp.wszSource, ARRAYSIZE(comp.wszSource));
dwDataLength = ARRAYSIZE(comp.wszSource);
ParseURLFromOutsideSourceW(comp.wszSource, comp.wszSource, &dwDataLength, NULL);
if (lstrcmp(aszAdminComp[i], REG_VAL_ADMINCOMP_ADD) == 0)
{
AddUrl(NULL, (LPCWSTR)comp.wszSource, &comp, ADDURL_SILENT);
fComponentsZoomDirty = TRUE;
}
else
{
RemoveDesktopItem((LPCOMPONENT)&comp, 0);
}
}
}
i++;
}
}
// Go through the entire list of desktop components and ensure any split/fullscreen
// components are at their correct size/location.
if (fComponentsZoomDirty)
{
if (_hdsaComponent)
{
int i;
for (i = 0; i < DSA_GetItemCount(_hdsaComponent); i++)
{
COMPONENTA * pcompT;
if (pcompT = (COMPONENTA *)DSA_GetItemPtr(_hdsaComponent, i))
{
if (ISZOOMED(pcompT))
{
BOOL fAdjustListview = (pcompT->dwCurItemState & IS_ADJUSTLISTVIEW);
ZoomComponent(&pcompT->cpPos, pcompT->dwCurItemState, fAdjustListview);
if (fAdjustListview)
pcompT->dwCurItemState &= ~IS_ADJUSTLISTVIEW;
}
}
}
SetDesktopFlags(COMPONENTS_ZOOMDIRTY, 0);
}
}
// NOTE #1: The above initialization would have changed the Z-order because of
// SortAndRationalize and so we need to APPLY_SAVE here.
// Warning: APPLY_SAVE changes the dwID field of components. This should not
// be a problem because we do this just before generating a new HTML file.
// NOTE #2: Do NOT use AD_APPLY_FORCE here. That sets the _fPatternDirty too and
// that causes a SystemParametersInfo() call which results in WM_SYSCOLORCHANGE
// and this causes a refresh. So, we set the dirty bit explicitly here.
_fDirty = TRUE; // See Note#2 above.
ApplyChanges(AD_APPLY_SAVE | AD_APPLY_HTMLGEN);
lRet = ERROR_SUCCESS;
if (dwRestrictUpdate != dwRestrict)
SHSetValue(HKEY_CURRENT_USER, REG_DESKCOMP_COMPONENTS_ROOT, REG_VAL_GENERAL_RESTRICTUPDATE, NULL, (LPVOID)&dwRestrict, SIZEOF(dwRestrict));
}
if (hkey)
{
RegCloseKey(hkey);
SHDeleteKey(HKEY_CURRENT_USER, REG_DESKCOMP_ADMINCOMP_ROOT);
}
return (lRet == ERROR_SUCCESS ? S_OK : E_FAIL);
}
// ReReadWallpaper()
// If the wallpaper was read when the active desktop was disabled, we would have read it from
// the old location. Now, if the active desktop is turned ON, then we need to re-read the wallpaper
// from the new location. We need to do this iff the wallpaper has not been changed in the mean-while
HRESULT CActiveDesktop::ReReadWallpaper(void)
{
if ((!_fDirty) || (!_co.fActiveDesktop)) //If nothing has changed OR if active desktop is OFF,
return(S_FALSE); // then nothing to do!
//ActiveDesktop is ON in our object. Read current shell state.
SHELLSTATE ss = { 0 };
SHGetSetSettings(&ss, SSF_DESKTOPHTML, FALSE);
if (ss.fDesktopHTML)
return(S_FALSE); // Active Desktop state hasn't changed. So, nothing to do!
//So, Active desktop was originally OFF and now it is turned ON.
//If if someone changed the wallpaper, we should not mess with it.
if (_fWallpaperDirty || _fWallpaperChangedDuringInit)
return(S_FALSE);
// No one has changed the wallpaper. So, we must re-read it from the new wallpaper location
// sothat we get the correct wallpaper for the active desktop mode.
_ReadWallpaper(TRUE);
return(S_OK);
}
// GetADObjectFlags()
// Get the Active Desktop object's internal flags
HRESULT CActiveDesktop::GetADObjectFlags(LPDWORD lpdwFlags, DWORD dwMask)
{
ASSERT(lpdwFlags);
*lpdwFlags = 0; //Init the flags
if ((dwMask & GADOF_DIRTY) && _fDirty)
*lpdwFlags |= GADOF_DIRTY;
return(S_OK);
}
#ifdef _WIN64
/*
*char *StrTokEx(pstring, control) - tokenize string with delimiter in control
*Purpose:
* StrTokEx considers the string to consist of a sequence of zero or more
* text tokens separated by spans of one or more control chars. the first
* call, with string specified, returns a pointer to the first char of the
* first token, and will write a null char into pstring immediately
* following the returned token. when no tokens remain
* in pstring a NULL pointer is returned. remember the control chars with a
* bit map, one bit per ascii char. the null char is always a control char.
*Entry:
* char **pstring - ptr to ptr to string to tokenize
* char *control - string of characters to use as delimiters
*Exit:
* returns pointer to first token in string,
* returns NULL when no more tokens remain.
* pstring points to the beginning of the next token.
*WARNING!!!
* upon exit, the first delimiter in the input string will be replaced with '\0'
* copied from iert.lib
*/
extern "C" char * __cdecl StrTokEx(char ** spstring, const char * scontrol)
{
unsigned char **pstring = (unsigned char**)spstring;
unsigned char *control = (unsigned char*)scontrol;
unsigned char *str;
const unsigned char *ctrl = control;
unsigned char map[32];
int count;
unsigned char *tokenstr;
if (*pstring == NULL)
return NULL;
/* Clear control map */
for (count = 0; count < 32; count++)
map[count] = 0;
/* Set bits in delimiter table */
do
{
map[*ctrl >> 3] |= (1 << (*ctrl & 7));
} while (*ctrl++);
/* Initialize str. */
str = *pstring;
/* Find beginning of token (skip over leading delimiters). Note that
* there is no token if this loop sets str to point to the terminal
* null (*str == '\0') */
while ((map[*str >> 3] & (1 << (*str & 7))) && *str)
str++;
tokenstr = str;
/* Find the end of the token. If it is not the end of the string,
* put a null there. */
for (; *str; str++)
{
if (map[*str >> 3] & (1 << (*str & 7)))
{
*str++ = '\0';
break;
}
}
/* string now points to beginning of next token */
*pstring = str;
/* Determine if a token has been found. */
if (tokenstr == str)
return NULL;
else
return (char*)tokenstr;
}
#endif //_WIN64