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.

1084 lines
27 KiB

  1. //---------------------------------------------------------------------------
  2. //
  3. // Copyright (c) Microsoft Corporation 1993-1994
  4. //
  5. // File: comm.c
  6. //
  7. // This files contains all common utility routines
  8. //
  9. // History:
  10. // 08-06-93 ScottH Transferred from twin code
  11. //
  12. //---------------------------------------------------------------------------
  13. ///////////////////////////////////////////////////// INCLUDES
  14. #include "brfprv.h" // common s
  15. #include "res.h"
  16. // Some of these are replacements for the C runtime routines.
  17. // This is so we don't have to link to the CRT libs.
  18. //
  19. /*----------------------------------------------------------
  20. Purpose: memset
  21. Swiped from the C 7.0 runtime sources.
  22. Returns:
  23. Cond:
  24. */
  25. CHAR * PUBLIC lmemset( // DO NO UNICODIZE
  26. CHAR * dst,
  27. CHAR val,
  28. UINT count)
  29. {
  30. CHAR * start = dst;
  31. while (count--)
  32. *dst++ = val;
  33. return(start);
  34. }
  35. /*----------------------------------------------------------
  36. Purpose: memmove
  37. Swiped from the C 7.0 runtime sources.
  38. Returns:
  39. Cond:
  40. */
  41. CHAR * PUBLIC lmemmove(
  42. CHAR * dst,
  43. CHAR * src,
  44. int count)
  45. {
  46. CHAR * ret = dst;
  47. if (dst <= src || dst >= (src + count)) {
  48. /*
  49. * Non-Overlapping Buffers
  50. * copy from lower addresses to higher addresses
  51. */
  52. while (count--)
  53. *dst++ = *src++;
  54. }
  55. else {
  56. /*
  57. * Overlapping Buffers
  58. * copy from higher addresses to lower addresses
  59. */
  60. dst += count - 1;
  61. src += count - 1;
  62. while (count--)
  63. *dst-- = *src--;
  64. }
  65. return(ret);
  66. }
  67. /*----------------------------------------------------------
  68. Purpose: My verion of atoi. Supports hexadecimal too.
  69. Returns: integer
  70. Cond: --
  71. */
  72. int PUBLIC AnsiToInt(
  73. LPCTSTR pszString)
  74. {
  75. int n;
  76. BOOL bNeg = FALSE;
  77. LPCTSTR psz;
  78. LPCTSTR pszAdj;
  79. // Skip leading whitespace
  80. //
  81. for (psz = pszString; *psz == TEXT(' ') || *psz == TEXT('\n') || *psz == TEXT('\t'); psz = CharNext(psz))
  82. ;
  83. // Determine possible explicit signage
  84. //
  85. if (*psz == TEXT('+') || *psz == TEXT('-'))
  86. {
  87. bNeg = (*psz == TEXT('+')) ? FALSE : TRUE;
  88. psz = CharNext(psz);
  89. }
  90. // Or is this hexadecimal?
  91. //
  92. pszAdj = CharNext(psz);
  93. if (*psz == TEXT('0') && (*pszAdj == TEXT('x') || *pszAdj == TEXT('X')))
  94. {
  95. bNeg = FALSE; // Never allow negative sign with hexadecimal numbers
  96. psz = CharNext(pszAdj);
  97. // Do the conversion
  98. //
  99. for (n = 0; ; psz = CharNext(psz))
  100. {
  101. if (*psz >= TEXT('0') && *psz <= TEXT('9'))
  102. n = 0x10 * n + *psz - TEXT('0');
  103. else
  104. {
  105. TCHAR ch = *psz;
  106. int n2;
  107. if (ch >= TEXT('a'))
  108. ch -= TEXT('a') - TEXT('A');
  109. n2 = ch - TEXT('A') + 0xA;
  110. if (n2 >= 0xA && n2 <= 0xF)
  111. n = 0x10 * n + n2;
  112. else
  113. break;
  114. }
  115. }
  116. }
  117. else
  118. {
  119. for (n = 0; *psz >= TEXT('0') && *psz <= TEXT('9'); psz = CharNext(psz))
  120. n = 10 * n + *psz - TEXT('0');
  121. }
  122. return bNeg ? -n : n;
  123. }
  124. /*----------------------------------------------------------
  125. Purpose: General front end to invoke dialog boxes
  126. Returns: result from EndDialog
  127. Cond: --
  128. */
  129. INT_PTR PUBLIC DoModal(
  130. HWND hwndParent, // owner of dialog
  131. DLGPROC lpfnDlgProc, // dialog proc
  132. UINT uID, // dialog template ID
  133. LPARAM lParam) // extra parm to pass to dialog (may be NULL)
  134. {
  135. INT_PTR nResult = -1;
  136. nResult = DialogBoxParam(g_hinst, MAKEINTRESOURCE(uID), hwndParent,
  137. lpfnDlgProc, lParam);
  138. return nResult;
  139. }
  140. /*----------------------------------------------------------
  141. Purpose: Sets the rectangle with the bounding extent of the given string.
  142. Returns: Rectangle
  143. Cond: --
  144. */
  145. void PUBLIC SetRectFromExtent(
  146. HDC hdc,
  147. LPRECT lprect,
  148. LPCTSTR lpcsz)
  149. {
  150. SIZE size;
  151. GetTextExtentPoint(hdc, lpcsz, lstrlen(lpcsz), &size);
  152. SetRect(lprect, 0, 0, size.cx, size.cy);
  153. }
  154. /*----------------------------------------------------------
  155. Purpose: Sees whether the entire string will fit in *prc.
  156. If not, compute the numbder of chars that will fit
  157. (including ellipses). Returns length of string in
  158. *pcchDraw.
  159. Taken from COMMCTRL.
  160. Returns: TRUE if the string needed ellipses
  161. Cond: --
  162. */
  163. BOOL PRIVATE NeedsEllipses(
  164. HDC hdc,
  165. LPCTSTR pszText,
  166. RECT * prc,
  167. int * pcchDraw,
  168. int cxEllipses)
  169. {
  170. int cchText;
  171. int cxRect;
  172. int ichMin, ichMax, ichMid;
  173. SIZE siz;
  174. cxRect = prc->right - prc->left;
  175. cchText = lstrlen(pszText);
  176. if (cchText == 0)
  177. {
  178. *pcchDraw = cchText;
  179. return FALSE;
  180. }
  181. GetTextExtentPoint(hdc, pszText, cchText, &siz);
  182. if (siz.cx <= cxRect)
  183. {
  184. *pcchDraw = cchText;
  185. return FALSE;
  186. }
  187. cxRect -= cxEllipses;
  188. // If no room for ellipses, always show first character.
  189. //
  190. ichMax = 1;
  191. if (cxRect > 0)
  192. {
  193. // Binary search to find character that will fit
  194. ichMin = 0;
  195. ichMax = cchText;
  196. while (ichMin < ichMax)
  197. {
  198. // Be sure to round up, to make sure we make progress in
  199. // the loop if ichMax == ichMin + 1.
  200. //
  201. ichMid = (ichMin + ichMax + 1) / 2;
  202. GetTextExtentPoint(hdc, &pszText[ichMin], ichMid - ichMin, &siz);
  203. if (siz.cx < cxRect)
  204. {
  205. ichMin = ichMid;
  206. cxRect -= siz.cx;
  207. }
  208. else if (siz.cx > cxRect)
  209. {
  210. ichMax = ichMid - 1;
  211. }
  212. else
  213. {
  214. // Exact match up up to ichMid: just exit.
  215. //
  216. ichMax = ichMid;
  217. break;
  218. }
  219. }
  220. // Make sure we always show at least the first character...
  221. //
  222. if (ichMax < 1)
  223. ichMax = 1;
  224. }
  225. *pcchDraw = ichMax;
  226. return TRUE;
  227. }
  228. #define CCHELLIPSES 3
  229. #define DT_LVWRAP (DT_CENTER | DT_WORDBREAK | DT_NOPREFIX | DT_EDITCONTROL)
  230. /*----------------------------------------------------------
  231. Purpose: Draws text the shell's way.
  232. Taken from COMMCTRL.
  233. Returns: --
  234. Cond: This function requires TRANSPARENT background mode
  235. and a properly selected font.
  236. */
  237. void PUBLIC MyDrawText(
  238. HDC hdc,
  239. LPCTSTR pszText,
  240. RECT * prc,
  241. UINT flags,
  242. int cyChar,
  243. int cxEllipses,
  244. COLORREF clrText,
  245. COLORREF clrTextBk)
  246. {
  247. int cchText;
  248. COLORREF clrSave;
  249. COLORREF clrSaveBk;
  250. UINT uETOFlags = 0;
  251. RECT rc;
  252. TCHAR ach[MAX_PATH + CCHELLIPSES];
  253. // REVIEW: Performance idea:
  254. // We could cache the currently selected text color
  255. // so we don't have to set and restore it each time
  256. // when the color is the same.
  257. //
  258. if (!pszText)
  259. return;
  260. rc = *prc;
  261. // If needed, add in a little extra margin...
  262. //
  263. if (IsFlagSet(flags, MDT_EXTRAMARGIN))
  264. {
  265. rc.left += g_cxLabelMargin * 3;
  266. rc.right -= g_cxLabelMargin * 3;
  267. }
  268. else
  269. {
  270. rc.left += g_cxLabelMargin;
  271. rc.right -= g_cxLabelMargin;
  272. }
  273. if (IsFlagSet(flags, MDT_ELLIPSES) &&
  274. NeedsEllipses(hdc, pszText, &rc, &cchText, cxEllipses))
  275. {
  276. hmemcpy(ach, pszText, cchText * sizeof(TCHAR));
  277. lstrcpy(ach + cchText, c_szEllipses);
  278. pszText = ach;
  279. // Left-justify, in case there's no room for all of ellipses
  280. //
  281. ClearFlag(flags, (MDT_RIGHT | MDT_CENTER));
  282. SetFlag(flags, MDT_LEFT);
  283. cchText += CCHELLIPSES;
  284. }
  285. else
  286. {
  287. cchText = lstrlen(pszText);
  288. }
  289. if (IsFlagSet(flags, MDT_TRANSPARENT))
  290. {
  291. clrSave = SetTextColor(hdc, 0x000000);
  292. }
  293. else
  294. {
  295. uETOFlags |= ETO_OPAQUE;
  296. if (IsFlagSet(flags, MDT_SELECTED))
  297. {
  298. clrSave = SetTextColor(hdc, g_clrHighlightText);
  299. clrSaveBk = SetBkColor(hdc, g_clrHighlight);
  300. if (IsFlagSet(flags, MDT_DRAWTEXT))
  301. {
  302. FillRect(hdc, prc, g_hbrHighlight);
  303. }
  304. }
  305. else
  306. {
  307. if (clrText == CLR_DEFAULT && clrTextBk == CLR_DEFAULT)
  308. {
  309. clrSave = SetTextColor(hdc, g_clrWindowText);
  310. clrSaveBk = SetBkColor(hdc, g_clrWindow);
  311. if (IsFlagSet(flags, MDT_DRAWTEXT | MDT_DESELECTED))
  312. {
  313. FillRect(hdc, prc, g_hbrWindow);
  314. }
  315. }
  316. else
  317. {
  318. HBRUSH hbr;
  319. if (clrText == CLR_DEFAULT)
  320. clrText = g_clrWindowText;
  321. if (clrTextBk == CLR_DEFAULT)
  322. clrTextBk = g_clrWindow;
  323. clrSave = SetTextColor(hdc, clrText);
  324. clrSaveBk = SetBkColor(hdc, clrTextBk);
  325. if (IsFlagSet(flags, MDT_DRAWTEXT | MDT_DESELECTED))
  326. {
  327. hbr = CreateSolidBrush(GetNearestColor(hdc, clrTextBk));
  328. if (hbr)
  329. {
  330. FillRect(hdc, prc, hbr);
  331. DeleteObject(hbr);
  332. }
  333. else
  334. FillRect(hdc, prc, GetStockObject(WHITE_BRUSH));
  335. }
  336. }
  337. }
  338. }
  339. // If we want the item to display as if it was depressed, we will
  340. // offset the text rectangle down and to the left
  341. if (IsFlagSet(flags, MDT_DEPRESSED))
  342. OffsetRect(&rc, g_cxBorder, g_cyBorder);
  343. if (IsFlagSet(flags, MDT_DRAWTEXT))
  344. {
  345. UINT uDTFlags = DT_LVWRAP;
  346. if (IsFlagClear(flags, MDT_CLIPPED))
  347. uDTFlags |= DT_NOCLIP;
  348. DrawText(hdc, pszText, cchText, &rc, uDTFlags);
  349. }
  350. else
  351. {
  352. if (IsFlagClear(flags, MDT_LEFT))
  353. {
  354. SIZE siz;
  355. GetTextExtentPoint(hdc, pszText, cchText, &siz);
  356. if (IsFlagSet(flags, MDT_CENTER))
  357. rc.left = (rc.left + rc.right - siz.cx) / 2;
  358. else
  359. {
  360. ASSERT(IsFlagSet(flags, MDT_RIGHT));
  361. rc.left = rc.right - siz.cx;
  362. }
  363. }
  364. if (IsFlagSet(flags, MDT_VCENTER))
  365. {
  366. // Center vertically
  367. rc.top += (rc.bottom - rc.top - cyChar) / 2;
  368. }
  369. if (IsFlagSet(flags, MDT_CLIPPED))
  370. uETOFlags |= ETO_CLIPPED;
  371. ExtTextOut(hdc, rc.left, rc.top, uETOFlags, prc, pszText, cchText, NULL);
  372. }
  373. if (flags & (MDT_SELECTED | MDT_DESELECTED | MDT_TRANSPARENT))
  374. {
  375. SetTextColor(hdc, clrSave);
  376. if (IsFlagClear(flags, MDT_TRANSPARENT))
  377. SetBkColor(hdc, clrSaveBk);
  378. }
  379. }
  380. /*----------------------------------------------------------
  381. Purpose: Takes a DWORD value and converts it to a string, adding
  382. commas on the way.
  383. This was taken from the shell.
  384. Returns: Pointer to buffer
  385. Cond: --
  386. */
  387. // REARCHITECT The shell has an AddCommas. Can it be used instead?
  388. LPTSTR PRIVATE BrfAddCommas(
  389. DWORD dw,
  390. LPTSTR pszBuffer,
  391. UINT cbBuffer)
  392. {
  393. TCHAR szTemp[30];
  394. TCHAR szSep[5];
  395. NUMBERFMT nfmt;
  396. nfmt.NumDigits=0;
  397. nfmt.LeadingZero=0;
  398. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_SGROUPING, szSep, ARRAYSIZE(szSep));
  399. nfmt.Grouping = StrToInt(szSep);
  400. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, szSep, ARRAYSIZE(szSep));
  401. nfmt.lpDecimalSep = nfmt.lpThousandSep = szSep;
  402. nfmt.NegativeOrder= 0;
  403. wsprintf(szTemp, TEXT("%lu"), dw);
  404. GetNumberFormat(LOCALE_USER_DEFAULT, 0, szTemp, &nfmt, pszBuffer, cbBuffer);
  405. return pszBuffer;
  406. }
  407. const short s_rgidsOrders[] = {IDS_BYTES, IDS_ORDERKB, IDS_ORDERMB, IDS_ORDERGB, IDS_ORDERTB};
  408. // REARCHITECT This is in the shell too, isn't it?
  409. /*----------------------------------------------------------
  410. Purpose: Converts a number into a short, string format.
  411. This code was taken from the shell.
  412. 532 -> 523 bytes
  413. 1340 -> 1.3KB
  414. 23506 -> 23.5KB
  415. -> 2.4MB
  416. -> 5.2GB
  417. Returns: pointer to buffer
  418. Cond: --
  419. */
  420. LPTSTR PRIVATE ShortSizeFormat64(
  421. __int64 dw64,
  422. LPTSTR szBuf)
  423. {
  424. int i;
  425. UINT wInt, wLen, wDec;
  426. TCHAR szTemp[10], szOrder[20], szFormat[5];
  427. if (dw64 < 1000)
  428. {
  429. wsprintf(szTemp, TEXT("%d"), LODWORD(dw64));
  430. i = 0;
  431. goto AddOrder;
  432. }
  433. for (i = 1; i<ARRAYSIZE(s_rgidsOrders)-1 && dw64 >= 1000L * 1024L; dw64 >>= 10, i++);
  434. /* do nothing */
  435. wInt = LODWORD(dw64 >> 10);
  436. BrfAddCommas(wInt, szTemp, ARRAYSIZE(szTemp));
  437. wLen = lstrlen(szTemp);
  438. if (wLen < 3)
  439. {
  440. wDec = LODWORD(dw64 - (__int64)wInt * 1024L) * 1000 / 1024;
  441. // At this point, wDec should be between 0 and 1000
  442. // we want get the top one (or two) digits.
  443. wDec /= 10;
  444. if (wLen == 2)
  445. wDec /= 10;
  446. // Note that we need to set the format before getting the
  447. // intl char.
  448. lstrcpy(szFormat, TEXT("%02d"));
  449. szFormat[2] = TEXT('0') + 3 - wLen;
  450. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_SDECIMAL,
  451. szTemp+wLen, ARRAYSIZE(szTemp)-wLen);
  452. wLen = lstrlen(szTemp);
  453. wLen += wsprintf(szTemp+wLen, szFormat, wDec);
  454. }
  455. AddOrder:
  456. LoadString(g_hinst, s_rgidsOrders[i], szOrder, ARRAYSIZE(szOrder));
  457. wsprintf(szBuf, szOrder, (LPTSTR)szTemp);
  458. return szBuf;
  459. }
  460. /*----------------------------------------------------------
  461. Purpose: Converts a number into a short, string format.
  462. This code was taken from the shell.
  463. 532 -> 523 bytes
  464. 1340 -> 1.3KB
  465. 23506 -> 23.5KB
  466. -> 2.4MB
  467. -> 5.2GB
  468. Returns: pointer to buffer
  469. Cond: --
  470. */
  471. LPTSTR PRIVATE ShortSizeFormat(DWORD dw, LPTSTR szBuf)
  472. {
  473. return(ShortSizeFormat64((__int64)dw, szBuf));
  474. }
  475. /*----------------------------------------------------------
  476. Purpose: Gets the file info given a path. If the path refers
  477. to a directory, then simply the path field is filled.
  478. If himl != NULL, then the function will add the file's
  479. image to the provided image list and set the image index
  480. field in the *ppfi.
  481. Returns: standard hresult
  482. Cond: --
  483. */
  484. HRESULT PUBLIC FICreate(
  485. LPCTSTR pszPath,
  486. FileInfo ** ppfi,
  487. UINT uFlags)
  488. {
  489. HRESULT hres = ResultFromScode(E_OUTOFMEMORY);
  490. int cchPath;
  491. SHFILEINFO sfi;
  492. UINT uInfoFlags = SHGFI_DISPLAYNAME | SHGFI_ATTRIBUTES;
  493. DWORD dwAttr;
  494. ASSERT(pszPath);
  495. ASSERT(ppfi);
  496. // Get shell file info
  497. if (IsFlagSet(uFlags, FIF_ICON))
  498. uInfoFlags |= SHGFI_ICON;
  499. if (IsFlagSet(uFlags, FIF_DONTTOUCH))
  500. {
  501. uInfoFlags |= SHGFI_USEFILEATTRIBUTES;
  502. // Today, FICreate is not called for folders, so this is ifdef'd out
  503. #ifdef SUPPORT_FOLDERS
  504. dwAttr = IsFlagSet(uFlags, FIF_FOLDER) ? FILE_ATTRIBUTE_DIRECTORY : 0;
  505. #else
  506. dwAttr = 0;
  507. #endif
  508. }
  509. else
  510. dwAttr = 0;
  511. if (SHGetFileInfo(pszPath, dwAttr, &sfi, sizeof(sfi), uInfoFlags))
  512. {
  513. // Allocate enough for the structure, plus buffer for the fully qualified
  514. // path and buffer for the display name (and extra null terminator).
  515. cchPath = lstrlen(pszPath);
  516. *ppfi = GAlloc(sizeof(FileInfo) +
  517. (cchPath+1) * sizeof(TCHAR) -
  518. sizeof((*ppfi)->szPath) +
  519. (lstrlen(sfi.szDisplayName)+1) * sizeof(TCHAR));
  520. if (*ppfi)
  521. {
  522. FileInfo * pfi = *ppfi;
  523. pfi->pszDisplayName = pfi->szPath+cchPath+1;
  524. lstrcpy(pfi->pszDisplayName, sfi.szDisplayName);
  525. if (IsFlagSet(uFlags, FIF_ICON))
  526. pfi->hicon = sfi.hIcon;
  527. pfi->dwAttributes = sfi.dwAttributes;
  528. // Does the path refer to a directory?
  529. if (FIIsFolder(pfi))
  530. {
  531. // Yes; just fill in the path field
  532. lstrcpy(pfi->szPath, pszPath);
  533. hres = NOERROR;
  534. }
  535. else
  536. {
  537. // No; assume the file exists?
  538. if (IsFlagClear(uFlags, FIF_DONTTOUCH))
  539. {
  540. // Yes; get the time, date and size of the file
  541. HANDLE hfile = CreateFile(pszPath, GENERIC_READ,
  542. FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL,
  543. NULL);
  544. if (hfile == INVALID_HANDLE_VALUE)
  545. {
  546. GFree(*ppfi);
  547. hres = ResultFromScode(E_HANDLE);
  548. }
  549. else
  550. {
  551. hres = NOERROR;
  552. lstrcpy(pfi->szPath, pszPath);
  553. pfi->dwSize = GetFileSize(hfile, NULL);
  554. GetFileTime(hfile, NULL, NULL, &pfi->ftMod);
  555. CloseHandle(hfile);
  556. }
  557. }
  558. else
  559. {
  560. // No; use what we have
  561. hres = NOERROR;
  562. lstrcpy(pfi->szPath, pszPath);
  563. }
  564. }
  565. }
  566. }
  567. else if (!PathExists(pszPath))
  568. {
  569. // Differentiate between out of memory and file not found
  570. hres = E_FAIL;
  571. }
  572. return hres;
  573. }
  574. /*----------------------------------------------------------
  575. Purpose: Get some file info of the given path.
  576. The returned string is of the format "# bytes <date>"
  577. If the path is a folder, the string is empty.
  578. Returns: FALSE if path is not found
  579. Cond: --
  580. */
  581. BOOL PUBLIC FIGetInfoString(
  582. FileInfo * pfi,
  583. LPTSTR pszBuf,
  584. int cchBuf)
  585. {
  586. BOOL bRet;
  587. ASSERT(pfi);
  588. ASSERT(pszBuf);
  589. *pszBuf = NULL_CHAR;
  590. if (pfi)
  591. {
  592. // Is this a file?
  593. if ( !FIIsFolder(pfi) )
  594. {
  595. // Yes
  596. TCHAR szSize[MAXMEDLEN];
  597. TCHAR szDate[MAXMEDLEN];
  598. TCHAR szTime[MAXMEDLEN];
  599. LPTSTR pszMsg;
  600. SYSTEMTIME st;
  601. FILETIME ftLocal;
  602. // Construct the string
  603. FileTimeToLocalFileTime(&pfi->ftMod, &ftLocal);
  604. FileTimeToSystemTime(&ftLocal, &st);
  605. GetDateFormat(LOCALE_USER_DEFAULT, DATE_LONGDATE, &st, NULL, szDate, ARRAYSIZE(szDate));
  606. GetTimeFormat(LOCALE_USER_DEFAULT, 0, &st, NULL, szTime, ARRAYSIZE(szTime));
  607. if (ConstructMessage(&pszMsg, g_hinst, MAKEINTRESOURCE(IDS_DATESIZELINE),
  608. ShortSizeFormat(FIGetSize(pfi), szSize), szDate, szTime))
  609. {
  610. lstrcpy(pszBuf, pszMsg);
  611. GFree(pszMsg);
  612. }
  613. else
  614. *pszBuf = 0;
  615. bRet = TRUE;
  616. }
  617. else
  618. bRet = FALSE;
  619. }
  620. else
  621. bRet = FALSE;
  622. return bRet;
  623. }
  624. /*----------------------------------------------------------
  625. Purpose: Set the path entry. This can move the pfi.
  626. Returns: FALSE on out of memory
  627. Cond: --
  628. */
  629. BOOL PUBLIC FISetPath(
  630. FileInfo ** ppfi,
  631. LPCTSTR pszPathNew,
  632. UINT uFlags)
  633. {
  634. ASSERT(ppfi);
  635. ASSERT(pszPathNew);
  636. FIFree(*ppfi);
  637. return SUCCEEDED(FICreate(pszPathNew, ppfi, uFlags));
  638. }
  639. /*----------------------------------------------------------
  640. Purpose: Free our file info struct
  641. Returns: --
  642. Cond: --
  643. */
  644. void PUBLIC FIFree(
  645. FileInfo * pfi)
  646. {
  647. if (pfi)
  648. {
  649. if (pfi->hicon)
  650. DestroyIcon(pfi->hicon);
  651. GFree(pfi); // This macro already checks for NULL pfi condition
  652. }
  653. }
  654. /*----------------------------------------------------------
  655. Purpose: Convert FILETIME struct to a readable string
  656. Returns: String
  657. Cond: --
  658. */
  659. void PUBLIC FileTimeToDateTimeString(
  660. LPFILETIME pft,
  661. LPTSTR pszBuf,
  662. int cchBuf)
  663. {
  664. SYSTEMTIME st;
  665. FILETIME ftLocal;
  666. FileTimeToLocalFileTime(pft, &ftLocal);
  667. FileTimeToSystemTime(&ftLocal, &st);
  668. // REARCHITECT: how do you know date comes before time???
  669. GetDateFormat(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &st, NULL, pszBuf, cchBuf/2);
  670. pszBuf += lstrlen(pszBuf);
  671. *pszBuf++ = TEXT(' ');
  672. GetTimeFormat(LOCALE_USER_DEFAULT, TIME_NOSECONDS, &st, NULL, pszBuf, cchBuf/2);
  673. }
  674. /*----------------------------------------------------------
  675. Purpose: Copies psz into *ppszBuf. Will alloc or realloc *ppszBuf
  676. accordingly.
  677. Returns: TRUE on success
  678. Cond: --
  679. */
  680. BOOL PUBLIC GSetString(
  681. LPTSTR * ppszBuf,
  682. LPCTSTR psz)
  683. {
  684. BOOL bRet = FALSE;
  685. DWORD cb;
  686. ASSERT(ppszBuf);
  687. ASSERT(psz);
  688. cb = CbFromCch(lstrlen(psz)+CCH_NUL);
  689. if (*ppszBuf)
  690. {
  691. // Need to reallocate?
  692. if (cb > GGetSize(*ppszBuf))
  693. {
  694. // Yes
  695. LPTSTR pszT = GReAlloc(*ppszBuf, cb);
  696. if (pszT)
  697. {
  698. *ppszBuf = pszT;
  699. bRet = TRUE;
  700. }
  701. }
  702. else
  703. {
  704. // No
  705. bRet = TRUE;
  706. }
  707. }
  708. else
  709. {
  710. *ppszBuf = (LPTSTR)GAlloc(cb);
  711. if (*ppszBuf)
  712. {
  713. bRet = TRUE;
  714. }
  715. }
  716. if (bRet)
  717. {
  718. ASSERT(*ppszBuf);
  719. lstrcpy(*ppszBuf, psz);
  720. }
  721. return bRet;
  722. }
  723. /*----------------------------------------------------------
  724. Purpose: Concatenates psz onto *ppszBuf. Will alloc or realloc *ppszBuf
  725. accordingly.
  726. Returns: TRUE on success
  727. Cond: --
  728. */
  729. BOOL PUBLIC GCatString(
  730. LPTSTR * ppszBuf,
  731. LPCTSTR psz)
  732. {
  733. BOOL bRet = FALSE;
  734. DWORD cb;
  735. ASSERT(ppszBuf);
  736. ASSERT(psz);
  737. cb = CbFromCch(lstrlen(psz)+CCH_NUL);
  738. if (*ppszBuf)
  739. {
  740. // (Don't need to count nul because it is already counted in cb)
  741. DWORD cbExisting = CbFromCch(lstrlen(*ppszBuf));
  742. // Need to reallocate?
  743. if ((cb+cbExisting) > GGetSize(*ppszBuf))
  744. {
  745. // Yes; realloc at least MAXBUFLEN to cut down on the amount
  746. // of calls in the future
  747. LPTSTR pszT = GReAlloc(*ppszBuf, cbExisting+max(cb, MAXBUFLEN));
  748. if (pszT)
  749. {
  750. *ppszBuf = pszT;
  751. bRet = TRUE;
  752. }
  753. }
  754. else
  755. {
  756. // No
  757. bRet = TRUE;
  758. }
  759. }
  760. else
  761. {
  762. *ppszBuf = (LPTSTR)GAlloc(max(cb, MAXBUFLEN));
  763. if (*ppszBuf)
  764. {
  765. bRet = TRUE;
  766. }
  767. }
  768. if (bRet)
  769. {
  770. ASSERT(*ppszBuf);
  771. lstrcat(*ppszBuf, psz);
  772. }
  773. return bRet;
  774. }
  775. /*----------------------------------------------------------
  776. Purpose: Waits for on object to signal. This function "does
  777. the right thing" to prevent deadlocks which can occur
  778. because the calculation thread calls SendMessage.
  779. Returns: value of MsgWaitForMultipleObjects
  780. Cond: --
  781. */
  782. DWORD PUBLIC MsgWaitObjectsSendMessage(
  783. DWORD cObjects,
  784. LPHANDLE phObjects,
  785. DWORD dwTimeout)
  786. {
  787. DWORD dwRet;
  788. while (TRUE)
  789. {
  790. dwRet = MsgWaitForMultipleObjects(cObjects, phObjects, FALSE,
  791. dwTimeout, QS_SENDMESSAGE);
  792. // If it is not a message, return
  793. if ((WAIT_OBJECT_0 + cObjects) != dwRet)
  794. {
  795. return dwRet;
  796. }
  797. else
  798. {
  799. // Process all the sent messages
  800. MSG msg;
  801. PeekMessage(&msg, NULL, 0, 0, PM_NOREMOVE);
  802. }
  803. }
  804. }
  805. /*----------------------------------------------------------
  806. Purpose: Call this if PeekMessage is going to be called during
  807. an expensive operation and a new window has (or is)
  808. appeared.
  809. Details: simply calling SetCursor to change the cursor
  810. to an hourglass, then calling an expensive operation
  811. which will call PeekMessage, will result in the cursor
  812. changing back prematurely. The reason is because SetCursorPos
  813. inserts a fake WM_MOUSEMOVE to set the cursor to the
  814. window class when a window appears for the first time.
  815. Since PeekMessage is processing this message, the cursor
  816. gets changed to the window class cursor.
  817. The trick is to remove the WM_MOUSEMOVE messages from
  818. the queue.
  819. Returns: Previous cursor
  820. Cond: --
  821. */
  822. HCURSOR PUBLIC SetCursorRemoveWigglies(
  823. HCURSOR hcur)
  824. {
  825. MSG msg;
  826. // Remove any mouse moves
  827. while (PeekMessage(&msg, NULL, WM_MOUSEMOVE, WM_MOUSEMOVE, PM_REMOVE))
  828. ;
  829. return SetCursor(hcur);
  830. }
  831. /*----------------------------------------------------------
  832. Purpose: Load the string (if necessary) and format the string
  833. properly.
  834. Returns: A pointer to the allocated string containing the formatted
  835. message or
  836. NULL if out of memory
  837. Cond: --
  838. */
  839. LPTSTR PUBLIC _ConstructMessageString(
  840. HINSTANCE hinst,
  841. LPCTSTR pszMsg,
  842. va_list *ArgList)
  843. {
  844. TCHAR szTemp[MAXBUFLEN];
  845. LPTSTR pszRet;
  846. LPTSTR pszRes;
  847. if (HIWORD(pszMsg))
  848. pszRes = (LPTSTR)pszMsg;
  849. else if (LOWORD(pszMsg) && LoadString(hinst, LOWORD(pszMsg), szTemp, ARRAYSIZE(szTemp)))
  850. pszRes = szTemp;
  851. else
  852. pszRes = NULL;
  853. if (pszRes)
  854. {
  855. if (!FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_STRING,
  856. pszRes, 0, 0, (LPTSTR)&pszRet, 0, ArgList))
  857. {
  858. pszRet = NULL;
  859. }
  860. }
  861. else
  862. {
  863. // Bad parameter
  864. pszRet = NULL;
  865. }
  866. return pszRet; // free with LocalFree()
  867. }
  868. /*----------------------------------------------------------
  869. Purpose: Constructs a formatted string. The returned string
  870. must be freed using GFree().
  871. Returns: TRUE on success
  872. Cond: --
  873. */
  874. BOOL PUBLIC ConstructMessage(
  875. LPTSTR * ppsz,
  876. HINSTANCE hinst,
  877. LPCTSTR pszMsg, ...)
  878. {
  879. BOOL bRet;
  880. LPTSTR pszRet;
  881. va_list ArgList;
  882. va_start(ArgList, pszMsg);
  883. pszRet = _ConstructMessageString(hinst, pszMsg, &ArgList);
  884. va_end(ArgList);
  885. *ppsz = NULL;
  886. if (pszRet)
  887. {
  888. bRet = GSetString(ppsz, pszRet);
  889. LocalFree(pszRet);
  890. }
  891. else
  892. bRet = FALSE;
  893. return bRet;
  894. }