1 /* 2 * Shlwapi string functions 3 * 4 * Copyright 1998 Juergen Schmied 5 * Copyright 2002 Jon Griffiths 6 * 7 * This library is free software; you can redistribute it and/or 8 * modify it under the terms of the GNU Lesser General Public 9 * License as published by the Free Software Foundation; either 10 * version 2.1 of the License, or (at your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, 13 * but WITHOUT ANY WARRANTY; without even the implied warranty of 14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15 * Lesser General Public License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public 18 * License along with this library; if not, write to the Free Software 19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA 20 */ 21 22 #include "config.h" 23 #include "wine/port.h" 24 25 #include <math.h> 26 #include <stdarg.h> 27 #include <stdio.h> 28 #include <string.h> 29 30 #define NONAMELESSUNION 31 32 #include "windef.h" 33 #include "winbase.h" 34 #define NO_SHLWAPI_REG 35 #define NO_SHLWAPI_STREAM 36 #include "shlwapi.h" 37 #include "wingdi.h" 38 #include "winuser.h" 39 #include "shlobj.h" 40 #include "mlang.h" 41 #include "ddeml.h" 42 #include "wine/unicode.h" 43 #include "wine/debug.h" 44 45 #include "resource.h" 46 47 WINE_DEFAULT_DEBUG_CHANNEL(shell); 48 49 extern HINSTANCE shlwapi_hInstance; 50 51 static HRESULT _SHStrDupAA(LPCSTR,LPSTR*); 52 static HRESULT _SHStrDupAW(LPCWSTR,LPSTR*); 53 54 55 static void FillNumberFmt(NUMBERFMTW *fmt, LPWSTR decimal_buffer, int decimal_bufwlen, 56 LPWSTR thousand_buffer, int thousand_bufwlen) 57 { 58 WCHAR grouping[64]; 59 WCHAR *c; 60 61 GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_ILZERO|LOCALE_RETURN_NUMBER, (LPWSTR)&fmt->LeadingZero, sizeof(fmt->LeadingZero)/sizeof(WCHAR)); 62 GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_INEGNUMBER|LOCALE_RETURN_NUMBER, (LPWSTR)&fmt->LeadingZero, sizeof(fmt->NegativeOrder)/sizeof(WCHAR)); 63 fmt->NumDigits = 0; 64 GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SDECIMAL, decimal_buffer, decimal_bufwlen); 65 GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, thousand_buffer, thousand_bufwlen); 66 fmt->lpThousandSep = thousand_buffer; 67 fmt->lpDecimalSep = decimal_buffer; 68 69 /* 70 * Converting grouping string to number as described on 71 * http://blogs.msdn.com/oldnewthing/archive/2006/04/18/578251.aspx 72 */ 73 fmt->Grouping = 0; 74 GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SGROUPING, grouping, sizeof(grouping)/sizeof(WCHAR)); 75 for (c = grouping; *c; c++) 76 if (*c >= '0' && *c < '9') 77 { 78 fmt->Grouping *= 10; 79 fmt->Grouping += *c - '0'; 80 } 81 82 if (fmt->Grouping % 10 == 0) 83 fmt->Grouping /= 10; 84 else 85 fmt->Grouping *= 10; 86 } 87 88 /************************************************************************* 89 * FormatInt [internal] 90 * 91 * Format an integer according to the current locale 92 * 93 * RETURNS 94 * The number of characters written on success or 0 on failure 95 */ 96 static int FormatInt(LONGLONG qdwValue, LPWSTR pszBuf, int cchBuf) 97 { 98 NUMBERFMTW fmt; 99 WCHAR decimal[8], thousand[8]; 100 WCHAR buf[24]; 101 WCHAR *c; 102 BOOL neg = (qdwValue < 0); 103 104 FillNumberFmt(&fmt, decimal, sizeof decimal / sizeof (WCHAR), 105 thousand, sizeof thousand / sizeof (WCHAR)); 106 107 c = &buf[24]; 108 *(--c) = 0; 109 do 110 { 111 *(--c) = '0' + (qdwValue%10); 112 qdwValue /= 10; 113 } while (qdwValue > 0); 114 if (neg) 115 *(--c) = '-'; 116 117 return GetNumberFormatW(LOCALE_USER_DEFAULT, 0, c, &fmt, pszBuf, cchBuf); 118 } 119 120 /************************************************************************* 121 * FormatDouble [internal] 122 * 123 * Format an integer according to the current locale. Prints the specified number of digits 124 * after the decimal point 125 * 126 * RETURNS 127 * The number of characters written on success or 0 on failure 128 */ 129 static int FormatDouble(double value, int decimals, LPWSTR pszBuf, int cchBuf) 130 { 131 static const WCHAR flfmt[] = {'%','f',0}; 132 WCHAR buf[64]; 133 NUMBERFMTW fmt; 134 WCHAR decimal[8], thousand[8]; 135 136 snprintfW(buf, 64, flfmt, value); 137 138 FillNumberFmt(&fmt, decimal, sizeof decimal / sizeof (WCHAR), 139 thousand, sizeof thousand / sizeof (WCHAR)); 140 fmt.NumDigits = decimals; 141 return GetNumberFormatW(LOCALE_USER_DEFAULT, 0, buf, &fmt, pszBuf, cchBuf); 142 } 143 144 /************************************************************************* 145 * SHLWAPI_ChrCmpHelperA 146 * 147 * Internal helper for SHLWAPI_ChrCmpA/ChrCMPIA. 148 * 149 * NOTES 150 * Both this function and its Unicode counterpart are very inefficient. To 151 * fix this, CompareString must be completely implemented and optimised 152 * first. Then the core character test can be taken out of that function and 153 * placed here, so that it need never be called at all. Until then, do not 154 * attempt to optimise this code unless you are willing to test that it 155 * still performs correctly. 156 */ 157 static BOOL SHLWAPI_ChrCmpHelperA(WORD ch1, WORD ch2, DWORD dwFlags) 158 { 159 char str1[3], str2[3]; 160 161 str1[0] = LOBYTE(ch1); 162 if (IsDBCSLeadByte(str1[0])) 163 { 164 str1[1] = HIBYTE(ch1); 165 str1[2] = '\0'; 166 } 167 else 168 str1[1] = '\0'; 169 170 str2[0] = LOBYTE(ch2); 171 if (IsDBCSLeadByte(str2[0])) 172 { 173 str2[1] = HIBYTE(ch2); 174 str2[2] = '\0'; 175 } 176 else 177 str2[1] = '\0'; 178 179 return CompareStringA(GetThreadLocale(), dwFlags, str1, -1, str2, -1) - CSTR_EQUAL; 180 } 181 182 /************************************************************************* 183 * SHLWAPI_ChrCmpA 184 * 185 * Internal helper function. 186 */ 187 static BOOL WINAPI SHLWAPI_ChrCmpA(WORD ch1, WORD ch2) 188 { 189 return SHLWAPI_ChrCmpHelperA(ch1, ch2, 0); 190 } 191 192 /************************************************************************* 193 * ChrCmpIA (SHLWAPI.385) 194 * 195 * Compare two characters, ignoring case. 196 * 197 * PARAMS 198 * ch1 [I] First character to compare 199 * ch2 [I] Second character to compare 200 * 201 * RETURNS 202 * FALSE, if the characters are equal. 203 * Non-zero otherwise. 204 */ 205 BOOL WINAPI ChrCmpIA(WORD ch1, WORD ch2) 206 { 207 TRACE("(%d,%d)\n", ch1, ch2); 208 209 return SHLWAPI_ChrCmpHelperA(ch1, ch2, NORM_IGNORECASE); 210 } 211 212 /************************************************************************* 213 * ChrCmpIW [SHLWAPI.386] 214 * 215 * See ChrCmpIA. 216 */ 217 BOOL WINAPI ChrCmpIW(WCHAR ch1, WCHAR ch2) 218 { 219 return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, &ch1, 1, &ch2, 1) - CSTR_EQUAL; 220 } 221 222 /************************************************************************* 223 * StrChrA [SHLWAPI.@] 224 * 225 * Find a given character in a string. 226 * 227 * PARAMS 228 * lpszStr [I] String to search in. 229 * ch [I] Character to search for. 230 * 231 * RETURNS 232 * Success: A pointer to the first occurrence of ch in lpszStr, or NULL if 233 * not found. 234 * Failure: NULL, if any arguments are invalid. 235 */ 236 LPSTR WINAPI StrChrA(LPCSTR lpszStr, WORD ch) 237 { 238 TRACE("(%s,%i)\n", debugstr_a(lpszStr), ch); 239 240 if (lpszStr) 241 { 242 while (*lpszStr) 243 { 244 if (!SHLWAPI_ChrCmpA(*lpszStr, ch)) 245 return (LPSTR)lpszStr; 246 lpszStr = CharNextA(lpszStr); 247 } 248 } 249 return NULL; 250 } 251 252 /************************************************************************* 253 * StrChrW [SHLWAPI.@] 254 * 255 * See StrChrA. 256 */ 257 LPWSTR WINAPI StrChrW(LPCWSTR lpszStr, WCHAR ch) 258 { 259 LPWSTR lpszRet = NULL; 260 261 TRACE("(%s,%i)\n", debugstr_w(lpszStr), ch); 262 263 if (lpszStr) 264 lpszRet = strchrW(lpszStr, ch); 265 return lpszRet; 266 } 267 268 /************************************************************************* 269 * StrChrIA [SHLWAPI.@] 270 * 271 * Find a given character in a string, ignoring case. 272 * 273 * PARAMS 274 * lpszStr [I] String to search in. 275 * ch [I] Character to search for. 276 * 277 * RETURNS 278 * Success: A pointer to the first occurrence of ch in lpszStr, or NULL if 279 * not found. 280 * Failure: NULL, if any arguments are invalid. 281 */ 282 LPSTR WINAPI StrChrIA(LPCSTR lpszStr, WORD ch) 283 { 284 TRACE("(%s,%i)\n", debugstr_a(lpszStr), ch); 285 286 if (lpszStr) 287 { 288 while (*lpszStr) 289 { 290 if (!ChrCmpIA(*lpszStr, ch)) 291 return (LPSTR)lpszStr; 292 lpszStr = CharNextA(lpszStr); 293 } 294 } 295 return NULL; 296 } 297 298 /************************************************************************* 299 * StrChrIW [SHLWAPI.@] 300 * 301 * See StrChrA. 302 */ 303 LPWSTR WINAPI StrChrIW(LPCWSTR lpszStr, WCHAR ch) 304 { 305 TRACE("(%s,%i)\n", debugstr_w(lpszStr), ch); 306 307 if (lpszStr) 308 { 309 ch = toupperW(ch); 310 while (*lpszStr) 311 { 312 if (toupperW(*lpszStr) == ch) 313 return (LPWSTR)lpszStr; 314 lpszStr++; 315 } 316 lpszStr = NULL; 317 } 318 return (LPWSTR)lpszStr; 319 } 320 321 /************************************************************************* 322 * StrChrNW [SHLWAPI.@] 323 */ 324 LPWSTR WINAPI StrChrNW(LPCWSTR lpszStr, WCHAR ch, UINT cchMax) 325 { 326 TRACE("(%s(%i),%i)\n", debugstr_wn(lpszStr,cchMax), cchMax, ch); 327 328 if (lpszStr) 329 { 330 while (*lpszStr && cchMax-- > 0) 331 { 332 if (*lpszStr == ch) 333 return (LPWSTR)lpszStr; 334 lpszStr++; 335 } 336 } 337 return NULL; 338 } 339 340 /************************************************************************* 341 * StrCmpIW [SHLWAPI.@] 342 * 343 * Compare two strings, ignoring case. 344 * 345 * PARAMS 346 * lpszStr [I] First string to compare 347 * lpszComp [I] Second string to compare 348 * 349 * RETURNS 350 * An integer less than, equal to or greater than 0, indicating that 351 * lpszStr is less than, the same, or greater than lpszComp. 352 */ 353 int WINAPI StrCmpIW(LPCWSTR lpszStr, LPCWSTR lpszComp) 354 { 355 TRACE("(%s,%s)\n", debugstr_w(lpszStr),debugstr_w(lpszComp)); 356 return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, lpszStr, -1, lpszComp, -1) - CSTR_EQUAL; 357 } 358 359 /************************************************************************* 360 * StrCmpNA [SHLWAPI.@] 361 * 362 * Compare two strings, up to a maximum length. 363 * 364 * PARAMS 365 * lpszStr [I] First string to compare 366 * lpszComp [I] Second string to compare 367 * iLen [I] Number of chars to compare 368 * 369 * RETURNS 370 * An integer less than, equal to or greater than 0, indicating that 371 * lpszStr is less than, the same, or greater than lpszComp. 372 */ 373 INT WINAPI StrCmpNA(LPCSTR lpszStr, LPCSTR lpszComp, INT iLen) 374 { 375 TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszComp), iLen); 376 return CompareStringA(GetThreadLocale(), 0, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL; 377 } 378 379 /************************************************************************* 380 * StrCmpNW [SHLWAPI.@] 381 * 382 * See StrCmpNA. 383 */ 384 INT WINAPI StrCmpNW(LPCWSTR lpszStr, LPCWSTR lpszComp, INT iLen) 385 { 386 TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszComp), iLen); 387 return CompareStringW(GetThreadLocale(), 0, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL; 388 } 389 390 /************************************************************************* 391 * StrCmpNIA [SHLWAPI.@] 392 * 393 * Compare two strings, up to a maximum length, ignoring case. 394 * 395 * PARAMS 396 * lpszStr [I] First string to compare 397 * lpszComp [I] Second string to compare 398 * iLen [I] Number of chars to compare 399 * 400 * RETURNS 401 * An integer less than, equal to or greater than 0, indicating that 402 * lpszStr is less than, the same, or greater than lpszComp. 403 */ 404 int WINAPI StrCmpNIA(LPCSTR lpszStr, LPCSTR lpszComp, int iLen) 405 { 406 TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszComp), iLen); 407 return CompareStringA(GetThreadLocale(), NORM_IGNORECASE, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL; 408 } 409 410 /************************************************************************* 411 * StrCmpNIW [SHLWAPI.@] 412 * 413 * See StrCmpNIA. 414 */ 415 INT WINAPI StrCmpNIW(LPCWSTR lpszStr, LPCWSTR lpszComp, int iLen) 416 { 417 TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszComp), iLen); 418 return CompareStringW(GetThreadLocale(), NORM_IGNORECASE, lpszStr, iLen, lpszComp, iLen) - CSTR_EQUAL; 419 } 420 421 /************************************************************************* 422 * StrCmpW [SHLWAPI.@] 423 * 424 * Compare two strings. 425 * 426 * PARAMS 427 * lpszStr [I] First string to compare 428 * lpszComp [I] Second string to compare 429 * 430 * RETURNS 431 * An integer less than, equal to or greater than 0, indicating that 432 * lpszStr is less than, the same, or greater than lpszComp. 433 */ 434 int WINAPI StrCmpW(LPCWSTR lpszStr, LPCWSTR lpszComp) 435 { 436 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszComp)); 437 return CompareStringW(GetThreadLocale(), 0, lpszStr, -1, lpszComp, -1) - CSTR_EQUAL; 438 } 439 440 /************************************************************************* 441 * StrCatW [SHLWAPI.@] 442 * 443 * Concatenate two strings. 444 * 445 * PARAMS 446 * lpszStr [O] Initial string 447 * lpszSrc [I] String to concatenate 448 * 449 * RETURNS 450 * lpszStr. 451 */ 452 LPWSTR WINAPI StrCatW(LPWSTR lpszStr, LPCWSTR lpszSrc) 453 { 454 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSrc)); 455 456 if (lpszStr && lpszSrc) 457 strcatW(lpszStr, lpszSrc); 458 return lpszStr; 459 } 460 461 /************************************************************************* 462 * StrCatChainW [SHLWAPI.@] 463 * 464 * Concatenates two unicode strings. 465 * 466 * PARAMS 467 * lpszStr [O] Initial string 468 * cchMax [I] Length of destination buffer 469 * ichAt [I] Offset from the destination buffer to begin concatenation 470 * lpszCat [I] String to concatenate 471 * 472 * RETURNS 473 * The offset from the beginning of pszDst to the terminating NULL. 474 */ 475 DWORD WINAPI StrCatChainW(LPWSTR lpszStr, DWORD cchMax, DWORD ichAt, LPCWSTR lpszCat) 476 { 477 TRACE("(%s,%u,%d,%s)\n", debugstr_w(lpszStr), cchMax, ichAt, debugstr_w(lpszCat)); 478 479 if (ichAt == -1) 480 ichAt = strlenW(lpszStr); 481 482 if (!cchMax) 483 return ichAt; 484 485 if (ichAt == cchMax) 486 ichAt--; 487 488 if (lpszCat && ichAt < cchMax) 489 { 490 lpszStr += ichAt; 491 while (ichAt < cchMax - 1 && *lpszCat) 492 { 493 *lpszStr++ = *lpszCat++; 494 ichAt++; 495 } 496 *lpszStr = 0; 497 } 498 499 return ichAt; 500 } 501 502 /************************************************************************* 503 * StrCpyW [SHLWAPI.@] 504 * 505 * Copy a string to another string. 506 * 507 * PARAMS 508 * lpszStr [O] Destination string 509 * lpszSrc [I] Source string 510 * 511 * RETURNS 512 * lpszStr. 513 */ 514 LPWSTR WINAPI StrCpyW(LPWSTR lpszStr, LPCWSTR lpszSrc) 515 { 516 TRACE("(%p,%s)\n", lpszStr, debugstr_w(lpszSrc)); 517 518 if (lpszStr && lpszSrc) 519 strcpyW(lpszStr, lpszSrc); 520 return lpszStr; 521 } 522 523 /************************************************************************* 524 * StrCpyNW [SHLWAPI.@] 525 * 526 * Copy a string to another string, up to a maximum number of characters. 527 * 528 * PARAMS 529 * dst [O] Destination string 530 * src [I] Source string 531 * count [I] Maximum number of chars to copy 532 * 533 * RETURNS 534 * dst. 535 */ 536 LPWSTR WINAPI StrCpyNW(LPWSTR dst, LPCWSTR src, int count) 537 { 538 LPWSTR d = dst; 539 LPCWSTR s = src; 540 541 TRACE("(%p,%s,%i)\n", dst, debugstr_w(src), count); 542 543 if (s) 544 { 545 while ((count > 1) && *s) 546 { 547 count--; 548 *d++ = *s++; 549 } 550 } 551 if (count) *d = 0; 552 553 return dst; 554 } 555 556 /************************************************************************* 557 * SHLWAPI_StrStrHelperA 558 * 559 * Internal implementation of StrStrA/StrStrIA 560 */ 561 static LPSTR SHLWAPI_StrStrHelperA(LPCSTR lpszStr, LPCSTR lpszSearch, 562 INT (WINAPI *pStrCmpFn)(LPCSTR,LPCSTR,INT)) 563 { 564 size_t iLen; 565 LPCSTR end; 566 567 if (!lpszStr || !lpszSearch || !*lpszSearch) 568 return NULL; 569 570 iLen = strlen(lpszSearch); 571 end = lpszStr + strlen(lpszStr); 572 573 while (lpszStr + iLen <= end) 574 { 575 if (!pStrCmpFn(lpszStr, lpszSearch, iLen)) 576 return (LPSTR)lpszStr; 577 lpszStr = CharNextA(lpszStr); 578 } 579 return NULL; 580 } 581 582 /************************************************************************* 583 * StrStrA [SHLWAPI.@] 584 * 585 * Find a substring within a string. 586 * 587 * PARAMS 588 * lpszStr [I] String to search in 589 * lpszSearch [I] String to look for 590 * 591 * RETURNS 592 * The start of lpszSearch within lpszStr, or NULL if not found. 593 */ 594 LPSTR WINAPI StrStrA(LPCSTR lpszStr, LPCSTR lpszSearch) 595 { 596 TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch)); 597 598 return SHLWAPI_StrStrHelperA(lpszStr, lpszSearch, StrCmpNA); 599 } 600 601 /************************************************************************* 602 * StrStrW [SHLWAPI.@] 603 * 604 * See StrStrA. 605 */ 606 LPWSTR WINAPI StrStrW(LPCWSTR lpszStr, LPCWSTR lpszSearch) 607 { 608 TRACE("(%s, %s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch)); 609 610 if (!lpszStr || !lpszSearch || !*lpszSearch) return NULL; 611 return strstrW( lpszStr, lpszSearch ); 612 } 613 614 /************************************************************************* 615 * StrRStrIA [SHLWAPI.@] 616 * 617 * Find the last occurrence of a substring within a string. 618 * 619 * PARAMS 620 * lpszStr [I] String to search in 621 * lpszEnd [I] End of lpszStr 622 * lpszSearch [I] String to look for 623 * 624 * RETURNS 625 * The last occurrence lpszSearch within lpszStr, or NULL if not found. 626 */ 627 LPSTR WINAPI StrRStrIA(LPCSTR lpszStr, LPCSTR lpszEnd, LPCSTR lpszSearch) 628 { 629 LPSTR lpszRet = NULL; 630 WORD ch1, ch2; 631 INT iLen; 632 633 TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch)); 634 635 if (!lpszStr || !lpszSearch || !*lpszSearch) 636 return NULL; 637 638 if (IsDBCSLeadByte(*lpszSearch)) 639 ch1 = *lpszSearch << 8 | (UCHAR)lpszSearch[1]; 640 else 641 ch1 = *lpszSearch; 642 iLen = lstrlenA(lpszSearch); 643 644 if (!lpszEnd) 645 lpszEnd = lpszStr + lstrlenA(lpszStr); 646 else /* reproduce the broken behaviour on Windows */ 647 lpszEnd += min(iLen - 1, lstrlenA(lpszEnd)); 648 649 while (lpszStr + iLen <= lpszEnd && *lpszStr) 650 { 651 ch2 = IsDBCSLeadByte(*lpszStr)? *lpszStr << 8 | (UCHAR)lpszStr[1] : *lpszStr; 652 if (!ChrCmpIA(ch1, ch2)) 653 { 654 if (!StrCmpNIA(lpszStr, lpszSearch, iLen)) 655 lpszRet = (LPSTR)lpszStr; 656 } 657 lpszStr = CharNextA(lpszStr); 658 } 659 return lpszRet; 660 } 661 662 /************************************************************************* 663 * StrRStrIW [SHLWAPI.@] 664 * 665 * See StrRStrIA. 666 */ 667 LPWSTR WINAPI StrRStrIW(LPCWSTR lpszStr, LPCWSTR lpszEnd, LPCWSTR lpszSearch) 668 { 669 LPWSTR lpszRet = NULL; 670 INT iLen; 671 672 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch)); 673 674 if (!lpszStr || !lpszSearch || !*lpszSearch) 675 return NULL; 676 677 iLen = strlenW(lpszSearch); 678 679 if (!lpszEnd) 680 lpszEnd = lpszStr + strlenW(lpszStr); 681 else /* reproduce the broken behaviour on Windows */ 682 lpszEnd += min(iLen - 1, lstrlenW(lpszEnd)); 683 684 while (lpszStr + iLen <= lpszEnd && *lpszStr) 685 { 686 if (!ChrCmpIW(*lpszSearch, *lpszStr)) 687 { 688 if (!StrCmpNIW(lpszStr, lpszSearch, iLen)) 689 lpszRet = (LPWSTR)lpszStr; 690 } 691 lpszStr++; 692 } 693 return lpszRet; 694 } 695 696 /************************************************************************* 697 * StrStrIA [SHLWAPI.@] 698 * 699 * Find a substring within a string, ignoring case. 700 * 701 * PARAMS 702 * lpszStr [I] String to search in 703 * lpszSearch [I] String to look for 704 * 705 * RETURNS 706 * The start of lpszSearch within lpszStr, or NULL if not found. 707 */ 708 LPSTR WINAPI StrStrIA(LPCSTR lpszStr, LPCSTR lpszSearch) 709 { 710 TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszSearch)); 711 712 return SHLWAPI_StrStrHelperA(lpszStr, lpszSearch, StrCmpNIA); 713 } 714 715 /************************************************************************* 716 * StrStrIW [SHLWAPI.@] 717 * 718 * See StrStrIA. 719 */ 720 LPWSTR WINAPI StrStrIW(LPCWSTR lpszStr, LPCWSTR lpszSearch) 721 { 722 int iLen; 723 LPCWSTR end; 724 725 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszSearch)); 726 727 if (!lpszStr || !lpszSearch || !*lpszSearch) 728 return NULL; 729 730 iLen = strlenW(lpszSearch); 731 end = lpszStr + strlenW(lpszStr); 732 733 while (lpszStr + iLen <= end) 734 { 735 if (!StrCmpNIW(lpszStr, lpszSearch, iLen)) 736 return (LPWSTR)lpszStr; 737 lpszStr++; 738 } 739 return NULL; 740 } 741 742 /************************************************************************* 743 * StrStrNW [SHLWAPI.@] 744 * 745 * Find a substring within a string up to a given number of initial characters. 746 * 747 * PARAMS 748 * lpFirst [I] String to search in 749 * lpSrch [I] String to look for 750 * cchMax [I] Maximum number of initial search characters 751 * 752 * RETURNS 753 * The start of lpFirst within lpSrch, or NULL if not found. 754 */ 755 LPWSTR WINAPI StrStrNW(LPCWSTR lpFirst, LPCWSTR lpSrch, UINT cchMax) 756 { 757 UINT i; 758 int len; 759 760 TRACE("(%s, %s, %u)\n", debugstr_w(lpFirst), debugstr_w(lpSrch), cchMax); 761 762 if (!lpFirst || !lpSrch || !*lpSrch || !cchMax) 763 return NULL; 764 765 len = strlenW(lpSrch); 766 767 for (i = cchMax; *lpFirst && (i > 0); i--, lpFirst++) 768 { 769 if (!strncmpW(lpFirst, lpSrch, len)) 770 return (LPWSTR)lpFirst; 771 } 772 773 return NULL; 774 } 775 776 /************************************************************************* 777 * StrStrNIW [SHLWAPI.@] 778 * 779 * Find a substring within a string up to a given number of initial characters, 780 * ignoring case. 781 * 782 * PARAMS 783 * lpFirst [I] String to search in 784 * lpSrch [I] String to look for 785 * cchMax [I] Maximum number of initial search characters 786 * 787 * RETURNS 788 * The start of lpFirst within lpSrch, or NULL if not found. 789 */ 790 LPWSTR WINAPI StrStrNIW(LPCWSTR lpFirst, LPCWSTR lpSrch, UINT cchMax) 791 { 792 UINT i; 793 int len; 794 795 TRACE("(%s, %s, %u)\n", debugstr_w(lpFirst), debugstr_w(lpSrch), cchMax); 796 797 if (!lpFirst || !lpSrch || !*lpSrch || !cchMax) 798 return NULL; 799 800 len = strlenW(lpSrch); 801 802 for (i = cchMax; *lpFirst && (i > 0); i--, lpFirst++) 803 { 804 if (!strncmpiW(lpFirst, lpSrch, len)) 805 return (LPWSTR)lpFirst; 806 } 807 808 return NULL; 809 } 810 811 /************************************************************************* 812 * StrToIntA [SHLWAPI.@] 813 * 814 * Read a signed integer from a string. 815 * 816 * PARAMS 817 * lpszStr [I] String to read integer from 818 * 819 * RETURNS 820 * The signed integer value represented by the string, or 0 if no integer is 821 * present. 822 * 823 * NOTES 824 * No leading space is allowed before the number, although a leading '-' is. 825 */ 826 int WINAPI StrToIntA(LPCSTR lpszStr) 827 { 828 int iRet = 0; 829 830 TRACE("(%s)\n", debugstr_a(lpszStr)); 831 832 if (!lpszStr) 833 { 834 WARN("Invalid lpszStr would crash under Win32!\n"); 835 return 0; 836 } 837 838 if (*lpszStr == '-' || isdigit(*lpszStr)) 839 StrToIntExA(lpszStr, 0, &iRet); 840 return iRet; 841 } 842 843 /************************************************************************* 844 * StrToIntW [SHLWAPI.@] 845 * 846 * See StrToIntA. 847 */ 848 int WINAPI StrToIntW(LPCWSTR lpszStr) 849 { 850 int iRet = 0; 851 852 TRACE("(%s)\n", debugstr_w(lpszStr)); 853 854 if (!lpszStr) 855 { 856 WARN("Invalid lpszStr would crash under Win32!\n"); 857 return 0; 858 } 859 860 if (*lpszStr == '-' || isdigitW(*lpszStr)) 861 StrToIntExW(lpszStr, 0, &iRet); 862 return iRet; 863 } 864 865 /************************************************************************* 866 * StrToIntExA [SHLWAPI.@] 867 * 868 * Read an integer from a string. 869 * 870 * PARAMS 871 * lpszStr [I] String to read integer from 872 * dwFlags [I] Flags controlling the conversion 873 * lpiRet [O] Destination for read integer. 874 * 875 * RETURNS 876 * Success: TRUE. lpiRet contains the integer value represented by the string. 877 * Failure: FALSE, if the string is invalid, or no number is present. 878 * 879 * NOTES 880 * Leading whitespace, '-' and '+' are allowed before the number. If 881 * dwFlags includes STIF_SUPPORT_HEX, hexadecimal numbers are allowed, if 882 * preceded by '0x'. If this flag is not set, or there is no '0x' prefix, 883 * the string is treated as a decimal string. A leading '-' is ignored for 884 * hexadecimal numbers. 885 */ 886 BOOL WINAPI StrToIntExA(LPCSTR lpszStr, DWORD dwFlags, int *lpiRet) 887 { 888 LONGLONG li; 889 BOOL bRes; 890 891 TRACE("(%s,%08X,%p)\n", debugstr_a(lpszStr), dwFlags, lpiRet); 892 893 bRes = StrToInt64ExA(lpszStr, dwFlags, &li); 894 if (bRes) *lpiRet = li; 895 return bRes; 896 } 897 898 /************************************************************************* 899 * StrToInt64ExA [SHLWAPI.@] 900 * 901 * See StrToIntExA. 902 */ 903 BOOL WINAPI StrToInt64ExA(LPCSTR lpszStr, DWORD dwFlags, LONGLONG *lpiRet) 904 { 905 BOOL bNegative = FALSE; 906 LONGLONG iRet = 0; 907 908 TRACE("(%s,%08X,%p)\n", debugstr_a(lpszStr), dwFlags, lpiRet); 909 910 if (!lpszStr || !lpiRet) 911 { 912 WARN("Invalid parameter would crash under Win32!\n"); 913 return FALSE; 914 } 915 if (dwFlags > STIF_SUPPORT_HEX) WARN("Unknown flags %08x\n", dwFlags); 916 917 /* Skip leading space, '+', '-' */ 918 while (isspace(*lpszStr)) 919 lpszStr = CharNextA(lpszStr); 920 921 if (*lpszStr == '-') 922 { 923 bNegative = TRUE; 924 lpszStr++; 925 } 926 else if (*lpszStr == '+') 927 lpszStr++; 928 929 if (dwFlags & STIF_SUPPORT_HEX && 930 *lpszStr == '0' && tolower(lpszStr[1]) == 'x') 931 { 932 /* Read hex number */ 933 lpszStr += 2; 934 935 if (!isxdigit(*lpszStr)) 936 return FALSE; 937 938 while (isxdigit(*lpszStr)) 939 { 940 iRet = iRet * 16; 941 if (isdigit(*lpszStr)) 942 iRet += (*lpszStr - '0'); 943 else 944 iRet += 10 + (tolower(*lpszStr) - 'a'); 945 lpszStr++; 946 } 947 *lpiRet = iRet; 948 return TRUE; 949 } 950 951 /* Read decimal number */ 952 if (!isdigit(*lpszStr)) 953 return FALSE; 954 955 while (isdigit(*lpszStr)) 956 { 957 iRet = iRet * 10; 958 iRet += (*lpszStr - '0'); 959 lpszStr++; 960 } 961 *lpiRet = bNegative ? -iRet : iRet; 962 return TRUE; 963 } 964 965 /************************************************************************* 966 * StrToIntExW [SHLWAPI.@] 967 * 968 * See StrToIntExA. 969 */ 970 BOOL WINAPI StrToIntExW(LPCWSTR lpszStr, DWORD dwFlags, int *lpiRet) 971 { 972 LONGLONG li; 973 BOOL bRes; 974 975 TRACE("(%s,%08X,%p)\n", debugstr_w(lpszStr), dwFlags, lpiRet); 976 977 bRes = StrToInt64ExW(lpszStr, dwFlags, &li); 978 if (bRes) *lpiRet = li; 979 return bRes; 980 } 981 982 /************************************************************************* 983 * StrToInt64ExW [SHLWAPI.@] 984 * 985 * See StrToIntExA. 986 */ 987 BOOL WINAPI StrToInt64ExW(LPCWSTR lpszStr, DWORD dwFlags, LONGLONG *lpiRet) 988 { 989 BOOL bNegative = FALSE; 990 LONGLONG iRet = 0; 991 992 TRACE("(%s,%08X,%p)\n", debugstr_w(lpszStr), dwFlags, lpiRet); 993 994 if (!lpszStr || !lpiRet) 995 { 996 WARN("Invalid parameter would crash under Win32!\n"); 997 return FALSE; 998 } 999 if (dwFlags > STIF_SUPPORT_HEX) WARN("Unknown flags %08x\n", dwFlags); 1000 1001 /* Skip leading space, '+', '-' */ 1002 while (isspaceW(*lpszStr)) lpszStr++; 1003 1004 if (*lpszStr == '-') 1005 { 1006 bNegative = TRUE; 1007 lpszStr++; 1008 } 1009 else if (*lpszStr == '+') 1010 lpszStr++; 1011 1012 if (dwFlags & STIF_SUPPORT_HEX && 1013 *lpszStr == '0' && tolowerW(lpszStr[1]) == 'x') 1014 { 1015 /* Read hex number */ 1016 lpszStr += 2; 1017 1018 if (!isxdigitW(*lpszStr)) 1019 return FALSE; 1020 1021 while (isxdigitW(*lpszStr)) 1022 { 1023 iRet = iRet * 16; 1024 if (isdigitW(*lpszStr)) 1025 iRet += (*lpszStr - '0'); 1026 else 1027 iRet += 10 + (tolowerW(*lpszStr) - 'a'); 1028 lpszStr++; 1029 } 1030 *lpiRet = iRet; 1031 return TRUE; 1032 } 1033 1034 /* Read decimal number */ 1035 if (!isdigitW(*lpszStr)) 1036 return FALSE; 1037 1038 while (isdigitW(*lpszStr)) 1039 { 1040 iRet = iRet * 10; 1041 iRet += (*lpszStr - '0'); 1042 lpszStr++; 1043 } 1044 *lpiRet = bNegative ? -iRet : iRet; 1045 return TRUE; 1046 } 1047 1048 /************************************************************************* 1049 * StrDupA [SHLWAPI.@] 1050 * 1051 * Duplicate a string. 1052 * 1053 * PARAMS 1054 * lpszStr [I] String to duplicate. 1055 * 1056 * RETURNS 1057 * Success: A pointer to a new string containing the contents of lpszStr 1058 * Failure: NULL, if memory cannot be allocated 1059 * 1060 * NOTES 1061 * The string memory is allocated with LocalAlloc(), and so should be released 1062 * by calling LocalFree(). 1063 */ 1064 LPSTR WINAPI StrDupA(LPCSTR lpszStr) 1065 { 1066 int iLen; 1067 LPSTR lpszRet; 1068 1069 TRACE("(%s)\n",debugstr_a(lpszStr)); 1070 1071 #ifdef __REACTOS__ 1072 if (!lpszStr) 1073 return NULL; 1074 #endif 1075 iLen = lpszStr ? strlen(lpszStr) + 1 : 1; 1076 lpszRet = LocalAlloc(LMEM_FIXED, iLen); 1077 1078 if (lpszRet) 1079 { 1080 if (lpszStr) 1081 memcpy(lpszRet, lpszStr, iLen); 1082 else 1083 *lpszRet = '\0'; 1084 } 1085 return lpszRet; 1086 } 1087 1088 /************************************************************************* 1089 * StrDupW [SHLWAPI.@] 1090 * 1091 * See StrDupA. 1092 */ 1093 LPWSTR WINAPI StrDupW(LPCWSTR lpszStr) 1094 { 1095 int iLen; 1096 LPWSTR lpszRet; 1097 1098 TRACE("(%s)\n",debugstr_w(lpszStr)); 1099 1100 #ifdef __REACTOS__ 1101 if (!lpszStr) 1102 return NULL; 1103 #endif 1104 iLen = (lpszStr ? strlenW(lpszStr) + 1 : 1) * sizeof(WCHAR); 1105 lpszRet = LocalAlloc(LMEM_FIXED, iLen); 1106 1107 if (lpszRet) 1108 { 1109 if (lpszStr) 1110 memcpy(lpszRet, lpszStr, iLen); 1111 else 1112 *lpszRet = '\0'; 1113 } 1114 return lpszRet; 1115 } 1116 1117 /************************************************************************* 1118 * SHLWAPI_StrSpnHelperA 1119 * 1120 * Internal implementation of StrSpnA/StrCSpnA/StrCSpnIA 1121 */ 1122 static int SHLWAPI_StrSpnHelperA(LPCSTR lpszStr, LPCSTR lpszMatch, 1123 LPSTR (WINAPI *pStrChrFn)(LPCSTR,WORD), 1124 BOOL bInvert) 1125 { 1126 LPCSTR lpszRead = lpszStr; 1127 if (lpszStr && *lpszStr && lpszMatch) 1128 { 1129 while (*lpszRead) 1130 { 1131 LPCSTR lpszTest = pStrChrFn(lpszMatch, *lpszRead); 1132 1133 if (!bInvert && !lpszTest) 1134 break; 1135 if (bInvert && lpszTest) 1136 break; 1137 lpszRead = CharNextA(lpszRead); 1138 }; 1139 } 1140 return lpszRead - lpszStr; 1141 } 1142 1143 /************************************************************************* 1144 * StrSpnA [SHLWAPI.@] 1145 * 1146 * Find the length of the start of a string that contains only certain 1147 * characters. 1148 * 1149 * PARAMS 1150 * lpszStr [I] String to search 1151 * lpszMatch [I] Characters that can be in the substring 1152 * 1153 * RETURNS 1154 * The length of the part of lpszStr containing only chars from lpszMatch, 1155 * or 0 if any parameter is invalid. 1156 */ 1157 int WINAPI StrSpnA(LPCSTR lpszStr, LPCSTR lpszMatch) 1158 { 1159 TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch)); 1160 1161 return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrA, FALSE); 1162 } 1163 1164 /************************************************************************* 1165 * StrSpnW [SHLWAPI.@] 1166 * 1167 * See StrSpnA. 1168 */ 1169 int WINAPI StrSpnW(LPCWSTR lpszStr, LPCWSTR lpszMatch) 1170 { 1171 if (!lpszStr || !lpszMatch) return 0; 1172 return strspnW( lpszStr, lpszMatch ); 1173 } 1174 1175 /************************************************************************* 1176 * StrCSpnA [SHLWAPI.@] 1177 * 1178 * Find the length of the start of a string that does not contain certain 1179 * characters. 1180 * 1181 * PARAMS 1182 * lpszStr [I] String to search 1183 * lpszMatch [I] Characters that cannot be in the substring 1184 * 1185 * RETURNS 1186 * The length of the part of lpszStr containing only chars not in lpszMatch, 1187 * or 0 if any parameter is invalid. 1188 */ 1189 int WINAPI StrCSpnA(LPCSTR lpszStr, LPCSTR lpszMatch) 1190 { 1191 TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch)); 1192 1193 return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrA, TRUE); 1194 } 1195 1196 /************************************************************************* 1197 * StrCSpnW [SHLWAPI.@] 1198 * 1199 * See StrCSpnA. 1200 */ 1201 int WINAPI StrCSpnW(LPCWSTR lpszStr, LPCWSTR lpszMatch) 1202 { 1203 if (!lpszStr || !lpszMatch) return 0; 1204 return strcspnW( lpszStr, lpszMatch ); 1205 } 1206 1207 /************************************************************************* 1208 * StrCSpnIA [SHLWAPI.@] 1209 * 1210 * Find the length of the start of a string that does not contain certain 1211 * characters, ignoring case. 1212 * 1213 * PARAMS 1214 * lpszStr [I] String to search 1215 * lpszMatch [I] Characters that cannot be in the substring 1216 * 1217 * RETURNS 1218 * The length of the part of lpszStr containing only chars not in lpszMatch, 1219 * or 0 if any parameter is invalid. 1220 */ 1221 int WINAPI StrCSpnIA(LPCSTR lpszStr, LPCSTR lpszMatch) 1222 { 1223 TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch)); 1224 1225 return SHLWAPI_StrSpnHelperA(lpszStr, lpszMatch, StrChrIA, TRUE); 1226 } 1227 1228 /************************************************************************* 1229 * StrCSpnIW [SHLWAPI.@] 1230 * 1231 * See StrCSpnIA. 1232 */ 1233 int WINAPI StrCSpnIW(LPCWSTR lpszStr, LPCWSTR lpszMatch) 1234 { 1235 LPCWSTR lpszRead = lpszStr; 1236 1237 TRACE("(%s,%s)\n",debugstr_w(lpszStr), debugstr_w(lpszMatch)); 1238 1239 if (lpszStr && *lpszStr && lpszMatch) 1240 { 1241 while (*lpszRead) 1242 { 1243 if (StrChrIW(lpszMatch, *lpszRead)) break; 1244 lpszRead++; 1245 } 1246 } 1247 return lpszRead - lpszStr; 1248 } 1249 1250 /************************************************************************* 1251 * StrPBrkA [SHLWAPI.@] 1252 * 1253 * Search a string for any of a group of characters. 1254 * 1255 * PARAMS 1256 * lpszStr [I] String to search 1257 * lpszMatch [I] Characters to match 1258 * 1259 * RETURNS 1260 * A pointer to the first matching character in lpszStr, or NULL if no 1261 * match was found. 1262 */ 1263 LPSTR WINAPI StrPBrkA(LPCSTR lpszStr, LPCSTR lpszMatch) 1264 { 1265 TRACE("(%s,%s)\n",debugstr_a(lpszStr), debugstr_a(lpszMatch)); 1266 1267 if (lpszStr && lpszMatch && *lpszMatch) 1268 { 1269 while (*lpszStr) 1270 { 1271 if (StrChrA(lpszMatch, *lpszStr)) 1272 return (LPSTR)lpszStr; 1273 lpszStr = CharNextA(lpszStr); 1274 } 1275 } 1276 return NULL; 1277 } 1278 1279 /************************************************************************* 1280 * StrPBrkW [SHLWAPI.@] 1281 * 1282 * See StrPBrkA. 1283 */ 1284 LPWSTR WINAPI StrPBrkW(LPCWSTR lpszStr, LPCWSTR lpszMatch) 1285 { 1286 if (!lpszStr || !lpszMatch) return NULL; 1287 return strpbrkW( lpszStr, lpszMatch ); 1288 } 1289 1290 /************************************************************************* 1291 * SHLWAPI_StrRChrHelperA 1292 * 1293 * Internal implementation of StrRChrA/StrRChrIA. 1294 */ 1295 static LPSTR SHLWAPI_StrRChrHelperA(LPCSTR lpszStr, 1296 LPCSTR lpszEnd, WORD ch, 1297 BOOL (WINAPI *pChrCmpFn)(WORD,WORD)) 1298 { 1299 LPCSTR lpszRet = NULL; 1300 1301 if (lpszStr) 1302 { 1303 WORD ch2; 1304 1305 if (!lpszEnd) 1306 lpszEnd = lpszStr + lstrlenA(lpszStr); 1307 1308 while (*lpszStr && lpszStr <= lpszEnd) 1309 { 1310 ch2 = IsDBCSLeadByte(*lpszStr)? *lpszStr << 8 | lpszStr[1] : *lpszStr; 1311 1312 if (!pChrCmpFn(ch, ch2)) 1313 lpszRet = lpszStr; 1314 lpszStr = CharNextA(lpszStr); 1315 } 1316 } 1317 return (LPSTR)lpszRet; 1318 } 1319 1320 /************************************************************************** 1321 * StrRChrA [SHLWAPI.@] 1322 * 1323 * Find the last occurrence of a character in string. 1324 * 1325 * PARAMS 1326 * lpszStr [I] String to search in 1327 * lpszEnd [I] Place to end search, or NULL to search until the end of lpszStr 1328 * ch [I] Character to search for. 1329 * 1330 * RETURNS 1331 * Success: A pointer to the last occurrence of ch in lpszStr before lpszEnd, 1332 * or NULL if not found. 1333 * Failure: NULL, if any arguments are invalid. 1334 */ 1335 LPSTR WINAPI StrRChrA(LPCSTR lpszStr, LPCSTR lpszEnd, WORD ch) 1336 { 1337 TRACE("(%s,%s,%x)\n", debugstr_a(lpszStr), debugstr_a(lpszEnd), ch); 1338 1339 return SHLWAPI_StrRChrHelperA(lpszStr, lpszEnd, ch, SHLWAPI_ChrCmpA); 1340 } 1341 1342 /************************************************************************** 1343 * StrRChrW [SHLWAPI.@] 1344 * 1345 * See StrRChrA. 1346 */ 1347 LPWSTR WINAPI StrRChrW(LPCWSTR str, LPCWSTR end, WORD ch) 1348 { 1349 WCHAR *ret = NULL; 1350 1351 if (!str) return NULL; 1352 if (!end) end = str + strlenW(str); 1353 while (str < end) 1354 { 1355 if (*str == ch) ret = (WCHAR *)str; 1356 str++; 1357 } 1358 return ret; 1359 } 1360 1361 /************************************************************************** 1362 * StrRChrIA [SHLWAPI.@] 1363 * 1364 * Find the last occurrence of a character in string, ignoring case. 1365 * 1366 * PARAMS 1367 * lpszStr [I] String to search in 1368 * lpszEnd [I] Place to end search, or NULL to search until the end of lpszStr 1369 * ch [I] Character to search for. 1370 * 1371 * RETURNS 1372 * Success: A pointer to the last occurrence of ch in lpszStr before lpszEnd, 1373 * or NULL if not found. 1374 * Failure: NULL, if any arguments are invalid. 1375 */ 1376 LPSTR WINAPI StrRChrIA(LPCSTR lpszStr, LPCSTR lpszEnd, WORD ch) 1377 { 1378 TRACE("(%s,%s,%x)\n", debugstr_a(lpszStr), debugstr_a(lpszEnd), ch); 1379 1380 return SHLWAPI_StrRChrHelperA(lpszStr, lpszEnd, ch, ChrCmpIA); 1381 } 1382 1383 /************************************************************************** 1384 * StrRChrIW [SHLWAPI.@] 1385 * 1386 * See StrRChrIA. 1387 */ 1388 LPWSTR WINAPI StrRChrIW(LPCWSTR str, LPCWSTR end, WORD ch) 1389 { 1390 WCHAR *ret = NULL; 1391 1392 if (!str) return NULL; 1393 if (!end) end = str + strlenW(str); 1394 while (str < end) 1395 { 1396 if (!ChrCmpIW(*str, ch)) ret = (WCHAR *)str; 1397 str++; 1398 } 1399 return ret; 1400 } 1401 1402 /************************************************************************* 1403 * StrCatBuffA [SHLWAPI.@] 1404 * 1405 * Concatenate two strings together. 1406 * 1407 * PARAMS 1408 * lpszStr [O] String to concatenate to 1409 * lpszCat [I] String to add to lpszCat 1410 * cchMax [I] Maximum number of characters for the whole string 1411 * 1412 * RETURNS 1413 * lpszStr. 1414 * 1415 * NOTES 1416 * cchMax determines the number of characters in the final length of the 1417 * string, not the number appended to lpszStr from lpszCat. 1418 */ 1419 LPSTR WINAPI StrCatBuffA(LPSTR lpszStr, LPCSTR lpszCat, INT cchMax) 1420 { 1421 INT iLen; 1422 1423 TRACE("(%p,%s,%d)\n", lpszStr, debugstr_a(lpszCat), cchMax); 1424 1425 if (!lpszStr) 1426 { 1427 WARN("Invalid lpszStr would crash under Win32!\n"); 1428 return NULL; 1429 } 1430 1431 iLen = strlen(lpszStr); 1432 cchMax -= iLen; 1433 1434 if (cchMax > 0) 1435 StrCpyNA(lpszStr + iLen, lpszCat, cchMax); 1436 return lpszStr; 1437 } 1438 1439 /************************************************************************* 1440 * StrCatBuffW [SHLWAPI.@] 1441 * 1442 * See StrCatBuffA. 1443 */ 1444 LPWSTR WINAPI StrCatBuffW(LPWSTR lpszStr, LPCWSTR lpszCat, INT cchMax) 1445 { 1446 INT iLen; 1447 1448 TRACE("(%p,%s,%d)\n", lpszStr, debugstr_w(lpszCat), cchMax); 1449 1450 if (!lpszStr) 1451 { 1452 WARN("Invalid lpszStr would crash under Win32!\n"); 1453 return NULL; 1454 } 1455 1456 iLen = strlenW(lpszStr); 1457 cchMax -= iLen; 1458 1459 if (cchMax > 0) 1460 StrCpyNW(lpszStr + iLen, lpszCat, cchMax); 1461 return lpszStr; 1462 } 1463 1464 /************************************************************************* 1465 * StrRetToBufA [SHLWAPI.@] 1466 * 1467 * Convert a STRRET to a normal string. 1468 * 1469 * PARAMS 1470 * lpStrRet [O] STRRET to convert 1471 * pIdl [I] ITEMIDLIST for lpStrRet->uType == STRRET_OFFSET 1472 * lpszDest [O] Destination for normal string 1473 * dwLen [I] Length of lpszDest 1474 * 1475 * RETURNS 1476 * Success: S_OK. lpszDest contains up to dwLen characters of the string. 1477 * If lpStrRet is of type STRRET_WSTR, its memory is freed with 1478 * CoTaskMemFree() and its type set to STRRET_CSTRA. 1479 * Failure: E_FAIL, if any parameters are invalid. 1480 */ 1481 HRESULT WINAPI StrRetToBufA (LPSTRRET src, const ITEMIDLIST *pidl, LPSTR dest, UINT len) 1482 { 1483 /* NOTE: 1484 * This routine is identical to that in dlls/shell32/shellstring.c. 1485 * It was duplicated because not every version of Shlwapi.dll exports 1486 * StrRetToBufA. If you change one routine, change them both. 1487 */ 1488 TRACE("dest=%p len=0x%x strret=%p pidl=%p\n", dest, len, src, pidl); 1489 1490 if (!src) 1491 { 1492 WARN("Invalid lpStrRet would crash under Win32!\n"); 1493 if (dest) 1494 *dest = '\0'; 1495 return E_FAIL; 1496 } 1497 1498 if (!dest || !len) 1499 return E_FAIL; 1500 1501 *dest = '\0'; 1502 1503 switch (src->uType) 1504 { 1505 case STRRET_WSTR: 1506 WideCharToMultiByte(CP_ACP, 0, src->u.pOleStr, -1, dest, len, NULL, NULL); 1507 CoTaskMemFree(src->u.pOleStr); 1508 break; 1509 1510 case STRRET_CSTR: 1511 lstrcpynA(dest, src->u.cStr, len); 1512 break; 1513 1514 case STRRET_OFFSET: 1515 lstrcpynA((LPSTR)dest, ((LPCSTR)&pidl->mkid)+src->u.uOffset, len); 1516 break; 1517 1518 default: 1519 FIXME("unknown type!\n"); 1520 return E_NOTIMPL; 1521 } 1522 return S_OK; 1523 } 1524 1525 /************************************************************************* 1526 * StrRetToBufW [SHLWAPI.@] 1527 * 1528 * See StrRetToBufA. 1529 */ 1530 HRESULT WINAPI StrRetToBufW (LPSTRRET src, const ITEMIDLIST *pidl, LPWSTR dest, UINT len) 1531 { 1532 TRACE("dest=%p len=0x%x strret=%p pidl=%p\n", dest, len, src, pidl); 1533 1534 if (!dest || !len) 1535 return E_FAIL; 1536 1537 if (!src) 1538 { 1539 WARN("Invalid lpStrRet would crash under Win32!\n"); 1540 if (dest) 1541 *dest = '\0'; 1542 return E_FAIL; 1543 } 1544 1545 *dest = '\0'; 1546 1547 switch (src->uType) { 1548 case STRRET_WSTR: { 1549 size_t dst_len; 1550 if (!src->u.pOleStr) 1551 return E_FAIL; 1552 dst_len = strlenW(src->u.pOleStr); 1553 memcpy(dest, src->u.pOleStr, min(dst_len, len-1) * sizeof(WCHAR)); 1554 dest[min(dst_len, len-1)] = 0; 1555 CoTaskMemFree(src->u.pOleStr); 1556 if (len <= dst_len) 1557 { 1558 dest[0] = 0; 1559 return E_NOT_SUFFICIENT_BUFFER; 1560 } 1561 break; 1562 } 1563 1564 case STRRET_CSTR: 1565 if (!MultiByteToWideChar( CP_ACP, 0, src->u.cStr, -1, dest, len )) 1566 dest[len-1] = 0; 1567 break; 1568 1569 case STRRET_OFFSET: 1570 if (pidl) 1571 { 1572 if (!MultiByteToWideChar( CP_ACP, 0, ((LPCSTR)&pidl->mkid)+src->u.uOffset, -1, 1573 dest, len )) 1574 dest[len-1] = 0; 1575 } 1576 break; 1577 1578 default: 1579 FIXME("unknown type!\n"); 1580 return E_NOTIMPL; 1581 } 1582 1583 return S_OK; 1584 } 1585 1586 /************************************************************************* 1587 * StrRetToStrA [SHLWAPI.@] 1588 * 1589 * Converts a STRRET to a normal string. 1590 * 1591 * PARAMS 1592 * lpStrRet [O] STRRET to convert 1593 * pidl [I] ITEMIDLIST for lpStrRet->uType == STRRET_OFFSET 1594 * ppszName [O] Destination for converted string 1595 * 1596 * RETURNS 1597 * Success: S_OK. ppszName contains the new string, allocated with CoTaskMemAlloc(). 1598 * Failure: E_FAIL, if any parameters are invalid. 1599 */ 1600 HRESULT WINAPI StrRetToStrA(LPSTRRET lpStrRet, const ITEMIDLIST *pidl, LPSTR *ppszName) 1601 { 1602 HRESULT hRet = E_FAIL; 1603 1604 switch (lpStrRet->uType) 1605 { 1606 case STRRET_WSTR: 1607 hRet = _SHStrDupAW(lpStrRet->u.pOleStr, ppszName); 1608 CoTaskMemFree(lpStrRet->u.pOleStr); 1609 break; 1610 1611 case STRRET_CSTR: 1612 hRet = _SHStrDupAA(lpStrRet->u.cStr, ppszName); 1613 break; 1614 1615 case STRRET_OFFSET: 1616 hRet = _SHStrDupAA(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, ppszName); 1617 break; 1618 1619 default: 1620 *ppszName = NULL; 1621 } 1622 1623 return hRet; 1624 } 1625 1626 /************************************************************************* 1627 * StrRetToStrW [SHLWAPI.@] 1628 * 1629 * See StrRetToStrA. 1630 */ 1631 HRESULT WINAPI StrRetToStrW(LPSTRRET lpStrRet, const ITEMIDLIST *pidl, LPWSTR *ppszName) 1632 { 1633 HRESULT hRet = E_FAIL; 1634 1635 switch (lpStrRet->uType) 1636 { 1637 case STRRET_WSTR: 1638 #ifdef __REACTOS__ 1639 hRet = lpStrRet->u.pOleStr ? S_OK : E_FAIL; 1640 *ppszName = lpStrRet->u.pOleStr; 1641 lpStrRet->u.pOleStr = NULL; /* Windows does this, presumably in case someone calls SHFree */ 1642 #else 1643 hRet = SHStrDupW(lpStrRet->u.pOleStr, ppszName); 1644 CoTaskMemFree(lpStrRet->u.pOleStr); 1645 #endif 1646 break; 1647 1648 case STRRET_CSTR: 1649 hRet = SHStrDupA(lpStrRet->u.cStr, ppszName); 1650 break; 1651 1652 case STRRET_OFFSET: 1653 hRet = SHStrDupA(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, ppszName); 1654 break; 1655 1656 default: 1657 *ppszName = NULL; 1658 } 1659 1660 return hRet; 1661 } 1662 1663 /* Create an ASCII string copy using SysAllocString() */ 1664 static HRESULT _SHStrDupAToBSTR(LPCSTR src, BSTR *pBstrOut) 1665 { 1666 *pBstrOut = NULL; 1667 1668 if (src) 1669 { 1670 INT len = MultiByteToWideChar(CP_ACP, 0, src, -1, NULL, 0); 1671 WCHAR* szTemp = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); 1672 1673 if (szTemp) 1674 { 1675 MultiByteToWideChar(CP_ACP, 0, src, -1, szTemp, len); 1676 *pBstrOut = SysAllocString(szTemp); 1677 HeapFree(GetProcessHeap(), 0, szTemp); 1678 1679 if (*pBstrOut) 1680 return S_OK; 1681 } 1682 } 1683 return E_OUTOFMEMORY; 1684 } 1685 1686 /************************************************************************* 1687 * StrRetToBSTR [SHLWAPI.@] 1688 * 1689 * Converts a STRRET to a BSTR. 1690 * 1691 * PARAMS 1692 * lpStrRet [O] STRRET to convert 1693 * pidl [I] ITEMIDLIST for lpStrRet->uType = STRRET_OFFSET 1694 * pBstrOut [O] Destination for converted BSTR 1695 * 1696 * RETURNS 1697 * Success: S_OK. pBstrOut contains the new string. 1698 * Failure: E_FAIL, if any parameters are invalid. 1699 */ 1700 HRESULT WINAPI StrRetToBSTR(STRRET *lpStrRet, LPCITEMIDLIST pidl, BSTR* pBstrOut) 1701 { 1702 HRESULT hRet = E_FAIL; 1703 1704 switch (lpStrRet->uType) 1705 { 1706 case STRRET_WSTR: 1707 *pBstrOut = SysAllocString(lpStrRet->u.pOleStr); 1708 if (*pBstrOut) 1709 hRet = S_OK; 1710 CoTaskMemFree(lpStrRet->u.pOleStr); 1711 break; 1712 1713 case STRRET_CSTR: 1714 hRet = _SHStrDupAToBSTR(lpStrRet->u.cStr, pBstrOut); 1715 break; 1716 1717 case STRRET_OFFSET: 1718 hRet = _SHStrDupAToBSTR(((LPCSTR)&pidl->mkid) + lpStrRet->u.uOffset, pBstrOut); 1719 break; 1720 1721 default: 1722 *pBstrOut = NULL; 1723 } 1724 1725 return hRet; 1726 } 1727 1728 /************************************************************************* 1729 * StrFormatKBSizeA [SHLWAPI.@] 1730 * 1731 * Create a formatted string containing a byte count in Kilobytes. 1732 * 1733 * PARAMS 1734 * llBytes [I] Byte size to format 1735 * lpszDest [I] Destination for formatted string 1736 * cchMax [I] Size of lpszDest 1737 * 1738 * RETURNS 1739 * lpszDest. 1740 */ 1741 LPSTR WINAPI StrFormatKBSizeA(LONGLONG llBytes, LPSTR lpszDest, UINT cchMax) 1742 { 1743 WCHAR wszBuf[256]; 1744 1745 if (!StrFormatKBSizeW(llBytes, wszBuf, 256)) 1746 return NULL; 1747 if (!WideCharToMultiByte(CP_ACP, 0, wszBuf, -1, lpszDest, cchMax, NULL, NULL)) 1748 return NULL; 1749 return lpszDest; 1750 } 1751 1752 /************************************************************************* 1753 * StrFormatKBSizeW [SHLWAPI.@] 1754 * 1755 * See StrFormatKBSizeA. 1756 */ 1757 LPWSTR WINAPI StrFormatKBSizeW(LONGLONG llBytes, LPWSTR lpszDest, UINT cchMax) 1758 { 1759 static const WCHAR kb[] = {' ','K','B',0}; 1760 LONGLONG llKB = (llBytes + 1023) >> 10; 1761 int len; 1762 1763 TRACE("(0x%s,%p,%d)\n", wine_dbgstr_longlong(llBytes), lpszDest, cchMax); 1764 1765 if (!FormatInt(llKB, lpszDest, cchMax)) 1766 return NULL; 1767 1768 len = lstrlenW(lpszDest); 1769 if (cchMax - len < 4) 1770 return NULL; 1771 lstrcatW(lpszDest, kb); 1772 return lpszDest; 1773 } 1774 1775 /************************************************************************* 1776 * StrNCatA [SHLWAPI.@] 1777 * 1778 * Concatenate two strings together. 1779 * 1780 * PARAMS 1781 * lpszStr [O] String to concatenate to 1782 * lpszCat [I] String to add to lpszCat 1783 * cchMax [I] Maximum number of characters to concatenate 1784 * 1785 * RETURNS 1786 * lpszStr. 1787 * 1788 * NOTES 1789 * cchMax determines the number of characters that are appended to lpszStr, 1790 * not the total length of the string. 1791 */ 1792 LPSTR WINAPI StrNCatA(LPSTR lpszStr, LPCSTR lpszCat, INT cchMax) 1793 { 1794 LPSTR lpszRet = lpszStr; 1795 1796 TRACE("(%s,%s,%i)\n", debugstr_a(lpszStr), debugstr_a(lpszCat), cchMax); 1797 1798 if (!lpszStr) 1799 { 1800 WARN("Invalid lpszStr would crash under Win32!\n"); 1801 return NULL; 1802 } 1803 1804 StrCpyNA(lpszStr + strlen(lpszStr), lpszCat, cchMax); 1805 return lpszRet; 1806 } 1807 1808 /************************************************************************* 1809 * StrNCatW [SHLWAPI.@] 1810 * 1811 * See StrNCatA. 1812 */ 1813 LPWSTR WINAPI StrNCatW(LPWSTR lpszStr, LPCWSTR lpszCat, INT cchMax) 1814 { 1815 LPWSTR lpszRet = lpszStr; 1816 1817 TRACE("(%s,%s,%i)\n", debugstr_w(lpszStr), debugstr_w(lpszCat), cchMax); 1818 1819 if (!lpszStr) 1820 { 1821 WARN("Invalid lpszStr would crash under Win32\n"); 1822 return NULL; 1823 } 1824 1825 StrCpyNW(lpszStr + strlenW(lpszStr), lpszCat, cchMax); 1826 return lpszRet; 1827 } 1828 1829 /************************************************************************* 1830 * StrTrimA [SHLWAPI.@] 1831 * 1832 * Remove characters from the start and end of a string. 1833 * 1834 * PARAMS 1835 * lpszStr [O] String to remove characters from 1836 * lpszTrim [I] Characters to remove from lpszStr 1837 * 1838 * RETURNS 1839 * TRUE If lpszStr was valid and modified 1840 * FALSE Otherwise 1841 */ 1842 BOOL WINAPI StrTrimA(LPSTR lpszStr, LPCSTR lpszTrim) 1843 { 1844 DWORD dwLen; 1845 LPSTR lpszRead = lpszStr; 1846 BOOL bRet = FALSE; 1847 1848 TRACE("(%s,%s)\n", debugstr_a(lpszStr), debugstr_a(lpszTrim)); 1849 1850 if (lpszRead && *lpszRead) 1851 { 1852 while (*lpszRead && StrChrA(lpszTrim, *lpszRead)) 1853 lpszRead = CharNextA(lpszRead); /* Skip leading matches */ 1854 1855 dwLen = strlen(lpszRead); 1856 1857 if (lpszRead != lpszStr) 1858 { 1859 memmove(lpszStr, lpszRead, dwLen + 1); 1860 bRet = TRUE; 1861 } 1862 if (dwLen > 0) 1863 { 1864 lpszRead = lpszStr + dwLen; 1865 while (StrChrA(lpszTrim, lpszRead[-1])) 1866 lpszRead = CharPrevA(lpszStr, lpszRead); /* Skip trailing matches */ 1867 1868 if (lpszRead != lpszStr + dwLen) 1869 { 1870 *lpszRead = '\0'; 1871 bRet = TRUE; 1872 } 1873 } 1874 } 1875 return bRet; 1876 } 1877 1878 /************************************************************************* 1879 * StrTrimW [SHLWAPI.@] 1880 * 1881 * See StrTrimA. 1882 */ 1883 BOOL WINAPI StrTrimW(LPWSTR lpszStr, LPCWSTR lpszTrim) 1884 { 1885 DWORD dwLen; 1886 LPWSTR lpszRead = lpszStr; 1887 BOOL bRet = FALSE; 1888 1889 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszTrim)); 1890 1891 if (lpszRead && *lpszRead) 1892 { 1893 while (*lpszRead && StrChrW(lpszTrim, *lpszRead)) lpszRead++; 1894 1895 dwLen = strlenW(lpszRead); 1896 1897 if (lpszRead != lpszStr) 1898 { 1899 memmove(lpszStr, lpszRead, (dwLen + 1) * sizeof(WCHAR)); 1900 bRet = TRUE; 1901 } 1902 if (dwLen > 0) 1903 { 1904 lpszRead = lpszStr + dwLen; 1905 while (StrChrW(lpszTrim, lpszRead[-1])) 1906 lpszRead--; /* Skip trailing matches */ 1907 1908 if (lpszRead != lpszStr + dwLen) 1909 { 1910 *lpszRead = '\0'; 1911 bRet = TRUE; 1912 } 1913 } 1914 } 1915 return bRet; 1916 } 1917 1918 /************************************************************************* 1919 * _SHStrDupAA [INTERNAL] 1920 * 1921 * Duplicates a ASCII string to ASCII. The destination buffer is allocated. 1922 */ 1923 static HRESULT _SHStrDupAA(LPCSTR src, LPSTR * dest) 1924 { 1925 HRESULT hr; 1926 int len = 0; 1927 1928 if (src) { 1929 len = lstrlenA(src) + 1; 1930 *dest = CoTaskMemAlloc(len); 1931 } else { 1932 *dest = NULL; 1933 } 1934 1935 if (*dest) { 1936 lstrcpynA(*dest,src, len); 1937 hr = S_OK; 1938 } else { 1939 hr = E_OUTOFMEMORY; 1940 } 1941 1942 TRACE("%s->(%p)\n", debugstr_a(src), *dest); 1943 return hr; 1944 } 1945 1946 /************************************************************************* 1947 * SHStrDupA [SHLWAPI.@] 1948 * 1949 * Return a Unicode copy of a string, in memory allocated by CoTaskMemAlloc(). 1950 * 1951 * PARAMS 1952 * lpszStr [I] String to copy 1953 * lppszDest [O] Destination for the new string copy 1954 * 1955 * RETURNS 1956 * Success: S_OK. lppszDest contains the new string in Unicode format. 1957 * Failure: E_OUTOFMEMORY, If any arguments are invalid or memory allocation 1958 * fails. 1959 */ 1960 HRESULT WINAPI SHStrDupA(LPCSTR lpszStr, LPWSTR * lppszDest) 1961 { 1962 HRESULT hRet; 1963 int len = 0; 1964 1965 if (lpszStr) 1966 { 1967 len = MultiByteToWideChar(CP_ACP, 0, lpszStr, -1, NULL, 0) * sizeof(WCHAR); 1968 *lppszDest = CoTaskMemAlloc(len); 1969 } 1970 else 1971 *lppszDest = NULL; 1972 1973 if (*lppszDest) 1974 { 1975 MultiByteToWideChar(CP_ACP, 0, lpszStr, -1, *lppszDest, len/sizeof(WCHAR)); 1976 hRet = S_OK; 1977 } 1978 else 1979 hRet = E_OUTOFMEMORY; 1980 1981 TRACE("%s->(%p)\n", debugstr_a(lpszStr), *lppszDest); 1982 return hRet; 1983 } 1984 1985 /************************************************************************* 1986 * _SHStrDupAW [INTERNAL] 1987 * 1988 * Duplicates a UNICODE to a ASCII string. The destination buffer is allocated. 1989 */ 1990 static HRESULT _SHStrDupAW(LPCWSTR src, LPSTR * dest) 1991 { 1992 HRESULT hr; 1993 int len = 0; 1994 1995 if (src) { 1996 len = WideCharToMultiByte(CP_ACP, 0, src, -1, NULL, 0, NULL, NULL); 1997 *dest = CoTaskMemAlloc(len); 1998 } else { 1999 *dest = NULL; 2000 } 2001 2002 if (*dest) { 2003 WideCharToMultiByte(CP_ACP, 0, src, -1, *dest, len, NULL, NULL); 2004 hr = S_OK; 2005 } else { 2006 hr = E_OUTOFMEMORY; 2007 } 2008 2009 TRACE("%s->(%p)\n", debugstr_w(src), *dest); 2010 return hr; 2011 } 2012 2013 /************************************************************************* 2014 * SHStrDupW [SHLWAPI.@] 2015 * 2016 * See SHStrDupA. 2017 */ 2018 HRESULT WINAPI SHStrDupW(LPCWSTR src, LPWSTR * dest) 2019 { 2020 HRESULT hr; 2021 int len = 0; 2022 2023 if (src) { 2024 len = (lstrlenW(src) + 1) * sizeof(WCHAR); 2025 *dest = CoTaskMemAlloc(len); 2026 } else { 2027 *dest = NULL; 2028 } 2029 2030 if (*dest) { 2031 memcpy(*dest, src, len); 2032 hr = S_OK; 2033 } else { 2034 hr = E_OUTOFMEMORY; 2035 } 2036 2037 TRACE("%s->(%p)\n", debugstr_w(src), *dest); 2038 return hr; 2039 } 2040 2041 /************************************************************************* 2042 * SHLWAPI_WriteReverseNum 2043 * 2044 * Internal helper for SHLWAPI_WriteTimeClass. 2045 */ 2046 static inline LPWSTR SHLWAPI_WriteReverseNum(LPWSTR lpszOut, DWORD dwNum) 2047 { 2048 *lpszOut-- = '\0'; 2049 2050 /* Write a decimal number to a string, backwards */ 2051 do 2052 { 2053 DWORD dwNextDigit = dwNum % 10; 2054 *lpszOut-- = '0' + dwNextDigit; 2055 dwNum = (dwNum - dwNextDigit) / 10; 2056 } while (dwNum > 0); 2057 2058 return lpszOut; 2059 } 2060 2061 /************************************************************************* 2062 * SHLWAPI_FormatSignificant 2063 * 2064 * Internal helper for SHLWAPI_WriteTimeClass. 2065 */ 2066 static inline int SHLWAPI_FormatSignificant(LPWSTR lpszNum, int dwDigits) 2067 { 2068 /* Zero non significant digits, return remaining significant digits */ 2069 while (*lpszNum) 2070 { 2071 lpszNum++; 2072 if (--dwDigits == 0) 2073 { 2074 while (*lpszNum) 2075 *lpszNum++ = '0'; 2076 return 0; 2077 } 2078 } 2079 return dwDigits; 2080 } 2081 2082 /************************************************************************* 2083 * SHLWAPI_WriteTimeClass 2084 * 2085 * Internal helper for StrFromTimeIntervalW. 2086 */ 2087 static int SHLWAPI_WriteTimeClass(LPWSTR lpszOut, DWORD dwValue, 2088 UINT uClassStringId, int iDigits) 2089 { 2090 WCHAR szBuff[64], *szOut = szBuff + 32; 2091 2092 szOut = SHLWAPI_WriteReverseNum(szOut, dwValue); 2093 iDigits = SHLWAPI_FormatSignificant(szOut + 1, iDigits); 2094 *szOut = ' '; 2095 LoadStringW(shlwapi_hInstance, uClassStringId, szBuff + 32, 32); 2096 strcatW(lpszOut, szOut); 2097 return iDigits; 2098 } 2099 2100 /************************************************************************* 2101 * StrFromTimeIntervalA [SHLWAPI.@] 2102 * 2103 * Format a millisecond time interval into a string 2104 * 2105 * PARAMS 2106 * lpszStr [O] Output buffer for formatted time interval 2107 * cchMax [I] Size of lpszStr 2108 * dwMS [I] Number of milliseconds 2109 * iDigits [I] Number of digits to print 2110 * 2111 * RETURNS 2112 * The length of the formatted string, or 0 if any parameter is invalid. 2113 * 2114 * NOTES 2115 * This implementation mimics the Win32 behaviour of always writing a leading 2116 * space before the time interval begins. 2117 * 2118 * iDigits is used to provide approximate times if accuracy is not important. 2119 * This number of digits will be written of the first non-zero time class 2120 * (hours/minutes/seconds). If this does not complete the time classification, 2121 * the remaining digits are changed to zeros (i.e. The time is _not_ rounded). 2122 * If there are digits remaining following the writing of a time class, the 2123 * next time class will be written. 2124 * 2125 * For example, given dwMS represents 138 hours,43 minutes and 15 seconds, the 2126 * following will result from the given values of iDigits: 2127 * 2128 *| iDigits 1 2 3 4 5 ... 2129 *| lpszStr "100 hr" "130 hr" "138 hr" "138 hr 40 min" "138 hr 43 min" ... 2130 */ 2131 INT WINAPI StrFromTimeIntervalA(LPSTR lpszStr, UINT cchMax, DWORD dwMS, 2132 int iDigits) 2133 { 2134 INT iRet = 0; 2135 2136 TRACE("(%p,%d,%d,%d)\n", lpszStr, cchMax, dwMS, iDigits); 2137 2138 if (lpszStr && cchMax) 2139 { 2140 WCHAR szBuff[128]; 2141 StrFromTimeIntervalW(szBuff, sizeof(szBuff)/sizeof(WCHAR), dwMS, iDigits); 2142 WideCharToMultiByte(CP_ACP,0,szBuff,-1,lpszStr,cchMax,0,0); 2143 } 2144 return iRet; 2145 } 2146 2147 2148 /************************************************************************* 2149 * StrFromTimeIntervalW [SHLWAPI.@] 2150 * 2151 * See StrFromTimeIntervalA. 2152 */ 2153 INT WINAPI StrFromTimeIntervalW(LPWSTR lpszStr, UINT cchMax, DWORD dwMS, 2154 int iDigits) 2155 { 2156 INT iRet = 0; 2157 2158 TRACE("(%p,%d,%d,%d)\n", lpszStr, cchMax, dwMS, iDigits); 2159 2160 if (lpszStr && cchMax) 2161 { 2162 WCHAR szCopy[128]; 2163 DWORD dwHours, dwMinutes; 2164 2165 if (!iDigits || cchMax == 1) 2166 { 2167 *lpszStr = '\0'; 2168 return 0; 2169 } 2170 2171 /* Calculate the time classes */ 2172 dwMS = (dwMS + 500) / 1000; 2173 dwHours = dwMS / 3600; 2174 dwMS -= dwHours * 3600; 2175 dwMinutes = dwMS / 60; 2176 dwMS -= dwMinutes * 60; 2177 2178 szCopy[0] = '\0'; 2179 2180 if (dwHours) 2181 iDigits = SHLWAPI_WriteTimeClass(szCopy, dwHours, IDS_TIME_INTERVAL_HOURS, iDigits); 2182 2183 if (dwMinutes && iDigits) 2184 iDigits = SHLWAPI_WriteTimeClass(szCopy, dwMinutes, IDS_TIME_INTERVAL_MINUTES, iDigits); 2185 2186 if (iDigits) /* Always write seconds if we have significant digits */ 2187 SHLWAPI_WriteTimeClass(szCopy, dwMS, IDS_TIME_INTERVAL_SECONDS, iDigits); 2188 2189 lstrcpynW(lpszStr, szCopy, cchMax); 2190 iRet = strlenW(lpszStr); 2191 } 2192 return iRet; 2193 } 2194 2195 /************************************************************************* 2196 * StrIsIntlEqualA [SHLWAPI.@] 2197 * 2198 * Compare two strings. 2199 * 2200 * PARAMS 2201 * bCase [I] Whether to compare case sensitively 2202 * lpszStr [I] First string to compare 2203 * lpszComp [I] Second string to compare 2204 * iLen [I] Length to compare 2205 * 2206 * RETURNS 2207 * TRUE If the strings are equal. 2208 * FALSE Otherwise. 2209 */ 2210 BOOL WINAPI StrIsIntlEqualA(BOOL bCase, LPCSTR lpszStr, LPCSTR lpszComp, 2211 int iLen) 2212 { 2213 DWORD dwFlags; 2214 2215 TRACE("(%d,%s,%s,%d)\n", bCase, 2216 debugstr_a(lpszStr), debugstr_a(lpszComp), iLen); 2217 2218 /* FIXME: This flag is undocumented and unknown by our CompareString. 2219 * We need a define for it. 2220 */ 2221 dwFlags = 0x10000000; 2222 if (!bCase) dwFlags |= NORM_IGNORECASE; 2223 2224 return (CompareStringA(GetThreadLocale(), dwFlags, lpszStr, iLen, lpszComp, iLen) == CSTR_EQUAL); 2225 } 2226 2227 /************************************************************************* 2228 * StrIsIntlEqualW [SHLWAPI.@] 2229 * 2230 * See StrIsIntlEqualA. 2231 */ 2232 BOOL WINAPI StrIsIntlEqualW(BOOL bCase, LPCWSTR lpszStr, LPCWSTR lpszComp, 2233 int iLen) 2234 { 2235 DWORD dwFlags; 2236 2237 TRACE("(%d,%s,%s,%d)\n", bCase, 2238 debugstr_w(lpszStr),debugstr_w(lpszComp), iLen); 2239 2240 /* FIXME: This flag is undocumented and unknown by our CompareString. 2241 * We need a define for it. 2242 */ 2243 dwFlags = 0x10000000; 2244 if (!bCase) dwFlags |= NORM_IGNORECASE; 2245 2246 return (CompareStringW(GetThreadLocale(), dwFlags, lpszStr, iLen, lpszComp, iLen) == CSTR_EQUAL); 2247 } 2248 2249 /************************************************************************* 2250 * @ [SHLWAPI.399] 2251 * 2252 * Copy a string to another string, up to a maximum number of characters. 2253 * 2254 * PARAMS 2255 * lpszDest [O] Destination string 2256 * lpszSrc [I] Source string 2257 * iLen [I] Maximum number of chars to copy 2258 * 2259 * RETURNS 2260 * Success: A pointer to the last character written to lpszDest. 2261 * Failure: lpszDest, if any arguments are invalid. 2262 */ 2263 LPSTR WINAPI StrCpyNXA(LPSTR lpszDest, LPCSTR lpszSrc, int iLen) 2264 { 2265 TRACE("(%p,%s,%i)\n", lpszDest, debugstr_a(lpszSrc), iLen); 2266 2267 if (lpszDest && lpszSrc && iLen > 0) 2268 { 2269 while ((iLen-- > 1) && *lpszSrc) 2270 *lpszDest++ = *lpszSrc++; 2271 if (iLen >= 0) 2272 *lpszDest = '\0'; 2273 } 2274 return lpszDest; 2275 } 2276 2277 /************************************************************************* 2278 * @ [SHLWAPI.400] 2279 * 2280 * Unicode version of StrCpyNXA. 2281 */ 2282 LPWSTR WINAPI StrCpyNXW(LPWSTR lpszDest, LPCWSTR lpszSrc, int iLen) 2283 { 2284 TRACE("(%p,%s,%i)\n", lpszDest, debugstr_w(lpszSrc), iLen); 2285 2286 if (lpszDest && lpszSrc && iLen > 0) 2287 { 2288 while ((iLen-- > 1) && *lpszSrc) 2289 *lpszDest++ = *lpszSrc++; 2290 if (iLen >= 0) 2291 *lpszDest = '\0'; 2292 } 2293 return lpszDest; 2294 } 2295 2296 /************************************************************************* 2297 * StrCmpLogicalW [SHLWAPI.@] 2298 * 2299 * Compare two strings, ignoring case and comparing digits as numbers. 2300 * 2301 * PARAMS 2302 * lpszStr [I] First string to compare 2303 * lpszComp [I] Second string to compare 2304 * iLen [I] Length to compare 2305 * 2306 * RETURNS 2307 * TRUE If the strings are equal. 2308 * FALSE Otherwise. 2309 */ 2310 INT WINAPI StrCmpLogicalW(LPCWSTR lpszStr, LPCWSTR lpszComp) 2311 { 2312 INT iDiff; 2313 2314 TRACE("(%s,%s)\n", debugstr_w(lpszStr), debugstr_w(lpszComp)); 2315 2316 if (lpszStr && lpszComp) 2317 { 2318 while (*lpszStr) 2319 { 2320 if (!*lpszComp) 2321 return 1; 2322 else if (isdigitW(*lpszStr)) 2323 { 2324 int iStr, iComp; 2325 2326 if (!isdigitW(*lpszComp)) 2327 return -1; 2328 2329 /* Compare the numbers */ 2330 StrToIntExW(lpszStr, 0, &iStr); 2331 StrToIntExW(lpszComp, 0, &iComp); 2332 2333 if (iStr < iComp) 2334 return -1; 2335 else if (iStr > iComp) 2336 return 1; 2337 2338 /* Skip */ 2339 while (isdigitW(*lpszStr)) 2340 lpszStr++; 2341 while (isdigitW(*lpszComp)) 2342 lpszComp++; 2343 } 2344 else if (isdigitW(*lpszComp)) 2345 return 1; 2346 else 2347 { 2348 iDiff = ChrCmpIW(*lpszStr,*lpszComp); 2349 if (iDiff > 0) 2350 return 1; 2351 else if (iDiff < 0) 2352 return -1; 2353 2354 lpszStr++; 2355 lpszComp++; 2356 } 2357 } 2358 if (*lpszComp) 2359 return -1; 2360 } 2361 return 0; 2362 } 2363 2364 /* Structure for formatting byte strings */ 2365 typedef struct tagSHLWAPI_BYTEFORMATS 2366 { 2367 LONGLONG dLimit; 2368 double dDivisor; 2369 double dNormaliser; 2370 int nDecimals; 2371 #ifdef __REACTOS__ 2372 UINT nFormatID; 2373 #else 2374 WCHAR wPrefix; 2375 #endif 2376 } SHLWAPI_BYTEFORMATS; 2377 2378 /************************************************************************* 2379 * StrFormatByteSizeW [SHLWAPI.@] 2380 * 2381 * Create a string containing an abbreviated byte count of up to 2^63-1. 2382 * 2383 * PARAMS 2384 * llBytes [I] Byte size to format 2385 * lpszDest [I] Destination for formatted string 2386 * cchMax [I] Size of lpszDest 2387 * 2388 * RETURNS 2389 * lpszDest. 2390 * 2391 * NOTES 2392 * There is no StrFormatByteSize64W function, it is called StrFormatByteSizeW(). 2393 */ 2394 LPWSTR WINAPI StrFormatByteSizeW(LONGLONG llBytes, LPWSTR lpszDest, UINT cchMax) 2395 { 2396 #define KB ((ULONGLONG)1024) 2397 #define MB (KB*KB) 2398 #define GB (KB*KB*KB) 2399 #define TB (KB*KB*KB*KB) 2400 #define PB (KB*KB*KB*KB*KB) 2401 2402 static const SHLWAPI_BYTEFORMATS bfFormats[] = 2403 { 2404 #ifdef __REACTOS__ 2405 { 10*KB, 10.24, 100.0, 2, IDS_KB_FORMAT }, /* 10 KB */ 2406 { 100*KB, 102.4, 10.0, 1, IDS_KB_FORMAT }, /* 100 KB */ 2407 { 1000*KB, 1024.0, 1.0, 0, IDS_KB_FORMAT }, /* 1000 KB */ 2408 { 10*MB, 10485.76, 100.0, 2, IDS_MB_FORMAT }, /* 10 MB */ 2409 { 100*MB, 104857.6, 10.0, 1, IDS_MB_FORMAT }, /* 100 MB */ 2410 { 1000*MB, 1048576.0, 1.0, 0, IDS_MB_FORMAT }, /* 1000 MB */ 2411 { 10*GB, 10737418.24, 100.0, 2, IDS_GB_FORMAT }, /* 10 GB */ 2412 { 100*GB, 107374182.4, 10.0, 1, IDS_GB_FORMAT }, /* 100 GB */ 2413 { 1000*GB, 1073741824.0, 1.0, 0, IDS_GB_FORMAT }, /* 1000 GB */ 2414 { 10*TB, 10485.76, 100.0, 2, IDS_TB_FORMAT }, /* 10 TB */ 2415 { 100*TB, 104857.6, 10.0, 1, IDS_TB_FORMAT }, /* 100 TB */ 2416 { 1000*TB, 1048576.0, 1.0, 0, IDS_TB_FORMAT }, /* 1000 TB */ 2417 { 10*PB, 10737418.24, 100.00, 2, IDS_PB_FORMAT }, /* 10 PB */ 2418 { 100*PB, 107374182.4, 10.00, 1, IDS_PB_FORMAT }, /* 100 PB */ 2419 { 1000*PB, 1073741824.0, 1.00, 0, IDS_PB_FORMAT }, /* 1000 PB */ 2420 { 0, 10995116277.76, 100.00, 2, IDS_EB_FORMAT } /* EB's, catch all */ 2421 #else 2422 { 10*KB, 10.24, 100.0, 2, 'K' }, /* 10 KB */ 2423 { 100*KB, 102.4, 10.0, 1, 'K' }, /* 100 KB */ 2424 { 1000*KB, 1024.0, 1.0, 0, 'K' }, /* 1000 KB */ 2425 { 10*MB, 10485.76, 100.0, 2, 'M' }, /* 10 MB */ 2426 { 100*MB, 104857.6, 10.0, 1, 'M' }, /* 100 MB */ 2427 { 1000*MB, 1048576.0, 1.0, 0, 'M' }, /* 1000 MB */ 2428 { 10*GB, 10737418.24, 100.0, 2, 'G' }, /* 10 GB */ 2429 { 100*GB, 107374182.4, 10.0, 1, 'G' }, /* 100 GB */ 2430 { 1000*GB, 1073741824.0, 1.0, 0, 'G' }, /* 1000 GB */ 2431 { 10*TB, 10485.76, 100.0, 2, 'T' }, /* 10 TB */ 2432 { 100*TB, 104857.6, 10.0, 1, 'T' }, /* 100 TB */ 2433 { 1000*TB, 1048576.0, 1.0, 0, 'T' }, /* 1000 TB */ 2434 { 10*PB, 10737418.24, 100.00, 2, 'P' }, /* 10 PB */ 2435 { 100*PB, 107374182.4, 10.00, 1, 'P' }, /* 100 PB */ 2436 { 1000*PB, 1073741824.0, 1.00, 0, 'P' }, /* 1000 PB */ 2437 { 0, 10995116277.76, 100.00, 2, 'E' } /* EB's, catch all */ 2438 #endif 2439 }; 2440 #ifdef __REACTOS__ 2441 WCHAR szBuff[40], wszFormat[40]; 2442 #else 2443 WCHAR wszAdd[] = {' ','?','B',0}; 2444 #endif 2445 double dBytes; 2446 UINT i = 0; 2447 2448 TRACE("(0x%s,%p,%d)\n", wine_dbgstr_longlong(llBytes), lpszDest, cchMax); 2449 2450 if (!lpszDest || !cchMax) 2451 return lpszDest; 2452 2453 if (llBytes < 1024) /* 1K */ 2454 { 2455 WCHAR wszBytesFormat[64]; 2456 LoadStringW(shlwapi_hInstance, IDS_BYTES_FORMAT, wszBytesFormat, 64); 2457 snprintfW(lpszDest, cchMax, wszBytesFormat, (int)llBytes); 2458 return lpszDest; 2459 } 2460 2461 /* Note that if this loop completes without finding a match, i will be 2462 * pointing at the last entry, which is a catch all for > 1000 PB 2463 */ 2464 while (i < sizeof(bfFormats) / sizeof(SHLWAPI_BYTEFORMATS) - 1) 2465 { 2466 if (llBytes < bfFormats[i].dLimit) 2467 break; 2468 i++; 2469 } 2470 /* Above 1 TB we encounter problems with FP accuracy. So for amounts above 2471 * this number we integer shift down by 1 MB first. The table above has 2472 * the divisors scaled down from the '< 10 TB' entry onwards, to account 2473 * for this. We also add a small fudge factor to get the correct result for 2474 * counts that lie exactly on a 1024 byte boundary. 2475 */ 2476 if (i > 8) 2477 dBytes = (double)(llBytes >> 20) + 0.001; /* Scale down by 1 MB */ 2478 else 2479 dBytes = (double)llBytes + 0.00001; 2480 2481 dBytes = floor(dBytes / bfFormats[i].dDivisor) / bfFormats[i].dNormaliser; 2482 2483 #ifdef __REACTOS__ 2484 if (!FormatDouble(dBytes, bfFormats[i].nDecimals, szBuff, ARRAYSIZE(szBuff))) 2485 return NULL; 2486 LoadStringW(shlwapi_hInstance, bfFormats[i].nFormatID, wszFormat, ARRAYSIZE(wszFormat)); 2487 snprintfW(lpszDest, cchMax, wszFormat, szBuff); 2488 #else 2489 if (!FormatDouble(dBytes, bfFormats[i].nDecimals, lpszDest, cchMax)) 2490 return NULL; 2491 wszAdd[1] = bfFormats[i].wPrefix; 2492 StrCatBuffW(lpszDest, wszAdd, cchMax); 2493 #endif 2494 return lpszDest; 2495 } 2496 2497 /************************************************************************* 2498 * StrFormatByteSize64A [SHLWAPI.@] 2499 * 2500 * See StrFormatByteSizeW. 2501 */ 2502 LPSTR WINAPI StrFormatByteSize64A(LONGLONG llBytes, LPSTR lpszDest, UINT cchMax) 2503 { 2504 WCHAR wszBuff[32]; 2505 2506 StrFormatByteSizeW(llBytes, wszBuff, sizeof(wszBuff)/sizeof(WCHAR)); 2507 2508 if (lpszDest) 2509 WideCharToMultiByte(CP_ACP, 0, wszBuff, -1, lpszDest, cchMax, 0, 0); 2510 return lpszDest; 2511 } 2512 2513 /************************************************************************* 2514 * StrFormatByteSizeA [SHLWAPI.@] 2515 * 2516 * Create a string containing an abbreviated byte count of up to 2^31-1. 2517 * 2518 * PARAMS 2519 * dwBytes [I] Byte size to format 2520 * lpszDest [I] Destination for formatted string 2521 * cchMax [I] Size of lpszDest 2522 * 2523 * RETURNS 2524 * lpszDest. 2525 * 2526 * NOTES 2527 * The Ascii and Unicode versions of this function accept a different 2528 * integer type for dwBytes. See StrFormatByteSize64A(). 2529 */ 2530 LPSTR WINAPI StrFormatByteSizeA(DWORD dwBytes, LPSTR lpszDest, UINT cchMax) 2531 { 2532 TRACE("(%d,%p,%d)\n", dwBytes, lpszDest, cchMax); 2533 2534 return StrFormatByteSize64A(dwBytes, lpszDest, cchMax); 2535 } 2536 2537 /************************************************************************* 2538 * @ [SHLWAPI.162] 2539 * 2540 * Remove a hanging lead byte from the end of a string, if present. 2541 * 2542 * PARAMS 2543 * lpStr [I] String to check for a hanging lead byte 2544 * size [I] Length of lpStr 2545 * 2546 * RETURNS 2547 * Success: The new length of the string. Any hanging lead bytes are removed. 2548 * Failure: 0, if any parameters are invalid. 2549 */ 2550 DWORD WINAPI SHTruncateString(LPSTR lpStr, DWORD size) 2551 { 2552 if (lpStr && size) 2553 { 2554 LPSTR lastByte = lpStr + size - 1; 2555 2556 while(lpStr < lastByte) 2557 lpStr += IsDBCSLeadByte(*lpStr) ? 2 : 1; 2558 2559 if(lpStr == lastByte && IsDBCSLeadByte(*lpStr)) 2560 { 2561 *lpStr = '\0'; 2562 size--; 2563 } 2564 return size; 2565 } 2566 return 0; 2567 } 2568 2569 /************************************************************************* 2570 * @ [SHLWAPI.203] 2571 * 2572 * Remove a single non-trailing ampersand ('&') from a string. 2573 * 2574 * PARAMS 2575 * lpszStr [I/O] String to remove ampersand from. 2576 * 2577 * RETURNS 2578 * The character after the first ampersand in lpszStr, or the first character 2579 * in lpszStr if there is no ampersand in the string. 2580 */ 2581 char WINAPI SHStripMneumonicA(LPCSTR lpszStr) 2582 { 2583 LPSTR lpszIter, lpszTmp; 2584 char ch; 2585 2586 TRACE("(%s)\n", debugstr_a(lpszStr)); 2587 2588 ch = *lpszStr; 2589 2590 if ((lpszIter = StrChrA(lpszStr, '&'))) 2591 { 2592 lpszTmp = CharNextA(lpszIter); 2593 if (*lpszTmp) 2594 { 2595 if (*lpszTmp != '&') 2596 ch = *lpszTmp; 2597 2598 memmove( lpszIter, lpszTmp, strlen(lpszTmp) + 1 ); 2599 } 2600 } 2601 2602 return ch; 2603 } 2604 2605 /************************************************************************* 2606 * @ [SHLWAPI.225] 2607 * 2608 * Unicode version of SHStripMneumonicA. 2609 */ 2610 WCHAR WINAPI SHStripMneumonicW(LPCWSTR lpszStr) 2611 { 2612 LPWSTR lpszIter, lpszTmp; 2613 WCHAR ch; 2614 2615 TRACE("(%s)\n", debugstr_w(lpszStr)); 2616 2617 ch = *lpszStr; 2618 2619 if ((lpszIter = StrChrW(lpszStr, '&'))) 2620 { 2621 lpszTmp = lpszIter + 1; 2622 if (*lpszTmp) 2623 { 2624 if (*lpszTmp != '&') 2625 ch = *lpszTmp; 2626 2627 memmove( lpszIter, lpszTmp, (strlenW(lpszTmp) + 1) * sizeof(WCHAR) ); 2628 } 2629 } 2630 2631 return ch; 2632 } 2633 2634 /************************************************************************* 2635 * @ [SHLWAPI.216] 2636 * 2637 * Convert an Ascii string to Unicode. 2638 * 2639 * PARAMS 2640 * dwCp [I] Code page for the conversion 2641 * lpSrcStr [I] Source Ascii string to convert 2642 * lpDstStr [O] Destination for converted Unicode string 2643 * iLen [I] Length of lpDstStr 2644 * 2645 * RETURNS 2646 * The return value of the MultiByteToWideChar() function called on lpSrcStr. 2647 */ 2648 DWORD WINAPI SHAnsiToUnicodeCP(DWORD dwCp, LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen) 2649 { 2650 DWORD dwRet; 2651 2652 dwRet = MultiByteToWideChar(dwCp, 0, lpSrcStr, -1, lpDstStr, iLen); 2653 TRACE("%s->%s,ret=%d\n", debugstr_a(lpSrcStr), debugstr_w(lpDstStr), dwRet); 2654 return dwRet; 2655 } 2656 2657 /************************************************************************* 2658 * @ [SHLWAPI.215] 2659 * 2660 * Convert an Ascii string to Unicode. 2661 * 2662 * PARAMS 2663 * lpSrcStr [I] Source Ascii string to convert 2664 * lpDstStr [O] Destination for converted Unicode string 2665 * iLen [I] Length of lpDstStr 2666 * 2667 * RETURNS 2668 * The return value of the MultiByteToWideChar() function called on lpSrcStr. 2669 * 2670 * NOTES 2671 * This function simply calls SHAnsiToUnicodeCP with code page CP_ACP. 2672 */ 2673 DWORD WINAPI SHAnsiToUnicode(LPCSTR lpSrcStr, LPWSTR lpDstStr, int iLen) 2674 { 2675 return SHAnsiToUnicodeCP(CP_ACP, lpSrcStr, lpDstStr, iLen); 2676 } 2677 2678 /************************************************************************* 2679 * @ [SHLWAPI.218] 2680 * 2681 * Convert a Unicode string to Ascii. 2682 * 2683 * PARAMS 2684 * CodePage [I] Code page to use for the conversion 2685 * lpSrcStr [I] Source Unicode string to convert 2686 * lpDstStr [O] Destination for converted Ascii string 2687 * dstlen [I] Length of buffer at lpDstStr 2688 * 2689 * RETURNS 2690 * Success: The length in bytes of the result at lpDstStr (including the terminator) 2691 * Failure: When using CP_UTF8, CP_UTF7 or 0xc350 as codePage, 0 is returned and 2692 * the result is not nul-terminated. 2693 * When using a different codepage, the length in bytes of the truncated 2694 * result at lpDstStr (including the terminator) is returned and 2695 * lpDstStr is always nul-terminated. 2696 * 2697 */ 2698 DWORD WINAPI SHUnicodeToAnsiCP(UINT CodePage, LPCWSTR lpSrcStr, LPSTR lpDstStr, int dstlen) 2699 { 2700 static const WCHAR emptyW[] = { '\0' }; 2701 int len , reqLen; 2702 LPSTR mem; 2703 2704 if (!lpDstStr || !dstlen) 2705 return 0; 2706 2707 if (!lpSrcStr) 2708 lpSrcStr = emptyW; 2709 2710 *lpDstStr = '\0'; 2711 2712 len = strlenW(lpSrcStr) + 1; 2713 2714 switch (CodePage) 2715 { 2716 case CP_WINUNICODE: 2717 CodePage = CP_UTF8; /* Fall through... */ 2718 case 0x0000C350: /* FIXME: CP_ #define */ 2719 case CP_UTF7: 2720 case CP_UTF8: 2721 { 2722 DWORD dwMode = 0; 2723 INT lenW = len - 1; 2724 INT needed = dstlen - 1; 2725 HRESULT hr; 2726 2727 /* try the user supplied buffer first */ 2728 hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &lenW, lpDstStr, &needed); 2729 if (hr == S_OK) 2730 { 2731 lpDstStr[needed] = '\0'; 2732 return needed + 1; 2733 } 2734 2735 /* user buffer too small. exclude termination and copy as much as possible */ 2736 lenW = len; 2737 hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &lenW, NULL, &needed); 2738 needed++; 2739 mem = HeapAlloc(GetProcessHeap(), 0, needed); 2740 if (!mem) 2741 return 0; 2742 2743 hr = ConvertINetUnicodeToMultiByte(&dwMode, CodePage, lpSrcStr, &len, mem, &needed); 2744 if (hr == S_OK) 2745 { 2746 reqLen = SHTruncateString(mem, dstlen); 2747 if (reqLen > 0) memcpy(lpDstStr, mem, reqLen-1); 2748 } 2749 HeapFree(GetProcessHeap(), 0, mem); 2750 return 0; 2751 } 2752 default: 2753 break; 2754 } 2755 2756 /* try the user supplied buffer first */ 2757 reqLen = WideCharToMultiByte(CodePage, 0, lpSrcStr, len, lpDstStr, dstlen, NULL, NULL); 2758 2759 if (!reqLen && GetLastError() == ERROR_INSUFFICIENT_BUFFER) 2760 { 2761 reqLen = WideCharToMultiByte(CodePage, 0, lpSrcStr, len, NULL, 0, NULL, NULL); 2762 if (reqLen) 2763 { 2764 mem = HeapAlloc(GetProcessHeap(), 0, reqLen); 2765 if (mem) 2766 { 2767 WideCharToMultiByte(CodePage, 0, lpSrcStr, len, mem, reqLen, NULL, NULL); 2768 2769 reqLen = SHTruncateString(mem, dstlen -1); 2770 reqLen++; 2771 2772 lstrcpynA(lpDstStr, mem, reqLen); 2773 HeapFree(GetProcessHeap(), 0, mem); 2774 lpDstStr[reqLen-1] = '\0'; 2775 } 2776 } 2777 } 2778 return reqLen; 2779 } 2780 2781 /************************************************************************* 2782 * @ [SHLWAPI.217] 2783 * 2784 * Convert a Unicode string to Ascii. 2785 * 2786 * PARAMS 2787 * lpSrcStr [I] Source Unicode string to convert 2788 * lpDstStr [O] Destination for converted Ascii string 2789 * iLen [O] Length of lpDstStr in characters 2790 * 2791 * RETURNS 2792 * See SHUnicodeToAnsiCP 2793 2794 * NOTES 2795 * This function simply calls SHUnicodeToAnsiCP() with CodePage = CP_ACP. 2796 */ 2797 INT WINAPI SHUnicodeToAnsi(LPCWSTR lpSrcStr, LPSTR lpDstStr, INT iLen) 2798 { 2799 return SHUnicodeToAnsiCP(CP_ACP, lpSrcStr, lpDstStr, iLen); 2800 } 2801 2802 /************************************************************************* 2803 * @ [SHLWAPI.345] 2804 * 2805 * Copy one string to another. 2806 * 2807 * PARAMS 2808 * lpszSrc [I] Source string to copy 2809 * lpszDst [O] Destination for copy 2810 * iLen [I] Length of lpszDst in characters 2811 * 2812 * RETURNS 2813 * The length of the copied string, including the terminating NUL. lpszDst 2814 * contains iLen characters of lpszSrc. 2815 */ 2816 DWORD WINAPI SHAnsiToAnsi(LPCSTR lpszSrc, LPSTR lpszDst, int iLen) 2817 { 2818 LPSTR lpszRet; 2819 2820 TRACE("(%s,%p,0x%08x)\n", debugstr_a(lpszSrc), lpszDst, iLen); 2821 2822 lpszRet = StrCpyNXA(lpszDst, lpszSrc, iLen); 2823 return lpszRet - lpszDst + 1; 2824 } 2825 2826 /************************************************************************* 2827 * @ [SHLWAPI.346] 2828 * 2829 * Unicode version of SSHAnsiToAnsi. 2830 */ 2831 DWORD WINAPI SHUnicodeToUnicode(LPCWSTR lpszSrc, LPWSTR lpszDst, int iLen) 2832 { 2833 LPWSTR lpszRet; 2834 2835 TRACE("(%s,%p,0x%08x)\n", debugstr_w(lpszSrc), lpszDst, iLen); 2836 2837 lpszRet = StrCpyNXW(lpszDst, lpszSrc, iLen); 2838 return lpszRet - lpszDst + 1; 2839 } 2840 2841 /************************************************************************* 2842 * @ [SHLWAPI.364] 2843 * 2844 * Determine if an Ascii string converts to Unicode and back identically. 2845 * 2846 * PARAMS 2847 * lpSrcStr [I] Source Unicode string to convert 2848 * lpDst [O] Destination for resulting Ascii string 2849 * iLen [I] Length of lpDst in characters 2850 * 2851 * RETURNS 2852 * TRUE, since Ascii strings always convert identically. 2853 */ 2854 BOOL WINAPI DoesStringRoundTripA(LPCSTR lpSrcStr, LPSTR lpDst, INT iLen) 2855 { 2856 lstrcpynA(lpDst, lpSrcStr, iLen); 2857 return TRUE; 2858 } 2859 2860 /************************************************************************* 2861 * @ [SHLWAPI.365] 2862 * 2863 * Determine if a Unicode string converts to Ascii and back identically. 2864 * 2865 * PARAMS 2866 * lpSrcStr [I] Source Unicode string to convert 2867 * lpDst [O] Destination for resulting Ascii string 2868 * iLen [I] Length of lpDst in characters 2869 * 2870 * RETURNS 2871 * TRUE, if lpSrcStr converts to Ascii and back identically, 2872 * FALSE otherwise. 2873 */ 2874 BOOL WINAPI DoesStringRoundTripW(LPCWSTR lpSrcStr, LPSTR lpDst, INT iLen) 2875 { 2876 WCHAR szBuff[MAX_PATH]; 2877 2878 SHUnicodeToAnsi(lpSrcStr, lpDst, iLen); 2879 SHAnsiToUnicode(lpDst, szBuff, MAX_PATH); 2880 return !strcmpW(lpSrcStr, szBuff); 2881 } 2882 2883 /************************************************************************* 2884 * SHLoadIndirectString [SHLWAPI.@] 2885 * 2886 * If passed a string that begins with '@', extract the string from the 2887 * appropriate resource, otherwise do a straight copy. 2888 * 2889 */ 2890 HRESULT WINAPI SHLoadIndirectString(LPCWSTR src, LPWSTR dst, UINT dst_len, void **reserved) 2891 { 2892 WCHAR *dllname = NULL; 2893 HMODULE hmod = NULL; 2894 HRESULT hr = E_FAIL; 2895 #ifdef __REACTOS__ 2896 WCHAR szExpanded[512]; 2897 #endif 2898 2899 TRACE("(%s %p %08x %p)\n", debugstr_w(src), dst, dst_len, reserved); 2900 2901 if(src[0] == '@') 2902 { 2903 WCHAR *index_str; 2904 int index; 2905 2906 #ifdef __REACTOS__ 2907 if (wcschr(src, '%') != NULL) 2908 { 2909 ExpandEnvironmentStringsW(src, szExpanded, ARRAY_SIZE(szExpanded)); 2910 src = szExpanded; 2911 } 2912 #endif 2913 dst[0] = 0; 2914 dllname = StrDupW(src + 1); 2915 index_str = strchrW(dllname, ','); 2916 2917 if(!index_str) goto end; 2918 2919 *index_str = 0; 2920 index_str++; 2921 index = atoiW(index_str); 2922 2923 #ifdef __REACTOS__ 2924 hmod = LoadLibraryExW(dllname, NULL, LOAD_LIBRARY_AS_DATAFILE); 2925 #else 2926 hmod = LoadLibraryW(dllname); 2927 #endif 2928 if(!hmod) goto end; 2929 2930 if(index < 0) 2931 { 2932 if(LoadStringW(hmod, -index, dst, dst_len)) 2933 hr = S_OK; 2934 } 2935 else 2936 FIXME("can't handle non-negative indices (%d)\n", index); 2937 } 2938 else 2939 { 2940 if(dst != src) 2941 lstrcpynW(dst, src, dst_len); 2942 hr = S_OK; 2943 } 2944 2945 TRACE("returning %s\n", debugstr_w(dst)); 2946 end: 2947 if(hmod) FreeLibrary(hmod); 2948 LocalFree(dllname); 2949 return hr; 2950 } 2951 2952 BOOL WINAPI IsCharSpaceA(CHAR c) 2953 { 2954 WORD CharType; 2955 return GetStringTypeA(GetSystemDefaultLCID(), CT_CTYPE1, &c, 1, &CharType) && (CharType & C1_SPACE); 2956 } 2957 2958 /************************************************************************* 2959 * @ [SHLWAPI.29] 2960 * 2961 * Determine if a Unicode character is a space. 2962 * 2963 * PARAMS 2964 * wc [I] Character to check. 2965 * 2966 * RETURNS 2967 * TRUE, if wc is a space, 2968 * FALSE otherwise. 2969 */ 2970 BOOL WINAPI IsCharSpaceW(WCHAR wc) 2971 { 2972 WORD CharType; 2973 2974 return GetStringTypeW(CT_CTYPE1, &wc, 1, &CharType) && (CharType & C1_SPACE); 2975 } 2976