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.

4117 lines
115 KiB

  1. #include "priv.h"
  2. #include "sccls.h"
  3. #include "mlang.h" // fo char conversion
  4. #include "bandprxy.h"
  5. #include "resource.h"
  6. #include <shdocvw.h>
  7. #include <icwcfg.h>
  8. #include <advpub.h> // for IE activesetup GUID
  9. #include <shellapi.h>
  10. #include "apithk.h" //for WM_KEYBOARDCUES msg
  11. #include <platform.h>
  12. #include <mobsync.h>
  13. #include <mobsyncp.h>
  14. #include "..\shell32\shitemid.h" // for SHID_XX
  15. #ifdef UNIX
  16. #include "unixstuff.h"
  17. #endif /* UNIX */
  18. #define MLUI_INIT
  19. #include "mluisupp.h"
  20. //small (previously duplicated) functions shared between shdcovw and browseui
  21. #include "..\inc\brutil.cpp"
  22. // #define MLUI_SUPPORT 1
  23. LCID g_lcidLocale = MAKELCID(LANG_USER_DEFAULT, SORT_DEFAULT);
  24. #define DM_NAV TF_SHDNAVIGATE
  25. #define DM_ZONE TF_SHDNAVIGATE
  26. #define DM_IEDDE DM_TRACE
  27. #define DM_CANCELMODE 0
  28. #define DM_UIWINDOW 0
  29. #define DM_ENABLEMODELESS 0
  30. #define DM_EXPLORERMENU 0
  31. #define DM_BACKFORWARD 0
  32. #define DM_PROTOCOL 0
  33. #define DM_ITBAR 0
  34. #define DM_STARTUP 0
  35. #define DM_AUTOLIFE 0
  36. #define DM_PALETTE 0
  37. const VARIANT c_vaEmpty = {0};
  38. const LARGE_INTEGER c_li0 = { 0, 0 };
  39. #ifdef DEBUG
  40. const DWORD TempBuffer::s_dummy = 0;
  41. #endif
  42. #undef VariantCopy
  43. WINOLEAUTAPI VariantCopyLazy(VARIANTARG * pvargDest, VARIANTARG * pvargSrc)
  44. {
  45. VariantClearLazy(pvargDest);
  46. switch(pvargSrc->vt)
  47. {
  48. case VT_I4:
  49. case VT_UI4:
  50. case VT_BOOL:
  51. // we can add more
  52. *pvargDest = *pvargSrc;
  53. return S_OK;
  54. case VT_UNKNOWN:
  55. if (pvargDest)
  56. {
  57. *pvargDest = *pvargSrc;
  58. pvargDest->punkVal->AddRef();
  59. return S_OK;
  60. }
  61. ASSERT(0);
  62. return E_INVALIDARG;
  63. }
  64. return VariantCopy(pvargDest, pvargSrc);
  65. }
  66. //
  67. // WARNING: This function must be placed at the end because we #undef
  68. // VariantClear
  69. //
  70. #undef VariantClear
  71. HRESULT VariantClearLazy(VARIANTARG *pvarg)
  72. {
  73. switch(pvarg->vt)
  74. {
  75. case VT_I4:
  76. case VT_UI4:
  77. case VT_EMPTY:
  78. case VT_BOOL:
  79. // No operation
  80. break;
  81. default:
  82. return VariantClear(pvarg);
  83. }
  84. return S_OK;
  85. }
  86. HRESULT QueryService_SID_IBandProxy(IUnknown * punkParent, REFIID riid, IBandProxy ** ppbp, void **ppvObj)
  87. {
  88. HRESULT hr = E_FAIL;
  89. if (ppbp)
  90. {
  91. if (NULL == (*ppbp))
  92. hr = IUnknown_QueryService(punkParent, SID_IBandProxy, IID_PPV_ARG(IBandProxy, ppbp));
  93. if (*ppbp && ppvObj)
  94. hr = (*ppbp)->QueryInterface(riid, ppvObj); // They already have the object.
  95. }
  96. return hr;
  97. }
  98. HRESULT CreateIBandProxyAndSetSite(IUnknown * punkParent, REFIID riid, IBandProxy ** ppbp, void **ppvObj)
  99. {
  100. ASSERT(ppbp);
  101. HRESULT hr = CoCreateInstance(CLSID_BandProxy, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IBandProxy, ppbp));
  102. if (SUCCEEDED(hr))
  103. {
  104. // Set the site
  105. ASSERT(*ppbp);
  106. (*ppbp)->SetSite(punkParent);
  107. if (ppvObj)
  108. hr = (*ppbp)->QueryInterface(riid, ppvObj);
  109. }
  110. else
  111. {
  112. if (ppvObj)
  113. *ppvObj = NULL;
  114. }
  115. return hr;
  116. }
  117. HRESULT IUnknown_FileSysChange(IUnknown* punk, DWORD dwEvent, LPCITEMIDLIST* ppidl)
  118. {
  119. HRESULT hres = E_FAIL;
  120. if (punk)
  121. {
  122. IAddressBand * pab;
  123. hres = punk->QueryInterface(IID_PPV_ARG(IAddressBand, &pab));
  124. if (SUCCEEDED(hres))
  125. {
  126. hres = pab->FileSysChange(dwEvent, ppidl);
  127. pab->Release();
  128. }
  129. }
  130. return hres;
  131. }
  132. UINT g_cfURL = 0;
  133. UINT g_cfHIDA = 0;
  134. UINT g_cfFileDescA = 0;
  135. UINT g_cfFileDescW = 0;
  136. UINT g_cfFileContents = 0;
  137. UINT g_cfPreferedEffect = 0;
  138. void InitClipboardFormats()
  139. {
  140. if (g_cfURL == 0)
  141. {
  142. g_cfURL = RegisterClipboardFormat(CFSTR_SHELLURL);
  143. g_cfHIDA = RegisterClipboardFormat(CFSTR_SHELLIDLIST);
  144. g_cfFileDescA = RegisterClipboardFormat(CFSTR_FILEDESCRIPTORA);
  145. g_cfFileDescW = RegisterClipboardFormat(CFSTR_FILEDESCRIPTORW);
  146. g_cfFileContents = RegisterClipboardFormat(CFSTR_FILECONTENTS);
  147. g_cfPreferedEffect = RegisterClipboardFormat(CFSTR_PREFERREDDROPEFFECT);
  148. }
  149. }
  150. DEFFOLDERSETTINGS g_dfs = INIT_DEFFOLDERSETTINGS;
  151. void _InitDefaultFolderSettings()
  152. {
  153. if (GetSystemMetrics(SM_CLEANBOOT))
  154. return;
  155. g_dfs.vid = g_bRunOnNT5 ? VID_LargeIcons : DFS_VID_Default;
  156. DEFFOLDERSETTINGS dfs = g_dfs;
  157. DWORD dwType, cbData = sizeof(dfs);
  158. if (SUCCEEDED(SKGetValue(SHELLKEY_HKCU_EXPLORER, REGVALUE_STREAMS, TEXT("Settings"), &dwType, &dfs, &cbData))
  159. && dwType == REG_BINARY)
  160. {
  161. if (cbData < sizeof(DEFFOLDERSETTINGS_W2K) || dfs.dwStructVersion < DFS_NASH_VER)
  162. {
  163. dfs.vid = g_bRunOnNT5 ? VID_LargeIcons : DFS_VID_Default;
  164. dfs.dwStructVersion = DFS_NASH_VER;
  165. dfs.bUseVID = TRUE;
  166. }
  167. if (cbData < sizeof(DEFFOLDERSETTINGS) || dfs.dwStructVersion < DFS_WHISTLER_VER)
  168. {
  169. dfs.dwViewPriority = VIEW_PRIORITY_CACHEMISS;
  170. dfs.dwStructVersion = DFS_WHISTLER_VER;
  171. }
  172. g_dfs = dfs;
  173. }
  174. }
  175. CABINETSTATE g_CabState = { 0 };
  176. extern HANDLE g_hCabStateChange;
  177. LONG g_lCabStateCount = -1; // never a valid count
  178. void GetCabState(CABINETSTATE *pcs)
  179. {
  180. if (g_hCabStateChange == NULL)
  181. g_hCabStateChange = SHGlobalCounterCreate(GUID_FolderSettingsChange);
  182. LONG lCabStateCur = SHGlobalCounterGetValue(g_hCabStateChange);
  183. if (g_lCabStateCount != lCabStateCur)
  184. {
  185. g_lCabStateCount = lCabStateCur;
  186. if (!ReadCabinetState(&g_CabState, sizeof(g_CabState)))
  187. {
  188. WriteCabinetState(&g_CabState);
  189. }
  190. }
  191. *pcs = g_CabState;
  192. }
  193. typedef struct tagINIPAIR
  194. {
  195. DWORD dwFlags;
  196. LPCTSTR pszSection;
  197. }
  198. INIPAIR, *PINIPAIR;
  199. const INIPAIR c_aIniPairs[] =
  200. {
  201. EICH_KINET, TEXT("Software\\Microsoft\\Internet Explorer"),
  202. EICH_KINETMAIN, TEXT("Software\\Microsoft\\Internet Explorer\\Main"),
  203. EICH_KWIN, TEXT("Software\\Microsoft\\Windows\\CurrentVersion"),
  204. EICH_KWINEXPLORER, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer"),
  205. EICH_KWINEXPLSMICO, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\SmallIcons"),
  206. EICH_KWINPOLICY, TEXT("Software\\Microsoft\\Windows\\CurrentVersion\\Policies"),
  207. EICH_SSAVETASKBAR, TEXT("SaveTaskbar"),
  208. EICH_SWINDOWMETRICS, TEXT("WindowMetrics"),
  209. EICH_SSHELLMENU, TEXT("ShellMenu"),
  210. EICH_SPOLICY, TEXT("Policy"),
  211. EICH_SWINDOWS, TEXT("Windows"),
  212. };
  213. DWORD SHIsExplorerIniChange(WPARAM wParam, LPARAM lParam)
  214. {
  215. DWORD dwFlags = 0;
  216. if (lParam == 0)
  217. {
  218. if (wParam == 0)
  219. {
  220. dwFlags = EICH_UNKNOWN;
  221. }
  222. }
  223. else
  224. {
  225. //
  226. // In the wacky world of browseui, UNICODE-ANSI doesn't vary from
  227. // window to window. Instead, on NT browseui registers all windows
  228. // UNICODE, while on 9x user browseui registers all windows ANSI.
  229. //
  230. USES_CONVERSION;
  231. LPCTSTR pszSection;
  232. if (g_fRunningOnNT)
  233. pszSection = W2CT((LPCWSTR)lParam);
  234. else
  235. pszSection = A2CT((LPCSTR)lParam);
  236. for (int i = 0; !dwFlags && i < ARRAYSIZE(c_aIniPairs); i++)
  237. {
  238. if (StrCmpI(pszSection, c_aIniPairs[i].pszSection) == 0)
  239. {
  240. dwFlags = c_aIniPairs[i].dwFlags;
  241. }
  242. }
  243. }
  244. return dwFlags;
  245. }
  246. void _InitAppGlobals()
  247. {
  248. static BOOL fInitialized = FALSE;
  249. if (!fInitialized)
  250. {
  251. _InitComCtl32();
  252. _InitDefaultFolderSettings();
  253. // dont put anything else here. instead init on demand
  254. fInitialized = TRUE; // allow a race on the above calls
  255. }
  256. }
  257. BOOL _InitComCtl32()
  258. {
  259. static BOOL fInitialized = FALSE;
  260. if (!fInitialized)
  261. {
  262. INITCOMMONCONTROLSEX icc;
  263. icc.dwSize = sizeof(INITCOMMONCONTROLSEX);
  264. icc.dwICC = ICC_USEREX_CLASSES | ICC_COOL_CLASSES | ICC_INTERNET_CLASSES | ICC_PAGESCROLLER_CLASS
  265. | ICC_NATIVEFNTCTL_CLASS;
  266. fInitialized = InitCommonControlsEx(&icc);
  267. }
  268. return fInitialized;
  269. }
  270. DWORD GetPreferedDropEffect(IDataObject *pdtobj)
  271. {
  272. InitClipboardFormats();
  273. DWORD dwEffect = 0;
  274. STGMEDIUM medium;
  275. DWORD *pdw = (DWORD *)DataObj_GetDataOfType(pdtobj, g_cfPreferedEffect, &medium);
  276. if (pdw)
  277. {
  278. dwEffect = *pdw;
  279. ReleaseStgMediumHGLOBAL(pdw,&medium);
  280. }
  281. return dwEffect;
  282. }
  283. HRESULT _SetPreferedDropEffect(IDataObject *pdtobj, DWORD dwEffect)
  284. {
  285. InitClipboardFormats();
  286. HRESULT hres = E_OUTOFMEMORY;
  287. DWORD *pdw = (DWORD *)GlobalAlloc(GPTR, sizeof(*pdw));
  288. if (pdw)
  289. {
  290. STGMEDIUM medium;
  291. FORMATETC fmte = {g_cfPreferedEffect, NULL, DVASPECT_CONTENT, -1, TYMED_HGLOBAL};
  292. *pdw = dwEffect;
  293. medium.tymed = TYMED_HGLOBAL;
  294. medium.hGlobal = pdw;
  295. medium.pUnkForRelease = NULL;
  296. hres = pdtobj->SetData(&fmte, &medium, TRUE);
  297. if (FAILED(hres))
  298. GlobalFree((HGLOBAL)pdw);
  299. }
  300. return hres;
  301. }
  302. //*** Reg_GetStrs -- get values from registry, assign to struct
  303. void Reg_GetStrs(HKEY hkey, const struct regstrs *tab, LPTSTR szBuf, int cchBuf, void *pv)
  304. {
  305. for (; tab->name != NULL; tab++)
  306. {
  307. ULONG cbTmp = cchBuf;
  308. // NOTE: IE4 did *NOT* support SHLoadRegUIString, so don't call Reg_GetStrs
  309. // on roamable data. (Or at least don't register plugui strings there.)
  310. if (ERROR_SUCCESS == SHLoadRegUIString(hkey, tab->name, szBuf, cbTmp))
  311. {
  312. // pv->field = StrDup(szBuf)
  313. *(LPTSTR *)((char *)pv + tab->off) = StrDup(szBuf);
  314. }
  315. }
  316. return;
  317. }
  318. BOOL g_fNewNotify = FALSE; // Are we using classic mode (W95 or new mode?
  319. PFNSHCHANGENOTIFYREGISTER g_pfnSHChangeNotifyRegister = NULL;
  320. PFNSHCHANGENOTIFYDEREGISTER g_pfnSHChangeNotifyDeregister = NULL;
  321. #define GET_PRIVATE_PROC_ADDRESS(_hinst, _fname, _ord) GetProcAddress(_hinst, _ord)
  322. BOOL _DelayLoadRegisterNotify(void)
  323. {
  324. // See if we need to still figure out which version of SHChange Notify to call?
  325. if (g_pfnSHChangeNotifyDeregister == NULL)
  326. {
  327. // This should never fail, since we are load-time-linked to SHELL32
  328. HMODULE hmodShell32 = GetModuleHandleA("SHELL32.DLL");
  329. if (hmodShell32)
  330. {
  331. g_pfnSHChangeNotifyRegister = (PFNSHCHANGENOTIFYREGISTER)GET_PRIVATE_PROC_ADDRESS(hmodShell32, "NTSHChangeNotifyRegister",MAKEINTRESOURCEA(640));
  332. if (g_pfnSHChangeNotifyRegister && (WhichPlatform() == PLATFORM_INTEGRATED))
  333. {
  334. g_pfnSHChangeNotifyDeregister = (PFNSHCHANGENOTIFYDEREGISTER)GET_PRIVATE_PROC_ADDRESS(hmodShell32,"NTSHChangeNotifyDeregister", MAKEINTRESOURCEA(641));
  335. g_fNewNotify = TRUE;
  336. }
  337. else
  338. {
  339. g_pfnSHChangeNotifyRegister = (PFNSHCHANGENOTIFYREGISTER)GET_PRIVATE_PROC_ADDRESS(hmodShell32, "SHChangeNotifyRegister", MAKEINTRESOURCEA(2));
  340. g_pfnSHChangeNotifyDeregister = (PFNSHCHANGENOTIFYDEREGISTER)GET_PRIVATE_PROC_ADDRESS(hmodShell32, "SHChangeNotifyDeregister",MAKEINTRESOURCEA(4));
  341. }
  342. }
  343. }
  344. return (NULL == g_pfnSHChangeNotifyDeregister) ? FALSE : TRUE;
  345. }
  346. ULONG RegisterNotify(HWND hwnd, UINT nMsg, LPCITEMIDLIST pidl, DWORD dwEvents, UINT uFlags, BOOL fRecursive)
  347. {
  348. if (_DelayLoadRegisterNotify())
  349. {
  350. SHChangeNotifyEntry fsne;
  351. if (g_fNewNotify)
  352. uFlags |= SHCNRF_NewDelivery;
  353. fsne.fRecursive = fRecursive;
  354. fsne.pidl = pidl;
  355. return g_pfnSHChangeNotifyRegister(hwnd, uFlags, dwEvents, nMsg, 1, &fsne);
  356. }
  357. return 0;
  358. }
  359. int PropBag_ReadInt4(IPropertyBag* pPropBag, LPWSTR pszKey, int iDefault)
  360. {
  361. SHPropertyBag_ReadInt(pPropBag, pszKey, &iDefault);
  362. return iDefault;
  363. }
  364. STDAPI_(BOOL) _EnsureLoaded(HINSTANCE *phinst, LPCSTR pszDLL)
  365. {
  366. if (*phinst == NULL)
  367. {
  368. #ifdef DEBUG
  369. if (g_dwDumpFlags & DF_DELAYLOADDLL)
  370. {
  371. TraceMsg(TF_ALWAYS, "DLLLOAD: Loading %s for the first time", pszDLL);
  372. }
  373. if (g_dwBreakFlags & 0x00000080)
  374. {
  375. DebugBreak();
  376. }
  377. #endif
  378. *phinst = LoadLibraryA(pszDLL);
  379. if (*phinst == NULL)
  380. {
  381. return FALSE;
  382. }
  383. }
  384. return TRUE;
  385. }
  386. // global g_hinst values
  387. HINSTANCE g_hinstSHDOCVW = NULL;
  388. HINSTANCE g_hinstShell32 = NULL;
  389. HINSTANCE HinstShdocvw()
  390. {
  391. _EnsureLoaded(&g_hinstSHDOCVW, "shdocvw.dll");
  392. return g_hinstSHDOCVW;
  393. }
  394. HINSTANCE HinstShell32()
  395. {
  396. _EnsureLoaded(&g_hinstShell32, "shell32.dll");
  397. return g_hinstShell32;
  398. }
  399. STDAPI_(BOOL) CallCoInternetQueryInfo(LPCTSTR pszURL, QUERYOPTION QueryOption)
  400. {
  401. DWORD fRetVal;
  402. DWORD dwSize;
  403. WCHAR wszURL[MAX_URL_STRING];
  404. SHTCharToUnicode(pszURL, wszURL, ARRAYSIZE(wszURL));
  405. return SUCCEEDED(CoInternetQueryInfo(
  406. wszURL, QueryOption,
  407. 0, &fRetVal, sizeof(fRetVal), &dwSize, 0)) && fRetVal;
  408. }
  409. HRESULT IURLQualifyW(IN LPCWSTR pcwzURL, DWORD dwFlags, OUT LPWSTR pwzTranslatedURL, LPBOOL pbWasSearchURL, LPBOOL pbWasCorrected)
  410. {
  411. return IURLQualify(pcwzURL, dwFlags, pwzTranslatedURL, pbWasSearchURL, pbWasCorrected);
  412. }
  413. BSTR LoadBSTR(HINSTANCE hinst, UINT uID)
  414. {
  415. WCHAR wszBuf[128];
  416. if (LoadStringW(hinst, uID, wszBuf, ARRAYSIZE(wszBuf)))
  417. {
  418. return SysAllocString(wszBuf);
  419. }
  420. return NULL;
  421. }
  422. HRESULT _SetStdLocation(LPTSTR szPath, UINT id)
  423. {
  424. HRESULT hres = E_FAIL;
  425. WCHAR szDefaultPath[MAX_URL_STRING];
  426. ASSERT(id == DVIDM_GOHOME);
  427. if (SUCCEEDED(URLSubLoadString(MLGetHinst(), IDS_DEF_HOME, szDefaultPath, SIZECHARS(szDefaultPath), URLSUB_ALL)))
  428. {
  429. if (!StrCmp(szDefaultPath, szPath))
  430. return S_OK; // We don't need to write out the name string.
  431. }
  432. DWORD cbSize = (lstrlen(szPath) + 1) * sizeof(TCHAR);
  433. if (ERROR_SUCCESS == SHSetValue(HKEY_CURRENT_USER, TEXT("Software\\Microsoft\\Internet Explorer\\Main"), (id==DVIDM_GOHOME) ? TEXT("Start Page") : TEXT("Search Page"),
  434. REG_SZ, (LPBYTE)szPath, cbSize))
  435. {
  436. hres = S_OK;
  437. }
  438. return hres;
  439. }
  440. //*** IsVK_TABCycler -- is key a TAB-equivalent
  441. // ENTRY/EXIT
  442. // dir 0 if not a TAB, non-0 if a TAB
  443. // NOTES
  444. // NYI: -1 for shift+tab, 1 for tab
  445. //
  446. int IsVK_TABCycler(MSG *pMsg)
  447. {
  448. int nDir = 0;
  449. if (!pMsg)
  450. return nDir;
  451. if (pMsg->message != WM_KEYDOWN)
  452. return nDir;
  453. if (! (pMsg->wParam == VK_TAB || pMsg->wParam == VK_F6))
  454. return nDir;
  455. nDir = (GetKeyState(VK_SHIFT) < 0) ? -1 : 1;
  456. #ifdef KEYBOARDCUES
  457. HWND hwndParent = GetParent(pMsg->hwnd);
  458. if (hwndParent)
  459. SendMessage(hwndParent, WM_CHANGEUISTATE, MAKEWPARAM(UIS_CLEAR, UISF_HIDEFOCUS), 0);
  460. #endif
  461. return nDir ;
  462. }
  463. BOOL IsVK_CtlTABCycler(MSG *pMsg)
  464. {
  465. if (IsVK_TABCycler(pMsg))
  466. {
  467. if (GetKeyState(VK_CONTROL) < 0 || (pMsg->wParam == VK_F6))
  468. return TRUE;
  469. }
  470. return FALSE;
  471. }
  472. const ITEMIDLIST s_idlNULL = { 0 } ;
  473. // Copied from shell32 (was _ILCreate), which does not export this.
  474. // The fsmenu code needs this function.
  475. STDAPI_(LPITEMIDLIST) IEILCreate(UINT cbSize)
  476. {
  477. LPITEMIDLIST pidl = (LPITEMIDLIST)SHAlloc(cbSize);
  478. if (pidl)
  479. memset(pidl, 0, cbSize); // needed for external task allicator
  480. return pidl;
  481. }
  482. void SaveDefaultFolderSettings(UINT flags)
  483. {
  484. ASSERT(!(flags & ~GFSS_VALID));
  485. if (flags & GFSS_SETASDEFAULT)
  486. g_dfs.dwDefRevCount++;
  487. SKSetValue(SHELLKEY_HKCU_EXPLORER, REGVALUE_STREAMS, TEXT("Settings"), REG_BINARY, &g_dfs, sizeof(g_dfs));
  488. }
  489. BOOL ViewIDFromViewMode(UINT uViewMode, SHELLVIEWID *pvid)
  490. {
  491. switch (uViewMode)
  492. {
  493. case FVM_ICON:
  494. *pvid = VID_LargeIcons;
  495. break;
  496. case FVM_SMALLICON:
  497. *pvid = VID_SmallIcons;
  498. break;
  499. case FVM_LIST:
  500. *pvid = VID_List;
  501. break;
  502. case FVM_DETAILS:
  503. *pvid = VID_Details;
  504. break;
  505. case FVM_THUMBNAIL:
  506. *pvid = VID_Thumbnails;
  507. break;
  508. case FVM_TILE:
  509. *pvid = VID_Tile;
  510. break;
  511. default:
  512. *pvid = VID_LargeIcons;
  513. return(FALSE);
  514. }
  515. return(TRUE);
  516. }
  517. // This is a hack for IE6 23652 Beta 2. Remove in Whistler RC 1.
  518. BOOL CheckForOutlookExpress()
  519. {
  520. HKEY hKeyMail = NULL;
  521. HKEY hKeyOE = NULL;
  522. DWORD dwErr = 0;
  523. DWORD dwSize = 0;
  524. TCHAR szBuf[MAX_PATH];
  525. DWORD dwType = 0;
  526. BOOL bRet = FALSE;
  527. // Open the key for default internet mail client
  528. // HKLM\Software\Clients\Mail
  529. dwErr = RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("Software\\Clients\\Mail"), 0, KEY_READ, &hKeyMail);
  530. if(dwErr != ERROR_SUCCESS)
  531. {
  532. // DebugTrace( TEXT("RegopenKey %s Failed -> %u\n"), szDefMailKey, dwErr);
  533. goto out;
  534. }
  535. dwSize = ARRAYSIZE(szBuf); // Expect ERROR_MORE_DATA
  536. dwErr = RegQueryValueEx( hKeyMail, NULL, NULL, &dwType, (LPBYTE)szBuf, &dwSize);
  537. if(dwErr != ERROR_SUCCESS)
  538. {
  539. goto out;
  540. }
  541. if(!lstrcmpi(szBuf, TEXT("Outlook Express")))
  542. {
  543. // Yes its outlook express ..
  544. bRet = TRUE;
  545. }
  546. out:
  547. if(hKeyOE)
  548. RegCloseKey(hKeyOE);
  549. if(hKeyMail)
  550. RegCloseKey(hKeyMail);
  551. return bRet;
  552. }
  553. HRESULT DropOnMailRecipient(IDataObject *pdtobj, DWORD grfKeyState)
  554. {
  555. IDropTarget *pdrop;
  556. HRESULT hres = CoCreateInstance(CLSID_MailRecipient,
  557. NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER,
  558. IID_PPV_ARG(IDropTarget, &pdrop));
  559. ULONG_PTR uCookie = 0;
  560. if (CheckForOutlookExpress())
  561. {
  562. SHActivateContext(&uCookie);
  563. }
  564. if (SUCCEEDED(hres))
  565. {
  566. hres = SHSimulateDrop(pdrop, pdtobj, grfKeyState, NULL, NULL);
  567. pdrop->Release();
  568. }
  569. if (uCookie)
  570. {
  571. SHDeactivateContext(uCookie);
  572. }
  573. return hres;
  574. }
  575. //
  576. // This function cannot return Non -NULL pointers if
  577. // it returns a FAILED(hr)
  578. //
  579. HRESULT CreateShortcutSetSiteAndGetDataObjectIfPIDLIsNetUrl(
  580. LPCITEMIDLIST pidl,
  581. IUnknown *pUnkSite,
  582. IUniformResourceLocator **ppUrlOut,
  583. IDataObject **ppdtobj
  584. )
  585. {
  586. HRESULT hr;
  587. TCHAR szUrl[MAX_URL_STRING];
  588. ASSERT(ppUrlOut);
  589. ASSERT(ppdtobj);
  590. *ppUrlOut = NULL;
  591. *ppdtobj = NULL;
  592. szUrl[0] = TEXT('\0');
  593. hr = IEGetNameAndFlags(pidl, SHGDN_FORPARSING, szUrl, SIZECHARS(szUrl), NULL);
  594. if ((S_OK == hr) && (*szUrl))
  595. {
  596. BOOL fIsHTML = FALSE;
  597. BOOL fHitsNet = UrlHitsNetW(szUrl);
  598. if (!fHitsNet)
  599. {
  600. if (URL_SCHEME_FILE == GetUrlScheme(szUrl))
  601. {
  602. TCHAR *szExt = PathFindExtension(szUrl);
  603. if (szExt)
  604. {
  605. fIsHTML = ((0 == StrCmpNI(szExt, TEXT(".htm"),4)) ||
  606. (0 == StrCmpNI(szExt, TEXT(".html"),5)));
  607. }
  608. }
  609. }
  610. if (fHitsNet || fIsHTML)
  611. {
  612. // Create a shortcut object and
  613. HRESULT hr = CoCreateInstance(CLSID_InternetShortcut, NULL, CLSCTX_INPROC_SERVER,
  614. IID_PPV_ARG(IUniformResourceLocator, ppUrlOut));
  615. if (SUCCEEDED(hr))
  616. {
  617. hr = (*ppUrlOut)->SetURL(szUrl, 0);
  618. if (S_OK == hr)
  619. {
  620. // Get the IDataObject and send that back for the Drag Drop
  621. hr = (*ppUrlOut)->QueryInterface(IID_PPV_ARG(IDataObject, ppdtobj));
  622. if (SUCCEEDED(hr))
  623. {
  624. IUnknown_SetSite(*ppUrlOut, pUnkSite); // Only set the site if we're sure of
  625. // returning SUCCESS
  626. }
  627. }
  628. }
  629. }
  630. else
  631. {
  632. hr = E_FAIL;
  633. }
  634. }
  635. if (FAILED(hr))
  636. {
  637. SAFERELEASE(*ppUrlOut);
  638. SAFERELEASE(*ppdtobj);
  639. }
  640. return hr;
  641. }
  642. HRESULT SendDocToMailRecipient(LPCITEMIDLIST pidl, UINT uiCodePage, DWORD grfKeyState, IUnknown *pUnkSite)
  643. {
  644. IDataObject *pdtobj = NULL;
  645. IUniformResourceLocator *purl = NULL;
  646. HRESULT hr = CreateShortcutSetSiteAndGetDataObjectIfPIDLIsNetUrl(pidl, pUnkSite, &purl, &pdtobj);
  647. if (FAILED(hr))
  648. {
  649. ASSERT(NULL == pdtobj);
  650. ASSERT(NULL == purl);
  651. hr = GetDataObjectForPidl(pidl, &pdtobj);
  652. }
  653. if (SUCCEEDED(hr))
  654. {
  655. IQueryCodePage * pQcp;
  656. if (SUCCEEDED(pdtobj->QueryInterface(IID_PPV_ARG(IQueryCodePage, &pQcp))))
  657. {
  658. pQcp->SetCodePage(uiCodePage);
  659. pQcp->Release();
  660. }
  661. hr = DropOnMailRecipient(pdtobj, grfKeyState);
  662. pdtobj->Release();
  663. }
  664. if (purl)
  665. {
  666. IUnknown_SetSite(purl, NULL);
  667. purl->Release();
  668. }
  669. return hr;
  670. }
  671. #ifdef DEBUG
  672. /****************************************************\
  673. FUNCTION: Dbg_PidlStr
  674. DESCRIPTION:
  675. Create a display name for the pidl passed in
  676. and store the display name in pszBuffer.
  677. \****************************************************/
  678. LPTSTR Dbg_PidlStr(LPCITEMIDLIST pidl, LPTSTR pszBuffer, DWORD cchBufferSize)
  679. {
  680. if (pidl)
  681. {
  682. if (ILIsRooted(pidl))
  683. StrCpyN(pszBuffer, TEXT("<ROOTED>"), cchBufferSize);
  684. else
  685. {
  686. IEGetNameAndFlags(pidl, SHGDN_FORPARSING, pszBuffer, cchBufferSize, NULL);
  687. }
  688. }
  689. else
  690. StrCpyN(pszBuffer, TEXT("<NULL>"), cchBufferSize);
  691. return pszBuffer;
  692. }
  693. #endif // DEBUG
  694. #ifdef DEBUG
  695. #define MAX_DEPTH 8
  696. void Dbg_RecursiveDumpMenu(HMENU hmenu, int iDepth)
  697. {
  698. if (!hmenu || iDepth > MAX_DEPTH)
  699. return;
  700. TCHAR szTabs[MAX_DEPTH + 1];
  701. for (int i = 0; i < iDepth; i++)
  702. {
  703. szTabs[i] = '\t';
  704. }
  705. szTabs[iDepth] = '\0';
  706. int cItems = GetMenuItemCount(hmenu);
  707. for (i = 0; i < cItems; i++)
  708. {
  709. MENUITEMINFO mii;
  710. mii.cbSize = sizeof(MENUITEMINFO);
  711. mii.fMask = MIIM_ID | MIIM_SUBMENU | MIIM_TYPE;
  712. TCHAR szTmp[64];
  713. mii.dwTypeData = szTmp;
  714. mii.cch = ARRAYSIZE(szTmp);
  715. if (GetMenuItemInfoWrap(hmenu, i, TRUE, &mii))
  716. {
  717. LPTSTR pszType;
  718. if (mii.fType == MFT_STRING && mii.dwTypeData)
  719. pszType = mii.dwTypeData;
  720. else
  721. pszType = TEXT("");
  722. TraceMsg(TF_ALWAYS, "%swID %x\tfType %x\t%s", szTabs, mii.wID, mii.fType, pszType);
  723. if (mii.hSubMenu)
  724. {
  725. Dbg_RecursiveDumpMenu(mii.hSubMenu, iDepth + 1);
  726. }
  727. }
  728. }
  729. }
  730. // FUNCTION: Dbg_DumpMenu
  731. //
  732. // walk hmenu & dump every item
  733. void Dbg_DumpMenu(LPCTSTR psz, HMENU hmenu)
  734. {
  735. if (IsFlagSet(g_dwDumpFlags, DF_DEBUGMENU))
  736. {
  737. TraceMsg(TF_ALWAYS, "Dumping hmenu %x (%s)", hmenu, psz);
  738. Dbg_RecursiveDumpMenu(hmenu, 0);
  739. TraceMsg(TF_ALWAYS, "End hmenu dump");
  740. }
  741. }
  742. #endif
  743. // evil evil evil. for browse only mode support. not the right way to do things
  744. STDAPI LookForDesktopIniText(IShellFolder *psf, LPCITEMIDLIST pidl, LPCTSTR pszKey, LPTSTR pszBuffer, DWORD cbSize);
  745. #define CLSID_SIZE 40
  746. HRESULT LoadHandler(const CLSID * pCLSID, LPCWSTR pszBuffer, REFIID riid, void **ppvObj)
  747. {
  748. ASSERT(pszBuffer);
  749. CLSID clsid;
  750. if (!pCLSID)
  751. {
  752. // find the extension first ....
  753. // REARCHITECT - Shouldn't this be PathFindExtension?
  754. // Otherwise we will get confused by "foo.bar\baz"
  755. LPCWSTR pszDot = StrRChrW(pszBuffer, NULL, WCHAR('.'));
  756. if (!pszDot)
  757. {
  758. return E_NOINTERFACE;
  759. }
  760. HKEY hKey;
  761. USES_CONVERSION;
  762. LONG lRes = RegOpenKey(HKEY_CLASSES_ROOT, W2CT(pszDot), &hKey);
  763. if (lRes != ERROR_SUCCESS)
  764. {
  765. return E_NOINTERFACE;
  766. }
  767. TCHAR szSubKey[CLSID_SIZE + 10];
  768. StrCpy(szSubKey, TEXT("shellex\\"));
  769. SHStringFromGUID(riid, szSubKey + 8, CLSID_SIZE);
  770. TCHAR szCLSID[CLSID_SIZE];
  771. DWORD cbSize = sizeof(szCLSID);
  772. DWORD dwType;
  773. // should we test for a value as well as a key ?
  774. lRes = SHGetValue(hKey, szSubKey, TEXT(""), &dwType, szCLSID, &cbSize);
  775. RegCloseKey(hKey);
  776. if (lRes != ERROR_SUCCESS || dwType != REG_SZ)
  777. {
  778. return E_NOINTERFACE;
  779. }
  780. if (!GUIDFromString(szCLSID, &clsid))
  781. {
  782. return E_NOINTERFACE;
  783. }
  784. pCLSID = &clsid;
  785. }
  786. ASSERT(pCLSID);
  787. IPersistFile *pFile;
  788. HRESULT hr = CoCreateInstance(*pCLSID, NULL, CLSCTX_INPROC_SERVER,
  789. IID_PPV_ARG(IPersistFile, &pFile));
  790. if (FAILED(hr))
  791. {
  792. return E_NOINTERFACE;
  793. }
  794. *ppvObj = NULL;
  795. hr = pFile->Load(pszBuffer, TRUE);
  796. if (SUCCEEDED(hr))
  797. {
  798. hr = pFile->QueryInterface(riid, ppvObj);
  799. }
  800. ATOMICRELEASE(pFile);
  801. return hr;
  802. }
  803. // routine used to make us think it really came from the right place....
  804. HRESULT FakeGetUIObjectOf(IShellFolder *psf, LPCITEMIDLIST pidl, UINT * prgfFlags, REFIID riid, void **ppvObj)
  805. {
  806. HRESULT hr = E_NOINTERFACE;
  807. if (WhichPlatform() == PLATFORM_INTEGRATED)
  808. {
  809. // we are on Nashville try the new mechanism first...
  810. hr = psf->GetUIObjectOf(NULL, 1, & pidl, riid, NULL, ppvObj);
  811. if (SUCCEEDED(hr))
  812. {
  813. return hr;
  814. }
  815. }
  816. // failure cases...
  817. if (riid == IID_IExtractImage || riid == IID_IExtractLogo || riid == IID_IQueryInfo)
  818. {
  819. // make sure this hacked up code is only executed for browser only release....
  820. // otherwise people will not register their stuff right and what a mess that will be.....
  821. if (WhichPlatform() == PLATFORM_INTEGRATED)
  822. {
  823. return hr;
  824. }
  825. // try the IconExtractor first ....
  826. IExtractIconA *pIcon;
  827. hr = psf->GetUIObjectOf(NULL, 1, &pidl, IID_X_PPV_ARG(IExtractIconA, NULL, &pIcon));
  828. if (SUCCEEDED(hr))
  829. {
  830. if (riid != IID_IQueryInfo)
  831. {
  832. hr = pIcon->QueryInterface(IID_IExtractLogo, ppvObj);
  833. ATOMICRELEASE(pIcon);
  834. if (SUCCEEDED(hr))
  835. return NOERROR;
  836. }
  837. else
  838. {
  839. hr = pIcon->QueryInterface(IID_IQueryInfo, ppvObj);
  840. ATOMICRELEASE(pIcon);
  841. //if someone is asking for an IQueryInfo, don't try giving them an IExtractImage
  842. return hr;
  843. }
  844. }
  845. // browser mode only hack so we can detect if we are asking for the normal logo or the wide one...
  846. LPCTSTR pszTag = TEXT("Logo");
  847. if (prgfFlags != NULL && *prgfFlags)
  848. {
  849. pszTag = TEXT("WideLogo");
  850. }
  851. TCHAR szBuffer[MAX_PATH];
  852. hr = LookForDesktopIniText(psf, pidl, pszTag, szBuffer, ARRAYSIZE(szBuffer));
  853. if (SUCCEEDED(hr))
  854. {
  855. // use IID_IExtractImage, this is the same interface as IExtractLogo, just IExtractLogo
  856. // allows us to restrict the things that show up in Logo View...
  857. USES_CONVERSION;
  858. hr = LoadHandler(NULL, T2W(szBuffer), IID_IExtractImage, ppvObj);
  859. }
  860. }
  861. return hr;
  862. }
  863. BOOL GetInfoTipEx(IShellFolder* psf, DWORD dwFlags, LPCITEMIDLIST pidl, LPTSTR pszText, int cchTextMax)
  864. {
  865. BOOL fRet = FALSE;
  866. *pszText = 0; // empty for failure
  867. if (pidl)
  868. {
  869. IQueryInfo *pqi;
  870. if (SUCCEEDED(psf->GetUIObjectOf(NULL, 1, &pidl, IID_X_PPV_ARG(IQueryInfo, NULL, &pqi))))
  871. {
  872. WCHAR *pwszTip;
  873. pqi->GetInfoTip(dwFlags, &pwszTip);
  874. if (pwszTip)
  875. {
  876. fRet = TRUE;
  877. SHUnicodeToTChar(pwszTip, pszText, cchTextMax);
  878. SHFree(pwszTip);
  879. }
  880. pqi->Release();
  881. }
  882. else if (SUCCEEDED(FakeGetUIObjectOf(psf, pidl, 0, IID_PPV_ARG(IQueryInfo, &pqi))))
  883. {
  884. WCHAR *pwszTip;
  885. pqi->GetInfoTip(0, &pwszTip);
  886. if (pwszTip)
  887. {
  888. fRet = TRUE;
  889. SHUnicodeToTChar(pwszTip, pszText, cchTextMax);
  890. SHFree(pwszTip);
  891. }
  892. pqi->Release();
  893. }
  894. }
  895. return fRet;
  896. }
  897. BOOL GetInfoTip(IShellFolder* psf, LPCITEMIDLIST pidl, LPTSTR pszText, int cchTextMax)
  898. {
  899. return GetInfoTipEx(psf, 0, pidl, pszText, cchTextMax);
  900. }
  901. #define MAX_CLASS 80 // From ..\shell32\fstreex.c
  902. BOOL IsBrowsableShellExt(LPCITEMIDLIST pidl)
  903. {
  904. DWORD cb;
  905. LPCTSTR pszExt;
  906. TCHAR szFile[MAX_PATH];
  907. TCHAR szProgID[MAX_CLASS];
  908. TCHAR szCLSID[GUIDSTR_MAX], szCATID[GUIDSTR_MAX];
  909. TCHAR szKey[GUIDSTR_MAX * 4];
  910. HKEY hkeyProgID = NULL;
  911. BOOL fRet = FALSE;
  912. for (;;)
  913. {
  914. // Make sure we have a file extension
  915. if (
  916. !SHGetPathFromIDList(pidl, szFile)
  917. ||
  918. ((pszExt = PathFindExtension(szFile)) == NULL)
  919. ||
  920. (pszExt[0] != TEXT('.'))
  921. )
  922. {
  923. break;
  924. }
  925. // Get the ProgID.
  926. cb = sizeof(szProgID);
  927. if (
  928. (SHGetValue(HKEY_CLASSES_ROOT, pszExt, NULL, NULL, szProgID, &cb) != ERROR_SUCCESS)
  929. ||
  930. (RegOpenKey(HKEY_CLASSES_ROOT, szProgID, &hkeyProgID) != ERROR_SUCCESS)
  931. )
  932. {
  933. break;
  934. }
  935. // From the ProgID, get the CLSID.
  936. cb = sizeof(szCLSID);
  937. if (SHGetValue(hkeyProgID, TEXT("CLSID"), NULL, NULL, szCLSID, &cb) != ERROR_SUCCESS)
  938. break;
  939. // Construct the registry key that detects if
  940. // a CLSID is a member of a CATID.
  941. SHStringFromGUID(CATID_BrowsableShellExt, szCATID, ARRAYSIZE(szCATID));
  942. wnsprintf(szKey, ARRAYSIZE(szKey), TEXT("CLSID\\%s\\Implemented Categories\\%s"),
  943. szCLSID, szCATID);
  944. // See if it's there.
  945. cb = 0;
  946. if (SHGetValue(HKEY_CLASSES_ROOT, szKey, NULL, NULL, NULL, &cb) != ERROR_SUCCESS)
  947. break;
  948. fRet = TRUE;
  949. break;
  950. }
  951. if (hkeyProgID != NULL)
  952. RegCloseKey(hkeyProgID);
  953. return fRet;
  954. }
  955. void OpenFolderPidl(LPCITEMIDLIST pidl)
  956. {
  957. SHELLEXECUTEINFO shei = { 0 };
  958. shei.cbSize = sizeof(shei);
  959. shei.fMask = SEE_MASK_INVOKEIDLIST;
  960. shei.nShow = SW_SHOWNORMAL;
  961. shei.lpIDList = (LPITEMIDLIST)pidl;
  962. ShellExecuteEx(&shei);
  963. }
  964. void OpenFolderPath(LPCTSTR pszPath)
  965. {
  966. LPITEMIDLIST pidl = ILCreateFromPath(pszPath);
  967. if (pidl)
  968. {
  969. OpenFolderPidl(pidl);
  970. ILFree(pidl);
  971. }
  972. }
  973. // NOTE: this is only called from browseui, why is it in the lib directory?
  974. STDAPI UpdateSubscriptions()
  975. {
  976. #ifndef DISABLE_SUBSCRIPTIONS
  977. HRESULT hr;
  978. if (!SHRestricted2W(REST_NoManualUpdates, NULL, 0))
  979. {
  980. ISyncMgrSynchronizeInvoke *pSyncMgrInvoke;
  981. hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL,
  982. IID_PPV_ARG(ISyncMgrSynchronizeInvoke, &pSyncMgrInvoke));
  983. if (SUCCEEDED(hr))
  984. {
  985. hr = pSyncMgrInvoke->UpdateAll();
  986. pSyncMgrInvoke->Release();
  987. }
  988. }
  989. else
  990. {
  991. SHRestrictedMessageBox(NULL);
  992. hr = S_FALSE;
  993. }
  994. return hr;
  995. #else /* !DISABLE_SUBSCRIPTIONS */
  996. return E_FAIL;
  997. #endif /* !DISABLE_SUBSCRIPTIONS */
  998. }
  999. STDAPI_(int) _SHHandleUpdateImage(LPCITEMIDLIST pidlExtra)
  1000. {
  1001. SHChangeUpdateImageIDList * pUs = (SHChangeUpdateImageIDList*) pidlExtra;
  1002. if (!pUs)
  1003. {
  1004. return -1;
  1005. }
  1006. // if in the same process, or an old style notification
  1007. if (pUs->dwProcessID == GetCurrentProcessId())
  1008. {
  1009. return (int) pUs->iCurIndex;
  1010. }
  1011. else
  1012. {
  1013. WCHAR szBuffer[MAX_PATH];
  1014. int iIconIndex = *(int UNALIGNED *)((BYTE *)&pUs->iIconIndex);
  1015. UINT uFlags = *(UINT UNALIGNED *)((BYTE *)&pUs->uFlags);
  1016. ualstrcpyW(szBuffer, pUs->szName);
  1017. // we are in a different process, look up the hash in our index to get the right one...
  1018. return Shell_GetCachedImageIndex(szBuffer, iIconIndex, uFlags);
  1019. }
  1020. }
  1021. // As perf, share IShellLink implementations between bands and ask
  1022. // the bandsite for an implementation. Don't rely on the bandsite
  1023. // because you never know who will host us in the future. (And bandsite
  1024. // can change to not have us hosted at save/load time. Ex: it doesn't
  1025. // set our site before loading us from the stream, which sounds buggy.)
  1026. //
  1027. HRESULT SavePidlAsLink(IUnknown* punkSite, IStream *pstm, LPCITEMIDLIST pidl)
  1028. {
  1029. HRESULT hr = E_FAIL;
  1030. IShellLinkA* psl;
  1031. if (punkSite)
  1032. hr = IUnknown_QueryService(punkSite, IID_IBandSite, IID_PPV_ARG(IShellLinkA, &psl));
  1033. if (FAILED(hr))
  1034. hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IShellLinkA, &psl));
  1035. if (SUCCEEDED(hr))
  1036. {
  1037. IPersistStream *pps;
  1038. hr = psl->QueryInterface(IID_PPV_ARG(IPersistStream, &pps));
  1039. if (EVAL(SUCCEEDED(hr)))
  1040. {
  1041. ASSERT(pidl);
  1042. psl->SetIDList(pidl);
  1043. hr = pps->Save(pstm, FALSE);
  1044. // Win95 and NT4 shell32 have a bug in the CShellLink implementation
  1045. // THEY DON'T NULL TERMINATE THEIR "EXTRA DATA SECTION". This causes
  1046. // the object to trash the rest of the stream when it reads back in.
  1047. // Fix this by writing the NULL out in the Browser Only case.
  1048. if (SUCCEEDED(hr) && (PLATFORM_BROWSERONLY == WhichPlatform()))
  1049. {
  1050. DWORD dw = 0;
  1051. pstm->Write(&dw, sizeof(dw), NULL);
  1052. }
  1053. pps->Release();
  1054. }
  1055. psl->Release();
  1056. }
  1057. return hr;
  1058. }
  1059. HRESULT LoadPidlAsLink(IUnknown* punkSite, IStream *pstm, LPITEMIDLIST *ppidl)
  1060. {
  1061. IShellLinkA* psl;
  1062. HRESULT hr = IUnknown_QueryService(punkSite, IID_IBandSite, IID_PPV_ARG(IShellLinkA, &psl));
  1063. if (FAILED(hr))
  1064. hr = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IShellLinkA, &psl));
  1065. if (SUCCEEDED(hr))
  1066. {
  1067. IPersistStream *pps;
  1068. hr = psl->QueryInterface(IID_PPV_ARG(IPersistStream, &pps));
  1069. if (EVAL(SUCCEEDED(hr)))
  1070. {
  1071. hr = pps->Load(pstm);
  1072. if (EVAL(SUCCEEDED(hr)))
  1073. {
  1074. hr = psl->GetIDList(ppidl);
  1075. // Don't make me resolve the link because it's soo slow because
  1076. // it often loads 80k of networking dlls.
  1077. if (!EVAL(SUCCEEDED(hr)))
  1078. {
  1079. hr = psl->Resolve(NULL, SLR_NOUPDATE | SLR_NO_UI);
  1080. if (EVAL(SUCCEEDED(hr)))
  1081. hr = psl->GetIDList(ppidl);
  1082. }
  1083. hr = *ppidl ? S_OK : E_FAIL;
  1084. }
  1085. pps->Release();
  1086. }
  1087. psl->Release();
  1088. }
  1089. return hr;
  1090. }
  1091. // AdjustECPosition
  1092. //
  1093. // purpose: because FE NT always uses WCHAR position for ComboBoxEx32
  1094. // even though we're ANSI module for EM_GETSEL/EM_SETSEL,
  1095. // we need to adjust between WCHAR and TCHAR position.
  1096. // iType: ADJUST_TO_WCHAR_POS or ADJUST_TO_TCHAR_POS
  1097. //
  1098. int AdjustECPosition(char *psz, int iPos, int iType)
  1099. {
  1100. char *pstr = psz;
  1101. int iNewPos = iPos;
  1102. if (psz && g_fRunOnFE && g_fRunningOnNT)
  1103. {
  1104. if (ADJUST_TO_WCHAR_POS == iType)
  1105. {
  1106. iNewPos = 0;
  1107. while (*pstr && (pstr - psz != iPos))
  1108. {
  1109. pstr = CharNextA(pstr);
  1110. iNewPos++;
  1111. }
  1112. }
  1113. else if (ADJUST_TO_TCHAR_POS == iType)
  1114. {
  1115. while (*pstr && iPos--)
  1116. pstr = CharNextA(pstr);
  1117. iNewPos = (int)(pstr - psz);
  1118. }
  1119. }
  1120. return iNewPos;
  1121. }
  1122. int CALLBACK _CompareIDs(LPARAM p1, LPARAM p2, LPARAM psf)
  1123. {
  1124. HRESULT hr = ((IShellFolder*)psf)->CompareIDs(0, (LPITEMIDLIST)p1, (LPITEMIDLIST)p2);
  1125. //ASSERT(SUCCEEDED(hr))
  1126. return (short)HRESULT_CODE(hr);
  1127. }
  1128. HDPA GetSortedIDList(LPITEMIDLIST pidl)
  1129. {
  1130. HDPA hdpa = DPA_Create(4);
  1131. if (hdpa)
  1132. {
  1133. IShellFolder* psf;
  1134. if (SUCCEEDED(IEBindToObject(pidl, &psf)))
  1135. {
  1136. LPENUMIDLIST penum;
  1137. SHELLSTATE ss = {0};
  1138. SHGetSetSettings(&ss, SSF_SHOWALLOBJECTS, FALSE);
  1139. if (S_OK == IShellFolder_EnumObjects(psf, NULL,
  1140. ss.fShowAllObjects ? SHCONTF_FOLDERS | SHCONTF_INCLUDEHIDDEN : SHCONTF_FOLDERS,
  1141. &penum))
  1142. {
  1143. LPITEMIDLIST pidl;
  1144. ULONG celt;
  1145. while (penum->Next(1, &pidl, &celt) == S_OK && celt == 1)
  1146. {
  1147. if (DPA_AppendPtr(hdpa, pidl) == -1)
  1148. {
  1149. SHFree(pidl);
  1150. }
  1151. }
  1152. penum->Release();
  1153. }
  1154. DPA_Sort(hdpa, (PFNDPACOMPARE)_CompareIDs, (LPARAM)psf);
  1155. psf->Release();
  1156. }
  1157. }
  1158. return hdpa;
  1159. }
  1160. int DPA_SHFreeCallback(void * p, void * d)
  1161. {
  1162. SHFree((LPITEMIDLIST)p);
  1163. return 1;
  1164. }
  1165. void FreeSortedIDList(HDPA hdpa)
  1166. {
  1167. DPA_DestroyCallback(hdpa, (PFNDPAENUMCALLBACK)DPA_SHFreeCallback, 0);
  1168. hdpa = NULL;
  1169. }
  1170. /****************************************************\
  1171. FUNCTION: StrCmpIWithRoot
  1172. PARAMETERS:
  1173. szDispNameIn - Str to see if it is the same as the
  1174. Display Name of the Root ISF.
  1175. fTotalStrCmp - If TRUE, pszDispNameIn has to completely equal the
  1176. Root's Display Name to succeed. If FALSE, only the first part
  1177. of pszDispNameIn needs to compare to the Root's Display Name
  1178. for this function to return successful.
  1179. ppszCachedRoot (In/Out Optional) - If this function will be called more than
  1180. once, this function will cache the string and make it run
  1181. quicker. The first time this function is called, (*ppszCachedRoot)
  1182. needs to be NULL. This function will allocate and the caller
  1183. needs to call LocalFree() when it's no longer needed.
  1184. DESCRIPTION:
  1185. This function will get the Display Name of the Root ISF (Desktop) and
  1186. see if the first cchDispNameComp chars of szDispNameIn
  1187. match that display name. S_OK will be returned if TRUE, and
  1188. S_FALSE if not.
  1189. \****************************************************/
  1190. HRESULT StrCmpIWithRoot(LPCTSTR pszDispNameIn, BOOL fTotalStrCmp, LPTSTR * ppszCachedRoot)
  1191. {
  1192. HRESULT hr;
  1193. TCHAR szDispNameTemp[MAX_PATH];
  1194. LPTSTR pszDispName = szDispNameTemp;
  1195. ASSERT(IS_VALID_STRING_PTR(pszDispNameIn, -1));
  1196. ASSERT(NULL == ppszCachedRoot || IS_VALID_WRITE_PTR(ppszCachedRoot, LPTSTR));
  1197. // Did the caller supply the display name of the namespace root?
  1198. if ((!ppszCachedRoot) ||
  1199. (ppszCachedRoot && !*ppszCachedRoot))
  1200. {
  1201. MLLoadString(IDS_DESKTOP, szDispNameTemp, SIZECHARS(szDispNameTemp));
  1202. // Cache this guy?
  1203. if (ppszCachedRoot)
  1204. {
  1205. // Yes
  1206. *ppszCachedRoot = StrDup(szDispNameTemp);
  1207. if (!*ppszCachedRoot)
  1208. return E_OUTOFMEMORY;
  1209. }
  1210. }
  1211. if (ppszCachedRoot && *ppszCachedRoot)
  1212. pszDispName = *ppszCachedRoot;
  1213. // Do we want to compare the entire string or just the first part of it?
  1214. if (fTotalStrCmp)
  1215. hr = (0 == lstrcmpi(pszDispName, pszDispNameIn)) ? S_OK : S_FALSE; // Entire String
  1216. else if (ppszCachedRoot)
  1217. {
  1218. // Compare the first part of the string
  1219. DWORD cchDispNameComp = lstrlen(*ppszCachedRoot);
  1220. hr = (0 == StrCmpNI(pszDispName, pszDispNameIn, cchDispNameComp)) ? S_OK : S_FALSE;
  1221. }
  1222. else
  1223. {
  1224. hr = S_FALSE;
  1225. }
  1226. return hr;
  1227. }
  1228. /****************************************************\
  1229. FUNCTION: GetMRUEntry
  1230. PARAMETERS:
  1231. hKey - Pointer to Registry Key to retrieve MRU entries from.
  1232. dwMRUIndex - 0 based MRU Index to retrieve.
  1233. pszMRUEntry - Location to store MRU Entry string.
  1234. cchMRUEntry - Size of Buffer in characters.
  1235. DESCRIPTION:
  1236. This function will retrieve the MRU Entry specified
  1237. by dwMRUIndex.
  1238. \****************************************************/
  1239. HRESULT GetMRUEntry(HKEY hKey, DWORD dwMRUIndex, LPTSTR pszMRUEntry, DWORD cchMRUEntry, LPITEMIDLIST * ppidl)
  1240. {
  1241. HRESULT hr = S_OK;
  1242. TCHAR szValueName[15]; // big enough for "url99999"
  1243. ASSERT(hKey);
  1244. ASSERT(pszMRUEntry);
  1245. ASSERT(cchMRUEntry);
  1246. // make a value name a la "url1" (1-based for historical reasons)
  1247. wnsprintf(szValueName, ARRAYSIZE(szValueName), SZ_REGVAL_MRUENTRY, dwMRUIndex+1);
  1248. cchMRUEntry *= sizeof(TCHAR);
  1249. if (ERROR_SUCCESS != SHQueryValueEx(hKey, szValueName, NULL, NULL, (LPBYTE) pszMRUEntry, &cchMRUEntry))
  1250. {
  1251. pszMRUEntry[0] = TEXT('\0');
  1252. hr = E_FAIL;
  1253. }
  1254. return hr;
  1255. }
  1256. /*----------------------------------------------------------
  1257. Purpose: Gets a registry value that is User Specifc.
  1258. This will open HKEY_CURRENT_USER if it exists,
  1259. otherwise it will open HKEY_LOCAL_MACHINE.
  1260. Returns: DWORD containing success or error code.
  1261. Cond: --
  1262. */
  1263. LONG OpenRegUSKey(LPCTSTR lpSubKey, DWORD ulOptions, REGSAM samDesired, PHKEY phkResult)
  1264. {
  1265. DWORD dwRet = RegOpenKeyEx(HKEY_CURRENT_USER, lpSubKey, ulOptions, samDesired, phkResult);
  1266. if (ERROR_SUCCESS != dwRet)
  1267. dwRet = RegOpenKeyEx(HKEY_LOCAL_MACHINE, lpSubKey, ulOptions, samDesired, phkResult);
  1268. return dwRet;
  1269. }
  1270. typedef struct tagSTREAMHEADER
  1271. {
  1272. DWORD dwHeaderSize;
  1273. DWORD dwDataSize;
  1274. DWORD dwSignature;
  1275. DWORD dwVersion;
  1276. } STREAMHEADER;
  1277. /****************************************************\
  1278. FUNCTION: LoadStreamHeader
  1279. PARAMETERS:
  1280. dwSignature - The signature that the caller supports
  1281. dwStartVersion - The lowest version the caller chooses to support.
  1282. dwEndVersion - The highest version the caller chooses to support.
  1283. pdwSize (OUT) - The size that the caller should read.
  1284. pdwVersionOut (OUT) - The version found.
  1285. return - if S_OK, then the caller needs to read pdwSize bytes
  1286. of data.
  1287. if S_FALSE, then the caller should use default settings
  1288. and return S_OK.
  1289. DESCRIPTION:
  1290. This function see if the caller owns this
  1291. data in the stream. If the caller does own the
  1292. stream segment, the size and version will be returned.
  1293. If the caller doesn't own the stream segment then S_FALSE is
  1294. returned to indicate that the caller should use default
  1295. data. If the caller doesn't claim to support the version
  1296. found (because of the dwStartVersion-dwEndVersion range),
  1297. then S_FALSE is returned to indicate to use default values
  1298. and this function skips over that segment in the stream
  1299. so the next segment can be parsed.
  1300. \****************************************************/
  1301. HRESULT LoadStreamHeader(IStream *pstm,
  1302. DWORD dwSignature, // What version?
  1303. DWORD dwStartVersion, // What is the earlies version supported?
  1304. DWORD dwEndVersion, // What is the oldest version supported?
  1305. DWORD * pdwSize, // What is the size to read?
  1306. DWORD * pdwVersionOut) // What version was found in the stream?
  1307. {
  1308. HRESULT hr;
  1309. STREAMHEADER shHeader;
  1310. BOOL fNotOurs = FALSE;
  1311. BOOL fSkipData = FALSE;
  1312. hr = pstm->Read(&shHeader, sizeof(shHeader), NULL);
  1313. ASSERT(pdwSize && pdwVersionOut);
  1314. *pdwSize = 0;
  1315. *pdwVersionOut = 0;
  1316. if (SUCCEEDED(hr))
  1317. {
  1318. if (shHeader.dwHeaderSize != sizeof(shHeader))
  1319. fNotOurs = TRUE;
  1320. else if (shHeader.dwSignature != dwSignature)
  1321. fNotOurs = TRUE;
  1322. else if (shHeader.dwVersion < dwStartVersion)
  1323. fSkipData = TRUE;
  1324. else if (shHeader.dwVersion > dwEndVersion)
  1325. fSkipData = TRUE;
  1326. if (fNotOurs)
  1327. {
  1328. // It's not, so reset it so the next guy will be able to read correctly.
  1329. LARGE_INTEGER li;
  1330. li.LowPart = (DWORD)-(LONG)sizeof(shHeader);
  1331. li.HighPart = 0;
  1332. hr = pstm->Seek(li, STREAM_SEEK_CUR, NULL);
  1333. hr = S_FALSE; // Means caller should use default data.
  1334. }
  1335. // Do we want to skip the Data for this part of the stream?
  1336. if (fSkipData)
  1337. {
  1338. ASSERT(STREAMSIZE_UNKNOWN != shHeader.dwDataSize); // SERIOUS, we cannot skip over data because we don't know size.
  1339. if (STREAMSIZE_UNKNOWN != shHeader.dwDataSize)
  1340. {
  1341. // Yes. The caller cannot read in this data because the caller doesn't support
  1342. // this version of the data. Therefore, we skip past the data and return S_FALSE
  1343. // to indicate to the caller that default settings should be used.
  1344. LARGE_INTEGER li;
  1345. li.LowPart = shHeader.dwDataSize;
  1346. li.HighPart = 0;
  1347. hr = pstm->Seek(li, STREAM_SEEK_CUR, NULL);
  1348. hr = S_FALSE; // Means caller should use default data.
  1349. }
  1350. }
  1351. if (!fNotOurs && !fSkipData)
  1352. {
  1353. *pdwSize = shHeader.dwDataSize;
  1354. *pdwVersionOut = shHeader.dwVersion;
  1355. }
  1356. }
  1357. return hr;
  1358. }
  1359. /****************************************************\
  1360. FUNCTION: SaveStreamHeader
  1361. DESCRIPTION:
  1362. This function will save a StreamHeader to
  1363. the stream that will allow the caller to verify
  1364. if he/she owns the data the next time it's read in.
  1365. It will also support the ability to ignore old
  1366. or future versions of data.
  1367. \****************************************************/
  1368. HRESULT SaveStreamHeader(IStream *pstm, DWORD dwSignature, DWORD dwVersion, DWORD dwSize)
  1369. {
  1370. HRESULT hr;
  1371. STREAMHEADER shHeader;
  1372. shHeader.dwHeaderSize = sizeof(STREAMHEADER);
  1373. shHeader.dwDataSize = dwSize;
  1374. shHeader.dwSignature = dwSignature;
  1375. shHeader.dwVersion = dwVersion;
  1376. hr = pstm->Write(&shHeader, sizeof(shHeader), NULL);
  1377. return hr;
  1378. }
  1379. //----------------------------------------------------------------------
  1380. //
  1381. // CMenuList
  1382. //
  1383. //----------------------------------------------------------------------
  1384. typedef struct
  1385. {
  1386. HMENU hmenu;
  1387. BITBOOL bObject:1; // TRUE: menu belongs to object
  1388. } MLITEM; // CMenuList item
  1389. CMenuList::CMenuList(void)
  1390. {
  1391. ASSERT(NULL == _hdsa);
  1392. }
  1393. CMenuList::~CMenuList(void)
  1394. {
  1395. if (_hdsa)
  1396. {
  1397. DSA_Destroy(_hdsa);
  1398. _hdsa = NULL;
  1399. }
  1400. }
  1401. /*----------------------------------------------------------
  1402. Purpose: Set the menu list (comparable to HOLEMENU) so we can
  1403. dispatch commands to the frame or the object correctly.
  1404. We do this since menu bands bypass OLE's FrameFilterWndProc.
  1405. We build the menu list by comparing the given hmenuShared
  1406. with hmenuFrame. Anything in hmenuShared that is not
  1407. in hmenuFrame belongs to the object.
  1408. */
  1409. void CMenuList::Set(HMENU hmenuShared, HMENU hmenuFrame)
  1410. {
  1411. ASSERT(NULL == hmenuShared || IS_VALID_HANDLE(hmenuShared, MENU));
  1412. ASSERT(NULL == hmenuFrame || IS_VALID_HANDLE(hmenuFrame, MENU));
  1413. if (_hdsa)
  1414. {
  1415. ASSERT(IS_VALID_HANDLE(_hdsa, DSA));
  1416. DSA_DeleteAllItems(_hdsa);
  1417. }
  1418. else
  1419. _hdsa = DSA_Create(sizeof(MLITEM), 10);
  1420. if (_hdsa && hmenuShared && hmenuFrame)
  1421. {
  1422. int i;
  1423. int iFrame = 0;
  1424. int cmenu = GetMenuItemCount(hmenuShared);
  1425. int cmenuFrame = GetMenuItemCount(hmenuFrame);
  1426. BOOL bMatched;
  1427. int iSaveFrame;
  1428. int iHaveFrame = -1;
  1429. TCHAR sz[64];
  1430. TCHAR szFrame[64];
  1431. MENUITEMINFO miiFrame;
  1432. MENUITEMINFO mii;
  1433. MLITEM mlitem;
  1434. miiFrame.cbSize = sizeof(miiFrame);
  1435. miiFrame.hSubMenu = NULL;
  1436. mii.cbSize = sizeof(mii);
  1437. for (i = 0; i < cmenu; i++)
  1438. {
  1439. mii.cch = SIZECHARS(sz);
  1440. mii.fMask = MIIM_SUBMENU | MIIM_TYPE;
  1441. mii.dwTypeData = sz;
  1442. EVAL(GetMenuItemInfoWrap(hmenuShared, i, TRUE, &mii));
  1443. ASSERT(IS_VALID_HANDLE(mii.hSubMenu, MENU));
  1444. mlitem.hmenu = mii.hSubMenu;
  1445. iSaveFrame = iFrame;
  1446. bMatched = FALSE;
  1447. // DocObject might have dropped some of our menus, like edit and view
  1448. // Need to be able to skip over dropped frame menus
  1449. while (1)
  1450. {
  1451. if (iHaveFrame != iFrame)
  1452. {
  1453. iHaveFrame = iFrame;
  1454. if (iFrame < cmenuFrame)
  1455. {
  1456. miiFrame.cch = SIZECHARS(szFrame);
  1457. miiFrame.fMask = MIIM_SUBMENU | MIIM_TYPE;
  1458. miiFrame.dwTypeData = szFrame;
  1459. EVAL(GetMenuItemInfoWrap(hmenuFrame, iFrame, TRUE, &miiFrame));
  1460. }
  1461. else
  1462. {
  1463. // Make it so it won't compare
  1464. miiFrame.hSubMenu = NULL;
  1465. *szFrame = 0;
  1466. }
  1467. }
  1468. ASSERT(iFrame >= cmenuFrame || IS_VALID_HANDLE(miiFrame.hSubMenu, MENU));
  1469. // The browser may have a menu that was not merged into
  1470. // the shared menu because the object put one in with
  1471. // the same name. Have we hit this case? Check by comparing
  1472. // sz and szFrame
  1473. if (mii.hSubMenu == miiFrame.hSubMenu || 0 == StrCmp(sz, szFrame))
  1474. {
  1475. bMatched = TRUE;
  1476. break;
  1477. }
  1478. else
  1479. {
  1480. if (iFrame >= cmenuFrame)
  1481. {
  1482. break;
  1483. }
  1484. iFrame++;
  1485. }
  1486. }
  1487. // Is this one of our menus?
  1488. mlitem.bObject = (mii.hSubMenu == miiFrame.hSubMenu) ? FALSE:TRUE;
  1489. if (bMatched)
  1490. {
  1491. iFrame++;
  1492. }
  1493. else
  1494. {
  1495. iFrame = iSaveFrame;
  1496. }
  1497. DSA_SetItem(_hdsa, i, &mlitem);
  1498. }
  1499. }
  1500. }
  1501. /*----------------------------------------------------------
  1502. Purpose: Adds the given hmenu to the list.
  1503. */
  1504. void CMenuList::AddMenu(HMENU hmenu)
  1505. {
  1506. ASSERT(NULL == hmenu || IS_VALID_HANDLE(hmenu, MENU));
  1507. if (_hdsa && hmenu)
  1508. {
  1509. MLITEM mlitem;
  1510. mlitem.hmenu = hmenu;
  1511. mlitem.bObject = TRUE;
  1512. DSA_AppendItem(_hdsa, &mlitem);
  1513. }
  1514. }
  1515. /*----------------------------------------------------------
  1516. Purpose: Removes the given hmenu from the list.
  1517. */
  1518. void CMenuList::RemoveMenu(HMENU hmenu)
  1519. {
  1520. ASSERT(NULL == hmenu || IS_VALID_HANDLE(hmenu, MENU));
  1521. if (_hdsa && hmenu)
  1522. {
  1523. int i = DSA_GetItemCount(_hdsa) - 1;
  1524. for (; i >= 0; i--)
  1525. {
  1526. MLITEM * pmlitem = (MLITEM *)DSA_GetItemPtr(_hdsa, i);
  1527. ASSERT(pmlitem);
  1528. if (hmenu == pmlitem->hmenu)
  1529. {
  1530. DSA_DeleteItem(_hdsa, i);
  1531. break;
  1532. }
  1533. }
  1534. }
  1535. }
  1536. /*----------------------------------------------------------
  1537. Purpose: Returns TRUE if the given hmenu belongs to the object.
  1538. */
  1539. BOOL CMenuList::IsObjectMenu(HMENU hmenu)
  1540. {
  1541. BOOL bRet = FALSE;
  1542. ASSERT(NULL == hmenu || IS_VALID_HANDLE(hmenu, MENU));
  1543. if (_hdsa && hmenu)
  1544. {
  1545. int i;
  1546. for (i = 0; i < DSA_GetItemCount(_hdsa); i++)
  1547. {
  1548. MLITEM * pmlitem = (MLITEM *)DSA_GetItemPtr(_hdsa, i);
  1549. ASSERT(pmlitem);
  1550. if (hmenu == pmlitem->hmenu)
  1551. {
  1552. bRet = pmlitem->bObject;
  1553. break;
  1554. }
  1555. }
  1556. }
  1557. return bRet;
  1558. }
  1559. #ifdef DEBUG
  1560. void CMenuList::Dump(LPCTSTR pszMsg)
  1561. {
  1562. if (IsFlagSet(g_dwDumpFlags, DF_DEBUGMENU))
  1563. {
  1564. TraceMsg(TF_ALWAYS, "CMenuList: Dumping menus for %#08x %s", this, pszMsg);
  1565. if (_hdsa)
  1566. {
  1567. int i;
  1568. for (i = 0; i < DSA_GetItemCount(_hdsa); i++)
  1569. {
  1570. MLITEM * pmlitem = (MLITEM *)DSA_GetItemPtr(_hdsa, i);
  1571. ASSERT(pmlitem);
  1572. TraceMsg(TF_ALWAYS, " [%d] = %x", i, pmlitem->hmenu);
  1573. }
  1574. }
  1575. }
  1576. }
  1577. #endif
  1578. #define REGVAL_FIRST_HOME_PAGE TEXT("First Home Page")
  1579. #define REGVAL_UPDATE_CHECK_PAGE TEXT("Update_Check_Page")
  1580. #define REGVAL_UPDATE_CHECK_INTERVAL TEXT("Update_Check_Interval")
  1581. #define REGVAL_LASTCHECKEDHI TEXT("LastCheckedHi")
  1582. #define REGSTR_PATH_INFODEL_REST TEXT("Software\\Policies\\Microsoft\\Internet Explorer\\Infodelivery\\Restrictions")
  1583. #define REGVAL_IEUPDATECHECK_REST TEXT("NoUpdateCheck")
  1584. #define DEFAULT_IEUPDATECHECK_PAGE TEXT("http://www.microsoft.com/isapi/redir.dll?Prd=ie&Pver=5.0&Ar=ie5update&O1=b1")
  1585. BOOL
  1586. IsUpdateCheckRestricted()
  1587. {
  1588. HKEY hkeyRest = 0;
  1589. BOOL bUpdateCheckRest = FALSE;
  1590. DWORD dwValue = 0;
  1591. DWORD dwLen = sizeof(DWORD);
  1592. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_INFODEL_REST, 0, KEY_READ, &hkeyRest) == ERROR_SUCCESS)
  1593. {
  1594. if (RegQueryValueEx(hkeyRest, REGVAL_IEUPDATECHECK_REST, NULL, NULL,
  1595. (LPBYTE)&dwValue, &dwLen) == ERROR_SUCCESS && dwValue)
  1596. bUpdateCheckRest = TRUE;
  1597. RegCloseKey(hkeyRest);
  1598. }
  1599. if (!bUpdateCheckRest)
  1600. {
  1601. // Check to see if the user has turned it off under advanced options
  1602. dwValue = 0;
  1603. dwLen = sizeof(DWORD);
  1604. if (SHRegGetUSValue(REGSTR_PATH_MAIN, REGVAL_IEUPDATECHECK_REST, NULL, (LPBYTE)&dwValue, &dwLen, 0,NULL,0) == ERROR_SUCCESS && dwValue)
  1605. bUpdateCheckRest = TRUE;
  1606. }
  1607. return bUpdateCheckRest;
  1608. }
  1609. HRESULT
  1610. CheckIEMinimalUpdate()
  1611. {
  1612. HRESULT hr = S_OK;
  1613. HKEY hkeyIE = 0;
  1614. TCHAR szUpdateUrl[MAX_URL_STRING];
  1615. DWORD dwSize;
  1616. DWORD dwType;
  1617. FILETIME ftlast, ftnow;
  1618. DWORD dwMagicDays = 0;
  1619. DWORD dwMagicPerDay = 201;
  1620. if (IsUpdateCheckRestricted())
  1621. {
  1622. hr = HRESULT_FROM_WIN32(ERROR_ACCESS_DENIED);
  1623. goto Exit;
  1624. }
  1625. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, REGSTR_PATH_MAIN, 0, KEY_READ, &hkeyIE) != ERROR_SUCCESS)
  1626. {
  1627. hr = HRESULT_FROM_WIN32(GetLastError());
  1628. goto Exit;
  1629. }
  1630. dwType = REG_DWORD;
  1631. dwSize = sizeof(DWORD);
  1632. if ((RegQueryValueEx(hkeyIE, REGVAL_UPDATE_CHECK_INTERVAL, NULL, &dwType,
  1633. (unsigned char *)&dwMagicDays, &dwSize) != ERROR_SUCCESS) || dwMagicDays == 0)
  1634. {
  1635. dwMagicDays = 30; // hardcode default to check every 30 days.
  1636. }
  1637. StrCpy(szUpdateUrl, DEFAULT_IEUPDATECHECK_PAGE);
  1638. dwType = REG_SZ;
  1639. dwSize = MAX_URL_STRING;
  1640. if (RegQueryValueEx(hkeyIE, REGVAL_UPDATE_CHECK_PAGE, NULL, &dwType,
  1641. (LPBYTE)szUpdateUrl, &dwSize) != ERROR_SUCCESS)
  1642. {
  1643. // go to default page on micorosoft.com
  1644. //hr = HRESULT_FROM_WIN32(GetLastError());
  1645. //goto Exit;
  1646. }
  1647. RegCloseKey(hkeyIE);
  1648. if (RegOpenKeyEx(HKEY_CURRENT_USER, REGSTR_PATH_MAIN, 0, KEY_READ|KEY_SET_VALUE, &hkeyIE) == ERROR_SUCCESS)
  1649. {
  1650. dwType = REG_SZ;
  1651. dwSize = MAX_URL_STRING;
  1652. if (RegQueryValueEx(hkeyIE, REGVAL_FIRST_HOME_PAGE, NULL, &dwType,
  1653. NULL, &dwSize) == ERROR_SUCCESS)
  1654. {
  1655. // if already exists then skip this write
  1656. hr = S_FALSE;
  1657. goto Exit;
  1658. }
  1659. GetSystemTimeAsFileTime(&ftnow);
  1660. ftnow.dwLowDateTime = 0;
  1661. ZeroMemory(&ftlast, sizeof(ftlast));
  1662. dwType = REG_DWORD;
  1663. dwSize = sizeof(DWORD);
  1664. if (RegQueryValueEx(hkeyIE, REGVAL_LASTCHECKEDHI, NULL, &dwType,
  1665. (unsigned char *)&ftlast.dwHighDateTime, &dwSize)==ERROR_SUCCESS)
  1666. {
  1667. ftlast.dwHighDateTime += (dwMagicPerDay * dwMagicDays);
  1668. }
  1669. if (CompareFileTime(&ftlast, &ftnow) > 0)
  1670. {
  1671. hr = S_FALSE;
  1672. }
  1673. else
  1674. {
  1675. RegSetValueEx(hkeyIE,REGVAL_FIRST_HOME_PAGE, NULL,
  1676. REG_SZ,(LPBYTE)szUpdateUrl, (lstrlen(szUpdateUrl)+1)*sizeof(TCHAR));
  1677. RegSetValueEx(hkeyIE, REGVAL_LASTCHECKEDHI, NULL, REG_DWORD,
  1678. (unsigned char *)&ftnow.dwHighDateTime, sizeof(DWORD));
  1679. }
  1680. RegCloseKey(hkeyIE);
  1681. }
  1682. Exit:
  1683. return hr;
  1684. }
  1685. static BOOL s_fSUCheckComplete = FALSE;
  1686. // returns:
  1687. // TRUE The user clicked Update Now and we ShellExe'ed
  1688. // the update URL.
  1689. // FALSE We did not launch a browser to the update page.
  1690. // NOTE: the "run-once-ness" of this is controlled by the ICW check
  1691. // variable g_fICWCheckComplete.
  1692. BOOL CheckSoftwareUpdateUI(HWND hwndOwner, IShellBrowser *pisb)
  1693. {
  1694. BOOL fLaunchUpdate = FALSE;
  1695. #ifndef UNIX
  1696. HRESULT hr = S_OK;
  1697. int nRes;
  1698. SOFTDISTINFO sdi = { 0 };
  1699. sdi.cbSize = sizeof(SOFTDISTINFO);
  1700. if (s_fSUCheckComplete)
  1701. return FALSE;
  1702. else
  1703. s_fSUCheckComplete = TRUE;
  1704. // We're putting up a message box, so make the msg pump modal
  1705. pisb->EnableModelessSB(FALSE);
  1706. nRes = SoftwareUpdateMessageBox(hwndOwner, awchMSIE4GUID, 0, &sdi);
  1707. pisb->EnableModelessSB(TRUE);
  1708. if (nRes != IDABORT)
  1709. {
  1710. if (nRes == IDYES)
  1711. {
  1712. // Okay, we tried to do this a couple of different ways.
  1713. // Originally, this was done with ShellExecEx. This failed
  1714. // because the http hook wasn't 100% reliable on Win95.
  1715. // The next stab was to:
  1716. //LPITEMIDLIST pidl;
  1717. // The user wants to navigate to the install page.
  1718. //hr = pibs->IEParseDisplayName(CP_ACP, sdi.szHREF, &pidl);
  1719. //if (SUCCEEDED(hr))
  1720. //{
  1721. // OpenFolderPidl(pidl);
  1722. // ILFree(pidl);
  1723. //}
  1724. hr = NavToUrlUsingIEW(sdi.szHREF, TRUE);
  1725. } // if user wants update
  1726. if (sdi.szTitle != NULL)
  1727. CoTaskMemFree(sdi.szTitle);
  1728. if (sdi.szAbstract != NULL)
  1729. CoTaskMemFree(sdi.szAbstract);
  1730. if (sdi.szHREF != NULL)
  1731. CoTaskMemFree(sdi.szHREF);
  1732. fLaunchUpdate = nRes == IDYES && SUCCEEDED(hr);
  1733. }
  1734. if (!fLaunchUpdate)
  1735. {
  1736. // for minimal install of IE every N days or so we want to
  1737. // hijack the home page to check if an update is available
  1738. // for us.
  1739. CheckIEMinimalUpdate();
  1740. }
  1741. #endif
  1742. return fLaunchUpdate;
  1743. }
  1744. BOOL g_fICWCheckComplete = FALSE;
  1745. // returns:
  1746. // TRUE Internet Connection Wizard (ICW) was run, and we should exit
  1747. // the browser since we likely need to restart the system
  1748. // FALSE did not run the ICW, continue on as normal
  1749. BOOL CheckRunICW(LPCTSTR pszURL)
  1750. {
  1751. if (g_fICWCheckComplete)
  1752. return FALSE;
  1753. DWORD dwICWCompleted = 0;
  1754. BOOL fRet = FALSE;
  1755. // Check if ICW has already been run
  1756. DWORD dwSize = sizeof(dwICWCompleted);
  1757. SHGetValue(HKEY_CURRENT_USER, TEXT(ICW_REGPATHSETTINGS), TEXT(ICW_REGKEYCOMPLETED), NULL, &dwICWCompleted, &dwSize);
  1758. if (!dwICWCompleted)
  1759. {
  1760. HINSTANCE hInetCfgDll = LoadLibrary(TEXT("inetcfg.dll"));
  1761. // set this to TRUE here so that if there's an error in loading the dll, or getting the proc address,
  1762. // we don't keep trying to do that.
  1763. g_fICWCheckComplete = TRUE;
  1764. if (hInetCfgDll)
  1765. {
  1766. PFNCHECKCONNECTIONWIZARD fp = (PFNCHECKCONNECTIONWIZARD)GetProcAddress(hInetCfgDll, "CheckConnectionWizard");
  1767. if (fp)
  1768. {
  1769. DWORD dwRet;
  1770. DWORD dwFlags = ICW_LAUNCHFULL | ICW_LAUNCHMANUAL | ICW_FULL_SMARTSTART;
  1771. if (pszURL)
  1772. {
  1773. PFNSETSHELLNEXT fpSetShellNext = (PFNSETSHELLNEXT)GetProcAddress(hInetCfgDll, "SetShellNext");
  1774. if (fpSetShellNext)
  1775. {
  1776. CHAR szAnsiUrl[MAX_URL_STRING];
  1777. SHTCharToAnsi(pszURL, szAnsiUrl, ARRAYSIZE(szAnsiUrl));
  1778. dwFlags |= ICW_USE_SHELLNEXT;
  1779. fpSetShellNext(szAnsiUrl);
  1780. }
  1781. }
  1782. // if we get this far, set the fICWCheckComplete back to FALSE (had to be false since we didn't early out)
  1783. // and let the ICW set the reg key. this is so that if the user decides to cancel and come back later,
  1784. // we respect that.
  1785. g_fICWCheckComplete = FALSE;
  1786. // Launch ICW full or manual path, whichever is available
  1787. // NOTE: the ICW code makes sure only a single instance is up
  1788. fp(dwFlags, &dwRet);
  1789. // If it was launched successfully, we need to exit
  1790. // since ICW may restart the machine if it needs to
  1791. // install system files.
  1792. if (dwRet & (ICW_LAUNCHEDFULL | ICW_LAUNCHEDMANUAL))
  1793. {
  1794. fRet = TRUE;
  1795. }
  1796. }
  1797. FreeLibrary(hInetCfgDll);
  1798. }
  1799. }
  1800. else
  1801. {
  1802. g_fICWCheckComplete = TRUE;
  1803. }
  1804. return fRet;
  1805. }
  1806. int GetColorComponent(LPSTR *ppsz)
  1807. {
  1808. int iColor = 0;
  1809. if (*ppsz)
  1810. {
  1811. LPSTR pBuf = *ppsz;
  1812. iColor = StrToIntA(pBuf);
  1813. // find the next comma
  1814. while(pBuf && *pBuf && *pBuf!=L',')
  1815. pBuf++;
  1816. // if valid and not NULL...
  1817. if (pBuf && *pBuf)
  1818. pBuf++; // increment
  1819. *ppsz = pBuf;
  1820. }
  1821. return iColor;
  1822. }
  1823. // Read the registry for a string (REG_SZ) of comma separated RGB values
  1824. COLORREF RegGetColorRefString(HKEY hkey, LPTSTR RegValue, COLORREF Value)
  1825. {
  1826. CHAR SmallBuf[80];
  1827. CHAR szRegKey[MAXIMUM_SUB_KEY_LENGTH];
  1828. LPSTR pszBuf;
  1829. DWORD cb;
  1830. int iRed, iGreen, iBlue;
  1831. SHTCharToAnsi(RegValue, szRegKey, ARRAYSIZE(szRegKey));
  1832. cb = sizeof(SmallBuf);
  1833. if (SHQueryValueExA(hkey, szRegKey, NULL, NULL, (LPBYTE)&SmallBuf, &cb)
  1834. == ERROR_SUCCESS)
  1835. {
  1836. pszBuf = SmallBuf;
  1837. iRed = GetColorComponent(&pszBuf);
  1838. iGreen = GetColorComponent(&pszBuf);
  1839. iBlue = GetColorComponent(&pszBuf);
  1840. // make sure all values are valid
  1841. iRed %= 256;
  1842. iGreen %= 256;
  1843. iBlue %= 256;
  1844. Value = RGB(iRed, iGreen, iBlue);
  1845. }
  1846. return Value;
  1847. }
  1848. LRESULT SetHyperlinkCursor(IShellFolder* pShellFolder, LPCITEMIDLIST pidl)
  1849. {
  1850. HCURSOR hCursor;
  1851. BOOL fCursorSet = FALSE;
  1852. if (!pidl)
  1853. return 0;
  1854. if (SHIsGlobalOffline())
  1855. {
  1856. IQueryInfo *pqi;
  1857. if (SUCCEEDED(pShellFolder->GetUIObjectOf(NULL, 1, &pidl, IID_X_PPV_ARG(IQueryInfo, NULL, &pqi))))
  1858. {
  1859. DWORD dwFlags = 0;
  1860. if (SUCCEEDED(pqi->GetInfoFlags(&dwFlags)))
  1861. {
  1862. if (0 == (dwFlags & QIF_CACHED))
  1863. {
  1864. // Load Offline cursor since not cached
  1865. hCursor = (HCURSOR)LoadCursor(HINST_THISDLL, MAKEINTRESOURCE(IDC_OFFLINE_HAND));
  1866. if (hCursor)
  1867. {
  1868. SetCursor(hCursor);
  1869. fCursorSet = TRUE;
  1870. }
  1871. }
  1872. }
  1873. pqi->Release();
  1874. }
  1875. }
  1876. if (!fCursorSet)
  1877. {
  1878. // For whatever reason, offline cursor was not set
  1879. hCursor = LoadHandCursor(0);
  1880. if (hCursor)
  1881. SetCursor(hCursor);
  1882. }
  1883. return 1;
  1884. }
  1885. BOOL IsSubscribableA(LPCSTR pszUrl)
  1886. {
  1887. // REARCHITECT: this should be method on the subscription mgr interface - zekel
  1888. DWORD dwScheme = GetUrlSchemeA(pszUrl);
  1889. return (dwScheme == URL_SCHEME_HTTP) || (dwScheme == URL_SCHEME_HTTPS);
  1890. }
  1891. BOOL IsSubscribableW(LPCWSTR pwzUrl)
  1892. {
  1893. // REARCHITECT: this should be method on the subscription mgr interface - zekel
  1894. DWORD dwScheme = GetUrlSchemeW(pwzUrl);
  1895. return (dwScheme == URL_SCHEME_HTTP) || (dwScheme == URL_SCHEME_HTTPS);
  1896. }
  1897. HWND GetTrayWindow()
  1898. {
  1899. #ifndef UNIX
  1900. static HWND s_hwndTray = NULL;
  1901. if (!IsWindow(s_hwndTray))
  1902. {
  1903. s_hwndTray = FindWindow(TEXT(WNDCLASS_TRAYNOTIFY), NULL);
  1904. }
  1905. return s_hwndTray;
  1906. #else
  1907. return NULL;
  1908. #endif
  1909. }
  1910. void FireEventSzA(LPCSTR szEvent)
  1911. {
  1912. HANDLE hEvent = OpenEventA(EVENT_MODIFY_STATE, FALSE, szEvent);
  1913. if (hEvent)
  1914. {
  1915. SetEvent(hEvent);
  1916. CloseHandle(hEvent);
  1917. }
  1918. }
  1919. void FireEventSzW(LPCWSTR pszEvent)
  1920. {
  1921. USES_CONVERSION;
  1922. FireEventSzA(W2A(pszEvent));
  1923. }
  1924. BOOL IsNamedWindow(HWND hwnd, LPCTSTR pszClass)
  1925. {
  1926. #ifndef UNIX
  1927. TCHAR szClass[32];
  1928. #else // UNIX use this function for trident dialog window
  1929. TCHAR szClass[64];
  1930. #endif
  1931. GetClassName(hwnd, szClass, ARRAYSIZE(szClass));
  1932. return lstrcmp(szClass, pszClass) == 0;
  1933. }
  1934. BOOL IsExplorerWindow(HWND hwnd)
  1935. {
  1936. return IsNamedWindow(hwnd, c_szExploreClass);
  1937. }
  1938. BOOL IsFolderWindow(HWND hwnd)
  1939. {
  1940. TCHAR szClass[32];
  1941. GetClassName(hwnd, szClass, ARRAYSIZE(szClass));
  1942. return (lstrcmp(szClass, c_szCabinetClass) == 0) || (lstrcmp(szClass, c_szIExploreClass) == 0);
  1943. }
  1944. // returns TRUE if the unknown is on a window that was opened as IE
  1945. // returns FALSE if the window was opened on the shell namespace, even if it's now showing a web page
  1946. // returns FALSE in other cases e.g. on the taskbar
  1947. STDAPI_(BOOL) WasOpenedAsBrowser(IUnknown *punkSite)
  1948. {
  1949. // this is a more reliable way of distinguishing windows opened for a URL. Checking
  1950. // the hwnd's classname does not work -- clicking on a hyperlink from Outlook 98 opens
  1951. // a browser window with a shell window's classname.
  1952. return (S_OK == IUnknown_QueryServiceExec(punkSite, SID_STopLevelBrowser, &CGID_Explorer, SBCMDID_STARTEDFORINTERNET, 0, NULL, NULL));
  1953. }
  1954. #define DXTRACK 1
  1955. void FrameTrack(HDC hdc, LPRECT prc, UINT uFlags)
  1956. {
  1957. COLORREF clrSave, clr;
  1958. RECT rc;
  1959. // upperleft
  1960. switch (uFlags)
  1961. {
  1962. case TRACKHOT:
  1963. clr = GetSysColor(COLOR_BTNHILIGHT);
  1964. break;
  1965. case TRACKNOCHILD:
  1966. case TRACKEXPAND:
  1967. clr = GetSysColor(COLOR_BTNSHADOW);
  1968. break;
  1969. default:
  1970. ASSERT(FALSE);
  1971. break;
  1972. }
  1973. clrSave = SetBkColor(hdc, clr);
  1974. rc = *prc;
  1975. rc.bottom = rc.top + DXTRACK;
  1976. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  1977. rc.bottom = prc->bottom;
  1978. rc.right = rc.left + DXTRACK;
  1979. rc.top = prc->top + DXTRACK;
  1980. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  1981. // lowerright
  1982. switch (uFlags)
  1983. {
  1984. case TRACKHOT:
  1985. clr = GetSysColor(COLOR_BTNSHADOW);
  1986. break;
  1987. case TRACKNOCHILD:
  1988. case TRACKEXPAND:
  1989. clr = GetSysColor(COLOR_BTNHILIGHT);
  1990. break;
  1991. default:
  1992. ASSERT(FALSE);
  1993. break;
  1994. }
  1995. SetBkColor(hdc, clr);
  1996. if (uFlags & (TRACKHOT | TRACKNOCHILD))
  1997. {
  1998. rc.right = prc->right;
  1999. rc.top = rc.bottom - DXTRACK;
  2000. rc.left = prc->left;
  2001. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  2002. }
  2003. rc.right = prc->right;
  2004. rc.left = prc->right - DXTRACK;
  2005. rc.top = prc->top;
  2006. rc.bottom = prc->bottom;
  2007. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  2008. SetBkColor(hdc, clrSave);
  2009. return;
  2010. }
  2011. #ifdef DEBUG // {
  2012. //*** SearchDW -- scan for DWORD in buffer
  2013. // ENTRY/EXIT
  2014. // pdwBuf buffer
  2015. // cbBuf size of buffer in *bytes* (*not* DWORDs)
  2016. // dwVal DWORD we're looking for
  2017. // dOff (return) byte offset in buffer; o.w. -1 if not found
  2018. //
  2019. int SearchDWP(DWORD_PTR *pdwBuf, int cbBuf, DWORD_PTR dwVal)
  2020. {
  2021. int dOff;
  2022. for (dOff = 0; dOff < cbBuf; dOff += sizeof(DWORD_PTR), pdwBuf++)
  2023. {
  2024. if (*pdwBuf == dwVal)
  2025. return dOff;
  2026. }
  2027. return -1;
  2028. }
  2029. #endif // }
  2030. int CAssociationList::FindEntry(DWORD dwKey)
  2031. {
  2032. if (_hdsa)
  2033. {
  2034. for (int i = 0; i < DSA_GetItemCount(_hdsa); i++)
  2035. {
  2036. ASSOCDATA* pad;
  2037. pad = (ASSOCDATA*)DSA_GetItemPtr(_hdsa, i);
  2038. if (pad->dwKey == dwKey)
  2039. return i;
  2040. }
  2041. }
  2042. return -1;
  2043. }
  2044. HRESULT CAssociationList::Find(DWORD dwKey, void ** ppData)
  2045. {
  2046. HRESULT hr = E_FAIL;
  2047. ENTERCRITICAL;
  2048. int i = FindEntry(dwKey);
  2049. if (i != -1)
  2050. {
  2051. ASSOCDATA* pad = (ASSOCDATA*)DSA_GetItemPtr(_hdsa, i);
  2052. ASSERT(dwKey == pad->dwKey);
  2053. *ppData = pad->lpData;
  2054. hr = S_OK;
  2055. }
  2056. LEAVECRITICAL;
  2057. return hr;
  2058. }
  2059. void CAssociationList::Delete(DWORD dwKey)
  2060. {
  2061. ENTERCRITICAL;
  2062. int i = FindEntry(dwKey);
  2063. if (i != -1)
  2064. {
  2065. DSA_DeleteItem(_hdsa, i);
  2066. }
  2067. LEAVECRITICAL;
  2068. }
  2069. BOOL CAssociationList::Add(DWORD dwKey, void *lpData)
  2070. {
  2071. ENTERCRITICAL;
  2072. if (!_hdsa)
  2073. {
  2074. _hdsa = DSA_Create(sizeof(ASSOCDATA), 4);
  2075. }
  2076. LEAVECRITICAL;
  2077. BOOL fRet = FALSE;
  2078. if (_hdsa)
  2079. {
  2080. ASSOCDATA ad;
  2081. ad.dwKey = dwKey;
  2082. ad.lpData = lpData;
  2083. ENTERCRITICAL;
  2084. fRet = DSA_AppendItem(_hdsa, &ad) != -1;
  2085. LEAVECRITICAL;
  2086. }
  2087. return fRet;
  2088. }
  2089. int g_cxSmIcon = 0;
  2090. int g_cySmIcon = 0;
  2091. HIMAGELIST g_himlSysSmall = NULL;
  2092. void _InitSmallImageList()
  2093. {
  2094. if (!g_himlSysSmall)
  2095. {
  2096. Shell_GetImageLists(NULL, &g_himlSysSmall);
  2097. ImageList_GetIconSize(g_himlSysSmall, &g_cxSmIcon, &g_cySmIcon);
  2098. }
  2099. }
  2100. #define CXIMAGEGAP 6
  2101. STDAPI_(void) DrawMenuItem(DRAWITEMSTRUCT* lpdi, LPCTSTR lpszMenuText, UINT iIcon)
  2102. {
  2103. _InitSmallImageList();
  2104. if ((lpdi->itemAction & ODA_SELECT) || (lpdi->itemAction & ODA_DRAWENTIRE))
  2105. {
  2106. int x, y;
  2107. SIZE sz;
  2108. RECT rc;
  2109. // Draw the image (if there is one).
  2110. GetTextExtentPoint32(lpdi->hDC, lpszMenuText, lstrlen(lpszMenuText), &sz);
  2111. if (lpdi->itemState & ODS_SELECTED)
  2112. {
  2113. SetBkColor(lpdi->hDC, GetSysColor(COLOR_HIGHLIGHT));
  2114. SetTextColor(lpdi->hDC, GetSysColor(COLOR_HIGHLIGHTTEXT));
  2115. FillRect(lpdi->hDC,&lpdi->rcItem,GetSysColorBrush(COLOR_HIGHLIGHT));
  2116. }
  2117. else
  2118. {
  2119. SetTextColor(lpdi->hDC, GetSysColor(COLOR_MENUTEXT));
  2120. FillRect(lpdi->hDC,&lpdi->rcItem,GetSysColorBrush(COLOR_MENU));
  2121. }
  2122. rc = lpdi->rcItem;
  2123. rc.left += +2 * CXIMAGEGAP + g_cxSmIcon;
  2124. DrawText(lpdi->hDC, lpszMenuText, lstrlen(lpszMenuText), &rc, DT_SINGLELINE | DT_VCENTER | DT_EXPANDTABS);
  2125. if (iIcon != -1)
  2126. {
  2127. x = lpdi->rcItem.left + CXIMAGEGAP;
  2128. y = (lpdi->rcItem.bottom + lpdi->rcItem.top - g_cySmIcon) / 2;
  2129. ImageList_Draw(g_himlSysSmall, iIcon, lpdi->hDC, x, y, ILD_TRANSPARENT);
  2130. }
  2131. else
  2132. {
  2133. x = lpdi->rcItem.left + CXIMAGEGAP;
  2134. y = (lpdi->rcItem.bottom + lpdi->rcItem.top - g_cySmIcon) / 2;
  2135. }
  2136. }
  2137. }
  2138. STDAPI_(LRESULT) MeasureMenuItem(MEASUREITEMSTRUCT *lpmi, LPCTSTR lpszMenuText)
  2139. {
  2140. LRESULT lres = FALSE;
  2141. if (0 == g_cxSmIcon)
  2142. {
  2143. _InitSmallImageList();
  2144. }
  2145. // Get the rough height of an item so we can work out when to break the
  2146. // menu. User should really do this for us but that would be useful.
  2147. HDC hdc = GetDC(NULL);
  2148. if (hdc)
  2149. {
  2150. // REVIEW cache out the menu font?
  2151. NONCLIENTMETRICSA ncm;
  2152. ncm.cbSize = sizeof(ncm);
  2153. if (SystemParametersInfoA(SPI_GETNONCLIENTMETRICS, sizeof(ncm), &ncm, FALSE))
  2154. {
  2155. HFONT hfont = CreateFontIndirectA(&ncm.lfMenuFont);
  2156. if (hfont)
  2157. {
  2158. SIZE sz;
  2159. HFONT hfontOld = (HFONT)SelectObject(hdc, hfont);
  2160. GetTextExtentPoint32(hdc, lpszMenuText, lstrlen(lpszMenuText), &sz);
  2161. lpmi->itemHeight = max (g_cySmIcon+CXIMAGEGAP/2, ncm.iMenuHeight);
  2162. lpmi->itemWidth = g_cxSmIcon + 2*CXIMAGEGAP + sz.cx;
  2163. SelectObject(hdc, hfontOld);
  2164. DeleteObject(hfont);
  2165. lres = TRUE;
  2166. }
  2167. }
  2168. ReleaseDC(NULL, hdc);
  2169. }
  2170. return lres;
  2171. }
  2172. //+-------------------------------------------------------------------------
  2173. // This function scans the document for the given HTML tag and returns the
  2174. // result in a collection.
  2175. //--------------------------------------------------------------------------
  2176. HRESULT GetDocumentTags
  2177. (
  2178. IHTMLDocument2 * pHTMLDocument, // doc to search
  2179. LPOLESTR pszTagName, // tag name to search for
  2180. IHTMLElementCollection ** ppTagsCollection // returned collection
  2181. )
  2182. {
  2183. HRESULT hr;
  2184. *ppTagsCollection = NULL;
  2185. //
  2186. // First get all document elements
  2187. //
  2188. IHTMLElementCollection * pAllCollection;
  2189. if (SUCCEEDED(hr = pHTMLDocument->get_all(&pAllCollection)))
  2190. {
  2191. //
  2192. // Now get all the elements with tags == pszTagName
  2193. //
  2194. VARIANT v;
  2195. v.vt = VT_BSTR;
  2196. v.bstrVal = ::SysAllocString(pszTagName);
  2197. if (v.bstrVal)
  2198. {
  2199. IDispatch * pDispTagsCollection;
  2200. if (SUCCEEDED(hr = pAllCollection->tags(v, &pDispTagsCollection)))
  2201. {
  2202. hr = pDispTagsCollection->QueryInterface(IID_PPV_ARG(IHTMLElementCollection, ppTagsCollection));
  2203. pDispTagsCollection->Release();
  2204. }
  2205. pAllCollection->Release();
  2206. VariantClear(&v);
  2207. }
  2208. }
  2209. return hr;
  2210. }
  2211. // This function uses the memory allocator from comctrl (which differs between NT and W95)
  2212. BOOL WINAPI Str_SetPtrPrivateW(WCHAR FAR * UNALIGNED * ppwzCurrent, LPCWSTR pwzNew)
  2213. {
  2214. LPWSTR pwzNewCopy = NULL;
  2215. if (pwzNew)
  2216. {
  2217. pwzNewCopy = StrDup(pwzNew);
  2218. if (!pwzNewCopy)
  2219. return FALSE;
  2220. }
  2221. LPWSTR pwzOld = (LPWSTR)InterlockedExchangePointer((void * *)ppwzCurrent, (void *)pwzNewCopy);
  2222. if (pwzOld)
  2223. LocalFree(pwzOld);
  2224. return TRUE;
  2225. }
  2226. // This function is compatible with API's that use LocalAlloc for string memory
  2227. BOOL WINAPI SetStr(WCHAR FAR * UNALIGNED * ppwzCurrent, LPCWSTR pwzNew)
  2228. {
  2229. int cchLength;
  2230. LPWSTR pwzOld;
  2231. LPWSTR pwzNewCopy = NULL;
  2232. if (pwzNew)
  2233. {
  2234. cchLength = lstrlenW(pwzNew);
  2235. // alloc a new buffer w/ room for the null terminator
  2236. pwzNewCopy = (LPWSTR)LocalAlloc(LPTR, (cchLength + 1) * sizeof(WCHAR));
  2237. if (!pwzNewCopy)
  2238. return FALSE;
  2239. StrCpyNW(pwzNewCopy, pwzNew, cchLength + 1);
  2240. }
  2241. pwzOld = (LPWSTR)InterlockedExchangePointer((void * *)ppwzCurrent, (void *)pwzNewCopy);
  2242. if (pwzOld)
  2243. LocalFree(pwzOld);
  2244. return TRUE;
  2245. }
  2246. //---------------------------------------------------------------------------
  2247. // If the string contains &ch or begins with ch then return TRUE.
  2248. BOOL _MenuCharMatch(LPCTSTR lpsz, TCHAR ch, BOOL fIgnoreAmpersand)
  2249. {
  2250. LPTSTR pchAS = StrChr(lpsz, TEXT('&')); // Find the first ampersand.
  2251. if (pchAS && !fIgnoreAmpersand)
  2252. {
  2253. // Yep, is the next char the one we want.
  2254. if (CharUpperChar(*CharNext(pchAS)) == CharUpperChar(ch))
  2255. {
  2256. // Yep.
  2257. return TRUE;
  2258. }
  2259. }
  2260. else if (CharUpperChar(*lpsz) == CharUpperChar(ch))
  2261. {
  2262. return TRUE;
  2263. }
  2264. return FALSE;
  2265. }
  2266. // Review chrisny: this can be moved into an object easily to handle generic droptarget, dropcursor
  2267. // , autoscrool, etc. . .
  2268. void _DragEnter(HWND hwndTarget, const POINTL ptStart, IDataObject *pdtObject)
  2269. {
  2270. RECT rc;
  2271. POINT pt;
  2272. GetWindowRect(hwndTarget, &rc);
  2273. //
  2274. // If hwndTarget is RTL mirrored, then measure the
  2275. // the client point from the visual right edge
  2276. // (near edge in RTL mirrored windows). [samera]
  2277. //
  2278. if (IS_WINDOW_RTL_MIRRORED(hwndTarget))
  2279. pt.x = rc.right - ptStart.x;
  2280. else
  2281. pt.x = ptStart.x - rc.left;
  2282. pt.y = ptStart.y - rc.top;
  2283. DAD_DragEnterEx2(hwndTarget, pt, pdtObject);
  2284. return;
  2285. }
  2286. void _DragMove(HWND hwndTarget, const POINTL ptStart)
  2287. {
  2288. RECT rc;
  2289. POINT pt;
  2290. GetWindowRect(hwndTarget, &rc);
  2291. //
  2292. // If hwndTarget is RTL mirrored, then measure the
  2293. // the client point from the visual right edge
  2294. // (near edge in RTL mirrored windows). [samera]
  2295. //
  2296. if (IS_WINDOW_RTL_MIRRORED(hwndTarget))
  2297. pt.x = rc.right - ptStart.x;
  2298. else
  2299. pt.x = ptStart.x - rc.left;
  2300. pt.y = ptStart.y - rc.top;
  2301. DAD_DragMove(pt);
  2302. return;
  2303. }
  2304. HRESULT CheckDesktopIni(LPCTSTR pszPath, LPCTSTR pszKey, LPTSTR pszBuffer, DWORD cbSize)
  2305. {
  2306. // NOTE:
  2307. // NOTE: DO NOT COPY THIS CODE. We only do this here for channels because we expect
  2308. // NOTE: the 99% case to be that it succeeds. If you need to find out if it is a
  2309. // NOTE: system folder, then you need to hack the pidl to get the system bit
  2310. // NOTE:
  2311. DWORD dwAttrs = GetFileAttributes(pszPath);
  2312. if (dwAttrs == (DWORD) -1 || !(dwAttrs & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_READONLY)))
  2313. return E_NOINTERFACE;
  2314. TCHAR szDIPath[MAX_PATH];
  2315. PathCombine(szDIPath, pszPath, TEXT("desktop.ini"));
  2316. if (pszKey == NULL)
  2317. {
  2318. if (GetFileAttributes(szDIPath) == (DWORD) -1)
  2319. {
  2320. return E_FAIL;
  2321. }
  2322. }
  2323. else
  2324. {
  2325. GetPrivateProfileString(TEXT(".ShellClassInfo"), pszKey, TEXT(""), pszBuffer, cbSize, szDIPath);
  2326. if (*pszBuffer == 0)
  2327. return E_NOINTERFACE;
  2328. // if its not a URL, then
  2329. if (!PathIsURL(pszBuffer))
  2330. {
  2331. PathCombine(pszBuffer, pszPath, pszBuffer);
  2332. }
  2333. }
  2334. return NOERROR;
  2335. }
  2336. STDAPI LookForDesktopIniText(IShellFolder *psf, LPCITEMIDLIST pidl, LPCTSTR pszKey, LPTSTR pszBuffer, DWORD cbSize)
  2337. {
  2338. TCHAR szPath[MAX_PATH];
  2339. DWORD ulFlags = SFGAO_FOLDER | SFGAO_FILESYSTEM;
  2340. HRESULT hr = GetPathForItem(psf, pidl, szPath, &ulFlags);
  2341. if (SUCCEEDED(hr) && (ulFlags & SFGAO_FOLDER))
  2342. {
  2343. hr = CheckDesktopIni(szPath, pszKey, pszBuffer, cbSize);
  2344. }
  2345. return hr;
  2346. }
  2347. // this is for channel category folders
  2348. HRESULT FakeGetNavigateTarget(IShellFolder *psf, LPCITEMIDLIST pidl, LPITEMIDLIST *ppidl)
  2349. {
  2350. HRESULT hres = E_FAIL;
  2351. WIN32_FIND_DATAA wfd;
  2352. // before looking for a desktop.ini (which hits the disk), cheaply
  2353. // see if it's a system folder.
  2354. //
  2355. // SHGetDataFromIDListA returns E_INVALIDARG on IE4.0 shell32 and
  2356. // IE4.01 shell32. It is fixed in IE4.01qfe shell32 and IE4.01sp1
  2357. // shell32. It also appears to work on NT4 and W95 shell32. If
  2358. // SHGetDataFromIDListA returns E_INVALIDARG we drop through and
  2359. // do the slow LookForDesktopIniText call.
  2360. //
  2361. HRESULT hresTemp = SHGetDataFromIDListA(psf, pidl, SHGDFIL_FINDDATA, &wfd, sizeof(wfd));
  2362. // on win95 non integrated, only the A version is implemented
  2363. if ((E_INVALIDARG == hresTemp) ||
  2364. (SUCCEEDED(hresTemp) &&
  2365. (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) &&
  2366. (wfd.dwFileAttributes & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_READONLY))))
  2367. {
  2368. // final failure case, check to see if it is a system folder with a desktop.ini file. This is gross and slow,
  2369. // but this is only called when the user has navigated so we can be a little slower. Please do NOT COPY this
  2370. // code or DavidDS will get really really upset and perf will be bad.
  2371. TCHAR szBuffer[MAX_URL_STRING];
  2372. // this will check for SFGAO_FOLDER & SFGAO_FILESYSTEM first...
  2373. hres = LookForDesktopIniText(psf, pidl, TEXT("URL"), szBuffer, ARRAYSIZE(szBuffer));
  2374. if (SUCCEEDED(hres))
  2375. {
  2376. DWORD dwChar = ARRAYSIZE(szBuffer);
  2377. // this call uses a temp, so we can reuse the buffer...
  2378. hres = UrlCreateFromPath(szBuffer, szBuffer, &dwChar, 0);
  2379. if (SUCCEEDED(hres))
  2380. hres = IECreateFromPath(szBuffer, ppidl);
  2381. }
  2382. }
  2383. else
  2384. {
  2385. hres = E_FAIL;
  2386. }
  2387. return hres;
  2388. }
  2389. // Can we browse or navigate to this pidl? If not, need
  2390. BOOL ILIsBrowsable(LPCITEMIDLIST pidl, BOOL *pfIsFolder)
  2391. {
  2392. if (!pidl)
  2393. return FALSE;
  2394. DWORD dwAttributes = SFGAO_FOLDER | SFGAO_BROWSABLE;
  2395. HRESULT hr = IEGetAttributesOf(pidl, &dwAttributes);
  2396. if (pfIsFolder && SUCCEEDED(hr))
  2397. *pfIsFolder = dwAttributes & SFGAO_FOLDER;
  2398. return SUCCEEDED(hr) && (dwAttributes & (SFGAO_FOLDER | SFGAO_BROWSABLE));
  2399. }
  2400. // gets a target pidl given a name space item. typically this is a .lnk or .url file
  2401. //
  2402. // in:
  2403. // psf shell folder for item
  2404. // pidl item relative to psf, single level
  2405. //
  2406. // in/out
  2407. // pdwAttribs [optional] attributes mask to filter on (returned).
  2408. // must be initalized
  2409. //
  2410. //
  2411. // returns
  2412. // *ppidl the target pidl
  2413. // *pdwAttribs [optional] attributes of the source object
  2414. STDAPI SHGetNavigateTarget(IShellFolder *psf, LPCITEMIDLIST pidl,
  2415. LPITEMIDLIST *ppidl, DWORD *pdwAttribs)
  2416. {
  2417. ASSERT(IS_VALID_CODE_PTR(psf, IShellFolder));
  2418. ASSERT(IS_VALID_PIDL(pidl));
  2419. ASSERT(NULL == pdwAttribs || IS_VALID_WRITE_PTR(pdwAttribs, DWORD));
  2420. ASSERT(ILFindLastID(pidl) == pidl); // must be single level PIDL
  2421. *ppidl = NULL; // assume failure
  2422. DWORD dwAttribs = SFGAO_FILESYSTEM | SFGAO_FOLDER | SFGAO_LINK | SFGAO_BROWSABLE;
  2423. if (pdwAttribs)
  2424. dwAttribs |= *pdwAttribs;
  2425. HRESULT hres = psf->GetAttributesOf(1, &pidl, &dwAttribs);
  2426. if (SUCCEEDED(hres))
  2427. {
  2428. // first try the most efficient way
  2429. IShellLinkA *psl; // "A" so this works on Win95
  2430. hres = psf->GetUIObjectOf(NULL, 1, &pidl, IID_X_PPV_ARG(IShellLinkA, NULL, &psl));
  2431. if (SUCCEEDED(hres))
  2432. {
  2433. hres = psl->GetIDList(ppidl);
  2434. psl->Release();
  2435. }
  2436. // this is for .lnk and .url files that don't register properly
  2437. if (FAILED(hres) && (dwAttribs & (SFGAO_FILESYSTEM | SFGAO_LINK)) == (SFGAO_FILESYSTEM | SFGAO_LINK))
  2438. {
  2439. TCHAR szPath[MAX_PATH];
  2440. hres = GetPathForItem(psf, pidl, szPath, NULL);
  2441. if (SUCCEEDED(hres))
  2442. hres = GetLinkTargetIDList(szPath, NULL, 0, ppidl);
  2443. }
  2444. // .doc or .html. return the pidl for this.
  2445. // (fully qualify against the folder pidl)
  2446. if (FAILED(hres) && (dwAttribs & SFGAO_BROWSABLE))
  2447. {
  2448. LPITEMIDLIST pidlFolder;
  2449. hres = SHGetIDListFromUnk(psf, &pidlFolder);
  2450. if (SUCCEEDED(hres))
  2451. {
  2452. *ppidl = ILCombine(pidlFolder, pidl); // navigate to this thing...
  2453. hres = *ppidl ? S_OK : E_OUTOFMEMORY;
  2454. ILFree(pidlFolder);
  2455. }
  2456. }
  2457. // channel name space items on non integrated
  2458. if (FAILED(hres) && WhichPlatform() != PLATFORM_INTEGRATED)
  2459. {
  2460. IExtractIconA *pxicon; // Use IID_IExtractIconA so we work on W95.
  2461. hres = psf->GetUIObjectOf(NULL, 1, &pidl, IID_X_PPV_ARG(IExtractIconA, NULL, &pxicon));
  2462. if (SUCCEEDED(hres))
  2463. {
  2464. hres = pxicon->QueryInterface(IID_PPV_ARG(IShellLinkA, &psl));
  2465. if (SUCCEEDED(hres))
  2466. {
  2467. hres = psl->GetIDList(ppidl);
  2468. psl->Release();
  2469. }
  2470. pxicon->Release();
  2471. }
  2472. }
  2473. #ifdef ENABLE_CHANNEL_CATEGORY_FOLDERS
  2474. // channel category folders
  2475. if (FAILED(hres))
  2476. hres = FakeGetNavigateTarget(psf, pidl, ppidl);
  2477. #endif //ENABLE_CHANNEL_CATEGORY_FOLDERS
  2478. // Callers of SHGetNavigateTarget assume that the returned pidl
  2479. // is navigatable (SFGAO_FOLDER or SFGAO_BROWSER), which isn't
  2480. // the case for a link (it could be a link to an exe).
  2481. //
  2482. if (SUCCEEDED(hres) && !ILIsBrowsable(*ppidl, NULL))
  2483. {
  2484. ILFree(*ppidl);
  2485. *ppidl = NULL;
  2486. hres = E_FAIL;
  2487. }
  2488. if (SUCCEEDED(hres) && pdwAttribs)
  2489. *pdwAttribs = dwAttribs;
  2490. }
  2491. return hres;
  2492. }
  2493. BOOL CreateShortcutAndDoDragDropIfPIDLIsNetUrl(IOleCommandTarget *pcmdt, LPITEMIDLIST pidl, HWND hwnd)
  2494. {
  2495. IUniformResourceLocator *purl;
  2496. IDataObject *pdtobj;
  2497. HRESULT hr = CreateShortcutSetSiteAndGetDataObjectIfPIDLIsNetUrl(pidl, pcmdt, &purl, &pdtobj);
  2498. if (SUCCEEDED(hr))
  2499. {
  2500. ASSERT(pdtobj);
  2501. ASSERT(purl);
  2502. // REARCHITECT: we should be binding to the parent and getting the attributes
  2503. // to determine the allowed effects - like we do in DragDrop()
  2504. DWORD dwEffect = (DROPEFFECT_COPY | DROPEFFECT_LINK);
  2505. ::_SetPreferedDropEffect(pdtobj, DROPEFFECT_LINK);
  2506. // Win95 Browser Only - the shell32 in this process doesn't know
  2507. // ole is loaded, even though it is.
  2508. SHLoadOLE(SHELLNOTIFY_OLELOADED);
  2509. hr = SHDoDragDrop(hwnd, pdtobj, NULL, dwEffect, &dwEffect);
  2510. // the returned value is not S_OK even tho' the drag drop succeeded
  2511. // however it is a success return
  2512. if (SUCCEEDED(hr))
  2513. {
  2514. // Since drag drop succeeded
  2515. // Bring down the icon for this shortcut
  2516. IUnknown_Exec(purl, &CGID_ShortCut, ISHCUTCMDID_DOWNLOADICON, 0, NULL, NULL);
  2517. }
  2518. pdtobj->Release();
  2519. IUnknown_SetSite(purl, NULL);
  2520. purl->Release();
  2521. }
  2522. return SUCCEEDED(hr);
  2523. }
  2524. BOOL DoDragDropWithInternetShortcut(IOleCommandTarget *pcmdt, LPITEMIDLIST pidl, HWND hwnd)
  2525. {
  2526. BOOL fDragDropDone = CreateShortcutAndDoDragDropIfPIDLIsNetUrl(pcmdt, pidl, hwnd);
  2527. if (FALSE == fDragDropDone)
  2528. {
  2529. // simply use PIDL and get none of the persistence effect
  2530. fDragDropDone = SUCCEEDED(DragDrop(hwnd, NULL, pidl, DROPEFFECT_LINK, NULL));
  2531. }
  2532. return fDragDropDone;
  2533. }
  2534. STDAPI_(HWND) GetTopLevelAncestor(HWND hWnd)
  2535. {
  2536. HWND hwndTemp;
  2537. while ((hwndTemp=GetParent(hWnd)) != NULL)
  2538. {
  2539. hWnd = hwndTemp;
  2540. }
  2541. return(hWnd);
  2542. }
  2543. #if 0
  2544. BOOL IsIERepairOn()
  2545. {
  2546. static DWORD dwChecked = -1;
  2547. if (dwChecked == -1)
  2548. {
  2549. DWORD dwSize, dwType;
  2550. // First check the OS setting. On NT5 and Win98-OSR, Repair is Off.
  2551. // OS turned Off Repair ==> "DisableRepair" RegValue is set to 1.
  2552. dwChecked = 1; // The default Repair is ON
  2553. dwSize = sizeof(dwChecked);
  2554. if (SHRegGetUSValue(SZ_REGKEY_ACTIVE_SETUP, SZ_REGVALUE_DISABLE_REPAIR, &dwType, (void *) &dwChecked, &dwSize, TRUE, (void *)NULL, 0) == ERROR_SUCCESS)
  2555. {
  2556. // OS Reg setting of 0 ==> Repair is ON
  2557. // OS Reg setting of 1 ==> Repair is OFF
  2558. dwChecked = (dwChecked == 0) ? 1 : 0;
  2559. }
  2560. else
  2561. {
  2562. dwChecked = 1; // if we fail to read Reg, go back to default.
  2563. }
  2564. // Check for Admin policy only if OS setting leaves Repair On.
  2565. if (dwChecked == 1)
  2566. {
  2567. dwSize = sizeof(dwChecked);
  2568. if (SHRegGetUSValue(SZ_REGKEY_IE_POLICIES, SZ_REGVALUE_IEREPAIR, &dwType, (void *) &dwChecked, &dwSize, TRUE, (void *)NULL, 0) != ERROR_SUCCESS)
  2569. {
  2570. dwChecked = 1; // if we fail to read Reg, go back to default.
  2571. }
  2572. }
  2573. }
  2574. return (dwChecked == 1);
  2575. }
  2576. #endif
  2577. BOOL IsResetWebSettingsEnabled(void)
  2578. {
  2579. static BOOL fUseCache = FALSE; // have we already looked up the answer in the registry?
  2580. static BOOL fEnabled; // is the feature enabled or disabled?
  2581. if (!fUseCache)
  2582. {
  2583. DWORD dwData;
  2584. DWORD dwSize = sizeof(dwData);
  2585. DWORD dwType;
  2586. //
  2587. // Next time, we'll use the cached value instead of
  2588. // looking in the registry
  2589. //
  2590. fUseCache = TRUE;
  2591. //
  2592. // Look up the appropriate ieak value in the registry
  2593. //
  2594. if (ERROR_SUCCESS == SHRegGetUSValue(
  2595. SZ_REGKEY_INETCPL_POLICIES,
  2596. SZ_REGVALUE_RESETWEBSETTINGS,
  2597. &dwType,
  2598. (void *)&dwData,
  2599. &dwSize,
  2600. FALSE,
  2601. NULL,
  2602. 0))
  2603. {
  2604. //
  2605. // If the value was found in the registry, then
  2606. // set fEnabled accordingly
  2607. //
  2608. fEnabled = !dwData;
  2609. }
  2610. else
  2611. {
  2612. //
  2613. // If the value is missing from the registry, then
  2614. // assume the feature is enabled
  2615. //
  2616. fEnabled = TRUE;
  2617. }
  2618. }
  2619. return fEnabled;
  2620. }
  2621. STDAPI_(BOOL) InitOCHostClass(const SHDRC * pshdrc)
  2622. {
  2623. // It would be nice to remove this, but since it was exported, we keep it here for compat
  2624. RIPMSG(FALSE, "This export is dead, caller needs to call SHDOCVW!DllRegisterWindowClasses directly");
  2625. return DllRegisterWindowClasses(pshdrc);
  2626. }
  2627. STDAPI SHNavigateToFavorite(IShellFolder* psf, LPCITEMIDLIST pidl, IUnknown* punkSite, DWORD dwFlags)
  2628. {
  2629. HRESULT hres = S_FALSE;
  2630. TCHAR szPath[MAX_PATH];
  2631. // Can we navigate to this favorite?
  2632. BOOL fNavigateDone = SUCCEEDED(GetPathForItem(psf, pidl, szPath, NULL)) &&
  2633. SUCCEEDED(NavFrameWithFile(szPath, punkSite));
  2634. if (fNavigateDone)
  2635. return S_OK;
  2636. LPITEMIDLIST pidlGoto;
  2637. ASSERT(!(dwFlags & (SBSP_NEWBROWSER | SBSP_SAMEBROWSER)));
  2638. if (SUCCEEDED(SHGetNavigateTarget(psf, pidl, &pidlGoto, NULL)))
  2639. {
  2640. IShellBrowser* psb;
  2641. if (SUCCEEDED(IUnknown_QueryService(punkSite, SID_STopLevelBrowser,
  2642. IID_PPV_ARG(IShellBrowser, &psb))))
  2643. {
  2644. hres = psb->BrowseObject(pidlGoto, dwFlags | SBSP_SAMEBROWSER);
  2645. psb->Release();
  2646. }
  2647. ILFree(pidlGoto);
  2648. }
  2649. return hres;
  2650. }
  2651. STDAPI SHGetTopBrowserWindow(IUnknown* punk, HWND* phwnd)
  2652. {
  2653. IOleWindow* pOleWindow;
  2654. HRESULT hr = IUnknown_QueryService(punk, SID_STopLevelBrowser, IID_PPV_ARG(IOleWindow, &pOleWindow));
  2655. if (SUCCEEDED(hr))
  2656. {
  2657. hr = pOleWindow->GetWindow(phwnd);
  2658. pOleWindow->Release();
  2659. }
  2660. return hr;
  2661. }
  2662. BOOL ILIsFolder(LPCITEMIDLIST pidl)
  2663. {
  2664. BOOL fIsFolder = FALSE;
  2665. DWORD dwAttributes = SFGAO_FOLDER;
  2666. HRESULT hr = IEGetAttributesOf(pidl, &dwAttributes);
  2667. if (SFGAO_FOLDER == dwAttributes)
  2668. fIsFolder = TRUE;
  2669. return fIsFolder;
  2670. }
  2671. STDAPI_(LPITEMIDLIST) IEGetInternetRootID(void)
  2672. {
  2673. LPITEMIDLIST pidl;
  2674. //
  2675. // HACKHACK - we want the pidl to the Internet SF
  2676. // so we make a dummy URL and parse it. then
  2677. // we know its parent will be the Internet SF.
  2678. //
  2679. if (SUCCEEDED(IECreateFromPath(TEXT("dummy://url"), &pidl)))
  2680. {
  2681. ASSERT(!ILIsEmpty(_ILNext(pidl)));
  2682. ASSERT(IsURLChild(pidl, FALSE));
  2683. // we only want the parent Internt SF
  2684. _ILNext(pidl)->mkid.cb = 0;
  2685. return pidl;
  2686. }
  2687. return NULL;
  2688. }
  2689. STDAPI_(void) UpdateButtonArray(TBBUTTON *ptbDst, const TBBUTTON *ptbSrc, int ctb, LONG_PTR lStrOffset)
  2690. {
  2691. memcpy(ptbDst, ptbSrc, ctb*sizeof(TBBUTTON));
  2692. if (lStrOffset == -1)
  2693. {
  2694. // handle failure case
  2695. for (int i = 0; i < ctb; i++)
  2696. ptbDst[i].iString = 0;
  2697. }
  2698. else
  2699. {
  2700. for (int i = 0; i < ctb; i++)
  2701. ptbDst[i].iString += lStrOffset;
  2702. }
  2703. }
  2704. //----------------------------------------------------------------------------
  2705. // <Swipped from the NT5 version of Shell32>
  2706. //
  2707. STDAPI_(void) PathToAppPathKey(LPCTSTR pszPath, LPTSTR pszKey, int cchKey)
  2708. {
  2709. // Use the szTemp variable of pseem to build key to the programs specific
  2710. // key in the registry as well as other things...
  2711. StrCpyN(pszKey, REGSTR_PATH_APPPATHS, cchKey);
  2712. StrCatBuff(pszKey, TEXT("\\"), cchKey);
  2713. StrCatBuff(pszKey, PathFindFileName(pszPath), cchKey);
  2714. // Currently we will only look up .EXE if an extension is not
  2715. // specified
  2716. if (*PathFindExtension(pszKey) == 0)
  2717. {
  2718. StrCatBuff(pszKey, TEXT(".exe"), cchKey);
  2719. }
  2720. }
  2721. //----------------------------------------------------------------------------
  2722. // <Swipped from the NT5 version of Shell32>
  2723. //
  2724. // this function checks for the existance of a value called "useURL" under the
  2725. // App Paths key in the registry associated with the app that is passed in.
  2726. STDAPI_(BOOL) DoesAppWantUrl(LPCTSTR pszCmdLine)
  2727. {
  2728. TCHAR szRegKeyName[MAX_PATH];
  2729. HKEY hKeyAppPaths;
  2730. BOOL bRet = FALSE;
  2731. // bug 61538 - The edit button never passes in args or quotes and the
  2732. // code below was screwing up if there were spaces in the path.
  2733. //
  2734. // need to copy the string since PathRemoveArgs whacks in a \0
  2735. // TCHAR szTemp[MAX_PATH];
  2736. // lstrcpyn(szTemp, pszCmdLine, ARRAYSIZE(szTemp));
  2737. // PathRemoveArgs(szTemp);
  2738. // PathUnquoteSpaces(szTemp);
  2739. PathToAppPathKey(pszCmdLine, szRegKeyName, ARRAYSIZE(szRegKeyName));
  2740. if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, szRegKeyName, 0L, KEY_QUERY_VALUE, &hKeyAppPaths) == ERROR_SUCCESS)
  2741. {
  2742. bRet = RegQueryValueEx(hKeyAppPaths, TEXT("UseURL"), NULL, NULL, NULL, NULL) == ERROR_SUCCESS;
  2743. RegCloseKey(hKeyAppPaths);
  2744. }
  2745. return bRet;
  2746. }
  2747. // thread reference count object, this uses SHSetThreadRef()to let other code
  2748. // in this process hold a reference to this main thread, and thus the main thread in this process
  2749. class CRefThread : public IUnknown
  2750. {
  2751. public:
  2752. // IUnknown
  2753. virtual STDMETHODIMP QueryInterface(REFIID riid, void **ppvObj);
  2754. virtual STDMETHODIMP_(ULONG) AddRef(void);
  2755. virtual STDMETHODIMP_(ULONG) Release(void);
  2756. CRefThread(LONG *pcRef);
  2757. private:
  2758. ~CRefThread();
  2759. LONG *_pcRef;
  2760. UINT _idThread;
  2761. };
  2762. CRefThread::CRefThread(LONG *pcRef)
  2763. {
  2764. _idThread = GetCurrentThreadId();
  2765. _pcRef = pcRef;
  2766. *_pcRef = 1;
  2767. }
  2768. //
  2769. // Note that this code tightens but does not close a race window.
  2770. // Although we nuke the process reference, the class factory for
  2771. // the web browser has yet to be deregistered, so if somebody decides
  2772. // to create one, our class factory will wake up and create a
  2773. // shell folder, which will flake out because it can't get a
  2774. // process reference.
  2775. //
  2776. CRefThread::~CRefThread()
  2777. {
  2778. // Avoid re-entrancy during destruction
  2779. *_pcRef = 1000;
  2780. // If we are the process reference, then revoke the process reference
  2781. // since we're going away.
  2782. IUnknown *punk;
  2783. SHGetInstanceExplorer(&punk);
  2784. if (punk == this)
  2785. SHSetInstanceExplorer(NULL);
  2786. ATOMICRELEASE(punk);
  2787. // Nobody should've rescued our reference
  2788. ASSERT(*_pcRef == 1000);
  2789. *_pcRef = 0;
  2790. // get the other thread out of WaitMessage() or GetMessage()
  2791. PostThreadMessage(_idThread, WM_NULL, 0, 0);
  2792. }
  2793. HRESULT CRefThread::QueryInterface(REFIID riid, void **ppvObj)
  2794. {
  2795. static const QITAB qit[] = { { 0 }, };
  2796. return QISearch(this, qit, riid, ppvObj);
  2797. }
  2798. ULONG CRefThread::AddRef()
  2799. {
  2800. return InterlockedIncrement(_pcRef);
  2801. }
  2802. ULONG CRefThread::Release()
  2803. {
  2804. if (InterlockedDecrement(_pcRef))
  2805. return *_pcRef;
  2806. delete this;
  2807. return 0;
  2808. }
  2809. STDAPI SHCreateThreadRef(LONG *pcRef, IUnknown **ppunk)
  2810. {
  2811. *ppunk = new CRefThread(pcRef);
  2812. if (*ppunk)
  2813. return S_OK;
  2814. *pcRef = 0;
  2815. *ppunk = NULL;
  2816. return E_OUTOFMEMORY;
  2817. }
  2818. //
  2819. // Returns the cache file associated with a URL. For file: urls, the associated
  2820. // disk file is returned. Not that we don't use URLDownloadToCacheFile because
  2821. // it causes another I-M-S GET to to be sent to the server
  2822. //
  2823. HRESULT URLToCacheFile
  2824. (
  2825. LPCWSTR pszUrl,
  2826. LPWSTR pszFile,
  2827. int cchFile
  2828. )
  2829. {
  2830. HRESULT hr = E_FAIL;
  2831. DWORD dwScheme = GetUrlScheme(pszUrl);
  2832. if (URL_SCHEME_FILE == dwScheme)
  2833. {
  2834. ULONG cch = cchFile;
  2835. hr = PathCreateFromUrl(pszUrl, pszFile, &cch, 0);
  2836. }
  2837. else
  2838. {
  2839. // bug 73386 - GetUrlCacheEntryInfoExW fails to find entries if there is an anchor
  2840. // so we have to whack it off.
  2841. //
  2842. // We should really fix GetUrlCacheEntryInfoExW instead, but apparently
  2843. // this is risky for 5.x
  2844. //
  2845. WCHAR szUrlBuf[MAX_URL_STRING];
  2846. if (URL_SCHEME_HTTP == dwScheme || URL_SCHEME_HTTPS == dwScheme)
  2847. {
  2848. LPWSTR pszAnchor = StrChr(pszUrl, L'#');
  2849. if (pszAnchor)
  2850. {
  2851. int cch = (int)min((int)(pszAnchor - pszUrl + 1), ARRAYSIZE(szUrlBuf));
  2852. StrCpyN(szUrlBuf, pszUrl, cch);
  2853. pszUrl = szUrlBuf;
  2854. }
  2855. }
  2856. char szBuf[1024];
  2857. LPINTERNET_CACHE_ENTRY_INFOW pCE = (LPINTERNET_CACHE_ENTRY_INFOW)szBuf;
  2858. DWORD dwEntrySize = sizeof(szBuf);
  2859. BOOL fGotCacheInfo = GetUrlCacheEntryInfoExW(pszUrl, pCE, &dwEntrySize, NULL, NULL, NULL, 0);
  2860. if (!fGotCacheInfo)
  2861. {
  2862. if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
  2863. {
  2864. // We guessed too small for the buffer so allocate the correct size & retry
  2865. pCE = (LPINTERNET_CACHE_ENTRY_INFOW)LocalAlloc(LPTR, dwEntrySize);
  2866. if (pCE)
  2867. {
  2868. fGotCacheInfo = GetUrlCacheEntryInfoEx(pszUrl, pCE, &dwEntrySize, NULL, NULL, NULL, 0);
  2869. }
  2870. }
  2871. else
  2872. {
  2873. // Retry using UTF8 encoding
  2874. //
  2875. // This fix belongs in GetUrlCacheEntryInfoEx (StevePro 01/19/99)
  2876. //
  2877. char szUrl[MAX_URL_STRING];
  2878. if (SHUnicodeToAnsiCP(CP_UTF8, pszUrl, szUrl, ARRAYSIZE(szUrl)))
  2879. {
  2880. szUrl[ARRAYSIZE(szUrl)-1] = '\0'; // paranoia
  2881. // UrlEscapeA Internally converts to unicode which messes up utf8. So we
  2882. // copy the string to a WCHAR buffer without coverting and call the unicode version.
  2883. // Yuk!
  2884. WCHAR wzUrl[ARRAYSIZE(szUrl)];
  2885. char* psz = szUrl;
  2886. WCHAR* pwz = wzUrl;
  2887. while (*psz!= NULL)
  2888. {
  2889. *pwz++ = ((WCHAR)*psz++) & 0xff;
  2890. }
  2891. *pwz = L'\0';
  2892. ULONG cch = ARRAYSIZE(wzUrl);
  2893. UrlEscapeW(wzUrl, wzUrl, &cch, /*URL_ESCAPE_PERCENT*/0);
  2894. psz = szUrl;
  2895. pwz = wzUrl;
  2896. while (*pwz!= NULL)
  2897. {
  2898. *psz++ = (char)LOWORD(*pwz++);
  2899. }
  2900. *psz = '\0';
  2901. LPINTERNET_CACHE_ENTRY_INFOA pCEA = (LPINTERNET_CACHE_ENTRY_INFOA)szBuf;
  2902. dwEntrySize = sizeof(szBuf);
  2903. BOOL fUtf8Worked = GetUrlCacheEntryInfoExA(szUrl, pCEA, &dwEntrySize, NULL, NULL, NULL, 0);
  2904. if (!fUtf8Worked)
  2905. {
  2906. if (GetLastError() == ERROR_INSUFFICIENT_BUFFER)
  2907. {
  2908. // We guessed too small for the buffer so allocate the correct size & retry
  2909. pCEA = (LPINTERNET_CACHE_ENTRY_INFOA)LocalAlloc(LPTR, dwEntrySize);
  2910. if (pCEA)
  2911. {
  2912. fUtf8Worked = GetUrlCacheEntryInfoExA(szUrl, pCEA, &dwEntrySize, NULL, NULL, NULL, 0);
  2913. }
  2914. }
  2915. }
  2916. if (fUtf8Worked)
  2917. {
  2918. SHAnsiToUnicode(pCEA->lpszLocalFileName, pszFile, cchFile);
  2919. hr = S_OK;
  2920. }
  2921. if ((char *)pCEA != szBuf)
  2922. {
  2923. LocalFree((HLOCAL)pCEA);
  2924. }
  2925. }
  2926. }
  2927. }
  2928. if (fGotCacheInfo)
  2929. {
  2930. StrCpyN(pszFile, pCE->lpszLocalFileName, cchFile);
  2931. hr = S_OK;
  2932. }
  2933. // Free our GetUrlCacheEntryInfo buffer if we allocated one
  2934. if ((char *)pCE != szBuf)
  2935. {
  2936. LocalFree((HLOCAL)pCE);
  2937. }
  2938. }
  2939. return hr;
  2940. }
  2941. #ifdef DEBUG
  2942. void DebugDumpPidl(DWORD dwDumpFlag, LPTSTR pszOutputString, LPCITEMIDLIST pidl)
  2943. {
  2944. if (g_dwDumpFlags & dwDumpFlag)
  2945. {
  2946. TCHAR szPath[MAX_PATH];
  2947. LPTSTR lpsz;
  2948. if (pidl)
  2949. {
  2950. lpsz = szPath;
  2951. SHGetPathFromIDList(pidl, szPath);
  2952. }
  2953. else
  2954. {
  2955. lpsz = TEXT("(NULL)");
  2956. }
  2957. TraceMsg(TF_ALWAYS, "%s: \"%s\"", pszOutputString, lpsz);
  2958. }
  2959. }
  2960. #endif
  2961. // Variable argument version that ultimately call FormatMessageLiteW
  2962. BOOL __cdecl _FormatMessage(LPCWSTR szTemplate, LPWSTR szBuf, UINT cchBuf, ...)
  2963. {
  2964. BOOL fRet;
  2965. va_list ArgList;
  2966. va_start(ArgList, cchBuf);
  2967. fRet = FormatMessage(FORMAT_MESSAGE_FROM_STRING, szTemplate, 0, 0, szBuf, cchBuf, &ArgList);
  2968. va_end(ArgList);
  2969. return fRet;
  2970. }
  2971. // [msadek], On win9x we get the message thru a chain from explorer /iexplore (ANSI app.).
  2972. // and pass it to comctl32 (Unicode) so it will fail to match the hot key.
  2973. // the system sends the message with ANSI char and we treated it as Unicode.
  2974. // It looks like noone is affected with this bug (US, FE) since they have hot keys always in Latin.
  2975. // Bidi platforms are affected since they do have hot keys in native language.
  2976. WPARAM AnsiWparamToUnicode(WPARAM wParam)
  2977. {
  2978. char szCh[2];
  2979. WCHAR wszCh[2];
  2980. szCh[0] = (BYTE)wParam;
  2981. szCh[1] = '\0';
  2982. if (MultiByteToWideChar(CP_ACP, 0, (LPCSTR)szCh, ARRAYSIZE(szCh),
  2983. wszCh, ARRAYSIZE(wszCh)))
  2984. {
  2985. memcpy(&wParam, wszCh, sizeof(WCHAR));
  2986. }
  2987. return wParam;
  2988. }
  2989. void SHOutlineRect(HDC hdc, const RECT* prc, COLORREF cr)
  2990. {
  2991. RECT rc;
  2992. COLORREF clrSave = SetBkColor(hdc, cr);
  2993. //top
  2994. rc.left = prc->left;
  2995. rc.top = prc->top;
  2996. rc.right = prc->right;
  2997. rc.bottom = prc->top + 1;
  2998. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  2999. //left
  3000. rc.left = prc->left;
  3001. rc.top = prc->top;
  3002. rc.right = prc->left + 1;
  3003. rc.bottom = prc->bottom;
  3004. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  3005. //right
  3006. rc.left = prc->right - 1;
  3007. rc.top = prc->top;
  3008. rc.right = prc->right;
  3009. rc.bottom = prc->bottom;
  3010. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  3011. // bottom
  3012. rc.left = prc->left;
  3013. rc.top = prc->bottom - 1;
  3014. rc.right = prc->right;
  3015. rc.bottom = prc->bottom;
  3016. ExtTextOut(hdc, 0, 0, ETO_OPAQUE, &rc, NULL, 0, NULL);
  3017. SetBkColor(hdc, clrSave);
  3018. }
  3019. HMONITOR GetPrimaryMonitor()
  3020. {
  3021. POINT pt = {0,0};
  3022. return MonitorFromPoint(pt, MONITOR_DEFAULTTOPRIMARY);
  3023. }
  3024. // Gets the Monitor's bounding or work rectangle, if the hMon is bad, return
  3025. // the primary monitor's bounding rectangle.
  3026. BOOL GetMonitorRects(HMONITOR hMon, LPRECT prc, BOOL bWork)
  3027. {
  3028. MONITORINFO mi;
  3029. mi.cbSize = sizeof(mi);
  3030. if (hMon && GetMonitorInfo(hMon, &mi))
  3031. {
  3032. if (!prc)
  3033. return TRUE;
  3034. else if (bWork)
  3035. CopyRect(prc, &mi.rcWork);
  3036. else
  3037. CopyRect(prc, &mi.rcMonitor);
  3038. return TRUE;
  3039. }
  3040. if (prc)
  3041. SetRect(prc, 0, 0, GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN));
  3042. return FALSE;
  3043. }
  3044. ///////////////////////////////////////////////////////////////////////////////
  3045. ///////////////////////////////////////////////////////////////////////////////
  3046. // Utils to load background bitmap for toolbars etc.
  3047. ///////////////////////////////////////////////////////////////////////////////
  3048. ///////////////////////////////////////////////////////////////////////////////
  3049. //------------------------------------------------------------------------
  3050. // determine source name for bitmap, sift thru IE history....
  3051. HRESULT _GetBackBitmapLocation(LPTSTR psz, BOOL fInternet)
  3052. {
  3053. HRESULT hres = E_FAIL;
  3054. DWORD dwType;
  3055. DWORD dwcbData;
  3056. static const TCHAR c_szRegKeyCoolbar[] = TSZIEPATH TEXT("\\Toolbar");
  3057. // IE4 shipped back bitmap customization affecting both browser and shell.
  3058. // IE5 wants these to be separate customizations. But in the roaming
  3059. // case a customized IE4 customer shouldn't lose customization when going
  3060. // to the IE5 machine. So we might need to check twice:
  3061. //
  3062. if (fInternet)
  3063. {
  3064. // Try the IE5 internet location.
  3065. dwcbData = MAX_PATH * sizeof(TCHAR);
  3066. hres = SHGetValue(HKEY_CURRENT_USER, c_szRegKeyCoolbar, TEXT("BackBitmapIE5"), &dwType, psz, &dwcbData);
  3067. }
  3068. else
  3069. {
  3070. // Try the NT5 shell location.
  3071. dwcbData = MAX_PATH * sizeof(TCHAR);
  3072. hres = SHGetValue(HKEY_CURRENT_USER, c_szRegKeyCoolbar, TEXT("BackBitmapShell"), &dwType, psz, &dwcbData);
  3073. }
  3074. if (ERROR_SUCCESS != hres)
  3075. {
  3076. // Try the old combined internet/shell location
  3077. dwcbData = MAX_PATH * sizeof(TCHAR);
  3078. hres = SHGetValue(HKEY_CURRENT_USER, c_szRegKeyCoolbar, TEXT("BackBitmap"), &dwType, psz, &dwcbData);
  3079. }
  3080. return hres;
  3081. }
  3082. //------------------------------------------------------------------------
  3083. // determine background settings and source for toolbar,
  3084. // load bitmap (file/resource) and update cache
  3085. HBITMAP LoadToolbarBackBmp(LPTSTR * ppszBitmap, BMPCACHE * pbmpCache, BOOL fInternet)
  3086. {
  3087. HIGHCONTRAST hc;
  3088. HBITMAP hbmp = pbmpCache->hbmp;
  3089. COLORREF cr3D = GetSysColor(COLOR_3DFACE);
  3090. TCHAR szScratch[MAX_PATH];
  3091. LPTSTR pszBitmap = NULL;
  3092. BOOL fBitmapInvalid = FALSE;
  3093. ENTERCRITICAL;
  3094. // If the stashed hbmp's cr3D color changed, we need to mark invalid
  3095. if (pbmpCache->hbmp && pbmpCache->cr3D != cr3D)
  3096. fBitmapInvalid = TRUE;
  3097. // get the location spec for the bitmap
  3098. hc.cbSize = sizeof(HIGHCONTRAST);
  3099. if ((SystemParametersInfoA(SPI_GETHIGHCONTRAST, hc.cbSize, (LPVOID) &hc, FALSE)) &&
  3100. (hc.dwFlags & HCF_HIGHCONTRASTON))
  3101. {
  3102. // we have no bitmap in high contrast
  3103. }
  3104. else if (SUCCEEDED(_GetBackBitmapLocation(szScratch, fInternet)))
  3105. {
  3106. pszBitmap = szScratch;
  3107. }
  3108. // if they are removing the bitmap, we need to mark invalid
  3109. if (!pszBitmap && *ppszBitmap)
  3110. fBitmapInvalid = TRUE;
  3111. // or it's location has been changed, we need to mark invalid
  3112. if (pszBitmap && (!*ppszBitmap || lstrcmpi(pszBitmap, *ppszBitmap)))
  3113. fBitmapInvalid = TRUE;
  3114. if (fBitmapInvalid)
  3115. {
  3116. TraceMsg(DM_ITBAR, "LoadToolbarBackBmp: Loading Background Bitmap");
  3117. Str_SetPtr(ppszBitmap, pszBitmap);
  3118. hbmp=NULL;
  3119. if (*ppszBitmap)
  3120. {
  3121. if ((*ppszBitmap)[0])
  3122. {
  3123. hbmp = (HBITMAP) LoadImage(NULL, szScratch, IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE | LR_CREATEDIBSECTION | LR_LOADFROMFILE | LR_LOADMAP3DCOLORS );
  3124. }
  3125. if (!hbmp)
  3126. {
  3127. #ifdef OLD_SWIRLY_BACKDROP
  3128. if (SHGetCurColorRes() <= 8)
  3129. hbmp = (HBITMAP) LoadImage(HINST_THISDLL, MAKEINTRESOURCE(IDB_BACK), IMAGE_BITMAP, 0, 0, LR_DEFAULTSIZE | LR_CREATEDIBSECTION | LR_LOADMAP3DCOLORS );
  3130. #endif
  3131. }
  3132. }
  3133. #ifdef OLD_LEGACY_BAD_COLOUR_CODE
  3134. if (hbmp)
  3135. {
  3136. // mapping needed ?
  3137. // DONTWORRYABOUTTHIS: this will be removed as soon as I get the new backdrop....
  3138. if ( /* cr3D != RGB(192,192,192) */ FALSE)
  3139. {
  3140. RGBQUAD rgbTable[256];
  3141. RGBQUAD rgbFace;
  3142. HDC dc;
  3143. HBITMAP hbmSave;
  3144. UINT i;
  3145. UINT n;
  3146. dc = CreateCompatibleDC(NULL);
  3147. hbmSave = (HBITMAP)SelectObject(dc, hbmp);
  3148. n = GetDIBColorTable(dc, 0, 256, rgbTable);
  3149. rgbFace.rgbRed = GetRValue(cr3D);
  3150. rgbFace.rgbGreen = GetGValue(cr3D);
  3151. rgbFace.rgbBlue = GetBValue(cr3D);
  3152. for (i = 0; i < n; i++)
  3153. {
  3154. if ( rgbTable[i].rgbRed == 192 && rgbTable[i].rgbGreen == 192 && rgbTable[i].rgbBlue == 192 )
  3155. {
  3156. rgbTable[i] = rgbFace;
  3157. }
  3158. else
  3159. {
  3160. rgbTable[i].rgbRed = (rgbTable[i].rgbRed * rgbFace.rgbRed ) / 192;
  3161. rgbTable[i].rgbGreen = (rgbTable[i].rgbGreen * rgbFace.rgbGreen) / 192;
  3162. rgbTable[i].rgbBlue = (rgbTable[i].rgbBlue * rgbFace.rgbBlue ) / 192;
  3163. }
  3164. }
  3165. SetDIBColorTable(dc, 0, n, rgbTable);
  3166. SelectObject(dc, hbmSave);
  3167. DeleteDC(dc);
  3168. }
  3169. }
  3170. #endif
  3171. if (pbmpCache->hbmp)
  3172. DeleteObject(pbmpCache->hbmp);
  3173. pbmpCache->hbmp = hbmp;
  3174. pbmpCache->cr3D = cr3D;
  3175. }
  3176. LEAVECRITICAL;
  3177. return hbmp;
  3178. }
  3179. VOID StripDecorations(PTSTR pszTitle, BOOL fStripAmp)
  3180. {
  3181. LPTSTR pszCleaned = pszTitle; // work in-place
  3182. LPCTSTR psz = pszTitle;
  3183. while (*psz && (*psz != TEXT('\t')))
  3184. {
  3185. if (*psz != TEXT('&') || !fStripAmp)
  3186. {
  3187. *pszCleaned = *psz;
  3188. pszCleaned++;
  3189. }
  3190. psz++;
  3191. }
  3192. *pszCleaned = TEXT('\0');
  3193. }
  3194. //------------------------------------------------------------------------
  3195. LPCTSTR UnescapeDoubleAmpersand(LPTSTR pszTitle)
  3196. {
  3197. LPTSTR pszCleaned = pszTitle; // work in-place
  3198. LPCTSTR psz = pszTitle;
  3199. bool fEscapedAmp = false;
  3200. while (*psz)
  3201. {
  3202. if (*psz != TEXT('&') || fEscapedAmp)
  3203. {
  3204. // copy character
  3205. *pszCleaned = *psz;
  3206. pszCleaned++;
  3207. fEscapedAmp = false;
  3208. }
  3209. else
  3210. {
  3211. LPCTSTR pszNext = psz + 1;
  3212. if (pszNext && (*pszNext == TEXT('&'))) {
  3213. fEscapedAmp = true; // keep next ampersand
  3214. }
  3215. }
  3216. psz++;
  3217. }
  3218. *pszCleaned = TEXT('\0');
  3219. return pszTitle;
  3220. }
  3221. UINT MapClsidToID(REFCLSID rclsid)
  3222. {
  3223. UINT nCmdID;
  3224. nCmdID = 0;
  3225. if (IsEqualCLSID(CLSID_SearchBand, rclsid))
  3226. nCmdID = FCIDM_VBBSEARCHBAND;
  3227. else if (IsEqualCLSID(CLSID_FavBand, rclsid))
  3228. nCmdID = FCIDM_VBBFAVORITESBAND;
  3229. else if (IsEqualCLSID(CLSID_HistBand, rclsid))
  3230. nCmdID = FCIDM_VBBHISTORYBAND;
  3231. else if (IsEqualCLSID(CLSID_ExplorerBand, rclsid))
  3232. nCmdID = FCIDM_VBBEXPLORERBAND;
  3233. else if (IsEqualCLSID(CLSID_FileSearchBand, rclsid))
  3234. nCmdID = FCIDM_VBBSEARCHBAND;
  3235. return nCmdID;
  3236. }
  3237. // Create mask from given bitmap, use color at pixel (x/y) as transparent color
  3238. HBITMAP CreateMaskBitmap(HDC hdc, int x, int y, HBITMAP hbmpImage)
  3239. {
  3240. ASSERT(hbmpImage);
  3241. BITMAP bm;
  3242. if (::GetObject(hbmpImage, sizeof(BITMAP), &bm) != sizeof(BITMAP)) {
  3243. return FALSE;
  3244. }
  3245. HDC hdcImg = NULL;
  3246. HDC hdcMask = NULL;
  3247. HBITMAP hbmpMask = NULL;
  3248. HBITMAP hbmpOldImg = NULL;
  3249. HBITMAP hbmpOldMsk = NULL;
  3250. COLORREF clrTransparent = 0;
  3251. hdcImg = ::CreateCompatibleDC(hdc);
  3252. if (hdcImg == NULL) goto _CMBcleanup;
  3253. hdcMask = ::CreateCompatibleDC(hdc);
  3254. if (hdcMask == NULL) goto _CMBcleanup;
  3255. hbmpMask = ::CreateBitmap(bm.bmWidth, bm.bmHeight, 1, 1, NULL);
  3256. if (hbmpMask == NULL) goto _CMBcleanup;
  3257. hbmpOldImg = (HBITMAP) ::SelectObject(hdcImg, hbmpImage);
  3258. hbmpOldMsk = (HBITMAP) ::SelectObject(hdcMask, hbmpMask);
  3259. clrTransparent = ::GetPixel(hdcImg, 0, 0);
  3260. ::SetBkColor(hdcImg, clrTransparent);
  3261. ::BitBlt(hdcMask, 0, 0, bm.bmWidth, bm.bmHeight, hdcImg, 0, 0, SRCCOPY);
  3262. _CMBcleanup:
  3263. if (hbmpOldImg && hdcImg)
  3264. SelectObject(hdcImg, hbmpOldImg);
  3265. if (hdcImg)
  3266. DeleteDC(hdcImg);
  3267. if (hbmpOldMsk && hdcMask)
  3268. SelectObject(hdcMask, hbmpOldMsk);
  3269. if (hdcMask)
  3270. DeleteDC(hdcMask);
  3271. return hbmpMask;
  3272. }
  3273. // draw bitmap transparently; on Win2K and up, one could use MaskBlt()
  3274. BOOL DrawTransparentBitmapPart(HDC hdc, int x, int y, int dx, int dy, HBITMAP hbmpImage, HBITMAP hbmpMask)
  3275. {
  3276. ASSERT(hbmpImage);
  3277. BITMAP bm;
  3278. if (::GetObject(hbmpImage, sizeof(BITMAP), &bm) != sizeof(BITMAP)) {
  3279. return FALSE;
  3280. }
  3281. HBITMAP hbmpTmpMask = NULL;
  3282. // create temporary mask bitmap if none supplied
  3283. if (hbmpMask == NULL) {
  3284. hbmpMask = hbmpTmpMask = CreateMaskBitmap(hdc, 0, 0, hbmpImage);
  3285. }
  3286. if (hbmpMask == NULL) {
  3287. return FALSE;
  3288. }
  3289. HDC hdcOffScr = NULL;
  3290. HBITMAP hbmOffScr = NULL;
  3291. HBITMAP hbmOldOffScr = NULL;
  3292. HDC hdcImage = NULL;
  3293. HBITMAP hbmOldImage = NULL;
  3294. HDC hdcMask = NULL;
  3295. HBITMAP hbmOldMask = NULL;
  3296. // draw.to offscreen bitmap
  3297. hdcOffScr = ::CreateCompatibleDC(hdc);
  3298. if (hdcOffScr == NULL) goto _DTBcleanup;
  3299. hbmOffScr = ::CreateBitmap(dx, dy,GetDeviceCaps(hdc, PLANES),
  3300. (BYTE)GetDeviceCaps(hdc, BITSPIXEL), NULL);
  3301. if (hbmOffScr == NULL) goto _DTBcleanup;
  3302. hbmOldOffScr = (HBITMAP)::SelectObject(hdcOffScr, hbmOffScr);
  3303. // Copy the image of the destination rectangle to the
  3304. // off-screen buffer DC, so we can play with it.
  3305. ::BitBlt(hdcOffScr, 0, 0, dx, dy, hdc, x, y, SRCCOPY);
  3306. // prepare DCs for both image and mask
  3307. hdcImage = ::CreateCompatibleDC(hdc);
  3308. if (hdcImage == NULL) goto _DTBcleanup;
  3309. hbmOldImage = (HBITMAP)::SelectObject(hdcImage, hbmpImage);
  3310. hdcMask = ::CreateCompatibleDC(hdc);
  3311. if (hdcMask == NULL) goto _DTBcleanup;
  3312. hbmOldMask = (HBITMAP)::SelectObject(hdcMask, hbmpMask);
  3313. ::SetBkColor(hdcOffScr, RGB(255,255,255));
  3314. ::BitBlt(hdcOffScr, 0, 0, dx, dy, hdcImage, 0, 0, SRCINVERT);
  3315. ::BitBlt(hdcOffScr, 0, 0, dx, dy, hdcMask, 0, 0, SRCAND);
  3316. ::BitBlt(hdcOffScr, 0, 0, dx, dy, hdcImage, 0, 0, SRCINVERT);
  3317. // Copy the resultant image back to the screen DC.
  3318. ::BitBlt(hdc, x, y, dx, dy, hdcOffScr, 0, 0, SRCCOPY);
  3319. _DTBcleanup:
  3320. if (hdcOffScr && hbmOldOffScr)
  3321. ::SelectObject(hdcOffScr, hbmOldOffScr);
  3322. if (hdcOffScr)
  3323. ::DeleteDC(hdcOffScr);
  3324. if (hbmOffScr)
  3325. ::DeleteObject(hbmOffScr);
  3326. if (hdcImage && hbmOldImage)
  3327. ::SelectObject(hdcImage, hbmOldImage);
  3328. if (hdcImage)
  3329. ::DeleteDC(hdcImage);
  3330. if (hdcMask && hbmOldMask)
  3331. ::SelectObject(hdcMask, hbmOldMask);
  3332. if (hdcMask)
  3333. ::DeleteDC(hdcMask);
  3334. if (hbmpTmpMask)
  3335. ::DeleteObject(hbmpTmpMask);
  3336. return TRUE;
  3337. }
  3338. // draw bitmap transparently; on Win2K and up, one could use MaskBlt()
  3339. BOOL DrawTransparentBitmap(HDC hdc, int x, int y, HBITMAP hbmpImage, HBITMAP hbmpMask)
  3340. {
  3341. ASSERT(hbmpImage);
  3342. BITMAP bm;
  3343. if (::GetObject(hbmpImage, sizeof(BITMAP), &bm) != sizeof(BITMAP)) {
  3344. return FALSE;
  3345. }
  3346. return DrawTransparentBitmapPart(hdc, x, y, bm.bmWidth, bm.bmHeight, hbmpImage, hbmpMask);
  3347. }
  3348. //------------------------------------------------------------------------
  3349. BOOL
  3350. DrawAlphaBitmap(HDC hdc, int x, int y, int dx, int dy, HBITMAP hbmpImage)
  3351. {
  3352. BLENDFUNCTION bf = {0};
  3353. HDC hdcImage = ::CreateCompatibleDC(hdc);
  3354. if (hdcImage == NULL) {
  3355. return false;
  3356. }
  3357. HBITMAP hbmOldImage = (HBITMAP)::SelectObject(hdcImage, hbmpImage);
  3358. bf.BlendOp = AC_SRC_OVER;
  3359. bf.SourceConstantAlpha = 255;
  3360. bf.AlphaFormat = AC_SRC_ALPHA;
  3361. AlphaBlend(hdc, x, y, dx, dy, hdcImage, 0, 0, dx, dy, bf);
  3362. if (hbmOldImage) {
  3363. SelectObject(hdcImage, hbmOldImage);
  3364. }
  3365. DESTROY_OBJ_WITH_HANDLE(hdcImage, DeleteObject);
  3366. return true;
  3367. }
  3368. STDAPI_(IDeskBand *) FindBandByClsidBS(IBandSite *pbs, REFCLSID clsidToFind)
  3369. {
  3370. DWORD dwBandID;
  3371. for (int i = 0; SUCCEEDED(pbs->EnumBands(i, &dwBandID)); i++)
  3372. {
  3373. IDeskBand *pstb;
  3374. HRESULT hr = pbs->QueryBand(dwBandID, &pstb, NULL, NULL, 0);
  3375. if (SUCCEEDED(hr))
  3376. {
  3377. CLSID clsid;
  3378. hr = IUnknown_GetClassID(pstb, &clsid);
  3379. if (SUCCEEDED(hr) && IsEqualGUID(clsidToFind, clsid))
  3380. {
  3381. return pstb;
  3382. }
  3383. pstb->Release();
  3384. }
  3385. }
  3386. return NULL;
  3387. }
  3388. HIMAGELIST CreateImageList(HINSTANCE hi, LPCTSTR lpbmp, int cx, int cGrow, COLORREF crMask,
  3389. UINT uType, UINT uFlags, BOOL bUseNewMirroringSupport)
  3390. {
  3391. HBITMAP hbmImage;
  3392. HIMAGELIST piml = NULL;
  3393. BITMAP bm;
  3394. int cy, cInitial;
  3395. UINT flags;
  3396. hbmImage = (HBITMAP)LoadImage(hi, lpbmp, uType, 0, 0, uFlags);
  3397. if (hbmImage && (sizeof(bm) == GetObject(hbmImage, sizeof(bm), &bm)))
  3398. {
  3399. // If cx is not stated assume it is the same as cy.
  3400. // ASSERT(cx);
  3401. cy = bm.bmHeight;
  3402. if (cx == 0)
  3403. cx = cy;
  3404. cInitial = bm.bmWidth / cx;
  3405. ENTERCRITICAL;
  3406. if (bUseNewMirroringSupport)
  3407. {
  3408. flags = ILC_MIRROR | PrivateILC_PERITEMMIRROR;
  3409. }
  3410. else
  3411. {
  3412. flags = 0;
  3413. }
  3414. if (crMask != CLR_NONE)
  3415. flags |= ILC_MASK;
  3416. if (bm.bmBits)
  3417. flags |= (bm.bmBitsPixel & ILC_COLORMASK);
  3418. piml = ImageList_Create(cx, cy, flags, cInitial, cGrow);
  3419. if (piml)
  3420. {
  3421. int added;
  3422. if (crMask == CLR_NONE)
  3423. added = ImageList_Add(piml, hbmImage, NULL);
  3424. else
  3425. added = ImageList_AddMasked(piml, hbmImage, crMask);
  3426. if (added < 0)
  3427. {
  3428. ImageList_Destroy(piml);
  3429. piml = NULL;
  3430. }
  3431. }
  3432. LEAVECRITICAL;
  3433. }
  3434. if (hbmImage)
  3435. DeleteObject(hbmImage);
  3436. return piml;
  3437. }