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.

2286 lines
65 KiB

  1. /*****************************************************************************\
  2. FILE: util.cpp
  3. DESCRIPTION:
  4. Shared stuff that operates on all classes.
  5. BryanSt 4/4/2000 (Bryan Starbuck)
  6. Copyright (C) Microsoft Corp 2000-2000. All rights reserved.
  7. \*****************************************************************************/
  8. #include "priv.h"
  9. #include <atlbase.h> // USES_CONVERSION
  10. #include <comdef.h>
  11. #include <errors.h> // \\themes\\inc
  12. #include <ctxdef.h> // hydra stuff
  13. #include <regapi.h> // WINSTATION_REG_NAME
  14. #include "WMPAPITemp.h"
  15. #define SECURITY_WIN32
  16. #include <sspi.h>
  17. extern "C" {
  18. #include <Secext.h> // for GetUserNameEx()
  19. }
  20. #define DECL_CRTFREE
  21. #include <crtfree.h>
  22. #include "util.h"
  23. /////////////////////////////////////////////////////////////////////
  24. // String Helpers
  25. /////////////////////////////////////////////////////////////////////
  26. HINSTANCE g_hinst; // My instance handle
  27. HANDLE g_hLogFile = INVALID_HANDLE_VALUE;
  28. #ifdef DEBUG
  29. DWORD g_TLSliStopWatchStartHi = 0xFFFFFFFF;
  30. DWORD g_TLSliStopWatchStartLo = 0xFFFFFFFF;
  31. LARGE_INTEGER g_liStopWatchFreq = {0};
  32. #endif // DEBUG
  33. /////////////////////////////////////////////////////////////////////
  34. // Debug Timing Helpers
  35. /////////////////////////////////////////////////////////////////////
  36. #ifdef DEBUG
  37. void DebugStartWatch(void)
  38. {
  39. LARGE_INTEGER liStopWatchStart;
  40. if (-1 == g_TLSliStopWatchStartHi)
  41. {
  42. g_TLSliStopWatchStartHi = TlsAlloc();
  43. g_TLSliStopWatchStartLo = TlsAlloc();
  44. liStopWatchStart.QuadPart = 0;
  45. QueryPerformanceFrequency(&g_liStopWatchFreq); // Only a one time call since it's value can't change while the system is running.
  46. }
  47. else
  48. {
  49. liStopWatchStart.HighPart = PtrToUlong(TlsGetValue(g_TLSliStopWatchStartHi));
  50. liStopWatchStart.LowPart = PtrToUlong(TlsGetValue(g_TLSliStopWatchStartLo));
  51. }
  52. AssertMsg((0 == liStopWatchStart.QuadPart), TEXT("Someone else is using our perf timer. Stop nesting.")); // If you hit this, then the stopwatch is nested.
  53. QueryPerformanceCounter(&liStopWatchStart);
  54. TlsSetValue(g_TLSliStopWatchStartHi, IntToPtr(liStopWatchStart.HighPart));
  55. TlsSetValue(g_TLSliStopWatchStartLo, IntToPtr(liStopWatchStart.LowPart));
  56. }
  57. DWORD DebugStopWatch(void)
  58. {
  59. LARGE_INTEGER liDiff;
  60. LARGE_INTEGER liStopWatchStart;
  61. QueryPerformanceCounter(&liDiff);
  62. liStopWatchStart.HighPart = PtrToUlong(TlsGetValue(g_TLSliStopWatchStartHi));
  63. liStopWatchStart.LowPart = PtrToUlong(TlsGetValue(g_TLSliStopWatchStartLo));
  64. liDiff.QuadPart -= liStopWatchStart.QuadPart;
  65. ASSERT(0 != g_liStopWatchFreq.QuadPart); // I don't like to fault with div 0.
  66. DWORD dwTime = (DWORD)((liDiff.QuadPart * 1000) / g_liStopWatchFreq.QuadPart);
  67. TlsSetValue(g_TLSliStopWatchStartHi, (LPVOID) 0);
  68. TlsSetValue(g_TLSliStopWatchStartLo, (LPVOID) 0);
  69. return dwTime;
  70. }
  71. #endif // DEBUG
  72. /////////////////////////////////////////////////////////////////////
  73. // String Helpers
  74. /////////////////////////////////////////////////////////////////////
  75. #undef SysAllocStringA
  76. BSTR SysAllocStringA(LPCSTR pszStr)
  77. {
  78. BSTR bstrOut = NULL;
  79. if (pszStr)
  80. {
  81. DWORD cchSize = (lstrlenA(pszStr) + 1);
  82. LPWSTR pwszThunkTemp = (LPWSTR) LocalAlloc(LPTR, (sizeof(pwszThunkTemp[0]) * cchSize)); // assumes INFOTIPSIZE number of chars max
  83. if (pwszThunkTemp)
  84. {
  85. SHAnsiToUnicode(pszStr, pwszThunkTemp, cchSize);
  86. bstrOut = SysAllocString(pwszThunkTemp);
  87. LocalFree(pwszThunkTemp);
  88. }
  89. }
  90. return bstrOut;
  91. }
  92. HRESULT HrSysAllocStringA(IN LPCSTR pszSource, OUT BSTR * pbstrDest)
  93. {
  94. HRESULT hr = S_OK;
  95. if (pbstrDest)
  96. {
  97. *pbstrDest = SysAllocStringA(pszSource);
  98. if (pszSource)
  99. {
  100. if (*pbstrDest)
  101. hr = S_OK;
  102. else
  103. hr = E_OUTOFMEMORY;
  104. }
  105. }
  106. return hr;
  107. }
  108. HRESULT HrSysAllocStringW(IN const OLECHAR * pwzSource, OUT BSTR * pbstrDest)
  109. {
  110. HRESULT hr = S_OK;
  111. if (pbstrDest)
  112. {
  113. *pbstrDest = SysAllocString(pwzSource);
  114. if (pwzSource)
  115. {
  116. if (*pbstrDest)
  117. hr = S_OK;
  118. else
  119. hr = E_OUTOFMEMORY;
  120. }
  121. }
  122. return hr;
  123. }
  124. LPSTR AllocStringFromBStr(BSTR bstr)
  125. {
  126. USES_CONVERSION; // atlbase.h
  127. char *a = W2A((bstr ? bstr : L""));
  128. int len = 1 + lstrlenA(a);
  129. char *p = (char *)LocalAlloc(LPTR, len);
  130. if (p)
  131. {
  132. StrCpyA(p, a);
  133. }
  134. return p;
  135. }
  136. HRESULT BSTRFromStream(IStream * pStream, BSTR * pbstr)
  137. {
  138. STATSTG statStg = {0};
  139. HRESULT hr = pStream->Stat(&statStg, STATFLAG_NONAME);
  140. if (S_OK == hr)
  141. {
  142. DWORD cchSize = statStg.cbSize.LowPart;
  143. *pbstr = SysAllocStringLen(NULL, cchSize + 4);
  144. if (*pbstr)
  145. {
  146. LPSTR pszTemp = (LPSTR) LocalAlloc(LPTR, sizeof(pszTemp[0]) * (cchSize + 4));
  147. if (pszTemp)
  148. {
  149. ULONG cbRead;
  150. hr = pStream->Read(pszTemp, cchSize, &cbRead);
  151. pszTemp[cchSize] = 0;
  152. SHAnsiToUnicode(pszTemp, *pbstr, (cchSize + 1));
  153. LocalFree(pszTemp);
  154. }
  155. else
  156. hr = E_OUTOFMEMORY;
  157. }
  158. else
  159. hr = E_OUTOFMEMORY;
  160. }
  161. return hr;
  162. }
  163. // --------------------------------------------------------------------------------
  164. // HrCopyStream
  165. // --------------------------------------------------------------------------------
  166. HRESULT HrCopyStream(LPSTREAM pstmIn, LPSTREAM pstmOut, ULONG *pcb)
  167. {
  168. HRESULT hr = S_OK;
  169. BYTE buf[4096];
  170. ULONG cbRead=0,
  171. cbTotal=0;
  172. do
  173. {
  174. hr = pstmIn->Read(buf, sizeof(buf), &cbRead);
  175. if (FAILED(hr) || cbRead == 0)
  176. {
  177. break;
  178. }
  179. hr = pstmOut->Write(buf, cbRead, NULL);
  180. if (FAILED(hr))
  181. {
  182. break;
  183. }
  184. cbTotal += cbRead;
  185. }
  186. while (cbRead == sizeof (buf));
  187. if (pcb && SUCCEEDED(hr))
  188. *pcb = cbTotal;
  189. return hr;
  190. }
  191. HRESULT CreateBStrVariantFromWStr(IN OUT VARIANT * pvar, IN LPCWSTR pwszString)
  192. {
  193. HRESULT hr = E_INVALIDARG;
  194. if (pvar)
  195. {
  196. pvar->bstrVal = SysAllocString(pwszString);
  197. if (pvar->bstrVal)
  198. {
  199. pvar->vt = VT_BSTR;
  200. hr = S_OK;
  201. }
  202. else
  203. {
  204. pvar->vt = VT_EMPTY;
  205. hr = E_OUTOFMEMORY;
  206. }
  207. }
  208. return hr;
  209. }
  210. HRESULT HrSysAllocString(IN const OLECHAR * pwzSource, OUT BSTR * pbstrDest)
  211. {
  212. HRESULT hr = S_OK;
  213. if (pbstrDest)
  214. {
  215. *pbstrDest = SysAllocString(pwzSource);
  216. if (pwzSource)
  217. {
  218. if (*pbstrDest)
  219. hr = S_OK;
  220. else
  221. hr = E_OUTOFMEMORY;
  222. }
  223. }
  224. return hr;
  225. }
  226. HRESULT UnEscapeHTML(BSTR bstrEscaped, BSTR * pbstrUnEscaped)
  227. {
  228. HRESULT hr = HrSysAllocString(bstrEscaped, pbstrUnEscaped);
  229. if (SUCCEEDED(hr))
  230. {
  231. // Find %xx and replace.
  232. LPWSTR pwszEscapedSequence = StrChrW(*pbstrUnEscaped, CH_HTML_ESCAPE);
  233. WCHAR wzEscaped[5] = L"0xXX";
  234. while (pwszEscapedSequence && (3 <= lstrlenW(pwszEscapedSequence)))
  235. {
  236. int nCharCode;
  237. wzEscaped[2] = pwszEscapedSequence[1];
  238. wzEscaped[3] = pwszEscapedSequence[2];
  239. StrToIntExW(wzEscaped, STIF_SUPPORT_HEX, &nCharCode);
  240. // Replace the '%' with the real char.
  241. pwszEscapedSequence[0] = (WCHAR) nCharCode;
  242. pwszEscapedSequence = CharNextW(pwszEscapedSequence); // Skip pasted the replaced char.
  243. // Over write the 0xXX value.
  244. StrCpyW(pwszEscapedSequence, &pwszEscapedSequence[2]);
  245. // Next...
  246. pwszEscapedSequence = StrChrW(pwszEscapedSequence, CH_HTML_ESCAPE);
  247. }
  248. }
  249. return hr;
  250. }
  251. /*****************************************************************************\
  252. PARAMETERS:
  253. If fBoolean is TRUE, return "True" else "False".
  254. HRESULT BOOLToString(BOOL fBoolean, BSTR * pbstrValue)
  255. {
  256. HRESULT hr = E_INVALIDARG;
  257. if (pbstrValue)
  258. {
  259. LPCWSTR pwszValue;
  260. *pbstrValue = NULL;
  261. if (TRUE == fBoolean)
  262. {
  263. pwszValue = SZ_QUERYDATA_TRUE;
  264. }
  265. else
  266. {
  267. pwszValue = SZ_QUERYDATA_FALSE;
  268. }
  269. hr = HrSysAllocString(pwszValue, pbstrValue);
  270. }
  271. return hr;
  272. }
  273. \*****************************************************************************/
  274. #define SZ_VALID_XML L"<?xml"
  275. /////////////////////////////////////////////////////////////////////
  276. // XML Related Helpers
  277. /////////////////////////////////////////////////////////////////////
  278. HRESULT XMLDOMFromBStr(BSTR bstrXML, IXMLDOMDocument ** ppXMLDoc)
  279. {
  280. HRESULT hr = E_FAIL;
  281. // We don't even want to
  282. // bother passing it to the XML DOM because they throw exceptions. These
  283. // are caught and handled but we still don't want this to happen. We try
  284. // to get XML from the web server, but we get HTML instead if the web server
  285. // fails or the web proxy returns HTML if the site isn't found.
  286. if (!StrCmpNIW(SZ_VALID_XML, bstrXML, (ARRAYSIZE(SZ_VALID_XML) - 1)))
  287. {
  288. hr = CoCreateInstance(CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IXMLDOMDocument, ppXMLDoc));
  289. if (SUCCEEDED(hr))
  290. {
  291. VARIANT_BOOL fIsSuccessful;
  292. // NOTE: This will throw an 0xE0000001 exception in MSXML if the XML is invalid.
  293. // This is not good but there isn't much we can do about it. The problem is
  294. // that web proxies give back HTML which fails to parse.
  295. hr = (*ppXMLDoc)->loadXML(bstrXML, &fIsSuccessful);
  296. if (SUCCEEDED(hr))
  297. {
  298. if (VARIANT_TRUE != fIsSuccessful)
  299. {
  300. hr = E_FAIL;
  301. }
  302. }
  303. }
  304. if (FAILED(hr))
  305. {
  306. (*ppXMLDoc)->Release();
  307. *ppXMLDoc = NULL;
  308. }
  309. }
  310. return hr;
  311. }
  312. HRESULT XMLBStrFromDOM(IXMLDOMDocument * pXMLDoc, BSTR * pbstrXML)
  313. {
  314. IStream * pStream;
  315. HRESULT hr = pXMLDoc->QueryInterface(IID_PPV_ARG(IStream, &pStream)); // check the return value
  316. if (S_OK == hr)
  317. {
  318. hr = BSTRFromStream(pStream, pbstrXML);
  319. pStream->Release();
  320. }
  321. return hr;
  322. }
  323. HRESULT XMLAppendElement(IXMLDOMElement * pXMLElementRoot, IXMLDOMElement * pXMLElementToAppend)
  324. {
  325. IXMLDOMNode * pXMLNodeRoot;
  326. HRESULT hr = pXMLElementRoot->QueryInterface(IID_PPV_ARG(IXMLDOMNode, &pXMLNodeRoot));
  327. if (EVAL(SUCCEEDED(hr)))
  328. {
  329. IXMLDOMNode * pXMLNodeToAppend;
  330. hr = pXMLElementToAppend->QueryInterface(IID_PPV_ARG(IXMLDOMNode, &pXMLNodeToAppend));
  331. if (EVAL(SUCCEEDED(hr)))
  332. {
  333. hr = pXMLNodeRoot->appendChild(pXMLNodeToAppend, NULL);
  334. pXMLNodeToAppend->Release();
  335. }
  336. pXMLNodeRoot->Release();
  337. }
  338. return hr;
  339. }
  340. HRESULT XMLDOMFromFile(IN LPCWSTR pwzPath, OUT IXMLDOMDocument ** ppXMLDOMDoc)
  341. {
  342. HRESULT hr = CoCreateInstance(CLSID_DOMDocument, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IXMLDOMDocument, ppXMLDOMDoc));
  343. if (SUCCEEDED(hr))
  344. {
  345. VARIANT xmlSource;
  346. xmlSource.vt = VT_BSTR;
  347. xmlSource.bstrVal = SysAllocString(pwzPath);
  348. if (xmlSource.bstrVal)
  349. {
  350. VARIANT_BOOL fIsSuccessful = VARIANT_TRUE;
  351. hr = (*ppXMLDOMDoc)->load(xmlSource, &fIsSuccessful);
  352. if ((S_FALSE == hr) || (VARIANT_FALSE == fIsSuccessful))
  353. {
  354. // This happens when the file isn't a valid XML file.
  355. hr = E_FAIL;
  356. }
  357. VariantClear(&xmlSource);
  358. }
  359. if (FAILED(hr))
  360. {
  361. ATOMICRELEASE(*ppXMLDOMDoc);
  362. }
  363. }
  364. return hr;
  365. }
  366. HRESULT XMLElem_VerifyTagName(IN IXMLDOMElement * pXMLElementMessage, IN LPCWSTR pwszTagName)
  367. {
  368. BSTR bstrTagName;
  369. HRESULT hr = pXMLElementMessage->get_tagName(&bstrTagName);
  370. if (S_FALSE == hr)
  371. {
  372. hr = E_FAIL;
  373. }
  374. else if (SUCCEEDED(hr))
  375. {
  376. if (!bstrTagName || !pwszTagName || StrCmpIW(bstrTagName, pwszTagName))
  377. {
  378. hr = E_FAIL;
  379. }
  380. SysFreeString(bstrTagName);
  381. }
  382. return hr;
  383. }
  384. HRESULT XMLElem_GetElementsByTagName(IN IXMLDOMElement * pXMLElementMessage, IN LPCWSTR pwszTagName, OUT IXMLDOMNodeList ** ppNodeList)
  385. {
  386. BSTR bstrTagName = SysAllocString(pwszTagName);
  387. HRESULT hr = E_OUTOFMEMORY;
  388. *ppNodeList = NULL;
  389. if (bstrTagName)
  390. {
  391. hr = pXMLElementMessage->getElementsByTagName(bstrTagName, ppNodeList);
  392. if (S_FALSE == hr)
  393. {
  394. hr = E_FAIL;
  395. }
  396. SysFreeString(bstrTagName);
  397. }
  398. return hr;
  399. }
  400. HRESULT XMLNode_GetAttributeValue(IN IXMLDOMNode * pXMLNode, IN LPCWSTR pwszAttributeName, OUT BSTR * pbstrValue)
  401. {
  402. BSTR bstrAttributeName = SysAllocString(pwszAttributeName);
  403. HRESULT hr = E_OUTOFMEMORY;
  404. *pbstrValue = NULL;
  405. if (bstrAttributeName)
  406. {
  407. IXMLDOMNamedNodeMap * pNodeAttributes;
  408. hr = pXMLNode->get_attributes(&pNodeAttributes);
  409. if (S_FALSE == hr) hr = E_FAIL;
  410. if (SUCCEEDED(hr))
  411. {
  412. IXMLDOMNode * pTypeAttribute;
  413. hr = pNodeAttributes->getNamedItem(bstrAttributeName, &pTypeAttribute);
  414. if (S_FALSE == hr) hr = ResultFromWin32(ERROR_NOT_FOUND);
  415. if (SUCCEEDED(hr))
  416. {
  417. VARIANT varAtribValue = {0};
  418. hr = pTypeAttribute->get_nodeValue(&varAtribValue);
  419. if (S_FALSE == hr) hr = E_FAIL;
  420. if (SUCCEEDED(hr) && (VT_BSTR == varAtribValue.vt))
  421. {
  422. *pbstrValue = SysAllocString(varAtribValue.bstrVal);
  423. }
  424. VariantClear(&varAtribValue);
  425. pTypeAttribute->Release();
  426. }
  427. pNodeAttributes->Release();
  428. }
  429. SysFreeString(bstrAttributeName);
  430. }
  431. return hr;
  432. }
  433. HRESULT XMLNode_GetChildTag(IN IXMLDOMNode * pXMLNode, IN LPCWSTR pwszTagName, OUT IXMLDOMNode ** ppChildNode)
  434. {
  435. HRESULT hr = E_INVALIDARG;
  436. *ppChildNode = NULL;
  437. if (pXMLNode)
  438. {
  439. IXMLDOMElement * pXMLElement;
  440. hr = pXMLNode->QueryInterface(IID_PPV_ARG(IXMLDOMElement, &pXMLElement));
  441. if (SUCCEEDED(hr))
  442. {
  443. IXMLDOMNodeList * pNodeList;
  444. hr = XMLElem_GetElementsByTagName(pXMLElement, pwszTagName, &pNodeList);
  445. if (SUCCEEDED(hr))
  446. {
  447. hr = XMLNodeList_GetChild(pNodeList, 0, ppChildNode);
  448. pNodeList->Release();
  449. }
  450. pXMLElement->Release();
  451. }
  452. }
  453. return hr;
  454. }
  455. HRESULT XMLNode_GetTagText(IN IXMLDOMNode * pXMLNode, OUT BSTR * pbstrValue)
  456. {
  457. DOMNodeType nodeType = NODE_TEXT;
  458. HRESULT hr = pXMLNode->get_nodeType(&nodeType);
  459. *pbstrValue = NULL;
  460. if (S_FALSE == hr) hr = E_FAIL;
  461. if (SUCCEEDED(hr))
  462. {
  463. if (NODE_TEXT == nodeType)
  464. {
  465. VARIANT varAtribValue = {0};
  466. hr = pXMLNode->get_nodeValue(&varAtribValue);
  467. if (S_FALSE == hr) hr = E_FAIL;
  468. if (SUCCEEDED(hr) && (VT_BSTR == varAtribValue.vt))
  469. {
  470. *pbstrValue = SysAllocString(varAtribValue.bstrVal);
  471. }
  472. VariantClear(&varAtribValue);
  473. }
  474. else
  475. {
  476. hr = pXMLNode->get_text(pbstrValue);
  477. }
  478. }
  479. return hr;
  480. }
  481. HRESULT XMLNodeList_GetChild(IN IXMLDOMNodeList * pNodeList, IN DWORD dwIndex, OUT IXMLDOMNode ** ppXMLChildNode)
  482. {
  483. HRESULT hr = pNodeList->get_item(dwIndex, ppXMLChildNode);
  484. if (S_FALSE == hr)
  485. {
  486. hr = ResultFromWin32(ERROR_NOT_FOUND);
  487. }
  488. return hr;
  489. }
  490. HRESULT XMLNode_GetChildTagTextValue(IN IXMLDOMNode * pXMLNode, IN BSTR bstrChildTag, OUT BSTR * pbstrValue)
  491. {
  492. IXMLDOMNode * pNodeType;
  493. HRESULT hr = XMLNode_GetChildTag(pXMLNode, bstrChildTag, &pNodeType);
  494. if (SUCCEEDED(hr))
  495. {
  496. hr = XMLNode_GetTagText(pNodeType, pbstrValue);
  497. pNodeType->Release();
  498. }
  499. return hr;
  500. }
  501. HRESULT XMLNode_GetChildTagTextValueToBool(IN IXMLDOMNode * pXMLNode, IN BSTR bstrChildTag, OUT BOOL * pfBoolean)
  502. {
  503. BSTR bstr;
  504. HRESULT hr = XMLNode_GetChildTagTextValue(pXMLNode, bstrChildTag, &bstr);
  505. if (SUCCEEDED(hr))
  506. {
  507. if (!StrCmpIW(bstr, L"on"))
  508. {
  509. *pfBoolean = TRUE;
  510. }
  511. else
  512. {
  513. *pfBoolean = FALSE;
  514. }
  515. SysFreeString(bstr);
  516. }
  517. return hr;
  518. }
  519. BOOL XML_IsChildTagTextEqual(IN IXMLDOMNode * pXMLNode, IN BSTR bstrChildTag, IN BSTR bstrText)
  520. {
  521. BOOL fIsChildTagTextEqual = FALSE;
  522. BSTR bstrChildText;
  523. HRESULT hr = XMLNode_GetChildTagTextValue(pXMLNode, bstrChildTag, &bstrChildText);
  524. if (SUCCEEDED(hr))
  525. {
  526. // Is this <TYPE>email</TYPE>?
  527. if (!StrCmpIW(bstrChildText, bstrText))
  528. {
  529. // No, so keep looking.
  530. fIsChildTagTextEqual = TRUE;
  531. }
  532. SysFreeString(bstrChildText);
  533. }
  534. return fIsChildTagTextEqual;
  535. }
  536. /////////////////////////////////////////////////////////////////////
  537. // File System Wrapping Helpers
  538. /////////////////////////////////////////////////////////////////////
  539. HRESULT CreateFileHrWrap(LPCTSTR lpFileName, DWORD dwDesiredAccess, DWORD dwShareMode, LPSECURITY_ATTRIBUTES lpSecurityAttributes,
  540. DWORD dwCreationDisposition, DWORD dwFlagsAndAttributes, HANDLE hTemplateFile, HANDLE * phFileHandle)
  541. {
  542. HRESULT hr = S_OK;
  543. HANDLE hTemp = NULL;
  544. DWORD dwError = 0;
  545. if (!phFileHandle)
  546. phFileHandle = &hTemp;
  547. *phFileHandle = CreateFile(lpFileName, dwDesiredAccess, dwShareMode, lpSecurityAttributes, dwCreationDisposition, dwFlagsAndAttributes, hTemplateFile);
  548. if (INVALID_HANDLE_VALUE == *phFileHandle)
  549. {
  550. dwError = GetLastError();
  551. hr = ResultFromWin32(dwError);
  552. }
  553. if (hTemp)
  554. CloseHandle(hTemp);
  555. return hr;
  556. }
  557. HRESULT WriteFileWrap(HANDLE hFile, LPCVOID lpBuffer, DWORD nNumberOfBytesToWrite, LPDWORD lpNumberOfBytesWritten, LPOVERLAPPED lpOverlapped)
  558. {
  559. HRESULT hr = S_OK;
  560. DWORD dwError = 0;
  561. if (!WriteFile(hFile, lpBuffer, nNumberOfBytesToWrite, lpNumberOfBytesWritten, lpOverlapped))
  562. {
  563. dwError = GetLastError();
  564. hr = ResultFromWin32(dwError);
  565. }
  566. return hr;
  567. }
  568. HRESULT DeleteFileHrWrap(LPCWSTR pszPath)
  569. {
  570. HRESULT hr = S_OK;
  571. DWORD dwError = 0;
  572. if (!DeleteFileW(pszPath))
  573. {
  574. dwError = GetLastError();
  575. hr = ResultFromWin32(dwError);
  576. }
  577. return hr;
  578. }
  579. HRESULT HrSHFileOpDeleteFile(HWND hwnd, FILEOP_FLAGS dwFlags, LPTSTR pszPath)
  580. {
  581. HRESULT hr = S_OK;
  582. SHFILEOPSTRUCT FileOp = {0};
  583. pszPath[lstrlen(pszPath)+1] = 0; // Ensure double terminated.
  584. FileOp.wFunc = FO_DELETE;
  585. FileOp.fAnyOperationsAborted = TRUE;
  586. FileOp.hwnd = hwnd;
  587. FileOp.pFrom = pszPath;
  588. FileOp.fFlags = dwFlags;
  589. if (SHFileOperation(&FileOp))
  590. {
  591. hr = ResultFromLastError();
  592. }
  593. return hr;
  594. }
  595. HRESULT GetPrivateProfileStringHrWrap(LPCWSTR lpAppName, LPCWSTR lpKeyName, LPCWSTR lpDefault, LPWSTR lpReturnedString, DWORD nSize, LPCWSTR lpFileName)
  596. {
  597. HRESULT hr = S_OK;
  598. DWORD chGot = GetPrivateProfileStringW(lpAppName, lpKeyName, lpDefault, lpReturnedString, nSize, lpFileName);
  599. // What else can indicate an error value?
  600. if (0 == chGot)
  601. {
  602. hr = ResultFromLastError();
  603. if (SUCCEEDED(hr))
  604. hr = E_FAIL;
  605. }
  606. return hr;
  607. }
  608. /////////////////////////////////////////////////////////////////////
  609. // Registry Helpers
  610. /////////////////////////////////////////////////////////////////////
  611. HRESULT HrRegOpenKeyEx(HKEY hKey, LPCTSTR lpSubKey, DWORD ulOptions, REGSAM samDesired, PHKEY phkResult)
  612. {
  613. DWORD dwError = RegOpenKeyEx(hKey, lpSubKey, ulOptions, samDesired, phkResult);
  614. return ResultFromWin32(dwError);
  615. }
  616. HRESULT HrRegCreateKeyEx(HKEY hKey, LPCTSTR lpSubKey, DWORD Reserved, LPTSTR lpClass, DWORD dwOptions,
  617. REGSAM samDesired, LPSECURITY_ATTRIBUTES lpSecurityAttributes, PHKEY phkResult, LPDWORD lpdwDisposition)
  618. {
  619. DWORD dwError = RegCreateKeyEx(hKey, lpSubKey, Reserved, lpClass, dwOptions, samDesired, lpSecurityAttributes, phkResult, lpdwDisposition);
  620. return ResultFromWin32(dwError);
  621. }
  622. HRESULT HrRegQueryValueEx(IN HKEY hKey, IN LPCTSTR lpValueName, IN LPDWORD lpReserved, IN LPDWORD lpType, IN LPBYTE lpData, IN LPDWORD lpcbData)
  623. {
  624. DWORD dwError = RegQueryValueEx(hKey, lpValueName, lpReserved, lpType, lpData, lpcbData);
  625. return ResultFromWin32(dwError);
  626. }
  627. HRESULT HrRegSetValueEx(IN HKEY hKey, IN LPCTSTR lpValueName, IN DWORD dwReserved, IN DWORD dwType, IN CONST BYTE *lpData, IN DWORD cbData)
  628. {
  629. DWORD dwError = RegSetValueEx(hKey, lpValueName, dwReserved, dwType, lpData, cbData);
  630. return ResultFromWin32(dwError);
  631. }
  632. HRESULT HrRegEnumKey(HKEY hKey, DWORD dwIndex, LPTSTR lpName, DWORD cbName)
  633. {
  634. DWORD dwError = RegEnumKey(hKey, dwIndex, lpName, cbName);
  635. return ResultFromWin32(dwError);
  636. }
  637. HRESULT HrRegEnumValue(HKEY hKey, DWORD dwIndex, LPTSTR lpValueName, LPDWORD lpcValueName, LPDWORD lpReserved,
  638. LPDWORD lpType, LPBYTE lpData, LPDWORD lpcbData)
  639. {
  640. DWORD dwError = RegEnumValue(hKey, dwIndex, lpValueName, lpcValueName, lpReserved, lpType, lpData, lpcbData);
  641. return ResultFromWin32(dwError);
  642. }
  643. HRESULT HrRegQueryInfoKey(HKEY hKey, LPTSTR lpClass, LPDWORD lpcClass, LPDWORD lpReserved, LPDWORD lpcSubKeys, LPDWORD lpcMaxSubKeyLen,
  644. LPDWORD lpcMaxClassLen, LPDWORD lpcValues, LPDWORD lpcMaxValueNameLen, LPDWORD lpcMaxValueLen, LPDWORD lpcbSecurityDescriptor, PFILETIME lpftLastWriteTime)
  645. {
  646. DWORD dwError = RegQueryInfoKey(hKey, lpClass, lpcClass, lpReserved, lpcSubKeys, lpcMaxSubKeyLen,
  647. lpcMaxClassLen, lpcValues, lpcMaxValueNameLen, lpcMaxValueLen, lpcbSecurityDescriptor, lpftLastWriteTime);
  648. return ResultFromWin32(dwError);
  649. }
  650. HRESULT HrBStrRegQueryValue(IN HKEY hKey, IN LPCTSTR lpValueName, OUT BSTR * pbstr)
  651. {
  652. TCHAR szValue[MAX_PATH];
  653. DWORD dwType;
  654. DWORD cbSize = sizeof(szValue);
  655. HRESULT hr = HrRegQueryValueEx(hKey, lpValueName, 0, &dwType, (BYTE *)szValue, &cbSize);
  656. *pbstr = NULL;
  657. if (SUCCEEDED(hr))
  658. {
  659. hr = HrSysAllocStringW(szValue, pbstr);
  660. }
  661. return hr;
  662. }
  663. HRESULT HrSHGetValue(IN HKEY hKey, IN LPCTSTR pszSubKey, OPTIONAL IN LPCTSTR pszValue, OPTIONAL OUT LPDWORD pdwType,
  664. OPTIONAL OUT LPVOID pvData, OPTIONAL OUT LPDWORD pcbData)
  665. {
  666. DWORD dwError = SHGetValue(hKey, pszSubKey, pszValue, pdwType, pvData, pcbData);
  667. return ResultFromWin32(dwError);
  668. }
  669. HRESULT HrSHSetValue(IN HKEY hkey, IN LPCTSTR pszSubKey, OPTIONAL IN LPCTSTR pszValue, DWORD dwType, OPTIONAL OUT LPVOID pvData, IN DWORD cbData)
  670. {
  671. DWORD dwError = SHSetValue(hkey, pszSubKey, pszValue, dwType, pvData, cbData);
  672. return ResultFromWin32(dwError);
  673. }
  674. HRESULT HrRegSetValueString(IN HKEY hKey, IN LPCTSTR pszSubKey, IN LPCTSTR pszValueName, OUT LPCWSTR pszString)
  675. {
  676. DWORD cbSize = ((lstrlenW(pszString) + 1) * sizeof(pszString[0]));
  677. return HrSHSetValue(hKey, pszSubKey, pszValueName, REG_SZ, (BYTE *)pszString, cbSize);
  678. }
  679. HRESULT HrRegGetValueString(IN HKEY hKey, IN LPCTSTR pszSubKey, IN LPCTSTR pszValueName, IN LPWSTR pszString, IN DWORD cchSize)
  680. {
  681. DWORD dwType;
  682. DWORD cbSize = (cchSize * sizeof(pszString[0]));
  683. HRESULT hr = HrSHGetValue(hKey, pszSubKey, pszValueName, &dwType, (BYTE *)pszString, &cbSize);
  684. if (SUCCEEDED(hr) && (REG_SZ != dwType))
  685. {
  686. hr = E_FAIL;
  687. }
  688. return hr;
  689. }
  690. /*****************************************************************************\
  691. DESCRIPTION:
  692. This function will store paths in the registry. The user calls the
  693. fuction with full paths are they are converted to relative path. The
  694. strings prefer to be stored in REG_EXPAND_SZ, but it will fallback to
  695. REG_SZ if needed.
  696. \*****************************************************************************/
  697. HRESULT HrRegSetPath(IN HKEY hKey, IN LPCTSTR pszSubKey, IN LPCTSTR pszValueName, BOOL fUseExpandSZ, OUT LPCWSTR pszPath)
  698. {
  699. TCHAR szFinalPath[MAX_PATH];
  700. if (!PathUnExpandEnvStrings(pszPath, szFinalPath, ARRAYSIZE(szFinalPath)))
  701. {
  702. StrCpyN(szFinalPath, pszPath, ARRAYSIZE(szFinalPath)); // We failed so use the original.
  703. }
  704. DWORD cbSize = ((lstrlenW(szFinalPath) + 1) * sizeof(szFinalPath[0]));
  705. HRESULT hr = E_FAIL;
  706. if (fUseExpandSZ)
  707. {
  708. hr = HrSHSetValue(hKey, pszSubKey, pszValueName, REG_EXPAND_SZ, (BYTE *)szFinalPath, cbSize);
  709. }
  710. if (FAILED(hr))
  711. {
  712. // Maybe it already exists as a REG_SZ so we will store it there. Note that we are still storing it
  713. // unexpanded even thought it's in REG_SZ. If the caller does not like it, use
  714. // another function like SHRegSetPath().
  715. cbSize = ((lstrlenW(szFinalPath) + 1) * sizeof(szFinalPath[0]));
  716. hr = HrSHSetValue(hKey, pszSubKey, pszValueName, REG_SZ, (BYTE *)szFinalPath, cbSize);
  717. }
  718. return hr;
  719. }
  720. HRESULT HrRegGetPath(IN HKEY hKey, IN LPCTSTR pszSubKey, IN LPCTSTR pszValueName, IN LPWSTR pszPath, IN DWORD cchSize)
  721. {
  722. TCHAR szFinalPath[MAX_PATH];
  723. DWORD dwType;
  724. DWORD cbSize = sizeof(szFinalPath);
  725. HRESULT hr = HrSHGetValue(hKey, pszSubKey, pszValueName, &dwType, (BYTE *)szFinalPath, &cbSize);
  726. if (SUCCEEDED(hr) &&
  727. ((REG_EXPAND_SZ == dwType) || (REG_SZ == dwType)))
  728. {
  729. if (0 == SHExpandEnvironmentStrings(szFinalPath, pszPath, cchSize))
  730. {
  731. StrCpyN(pszPath, szFinalPath, cchSize); // We failed so use the original.
  732. }
  733. }
  734. return hr;
  735. }
  736. HRESULT HrRegDeleteValue(IN HKEY hKey, IN LPCTSTR pszSubKey, IN LPCTSTR pszValueName)
  737. {
  738. HRESULT hr = S_OK;
  739. HKEY hKeySub = hKey;
  740. if (pszSubKey)
  741. {
  742. hr = HrRegOpenKeyEx(hKey, pszSubKey, 0, KEY_WRITE, &hKeySub);
  743. }
  744. if (SUCCEEDED(hr))
  745. {
  746. DWORD dwError = RegDeleteValue(hKeySub, pszValueName);
  747. hr = ResultFromWin32(dwError);
  748. }
  749. if (hKeySub == hKey)
  750. {
  751. RegCloseKey(hKeySub);
  752. }
  753. return hr;
  754. }
  755. DWORD HrRegGetDWORD(HKEY hKey, LPCWSTR szKey, LPCWSTR szValue, DWORD dwDefault)
  756. {
  757. DWORD dwResult = dwDefault;
  758. DWORD cbSize = sizeof(dwResult);
  759. DWORD dwType;
  760. DWORD dwError = SHGetValue(hKey, szKey, szValue, &dwType, &dwResult, &cbSize);
  761. if ((ERROR_SUCCESS != dwError) ||
  762. ((REG_DWORD != dwType) && (REG_BINARY != dwType)) || (sizeof(dwResult) != cbSize))
  763. {
  764. return dwDefault;
  765. }
  766. return dwResult;
  767. }
  768. HRESULT HrRegSetDWORD(HKEY hKey, LPCWSTR szKey, LPCWSTR szValue, DWORD dwData)
  769. {
  770. DWORD dwError = SHSetValue(hKey, szKey, szValue, REG_DWORD, &dwData, sizeof(dwData));
  771. return ResultFromWin32(dwError);
  772. }
  773. /////////////////////////////////////////////////////////////////////
  774. // Palette Helpers
  775. /////////////////////////////////////////////////////////////////////
  776. COLORREF GetNearestPaletteColor(HPALETTE hpal, COLORREF rgb)
  777. {
  778. PALETTEENTRY pe = {0};
  779. UINT nIndex = GetNearestPaletteIndex(hpal, rgb & 0x00FFFFFF);
  780. if (CLR_INVALID != nIndex)
  781. {
  782. GetPaletteEntries(hpal, nIndex, 1, &pe);
  783. }
  784. return RGB(pe.peRed, pe.peGreen, pe.peBlue);
  785. }
  786. BOOL IsPaletteColor(HPALETTE hpal, COLORREF rgb)
  787. {
  788. return GetNearestPaletteColor(hpal, rgb) == (rgb & 0xFFFFFF);
  789. }
  790. /////////////////////////////////////////////////////////////////////
  791. // Other Helpers
  792. /////////////////////////////////////////////////////////////////////
  793. HRESULT HrRewindStream(IStream * pstm)
  794. {
  795. LARGE_INTEGER liOrigin = {0,0};
  796. return pstm->Seek(liOrigin, STREAM_SEEK_SET, NULL);
  797. }
  798. #define SET_FLAG(dwAllFlags, dwFlag) ((dwAllFlags) |= (dwFlag))
  799. #define IS_FLAG_SET(dwAllFlags, dwFlag) ((BOOL)((dwAllFlags) & (dwFlag)))
  800. HRESULT HrByteToStream(LPSTREAM *lppstm, LPBYTE lpb, ULONG cb)
  801. {
  802. // Locals
  803. HRESULT hr=S_OK;
  804. LARGE_INTEGER liOrigin = {0,0};
  805. // Create H Global Stream
  806. hr = CreateStreamOnHGlobal (NULL, TRUE, lppstm);
  807. if (FAILED(hr))
  808. goto exit;
  809. // Write String
  810. hr = (*lppstm)->Write (lpb, cb, NULL);
  811. if (FAILED(hr))
  812. goto exit;
  813. // Rewind the steam
  814. hr = (*lppstm)->Seek(liOrigin, STREAM_SEEK_SET, NULL);
  815. if (FAILED(hr))
  816. goto exit;
  817. exit:
  818. // Done
  819. return hr;
  820. }
  821. const char szDayOfWeekArray[7][4] = { "Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat" } ;
  822. const char szMonthOfYearArray[12][4] = { "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" } ;
  823. void GetDateString(char * szSentDateString, ULONG stringLen)
  824. {
  825. // Sent Date
  826. SYSTEMTIME stSentTime;
  827. CHAR szMonth[10], szWeekDay[12] ;
  828. GetSystemTime(&stSentTime);
  829. lstrcpynA(szWeekDay, szDayOfWeekArray[stSentTime.wDayOfWeek], ARRAYSIZE(szWeekDay)) ;
  830. lstrcpynA(szMonth, szMonthOfYearArray[stSentTime.wMonth-1], ARRAYSIZE(szMonth)) ;
  831. wnsprintfA(szSentDateString, stringLen, "%s, %u %s %u %2d:%02d:%02d ", (LPSTR) szWeekDay, stSentTime.wDay,
  832. (LPSTR) szMonth, stSentTime.wYear, stSentTime.wHour,
  833. stSentTime.wMinute, stSentTime.wSecond) ;
  834. }
  835. /*****************************************************************************\
  836. PARAMETERS:
  837. RETURN: Win32 HRESULT (Not Script Safe).
  838. SUCCEEDED(hr) for OK and out params filled in.
  839. FAILED(hr) for all errors.
  840. \*****************************************************************************/
  841. HRESULT GetQueryStringValue(BSTR bstrURL, LPCWSTR pwszValue, LPWSTR pwszData, int cchSizeData)
  842. {
  843. HRESULT hr = E_FAIL;
  844. LPCWSTR pwszIterate = bstrURL;
  845. pwszIterate = StrChrW(pwszIterate, L'?'); // Advance to Query part of URL.
  846. while (pwszIterate && pwszIterate[0])
  847. {
  848. pwszIterate++; // Start at first value
  849. LPCWSTR pwszEndOfValue = StrChrW(pwszIterate, L'=');
  850. if (!pwszEndOfValue)
  851. break;
  852. int cchValueSize = (INT)(UINT)(pwszEndOfValue - pwszIterate);
  853. if (0 == StrCmpNIW(pwszValue, pwszIterate, cchValueSize))
  854. {
  855. int cchSizeToCopy = cchSizeData; // Copy rest of line by default.
  856. pwszIterate = StrChrW(pwszEndOfValue, L'&');
  857. if (pwszIterate)
  858. {
  859. cchSizeToCopy = (INT)(UINT)(pwszIterate - pwszEndOfValue);
  860. }
  861. // It matches, now get the Data.
  862. StrCpyNW(pwszData, (pwszEndOfValue + 1), cchSizeToCopy);
  863. hr = S_OK;
  864. break;
  865. }
  866. else
  867. {
  868. pwszIterate = StrChrW(pwszEndOfValue, L'&');
  869. }
  870. }
  871. return hr;
  872. }
  873. BOOL _InitComCtl32()
  874. {
  875. static BOOL fInitialized = FALSE;
  876. if (!fInitialized)
  877. {
  878. INITCOMMONCONTROLSEX icc;
  879. icc.dwSize = sizeof(INITCOMMONCONTROLSEX);
  880. icc.dwICC = (ICC_ANIMATE_CLASS | ICC_USEREX_CLASSES | ICC_COOL_CLASSES | ICC_INTERNET_CLASSES | ICC_PAGESCROLLER_CLASS | ICC_NATIVEFNTCTL_CLASS | ICC_LISTVIEW_CLASSES | ICC_LINK_CLASS);
  881. fInitialized = InitCommonControlsEx(&icc);
  882. }
  883. return fInitialized;
  884. }
  885. DWORD GetCurrentSessionID(void)
  886. {
  887. DWORD dwProcessID = (DWORD) -1;
  888. ProcessIdToSessionId(GetCurrentProcessId(), &dwProcessID);
  889. return dwProcessID;
  890. }
  891. typedef struct
  892. {
  893. LPCWSTR pszRegKey;
  894. LPCWSTR pszRegValue;
  895. } TSPERFFLAG_ITEM;
  896. const TSPERFFLAG_ITEM s_TSPerfFlagItems[] =
  897. {
  898. {L"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\Remote\\%d", L"ActiveDesktop"}, // TSPerFlag_NoADWallpaper
  899. {L"Remote\\%d\\Control Panel\\Desktop", L"Wallpaper"}, // TSPerFlag_NoWallpaper
  900. {L"Software\\Microsoft\\Windows\\CurrentVersion\\ThemeManager\\Remote\\%d", L"ThemeActive"}, // TSPerFlag_NoVisualStyles
  901. {L"Remote\\%d\\Control Panel\\Desktop", L"DragFullWindows"}, // TSPerFlag_NoWindowDrag
  902. {L"Remote\\%d\\Control Panel\\Desktop", L"SmoothScroll"}, // TSPerFlag_NoAnimation
  903. };
  904. BOOL IsTSPerfFlagEnabled(enumTSPerfFlag eTSFlag)
  905. {
  906. BOOL fIsTSFlagEnabled = FALSE;
  907. static BOOL s_fTSSession = -10;
  908. if (-10 == s_fTSSession)
  909. {
  910. s_fTSSession = GetSystemMetrics(SM_REMOTESESSION);
  911. }
  912. if (s_fTSSession)
  913. {
  914. TCHAR szTemp[MAX_PATH];
  915. DWORD dwType;
  916. DWORD cbSize = sizeof(szTemp);
  917. TCHAR szRegKey[MAX_PATH];
  918. wnsprintf(szRegKey, ARRAYSIZE(szRegKey), s_TSPerfFlagItems[eTSFlag].pszRegKey, GetCurrentSessionID());
  919. if (ERROR_SUCCESS == SHGetValueW(HKEY_CURRENT_USER, szRegKey, s_TSPerfFlagItems[eTSFlag].pszRegValue, &dwType, (void *)szTemp, &cbSize))
  920. {
  921. fIsTSFlagEnabled = TRUE;
  922. }
  923. }
  924. return fIsTSFlagEnabled;
  925. }
  926. HRESULT HrShellExecute(HWND hwnd, LPCTSTR lpVerb, LPCTSTR lpFile, LPCTSTR lpParameters, LPCTSTR lpDirectory, INT nShowCmd)
  927. {
  928. HRESULT hr = S_OK;
  929. HINSTANCE hReturn = ShellExecute(hwnd, lpVerb, lpFile, lpParameters, lpDirectory, nShowCmd);
  930. if ((HINSTANCE)32 > hReturn)
  931. {
  932. hr = ResultFromLastError();
  933. }
  934. return hr;
  935. }
  936. HRESULT StrReplaceToken(IN LPCTSTR pszToken, IN LPCTSTR pszReplaceValue, IN LPTSTR pszString, IN DWORD cchSize)
  937. {
  938. HRESULT hr = S_OK;
  939. LPTSTR pszTempLastHalf = NULL;
  940. LPTSTR pszNextToken = pszString;
  941. while (0 != (pszNextToken = StrStrI(pszNextToken, pszToken)))
  942. {
  943. // We found one.
  944. LPTSTR pszPastToken = pszNextToken + lstrlen(pszToken);
  945. Str_SetPtr(&pszTempLastHalf, pszPastToken); // Keep a copy because we will overwrite it.
  946. pszNextToken[0] = 0; // Remove the rest of the string.
  947. StrCatBuff(pszString, pszReplaceValue, cchSize);
  948. StrCatBuff(pszString, pszTempLastHalf, cchSize);
  949. pszNextToken += lstrlen(pszReplaceValue);
  950. }
  951. Str_SetPtr(&pszTempLastHalf, NULL);
  952. return hr;
  953. }
  954. HRESULT HrWritePrivateProfileStringW(LPCWSTR pszAppName, LPCWSTR pszKeyName, LPCWSTR pszString, LPCWSTR pszFileName)
  955. {
  956. HRESULT hr = S_OK;
  957. if (!WritePrivateProfileStringW(pszAppName, pszKeyName, pszString, pszFileName))
  958. {
  959. hr = ResultFromLastError();
  960. }
  961. return hr;
  962. }
  963. BOOL IUnknown_CompareCLSID(IN IUnknown * punk, IN CLSID clsid)
  964. {
  965. BOOL fIsEqual = FALSE;
  966. if (punk)
  967. {
  968. CLSID clsidPageID;
  969. HRESULT hr = IUnknown_GetClassID(punk, &clsidPageID);
  970. if (SUCCEEDED(hr) && IsEqualCLSID(clsidPageID, clsid))
  971. {
  972. fIsEqual = TRUE;
  973. }
  974. }
  975. return fIsEqual;
  976. }
  977. HRESULT IEnumUnknown_FindCLSID(IN IUnknown * punk, IN CLSID clsid, OUT IUnknown ** ppunkFound)
  978. {
  979. HRESULT hr = E_INVALIDARG;
  980. if (punk && ppunkFound)
  981. {
  982. IEnumUnknown * pEnum;
  983. *ppunkFound = NULL;
  984. hr = punk->QueryInterface(IID_PPV_ARG(IEnumUnknown, &pEnum));
  985. if (SUCCEEDED(hr))
  986. {
  987. IUnknown * punkToTry;
  988. ULONG ulFetched;
  989. pEnum->Reset();
  990. hr = E_FAIL;
  991. while (SUCCEEDED(pEnum->Next(1, &punkToTry, &ulFetched)) &&
  992. (1 == ulFetched))
  993. {
  994. if (IUnknown_CompareCLSID(punkToTry, clsid))
  995. {
  996. *ppunkFound = punkToTry;
  997. hr = S_OK;
  998. break;
  999. }
  1000. punkToTry->Release();
  1001. }
  1002. pEnum->Release();
  1003. }
  1004. }
  1005. return hr;
  1006. }
  1007. BYTE WINAPI MyStrToByte(LPCTSTR sz)
  1008. {
  1009. BYTE l=0;
  1010. while (*sz >= TEXT('0') && *sz <= TEXT('9'))
  1011. {
  1012. l = (BYTE) l*10 + (*sz++ - TEXT('0'));
  1013. }
  1014. return l;
  1015. }
  1016. COLORREF ConvertColor(LPTSTR pszColor)
  1017. {
  1018. BYTE RGBTemp[3];
  1019. LPTSTR pszTemp = pszColor;
  1020. UINT i;
  1021. if (!pszColor || !*pszColor)
  1022. {
  1023. return RGB(0,0,0);
  1024. }
  1025. for (i =0; i < 3; i++)
  1026. {
  1027. // Remove leading spaces
  1028. while (*pszTemp == TEXT(' '))
  1029. {
  1030. pszTemp++;
  1031. }
  1032. // Set pszColor to the beginning of the number
  1033. pszColor = pszTemp;
  1034. // Find the end of the number and null terminate
  1035. while ((*pszTemp) && (*pszTemp != TEXT(' ')))
  1036. {
  1037. pszTemp++;
  1038. }
  1039. if (*pszTemp != TEXT('\0'))
  1040. {
  1041. *pszTemp = TEXT('\0');
  1042. }
  1043. pszTemp++;
  1044. RGBTemp[i] = MyStrToByte(pszColor);
  1045. }
  1046. return (RGB(RGBTemp[0], RGBTemp[1], RGBTemp[2]));
  1047. }
  1048. // Paremeters:
  1049. // hwndOwner -- owner window
  1050. // idTemplate -- specifies template (e.g., "Can't open %2%s\n\n%1%s")
  1051. // hr -- specifies the HRESULT error code
  1052. // pszParam -- specifies the 2nd parameter to idTemplate
  1053. // dwFlags -- flags for MessageBox
  1054. UINT ErrorMessageBox(HWND hwndOwner, LPCTSTR pszTitle, UINT idTemplate, HRESULT hr, LPCTSTR pszParam, UINT dwFlags)
  1055. {
  1056. TCHAR szErrNumString[MAX_PATH * 2];
  1057. TCHAR szTemplate[MAX_PATH * 2];
  1058. TCHAR szErrMsg[MAX_PATH * 2];
  1059. if (!FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM, NULL, hr, 0, szErrNumString, ARRAYSIZE(szErrNumString), NULL))
  1060. {
  1061. szErrNumString[0] = 0; // We will not be able to display an error message.
  1062. }
  1063. // These error messages are so useless to customers, that we prefer to leave it blank.
  1064. if ((E_INVALIDARG == hr) ||
  1065. (ResultFromWin32(ERROR_INVALID_PARAMETER) == hr))
  1066. {
  1067. szErrNumString[0] = 0;
  1068. }
  1069. LoadString(HINST_THISDLL, idTemplate, szTemplate, ARRAYSIZE(szTemplate));
  1070. if (pszParam)
  1071. {
  1072. wnsprintf(szErrMsg, ARRAYSIZE(szErrMsg), szTemplate, szErrNumString, pszParam);
  1073. }
  1074. else
  1075. {
  1076. wnsprintf(szErrMsg, ARRAYSIZE(szErrMsg), szTemplate, szErrNumString);
  1077. }
  1078. return MessageBox(hwndOwner, szErrMsg, pszTitle, (MB_OK | MB_ICONERROR));
  1079. }
  1080. HRESULT DisplayThemeErrorDialog(HWND hwndParent, HRESULT hrError, UINT nTitle, UINT nTemplate)
  1081. {
  1082. HRESULT hr = S_OK;
  1083. if (FAILED(hrError))
  1084. {
  1085. hr = ResultFromWin32(ERROR_CANCELLED);
  1086. if (!g_fInSetup && // Don't display an error during setup.
  1087. (ResultFromWin32(ERROR_CANCELLED) != hrError))
  1088. {
  1089. //---- get error from theme manager ----
  1090. WCHAR szErrorMsg[MAX_PATH*2];
  1091. WCHAR szTitle[MAX_PATH];
  1092. szErrorMsg[0] = 0; // In case the error function fails.
  1093. if (FAILED(hrError))
  1094. {
  1095. PARSE_ERROR_INFO Info = {sizeof(Info)};
  1096. if (SUCCEEDED(GetThemeParseErrorInfo(&Info)))
  1097. {
  1098. lstrcpy(szErrorMsg, Info.szMsg);
  1099. }
  1100. else
  1101. {
  1102. *szErrorMsg = 0; // no error avail
  1103. }
  1104. }
  1105. // We want to display UI if an error occured here. We want to do
  1106. // it instead of our parent because THEMELOADPARAMS contains
  1107. // extra error information that we can't pass back to the caller.
  1108. // However, we will only display error UI if our caller wants us
  1109. // to. We determine that by the fact that they make an hwnd available
  1110. // to us. We get the hwnd by getting our site pointer and getting
  1111. // the hwnd via ::GetWindow().
  1112. LoadString(HINST_THISDLL, nTitle, szTitle, ARRAYSIZE(szTitle));
  1113. ErrorMessageBox(hwndParent, szTitle, nTemplate, hrError, szErrorMsg, (MB_OK | MB_ICONEXCLAMATION));
  1114. }
  1115. }
  1116. return hr;
  1117. }
  1118. BOOL IsOSNT(void)
  1119. {
  1120. OSVERSIONINFOA osVerInfoA;
  1121. osVerInfoA.dwOSVersionInfoSize = sizeof(osVerInfoA);
  1122. if (!GetVersionExA(&osVerInfoA))
  1123. return VER_PLATFORM_WIN32_WINDOWS; // Default to this.
  1124. return (VER_PLATFORM_WIN32_NT == osVerInfoA.dwPlatformId);
  1125. }
  1126. DWORD GetOSVer(void)
  1127. {
  1128. OSVERSIONINFOA osVerInfoA;
  1129. osVerInfoA.dwOSVersionInfoSize = sizeof(osVerInfoA);
  1130. if (!GetVersionExA(&osVerInfoA))
  1131. return VER_PLATFORM_WIN32_WINDOWS; // Default to this.
  1132. return osVerInfoA.dwMajorVersion;
  1133. }
  1134. BOOL HideBackgroundTabOnTermServices(void)
  1135. {
  1136. BOOL fHideThisPage = FALSE;
  1137. TCHAR szSessionName[WINSTATIONNAME_LENGTH * 2];
  1138. TCHAR szBuf[MAX_PATH*2];
  1139. TCHAR szActualValue[MAX_PATH*2];
  1140. DWORD dwLen;
  1141. DWORD i;
  1142. ZeroMemory((PVOID)szSessionName,sizeof(szSessionName));
  1143. dwLen = GetEnvironmentVariable(TEXT("SESSIONNAME"), szSessionName, ARRAYSIZE(szSessionName));
  1144. if (dwLen != 0)
  1145. {
  1146. // Now that we have the session name, search for the # character.
  1147. for(i = 0; i < dwLen; i++)
  1148. {
  1149. if (szSessionName[i] == TEXT('#'))
  1150. {
  1151. szSessionName[i] = TEXT('\0');
  1152. break;
  1153. }
  1154. }
  1155. // Here is what we are looking for in NT5:
  1156. // HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Terminal Server\
  1157. // WinStations\RDP-Tcp\UserOverride\Control Panel\Desktop
  1158. //
  1159. // The value is:
  1160. // Wallpaper
  1161. wnsprintf(szBuf, ARRAYSIZE(szBuf), TEXT("%s\\%s\\%s\\%s"), WINSTATION_REG_NAME, szSessionName, WIN_USEROVERRIDE, REGSTR_PATH_DESKTOP);
  1162. // See if we can get the wallpaper string. This will fail if the key
  1163. // doesn't exist. This means the policy isn't set.
  1164. if (SUCCEEDED(HrRegGetValueString(HKEY_LOCAL_MACHINE, szBuf, L"Wallpaper", szActualValue, ARRAYSIZE(szActualValue))))
  1165. {
  1166. fHideThisPage = TRUE;
  1167. }
  1168. }
  1169. return fHideThisPage;
  1170. }
  1171. extern BOOL FadeEffectAvailable(void);
  1172. void LogStartInformation(void)
  1173. {
  1174. BOOL fTemp;
  1175. // Frequently users will report that something is broken in the Display CPL.
  1176. // However, the real problem is that someone turned on a policy that locks UI
  1177. // and the user didn't know that the policy was enabled. We log those here so
  1178. // it's quick to find those issues.
  1179. if (SHRestricted(REST_NODISPLAYCPL)) LogStatus("POLICY ENABLED: Do not show the Display CPL.");
  1180. if (SHRestricted(REST_NODISPLAYAPPEARANCEPAGE)) LogStatus("POLICY ENABLED: Hide the Themes and Appearance tab.");
  1181. if (SHRestricted(REST_NOTHEMESTAB)) LogStatus("POLICY ENABLED: Hide the Themes tab.");
  1182. if (SHRestricted(REST_NODISPBACKGROUND)) LogStatus("POLICY ENABLED: Hide the Desktop tab.");
  1183. if (SHRestricted(REST_NODISPSCREENSAVEPG)) LogStatus("POLICY ENABLED: Hide the ScreenSaver tab.");
  1184. if (SHRestricted(REST_NODISPSETTINGSPG)) LogStatus("POLICY ENABLED: Hide the Settings tab.");
  1185. if (SHRestricted(REST_NOVISUALSTYLECHOICE)) LogStatus("POLICY ENABLED: User not allowed to change the Visual Style.");
  1186. if (SHRestricted(REST_NOCOLORCHOICE)) LogStatus("POLICY ENABLED: User Not allowed to change the Visual Style Color Selection.");
  1187. if (SHRestricted(REST_NOSIZECHOICE)) LogStatus("POLICY ENABLED: User not allowed to change the Visual Style size selection.");
  1188. if (0 != SHGetRestriction(NULL,POLICY_KEY_EXPLORER,POLICY_VALUE_ANIMATION)) LogStatus("POLICY ENABLED: Policy disallows fade effect. (Effects dialog)");
  1189. if (0 != SHGetRestriction(NULL,POLICY_KEY_EXPLORER, POLICY_VALUE_KEYBOARDNAV)) LogStatus("POLICY ENABLED: Policy disallows changing underline key accell. (Effects dialog)");
  1190. if (0 != SHGetRestriction(NULL,POLICY_KEY_ACTIVEDESKTOP, SZ_POLICY_NOCHANGEWALLPAPER)) LogStatus("POLICY ENABLED: Policy disallows changing wallpaper. (Desktop tab)");
  1191. if (0 != SHGetRestriction(NULL,POLICY_KEY_SYSTEM, SZ_POLICY_NODISPSCREENSAVERPG)) LogStatus("POLICY ENABLED: Policy hides ScreenSaver page.");
  1192. if (0 != SHGetRestriction(SZ_REGKEY_POLICIES_DESKTOP, NULL, SZ_POLICY_SCREENSAVEACTIVE)) LogStatus("POLICY ENABLED: Policy forces screensaver on or off");
  1193. if (0 != SHGetRestriction(NULL,POLICY_KEY_EXPLORER, POLICY_VALUE_KEYBOARDNAV)) LogStatus("POLICY ENABLED: Policy disallows changing underline key accell. (Effects dialog)");
  1194. if (IsTSPerfFlagEnabled(TSPerFlag_NoAnimation)) LogStatus("POLICY ENABLED: TS Perf Policy disallows animations. (Effects dialog)");
  1195. if (IsTSPerfFlagEnabled(TSPerFlag_NoWindowDrag)) LogStatus("POLICY ENABLED: TS Perf Policy disallows full window drag. (Effects dialog)");
  1196. if (IsTSPerfFlagEnabled(TSPerFlag_NoVisualStyles)) LogStatus("POLICY ENABLED: TS Perf Policy disallows visual styles.");
  1197. if (IsTSPerfFlagEnabled(TSPerFlag_NoWallpaper)) LogStatus("POLICY ENABLED: TS Perf Policy disallows Wallpaper.");
  1198. if (IsTSPerfFlagEnabled(TSPerFlag_NoADWallpaper)) LogStatus("POLICY ENABLED: TS Perf Policy disallows AD Wallpaper.");
  1199. if (HideBackgroundTabOnTermServices()) LogStatus("POLICY ENABLED: TS Set a policy forcing a certain wallpaper, so the Desktop tab is hidden.");
  1200. if (!FadeEffectAvailable()) LogStatus("POLICY ENABLED: A policy forces Fade Effects off (Effects dialog)");
  1201. if (!ClassicSystemParametersInfo(SPI_GETFONTSMOOTHINGTYPE, 0, (PVOID)&fTemp, 0)) LogStatus("POLICY ENABLED: SPI_GETFONTSMOOTHINGTYPE hides FontSmoothing. (Effects dialog)");
  1202. if (ClassicSystemParametersInfo(SPI_GETUIEFFECTS, 0, (PVOID) &fTemp, 0) && !fTemp) LogStatus("POLICY ENABLED: SPI_GETUIEFFECTS hides lots of UI effects. (Effects dialog)");
  1203. if (ClassicSystemParametersInfo(SPI_GETGRADIENTCAPTIONS, 0, (PVOID) &fTemp, 0) && !fTemp) LogStatus("POLICY ENABLED: SPI_GETGRADIENTCAPTIONS turns off Caption bar Gradients. (Advance Appearance)");
  1204. }
  1205. void LogStatus(LPCSTR pszMessage, ...)
  1206. {
  1207. static int nLogOn = -1;
  1208. va_list vaParamList;
  1209. va_start(vaParamList, pszMessage);
  1210. if (-1 == nLogOn)
  1211. {
  1212. nLogOn = (SHRegGetBoolUSValue(SZ_THEMES, SZ_REGVALUE_LOGINFO, FALSE, FALSE) ? 1 : 0);
  1213. }
  1214. if (1 == nLogOn)
  1215. {
  1216. if (INVALID_HANDLE_VALUE == g_hLogFile)
  1217. {
  1218. TCHAR szPath[MAX_PATH];
  1219. if (GetWindowsDirectory(szPath, ARRAYSIZE(szPath)))
  1220. {
  1221. PathAppend(szPath, TEXT("Theme.log"));
  1222. g_hLogFile = CreateFile(szPath, (GENERIC_READ | GENERIC_WRITE), FILE_SHARE_READ, NULL, OPEN_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL);
  1223. if (INVALID_HANDLE_VALUE != g_hLogFile)
  1224. {
  1225. WCHAR szUserName[MAX_PATH];
  1226. CHAR szTimeDate[MAX_PATH];
  1227. CHAR szHeader[MAX_PATH];
  1228. FILETIME ftCurrentUTC;
  1229. FILETIME ftCurrent;
  1230. SYSTEMTIME stCurrent;
  1231. DWORD cbWritten;
  1232. SetFilePointer(g_hLogFile, 0, NULL, FILE_END);
  1233. GetLocalTime(&stCurrent);
  1234. SystemTimeToFileTime(&stCurrent, &ftCurrent);
  1235. LocalFileTimeToFileTime(&ftCurrent, &ftCurrentUTC);
  1236. SHFormatDateTimeA(&ftCurrentUTC, NULL, szTimeDate, ARRAYSIZE(szTimeDate));
  1237. ULONG cchUserSize = ARRAYSIZE(szUserName);
  1238. if (!GetUserNameEx(NameDisplay, szUserName, &cchUserSize) &&
  1239. !GetUserNameEx(NameUserPrincipal, szUserName, &cchUserSize) &&
  1240. !GetUserNameEx(NameSamCompatible, szUserName, &cchUserSize) &&
  1241. !GetUserNameEx(NameUniqueId, szUserName, &cchUserSize))
  1242. {
  1243. szUserName[0] = 0;
  1244. }
  1245. TCHAR szProcess[MAX_PATH];
  1246. if (!GetModuleFileName(NULL, szProcess, ARRAYSIZE(szProcess)))
  1247. {
  1248. szProcess[0] = 0;
  1249. }
  1250. wnsprintfA(szHeader, ARRAYSIZE(szHeader), "\r\n\r\n%hs - USER: %ls (%ls)\r\n", szTimeDate, szUserName, szProcess);
  1251. WriteFile(g_hLogFile, szHeader, lstrlenA(szHeader), &cbWritten, NULL);
  1252. // Log information that we need to do on every startup. (Like Policies that are on that confuse people)
  1253. LogStartInformation();
  1254. }
  1255. }
  1256. }
  1257. if (INVALID_HANDLE_VALUE != g_hLogFile)
  1258. {
  1259. CHAR szMessage[4000];
  1260. DWORD cbWritten;
  1261. wvsprintfA(szMessage, pszMessage, vaParamList);
  1262. WriteFile(g_hLogFile, szMessage, lstrlenA(szMessage), &cbWritten, NULL);
  1263. }
  1264. }
  1265. va_end(vaParamList);
  1266. }
  1267. void LogSystemMetrics(LPCSTR pszMessage, SYSTEMMETRICSALL * pSystemMetrics)
  1268. {
  1269. CHAR szSysMetrics[1024]; // Random because it's big.
  1270. if (pSystemMetrics)
  1271. {
  1272. wnsprintfA(szSysMetrics, ARRAYSIZE(szSysMetrics), "Sz(Brdr=%d, Scrl=%d, Cap=%d, Menu=%d, Icon=%d, DXIn=%d) Ft(Cap=%d(%d), SmCap=%d(%d), Menu=%d(%d), Stus=%d(%d), Msg=%d(%d))",
  1273. pSystemMetrics->schemeData.ncm.iBorderWidth,
  1274. pSystemMetrics->schemeData.ncm.iScrollWidth,
  1275. pSystemMetrics->schemeData.ncm.iCaptionHeight,
  1276. pSystemMetrics->schemeData.ncm.iMenuHeight,
  1277. pSystemMetrics->nIcon,
  1278. pSystemMetrics->nDYIcon,
  1279. pSystemMetrics->schemeData.ncm.lfCaptionFont.lfHeight,
  1280. pSystemMetrics->schemeData.ncm.lfCaptionFont.lfCharSet,
  1281. pSystemMetrics->schemeData.ncm.lfSmCaptionFont.lfHeight,
  1282. pSystemMetrics->schemeData.ncm.lfSmCaptionFont.lfCharSet,
  1283. pSystemMetrics->schemeData.ncm.lfMenuFont.lfHeight,
  1284. pSystemMetrics->schemeData.ncm.lfMenuFont.lfCharSet,
  1285. pSystemMetrics->schemeData.ncm.lfStatusFont.lfHeight,
  1286. pSystemMetrics->schemeData.ncm.lfStatusFont.lfCharSet,
  1287. pSystemMetrics->schemeData.ncm.lfMessageFont.lfHeight,
  1288. pSystemMetrics->schemeData.ncm.lfMessageFont.lfCharSet);
  1289. }
  1290. else
  1291. {
  1292. szSysMetrics[0] = 0;
  1293. }
  1294. LogStatus("SYSMET: %s: %s\r\n", pszMessage, szSysMetrics);
  1295. }
  1296. #define SzFromInt(sz, n) (wsprintf((LPTSTR)sz, (LPTSTR)TEXT("%d"), n), (LPTSTR)sz)
  1297. int WritePrivateProfileInt(LPCTSTR szApp, LPCTSTR szKey, int nDefault, LPCTSTR pszFileName)
  1298. {
  1299. CHAR sz[7];
  1300. return WritePrivateProfileString(szApp, szKey, SzFromInt(sz, nDefault), pszFileName);
  1301. }
  1302. #define SZ_RESOURCEDIR L"Resources"
  1303. HRESULT SHGetResourcePath(BOOL fLocaleNode, IN LPWSTR pszPath, IN DWORD cchSize)
  1304. {
  1305. DWORD dwFlags = (CSIDL_FLAG_CREATE | CSIDL_RESOURCES);
  1306. return SHGetFolderPath(NULL, dwFlags, NULL, 0, pszPath);
  1307. }
  1308. #define SZ_RESOURCEDIR_TOKEN TEXT("%ResourceDir%")
  1309. #define SZ_RESOURCELDIR_TOKEN TEXT("%ResourceDirL%")
  1310. HRESULT ExpandResourceDir(IN LPWSTR pszPath, IN DWORD cchSize)
  1311. {
  1312. HRESULT hr = S_OK;
  1313. BOOL fLocalized = FALSE;
  1314. LPCTSTR pszToken = StrStrW(pszPath, SZ_RESOURCEDIR_TOKEN);
  1315. if (!pszToken)
  1316. {
  1317. pszToken = StrStrW(pszPath, SZ_RESOURCELDIR_TOKEN);
  1318. }
  1319. // Do we have stuff to replace?
  1320. if (pszToken)
  1321. {
  1322. // Yes, so get the replacement value.
  1323. WCHAR szResourceDir[MAX_PATH];
  1324. hr = SHGetResourcePath(fLocalized, szResourceDir, ARRAYSIZE(szResourceDir));
  1325. if (SUCCEEDED(hr))
  1326. {
  1327. hr = StrReplaceToken((fLocalized ? SZ_RESOURCELDIR_TOKEN : SZ_RESOURCEDIR_TOKEN), szResourceDir, pszPath, cchSize);
  1328. }
  1329. }
  1330. return hr;
  1331. }
  1332. STDAPI SHPropertyBag_WritePunk(IN IPropertyBag * pPropertyPage, IN LPCWSTR pwzPropName, IN IUnknown * punk)
  1333. {
  1334. HRESULT hr = E_INVALIDARG;
  1335. if (pPropertyPage && pwzPropName)
  1336. {
  1337. VARIANT va;
  1338. va.vt = VT_UNKNOWN;
  1339. va.punkVal = punk;
  1340. hr = pPropertyPage->Write(pwzPropName, &va);
  1341. }
  1342. return hr;
  1343. }
  1344. STDAPI SHPropertyBag_ReadByRef(IN IPropertyBag * pPropertyPage, IN LPCWSTR pwzPropName, IN void * p, IN SIZE_T cbSize)
  1345. {
  1346. HRESULT hr = E_INVALIDARG;
  1347. if (pPropertyPage && pwzPropName && p)
  1348. {
  1349. VARIANT va;
  1350. hr = pPropertyPage->Read(pwzPropName, &va, NULL);
  1351. if (SUCCEEDED(hr))
  1352. {
  1353. if ((VT_BYREF == va.vt) && va.byref)
  1354. {
  1355. CopyMemory(p, va.byref, cbSize);
  1356. }
  1357. else
  1358. {
  1359. hr = E_FAIL;
  1360. }
  1361. }
  1362. }
  1363. return hr;
  1364. }
  1365. STDAPI SHPropertyBag_WriteByRef(IN IPropertyBag * pPropertyPage, IN LPCWSTR pwzPropName, IN void * p)
  1366. {
  1367. HRESULT hr = E_INVALIDARG;
  1368. if (pPropertyPage && pwzPropName && p)
  1369. {
  1370. VARIANT va;
  1371. va.vt = VT_BYREF;
  1372. va.byref = p;
  1373. hr = pPropertyPage->Write(pwzPropName, &va);
  1374. }
  1375. return hr;
  1376. }
  1377. LONG s_cSpiDummy = -1;
  1378. LONG *g_pcSpiThreads = &s_cSpiDummy;
  1379. void SPISetThreadCounter(LONG *pcThreads)
  1380. {
  1381. if (!pcThreads)
  1382. pcThreads = &s_cSpiDummy;
  1383. InterlockedExchangePointer((void **) &g_pcSpiThreads, pcThreads);
  1384. }
  1385. typedef struct
  1386. {
  1387. LPTHREAD_START_ROUTINE pfnThreadProc;
  1388. void *pvData;
  1389. UINT idThread;
  1390. }SPITHREAD;
  1391. DWORD CALLBACK _SPIWrapperThreadProc(void *pv)
  1392. {
  1393. SPITHREAD *pspi = (SPITHREAD *)pv;
  1394. DWORD dwRet = pspi->pfnThreadProc(pspi->pvData);
  1395. // then we check to see
  1396. if (0 == InterlockedDecrement(g_pcSpiThreads))
  1397. {
  1398. PostThreadMessage(pspi->idThread, WM_NULL, 0, 0);
  1399. }
  1400. delete pspi;
  1401. return dwRet;
  1402. }
  1403. BOOL SPICreateThread(LPTHREAD_START_ROUTINE pfnThreadProc, void *pvData)
  1404. {
  1405. SPITHREAD *pspi = new SPITHREAD;
  1406. if (pspi)
  1407. {
  1408. pspi->idThread = GetCurrentThreadId();
  1409. pspi->pfnThreadProc = pfnThreadProc;
  1410. pspi->pvData = pvData;
  1411. InterlockedIncrement(g_pcSpiThreads);
  1412. return SHCreateThread(_SPIWrapperThreadProc, pspi, (CTF_COINIT | CTF_INSIST | CTF_FREELIBANDEXIT), NULL);
  1413. }
  1414. else
  1415. {
  1416. // CTF_INSIST
  1417. pfnThreadProc(pvData);
  1418. return TRUE;
  1419. }
  1420. }
  1421. void PostMessageBroadAsync(IN UINT Msg, IN WPARAM wParam, IN LPARAM lParam)
  1422. {
  1423. // We don't want to hang our UI if other apps are hung or slow when
  1424. // we need to tell them to update their changes. So we choose this
  1425. // mechanism.
  1426. //
  1427. // The alternatives are:
  1428. // SendMessageCallback: Except we don't need to do anything when the apps
  1429. // are done.
  1430. // SendMessageTimeout: Except we don't want to incure any timeout.
  1431. PostMessage(HWND_BROADCAST, Msg, wParam, lParam);
  1432. }
  1433. typedef struct
  1434. {
  1435. BOOL fFree; // Do you need to call LocalFree() on pvData?
  1436. UINT uiAction;
  1437. UINT uiParam;
  1438. UINT fWinIni;
  1439. void * pvData;
  1440. CDimmedWindow* pDimmedWindow;
  1441. } SPIS_INFO;
  1442. DWORD SystemParametersInfoAsync_WorkerThread(IN void *pv)
  1443. {
  1444. SPIS_INFO * pSpisInfo = (SPIS_INFO *) pv;
  1445. HINSTANCE hInstance = LoadLibrary(TEXT("desk.cpl"));
  1446. if (pSpisInfo)
  1447. {
  1448. ClassicSystemParametersInfo(pSpisInfo->uiAction, pSpisInfo->uiParam, pSpisInfo->pvData, pSpisInfo->fWinIni);
  1449. if (pSpisInfo->fFree && pSpisInfo->pvData)
  1450. {
  1451. LocalFree(pSpisInfo->pvData);
  1452. }
  1453. if (pSpisInfo->pDimmedWindow)
  1454. {
  1455. pSpisInfo->pDimmedWindow->Release();
  1456. }
  1457. LocalFree(pv);
  1458. }
  1459. if (hInstance)
  1460. {
  1461. FreeLibrary(hInstance);
  1462. }
  1463. return 0;
  1464. }
  1465. void SystemParametersInfoAsync(IN UINT uiAction, IN UINT uiParam, IN void * pvParam, IN DWORD cbSize, IN UINT fWinIni, IN CDimmedWindow* pDimmedWindow)
  1466. {
  1467. // ClassicSystemParametersInfo() will hang if a top level window is hung (#162570) and USER will not fix that bug.
  1468. // Therefore, we need to make that API call on a background thread because we need to
  1469. // be more rebust than to hang.
  1470. SPIS_INFO * pSpisInfo = (SPIS_INFO *) LocalAlloc(LPTR, sizeof(*pSpisInfo));
  1471. if (pSpisInfo)
  1472. {
  1473. BOOL fAsyncOK = TRUE;
  1474. pSpisInfo->fFree = (0 != cbSize);
  1475. pSpisInfo->pvData = pvParam;
  1476. pSpisInfo->uiAction = uiAction;
  1477. pSpisInfo->uiParam = uiParam;
  1478. pSpisInfo->fWinIni = fWinIni;
  1479. pSpisInfo->pDimmedWindow = pDimmedWindow;
  1480. // Spawning thread is responsible for addref dimmed window, but not releasing
  1481. // that is the repsonsibility of the spawned thread
  1482. if (pSpisInfo->pDimmedWindow)
  1483. {
  1484. pSpisInfo->pDimmedWindow->AddRef();
  1485. }
  1486. if (pSpisInfo->fFree)
  1487. {
  1488. pSpisInfo->pvData = LocalAlloc(LPTR, cbSize);
  1489. if (!pSpisInfo->pvData)
  1490. {
  1491. pSpisInfo->pvData = pvParam;
  1492. fAsyncOK = FALSE;
  1493. pSpisInfo->fFree = FALSE;
  1494. }
  1495. else
  1496. {
  1497. CopyMemory(pSpisInfo->pvData, pvParam, cbSize);
  1498. }
  1499. }
  1500. if (fAsyncOK)
  1501. SPICreateThread(SystemParametersInfoAsync_WorkerThread, (void *)pSpisInfo);
  1502. else
  1503. SystemParametersInfoAsync_WorkerThread((void *)pSpisInfo);
  1504. }
  1505. }
  1506. HRESULT GetCurrentUserCustomName(LPWSTR pszDisplayName, DWORD cchSize)
  1507. {
  1508. HRESULT hr = S_OK;
  1509. #ifdef FEATURE_USECURRENTNAME_INCUSTOMTHEME
  1510. WCHAR szUserName[MAX_PATH];
  1511. ULONG cchUserSize = ARRAYSIZE(szUserName);
  1512. if (GetUserNameEx(NameDisplay, szUserName, &cchUserSize))
  1513. {
  1514. // It succeeded, so use it.
  1515. WCHAR szTemplate[MAX_PATH];
  1516. LoadString(HINST_THISDLL, IDS_CURRENTTHEME_DISPLAYNAME, szTemplate, ARRAYSIZE(szTemplate));
  1517. wnsprintf(pszDisplayName, cchSize, szTemplate, szUserName);
  1518. }
  1519. else
  1520. #endif // FEATURE_USECURRENTNAME_INCUSTOMTHEME
  1521. {
  1522. // It failed, so load "My Custom Theme". This may happen on personal.
  1523. LoadString(HINST_THISDLL, IDS_MYCUSTOMTHEME, pszDisplayName, cchSize);
  1524. }
  1525. return hr;
  1526. }
  1527. HRESULT InstallVisualStyle(IThemeManager * pThemeManager, LPCTSTR pszVisualStylePath, LPCTSTR pszVisualStyleColor, LPCTSTR pszVisualStyleSize)
  1528. {
  1529. HRESULT hr = E_OUTOFMEMORY;
  1530. CComVariant varTheme(pszVisualStylePath);
  1531. if (varTheme.bstrVal)
  1532. {
  1533. IThemeScheme * pVisualStyle;
  1534. hr = pThemeManager->get_schemeItem(varTheme, &pVisualStyle);
  1535. if (SUCCEEDED(hr))
  1536. {
  1537. CComVariant varStyleName(pszVisualStyleColor);
  1538. if (!varStyleName.bstrVal)
  1539. hr = E_OUTOFMEMORY;
  1540. else
  1541. {
  1542. IThemeStyle * pThemeStyle;
  1543. hr = pVisualStyle->get_item(varStyleName, &pThemeStyle);
  1544. if (SUCCEEDED(hr))
  1545. {
  1546. CComVariant varSizeName(pszVisualStyleSize);
  1547. if (!varSizeName.bstrVal)
  1548. hr = E_OUTOFMEMORY;
  1549. else
  1550. {
  1551. IThemeSize * pThemeSize;
  1552. hr = pThemeStyle->get_item(varSizeName, &pThemeSize);
  1553. if (SUCCEEDED(hr))
  1554. {
  1555. hr = pThemeStyle->put_SelectedSize(pThemeSize);
  1556. if (SUCCEEDED(hr))
  1557. {
  1558. hr = pVisualStyle->put_SelectedStyle(pThemeStyle);
  1559. if (SUCCEEDED(hr))
  1560. {
  1561. hr = pThemeManager->put_SelectedScheme(pVisualStyle);
  1562. }
  1563. }
  1564. pThemeSize->Release();
  1565. }
  1566. }
  1567. pThemeStyle->Release();
  1568. }
  1569. }
  1570. pVisualStyle->Release();
  1571. }
  1572. }
  1573. return hr;
  1574. }
  1575. // {B2A7FD52-301F-4348-B93A-638C6DE49229}
  1576. DEFINE_GUID(CLSID_WMPSkinMngr, 0xB2A7FD52, 0x301F, 0x4348, 0xB9, 0x3A, 0x63, 0x8C, 0x6D, 0xE4, 0x92, 0x29);
  1577. // {076F2FA6-ED30-448B-8CC5-3F3EF3529C7A}
  1578. DEFINE_GUID(IID_IWMPSkinMngr, 0x076F2FA6, 0xED30, 0x448B, 0x8C, 0xC5, 0x3F, 0x3E, 0xF3, 0x52, 0x9C, 0x7A);
  1579. HRESULT ApplyVisualStyle(LPCTSTR pszVisualStylePath, LPCTSTR pszVisualStyleColor, LPCTSTR pszVisualStyleSize)
  1580. {
  1581. HRESULT hr = S_OK;
  1582. HTHEMEFILE hThemeFile = NULL;
  1583. DWORD dwFlags = 0;
  1584. if (pszVisualStylePath)
  1585. {
  1586. // Load the skin
  1587. hr = OpenThemeFile(pszVisualStylePath, pszVisualStyleColor, pszVisualStyleSize, &hThemeFile, TRUE);
  1588. LogStatus("OpenThemeFile(%ls. %ls, %ls) returned hr=%#08lx.\r\n", pszVisualStylePath, pszVisualStyleColor, pszVisualStyleSize, hr);
  1589. }
  1590. if (SUCCEEDED(hr))
  1591. {
  1592. hr = ApplyTheme(hThemeFile, dwFlags, NULL);
  1593. LogStatus("ApplyTheme(%hs) returned hr=%#08lx.\r\n", (hThemeFile ? "hThemeFile" : "NULL"), hr);
  1594. }
  1595. if (hThemeFile)
  1596. {
  1597. CloseThemeFile(hThemeFile); // don't need to hold this open anymore
  1598. }
  1599. if (SUCCEEDED(hr))
  1600. {
  1601. CComBSTR bstrPath(pszVisualStylePath);
  1602. if (pszVisualStylePath && !bstrPath)
  1603. {
  1604. hr = E_OUTOFMEMORY;
  1605. }
  1606. else
  1607. {
  1608. IWMPSkinMngr * pWMPSkinMngr;
  1609. // Ignore failures until we are guarenteed they are in setup.
  1610. if (SUCCEEDED(CoCreateInstance(CLSID_WMPSkinMngr, NULL, CLSCTX_INPROC_SERVER, IID_PPV_ARG(IWMPSkinMngr, &pWMPSkinMngr))))
  1611. {
  1612. pWMPSkinMngr->SetVisualStyle(bstrPath);
  1613. pWMPSkinMngr->Release();
  1614. }
  1615. }
  1616. }
  1617. return hr;
  1618. }
  1619. HRESULT GetPageByCLSID(IUnknown * punkSite, const GUID * pClsid, IPropertyBag ** ppPropertyBag)
  1620. {
  1621. HRESULT hr = E_FAIL;
  1622. *ppPropertyBag = NULL;
  1623. if (punkSite)
  1624. {
  1625. IThemeUIPages * pThemeUI;
  1626. hr = punkSite->QueryInterface(IID_PPV_ARG(IThemeUIPages, &pThemeUI));
  1627. if (SUCCEEDED(hr))
  1628. {
  1629. IEnumUnknown * pEnumUnknown;
  1630. hr = pThemeUI->GetBasePagesEnum(&pEnumUnknown);
  1631. if (SUCCEEDED(hr))
  1632. {
  1633. IUnknown * punk;
  1634. // This may not exit due to policy
  1635. hr = IEnumUnknown_FindCLSID(pEnumUnknown, *pClsid, &punk);
  1636. if (SUCCEEDED(hr))
  1637. {
  1638. hr = punk->QueryInterface(IID_PPV_ARG(IPropertyBag, ppPropertyBag));
  1639. punk->Release();
  1640. }
  1641. pEnumUnknown->Release();
  1642. }
  1643. pThemeUI->Release();
  1644. }
  1645. }
  1646. return hr;
  1647. }
  1648. DWORD QueryThemeServicesWrap(void)
  1649. {
  1650. DWORD dwResult = QueryThemeServices();
  1651. if (IsTSPerfFlagEnabled(TSPerFlag_NoVisualStyles))
  1652. {
  1653. dwResult = (dwResult & ~QTS_AVAILABLE); // Remove the QTS_AVAILABLE flag because they are forced of because of TS Perf Flags
  1654. LogStatus("Visual Styles Forced off because of TS Perf Flags\r\n");
  1655. }
  1656. LogStatus("QueryThemeServices() returned %d. In QueryThemeServicesWrap\r\n", dwResult);
  1657. return dwResult;
  1658. }
  1659. void PathUnExpandEnvStringsWrap(LPTSTR pszString, DWORD cchSize)
  1660. {
  1661. TCHAR szTemp[MAX_PATH];
  1662. StrCpyN(szTemp, pszString, ARRAYSIZE(szTemp));
  1663. if (!PathUnExpandEnvStrings(szTemp, pszString, cchSize))
  1664. {
  1665. StrCpyN(pszString, szTemp, cchSize);
  1666. }
  1667. }
  1668. void PathExpandEnvStringsWrap(LPTSTR pszString, DWORD cchSize)
  1669. {
  1670. TCHAR szTemp[MAX_PATH];
  1671. StrCpyN(szTemp, pszString, ARRAYSIZE(szTemp));
  1672. if (0 == SHExpandEnvironmentStrings(szTemp, pszString, cchSize))
  1673. {
  1674. StrCpyN(pszString, szTemp, cchSize);
  1675. }
  1676. }
  1677. // PERF: This API is INCREADIBLY slow so be very very careful when you use it.
  1678. BOOL EnumDisplaySettingsExWrap(LPCTSTR lpszDeviceName, DWORD iModeNum, LPDEVMODE lpDevMode, DWORD dwFlags)
  1679. {
  1680. DEBUG_CODE(DebugStartWatch());
  1681. BOOL fReturn = EnumDisplaySettingsEx(lpszDeviceName, iModeNum, lpDevMode, dwFlags);
  1682. DEBUG_CODE(TraceMsg(TF_THEMEUI_PERF, "EnumDisplaySettingsEx() took Time=%lums", DebugStopWatch()));
  1683. return fReturn;
  1684. }
  1685. #define DEFAULT_DPI 96.0f
  1686. void DPIScaleRect(RECT * pRect)
  1687. {
  1688. HDC hdcScreen = GetDC(NULL);
  1689. if (hdcScreen)
  1690. {
  1691. double dScaleX = (GetDeviceCaps(hdcScreen, LOGPIXELSX) / DEFAULT_DPI);
  1692. double dScaleY = (GetDeviceCaps(hdcScreen, LOGPIXELSY) / DEFAULT_DPI);
  1693. #define DPI_SCALEX(nSizeX) ((int) ((nSizeX) * dScaleX))
  1694. #define DPI_SCALEY(nSizeY) ((int) ((nSizeY) * dScaleY))
  1695. if ((DEFAULT_DPI != dScaleX) || (DEFAULT_DPI != dScaleY))
  1696. {
  1697. pRect->top = DPI_SCALEY(pRect->top);
  1698. pRect->bottom = DPI_SCALEY(pRect->bottom);
  1699. pRect->left = DPI_SCALEX(pRect->left);
  1700. pRect->right = DPI_SCALEX(pRect->right);
  1701. }
  1702. ReleaseDC(NULL, hdcScreen);
  1703. }
  1704. }
  1705. // We may want to move this to shlwapi
  1706. #define DEFAULT_DPI 96.0f
  1707. HBITMAP LoadBitmapAndDPIScale(HINSTANCE hInst, LPCTSTR pszBitmapName)
  1708. {
  1709. HBITMAP hBitmap = LoadBitmap(hInst, pszBitmapName);
  1710. HDC hdcScreen = GetDC(NULL);
  1711. if (hdcScreen)
  1712. {
  1713. double dScaleX = (GetDeviceCaps(hdcScreen, LOGPIXELSX) / DEFAULT_DPI);
  1714. double dScaleY = (GetDeviceCaps(hdcScreen, LOGPIXELSY) / DEFAULT_DPI);
  1715. #define DPI_SCALEX(nSizeX) ((int) ((nSizeX) * dScaleX))
  1716. #define DPI_SCALEY(nSizeY) ((int) ((nSizeY) * dScaleY))
  1717. if ((DEFAULT_DPI != dScaleX) || (DEFAULT_DPI != dScaleY))
  1718. {
  1719. // We need to scale the bitmap.
  1720. HDC hdcBitmapSrc = CreateCompatibleDC(hdcScreen);
  1721. if (hdcBitmapSrc)
  1722. {
  1723. HDC hdcBitmapDest = CreateCompatibleDC(hdcScreen);
  1724. SelectObject(hdcBitmapSrc, hBitmap); // Put the bitmap into the source DC
  1725. if (hdcBitmapDest)
  1726. {
  1727. BITMAP bitmapInfo;
  1728. if (GetObject(hBitmap, sizeof(bitmapInfo), &bitmapInfo))
  1729. {
  1730. SetStretchBltMode(hdcBitmapDest, HALFTONE);
  1731. if (StretchBlt(hdcBitmapDest, 0, 0, DPI_SCALEX(bitmapInfo.bmWidth), DPI_SCALEY(bitmapInfo.bmHeight), hdcBitmapSrc, 0, 0, bitmapInfo.bmWidth, bitmapInfo.bmHeight, SRCCOPY))
  1732. {
  1733. HBITMAP hBitmapScaled = (HBITMAP)SelectObject(hdcBitmapSrc, NULL); // Get the bitmap from the DC
  1734. if (hBitmapScaled)
  1735. {
  1736. DeleteObject(hBitmap);
  1737. hBitmap = hBitmapScaled;
  1738. }
  1739. }
  1740. }
  1741. DeleteDC(hdcBitmapDest);
  1742. }
  1743. DeleteDC(hdcBitmapSrc);
  1744. }
  1745. }
  1746. ReleaseDC(NULL, hdcScreen);
  1747. }
  1748. return hBitmap;
  1749. }