Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

2894 lines
90 KiB

  1. #include "stdafx.h"
  2. #include "utils.h"
  3. #include "..\\deskfldr.h"
  4. #include <cfgmgr32.h> // MAX_GUID_STRING_LEN
  5. #pragma hdrstop
  6. const TCHAR c_szSetup[] = REGSTR_PATH_SETUP TEXT("\\Setup");
  7. const TCHAR c_szSharedDir[] = TEXT("SharedDir");
  8. BOOL g_fDirtyAdvanced;
  9. BOOL g_fLaunchGallery; // If true, we launched the gallery, so close the dialog.
  10. DWORD g_dwApplyFlags = (AD_APPLY_ALL | AD_APPLY_DYNAMICREFRESH);
  11. // used to indicate if desktop cleanup settings have changes
  12. extern int g_iRunDesktopCleanup = BST_INDETERMINATE; // indicates uninitilized value.
  13. STDAPI ApplyDesktopCleanupSettings();
  14. BOOL _IsNonEnumPolicySet(const CLSID *pclsid);
  15. static const LPCTSTR c_rgpszWallpaperExt[] = {
  16. TEXT("BMP"), TEXT("GIF"),
  17. TEXT("JPG"), TEXT("JPE"),
  18. TEXT("JPEG"),TEXT("DIB"),
  19. TEXT("PNG"), TEXT("HTM"),
  20. TEXT("HTML")
  21. };
  22. const static DWORD aBackHelpIDs[] = {
  23. IDC_BACK_SELECT, IDH_DISPLAY_BACKGROUND_WALLPAPERLIST,
  24. IDC_BACK_WPLIST, IDH_DISPLAY_BACKGROUND_WALLPAPERLIST,
  25. IDC_BACK_BROWSE, IDH_DISPLAY_BACKGROUND_BROWSE_BUTTON,
  26. IDC_BACK_WEB, IDH_DISPLAY_BACKGROUND_DESKTOP_ITEMS,
  27. IDC_BACK_DISPLAY, IDH_DISPLAY_BACKGROUND_PICTUREDISPLAY,
  28. IDC_BACK_WPSTYLE, IDH_DISPLAY_BACKGROUND_PICTUREDISPLAY,
  29. IDC_BACK_PREVIEW, IDH_DISPLAY_BACKGROUND_MONITOR,
  30. IDC_BACK_COLORPICKERLABEL, IDH_DISPLAY_BACKGROUND_BACKGROUND_COLOR,
  31. IDC_BACK_COLORPICKER, IDH_DISPLAY_BACKGROUND_BACKGROUND_COLOR,
  32. 0, 0
  33. };
  34. #define SZ_HELPFILE_DESKTOPTAB TEXT("display.hlp")
  35. #define SZ_REGKEY_PROGRAMFILES TEXT("Software\\Microsoft\\Windows\\CurrentVersion")
  36. #define SZ_REGKEY_PLUS95DIR TEXT("Software\\Microsoft\\Plus!\\Setup") // PLUS95_KEY
  37. #define SZ_REGKEY_PLUS98DIR TEXT("Software\\Microsoft\\Plus!98") // PLUS98_KEY
  38. #define SZ_REGKEY_KIDSDIR TEXT("Software\\Microsoft\\Microsoft Kids\\Kids Plus!") // KIDS_KEY
  39. #define SZ_REGKEY_WALLPAPER TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Wallpaper")
  40. #define SZ_REGKEY_WALLPAPERMRU TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Wallpaper\\MRU")
  41. #define SZ_REGKEY_LASTTHEME TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Themes\\LastTheme")
  42. #define SZ_REGVALUE_PLUS95DIR TEXT("DestPath") // PLUS95_PATH
  43. #define SZ_REGVALUE_PLUS98DIR TEXT("Path") // PLUS98_PATH
  44. #define SZ_REGVALUE_KIDSDIR TEXT("InstallDir") // KIDS_PATH
  45. #define SZ_REGVALUE_PROGRAMFILESDIR TEXT("ProgramFilesDir")
  46. #define SZ_REGVALUE_PROGRAMFILESDIR TEXT("ProgramFilesDir")
  47. #define SZ_REGVALUE_USETILE TEXT("UseTile") // If it's not a watermark background, does the user want to default to "Center" or "Stretch". Different users like different settings.
  48. #define SZ_REGVALUE_LASTSCAN TEXT("LastScan") // When was the last time we scanned the file sizes?
  49. #ifndef RECTHEIGHT
  50. #define RECTHEIGHT(rc) ((rc).bottom - (rc).top)
  51. #define RECTWIDTH(rc) ((rc).right - (rc).left)
  52. #endif
  53. //===========================
  54. // *** Class Internals & Helpers ***
  55. //===========================
  56. #ifdef NEVER
  57. BOOL IsWallpaperDesktopV2(LPCTSTR lpszWallpaper)
  58. {
  59. TCHAR szBuf[MAX_PATH];
  60. TCHAR szDesktopV2path[MAX_PATH];
  61. TCHAR szDesktopV2[100];
  62. LoadString(HINST_THISDLL, IDS_MILLEN_DESKTOP, szDesktopV2, ARRAYSIZE(szDesktopV2));
  63. LoadString(HINST_THISDLL, IDS_MILLEN_DESKTOP_PATH, szDesktopV2path, ARRAYSIZE(szDesktopV2path));
  64. //Get the windows directory.
  65. if (!GetWindowsDirectory(szBuf, ARRAYSIZE(szBuf)))
  66. {
  67. szBuf[0] = 0;
  68. }
  69. //Append the path to Desktop V2.
  70. if(PathAppend(szBuf, szDesktopV2path))
  71. {
  72. //Append the wallpaper name.
  73. if(PathAppend(szBuf, szDesktopV2))
  74. {
  75. //Check the given wallpaper name against the DesktopV2 wallpaper name we just built.
  76. if(StrCmpIC(szBuf, lpszWallpaper) == 0)
  77. return(TRUE); //Yup! the wallpaper we have is indeed the DesktopV2 wallpaper.
  78. }
  79. }
  80. return FALSE; //Nope this is someother wallpaper.
  81. }
  82. #endif //NEVER
  83. HRESULT CBackPropSheetPage::_LoadState(void)
  84. {
  85. HRESULT hr = S_OK;
  86. if (!_fStateLoaded)
  87. {
  88. hr = _LoadIconState();
  89. if (SUCCEEDED(hr))
  90. {
  91. hr = _LoadDesktopOptionsState();
  92. if (SUCCEEDED(hr))
  93. {
  94. _fStateLoaded = TRUE;
  95. }
  96. }
  97. }
  98. return hr;
  99. }
  100. #define SZ_ICON_DEFAULTICON L"DefaultValue"
  101. HRESULT CBackPropSheetPage::_GetIconPath(IN CLSID clsid, IN LPCWSTR pszName, IN BOOL fOldIcon, IN LPWSTR pszPath, IN DWORD cchSize)
  102. {
  103. HRESULT hr = E_FAIL;
  104. int nIndex;
  105. if (!StrCmpIW(SZ_ICON_DEFAULTICON, pszName))
  106. {
  107. pszName = L"";
  108. }
  109. for (nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
  110. {
  111. if (IsEqualCLSID(*(c_aIconRegKeys[nIndex].pclsid), clsid) &&
  112. !StrCmpIW(pszName, c_aIconRegKeys[nIndex].szIconValue))
  113. {
  114. // We found it.
  115. if (fOldIcon)
  116. {
  117. wnsprintfW(pszPath, cchSize, L"%s,%d", _IconData[nIndex].szOldFile, _IconData[nIndex].iOldIndex);
  118. }
  119. else
  120. {
  121. wnsprintfW(pszPath, cchSize, L"%s,%d", _IconData[nIndex].szNewFile, _IconData[nIndex].iNewIndex);
  122. }
  123. hr = S_OK;
  124. break;
  125. }
  126. }
  127. return hr;
  128. }
  129. HRESULT CBackPropSheetPage::_SetIconPath(IN CLSID clsid, IN LPCWSTR pszName, IN LPCWSTR pszPath, IN int nResourceID)
  130. {
  131. HRESULT hr = E_FAIL;
  132. int nIndex;
  133. if (!StrCmpIW(SZ_ICON_DEFAULTICON, pszName))
  134. {
  135. pszName = L"";
  136. }
  137. for (nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
  138. {
  139. if (IsEqualCLSID(*(c_aIconRegKeys[nIndex].pclsid), clsid) &&
  140. !StrCmpIW(pszName, c_aIconRegKeys[nIndex].szIconValue))
  141. {
  142. TCHAR szTemp[MAX_PATH];
  143. if (!pszPath || !pszPath[0])
  144. {
  145. // The caller didn't specify an icon so use the default values.
  146. if (!SHExpandEnvironmentStrings(c_aIconRegKeys[nIndex].pszDefault, szTemp, ARRAYSIZE(szTemp)))
  147. {
  148. StrCpyN(szTemp, c_aIconRegKeys[nIndex].pszDefault, ARRAYSIZE(szTemp));
  149. }
  150. pszPath = szTemp;
  151. nResourceID = c_aIconRegKeys[nIndex].nDefaultIndex;
  152. }
  153. // We found it.
  154. StrCpyNW(_IconData[nIndex].szNewFile, pszPath, ARRAYSIZE(_IconData[nIndex].szNewFile));
  155. _IconData[nIndex].iNewIndex = nResourceID;
  156. hr = S_OK;
  157. break;
  158. }
  159. }
  160. return hr;
  161. }
  162. HRESULT CBackPropSheetPage::_LoadIconState(void)
  163. {
  164. HRESULT hr = S_OK;
  165. // load the icons and add them to the image lists
  166. // get the icon files and indexes from the registry, including for the Default recycle bin
  167. for (int nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
  168. {
  169. TCHAR szTemp[MAX_PATH];
  170. szTemp[0] = 0;
  171. IconGetRegValueString(c_aIconRegKeys[nIndex].pclsid, TEXT("DefaultIcon"), c_aIconRegKeys[nIndex].szIconValue, szTemp, ARRAYSIZE(szTemp));
  172. int iIndex = PathParseIconLocation(szTemp);
  173. // store the icon information
  174. StrCpyNW(_IconData[nIndex].szOldFile, szTemp, ARRAYSIZE(_IconData[nIndex].szOldFile));
  175. StrCpyNW(_IconData[nIndex].szNewFile, szTemp, ARRAYSIZE(_IconData[nIndex].szNewFile));
  176. _IconData[nIndex].iOldIndex = iIndex;
  177. _IconData[nIndex].iNewIndex = iIndex;
  178. }
  179. return hr;
  180. }
  181. HRESULT CBackPropSheetPage::_LoadDesktopOptionsState(void)
  182. {
  183. HRESULT hr = S_OK;
  184. int iStartPanel;
  185. TCHAR szRegPath[MAX_PATH];
  186. // i = 0 is for StartPanel off and i = 1 is for StartPanel ON!
  187. for(iStartPanel = 0; iStartPanel <= 1; iStartPanel++)
  188. {
  189. int iIndex;
  190. //Get the proper registry path based on if StartPanel is ON/OFF
  191. wsprintf(szRegPath, REGSTR_PATH_HIDDEN_DESKTOP_ICONS, c_apstrRegLocation[iStartPanel]);
  192. //Load the settings for all icons we are interested in.
  193. for(iIndex = 0; iIndex < NUM_DESKICONS; iIndex++)
  194. {
  195. TCHAR szValueName[MAX_GUID_STRING_LEN];
  196. SHUnicodeToTChar(c_aDeskIconId[iIndex].pwszCLSID, szValueName, ARRAYSIZE(szValueName));
  197. //Read the setting from the registry!
  198. _aHideDesktopIcon[iStartPanel][iIndex].fHideIcon = SHRegGetBoolUSValue(szRegPath, szValueName, FALSE, /* default */FALSE);
  199. _aHideDesktopIcon[iStartPanel][iIndex].fDirty = FALSE;
  200. //Update the NonEnum attribute data.
  201. if((c_aDeskIconId[iIndex].fCheckNonEnumAttrib) && (iStartPanel == 1))
  202. {
  203. TCHAR szAttriRegPath[MAX_PATH];
  204. DWORD dwSize = sizeof(_aDeskIconNonEnumData[iIndex].rgfAttributes);
  205. ULONG rgfDefault = 0; //By default the SFGAO_NONENUMERATED bit if off!
  206. wsprintf(szAttriRegPath, REGSTR_PATH_EXP_SHELLFOLDER, szValueName);
  207. //Read the attributes.
  208. SHRegGetUSValue(szAttriRegPath, REGVAL_ATTRIBUTES,
  209. NULL,
  210. &(_aDeskIconNonEnumData[iIndex].rgfAttributes),
  211. &dwSize,
  212. FALSE,
  213. &rgfDefault,
  214. sizeof(rgfDefault));
  215. //If SHGAO_NONENUMERATED bit is ON, then we need to hide the checkboxes in both modes.
  216. if(_aDeskIconNonEnumData[iIndex].rgfAttributes & SFGAO_NONENUMERATED)
  217. {
  218. //Overwrite what we read earlier! These icons are hidden in both modes!
  219. _aHideDesktopIcon[0][iIndex].fHideIcon = TRUE;
  220. _aHideDesktopIcon[1][iIndex].fHideIcon = TRUE;
  221. }
  222. }
  223. //Check the policy if needed!
  224. if((c_aDeskIconId[iIndex].fCheckNonEnumPolicy) && (iStartPanel == 1))
  225. {
  226. if(_IsNonEnumPolicySet(c_aDeskIconId[iIndex].pclsid))
  227. {
  228. //Remember that this policy is set. So that we can disable these controls in UI.
  229. _aDeskIconNonEnumData[iIndex].fNonEnumPolicySet = TRUE;
  230. //Remember to hide these icons in both modes!
  231. _aHideDesktopIcon[0][iIndex].fHideIcon = TRUE;
  232. _aHideDesktopIcon[1][iIndex].fHideIcon = TRUE;
  233. }
  234. }
  235. } //for (all desktop items)
  236. } //for both the modes (StartPanel off and On)
  237. return hr;
  238. }
  239. HRESULT CBackPropSheetPage::_SaveIconState(void)
  240. {
  241. HRESULT hr = S_OK;
  242. BOOL fDorked = FALSE;
  243. if (_fStateLoaded)
  244. {
  245. int nIndex;
  246. // Change the system icons
  247. for(nIndex = 0; nIndex < ARRAYSIZE(_IconData); nIndex++)
  248. {
  249. if ((lstrcmpi(_IconData[nIndex].szNewFile, _IconData[nIndex].szOldFile) != 0) ||
  250. (_IconData[nIndex].iNewIndex != _IconData[nIndex].iOldIndex))
  251. {
  252. TCHAR szTemp[MAX_PATH];
  253. wnsprintf(szTemp, ARRAYSIZE(szTemp), TEXT("%s,%d"), _IconData[nIndex].szNewFile, _IconData[nIndex].iNewIndex);
  254. IconSetRegValueString(c_aIconRegKeys[nIndex].pclsid, TEXT("DefaultIcon"), c_aIconRegKeys[nIndex].szIconValue, szTemp);
  255. // Next two lines necessary if the user does an Apply as opposed to OK
  256. StrCpyNW(_IconData[nIndex].szOldFile, _IconData[nIndex].szNewFile, ARRAYSIZE(_IconData[nIndex].szOldFile));
  257. _IconData[nIndex].iOldIndex = _IconData[nIndex].iNewIndex;
  258. fDorked = TRUE;
  259. }
  260. }
  261. }
  262. // Make the system notice we changed the system icons
  263. if (fDorked)
  264. {
  265. SHChangeNotify(SHCNE_ASSOCCHANGED, 0, NULL, NULL); // should do the trick!
  266. SHUpdateRecycleBinIcon();
  267. }
  268. return hr;
  269. }
  270. HRESULT CBackPropSheetPage::_SaveDesktopOptionsState(void)
  271. {
  272. HRESULT hr = S_OK;
  273. int iStartPanel;
  274. TCHAR szRegPath[MAX_PATH];
  275. BOOL fUpdateDesktop = FALSE;
  276. // i = 0 is for StartPanel off and i = 1 is for StartPanel ON!
  277. for(iStartPanel = 0; iStartPanel <= 1; iStartPanel++)
  278. {
  279. int iIndex;
  280. //Get the proper registry path based on if StartPanel is ON/OFF
  281. wsprintf(szRegPath, REGSTR_PATH_HIDDEN_DESKTOP_ICONS, c_apstrRegLocation[iStartPanel]);
  282. //Load the settings for all icons we are interested in.
  283. for(iIndex = 0; iIndex < NUM_DESKICONS; iIndex++)
  284. {
  285. //Update the registry only if the particular icon entry is dirty.
  286. if(_aHideDesktopIcon[iStartPanel][iIndex].fDirty)
  287. {
  288. TCHAR szValueName[MAX_GUID_STRING_LEN];
  289. SHUnicodeToTChar(c_aDeskIconId[iIndex].pwszCLSID, szValueName, ARRAYSIZE(szValueName));
  290. //Write the setting to the registry!
  291. SHRegSetUSValue(szRegPath, szValueName, REG_DWORD,
  292. &(_aHideDesktopIcon[iStartPanel][iIndex].fHideIcon),
  293. sizeof(_aHideDesktopIcon[iStartPanel][iIndex].fHideIcon),
  294. SHREGSET_FORCE_HKCU);
  295. _aHideDesktopIcon[iStartPanel][iIndex].fDirty = FALSE;
  296. fUpdateDesktop = TRUE; //Desktop window needs to be refreshed.
  297. // Note this will be done only once per index because SFGAO_NONENUMERATED bit is
  298. // reset in rgfAttributes.
  299. if((c_aDeskIconId[iIndex].fCheckNonEnumAttrib) &&
  300. (_aDeskIconNonEnumData[iIndex].rgfAttributes & SFGAO_NONENUMERATED) &&
  301. (_aHideDesktopIcon[iStartPanel][iIndex].fHideIcon == FALSE))
  302. {
  303. TCHAR szAttriRegPath[MAX_PATH];
  304. wsprintf(szAttriRegPath, REGSTR_PATH_EXP_SHELLFOLDER, szValueName);
  305. //We need to remove the SFGAO_NONENUMERATED attribute bit!
  306. //We assume here is that when we save to registry, we save the same value
  307. //for both the modes.
  308. ASSERT(_aHideDesktopIcon[0][iIndex].fHideIcon == _aHideDesktopIcon[1][iIndex].fHideIcon);
  309. //Strip out the NonEnum attribute!
  310. _aDeskIconNonEnumData[iIndex].rgfAttributes &= ~SFGAO_NONENUMERATED;
  311. //And save it in the registry!
  312. SHRegSetUSValue(szAttriRegPath, REGVAL_ATTRIBUTES,
  313. REG_DWORD,
  314. &(_aDeskIconNonEnumData[iIndex].rgfAttributes),
  315. sizeof(_aDeskIconNonEnumData[iIndex].rgfAttributes),
  316. SHREGSET_FORCE_HKCU);
  317. }
  318. }
  319. }
  320. }
  321. if(fUpdateDesktop)
  322. PostMessage(GetShellWindow(), WM_COMMAND, FCIDM_REFRESH, 0L); //Refresh desktop!
  323. _fHideDesktopIconDirty = FALSE; //We just saved. So, reset the dirty bit!
  324. return hr;
  325. }
  326. int CBackPropSheetPage::_AddAFileToLV(LPCTSTR pszDir, LPTSTR pszFile, UINT nBitmap)
  327. {
  328. int index = -1;
  329. TCHAR szTemp[MAX_PATH];
  330. if (pszDir)
  331. {
  332. lstrcpyn(szTemp, pszDir, ARRAYSIZE(szTemp));
  333. PathAppend(szTemp, pszFile);
  334. }
  335. else if (pszFile && *pszFile && (lstrcmpi(pszFile, g_szNone) != 0))
  336. {
  337. lstrcpyn(szTemp, pszFile, ARRAYSIZE(szTemp));
  338. }
  339. else
  340. {
  341. *szTemp = TEXT('\0');
  342. }
  343. TCHAR szLVIText[MAX_PATH];
  344. StrCpyN(szLVIText, PathFindFileName(pszFile), ARRAYSIZE(szLVIText));
  345. PathRemoveExtension(szLVIText);
  346. PathMakePretty(szLVIText);
  347. LV_ITEM lvi = {0};
  348. lvi.mask = LVIF_TEXT | LVIF_PARAM | (nBitmap != -1 ? LVIF_IMAGE : 0);
  349. lvi.iItem = 0x7FFFFFFF;
  350. lvi.pszText = szLVIText;
  351. lvi.iImage = nBitmap;
  352. lvi.lParam = (LPARAM)StrDup(szTemp);
  353. if (lvi.lParam)
  354. {
  355. index = ListView_InsertItem(_hwndLV, &lvi);
  356. if (index == -1)
  357. {
  358. LocalFree((HLOCAL)lvi.lParam);
  359. }
  360. else
  361. {
  362. ListView_SetColumnWidth(_hwndLV, 0, LVSCW_AUTOSIZE);
  363. }
  364. }
  365. return index;
  366. }
  367. void CBackPropSheetPage::_AddFilesToLV(LPCTSTR pszDir, LPCTSTR pszSpec, UINT nBitmap, BOOL fCount)
  368. {
  369. WIN32_FIND_DATA fd;
  370. HANDLE h;
  371. TCHAR szBuf[MAX_PATH];
  372. StrCpyN(szBuf, pszDir, ARRAYSIZE(szBuf));
  373. StrCatBuff(szBuf, TEXT("\\*."), SIZECHARS(szBuf));
  374. StrCatBuff(szBuf, pszSpec, SIZECHARS(szBuf));
  375. h = FindFirstFile(szBuf, &fd);
  376. if (h != INVALID_HANDLE_VALUE)
  377. {
  378. do
  379. {
  380. // Skip files that are "Super-hidden" like "Winnt.bmp" and "Winnt256.bmp"
  381. if ((fd.dwFileAttributes & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN)) != (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN))
  382. {
  383. if (!fCount)
  384. {
  385. _AddAFileToLV(pszDir, fd.cFileName, nBitmap);
  386. }
  387. else
  388. {
  389. _nFileCount++;
  390. }
  391. }
  392. }
  393. while (FindNextFile(h, &fd));
  394. FindClose(h);
  395. }
  396. }
  397. int CBackPropSheetPage::_FindWallpaper(LPCTSTR pszFile)
  398. {
  399. int nItems = ListView_GetItemCount(_hwndLV);
  400. int i;
  401. for (i=0; i<nItems; i++)
  402. {
  403. LV_ITEM lvi = {0};
  404. lvi.iItem = i;
  405. lvi.mask = LVIF_PARAM;
  406. ListView_GetItem(_hwndLV, &lvi);
  407. if (StrCmpIC(pszFile, (LPCTSTR)lvi.lParam) == 0)
  408. {
  409. return i;
  410. }
  411. }
  412. return -1;
  413. }
  414. void CBackPropSheetPage::_UpdatePreview(IN WPARAM flags, IN BOOL fUpdateThemePage)
  415. {
  416. WALLPAPEROPT wpo;
  417. wpo.dwSize = sizeof(WALLPAPEROPT);
  418. g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  419. if (wpo.dwStyle & WPSTYLE_TILE)
  420. {
  421. flags |= BP_TILE;
  422. }
  423. else if(wpo.dwStyle & WPSTYLE_STRETCH)
  424. {
  425. flags |= BP_STRETCH;
  426. }
  427. WCHAR wszWallpaper[INTERNET_MAX_URL_LENGTH];
  428. g_pActiveDesk->GetWallpaper(wszWallpaper, ARRAYSIZE(wszWallpaper), 0);
  429. HRESULT hr = S_OK;
  430. if (!_pThemePreview)
  431. {
  432. hr = CoCreateInstance(CLSID_ThemePreview, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IThemePreview, &_pThemePreview));
  433. if (SUCCEEDED(hr) && _punkSite)
  434. {
  435. IPropertyBag * pPropertyBag;
  436. hr = _punkSite->QueryInterface(IID_PPV_ARG(IPropertyBag, &pPropertyBag));
  437. ASSERT(SUCCEEDED(hr));
  438. if (SUCCEEDED(hr))
  439. {
  440. HWND hwndPlaceHolder = GetDlgItem(_hwnd, IDC_BACK_PREVIEW);
  441. if (hwndPlaceHolder)
  442. {
  443. RECT rcPreview;
  444. GetClientRect(hwndPlaceHolder, &rcPreview);
  445. MapWindowPoints(hwndPlaceHolder, _hwnd, (LPPOINT)&rcPreview, 2);
  446. hr = _pThemePreview->CreatePreview(_hwnd, TMPREV_SHOWMONITOR | TMPREV_SHOWBKGND, WS_VISIBLE | WS_CHILDWINDOW | WS_OVERLAPPED, 0, rcPreview.left, rcPreview.top, rcPreview.right - rcPreview.left, rcPreview.bottom - rcPreview.top, pPropertyBag, IDC_BACK_PREVIEW);
  447. if (SUCCEEDED(hr))
  448. {
  449. // If we succeeded, remove the dummy window.
  450. DestroyWindow(hwndPlaceHolder);
  451. hr = SHPropertyBag_WritePunk(pPropertyBag, SZ_PBPROP_PREVIEW2, _pThemePreview);
  452. _fThemePreviewCreated = TRUE;
  453. }
  454. }
  455. pPropertyBag->Release();
  456. }
  457. }
  458. }
  459. if (_punkSite)
  460. {
  461. IThemeUIPages * pThemeUIPages;
  462. HRESULT hr = _punkSite->QueryInterface(IID_PPV_ARG(IThemeUIPages, &pThemeUIPages));
  463. if (SUCCEEDED(hr))
  464. {
  465. hr = pThemeUIPages->UpdatePreview(0);
  466. pThemeUIPages->Release();
  467. }
  468. if (fUpdateThemePage)
  469. {
  470. IPropertyBag * pPropertyBag;
  471. hr = _punkSite->QueryInterface(IID_PPV_ARG(IPropertyBag, &pPropertyBag));
  472. if (SUCCEEDED(hr))
  473. {
  474. // Tell the theme that we have customized the values.
  475. hr = SHPropertyBag_WriteInt(pPropertyBag, SZ_PBPROP_CUSTOMIZE_THEME, 0);
  476. pPropertyBag->Release();
  477. }
  478. }
  479. }
  480. if (!_fThemePreviewCreated)
  481. {
  482. HWND hwndOldPreview = GetDlgItem(_hwnd, IDC_BACK_PREVIEW);
  483. if (hwndOldPreview)
  484. {
  485. SendDlgItemMessage(_hwnd, IDC_BACK_PREVIEW, WM_SETBACKINFO, flags, 0);
  486. }
  487. }
  488. }
  489. void CBackPropSheetPage::_EnableControls(void)
  490. {
  491. BOOL fEnable;
  492. WCHAR wszWallpaper[INTERNET_MAX_URL_LENGTH];
  493. LPTSTR pszWallpaper;
  494. g_pActiveDesk->GetWallpaper(wszWallpaper, ARRAYSIZE(wszWallpaper), 0);
  495. pszWallpaper = (LPTSTR)wszWallpaper;
  496. BOOL fIsPicture = IsWallpaperPicture(pszWallpaper);
  497. // Style combo only enabled if a non-null picture
  498. // is being viewed.
  499. fEnable = fIsPicture && (*pszWallpaper) && (!_fPolicyForStyle);
  500. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_WPSTYLE), fEnable);
  501. // 98/09/10 vtan #209753: Also remember to disable the corresponding
  502. // text item with the keyboard shortcut. Not disabling this will
  503. // allow the shortcut to be invoked but will cause the incorrect
  504. // dialog item to be "clicked".
  505. (BOOL)EnableWindow(GetDlgItem(_hwnd, IDC_BACK_DISPLAY), fEnable);
  506. }
  507. int CBackPropSheetPage::_GetImageIndex(LPCTSTR pszFile)
  508. {
  509. int iRet = 0;
  510. if (pszFile && *pszFile)
  511. {
  512. LPCTSTR pszExt = PathFindExtension(pszFile);
  513. if (*pszExt == TEXT('.'))
  514. {
  515. pszExt++;
  516. for (iRet=0; iRet<ARRAYSIZE(c_rgpszWallpaperExt); iRet++)
  517. {
  518. if (StrCmpIC(pszExt, c_rgpszWallpaperExt[iRet]) == 0)
  519. {
  520. //
  521. // Add one because 'none' took the 0th slot.
  522. //
  523. iRet++;
  524. return(iRet);
  525. }
  526. }
  527. //
  528. // If we fell off the end of the for loop here,
  529. // this is a file with unknown extension. So, we assume that
  530. // it is a normal wallpaper and it gets the Bitmap's icon
  531. //
  532. iRet = 1;
  533. }
  534. else
  535. {
  536. //
  537. // Unknown files get Bitmap's icon.
  538. //
  539. iRet = 1;
  540. }
  541. }
  542. return iRet;
  543. }
  544. // This function is called when another tab is trying to change our
  545. // Tile mode. This means that our tab may not have been activated yet
  546. // and may not activate until later.
  547. HRESULT CBackPropSheetPage::_SetNewWallpaperTile(IN DWORD dwMode, IN BOOL fUpdateThemePage)
  548. {
  549. HRESULT hr = E_UNEXPECTED;
  550. AssertMsg((NULL != g_pActiveDesk), TEXT("We need g_pActiveDesk or we can't change the background"));
  551. if (g_pActiveDesk)
  552. {
  553. WALLPAPEROPT wpo;
  554. wpo.dwSize = sizeof(wpo);
  555. g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  556. wpo.dwStyle = dwMode;
  557. hr = g_pActiveDesk->SetWallpaperOptions(&wpo, 0);
  558. }
  559. if (_hwndWPStyle)
  560. {
  561. ComboBox_SetCurSel(_hwndWPStyle, dwMode);
  562. _UpdatePreview(0, fUpdateThemePage);
  563. }
  564. return hr;
  565. }
  566. HRESULT CBackPropSheetPage::_SetNewWallpaper(LPCTSTR pszFileIn, IN BOOL fUpdateThemePageIn)
  567. {
  568. HRESULT hr = S_OK;
  569. WCHAR szFile[MAX_PATH];
  570. WCHAR szTemp[MAX_PATH];
  571. LPWSTR pszFile = szFile;
  572. DWORD cchFile = ARRAYSIZE(szFile);
  573. LPWSTR pszTemp = szTemp;
  574. DWORD cchTemp = ARRAYSIZE(szTemp);
  575. //
  576. // Make a copy of the file name.
  577. //
  578. DWORD cchFileIn = lstrlen(pszFileIn) + 1;
  579. if ( cchFileIn > cchFile )
  580. {
  581. cchFile = cchFileIn;
  582. pszFile = (LPWSTR) LocalAlloc(LPTR, cchFile * sizeof(WCHAR));
  583. if (NULL == pszFile)
  584. {
  585. hr = E_OUTOFMEMORY;
  586. goto Cleanup;
  587. }
  588. }
  589. StrCpyN(pszFile, pszFileIn, cchFile);
  590. //
  591. // Replace all "(none)" with empty strings.
  592. //
  593. if (lstrcmpi(pszFile, g_szNone) == 0)
  594. {
  595. pszFile[0] = TEXT('\0');
  596. }
  597. //
  598. // Replace net drives with UNC names.
  599. //
  600. if( pszFile[1] == TEXT(':') )
  601. {
  602. DWORD dwErr;
  603. TCHAR szDrive[3];
  604. //
  605. // Copy just the drive letter and see if it maps to a network drive.
  606. //
  607. lstrcpyn(szDrive, pszFile, ARRAYSIZE(szDrive));
  608. dwErr = SHWNetGetConnection(szDrive, pszTemp, &cchTemp );
  609. //
  610. // See if our buffer was too small. If so, make it bigger and try
  611. // again.
  612. //
  613. if ( ERROR_MORE_DATA == dwErr )
  614. {
  615. // Add the size of the rest of the filepath to the UNC path.
  616. cchTemp += cchFile;
  617. pszTemp = (LPWSTR) LocalAlloc( LPTR, cchTemp * sizeof(WCHAR) );
  618. if ( NULL == pszTemp )
  619. {
  620. hr = E_OUTOFMEMORY;
  621. goto Cleanup;
  622. }
  623. dwErr = SHWNetGetConnection(szDrive, pszTemp, &cchTemp );
  624. }
  625. //
  626. // If it maps to a network location, replace the network drive letter
  627. // with the UNC path.
  628. //
  629. if ( NO_ERROR == dwErr )
  630. {
  631. if (pszTemp[0] == TEXT('\\') && pszTemp[1] == TEXT('\\'))
  632. {
  633. DWORD cchLen;
  634. wcscat(pszTemp, pszFile + 2);
  635. //
  636. // See if the new string will fit into our file buffer.
  637. //
  638. cchLen = wcslen(pszTemp) + 1;
  639. if ( cchLen > cchFile )
  640. {
  641. if ( szFile != pszFile )
  642. {
  643. LocalFree( pszFile );
  644. }
  645. cchFile = cchLen;
  646. pszFile = (LPWSTR) LocalAlloc(LPTR, cchFile * sizeof(WCHAR) );
  647. if ( NULL == pszFile )
  648. {
  649. hr = E_OUTOFMEMORY;
  650. goto Cleanup;
  651. }
  652. }
  653. wcscpy(pszFile, pszTemp);
  654. }
  655. }
  656. }
  657. //
  658. // If necessary, update the desk state object.
  659. //
  660. hr = g_pActiveDesk->GetWallpaper(pszTemp, cchTemp, 0);
  661. if (FAILED(hr))
  662. goto Cleanup;
  663. //
  664. // If we need more room, allocate it on the heap and try again.
  665. //
  666. if ( MAKE_HRESULT( 0, FACILITY_WIN32, ERROR_MORE_DATA ) == hr )
  667. {
  668. if ( szTemp != pszTemp )
  669. {
  670. LocalFree( pszTemp );
  671. }
  672. cchTemp = INTERNET_MAX_URL_LENGTH;
  673. pszTemp = (LPWSTR) LocalAlloc( LPTR, cchTemp * sizeof(WCHAR) );
  674. if ( NULL == pszTemp )
  675. {
  676. hr = E_OUTOFMEMORY;
  677. goto Cleanup;
  678. }
  679. hr = g_pActiveDesk->GetWallpaper(pszTemp, cchTemp, 0);
  680. if (S_OK != hr)
  681. goto Cleanup;
  682. }
  683. //
  684. // Make sure the old background doesn't equal the new background.
  685. //
  686. if (StrCmpIC(pszTemp, pszFile) != 0)
  687. {
  688. //
  689. // Did they choose something other than a .bmp?
  690. // And is ActiveDesktop off?
  691. //
  692. if (!IsNormalWallpaper(pszFileIn))
  693. {
  694. Str_SetPtr(&_pszOriginalFile, pszFileIn);
  695. }
  696. else
  697. {
  698. // We may not need to have a temp file.
  699. Str_SetPtr(&_pszOriginalFile, NULL);
  700. }
  701. Str_SetPtr(&_pszLastSourcePath, pszFile);
  702. hr = g_pActiveDesk->SetWallpaper(pszFile, 0);
  703. if (SUCCEEDED(hr))
  704. {
  705. if (fUpdateThemePageIn)
  706. {
  707. _SetNewWallpaperTile(_GetStretchMode(pszFile), fUpdateThemePageIn);
  708. }
  709. }
  710. }
  711. //
  712. // Update the preview picture of the new wallpaper.
  713. //
  714. _UpdatePreview(0, fUpdateThemePageIn);
  715. //
  716. // If the wallpaper does not have a directory specified, (this may happen if other apps. change this value),
  717. // we have to figure it out.
  718. //
  719. //
  720. // REVIEW: gpease 17-MAY-2001
  721. // GetWallpaperWithPath() probably should return a value if it
  722. // fails or the buffer is too small. We might need to grow the
  723. // buffer to append strings to the pszFile. Currently, it looks
  724. // like this function takes "foobar.bmp"and prepends "c:\windows\"
  725. // to it so MAX_PATH is probably sufficent.
  726. //
  727. GetWallpaperWithPath(pszFile, pszTemp, cchTemp);
  728. LPTSTR pszFileForList = (_pszOriginalFile ? _pszOriginalFile : pszTemp);
  729. int iSelectionNew = (pszFileForList[0] ? _FindWallpaper(pszFileForList) : 0);
  730. if (iSelectionNew == -1)
  731. {
  732. iSelectionNew = _AddAFileToLV(NULL, pszFileForList, _GetImageIndex(pszFileForList));
  733. }
  734. _fSelectionFromUser = FALSE; // disable
  735. //
  736. // If necessary, select the item in the listview.
  737. //
  738. int iSelected = ListView_GetNextItem(_hwndLV, -1, LVNI_SELECTED);
  739. if (iSelected != iSelectionNew)
  740. {
  741. ListView_SetItemState(_hwndLV, iSelectionNew, LVIS_SELECTED | LVIS_FOCUSED, LVIS_SELECTED | LVIS_FOCUSED);
  742. }
  743. //
  744. // Put all controls in correct enabled state.
  745. //
  746. _EnableControls();
  747. //
  748. // Make sure the selected item is visible.
  749. //
  750. ListView_EnsureVisible(_hwndLV, iSelectionNew, FALSE);
  751. _fSelectionFromUser = TRUE; // re-enable
  752. Cleanup:
  753. if ( szFile != pszFile && NULL != pszFile )
  754. {
  755. LocalFree( pszFile );
  756. }
  757. if ( szTemp != pszTemp && NULL != pszTemp )
  758. {
  759. LocalFree( pszTemp );
  760. }
  761. return hr;
  762. }
  763. int CALLBACK CBackPropSheetPage::_SortBackgrounds(LPARAM lParam1, LPARAM lParam2, LPARAM lParamSort)
  764. {
  765. TCHAR szFile1[MAX_PATH], szFile2[MAX_PATH];
  766. LPTSTR lpszFile1, lpszFile2;
  767. StrCpyN(szFile1, (LPTSTR)lParam1, ARRAYSIZE(szFile1));
  768. lpszFile1 = PathFindFileName(szFile1);
  769. PathRemoveExtension(lpszFile1);
  770. PathMakePretty(lpszFile1);
  771. StrCpyN(szFile2, (LPTSTR)lParam2, ARRAYSIZE(szFile2));
  772. lpszFile2 = PathFindFileName(szFile2);
  773. PathRemoveExtension(lpszFile2);
  774. PathMakePretty(lpszFile2);
  775. return StrCmpIC(lpszFile1, lpszFile2);
  776. }
  777. HRESULT CBackPropSheetPage::_GetPlus95ThemesDir(LPTSTR pszPath, DWORD cchSize)
  778. {
  779. DWORD dwType;
  780. DWORD cbSize = (sizeof(pszPath[0]) * cchSize);
  781. HRESULT hr = HrSHGetValue(HKEY_LOCAL_MACHINE, SZ_REGKEY_PLUS95DIR, SZ_REGVALUE_PLUS95DIR, &dwType, pszPath, &cbSize);
  782. if (SUCCEEDED(hr))
  783. {
  784. TCHAR szSubDir[MAX_PATH];
  785. LPTSTR pszFile = PathFindFileName(pszPath);
  786. if (pszFile)
  787. {
  788. // Plus!95 DestPath has "Plus!.dll" on the end so get rid of that.
  789. pszFile[0] = 0;
  790. }
  791. // Tack on a "Themes" onto the path
  792. LoadString(HINST_THISDLL, IDS_THEMES_SUBDIR, szSubDir, ARRAYSIZE(szSubDir));
  793. PathAppend(pszPath, szSubDir);
  794. }
  795. return hr;
  796. }
  797. HRESULT CBackPropSheetPage::_GetPlus98ThemesDir(LPTSTR pszPath, DWORD cchSize)
  798. {
  799. DWORD dwType;
  800. DWORD cbSize = (sizeof(pszPath[0]) * cchSize);
  801. HRESULT hr = HrSHGetValue(HKEY_LOCAL_MACHINE, SZ_REGKEY_PLUS98DIR, SZ_REGVALUE_PLUS98DIR, &dwType, pszPath, &cbSize);
  802. if (SUCCEEDED(hr))
  803. {
  804. TCHAR szSubDir[MAX_PATH];
  805. LoadString(HINST_THISDLL, IDS_THEMES_SUBDIR, szSubDir, ARRAYSIZE(szSubDir));
  806. PathAppend(pszPath, szSubDir);
  807. }
  808. return hr;
  809. }
  810. HRESULT CBackPropSheetPage::_GetKidsThemesDir(LPTSTR pszPath, DWORD cchSize)
  811. {
  812. DWORD dwType;
  813. DWORD cbSize = (sizeof(pszPath[0]) * cchSize);
  814. HRESULT hr = HrSHGetValue(HKEY_LOCAL_MACHINE, SZ_REGKEY_KIDSDIR, SZ_REGVALUE_KIDSDIR, &dwType, pszPath, &cbSize);
  815. if (SUCCEEDED(hr))
  816. {
  817. TCHAR szSubDir[MAX_PATH];
  818. // Tack a "\Plus! for Kids\Themes" onto the path
  819. PathAppend(pszPath, TEXT("Plus! for Kids"));
  820. LoadString(HINST_THISDLL, IDS_THEMES_SUBDIR, szSubDir, ARRAYSIZE(szSubDir));
  821. PathAppend(pszPath, szSubDir);
  822. }
  823. return hr;
  824. }
  825. HRESULT CBackPropSheetPage::_GetHardDirThemesDir(LPTSTR pszPath, DWORD cchSize)
  826. {
  827. DWORD dwType;
  828. DWORD cbSize = (sizeof(pszPath[0]) * cchSize);
  829. HRESULT hr = HrSHGetValue(HKEY_LOCAL_MACHINE, SZ_REGKEY_PROGRAMFILES, SZ_REGVALUE_PROGRAMFILESDIR, &dwType, pszPath, &cbSize);
  830. if (SUCCEEDED(hr))
  831. {
  832. TCHAR szSubDir[MAX_PATH];
  833. // Tack a "\Plus! for Kids\Themes" onto the path
  834. PathAppend(pszPath, TEXT("Plus!"));
  835. LoadString(HINST_THISDLL, IDS_THEMES_SUBDIR, szSubDir, ARRAYSIZE(szSubDir));
  836. PathAppend(pszPath, szSubDir);
  837. }
  838. return hr;
  839. }
  840. BOOL CBackPropSheetPage::_DoesDirHaveMoreThanMax(LPCTSTR pszPath, int nMax)
  841. {
  842. _nFileCount = 0;
  843. _nFileMax = nMax;
  844. _AddPicturesFromDirRecursively(pszPath, TRUE);
  845. return (nMax < _nFileCount);
  846. }
  847. #define MAX_PICTURES_TOSTOPRECURSION 100 // PERF: If the directory (My Pictures) has more than this many pictures, only add the pictures in the top level.
  848. HRESULT CBackPropSheetPage::_AddFilesToList(void)
  849. {
  850. HRESULT hr = S_OK;
  851. TCHAR szPath[MAX_PATH];
  852. // Get the directory with the wallpaper files.
  853. if (!GetStringFromReg(HKEY_LOCAL_MACHINE, c_szSetup, c_szSharedDir, NULL, szPath, ARRAYSIZE(szPath)))
  854. {
  855. if (!GetWindowsDirectory(szPath, ARRAYSIZE(szPath)))
  856. {
  857. szPath[0] = 0;
  858. }
  859. }
  860. // Add only the *.BMP files in the windows directory.
  861. _AddFilesToLV(szPath, c_rgpszWallpaperExt[0], 1, FALSE);
  862. // Get the wallpaper Directory name
  863. GetWallpaperDirName(szPath, ARRAYSIZE(szPath));
  864. // Add all pictures from Wallpaper directory to the list!
  865. _AddPicturesFromDir(szPath, FALSE);
  866. // Get the path to the "My Pictures" folder
  867. // NOTE: don't create the My Pictures directory -- if it doesn't exist, we won't find anything there anyway!
  868. if (S_OK == SHGetFolderPath(NULL, CSIDL_MYPICTURES, NULL, 0, szPath))
  869. {
  870. // Add all pictures in "My Pictures" directory to the list!
  871. if (!_DoesDirHaveMoreThanMax(szPath, MAX_PICTURES_TOSTOPRECURSION))
  872. {
  873. hr = _AddPicturesFromDirRecursively(szPath, FALSE);
  874. }
  875. }
  876. //Get the path to the common "%UserProfile%\Application Data\" folder
  877. if (S_OK == SHGetFolderPath(NULL, CSIDL_APPDATA, NULL, 0, szPath))
  878. {
  879. // Add all pictures in common "%UserProfile%\Application Data\Microsoft Internet Explorer\" so we get the user's
  880. // "Internet Explorer Wallpaper.bmp" file.
  881. PathAppend(szPath, TEXT("Microsoft\\Internet Explorer"));
  882. _AddPicturesFromDir(szPath, FALSE);
  883. }
  884. // Add pictures from Theme Directories
  885. // The follwoing directories can contain themes:
  886. // Plus!98 Install Path\Themes
  887. // Plus!95 Install Path\Themes
  888. // Kids for Plus! Install Path\Themes
  889. // Program Files\Plus!\Themes
  890. if (SUCCEEDED(_GetPlus98ThemesDir(szPath, ARRAYSIZE(szPath))))
  891. {
  892. _AddPicturesFromDirRecursively(szPath, FALSE);
  893. }
  894. else if (SUCCEEDED(_GetPlus95ThemesDir(szPath, ARRAYSIZE(szPath))))
  895. {
  896. _AddPicturesFromDirRecursively(szPath, FALSE);
  897. }
  898. else if (SUCCEEDED(_GetKidsThemesDir(szPath, ARRAYSIZE(szPath))))
  899. {
  900. _AddPicturesFromDirRecursively(szPath, FALSE);
  901. }
  902. else if (SUCCEEDED(_GetHardDirThemesDir(szPath, ARRAYSIZE(szPath))))
  903. {
  904. _AddPicturesFromDirRecursively(szPath, FALSE);
  905. }
  906. return hr;
  907. }
  908. #define SZ_ALL_FILTER TEXT("*.*")
  909. HRESULT CBackPropSheetPage::_AddPicturesFromDirRecursively(IN LPCTSTR pszDirName, BOOL fCount)
  910. {
  911. HRESULT hr = S_OK;
  912. WIN32_FIND_DATA findFileData;
  913. TCHAR szSearchPath[MAX_PATH];
  914. _AddPicturesFromDir(pszDirName, fCount);
  915. StrCpyN(szSearchPath, pszDirName, ARRAYSIZE(szSearchPath));
  916. PathAppend(szSearchPath, SZ_ALL_FILTER);
  917. HANDLE hFindFiles = FindFirstFile(szSearchPath, &findFileData);
  918. if (hFindFiles && (INVALID_HANDLE_VALUE != hFindFiles))
  919. {
  920. while (hFindFiles && (INVALID_HANDLE_VALUE != hFindFiles) &&
  921. !(fCount && (_nFileMax < _nFileCount)))
  922. {
  923. if ((FILE_ATTRIBUTE_DIRECTORY & findFileData.dwFileAttributes) &&
  924. !PathIsDotOrDotDot(findFileData.cFileName))
  925. {
  926. TCHAR szSubDir[MAX_PATH];
  927. StrCpyN(szSubDir, pszDirName, ARRAYSIZE(szSubDir));
  928. PathAppend(szSubDir, findFileData.cFileName);
  929. hr = _AddPicturesFromDirRecursively(szSubDir, fCount);
  930. }
  931. if (!FindNextFile(hFindFiles, &findFileData))
  932. {
  933. break;
  934. }
  935. }
  936. FindClose(hFindFiles);
  937. }
  938. return hr;
  939. }
  940. void CBackPropSheetPage::_AddPicturesFromDir(LPCTSTR pszDirName, BOOL fCount)
  941. {
  942. int i;
  943. // Add only the *.BMP files first (They can be shown even when Active Desktop is OFF)
  944. _AddFilesToLV(pszDirName, c_rgpszWallpaperExt[0], 1, fCount);
  945. if(_fAllowHtml)
  946. {
  947. // BMPs have already been added; So, start from the next.
  948. for (i=1; i<ARRAYSIZE(c_rgpszWallpaperExt); i++)
  949. {
  950. // The .htm extension includes the .html files. Hence the .html extension is skipped.
  951. // The .jpe extension includes the .jpeg files too. So, it .jpeg is skipped too!
  952. if((lstrcmpi(c_rgpszWallpaperExt[i],TEXT("HTML")) == 0) ||
  953. (lstrcmpi(c_rgpszWallpaperExt[i],TEXT("JPEG")) == 0))
  954. continue;
  955. _AddFilesToLV(pszDirName, c_rgpszWallpaperExt[i], i+1, fCount);
  956. }
  957. }
  958. }
  959. HRESULT GetActiveDesktop(IActiveDesktop ** ppActiveDesktop)
  960. {
  961. HRESULT hr = S_OK;
  962. if (!*ppActiveDesktop)
  963. {
  964. IActiveDesktopP * piadp;
  965. if (SUCCEEDED(hr = CActiveDesktop_InternalCreateInstance((LPUNKNOWN *)&piadp, IID_IActiveDesktopP)))
  966. {
  967. WCHAR wszScheme[MAX_PATH];
  968. DWORD dwcch = ARRAYSIZE(wszScheme);
  969. // Get the global "edit" scheme and set ourselves us to read from and edit that scheme
  970. if (SUCCEEDED(piadp->GetScheme(wszScheme, &dwcch, SCHEME_GLOBAL | SCHEME_EDIT)))
  971. {
  972. piadp->SetScheme(wszScheme, SCHEME_LOCAL);
  973. }
  974. hr = piadp->QueryInterface(IID_PPV_ARG(IActiveDesktop, ppActiveDesktop));
  975. piadp->Release();
  976. }
  977. }
  978. else
  979. {
  980. (*ppActiveDesktop)->AddRef();
  981. }
  982. return hr;
  983. }
  984. HRESULT ReleaseActiveDesktop(IActiveDesktop ** ppActiveDesktop)
  985. {
  986. HRESULT hr = S_OK;
  987. if (*ppActiveDesktop)
  988. {
  989. if((*ppActiveDesktop)->Release() == 0)
  990. *ppActiveDesktop = NULL;
  991. }
  992. return hr;
  993. }
  994. #define SZ_REGKEY_CONTROLPANEL_DESKTOP TEXT("Control Panel\\Desktop")
  995. #define SZ_REGVALUE_CP_PATTERN TEXT("pattern")
  996. #define SZ_REGVALUE_CP_PATTERNUPGRADE TEXT("Pattern Upgrade")
  997. #define SZ_REGVALUE_CONVERTED_WALLPAPER TEXT("ConvertedWallpaper")
  998. #define SZ_REGVALUE_ORIGINAL_WALLPAPER TEXT("OriginalWallpaper") // We store this to find when someone changed the wallpaper around us
  999. #define SZ_REGVALUE_CONVERTED_WP_LASTWRITE TEXT("ConvertedWallpaper Last WriteTime")
  1000. HRESULT CBackPropSheetPage::_LoadTempWallpaperSettings(IN LPCWSTR pszWallpaperFile)
  1001. {
  1002. // When we converted a non-.BMP wallpaper to a .bmp temp file,
  1003. // we keep the name of the original wallpaper path stored in _pszOriginalFile.
  1004. // We need to load that in now.
  1005. TCHAR szTempWallPaper[MAX_PATH];
  1006. DWORD dwType;
  1007. DWORD cbSize = sizeof(szTempWallPaper);
  1008. // ISSUE: CONVERTED and ORIGINAL are backwards, but we shipped beta1 like this so we can't change it... blech
  1009. DWORD dwError = SHGetValue(HKEY_CURRENT_USER, SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_CONVERTED_WALLPAPER, &dwType, (void *) szTempWallPaper, &cbSize);
  1010. HRESULT hr = HRESULT_FROM_WIN32(dwError);
  1011. if (SUCCEEDED(hr) && szTempWallPaper[0] && !_fWallpaperChanged)
  1012. {
  1013. TCHAR szOriginalWallPaper[MAX_PATH];
  1014. cbSize = sizeof(szOriginalWallPaper);
  1015. DWORD dwError = SHGetValue(HKEY_CURRENT_USER, SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_ORIGINAL_WALLPAPER, &dwType, (void *) szOriginalWallPaper, &cbSize);
  1016. // It's possible that someone changed the wallpaper externally (IE's "Set as Wallpaper").
  1017. // We need to detect this and ignore the converted wallpaper regkey (SZ_REGVALUE_CONVERTED_WALLPAPER).
  1018. if ((ERROR_SUCCESS == dwError) && !StrCmpI(szOriginalWallPaper, pszWallpaperFile))
  1019. {
  1020. Str_SetPtr(&_pszOriginalFile, szTempWallPaper);
  1021. Str_SetPtr(&_pszOrigLastApplied, szTempWallPaper);
  1022. }
  1023. }
  1024. Str_SetPtrW(&_pszWallpaperInUse, pszWallpaperFile);
  1025. cbSize = sizeof(_ftLastWrite);
  1026. dwError = SHGetValue(HKEY_CURRENT_USER, SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_CONVERTED_WP_LASTWRITE, &dwType, (void *) &_ftLastWrite, &cbSize);
  1027. return S_OK; // Ignore the hr because it's fine if the value wasn't found.
  1028. }
  1029. #define POLICY_DISABLECOLORCUSTOMIZATION_ON 0x00000001
  1030. HRESULT CBackPropSheetPage::_LoadBackgroundColor(IN BOOL fInit)
  1031. {
  1032. HRESULT hr = E_INVALIDARG;
  1033. if (_punkSite)
  1034. {
  1035. IPropertyBag * pPropertyBag;
  1036. hr = _punkSite->QueryInterface(IID_PPV_ARG(IPropertyBag, &pPropertyBag));
  1037. if (SUCCEEDED(hr))
  1038. {
  1039. _rgbBkgdColor = 0x00000000;
  1040. hr = SHPropertyBag_ReadDWORD(pPropertyBag, SZ_PBPROP_BACKGROUND_COLOR, &_rgbBkgdColor);
  1041. if (fInit)
  1042. {
  1043. // Check the policy.
  1044. if (POLICY_DISABLECOLORCUSTOMIZATION_ON == SHRestricted(REST_NODISPLAYAPPEARANCEPAGE))
  1045. {
  1046. // We need to disable and hide the windows. We need to disable them so they can't get
  1047. // focus or accessibility.
  1048. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_COLORPICKER), FALSE);
  1049. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_COLORPICKERLABEL), FALSE);
  1050. ShowWindow(GetDlgItem(_hwnd, IDC_BACK_COLORPICKER), SW_HIDE);
  1051. ShowWindow(GetDlgItem(_hwnd, IDC_BACK_COLORPICKERLABEL), SW_HIDE);
  1052. }
  1053. else
  1054. {
  1055. _colorControl.Initialize(GetDlgItem(_hwnd, IDC_BACK_COLORPICKER), _rgbBkgdColor);
  1056. }
  1057. }
  1058. else
  1059. {
  1060. _colorControl.SetColor(_rgbBkgdColor);
  1061. }
  1062. pPropertyBag->Release();
  1063. }
  1064. }
  1065. return hr;
  1066. }
  1067. HRESULT CBackPropSheetPage::_Initialize(void)
  1068. {
  1069. HRESULT hr = S_OK;
  1070. if (!_fInitialized && g_pActiveDesk)
  1071. {
  1072. WCHAR szPath[MAX_PATH];
  1073. // Add & select the current setting.
  1074. hr = g_pActiveDesk->GetWallpaper(szPath, ARRAYSIZE(szPath), 0);
  1075. if (SUCCEEDED(hr))
  1076. {
  1077. hr = _LoadTempWallpaperSettings(szPath);
  1078. _fInitialized = TRUE;
  1079. }
  1080. }
  1081. return hr;
  1082. }
  1083. void CBackPropSheetPage::_OnInitDialog(HWND hwnd)
  1084. {
  1085. int i;
  1086. TCHAR szBuf[MAX_PATH];
  1087. _Initialize();
  1088. _colorControl.ChangeTheme(hwnd);
  1089. // Upgrade the pattern setting. Since we got rid of the UI, we want to
  1090. // get rid of the setting on upgrade. We only want to do this once since
  1091. // if the user added it back, we don't want to redelete it.
  1092. if (FALSE == SHRegGetBoolUSValue(SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_CP_PATTERNUPGRADE, FALSE, FALSE))
  1093. {
  1094. SHDeleteValue(HKEY_CURRENT_USER, SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_CP_PATTERN);
  1095. SHSetValue(HKEY_CURRENT_USER, SZ_REGKEY_CONTROLPANEL_DESKTOP, SZ_REGVALUE_CP_PATTERNUPGRADE, REG_SZ, TEXT("TRUE"), ((lstrlen(TEXT("TRUE")) + 1) * sizeof(TCHAR)));
  1096. }
  1097. //
  1098. // Set some member variables.
  1099. //
  1100. _hwnd = hwnd;
  1101. _hwndLV = GetDlgItem(hwnd, IDC_BACK_WPLIST);
  1102. _hwndWPStyle = GetDlgItem(hwnd, IDC_BACK_WPSTYLE);
  1103. HWND hWndPrev = GetDlgItem(hwnd, IDC_BACK_PREVIEW);
  1104. if (hWndPrev)
  1105. {
  1106. SetWindowBits(hWndPrev, GWL_EXSTYLE, RTL_MIRRORED_WINDOW, 0);
  1107. }
  1108. _UpdatePreview(0, FALSE);
  1109. InitDeskHtmlGlobals();
  1110. if (!g_pActiveDesk)
  1111. {
  1112. return;
  1113. }
  1114. //
  1115. // Read in the restrictions.
  1116. //
  1117. _fForceAD = SHRestricted(REST_FORCEACTIVEDESKTOPON);
  1118. _fAllowAD = _fForceAD || !PolicyNoActiveDesktop();
  1119. if (_fAllowAD == FALSE)
  1120. {
  1121. _fAllowHtml = FALSE;
  1122. }
  1123. else
  1124. {
  1125. _fAllowHtml = !SHRestricted(REST_NOHTMLWALLPAPER);
  1126. }
  1127. //
  1128. // Check to see if there is a policy for Wallpaper name and wallpaper style.
  1129. //
  1130. _fPolicyForWallpaper = ReadPolicyForWallpaper(NULL, 0);
  1131. _fPolicyForStyle = ReadPolicyForWPStyle(NULL);
  1132. //
  1133. // Get the images into the listview.
  1134. //
  1135. HIMAGELIST himl = ImageList_Create(GetSystemMetrics(SM_CXSMICON),
  1136. GetSystemMetrics(SM_CYSMICON), ILC_MASK | ILC_COLOR32, ARRAYSIZE(c_rgpszWallpaperExt),
  1137. ARRAYSIZE(c_rgpszWallpaperExt));
  1138. if (himl)
  1139. {
  1140. SHFILEINFO sfi;
  1141. // Add the 'None' icon.
  1142. HICON hIconNone = (HICON)LoadImage(HINST_THISDLL, MAKEINTRESOURCE(IDI_BACK_NONE),
  1143. IMAGE_ICON, GetSystemMetrics(SM_CXSMICON),
  1144. GetSystemMetrics(SM_CYSMICON), 0);
  1145. ImageList_AddIcon(himl, hIconNone);
  1146. int iPrefixLen = lstrlen(TEXT("foo."));
  1147. StrCpyN(szBuf, TEXT("foo."), ARRAYSIZE(szBuf)); //Pass "foo.bmp" etc., to SHGetFileInfo instead of ".bmp"
  1148. for (i=0; i<ARRAYSIZE(c_rgpszWallpaperExt); i++)
  1149. {
  1150. StrCpyN(szBuf+iPrefixLen, c_rgpszWallpaperExt[i], ARRAYSIZE(szBuf) - iPrefixLen);
  1151. if (SHGetFileInfo(szBuf, 0, &sfi, SIZEOF(sfi), SHGFI_ICON | SHGFI_SMALLICON | SHGFI_USEFILEATTRIBUTES))
  1152. {
  1153. ImageList_AddIcon(himl, sfi.hIcon);
  1154. DestroyIcon(sfi.hIcon);
  1155. }
  1156. }
  1157. ListView_SetImageList(_hwndLV, himl, LVSIL_SMALL);
  1158. }
  1159. // Add the single column that we want.
  1160. LV_COLUMN lvc;
  1161. lvc.mask = LVCF_FMT | LVCF_SUBITEM;
  1162. lvc.fmt = LVCFMT_LEFT;
  1163. lvc.iSubItem = 0;
  1164. ListView_InsertColumn(_hwndLV, 0, &lvc);
  1165. // Add 'none' option.
  1166. _AddAFileToLV(NULL, g_szNone, 0);
  1167. // Add the rest of the files.
  1168. _AddFilesToList();
  1169. // Sort the standard items.
  1170. ListView_SortItems(_hwndLV, _SortBackgrounds, 0);
  1171. WCHAR wszBuf[MAX_PATH];
  1172. LPTSTR pszBuf;
  1173. // Add & select the current setting.
  1174. g_pActiveDesk->GetWallpaper(wszBuf, ARRAYSIZE(wszBuf), 0);
  1175. //Convert wszBuf to szBuf.
  1176. pszBuf = (LPTSTR)wszBuf;
  1177. if (!_fAllowHtml && !IsNormalWallpaper(pszBuf))
  1178. {
  1179. *pszBuf = TEXT('\0');
  1180. }
  1181. _SetNewWallpaper(pszBuf, FALSE);
  1182. int iEndStyle = WPSTYLE_STRETCH;
  1183. for (i=0; i<= iEndStyle; i++)
  1184. {
  1185. LoadString(HINST_THISDLL, IDS_WPSTYLE+i, szBuf, ARRAYSIZE(szBuf));
  1186. ComboBox_AddString(_hwndWPStyle, szBuf);
  1187. }
  1188. WALLPAPEROPT wpo;
  1189. wpo.dwSize = sizeof(wpo);
  1190. g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  1191. ComboBox_SetCurSel(_hwndWPStyle, wpo.dwStyle);
  1192. // Adjust various UI components.
  1193. if (!_fAllowChanges)
  1194. {
  1195. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_DISPLAY), FALSE);
  1196. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_WPSTYLE), FALSE);
  1197. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_BROWSE), FALSE);
  1198. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_WPLIST), FALSE);
  1199. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_SELECT), FALSE);
  1200. }
  1201. // Disable controls based on the policies
  1202. if(_fPolicyForWallpaper)
  1203. {
  1204. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_BROWSE), FALSE);
  1205. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_WPLIST), FALSE);
  1206. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_SELECT), FALSE);
  1207. }
  1208. if(_fPolicyForStyle)
  1209. {
  1210. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_WPSTYLE), FALSE);
  1211. EnableWindow(GetDlgItem(_hwnd, IDC_BACK_DISPLAY), FALSE);
  1212. }
  1213. COMPONENTSOPT co;
  1214. co.dwSize = sizeof(COMPONENTSOPT);
  1215. g_pActiveDesk->GetDesktopItemOptions(&co, 0);
  1216. //if activedesktop is forced to be on, this overrides the NOACTIVEDESKTOP restriction
  1217. if (_fForceAD)
  1218. {
  1219. if(!co.fActiveDesktop)
  1220. {
  1221. co.fActiveDesktop = TRUE;
  1222. g_pActiveDesk->SetDesktopItemOptions(&co, 0);
  1223. }
  1224. }
  1225. else
  1226. {
  1227. //See if Active Desktop is to be turned off by restriction.
  1228. if (!_fAllowAD)
  1229. {
  1230. if (co.fActiveDesktop)
  1231. {
  1232. co.fActiveDesktop = FALSE;
  1233. g_pActiveDesk->SetDesktopItemOptions(&co, 0);
  1234. }
  1235. }
  1236. }
  1237. _EnableControls();
  1238. _LoadBackgroundColor(TRUE);
  1239. if (_fOpenAdvOnInit)
  1240. {
  1241. // Tell the Advanced dialog to open.
  1242. PostMessage(_hwnd, WM_COMMAND, (WPARAM)IDC_BACK_WEB, (LPARAM)GetDlgItem(_hwnd, IDC_BACK_WEB));
  1243. }
  1244. _StartSizeChecker();
  1245. }
  1246. // This function checks to see if the currently selected wallpaper is a HTML wallpaper
  1247. // and if so, it makes sure that the active desktop is enabled. If it is disabled
  1248. // then it prompts the user asking a question to see if the user wants to enable it.
  1249. //
  1250. void EnableADifHtmlWallpaper(HWND hwnd)
  1251. {
  1252. if(!g_pActiveDesk)
  1253. {
  1254. return;
  1255. }
  1256. // turn active desktop on or off, depending on background
  1257. ActiveDesktop_ApplyChanges();
  1258. }
  1259. void CBackPropSheetPage::_OnNotify(LPNMHDR lpnm)
  1260. {
  1261. //
  1262. // Start with a small stack allocation.
  1263. //
  1264. WCHAR wszBuf[MAX_PATH];
  1265. LPWSTR pszBuf = wszBuf;
  1266. DWORD cchBuf = ARRAYSIZE(wszBuf);
  1267. switch (lpnm->code)
  1268. {
  1269. case PSN_SETACTIVE:
  1270. {
  1271. HRESULT hr;
  1272. //
  1273. // Make sure the correct wallpaper is selected.
  1274. //
  1275. hr = g_pActiveDesk->GetWallpaper(pszBuf, cchBuf, 0);
  1276. if (FAILED(hr))
  1277. break;
  1278. //
  1279. // If we need more room, allocate it on the heap and try again.
  1280. //
  1281. if ( MAKE_HRESULT( 0, FACILITY_WIN32, ERROR_MORE_DATA ) == hr )
  1282. {
  1283. cchBuf = INTERNET_MAX_URL_LENGTH;
  1284. pszBuf = (LPWSTR) LocalAlloc( LPTR, cchBuf * sizeof(WCHAR) );
  1285. if ( NULL == pszBuf )
  1286. break;
  1287. hr = g_pActiveDesk->GetWallpaper(pszBuf, cchBuf, 0);
  1288. if (S_OK != hr)
  1289. break;
  1290. }
  1291. _LoadBackgroundColor(FALSE);
  1292. _SetNewWallpaper(pszBuf, FALSE);
  1293. if (g_pActiveDesk)
  1294. {
  1295. WALLPAPEROPT wpo;
  1296. wpo.dwSize = sizeof(wpo);
  1297. g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  1298. ComboBox_SetCurSel(_hwndWPStyle, wpo.dwStyle);
  1299. }
  1300. }
  1301. break;
  1302. case PSN_APPLY:
  1303. _OnApply();
  1304. break;
  1305. case LVN_ITEMCHANGED:
  1306. NM_LISTVIEW *pnmlv = (NM_LISTVIEW *)lpnm;
  1307. if ((pnmlv->uChanged & LVIF_STATE) &&
  1308. (pnmlv->uNewState & LVIS_SELECTED))
  1309. {
  1310. LV_ITEM lvi = {0};
  1311. lvi.iItem = pnmlv->iItem;
  1312. lvi.mask = LVIF_PARAM;
  1313. ListView_GetItem(_hwndLV, &lvi);
  1314. LPCTSTR pszSelectedNew = (LPCTSTR)lvi.lParam;
  1315. //
  1316. // Make sure the correct wallpaper is selected.
  1317. //
  1318. HRESULT hr = g_pActiveDesk->GetWallpaper(pszBuf, cchBuf, 0);
  1319. if (FAILED(hr))
  1320. break;
  1321. //
  1322. // If we need more room, allocate it on the heap and try again.
  1323. //
  1324. if ( MAKE_HRESULT( 0, FACILITY_WIN32, ERROR_MORE_DATA ) == hr )
  1325. {
  1326. cchBuf = INTERNET_MAX_URL_LENGTH;
  1327. pszBuf = (LPWSTR) LocalAlloc( LPTR, cchBuf * sizeof(WCHAR) );
  1328. if ( NULL == pszBuf )
  1329. break;
  1330. hr = g_pActiveDesk->GetWallpaper(pszBuf, cchBuf, 0);
  1331. if (S_OK != hr)
  1332. break;
  1333. }
  1334. if (lstrcmp(pszSelectedNew, pszBuf) != 0)
  1335. {
  1336. _SetNewWallpaper(pszSelectedNew, _fSelectionFromUser);
  1337. EnableApplyButton(_hwnd);
  1338. }
  1339. }
  1340. break;
  1341. }
  1342. //
  1343. // Free heap allocation (if any)
  1344. //
  1345. if ( wszBuf != pszBuf && NULL != pszBuf )
  1346. {
  1347. LocalFree( pszBuf );
  1348. }
  1349. }
  1350. #define MAX_PAGES 12
  1351. // Parameters:
  1352. // hwnd: We parent our UI on this hwnd.
  1353. // dwPage: Which page does the caller want to come up as default?
  1354. // Use ADP_DEFAULT if they don't care
  1355. // Return: S_OK if the user closed the dialog with OK.
  1356. // HRESULT_FROM_WIN32(ERROR_CANCELLED) will be returned
  1357. // if the user clicked the cancel button.
  1358. HRESULT CBackPropSheetPage::_LaunchAdvancedDisplayProperties(HWND hwnd)
  1359. {
  1360. HRESULT hr = E_OUTOFMEMORY;
  1361. CCompPropSheetPage * pWebDialog = new CCompPropSheetPage();
  1362. if (pWebDialog)
  1363. {
  1364. BOOL fEnableApply = FALSE;
  1365. IUnknown_SetSite(SAFECAST(pWebDialog, IObjectWithSite *), SAFECAST(this, IObjectWithSite *));
  1366. hr = pWebDialog->DisplayAdvancedDialog(_hwnd, SAFECAST(this, IPropertyBag *), &fEnableApply);
  1367. if (SUCCEEDED(hr) && (fEnableApply || _fHideDesktopIconDirty))
  1368. {
  1369. EnableApplyButton(_hwnd);
  1370. }
  1371. IUnknown_SetSite(SAFECAST(pWebDialog, IObjectWithSite *), NULL);
  1372. pWebDialog->Release();
  1373. }
  1374. return hr;
  1375. }
  1376. void CBackPropSheetPage::_OnCommand(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
  1377. {
  1378. WORD wNotifyCode = HIWORD(wParam);
  1379. WORD wID = LOWORD(wParam);
  1380. HWND hwndCtl = (HWND)lParam;
  1381. switch (wID)
  1382. {
  1383. case IDC_BACK_WPSTYLE:
  1384. switch (wNotifyCode)
  1385. {
  1386. case CBN_SELCHANGE:
  1387. WALLPAPEROPT wpo;
  1388. wpo.dwSize = sizeof(WALLPAPEROPT);
  1389. g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  1390. wpo.dwStyle = ComboBox_GetCurSel(_hwndWPStyle);
  1391. g_pActiveDesk->SetWallpaperOptions(&wpo, 0);
  1392. _EnableControls();
  1393. _UpdatePreview(0, TRUE);
  1394. EnableApplyButton(_hwnd);
  1395. break;
  1396. }
  1397. break;
  1398. case IDC_BACK_BROWSE:
  1399. _BrowseForBackground();
  1400. break;
  1401. case IDC_BACK_WEB:
  1402. _LaunchAdvancedDisplayProperties(_hwnd);
  1403. break;
  1404. case IDC_BACK_COLORPICKER:
  1405. {
  1406. COLORREF rbgColorNew;
  1407. if (SUCCEEDED(_colorControl.OnCommand(hdlg, uMsg, wParam, lParam)) &&
  1408. SUCCEEDED(_colorControl.GetColor(&rbgColorNew)) &&
  1409. (rbgColorNew != _rgbBkgdColor))
  1410. {
  1411. _rgbBkgdColor = rbgColorNew;
  1412. if (_punkSite)
  1413. {
  1414. IPropertyBag * pPropertyBag;
  1415. if (SUCCEEDED(_punkSite->QueryInterface(IID_PPV_ARG(IPropertyBag, &pPropertyBag))))
  1416. {
  1417. SHPropertyBag_WriteDWORD(pPropertyBag, SZ_PBPROP_BACKGROUND_COLOR, _rgbBkgdColor);
  1418. pPropertyBag->Release();
  1419. }
  1420. }
  1421. EnableApplyButton(_hwnd);
  1422. _UpdatePreview(0, TRUE);
  1423. }
  1424. }
  1425. break;
  1426. }
  1427. }
  1428. HRESULT CBackPropSheetPage::_BrowseForBackground(void)
  1429. {
  1430. HRESULT hr;
  1431. WCHAR wszFileName[INTERNET_MAX_URL_LENGTH];
  1432. hr = g_pActiveDesk->GetWallpaper(wszFileName, ARRAYSIZE(wszFileName), 0);
  1433. if (SUCCEEDED(hr))
  1434. {
  1435. WCHAR szTestPath[MAX_PATH];
  1436. WCHAR szPath[MAX_PATH];
  1437. StrCpyNW(szPath, wszFileName, ARRAYSIZE(szPath));
  1438. PathRemoveFileSpec(szPath);
  1439. if (GetWindowsDirectory(szTestPath, ARRAYSIZE(szTestPath)) &&
  1440. !StrCmpIW(szTestPath, szPath))
  1441. {
  1442. // If the directory is equal to the boring %windir%, then use "My Pictures" instead.
  1443. hr = (SHGetSpecialFolderPath(NULL, wszFileName, CSIDL_MYPICTURES, FALSE) ? S_OK : E_FAIL);
  1444. lstrcat(wszFileName, TEXT("\\"));
  1445. }
  1446. else if (GetSystemDirectory(szTestPath, ARRAYSIZE(szTestPath)) &&
  1447. !StrCmpIW(szTestPath, szPath))
  1448. {
  1449. // If the directory is equal to the boring %windir%\system32, then use "My Pictures" instead.
  1450. hr = (SHGetSpecialFolderPath(NULL, wszFileName, CSIDL_MYPICTURES, FALSE) ? S_OK : E_FAIL);
  1451. lstrcat(wszFileName, TEXT("\\"));
  1452. }
  1453. else if (GetWindowsDirectory(szTestPath, ARRAYSIZE(szTestPath)))
  1454. {
  1455. PathAppendW(szTestPath, L"Web\\Wallpaper");
  1456. if (!StrCmpIW(szTestPath, szPath))
  1457. {
  1458. // If the directory is equal to the boring %windir%\web\wallpaper, then use "My Pictures" instead.
  1459. hr = (SHGetSpecialFolderPath(NULL, wszFileName, CSIDL_MYPICTURES, FALSE) ? S_OK : E_FAIL);
  1460. lstrcat(wszFileName, TEXT("\\"));
  1461. }
  1462. }
  1463. }
  1464. else
  1465. {
  1466. hr = (SHGetSpecialFolderPath(NULL, wszFileName, CSIDL_MYPICTURES, TRUE) ? S_OK : E_FAIL);
  1467. lstrcat(wszFileName, TEXT("\\"));
  1468. }
  1469. if (SUCCEEDED(hr))
  1470. {
  1471. DWORD adwFlags[] = { GFN_PICTURE, GFN_PICTURE, 0, 0};
  1472. int aiTypes[] = { IDS_BACK_FILETYPES, IDS_ALL_PICTURES, 0, 0};
  1473. if (_fAllowHtml)
  1474. {
  1475. SetFlag(adwFlags[0], GFN_LOCALHTM);
  1476. SetFlag(adwFlags[2], GFN_LOCALHTM);
  1477. aiTypes[2] = IDS_HTMLDOC;
  1478. }
  1479. if (wszFileName[0] == TEXT('\0'))
  1480. {
  1481. if (!GetWindowsDirectory(wszFileName, ARRAYSIZE(wszFileName)))
  1482. {
  1483. wszFileName[0] = 0;
  1484. }
  1485. // GetFileName breaks up the string into a directory and file
  1486. // component, so we append a slash to make sure everything
  1487. // is considered part of the directory.
  1488. lstrcat(wszFileName, TEXT("\\"));
  1489. }
  1490. if (GetFileName(_hwnd, wszFileName, ARRAYSIZE(wszFileName), aiTypes, adwFlags) &&
  1491. ValidateFileName(_hwnd, wszFileName, IDS_BACK_TYPE1))
  1492. {
  1493. if (_fAllowHtml || IsNormalWallpaper(wszFileName))
  1494. {
  1495. _SetNewWallpaper(wszFileName, TRUE);
  1496. EnableApplyButton(_hwnd);
  1497. }
  1498. }
  1499. }
  1500. return hr;
  1501. }
  1502. void CBackPropSheetPage::_OnDestroy()
  1503. {
  1504. }
  1505. INT_PTR CALLBACK CBackPropSheetPage::BackgroundDlgProc(HWND hDlg, UINT wMsg, WPARAM wParam, LPARAM lParam)
  1506. {
  1507. CBackPropSheetPage * pThis = (CBackPropSheetPage *)GetWindowLongPtr(hDlg, DWLP_USER);
  1508. if (WM_INITDIALOG == wMsg)
  1509. {
  1510. PROPSHEETPAGE * pPropSheetPage = (PROPSHEETPAGE *) lParam;
  1511. if (pPropSheetPage)
  1512. {
  1513. SetWindowLongPtr(hDlg, DWLP_USER, pPropSheetPage->lParam);
  1514. pThis = (CBackPropSheetPage *)pPropSheetPage->lParam;
  1515. }
  1516. }
  1517. if (pThis)
  1518. return pThis->_BackgroundDlgProc(hDlg, wMsg, wParam, lParam);
  1519. return DefWindowProc(hDlg, wMsg, wParam, lParam);
  1520. }
  1521. BOOL_PTR CBackPropSheetPage::_BackgroundDlgProc(HWND hdlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
  1522. {
  1523. switch(uMsg)
  1524. {
  1525. case WM_INITDIALOG:
  1526. _OnInitDialog(hdlg);
  1527. break;
  1528. case WM_NOTIFY:
  1529. _OnNotify((LPNMHDR)lParam);
  1530. break;
  1531. case WM_COMMAND:
  1532. _OnCommand(hdlg, uMsg, wParam, lParam);
  1533. break;
  1534. case WM_SYSCOLORCHANGE:
  1535. case WM_SETTINGCHANGE:
  1536. case WM_DISPLAYCHANGE:
  1537. case WM_QUERYNEWPALETTE:
  1538. case WM_PALETTECHANGED:
  1539. SHPropagateMessage(hdlg, uMsg, wParam, lParam, TRUE);
  1540. break;
  1541. case WM_HELP:
  1542. WinHelp((HWND)((LPHELPINFO) lParam)->hItemHandle, SZ_HELPFILE_DESKTOPTAB, HELP_WM_HELP, (ULONG_PTR)aBackHelpIDs);
  1543. break;
  1544. case WM_CONTEXTMENU:
  1545. WinHelp((HWND) wParam, SZ_HELPFILE_DESKTOPTAB, HELP_CONTEXTMENU, (ULONG_PTR)(LPVOID) aBackHelpIDs);
  1546. break;
  1547. case WM_DESTROY:
  1548. {
  1549. TCHAR szFileName[MAX_PATH];
  1550. //Delete the tempoaray HTX file created for non-HTML wallpaper preview.
  1551. GetTempPath(ARRAYSIZE(szFileName), szFileName);
  1552. lstrcatn(szFileName, PREVIEW_PICTURE_FILENAME, ARRAYSIZE(szFileName));
  1553. DeleteFile(szFileName);
  1554. _OnDestroy();
  1555. }
  1556. break;
  1557. case WM_DRAWITEM:
  1558. switch (wParam)
  1559. {
  1560. case IDC_BACK_COLORPICKER:
  1561. _colorControl.OnDrawItem(hdlg, uMsg, wParam, lParam);
  1562. return TRUE;
  1563. }
  1564. break;
  1565. case WM_THEMECHANGED:
  1566. _colorControl.ChangeTheme(hdlg);
  1567. break;
  1568. }
  1569. return FALSE;
  1570. }
  1571. HRESULT CBackPropSheetPage::_OnApply(void)
  1572. {
  1573. // Our parent dialog will be notified of the Apply event and will call our
  1574. // IBasePropPage::OnApply() to do the real work.
  1575. return S_OK;
  1576. }
  1577. /*****************************************************************************\
  1578. DESCRIPTION:
  1579. This function will start a background thread which will make sure our
  1580. MRU of pictures in the Wallpaper list have been checked. They will be checked
  1581. to verify that we know their width & height. This way, when we choose a wallpaper, we can quickly
  1582. see if it should use "Stretch" or "Tile". We want to tile if they are very
  1583. small, indicating that they are probably watermark in nature.
  1584. \*****************************************************************************/
  1585. #define TIME_SCANFREQUENCY (8 * 60 /*Secs*/) // We don't want to rescan the files more than once per 8 minutes
  1586. HRESULT CBackPropSheetPage::_StartSizeChecker(void)
  1587. {
  1588. HRESULT hr = S_OK;
  1589. BOOL fSkipCheck = FALSE;
  1590. DWORD dwType;
  1591. FILETIME ftLastScan;
  1592. FILETIME ftCurrentTime;
  1593. SYSTEMTIME stCurrentTime;
  1594. DWORD cbSize = sizeof(ftLastScan);
  1595. GetSystemTime(&stCurrentTime);
  1596. SystemTimeToFileTime(&stCurrentTime, &ftCurrentTime);
  1597. // If we have recently scanned, skip the scan.
  1598. if ((ERROR_SUCCESS == SHGetValue(HKEY_CURRENT_USER, SZ_REGKEY_WALLPAPER, SZ_REGVALUE_LASTSCAN, &dwType, (LPBYTE) &ftLastScan, &cbSize)) &&
  1599. (REG_BINARY == dwType) &&
  1600. (sizeof(ftLastScan) == cbSize))
  1601. {
  1602. ULARGE_INTEGER * pulLastScan = (ULARGE_INTEGER *)&ftLastScan;
  1603. ULARGE_INTEGER * pulCurrent = (ULARGE_INTEGER *)&ftCurrentTime;
  1604. ULARGE_INTEGER ulDelta;
  1605. ulDelta.QuadPart = (pulCurrent->QuadPart - pulLastScan->QuadPart);
  1606. ulDelta.QuadPart /= 10000000; // units in a second
  1607. if (ulDelta.QuadPart < TIME_SCANFREQUENCY)
  1608. {
  1609. fSkipCheck = TRUE;
  1610. }
  1611. }
  1612. if (!fSkipCheck)
  1613. {
  1614. AddRef();
  1615. if (!SHCreateThread(CBackPropSheetPage::SizeCheckerThreadProc, this, (CTF_COINIT | CTF_FREELIBANDEXIT | CTF_PROCESS_REF), NULL))
  1616. {
  1617. hr = E_FAIL;
  1618. Release();
  1619. }
  1620. SHSetValue(HKEY_CURRENT_USER, SZ_REGKEY_WALLPAPER, SZ_REGVALUE_LASTSCAN, REG_BINARY, (LPCBYTE) &ftCurrentTime, sizeof(ftLastScan));
  1621. }
  1622. return hr;
  1623. }
  1624. typedef struct
  1625. {
  1626. TCHAR szPath[MAX_PATH];
  1627. DWORD dwSizeX;
  1628. DWORD dwSizeY;
  1629. FILETIME ftLastModified;
  1630. } WALLPAPERSIZE_STRUCT;
  1631. #define MAX_WALLPAPERSIZEMRU 500
  1632. HRESULT CBackPropSheetPage::_GetMRUObject(IMruDataList ** ppSizeMRU)
  1633. {
  1634. HRESULT hr = CoCreateInstance(CLSID_MruLongList, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IMruDataList, ppSizeMRU));
  1635. if (SUCCEEDED(hr))
  1636. {
  1637. hr = (*ppSizeMRU)->InitData(MAX_WALLPAPERSIZEMRU, MRULISTF_USE_STRCMPIW, HKEY_CURRENT_USER, SZ_REGKEY_WALLPAPERMRU, NULL);
  1638. if (FAILED(hr))
  1639. {
  1640. ATOMICRELEASE(*ppSizeMRU);
  1641. }
  1642. }
  1643. return hr;
  1644. }
  1645. BOOL IsGraphicsFile(LPCTSTR pszFile)
  1646. {
  1647. BOOL fIsGraphicsFile = FALSE;
  1648. if (pszFile && *pszFile)
  1649. {
  1650. LPCTSTR pszExt = PathFindExtension(pszFile);
  1651. if (*pszExt == TEXT('.'))
  1652. {
  1653. pszExt++;
  1654. for (int iRet = 0; iRet < ARRAYSIZE(c_rgpszWallpaperExt); iRet++)
  1655. {
  1656. if (StrCmpIC(pszExt, c_rgpszWallpaperExt[iRet]) == 0)
  1657. {
  1658. fIsGraphicsFile = TRUE;
  1659. }
  1660. }
  1661. }
  1662. }
  1663. return fIsGraphicsFile;
  1664. }
  1665. HRESULT CBackPropSheetPage::_CalcSizeForFile(IN LPCTSTR pszPath, IN WIN32_FIND_DATA * pfdFile, IN OUT DWORD * pdwAdded)
  1666. {
  1667. HRESULT hr = S_OK;
  1668. if (!_pSizeMRUBk)
  1669. {
  1670. hr = _GetMRUObject(&_pSizeMRUBk);
  1671. }
  1672. if (SUCCEEDED(hr))
  1673. {
  1674. WALLPAPERSIZE_STRUCT wallpaperSize;
  1675. int nIndex;
  1676. StrCpyNW(wallpaperSize.szPath, pszPath, ARRAYSIZE(wallpaperSize.szPath));
  1677. // Let's see if it's already in the MRU.
  1678. hr = _pSizeMRUBk->FindData((LPCBYTE) &wallpaperSize, sizeof(wallpaperSize), &nIndex);
  1679. if (SUCCEEDED(hr))
  1680. {
  1681. // If so, let's see if it's been modified since.
  1682. hr = _pSizeMRUBk->GetData(nIndex, (LPBYTE) &wallpaperSize, sizeof(wallpaperSize));
  1683. if (SUCCEEDED(hr))
  1684. {
  1685. if (CompareFileTime(&wallpaperSize.ftLastModified, &pfdFile->ftLastWriteTime))
  1686. {
  1687. // We want to delete this index because it would colide with the name of
  1688. // the new entry we are going to add below.
  1689. _pSizeMRUBk->Delete(nIndex);
  1690. hr = E_FAIL; // We need to rescan.
  1691. }
  1692. else
  1693. {
  1694. (*pdwAdded)++; // We are only going to check the first 500 files. We don't want to waste too much time on this heuristical feature.
  1695. }
  1696. }
  1697. }
  1698. if (FAILED(hr))
  1699. {
  1700. (*pdwAdded)++; // We are only going to check the first 500 files. We don't want to waste too much time on this heuristical feature.
  1701. hr = S_OK;
  1702. // We didn't find it so we need to add it.
  1703. if (!_pImgFactBk)
  1704. {
  1705. hr = CoCreateInstance(CLSID_ShellImageDataFactory, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IShellImageDataFactory, &_pImgFactBk));
  1706. }
  1707. if (SUCCEEDED(hr))
  1708. {
  1709. IShellImageData* pImage;
  1710. hr = _pImgFactBk->CreateImageFromFile(pszPath, &pImage);
  1711. if (SUCCEEDED(hr))
  1712. {
  1713. hr = pImage->Decode(SHIMGDEC_DEFAULT, 0, 0);
  1714. if (SUCCEEDED(hr))
  1715. {
  1716. SIZE size;
  1717. hr = pImage->GetSize(&size);
  1718. if (SUCCEEDED(hr) && size.cx && size.cy)
  1719. {
  1720. DWORD dwSlot = 0;
  1721. StrCpyNW(wallpaperSize.szPath, pszPath, ARRAYSIZE(wallpaperSize.szPath));
  1722. wallpaperSize.dwSizeX = size.cx;
  1723. wallpaperSize.dwSizeY = size.cy;
  1724. wallpaperSize.ftLastModified = pfdFile->ftLastWriteTime;
  1725. hr = _pSizeMRUBk->AddData((LPCBYTE) &wallpaperSize, sizeof(wallpaperSize), &dwSlot);
  1726. }
  1727. }
  1728. pImage->Release();
  1729. }
  1730. }
  1731. }
  1732. }
  1733. return hr;
  1734. }
  1735. HRESULT CBackPropSheetPage::_CalcSizeFromDir(IN LPCTSTR pszPath, IN OUT DWORD * pdwAdded, IN BOOL fRecursive)
  1736. {
  1737. HRESULT hr = S_OK;
  1738. if (MAX_WALLPAPERSIZEMRU > *pdwAdded)
  1739. {
  1740. WIN32_FIND_DATA findFileData;
  1741. TCHAR szSearchPath[MAX_PATH];
  1742. StrCpyN(szSearchPath, pszPath, ARRAYSIZE(szSearchPath));
  1743. PathAppend(szSearchPath, SZ_ALL_FILTER);
  1744. HANDLE hFindFiles = FindFirstFile(szSearchPath, &findFileData);
  1745. if (hFindFiles && (INVALID_HANDLE_VALUE != hFindFiles))
  1746. {
  1747. while (hFindFiles && (INVALID_HANDLE_VALUE != hFindFiles) &&
  1748. (MAX_WALLPAPERSIZEMRU > *pdwAdded))
  1749. {
  1750. if (!PathIsDotOrDotDot(findFileData.cFileName))
  1751. {
  1752. if (FILE_ATTRIBUTE_DIRECTORY & findFileData.dwFileAttributes)
  1753. {
  1754. if (fRecursive)
  1755. {
  1756. TCHAR szSubDir[MAX_PATH];
  1757. StrCpyN(szSubDir, pszPath, ARRAYSIZE(szSubDir));
  1758. PathAppend(szSubDir, findFileData.cFileName);
  1759. hr = _CalcSizeFromDir(szSubDir, pdwAdded, fRecursive);
  1760. }
  1761. }
  1762. else
  1763. {
  1764. if (IsGraphicsFile(findFileData.cFileName))
  1765. {
  1766. TCHAR szPath[MAX_PATH];
  1767. StrCpyN(szPath, pszPath, ARRAYSIZE(szPath));
  1768. PathAppend(szPath, findFileData.cFileName);
  1769. hr = _CalcSizeForFile(szPath, &findFileData, pdwAdded);
  1770. }
  1771. }
  1772. }
  1773. if (!FindNextFile(hFindFiles, &findFileData))
  1774. {
  1775. break;
  1776. }
  1777. }
  1778. FindClose(hFindFiles);
  1779. }
  1780. }
  1781. return hr;
  1782. }
  1783. /*****************************************************************************\
  1784. DESCRIPTION:
  1785. This function will create or update the sizes of the files we have in the
  1786. wallpaper list. We can later use this to decide if we want to select "Tile"
  1787. vs. "Stretch".
  1788. PERF:
  1789. We don't track more than 500 files because we want to keep the MRU from
  1790. growing too much. Users shouldn't have more than that many files normally.
  1791. And if so, they won't get this feature.
  1792. Max 500 Files: First scan will take ~35 seconds on a 300MHz 128MB machine.
  1793. Max 500 Files: Update scan will take ~3 seconds on a 300MHz 128MB machine.
  1794. \*****************************************************************************/
  1795. DWORD CBackPropSheetPage::_SizeCheckerThreadProc(void)
  1796. {
  1797. HRESULT hr;
  1798. DWORD dwAdded = 0;
  1799. TCHAR szPath[MAX_PATH];
  1800. // We want to make our priority low so we don't slow down the UI.
  1801. SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_BELOW_NORMAL);
  1802. // Get the directory with the wallpaper files.
  1803. if (!GetStringFromReg(HKEY_LOCAL_MACHINE, c_szSetup, c_szSharedDir, NULL, szPath, ARRAYSIZE(szPath)))
  1804. {
  1805. if (!GetWindowsDirectory(szPath, ARRAYSIZE(szPath)))
  1806. {
  1807. szPath[0] = 0;
  1808. }
  1809. }
  1810. // Add only the *.BMP files in the windows directory.
  1811. _CalcSizeFromDir(szPath, &dwAdded, FALSE);
  1812. // Get the wallpaper Directory name
  1813. szPath[0] = 0;
  1814. GetWallpaperDirName(szPath, ARRAYSIZE(szPath));
  1815. if (szPath[0])
  1816. {
  1817. // Add all pictures from Wallpaper directory to the list!
  1818. _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1819. }
  1820. //Get the path to the "My Pictures" folder
  1821. if (S_OK == SHGetFolderPath(NULL, CSIDL_MYPICTURES | CSIDL_FLAG_CREATE, NULL, 0, szPath))
  1822. {
  1823. // Add all pictures in "My Pictures" directory to the list!
  1824. hr = _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1825. }
  1826. //Get the path to the common "My Pictures" folder
  1827. if (S_OK == SHGetFolderPath(NULL, CSIDL_COMMON_PICTURES | CSIDL_FLAG_CREATE, NULL, 0, szPath))
  1828. {
  1829. // Add all pictures in common "My Pictures" directory to the list!
  1830. hr = _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1831. }
  1832. // Add pictures from Theme Directories
  1833. // The follwoing directories can contain themes:
  1834. // Plus!98 Install Path\Themes
  1835. // Plus!95 Install Path\Themes
  1836. // Kids for Plus! Install Path\Themes
  1837. // Program Files\Plus!\Themes
  1838. if (SUCCEEDED(_GetPlus98ThemesDir(szPath, ARRAYSIZE(szPath))))
  1839. {
  1840. _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1841. }
  1842. else if (SUCCEEDED(_GetPlus95ThemesDir(szPath, ARRAYSIZE(szPath))))
  1843. {
  1844. _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1845. }
  1846. else if (SUCCEEDED(_GetKidsThemesDir(szPath, ARRAYSIZE(szPath))))
  1847. {
  1848. _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1849. }
  1850. else if (SUCCEEDED(_GetHardDirThemesDir(szPath, ARRAYSIZE(szPath))))
  1851. {
  1852. _CalcSizeFromDir(szPath, &dwAdded, TRUE);
  1853. }
  1854. ATOMICRELEASE(_pSizeMRUBk);
  1855. ATOMICRELEASE(_pImgFactBk);
  1856. _fScanFinished = TRUE;
  1857. Release();
  1858. return 0;
  1859. }
  1860. // We decide to use tile mode if the width and height of the
  1861. // picture are 256x256 or smaller. Anything this small is most
  1862. // likely a watermark. Besides, it will always look bad stretched
  1863. // or centered.
  1864. // We make exceptions for the following wallpaper that we ship
  1865. // that is watermark:
  1866. // "Boiling Point.jpg", which is 163x293.
  1867. // "Fall Memories.jpg", which is 210x185.
  1868. // "Fly Away.jpg", which is 210x185.
  1869. // "Prairie Wind.jpg", which is 255x255.
  1870. // "Santa Fe Stucco.jpg", which is 256x256.
  1871. // "Soap Bubbles.jpg", which is 256x256.
  1872. // "Water Color.jpg", which is 218x162.
  1873. #define SHOULD_USE_TILE(xPicture, yPicture, xMonitor, yMonitor) ((((((LONG)(xPicture) * 6) < (xMonitor)) && (((LONG)(yPicture) * 6) < (yMonitor)))) || \
  1874. (((xPicture) <= 256) && ((yPicture) <= 256)) || \
  1875. (((xPicture) == 163) && ((yPicture) == 293)))
  1876. DWORD CBackPropSheetPage::_GetStretchMode(IN LPCTSTR pszPath)
  1877. {
  1878. HRESULT hr = S_OK;
  1879. DWORD dwStretchMode = WPSTYLE_STRETCH; // Default to Stretch.
  1880. if (_fScanFinished)
  1881. {
  1882. // If we just finished the Scan, we want to release _pSizeMRU and get a new one.
  1883. // The object will cache the registry state so it may have none or very few of the results
  1884. // because it was obtained before the background thread did all it's work.
  1885. ATOMICRELEASE(_pSizeMRU);
  1886. _fScanFinished = FALSE;
  1887. }
  1888. if (!_pSizeMRU)
  1889. {
  1890. hr = _GetMRUObject(&_pSizeMRU);
  1891. }
  1892. if (SUCCEEDED(hr))
  1893. {
  1894. WALLPAPERSIZE_STRUCT wallpaperSize;
  1895. int nIndex;
  1896. StrCpyNW(wallpaperSize.szPath, pszPath, ARRAYSIZE(wallpaperSize.szPath));
  1897. // Let's see if it's already in the MRU.
  1898. hr = _pSizeMRU->FindData((LPCBYTE) &wallpaperSize, sizeof(wallpaperSize), &nIndex);
  1899. if (SUCCEEDED(hr))
  1900. {
  1901. // If so, let's see if it's been modified since.
  1902. hr = _pSizeMRU->GetData(nIndex, (LPBYTE) &wallpaperSize, sizeof(wallpaperSize));
  1903. if (SUCCEEDED(hr))
  1904. {
  1905. WIN32_FIND_DATA findFileData;
  1906. HANDLE hFindFiles = FindFirstFile(pszPath, &findFileData);
  1907. if (hFindFiles && (INVALID_HANDLE_VALUE != hFindFiles))
  1908. {
  1909. // Yes, the value exists and is up to date.
  1910. if (!CompareFileTime(&wallpaperSize.ftLastModified, &findFileData.ftLastWriteTime))
  1911. {
  1912. RECT rc;
  1913. GetMonitorRects(GetPrimaryMonitor(), &rc, 0);
  1914. // We decide to use tile mode if the width and height of the
  1915. // picture are 1/6th the size of the default monitor.
  1916. if (SHOULD_USE_TILE(wallpaperSize.dwSizeX, wallpaperSize.dwSizeY, RECTWIDTH(rc), RECTHEIGHT(rc)))
  1917. {
  1918. dwStretchMode = WPSTYLE_TILE;
  1919. }
  1920. else
  1921. {
  1922. double dWidth = ((double)wallpaperSize.dwSizeX * ((double)RECTHEIGHT(rc) / ((double)RECTWIDTH(rc))));
  1923. // We want to use WPSTYLE_CENTER if it's more than 7% off of a 4x3 aspect ratio.
  1924. // We do this to prevent it from looking bad because it's stretched too much in
  1925. // one direction. The reason we use 7% is because some common screen solutions
  1926. // (1280 x 1024) aren't 4x3, but they are under 7% of that.
  1927. if (dWidth <= (wallpaperSize.dwSizeY * 1.07) && (dWidth >= (wallpaperSize.dwSizeY * 0.92)))
  1928. {
  1929. dwStretchMode = WPSTYLE_STRETCH; // This is within 4x3
  1930. }
  1931. else
  1932. {
  1933. dwStretchMode = WPSTYLE_CENTER;
  1934. }
  1935. }
  1936. }
  1937. FindClose(hFindFiles);
  1938. }
  1939. }
  1940. }
  1941. }
  1942. return dwStretchMode;
  1943. }
  1944. //===========================
  1945. // *** IBasePropPage Interface ***
  1946. //===========================
  1947. HRESULT CBackPropSheetPage::GetAdvancedDialog(OUT IAdvancedDialog ** ppAdvDialog)
  1948. {
  1949. HRESULT hr = E_INVALIDARG;
  1950. if (ppAdvDialog)
  1951. {
  1952. *ppAdvDialog = NULL;
  1953. hr = _LoadState();
  1954. if (SUCCEEDED(hr))
  1955. {
  1956. CCompPropSheetPage * pThis = new CCompPropSheetPage();
  1957. if (pThis)
  1958. {
  1959. hr = pThis->QueryInterface(IID_PPV_ARG(IAdvancedDialog, ppAdvDialog));
  1960. pThis->Release();
  1961. }
  1962. else
  1963. {
  1964. hr = E_OUTOFMEMORY;
  1965. }
  1966. }
  1967. }
  1968. return hr;
  1969. }
  1970. HRESULT CBackPropSheetPage::OnApply(IN PROPPAGEONAPPLY oaAction)
  1971. {
  1972. HRESULT hr = S_OK;
  1973. if (PPOAACTION_CANCEL != oaAction)
  1974. {
  1975. if (_fAllowChanges)
  1976. {
  1977. // The user clicked Okay in the dialog so merge the dirty state from the
  1978. // advanced dialog into the base dialog.
  1979. EnableADifHtmlWallpaper(_hwnd);
  1980. SetSafeMode(SSM_CLEAR);
  1981. g_pActiveDesk->SetWallpaper(_pszOriginalFile, 0);
  1982. SetSafeMode(SSM_CLEAR);
  1983. EnableADifHtmlWallpaper(_hwnd);
  1984. if (g_pActiveDesk)
  1985. {
  1986. g_pActiveDesk->ApplyChanges(g_dwApplyFlags);
  1987. }
  1988. }
  1989. if (SUCCEEDED(hr))
  1990. {
  1991. hr = _SaveIconState();
  1992. if (SUCCEEDED(hr))
  1993. {
  1994. hr = _SaveDesktopOptionsState();
  1995. }
  1996. }
  1997. if (g_iRunDesktopCleanup != BST_INDETERMINATE)
  1998. {
  1999. ApplyDesktopCleanupSettings(); // ignore return value, as nobody cares about it
  2000. }
  2001. SetWindowLongPtr(_hwnd, DWLP_MSGRESULT, PSNRET_NOERROR);
  2002. }
  2003. return hr;
  2004. }
  2005. BOOL IsIconHeaderProperty(IN LPCOLESTR pszPropName)
  2006. {
  2007. return (!StrCmpNIW(SZ_ICONHEADER, pszPropName, ARRAYSIZE(SZ_ICONHEADER) - 1) &&
  2008. ((ARRAYSIZE(SZ_ICONHEADER) + MAX_GUID_STRING_LEN - 1) < lstrlenW(pszPropName)));
  2009. }
  2010. BOOL IsShowDeskIconProperty(IN LPCOLESTR pszPropName)
  2011. {
  2012. return ((!StrCmpNIW(STARTPAGE_ON_PREFIX, pszPropName, LEN_PROP_PREFIX) ||
  2013. !StrCmpNIW(STARTPAGE_OFF_PREFIX, pszPropName, LEN_PROP_PREFIX) ||
  2014. !StrCmpNIW(POLICY_PREFIX, pszPropName, LEN_PROP_PREFIX)) &&
  2015. ((LEN_PROP_PREFIX + MAX_GUID_STRING_LEN - 1) <= lstrlenW(pszPropName)));
  2016. }
  2017. //===========================
  2018. // *** IPropertyBag Interface ***
  2019. //===========================
  2020. HRESULT CBackPropSheetPage::Read(IN LPCOLESTR pszPropName, IN VARIANT * pVar, IN IErrorLog *pErrorLog)
  2021. {
  2022. HRESULT hr = E_INVALIDARG;
  2023. if (pszPropName && pVar)
  2024. {
  2025. VARTYPE vtDesired = pVar->vt;
  2026. if (!StrCmpW(pszPropName, SZ_PBPROP_BACKGROUND_PATH)) // Get the real current wallpaper (after conversion to .bmp)
  2027. {
  2028. WCHAR szPath[MAX_PATH];
  2029. hr = g_pActiveDesk->GetWallpaper(szPath, ARRAYSIZE(szPath), 0);
  2030. if (SUCCEEDED(hr))
  2031. {
  2032. WCHAR szFullPath[MAX_PATH];
  2033. // The string may come back with environment variables.
  2034. if (0 == SHExpandEnvironmentStrings(szPath, szFullPath, ARRAYSIZE(szFullPath)))
  2035. {
  2036. StrCpyN(szFullPath, szPath, ARRAYSIZE(szFullPath)); // We failed so use the original.
  2037. }
  2038. hr = InitVariantFromStr(pVar, szFullPath);
  2039. }
  2040. }
  2041. else if (!StrCmpW(pszPropName, SZ_PBPROP_BACKGROUNDSRC_PATH)) // Get the original wallpaper (before convertion to .bmp)
  2042. {
  2043. WCHAR szPath[MAX_PATH];
  2044. hr = _Initialize();
  2045. if (SUCCEEDED(hr))
  2046. {
  2047. if (_pszLastSourcePath)
  2048. {
  2049. hr = InitVariantFromStr(pVar, _pszLastSourcePath);
  2050. }
  2051. else if (_pszOrigLastApplied)
  2052. {
  2053. hr = InitVariantFromStr(pVar, _pszOrigLastApplied);
  2054. }
  2055. else
  2056. {
  2057. hr = g_pActiveDesk->GetWallpaper(szPath, ARRAYSIZE(szPath), 0);
  2058. if (SUCCEEDED(hr))
  2059. {
  2060. hr = InitVariantFromStr(pVar, szPath);
  2061. }
  2062. }
  2063. }
  2064. }
  2065. else if (!StrCmpW(pszPropName, SZ_PBPROP_BACKGROUND_TILE))
  2066. {
  2067. if (g_pActiveDesk)
  2068. {
  2069. WALLPAPEROPT wpo;
  2070. wpo.dwSize = sizeof(wpo);
  2071. hr = g_pActiveDesk->GetWallpaperOptions(&wpo, 0);
  2072. if (SUCCEEDED(hr))
  2073. {
  2074. pVar->ulVal = wpo.dwStyle;
  2075. pVar->vt = VT_UI4;
  2076. }
  2077. }
  2078. }
  2079. else if (!StrCmpW(pszPropName, SZ_PBPROP_WEBCOMPONENTS))
  2080. {
  2081. if (g_pActiveDesk)
  2082. {
  2083. g_pActiveDesk->AddRef();
  2084. pVar->punkVal = g_pActiveDesk;
  2085. pVar->vt = VT_UNKNOWN;
  2086. hr = S_OK;
  2087. }
  2088. }
  2089. else if (IsIconHeaderProperty(pszPropName))
  2090. {
  2091. // The caller can pass us the string in the following format:
  2092. // pszPropName="CLSID\{<CLSID>}\DefaultIcon:<Item>" = "<FilePath>,<ResourceIndex>"
  2093. // For example:
  2094. // pszPropName="CLSID\{20D04FE0-3AEA-1069-A2D8-08002B30309D}\DefaultIcon:DefaultValue" = "%WinDir%SYSTEM\COOL.DLL,16"
  2095. hr = _LoadState();
  2096. if (SUCCEEDED(hr))
  2097. {
  2098. CLSID clsid;
  2099. WCHAR szTemp[MAX_PATH];
  2100. // Get the CLSID
  2101. StrCpyNW(szTemp, &(pszPropName[ARRAYSIZE(SZ_ICONHEADER) - 2]), ARRAYSIZE(szTemp));
  2102. hr = SHCLSIDFromString(szTemp, &clsid);
  2103. if (SUCCEEDED(hr))
  2104. {
  2105. // Get the name of the icon type. Normally this is "DefaultIcon", but it can be several states, like
  2106. // "full" and "empty" for the recycle bin.
  2107. LPCWSTR pszToken = StrChrW((pszPropName + ARRAYSIZE(SZ_ICONHEADER)), L':');
  2108. BOOL fOldIcon = FALSE;
  2109. // If they use a ";" instead of a ":" then they want the old icon.
  2110. if (!pszToken)
  2111. {
  2112. pszToken = StrChrW((pszPropName + ARRAYSIZE(SZ_ICONHEADER)), L';');
  2113. fOldIcon = TRUE;
  2114. }
  2115. hr = E_FAIL;
  2116. if (pszToken)
  2117. {
  2118. TCHAR szIconPath[MAX_PATH];
  2119. pszToken++;
  2120. hr = _GetIconPath(clsid, pszToken, FALSE, szIconPath, ARRAYSIZE(szIconPath));
  2121. if (SUCCEEDED(hr))
  2122. {
  2123. hr = InitVariantFromStr(pVar, szIconPath);
  2124. }
  2125. }
  2126. }
  2127. }
  2128. }
  2129. else if(IsShowDeskIconProperty(pszPropName))
  2130. {
  2131. int iStartPaneOn = (int)(StrCmpNIW(pszPropName, STARTPAGE_ON_PREFIX, LEN_PROP_PREFIX) == 0);
  2132. for(int iIndex = 0; iIndex < NUM_DESKICONS; iIndex++)
  2133. {
  2134. if(lstrcmpiW(pszPropName+LEN_PROP_PREFIX, c_aDeskIconId[iIndex].pwszCLSID) == 0)
  2135. {
  2136. BOOL fBoolValue = FALSE;
  2137. pVar->vt = VT_BOOL;
  2138. //Check if we are looking for the POLICY or for Show/Hide
  2139. if(!StrCmpNIW(POLICY_PREFIX, pszPropName, LEN_PROP_PREFIX))
  2140. {
  2141. //We are reading "whether-the-POLICY-is-set" property!
  2142. fBoolValue = _aDeskIconNonEnumData[iIndex].fNonEnumPolicySet;
  2143. }
  2144. else
  2145. {
  2146. //We are reading the fHideIcon property.
  2147. fBoolValue = _aHideDesktopIcon[iStartPaneOn][iIndex].fHideIcon;
  2148. }
  2149. pVar ->boolVal = fBoolValue ? VARIANT_TRUE : VARIANT_FALSE;
  2150. hr = S_OK;
  2151. break; //break out of the loop!
  2152. }
  2153. }
  2154. }
  2155. if (SUCCEEDED(hr))
  2156. hr = VariantChangeTypeForRead(pVar, vtDesired);
  2157. }
  2158. return hr;
  2159. }
  2160. HRESULT CBackPropSheetPage::Write(IN LPCOLESTR pszPropName, IN VARIANT *pVar)
  2161. {
  2162. HRESULT hr = E_INVALIDARG;
  2163. if (pszPropName && pVar)
  2164. {
  2165. if ((VT_UI4 == pVar->vt) &&
  2166. _fAllowChanges &&
  2167. !StrCmpW(pszPropName, SZ_PBPROP_BACKGROUND_TILE))
  2168. {
  2169. hr = _SetNewWallpaperTile(pVar->ulVal, FALSE);
  2170. }
  2171. if (!StrCmpW(pszPropName, SZ_PBPROP_OPENADVANCEDDLG) &&
  2172. (VT_BOOL == pVar->vt))
  2173. {
  2174. _fOpenAdvOnInit = (VARIANT_TRUE == pVar->boolVal);
  2175. hr = S_OK;
  2176. }
  2177. else if (!StrCmpW(pszPropName, SZ_PBPROP_WEBCOMPONENTS) &&
  2178. (VT_UNKNOWN == pVar->vt))
  2179. {
  2180. IUnknown_Set((IUnknown **) &g_pActiveDesk, pVar->punkVal);
  2181. hr = S_OK;
  2182. }
  2183. else if (VT_BSTR == pVar->vt)
  2184. {
  2185. if (_fAllowChanges && !StrCmpW(pszPropName, SZ_PBPROP_BACKGROUND_PATH))
  2186. {
  2187. _fWallpaperChanged = TRUE;
  2188. hr = _SetNewWallpaper(pVar->bstrVal, FALSE);
  2189. }
  2190. else if (IsIconHeaderProperty(pszPropName))
  2191. {
  2192. // The caller can pass us the string in the following format:
  2193. // pszPropName="CLSID\{<CLSID>}\DefaultIcon:<Item>" = "<FilePath>,<ResourceIndex>"
  2194. // For example:
  2195. // pszPropName="CLSID\{20D04FE0-3AEA-1069-A2D8-08002B30309D}\DefaultIcon:DefaultValue" = "%WinDir%SYSTEM\COOL.DLL,16"
  2196. hr = _LoadState();
  2197. if (SUCCEEDED(hr))
  2198. {
  2199. CLSID clsid;
  2200. WCHAR szTemp[MAX_PATH];
  2201. // Get the CLSID
  2202. StrCpyNW(szTemp, &(pszPropName[ARRAYSIZE(SZ_ICONHEADER) - 2]), ARRAYSIZE(szTemp));
  2203. hr = SHCLSIDFromString(szTemp, &clsid);
  2204. if (SUCCEEDED(hr))
  2205. {
  2206. // Get the name of the icon type. Normally this is "DefaultIcon", but it can be several states, like
  2207. // "full" and "empty" for the recycle bin.
  2208. LPCWSTR pszToken = StrChrW((pszPropName + ARRAYSIZE(SZ_ICONHEADER)), L':');
  2209. hr = E_FAIL;
  2210. if (pszToken)
  2211. {
  2212. pszToken++;
  2213. StrCpyNW(szTemp, pszToken, ARRAYSIZE(szTemp));
  2214. // Now the pVar->bstrVal is the icon path + "," + resourceID. Separate those two.
  2215. WCHAR szPath[MAX_PATH];
  2216. StrCpyNW(szPath, pVar->bstrVal, ARRAYSIZE(szPath));
  2217. int nResourceID = PathParseIconLocationW(szPath);
  2218. hr = _SetIconPath(clsid, szTemp, szPath, nResourceID);
  2219. }
  2220. }
  2221. }
  2222. }
  2223. }
  2224. else if((VT_BOOL == pVar->vt) && (IsShowDeskIconProperty(pszPropName)))
  2225. {
  2226. int iStartPaneOn = (int)(StrCmpNIW(pszPropName, STARTPAGE_ON_PREFIX, LEN_PROP_PREFIX) == 0);
  2227. for(int iIndex = 0; iIndex < NUM_DESKICONS; iIndex++)
  2228. {
  2229. if(lstrcmpiW(pszPropName+LEN_PROP_PREFIX, c_aDeskIconId[iIndex].pwszCLSID) == 0)
  2230. {
  2231. BOOL fNewHideIconStatus = (VARIANT_TRUE == pVar->boolVal);
  2232. //check if the new hide icon status is different from the old one.
  2233. if(_aHideDesktopIcon[iStartPaneOn][iIndex].fHideIcon != fNewHideIconStatus)
  2234. {
  2235. _aHideDesktopIcon[iStartPaneOn][iIndex].fHideIcon = fNewHideIconStatus;
  2236. _aHideDesktopIcon[iStartPaneOn][iIndex].fDirty = TRUE;
  2237. _fHideDesktopIconDirty = TRUE;
  2238. }
  2239. hr = S_OK;
  2240. break; //break out of the loop!
  2241. }
  2242. }
  2243. }
  2244. }
  2245. return hr;
  2246. }
  2247. //===========================
  2248. // *** IShellPropSheetExt Interface ***
  2249. //===========================
  2250. HRESULT CBackPropSheetPage::AddPages(IN LPFNSVADDPROPSHEETPAGE pfnAddPage, IN LPARAM lParam)
  2251. {
  2252. HRESULT hr = E_NOTIMPL;
  2253. PROPSHEETPAGE psp = {0};
  2254. // If classic shell is forced, then we desk.cpl will put-up the old background page
  2255. // So, we should not replace that with this new background page.
  2256. // (Note: All other ActiveDesktop restrictions are checked in dcomp.cpp to prevent
  2257. // the web tab from appearing there).
  2258. if (SHRestricted(REST_CLASSICSHELL))
  2259. {
  2260. // It's restricted, so don't add this page.
  2261. hr = E_ACCESSDENIED;
  2262. }
  2263. else
  2264. {
  2265. // Initialize a bunch of propsheetpage variables.
  2266. psp.dwSize = sizeof(psp);
  2267. psp.hInstance = HINST_THISDLL;
  2268. psp.dwFlags = PSP_DEFAULT | PSP_USECALLBACK;
  2269. psp.lParam = (LPARAM) this;
  2270. // psp.hIcon = NULL; // unused (PSP_USEICON is not set)
  2271. // psp.pszTitle = NULL; // unused (PSP_USETITLE is not set)
  2272. // psp.lParam = 0; // unused
  2273. // psp.pcRefParent = NULL;
  2274. psp.pszTemplate = MAKEINTRESOURCE(IDD_BACKGROUND);
  2275. psp.pfnDlgProc = CBackPropSheetPage::BackgroundDlgProc;
  2276. HPROPSHEETPAGE hpsp = CreatePropertySheetPage(&psp);
  2277. if (hpsp)
  2278. {
  2279. if (pfnAddPage(hpsp, lParam))
  2280. {
  2281. hr = S_OK;
  2282. }
  2283. else
  2284. {
  2285. DestroyPropertySheetPage(hpsp);
  2286. hr = E_FAIL;
  2287. }
  2288. }
  2289. else
  2290. {
  2291. hr = E_OUTOFMEMORY;
  2292. }
  2293. }
  2294. return hr;
  2295. }
  2296. //===========================
  2297. // *** IUnknown Interface ***
  2298. //===========================
  2299. ULONG CBackPropSheetPage::AddRef()
  2300. {
  2301. _cRef++;
  2302. return _cRef;
  2303. }
  2304. ULONG CBackPropSheetPage::Release()
  2305. {
  2306. ASSERT(_cRef > 0);
  2307. _cRef--;
  2308. if (_cRef > 0)
  2309. return _cRef;
  2310. delete this;
  2311. return 0;
  2312. }
  2313. HRESULT CBackPropSheetPage::QueryInterface(REFIID riid, void **ppvObj)
  2314. {
  2315. HRESULT hr = E_NOINTERFACE;
  2316. static const QITAB qit[] = {
  2317. QITABENT(CBackPropSheetPage, IObjectWithSite),
  2318. QITABENT(CBackPropSheetPage, IBasePropPage),
  2319. QITABENT(CBackPropSheetPage, IPersist),
  2320. QITABENT(CBackPropSheetPage, IPropertyBag),
  2321. QITABENTMULTI(CBackPropSheetPage, IShellPropSheetExt, IBasePropPage),
  2322. { 0 },
  2323. };
  2324. return QISearch(this, qit, riid, ppvObj);
  2325. }
  2326. //===========================
  2327. // *** Class Methods ***
  2328. //===========================
  2329. CBackPropSheetPage::CBackPropSheetPage(void) : CObjectCLSID(&PPID_Background)
  2330. {
  2331. _cRef = 1;
  2332. _punkSite = NULL;
  2333. _pszOriginalFile = NULL;
  2334. _pszLastSourcePath = NULL;
  2335. _pszOrigLastApplied = NULL;
  2336. _pszWallpaperInUse = NULL;
  2337. _pImgFactBk = NULL;
  2338. _pSizeMRU = NULL;
  2339. _pSizeMRUBk = NULL;
  2340. _fThemePreviewCreated = FALSE;
  2341. _pThemePreview = NULL;
  2342. _fWallpaperChanged = FALSE;
  2343. _fSelectionFromUser = TRUE;
  2344. _fStateLoaded = FALSE;
  2345. _fOpenAdvOnInit = FALSE;
  2346. _fScanFinished = FALSE;
  2347. _fInitialized = FALSE;
  2348. _fAllowChanges = (!SHRestricted(REST_NOCHANGINGWALLPAPER) && !IsTSPerfFlagEnabled(TSPerFlag_NoADWallpaper) && !IsTSPerfFlagEnabled(TSPerFlag_NoWallpaper));
  2349. g_dwApplyFlags = (AD_APPLY_ALL | AD_APPLY_DYNAMICREFRESH);
  2350. GetActiveDesktop(&g_pActiveDesk);
  2351. }
  2352. CBackPropSheetPage::~CBackPropSheetPage(void)
  2353. {
  2354. ASSERT(!_pSizeMRUBk); // Should have been released by the background thread.
  2355. ASSERT(!_pImgFactBk); // Should have been released by the background thread.
  2356. Str_SetPtr(&_pszOriginalFile, NULL);
  2357. Str_SetPtrW(&_pszOrigLastApplied, NULL);
  2358. Str_SetPtrW(&_pszWallpaperInUse, NULL);
  2359. Str_SetPtrW(&_pszLastSourcePath, NULL);
  2360. if (_pSizeMRU)
  2361. {
  2362. _pSizeMRU->Release();
  2363. _pSizeMRU = NULL;
  2364. }
  2365. if (_pThemePreview)
  2366. {
  2367. _pThemePreview->Release();
  2368. _pThemePreview = NULL;
  2369. }
  2370. ReleaseActiveDesktop(&g_pActiveDesk);
  2371. }