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.

1025 lines
22 KiB

  1. /*++
  2. Copyright (c) 1990-1995 Microsoft Corporation
  3. Module Name:
  4. resource.c
  5. Abstract:
  6. This module contains functions to load resources
  7. Author:
  8. 29-Aug-1995 Tue 12:29:27 created -by- Daniel Chou (danielc)
  9. [Environment:]
  10. NT Windows - Common Printer Driver UI DLL.
  11. [Notes:]
  12. Revision History:
  13. --*/
  14. #include "precomp.h"
  15. #pragma hdrstop
  16. #define DBG_CPSUIFILENAME DbgResource
  17. #define DBG_GETSTR0 0x00000001
  18. #define DBG_GETSTR1 0x00000002
  19. #define DBG_GETICON 0x00000004
  20. #define DBG_COMPOSESTR 0x00000008
  21. #define DBG_ADD_SPACE 0x00000010
  22. #define DBG_ADD_WCHAR 0x00000020
  23. #define DBG_AMPERCENT 0x00000040
  24. DEFINE_DBGVAR(0);
  25. extern HINSTANCE hInstDLL;
  26. //
  27. // Remove Ampercent will remove a '&' sign, if two '&' signs (ie. '&&') then
  28. // only one is removed, ALSO if a '(&X)' (ie. SPACE + & + Left Parenthesis +
  29. // Single Character + Right Parenthesis) then it consider to be a Localization
  30. // hot key indicator, then the whole ' (&X)' will be removed
  31. //
  32. #define REMOVE_AMPERCENT(CHTYPE) \
  33. { \
  34. CHTYPE *pOrg; \
  35. CHTYPE *pCopy; \
  36. UINT cRemoved; \
  37. CHTYPE ch; \
  38. \
  39. \
  40. cRemoved = 0; \
  41. pOrg = \
  42. pCopy = pStr; \
  43. \
  44. CPSUIDBG(DBG_AMPERCENT, ("RemoveAmpercent (ORG)='%ws'", pOrg)); \
  45. \
  46. do { \
  47. \
  48. while ((ch = *pStr++) && (ch != (CHTYPE)'&')) { \
  49. \
  50. if (cRemoved) { \
  51. \
  52. *pCopy = ch; \
  53. } \
  54. \
  55. ++pCopy; \
  56. } \
  57. \
  58. if (ch) { \
  59. \
  60. ++cRemoved; \
  61. \
  62. if (*pStr == (CHTYPE)'&') { \
  63. \
  64. *pCopy++ = *pStr++; \
  65. \
  66. } else if ((*(pCopy - 1) == (CHTYPE)'(') && \
  67. (*(pStr + 1) == (CHTYPE)')')) { \
  68. \
  69. cRemoved += 3; \
  70. ch = (CHTYPE)')'; \
  71. pCopy -= 1; \
  72. pStr += 2; \
  73. \
  74. if ((*pStr == (CHTYPE)' ') && \
  75. ((pCopy == pOrg) || \
  76. ((pCopy > pOrg) && \
  77. (*(pCopy - 1) == (CHTYPE)' ')))) { \
  78. \
  79. CPSUIDBG(DBG_AMPERCENT, ("Extra SPACE")); \
  80. \
  81. if (pCopy == pOrg) { \
  82. \
  83. ++pStr; \
  84. \
  85. } else { \
  86. \
  87. --pCopy; \
  88. } \
  89. \
  90. ++cRemoved; \
  91. } \
  92. } \
  93. } \
  94. \
  95. } while (ch); \
  96. \
  97. if (cRemoved) { \
  98. \
  99. *pCopy = (CHTYPE)'\0'; \
  100. \
  101. CPSUIDBG(DBG_AMPERCENT, (" RemoveAmpercent (%3ld)='%ws'", \
  102. cRemoved, pOrg)); \
  103. } \
  104. \
  105. return(cRemoved); \
  106. }
  107. UINT
  108. RemoveAmpersandA(
  109. LPSTR pStr
  110. )
  111. /*++
  112. Routine Description:
  113. This function remove ampersand from a string, the string must be writable
  114. Arguments:
  115. pStr - string to be serarch and remove the ampersand if found
  116. Return Value:
  117. UINT, count of ampersands removed
  118. Author:
  119. 19-Sep-1995 Tue 21:55:19 created -by- Daniel Chou (danielc)
  120. Revision History:
  121. --*/
  122. {
  123. REMOVE_AMPERCENT(CHAR);
  124. }
  125. UINT
  126. RemoveAmpersandW(
  127. LPWSTR pStr
  128. )
  129. /*++
  130. Routine Description:
  131. This function remove ampersand from a string, the string must be writable
  132. Arguments:
  133. pwStr - string to be serarch and remove the ampersand if found
  134. Return Value:
  135. UINT, count of ampersands removed
  136. Author:
  137. 19-Sep-1995 Tue 21:55:19 created -by- Daniel Chou (danielc)
  138. Revision History:
  139. --*/
  140. {
  141. REMOVE_AMPERCENT(WCHAR);
  142. }
  143. UINT
  144. DupAmpersandW(
  145. LPWSTR pwStr,
  146. INT cChar,
  147. INT cMaxChar
  148. )
  149. /*++
  150. Routine Description:
  151. This function remove ampersand from a string, the string must be writable
  152. Arguments:
  153. pwStr - string to be serarch and remove the ampersand if found
  154. Return Value:
  155. UINT, count of ampersands removed
  156. Author:
  157. 19-Sep-1995 Tue 21:55:19 created -by- Daniel Chou (danielc)
  158. Revision History:
  159. --*/
  160. {
  161. LPWSTR pw;
  162. LPWSTR pwCopy;
  163. INT i;
  164. INT cAdd = 0;
  165. if (((i = cChar) > 0) && ((cMaxChar -= cChar) > 0)) {
  166. pw = pwStr;
  167. while ((i--) && (cAdd < cMaxChar)) {
  168. if (*pw++ == L'&') {
  169. ++cAdd;
  170. }
  171. }
  172. if (cAdd) {
  173. pw = pwStr + cChar;
  174. pwCopy = pw + cAdd;
  175. CPSUIASSERT(0, "DupAmpersandW(): pwStr[%u] is not NULL",
  176. *pw == L'\0', IntToPtr(cChar));
  177. while (pwCopy >= pwStr) {
  178. if ((*pwCopy-- = *pw--) == L'&') {
  179. *pwCopy-- = L'&';
  180. }
  181. }
  182. }
  183. }
  184. return((UINT)cAdd);
  185. }
  186. UINT
  187. GetString(
  188. PGSBUF pGSBuf,
  189. LPTSTR pStr
  190. )
  191. /*++
  192. Routine Description:
  193. This function load the string either from caller (ANSI or UNICODE) or
  194. from common UI DLL, if the pStr is valid then it will just it, after
  195. getting the correct string, it will convert it to UNICODE as needed
  196. Arguments:
  197. pGSBuf - Pointer to GSBUF structure and following structure must set
  198. pTVWnd - Pointer to the TVWND which has all then information
  199. needed.
  200. pBuf - Pointer to the begining of the buffer (LPWSTR)
  201. pEndBuf - Pointer to the end of the buffer
  202. pStr - Pointer to the string to be converted
  203. Return Value:
  204. UINT - Count of character stored in the pBuf not include the NULL
  205. terminator, if the pBuf only has one character left then it
  206. always store a NULL and return 0
  207. Author:
  208. 29-Aug-1995 Tue 12:30:49 created -by- Daniel Chou (danielc)
  209. First version
  210. 31-Aug-1995 Thu 10:58:04 updated -by- Daniel Chou (danielc)
  211. Re-write to do UNICODE conversion and ANSI call checking
  212. 05-Feb-1996 Mon 12:20:28 updated -by- Daniel Chou (danielc)
  213. Fix the bug when UNICODE we do lstrcpy without checking the buffer
  214. size.
  215. Revision History:
  216. --*/
  217. {
  218. LPWSTR pBuf;
  219. WORD Flags;
  220. GSBUF GSBuf = *pGSBuf;
  221. UINT RemoveAmpersandOff = 0;
  222. INT cChar;
  223. INT Len = 0;
  224. //
  225. // Make pBuf pointed to the first available character
  226. //
  227. pBuf = pGSBuf->pBuf;
  228. Flags = pGSBuf->Flags;
  229. CPSUIASSERT(0, "GetString(pBuf=NULL)", pBuf, 0);
  230. if (pGSBuf->pEndBuf) {
  231. cChar = (INT)(pGSBuf->pEndBuf - pBuf);
  232. } else {
  233. cChar = MAX_RES_STR_CHARS;
  234. }
  235. //
  236. // Check if we have room to convert the string, make sure we reduced the
  237. // cChar by one for the NULL terminator
  238. //
  239. if ((pStr == NULL) || (cChar < 2)) {
  240. if (pStr) {
  241. CPSUIWARN(("GetString: pStr=%08lx, Buffer cChar=%ld too smaller",
  242. pStr, cChar));
  243. }
  244. *pBuf = L'\0';
  245. return(0);
  246. }
  247. if (pGSBuf->chPreAdd != L'\0') {
  248. CPSUIDBG(DBG_GETSTR0, ("GetString(): Pre-Add Char = '%wc'",
  249. pGSBuf->chPreAdd));
  250. //
  251. // If we pre-add character first then do it now
  252. //
  253. if ((*pBuf++ = pGSBuf->chPreAdd) == L'&') {
  254. RemoveAmpersandOff = 1;
  255. }
  256. cChar--;
  257. pGSBuf->chPreAdd = L'\0';
  258. }
  259. if (--cChar < 1) {
  260. CPSUIDBG(DBG_GETSTR1, ("GetString()=Only has one character for SPACE"));
  261. NULL;
  262. } else if (!VALID_PTR(pStr)) {
  263. HINSTANCE hInst;
  264. WORD ResID;
  265. //
  266. // Apperantly this is the resource ID, the LoadString() will not
  267. // write exceed the cChar included the NULL terminator according to
  268. // the Win32 help file. At here we know we either have to convert
  269. // the ASCII string to UNICODE or we load the UNICODE string to the
  270. // buffer already
  271. //
  272. ResID = LOWORD(LODWORD(pStr));
  273. CPSUIDBGBLK({
  274. if ((ResID >= IDI_CPSUI_ICONID_FIRST) &&
  275. (ResID <= IDI_CPSUI_ICONID_LAST)) {
  276. CPSUIERR(("ResID=%ld is in icon ID range, change it", ResID));
  277. ResID = ResID - IDI_CPSUI_ICONID_FIRST + IDS_CPSUI_STRID_FIRST;
  278. }
  279. })
  280. if ((ResID >= IDS_CPSUI_STRID_FIRST) &&
  281. (ResID <= IDS_CPSUI_STRID_LAST)) {
  282. hInst = hInstDLL;
  283. if (Flags & GBF_INT_NO_PREFIX) {
  284. Flags &= ~GBF_PREFIX_OK;
  285. }
  286. } else {
  287. hInst = (Flags & GBF_IDS_INT_CPSUI) ? hInstDLL : pGSBuf->hInst;
  288. CPSUIASSERT(0, "GetString(hInst=NULL, %08lx)", pStr, 0);
  289. }
  290. //
  291. // Now loaded from common UI DLL directly to the user buffer
  292. //
  293. if (Len = LoadString(hInst, ResID, pBuf, cChar)) {
  294. pBuf += Len;
  295. } else {
  296. pBuf = pGSBuf->pBuf;
  297. CPSUIERR(("LoadString(ID=%ld) FAILED", ResID));
  298. }
  299. } else if ((Flags & GBF_COPYWSTR) ||
  300. (!(Flags & GBF_ANSI_CALL))) {
  301. //
  302. // We have UNICODE string but may need to put into the buffer
  303. //
  304. if (Len = lstrlen(pStr)) {
  305. if (Len > cChar) {
  306. Len = cChar;
  307. }
  308. CopyMemory(pBuf, pStr, sizeof(WCHAR) * Len);
  309. pBuf += Len;
  310. }
  311. } else {
  312. //
  313. // We are loading the ANSI string
  314. //
  315. if (Len = lstrlenA((LPSTR)pStr)) {
  316. if (Len = MultiByteToWideChar(CP_ACP,
  317. 0,
  318. (LPCSTR)pStr,
  319. Len,
  320. pBuf,
  321. cChar)) {
  322. pBuf += Len;
  323. } else {
  324. //
  325. // Conversion is not complete so make sure it NULL terminated
  326. //
  327. pBuf = pGSBuf->pBuf;
  328. CPSUIWARN(("GetString: pstr='%hs', Buffer reach limit=%ld, Len=%ld",
  329. pStr, cChar, Len));
  330. }
  331. CPSUIDBG(DBG_GETSTR0, ("Convert to UNICODE, Len=%d, cChar=%d",
  332. Len, cChar));
  333. }
  334. }
  335. //
  336. // Save the new index back and return the len to the caller
  337. //
  338. *pBuf = L'\0';
  339. Len = (INT)(pBuf - pGSBuf->pBuf);
  340. if (!(Flags & GBF_PREFIX_OK)) {
  341. Len -= RemoveAmpersandW(pGSBuf->pBuf + RemoveAmpersandOff);
  342. }
  343. if (Flags & GBF_DUP_PREFIX) {
  344. Len += DupAmpersandW(pGSBuf->pBuf,
  345. Len,
  346. (INT)(UINT)(pGSBuf->pEndBuf - pGSBuf->pBuf));
  347. }
  348. CPSUIDBG(DBG_GETSTR1, ("GetString()=%ws (%d/%d)", pGSBuf->pBuf, Len, cChar));
  349. CPSUIASSERT(0, "GetString() : Len != Real Len (%ld)",
  350. Len == lstrlen(pGSBuf->pBuf), IntToPtr(lstrlen(pGSBuf->pBuf)));
  351. pGSBuf->pBuf += Len;
  352. return((UINT)Len);
  353. }
  354. UINT
  355. GSBufAddNumber(
  356. PGSBUF pGSBuf,
  357. DWORD Number,
  358. BOOL Sign
  359. )
  360. /*++
  361. Routine Description:
  362. Convert a number to a string with the limitation of GSBUF
  363. Arguments:
  364. pGSBuf - Pointer to GSBUF structure and following structure must set
  365. pTVWnd - Pointer to the TVWND which has all then information
  366. needed.
  367. pBuf - Pointer to the begining of the buffer (LPWSTR)
  368. pEndBuf - Pointer to the end of the buffer
  369. Number - LONG number to be converted
  370. Sign - if TRUE then Number is a sign long number else it is a unsigned
  371. DWORD
  372. Return Value:
  373. UINT total bytes converted to the string
  374. Author:
  375. 21-Feb-1996 Wed 12:17:00 created -by- Daniel Chou (danielc)
  376. Revision History:
  377. --*/
  378. {
  379. WCHAR wBuf[16];
  380. UINT cChar;
  381. UINT Len = 0;
  382. if (((cChar = (INT)(pGSBuf->pEndBuf - pGSBuf->pBuf - 1)) > 0) &&
  383. (Len = wsprintf(wBuf, (Sign) ? L"%ld" : L"%lu", Number))) {
  384. if (Len > cChar) {
  385. Len = cChar;
  386. }
  387. CopyMemory(pGSBuf->pBuf, wBuf, sizeof(WCHAR) * Len);
  388. pGSBuf->pBuf += Len;
  389. *(pGSBuf->pBuf) = L'\0';
  390. }
  391. return(Len);
  392. }
  393. UINT
  394. GSBufAddWChar(
  395. PGSBUF pGSBuf,
  396. UINT IntCharStrID,
  397. UINT Count
  398. )
  399. /*++
  400. Routine Description:
  401. Add a single character to the GSBuf
  402. Arguments:
  403. pGSBuf - Pointer to GSBUF structure and following structure must set
  404. pTVWnd - Pointer to the TVWND which has all then information
  405. needed.
  406. pBuf - Pointer to the begining of the buffer (LPWSTR)
  407. pEndBuf - Pointer to the end of the buffer
  408. wch - a single character to be added
  409. Return Value:
  410. BOOLEAN, true if succeed else false
  411. Author:
  412. 21-Feb-1996 Wed 12:00:24 created -by- Daniel Chou (danielc)
  413. Revision History:
  414. --*/
  415. {
  416. WCHAR wCh[2];
  417. UINT cAvai;
  418. if ((Count) &&
  419. ((cAvai = (UINT)(pGSBuf->pEndBuf - pGSBuf->pBuf)) > 1) &&
  420. (LoadString(hInstDLL, IntCharStrID, wCh, COUNT_ARRAY(wCh)))) {
  421. CPSUIDBG(DBG_ADD_WCHAR, ("GSBufAddWChar(%08lx, %u, %u)=%u of '%wc'",
  422. pGSBuf, IntCharStrID, Count,
  423. (Count > (cAvai - 1)) ? cAvai - 1 : Count, wCh[0]));
  424. if (Count > (cAvai -= 1)) {
  425. Count = cAvai;
  426. }
  427. cAvai = Count;
  428. while (cAvai--) {
  429. *(pGSBuf->pBuf)++ = wCh[0];
  430. }
  431. *(pGSBuf->pBuf) = L'\0';
  432. return(Count);
  433. } else {
  434. CPSUIERR(("GSBufAddWChar(%08lx, %u, %u) FAILED",
  435. pGSBuf, IntCharStrID, Count));
  436. return(0);
  437. }
  438. }
  439. UINT
  440. GSBufAddSpace(
  441. PGSBUF pGSBuf,
  442. UINT Count
  443. )
  444. /*++
  445. Routine Description:
  446. Arguments:
  447. Return Value:
  448. Author:
  449. 20-Jul-1996 Sat 00:59:47 created -by- Daniel Chou (danielc)
  450. Revision History:
  451. --*/
  452. {
  453. static WCHAR wSpace[2] = { 0, 0 };
  454. UINT cAvai;
  455. if (wSpace[0] == L'\0') {
  456. LoadString(hInstDLL,
  457. IDS_INT_CPSUI_SPACE_CHAR,
  458. wSpace,
  459. COUNT_ARRAY(wSpace));
  460. }
  461. if ((wSpace[0] != L'\0') &&
  462. (Count) &&
  463. ((cAvai = (UINT)(pGSBuf->pEndBuf - pGSBuf->pBuf)) > 1)) {
  464. CPSUIDBG(DBG_ADD_SPACE, ("GSBufAddSpace(%08lx, %u)=%u of '%wc'",
  465. pGSBuf, Count,
  466. (Count > (cAvai - 1)) ? cAvai - 1 : Count, wSpace[0]));
  467. if (Count > (cAvai -= 1)) {
  468. Count = cAvai;
  469. }
  470. cAvai = Count;
  471. while (cAvai--) {
  472. *(pGSBuf->pBuf)++ = wSpace[0];
  473. }
  474. *(pGSBuf->pBuf) = L'\0';
  475. return(Count);
  476. } else {
  477. CPSUIERR(("GSBufAddSpace(%08lx, %u) FAILED", pGSBuf, Count));
  478. return(0);
  479. }
  480. }
  481. UINT
  482. GetStringBuffer(
  483. HINSTANCE hInst,
  484. WORD GBFlags,
  485. WCHAR chPreAdd,
  486. LPTSTR pStr,
  487. LPWSTR pBuf,
  488. UINT cwBuf
  489. )
  490. /*++
  491. Routine Description:
  492. Arguments:
  493. Return Value:
  494. Author:
  495. 07-Sep-1995 Thu 10:45:09 created -by- Daniel Chou (danielc)
  496. Revision History:
  497. --*/
  498. {
  499. GSBUF GSBuf;
  500. GSBuf.hInst = hInst;
  501. GSBuf.Flags = GBFlags;
  502. GSBuf.pBuf = (LPWSTR)pBuf;
  503. GSBuf.pEndBuf = (LPWSTR)pBuf + cwBuf;
  504. GSBuf.chPreAdd = chPreAdd;
  505. return(GetString(&GSBuf, pStr));
  506. }
  507. LONG
  508. LoadCPSUIString(
  509. LPTSTR pStr,
  510. UINT cStr,
  511. UINT StrResID,
  512. BOOL AnsiCall
  513. )
  514. /*++
  515. Routine Description:
  516. Arguments:
  517. Return Value:
  518. Author:
  519. 08-Feb-1996 Thu 13:36:12 created -by- Daniel Chou (danielc)
  520. Revision History:
  521. --*/
  522. {
  523. if ((pStr) && (cStr)) {
  524. UINT Len = 0;
  525. if ((StrResID >= IDS_CPSUI_STRID_FIRST) &&
  526. (StrResID <= IDS_CPSUI_STRID_LAST)) {
  527. if (AnsiCall) {
  528. if (Len = LoadStringA(hInstDLL, StrResID, (LPSTR)pStr, cStr)) {
  529. Len -= RemoveAmpersandA((LPSTR)pStr);
  530. }
  531. } else {
  532. if (Len = LoadString(hInstDLL, StrResID, (LPWSTR)pStr, cStr)) {
  533. Len -= RemoveAmpersandW((LPWSTR)pStr);
  534. }
  535. }
  536. }
  537. return((LONG)Len);
  538. } else {
  539. return(-1);
  540. }
  541. }
  542. UINT
  543. ComposeStrData(
  544. HINSTANCE hInst,
  545. WORD GBFlags,
  546. LPWSTR pBuf,
  547. UINT cwBuf,
  548. UINT IntFormatStrID,
  549. LPTSTR pStr,
  550. DWORD dw1,
  551. DWORD dw2
  552. )
  553. /*++
  554. Routine Description:
  555. Arguments:
  556. Return Value:
  557. Author:
  558. 19-Jul-1996 Fri 17:11:19 created -by- Daniel Chou (danielc)
  559. Revision History:
  560. --*/
  561. {
  562. TCHAR szFormat[MAX_RES_STR_CHARS * 3];
  563. LPTSTR pData;
  564. LPTSTR pFinal;
  565. UINT Count;
  566. UINT i;
  567. UINT cb;
  568. ZeroMemory(szFormat, sizeof(szFormat));
  569. if ((IntFormatStrID) &&
  570. (i = LoadString(hInstDLL,
  571. IntFormatStrID,
  572. (LPTSTR)szFormat,
  573. MAX_RES_STR_CHARS))) {
  574. cb = ARRAYSIZE(szFormat) - i - 1;
  575. pData = szFormat + i + 1;
  576. if (!pStr) {
  577. //
  578. // Skip the pStr if it passed as NULL
  579. //
  580. Count = wnsprintf(pFinal = pData, cb, szFormat, dw1, dw2);
  581. } else {
  582. i = GetStringBuffer(hInst,
  583. (WORD)(GBFlags | GBF_INT_NO_PREFIX),
  584. (WCHAR)0,
  585. pStr,
  586. pData,
  587. MAX_RES_STR_CHARS);
  588. cb = cb - i - 1;
  589. pFinal = pData + i + 1;
  590. Count = wnsprintf(pFinal, cb, szFormat, pData, dw1, dw2);
  591. }
  592. if (Count > (cwBuf - 1)) {
  593. Count = cwBuf - 1;
  594. szFormat[Count] = '\0';
  595. }
  596. CopyMemory(pBuf, pFinal, (Count + 1) * sizeof(TCHAR));
  597. CPSUIDBG(DBG_COMPOSESTR, ("ComposeString('%ws', '%ws', %lu, %lu)='%ws' [%u]",
  598. szFormat, pData, dw1, dw2, pBuf, Count));
  599. } else {
  600. Count = GetStringBuffer(hInst, GBFlags, (WCHAR)0, pStr, pBuf, cwBuf);
  601. CPSUIDBG(DBG_COMPOSESTR, ("ComposeString(%08lx, %lu, %lu)=FAILED, '%ws' [%u]",
  602. pStr, dw1, dw2, pBuf, Count));
  603. }
  604. return(Count);
  605. }