1 /* 2 * Path Functions 3 * 4 * Copyright 1999, 2000 Juergen Schmied 5 * Copyright 2001, 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 <stdarg.h> 26 #include <string.h> 27 #include <stdlib.h> 28 29 #include "wine/unicode.h" 30 #include "windef.h" 31 #include "winbase.h" 32 #include "wingdi.h" 33 #include "winuser.h" 34 #include "winreg.h" 35 #include "winternl.h" 36 #define NO_SHLWAPI_STREAM 37 #include "shlwapi.h" 38 #include "wine/debug.h" 39 40 WINE_DEFAULT_DEBUG_CHANNEL(shell); 41 42 #ifdef __REACTOS__ 43 44 #include <shlobj.h> 45 #include <shlwapi_undoc.h> 46 47 int WINAPI IsNetDrive(int drive); 48 49 #else 50 51 /* Get a function pointer from a DLL handle */ 52 #define GET_FUNC(func, module, name, fail) \ 53 do { \ 54 if (!func) { \ 55 if (!SHLWAPI_h##module && !(SHLWAPI_h##module = LoadLibraryA(#module ".dll"))) return fail; \ 56 func = (fn##func)GetProcAddress(SHLWAPI_h##module, name); \ 57 if (!func) return fail; \ 58 } \ 59 } while (0) 60 61 /* DLL handles for late bound calls */ 62 static HMODULE SHLWAPI_hshell32; 63 64 /* Function pointers for GET_FUNC macro; these need to be global because of gcc bug */ 65 typedef BOOL (WINAPI *fnpIsNetDrive)(int); 66 static fnpIsNetDrive pIsNetDrive; 67 68 #endif /* __REACTOS__ */ 69 70 71 HRESULT WINAPI SHGetWebFolderFilePathW(LPCWSTR,LPWSTR,DWORD); 72 73 static inline WCHAR* heap_strdupAtoW(LPCSTR str) 74 { 75 WCHAR *ret = NULL; 76 77 if (str) 78 { 79 DWORD len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0); 80 ret = HeapAlloc(GetProcessHeap(), 0, len*sizeof(WCHAR)); 81 if (ret) 82 MultiByteToWideChar(CP_ACP, 0, str, -1, ret, len); 83 } 84 85 return ret; 86 } 87 88 /************************************************************************* 89 * PathAppendA [SHLWAPI.@] 90 * 91 * Append one path to another. 92 * 93 * PARAMS 94 * lpszPath [I/O] Initial part of path, and destination for output 95 * lpszAppend [I] Path to append 96 * 97 * RETURNS 98 * Success: TRUE. lpszPath contains the newly created path. 99 * Failure: FALSE, if either path is NULL, or PathCombineA() fails. 100 * 101 * NOTES 102 * lpszAppend must contain at least one backslash ('\') if not NULL. 103 * Because PathCombineA() is used to join the paths, the resulting 104 * path is also canonicalized. 105 */ 106 BOOL WINAPI PathAppendA (LPSTR lpszPath, LPCSTR lpszAppend) 107 { 108 TRACE("(%s,%s)\n",debugstr_a(lpszPath), debugstr_a(lpszAppend)); 109 110 if (lpszPath && lpszAppend) 111 { 112 if (!PathIsUNCA(lpszAppend)) 113 while (*lpszAppend == '\\') 114 lpszAppend++; 115 if (PathCombineA(lpszPath, lpszPath, lpszAppend)) 116 return TRUE; 117 } 118 return FALSE; 119 } 120 121 /************************************************************************* 122 * PathAppendW [SHLWAPI.@] 123 * 124 * See PathAppendA. 125 */ 126 BOOL WINAPI PathAppendW(LPWSTR lpszPath, LPCWSTR lpszAppend) 127 { 128 TRACE("(%s,%s)\n",debugstr_w(lpszPath), debugstr_w(lpszAppend)); 129 130 if (lpszPath && lpszAppend) 131 { 132 if (!PathIsUNCW(lpszAppend)) 133 while (*lpszAppend == '\\') 134 lpszAppend++; 135 if (PathCombineW(lpszPath, lpszPath, lpszAppend)) 136 return TRUE; 137 } 138 return FALSE; 139 } 140 141 /************************************************************************* 142 * PathCombineA [SHLWAPI.@] 143 * 144 * Combine two paths together. 145 * 146 * PARAMS 147 * lpszDest [O] Destination for combined path 148 * lpszDir [I] Directory path 149 * lpszFile [I] File path 150 * 151 * RETURNS 152 * Success: The output path 153 * Failure: NULL, if inputs are invalid. 154 * 155 * NOTES 156 * lpszDest should be at least MAX_PATH in size, and may point to the same 157 * memory location as lpszDir. The combined path is canonicalised. 158 */ 159 LPSTR WINAPI PathCombineA(LPSTR lpszDest, LPCSTR lpszDir, LPCSTR lpszFile) 160 { 161 WCHAR szDest[MAX_PATH]; 162 WCHAR szDir[MAX_PATH]; 163 WCHAR szFile[MAX_PATH]; 164 TRACE("(%p,%s,%s)\n", lpszDest, debugstr_a(lpszDir), debugstr_a(lpszFile)); 165 166 /* Invalid parameters */ 167 if (!lpszDest) 168 return NULL; 169 if (!lpszDir && !lpszFile) 170 goto fail; 171 172 if (lpszDir) 173 if (!MultiByteToWideChar(CP_ACP,0,lpszDir,-1,szDir,MAX_PATH)) 174 goto fail; 175 176 if (lpszFile) 177 if (!MultiByteToWideChar(CP_ACP,0,lpszFile,-1,szFile,MAX_PATH)) 178 goto fail; 179 180 if (PathCombineW(szDest, lpszDir ? szDir : NULL, lpszFile ? szFile : NULL)) 181 if (WideCharToMultiByte(CP_ACP,0,szDest,-1,lpszDest,MAX_PATH,0,0)) 182 return lpszDest; 183 184 fail: 185 lpszDest[0] = 0; 186 return NULL; 187 } 188 189 /************************************************************************* 190 * PathCombineW [SHLWAPI.@] 191 * 192 * See PathCombineA. 193 */ 194 LPWSTR WINAPI PathCombineW(LPWSTR lpszDest, LPCWSTR lpszDir, LPCWSTR lpszFile) 195 { 196 WCHAR szTemp[MAX_PATH]; 197 BOOL bUseBoth = FALSE, bStrip = FALSE; 198 199 TRACE("(%p,%s,%s)\n", lpszDest, debugstr_w(lpszDir), debugstr_w(lpszFile)); 200 201 /* Invalid parameters */ 202 if (!lpszDest) 203 return NULL; 204 if (!lpszDir && !lpszFile) 205 { 206 lpszDest[0] = 0; 207 return NULL; 208 } 209 210 if ((!lpszFile || !*lpszFile) && lpszDir) 211 { 212 /* Use dir only */ 213 lstrcpynW(szTemp, lpszDir, MAX_PATH); 214 } 215 else if (!lpszDir || !*lpszDir || !PathIsRelativeW(lpszFile)) 216 { 217 if (!lpszDir || !*lpszDir || *lpszFile != '\\' || PathIsUNCW(lpszFile)) 218 { 219 /* Use file only */ 220 lstrcpynW(szTemp, lpszFile, MAX_PATH); 221 } 222 else 223 { 224 bUseBoth = TRUE; 225 bStrip = TRUE; 226 } 227 } 228 else 229 bUseBoth = TRUE; 230 231 if (bUseBoth) 232 { 233 lstrcpynW(szTemp, lpszDir, MAX_PATH); 234 if (bStrip) 235 { 236 PathStripToRootW(szTemp); 237 lpszFile++; /* Skip '\' */ 238 } 239 if (!PathAddBackslashW(szTemp) || strlenW(szTemp) + strlenW(lpszFile) >= MAX_PATH) 240 { 241 lpszDest[0] = 0; 242 return NULL; 243 } 244 strcatW(szTemp, lpszFile); 245 } 246 247 PathCanonicalizeW(lpszDest, szTemp); 248 return lpszDest; 249 } 250 251 /************************************************************************* 252 * PathAddBackslashA [SHLWAPI.@] 253 * 254 * Append a backslash ('\') to a path if one doesn't exist. 255 * 256 * PARAMS 257 * lpszPath [I/O] The path to append a backslash to. 258 * 259 * RETURNS 260 * Success: The position of the last backslash in the path. 261 * Failure: NULL, if lpszPath is NULL or the path is too large. 262 */ 263 LPSTR WINAPI PathAddBackslashA(LPSTR lpszPath) 264 { 265 size_t iLen; 266 LPSTR prev = lpszPath; 267 268 TRACE("(%s)\n",debugstr_a(lpszPath)); 269 270 if (!lpszPath || (iLen = strlen(lpszPath)) >= MAX_PATH) 271 return NULL; 272 273 if (iLen) 274 { 275 do { 276 lpszPath = CharNextA(prev); 277 if (*lpszPath) 278 prev = lpszPath; 279 } while (*lpszPath); 280 if (*prev != '\\') 281 { 282 *lpszPath++ = '\\'; 283 *lpszPath = '\0'; 284 } 285 } 286 return lpszPath; 287 } 288 289 /************************************************************************* 290 * PathAddBackslashW [SHLWAPI.@] 291 * 292 * See PathAddBackslashA. 293 */ 294 LPWSTR WINAPI PathAddBackslashW( LPWSTR lpszPath ) 295 { 296 size_t iLen; 297 298 TRACE("(%s)\n",debugstr_w(lpszPath)); 299 300 if (!lpszPath || (iLen = strlenW(lpszPath)) >= MAX_PATH) 301 return NULL; 302 303 if (iLen) 304 { 305 lpszPath += iLen; 306 if (lpszPath[-1] != '\\') 307 { 308 *lpszPath++ = '\\'; 309 *lpszPath = '\0'; 310 } 311 } 312 return lpszPath; 313 } 314 315 /************************************************************************* 316 * PathBuildRootA [SHLWAPI.@] 317 * 318 * Create a root drive string (e.g. "A:\") from a drive number. 319 * 320 * PARAMS 321 * lpszPath [O] Destination for the drive string 322 * 323 * RETURNS 324 * lpszPath 325 * 326 * NOTES 327 * If lpszPath is NULL or drive is invalid, nothing is written to lpszPath. 328 */ 329 LPSTR WINAPI PathBuildRootA(LPSTR lpszPath, int drive) 330 { 331 TRACE("(%p,%d)\n", lpszPath, drive); 332 333 if (lpszPath && drive >= 0 && drive < 26) 334 { 335 lpszPath[0] = 'A' + drive; 336 lpszPath[1] = ':'; 337 lpszPath[2] = '\\'; 338 lpszPath[3] = '\0'; 339 } 340 return lpszPath; 341 } 342 343 /************************************************************************* 344 * PathBuildRootW [SHLWAPI.@] 345 * 346 * See PathBuildRootA. 347 */ 348 LPWSTR WINAPI PathBuildRootW(LPWSTR lpszPath, int drive) 349 { 350 TRACE("(%p,%d)\n", lpszPath, drive); 351 352 if (lpszPath && drive >= 0 && drive < 26) 353 { 354 lpszPath[0] = 'A' + drive; 355 lpszPath[1] = ':'; 356 lpszPath[2] = '\\'; 357 lpszPath[3] = '\0'; 358 } 359 return lpszPath; 360 } 361 362 /************************************************************************* 363 * PathFindFileNameA [SHLWAPI.@] 364 * 365 * Locate the start of the file name in a path 366 * 367 * PARAMS 368 * lpszPath [I] Path to search 369 * 370 * RETURNS 371 * A pointer to the first character of the file name 372 */ 373 LPSTR WINAPI PathFindFileNameA(LPCSTR lpszPath) 374 { 375 LPCSTR lastSlash = lpszPath; 376 377 TRACE("(%s)\n",debugstr_a(lpszPath)); 378 379 while (lpszPath && *lpszPath) 380 { 381 if ((*lpszPath == '\\' || *lpszPath == '/' || *lpszPath == ':') && 382 lpszPath[1] && lpszPath[1] != '\\' && lpszPath[1] != '/') 383 lastSlash = lpszPath + 1; 384 lpszPath = CharNextA(lpszPath); 385 } 386 return (LPSTR)lastSlash; 387 } 388 389 /************************************************************************* 390 * PathFindFileNameW [SHLWAPI.@] 391 * 392 * See PathFindFileNameA. 393 */ 394 LPWSTR WINAPI PathFindFileNameW(LPCWSTR lpszPath) 395 { 396 LPCWSTR lastSlash = lpszPath; 397 398 TRACE("(%s)\n",debugstr_w(lpszPath)); 399 400 while (lpszPath && *lpszPath) 401 { 402 if ((*lpszPath == '\\' || *lpszPath == '/' || *lpszPath == ':') && 403 lpszPath[1] && lpszPath[1] != '\\' && lpszPath[1] != '/') 404 lastSlash = lpszPath + 1; 405 lpszPath++; 406 } 407 return (LPWSTR)lastSlash; 408 } 409 410 /************************************************************************* 411 * PathFindExtensionA [SHLWAPI.@] 412 * 413 * Locate the start of the file extension in a path 414 * 415 * PARAMS 416 * lpszPath [I] The path to search 417 * 418 * RETURNS 419 * A pointer to the first character of the extension, the end of 420 * the string if the path has no extension, or NULL If lpszPath is NULL 421 */ 422 LPSTR WINAPI PathFindExtensionA( LPCSTR lpszPath ) 423 { 424 LPCSTR lastpoint = NULL; 425 426 TRACE("(%s)\n", debugstr_a(lpszPath)); 427 428 if (lpszPath) 429 { 430 while (*lpszPath) 431 { 432 if (*lpszPath == '\\' || *lpszPath==' ') 433 lastpoint = NULL; 434 else if (*lpszPath == '.') 435 lastpoint = lpszPath; 436 lpszPath = CharNextA(lpszPath); 437 } 438 } 439 return (LPSTR)(lastpoint ? lastpoint : lpszPath); 440 } 441 442 /************************************************************************* 443 * PathFindExtensionW [SHLWAPI.@] 444 * 445 * See PathFindExtensionA. 446 */ 447 LPWSTR WINAPI PathFindExtensionW( LPCWSTR lpszPath ) 448 { 449 LPCWSTR lastpoint = NULL; 450 451 TRACE("(%s)\n", debugstr_w(lpszPath)); 452 453 if (lpszPath) 454 { 455 while (*lpszPath) 456 { 457 if (*lpszPath == '\\' || *lpszPath==' ') 458 lastpoint = NULL; 459 else if (*lpszPath == '.') 460 lastpoint = lpszPath; 461 lpszPath++; 462 } 463 } 464 return (LPWSTR)(lastpoint ? lastpoint : lpszPath); 465 } 466 467 /************************************************************************* 468 * PathGetArgsA [SHLWAPI.@] 469 * 470 * Find the next argument in a string delimited by spaces. 471 * 472 * PARAMS 473 * lpszPath [I] The string to search for arguments in 474 * 475 * RETURNS 476 * The start of the next argument in lpszPath, or NULL if lpszPath is NULL 477 * 478 * NOTES 479 * Spaces in quoted strings are ignored as delimiters. 480 */ 481 LPSTR WINAPI PathGetArgsA(LPCSTR lpszPath) 482 { 483 BOOL bSeenQuote = FALSE; 484 485 TRACE("(%s)\n",debugstr_a(lpszPath)); 486 487 if (lpszPath) 488 { 489 while (*lpszPath) 490 { 491 if ((*lpszPath==' ') && !bSeenQuote) 492 return (LPSTR)lpszPath + 1; 493 if (*lpszPath == '"') 494 bSeenQuote = !bSeenQuote; 495 lpszPath = CharNextA(lpszPath); 496 } 497 } 498 return (LPSTR)lpszPath; 499 } 500 501 /************************************************************************* 502 * PathGetArgsW [SHLWAPI.@] 503 * 504 * See PathGetArgsA. 505 */ 506 LPWSTR WINAPI PathGetArgsW(LPCWSTR lpszPath) 507 { 508 BOOL bSeenQuote = FALSE; 509 510 TRACE("(%s)\n",debugstr_w(lpszPath)); 511 512 if (lpszPath) 513 { 514 while (*lpszPath) 515 { 516 if ((*lpszPath==' ') && !bSeenQuote) 517 return (LPWSTR)lpszPath + 1; 518 if (*lpszPath == '"') 519 bSeenQuote = !bSeenQuote; 520 lpszPath++; 521 } 522 } 523 return (LPWSTR)lpszPath; 524 } 525 526 /************************************************************************* 527 * PathGetDriveNumberA [SHLWAPI.@] 528 * 529 * Return the drive number from a path 530 * 531 * PARAMS 532 * lpszPath [I] Path to get the drive number from 533 * 534 * RETURNS 535 * Success: The drive number corresponding to the drive in the path 536 * Failure: -1, if lpszPath contains no valid drive 537 */ 538 int WINAPI PathGetDriveNumberA(LPCSTR lpszPath) 539 { 540 TRACE ("(%s)\n",debugstr_a(lpszPath)); 541 542 if (lpszPath && !IsDBCSLeadByte(*lpszPath) && lpszPath[1] == ':' && 543 tolower(*lpszPath) >= 'a' && tolower(*lpszPath) <= 'z') 544 return tolower(*lpszPath) - 'a'; 545 return -1; 546 } 547 548 /************************************************************************* 549 * PathGetDriveNumberW [SHLWAPI.@] 550 * 551 * See PathGetDriveNumberA. 552 */ 553 int WINAPI PathGetDriveNumberW(const WCHAR *path) 554 { 555 WCHAR drive; 556 557 static const WCHAR nt_prefixW[] = {'\\','\\','?','\\'}; 558 559 TRACE("(%s)\n", debugstr_w(path)); 560 561 if (!path) 562 return -1; 563 564 if (!strncmpW(path, nt_prefixW, 4)) 565 path += 4; 566 567 drive = tolowerW(path[0]); 568 if (drive < 'a' || drive > 'z' || path[1] != ':') 569 return -1; 570 571 return drive - 'a'; 572 } 573 574 /************************************************************************* 575 * PathRemoveFileSpecA [SHLWAPI.@] 576 * 577 * Remove the file specification from a path. 578 * 579 * PARAMS 580 * lpszPath [I/O] Path to remove the file spec from 581 * 582 * RETURNS 583 * TRUE If the path was valid and modified 584 * FALSE Otherwise 585 */ 586 BOOL WINAPI PathRemoveFileSpecA(LPSTR lpszPath) 587 { 588 LPSTR lpszFileSpec = lpszPath; 589 BOOL bModified = FALSE; 590 591 TRACE("(%s)\n",debugstr_a(lpszPath)); 592 593 if(lpszPath) 594 { 595 /* Skip directory or UNC path */ 596 if (*lpszPath == '\\') 597 lpszFileSpec = ++lpszPath; 598 if (*lpszPath == '\\') 599 lpszFileSpec = ++lpszPath; 600 601 while (*lpszPath) 602 { 603 if(*lpszPath == '\\') 604 lpszFileSpec = lpszPath; /* Skip dir */ 605 else if(*lpszPath == ':') 606 { 607 lpszFileSpec = ++lpszPath; /* Skip drive */ 608 if (*lpszPath == '\\') 609 lpszFileSpec++; 610 } 611 if (!(lpszPath = CharNextA(lpszPath))) 612 break; 613 } 614 615 if (*lpszFileSpec) 616 { 617 *lpszFileSpec = '\0'; 618 bModified = TRUE; 619 } 620 } 621 return bModified; 622 } 623 624 /************************************************************************* 625 * PathRemoveFileSpecW [SHLWAPI.@] 626 * 627 * See PathRemoveFileSpecA. 628 */ 629 BOOL WINAPI PathRemoveFileSpecW(LPWSTR lpszPath) 630 { 631 LPWSTR lpszFileSpec = lpszPath; 632 BOOL bModified = FALSE; 633 634 TRACE("(%s)\n",debugstr_w(lpszPath)); 635 636 if(lpszPath) 637 { 638 /* Skip directory or UNC path */ 639 if (*lpszPath == '\\') 640 lpszFileSpec = ++lpszPath; 641 if (*lpszPath == '\\') 642 lpszFileSpec = ++lpszPath; 643 644 while (*lpszPath) 645 { 646 if(*lpszPath == '\\') 647 lpszFileSpec = lpszPath; /* Skip dir */ 648 else if(*lpszPath == ':') 649 { 650 lpszFileSpec = ++lpszPath; /* Skip drive */ 651 if (*lpszPath == '\\') 652 lpszFileSpec++; 653 } 654 lpszPath++; 655 } 656 657 if (*lpszFileSpec) 658 { 659 *lpszFileSpec = '\0'; 660 bModified = TRUE; 661 } 662 } 663 return bModified; 664 } 665 666 /************************************************************************* 667 * PathStripPathA [SHLWAPI.@] 668 * 669 * Remove the initial path from the beginning of a filename 670 * 671 * PARAMS 672 * lpszPath [I/O] Path to remove the initial path from 673 * 674 * RETURNS 675 * Nothing. 676 */ 677 void WINAPI PathStripPathA(LPSTR lpszPath) 678 { 679 TRACE("(%s)\n", debugstr_a(lpszPath)); 680 681 if (lpszPath) 682 { 683 LPSTR lpszFileName = PathFindFileNameA(lpszPath); 684 if(lpszFileName != lpszPath) 685 RtlMoveMemory(lpszPath, lpszFileName, strlen(lpszFileName)+1); 686 } 687 } 688 689 /************************************************************************* 690 * PathStripPathW [SHLWAPI.@] 691 * 692 * See PathStripPathA. 693 */ 694 void WINAPI PathStripPathW(LPWSTR lpszPath) 695 { 696 LPWSTR lpszFileName; 697 698 TRACE("(%s)\n", debugstr_w(lpszPath)); 699 lpszFileName = PathFindFileNameW(lpszPath); 700 if(lpszFileName != lpszPath) 701 RtlMoveMemory(lpszPath, lpszFileName, (strlenW(lpszFileName)+1)*sizeof(WCHAR)); 702 } 703 704 /************************************************************************* 705 * PathStripToRootA [SHLWAPI.@] 706 * 707 * Reduce a path to its root. 708 * 709 * PARAMS 710 * lpszPath [I/O] the path to reduce 711 * 712 * RETURNS 713 * Success: TRUE if the stripped path is a root path 714 * Failure: FALSE if the path cannot be stripped or is NULL 715 */ 716 BOOL WINAPI PathStripToRootA(LPSTR lpszPath) 717 { 718 TRACE("(%s)\n", debugstr_a(lpszPath)); 719 720 if (!lpszPath) 721 return FALSE; 722 while(!PathIsRootA(lpszPath)) 723 if (!PathRemoveFileSpecA(lpszPath)) 724 return FALSE; 725 return TRUE; 726 } 727 728 /************************************************************************* 729 * PathStripToRootW [SHLWAPI.@] 730 * 731 * See PathStripToRootA. 732 */ 733 BOOL WINAPI PathStripToRootW(LPWSTR lpszPath) 734 { 735 TRACE("(%s)\n", debugstr_w(lpszPath)); 736 737 if (!lpszPath) 738 return FALSE; 739 while(!PathIsRootW(lpszPath)) 740 if (!PathRemoveFileSpecW(lpszPath)) 741 return FALSE; 742 return TRUE; 743 } 744 745 /************************************************************************* 746 * PathRemoveArgsA [SHLWAPI.@] 747 * 748 * Strip space separated arguments from a path. 749 * 750 * PARAMS 751 * lpszPath [I/O] Path to remove arguments from 752 * 753 * RETURNS 754 * Nothing. 755 */ 756 void WINAPI PathRemoveArgsA(LPSTR lpszPath) 757 { 758 TRACE("(%s)\n",debugstr_a(lpszPath)); 759 760 if(lpszPath) 761 { 762 LPSTR lpszArgs = PathGetArgsA(lpszPath); 763 if (*lpszArgs) 764 lpszArgs[-1] = '\0'; 765 else 766 { 767 LPSTR lpszLastChar = CharPrevA(lpszPath, lpszArgs); 768 if(*lpszLastChar == ' ') 769 *lpszLastChar = '\0'; 770 } 771 } 772 } 773 774 /************************************************************************* 775 * PathRemoveArgsW [SHLWAPI.@] 776 * 777 * See PathRemoveArgsA. 778 */ 779 void WINAPI PathRemoveArgsW(LPWSTR lpszPath) 780 { 781 TRACE("(%s)\n",debugstr_w(lpszPath)); 782 783 if(lpszPath) 784 { 785 LPWSTR lpszArgs = PathGetArgsW(lpszPath); 786 if (*lpszArgs || (lpszArgs > lpszPath && lpszArgs[-1] == ' ')) 787 lpszArgs[-1] = '\0'; 788 } 789 } 790 791 /************************************************************************* 792 * PathRemoveExtensionA [SHLWAPI.@] 793 * 794 * Remove the file extension from a path 795 * 796 * PARAMS 797 * lpszPath [I/O] Path to remove the extension from 798 * 799 * NOTES 800 * The NUL terminator must be written only if extension exists 801 * and if the pointed character is not already NUL. 802 * 803 * RETURNS 804 * Nothing. 805 */ 806 void WINAPI PathRemoveExtensionA(LPSTR lpszPath) 807 { 808 TRACE("(%s)\n", debugstr_a(lpszPath)); 809 810 if (lpszPath) 811 { 812 lpszPath = PathFindExtensionA(lpszPath); 813 if (lpszPath && *lpszPath != '\0') 814 *lpszPath = '\0'; 815 } 816 } 817 818 /************************************************************************* 819 * PathRemoveExtensionW [SHLWAPI.@] 820 * 821 * See PathRemoveExtensionA. 822 */ 823 void WINAPI PathRemoveExtensionW(LPWSTR lpszPath) 824 { 825 TRACE("(%s)\n", debugstr_w(lpszPath)); 826 827 if (lpszPath) 828 { 829 lpszPath = PathFindExtensionW(lpszPath); 830 if (lpszPath && *lpszPath != '\0') 831 *lpszPath = '\0'; 832 } 833 } 834 835 /************************************************************************* 836 * PathRemoveBackslashA [SHLWAPI.@] 837 * 838 * Remove a trailing backslash from a path. 839 * 840 * PARAMS 841 * lpszPath [I/O] Path to remove backslash from 842 * 843 * RETURNS 844 * Success: A pointer to the end of the path 845 * Failure: NULL, if lpszPath is NULL 846 */ 847 LPSTR WINAPI PathRemoveBackslashA( LPSTR lpszPath ) 848 { 849 LPSTR szTemp = NULL; 850 851 TRACE("(%s)\n", debugstr_a(lpszPath)); 852 853 if(lpszPath) 854 { 855 szTemp = CharPrevA(lpszPath, lpszPath + strlen(lpszPath)); 856 if (!PathIsRootA(lpszPath) && *szTemp == '\\') 857 *szTemp = '\0'; 858 } 859 return szTemp; 860 } 861 862 /************************************************************************* 863 * PathRemoveBackslashW [SHLWAPI.@] 864 * 865 * See PathRemoveBackslashA. 866 */ 867 LPWSTR WINAPI PathRemoveBackslashW( LPWSTR lpszPath ) 868 { 869 LPWSTR szTemp = NULL; 870 871 TRACE("(%s)\n", debugstr_w(lpszPath)); 872 873 if(lpszPath) 874 { 875 szTemp = lpszPath + strlenW(lpszPath); 876 if (szTemp > lpszPath) szTemp--; 877 if (!PathIsRootW(lpszPath) && *szTemp == '\\') 878 *szTemp = '\0'; 879 } 880 return szTemp; 881 } 882 883 /************************************************************************* 884 * PathRemoveBlanksA [SHLWAPI.@] 885 * 886 * Remove Spaces from the start and end of a path. 887 * 888 * PARAMS 889 * lpszPath [I/O] Path to strip blanks from 890 * 891 * RETURNS 892 * Nothing. 893 */ 894 VOID WINAPI PathRemoveBlanksA(LPSTR lpszPath) 895 { 896 TRACE("(%s)\n", debugstr_a(lpszPath)); 897 898 if(lpszPath && *lpszPath) 899 { 900 LPSTR start = lpszPath; 901 902 while (*lpszPath == ' ') 903 lpszPath = CharNextA(lpszPath); 904 905 while(*lpszPath) 906 *start++ = *lpszPath++; 907 908 if (start != lpszPath) 909 while (start[-1] == ' ') 910 start--; 911 *start = '\0'; 912 } 913 } 914 915 /************************************************************************* 916 * PathRemoveBlanksW [SHLWAPI.@] 917 * 918 * See PathRemoveBlanksA. 919 */ 920 VOID WINAPI PathRemoveBlanksW(LPWSTR lpszPath) 921 { 922 TRACE("(%s)\n", debugstr_w(lpszPath)); 923 924 if(lpszPath && *lpszPath) 925 { 926 LPWSTR start = lpszPath; 927 928 while (*lpszPath == ' ') 929 lpszPath++; 930 931 while(*lpszPath) 932 *start++ = *lpszPath++; 933 934 if (start != lpszPath) 935 while (start[-1] == ' ') 936 start--; 937 *start = '\0'; 938 } 939 } 940 941 /************************************************************************* 942 * PathQuoteSpacesA [SHLWAPI.@] 943 * 944 * Surround a path containing spaces in quotes. 945 * 946 * PARAMS 947 * lpszPath [I/O] Path to quote 948 * 949 * RETURNS 950 * Nothing. 951 * 952 * NOTES 953 * The path is not changed if it is invalid or has no spaces. 954 */ 955 VOID WINAPI PathQuoteSpacesA(LPSTR lpszPath) 956 { 957 TRACE("(%s)\n", debugstr_a(lpszPath)); 958 959 if(lpszPath && StrChrA(lpszPath,' ')) 960 { 961 size_t iLen = strlen(lpszPath) + 1; 962 963 if (iLen + 2 < MAX_PATH) 964 { 965 memmove(lpszPath + 1, lpszPath, iLen); 966 lpszPath[0] = '"'; 967 lpszPath[iLen] = '"'; 968 lpszPath[iLen + 1] = '\0'; 969 } 970 } 971 } 972 973 /************************************************************************* 974 * PathQuoteSpacesW [SHLWAPI.@] 975 * 976 * See PathQuoteSpacesA. 977 */ 978 VOID WINAPI PathQuoteSpacesW(LPWSTR lpszPath) 979 { 980 TRACE("(%s)\n", debugstr_w(lpszPath)); 981 982 if(lpszPath && StrChrW(lpszPath,' ')) 983 { 984 int iLen = strlenW(lpszPath) + 1; 985 986 if (iLen + 2 < MAX_PATH) 987 { 988 memmove(lpszPath + 1, lpszPath, iLen * sizeof(WCHAR)); 989 lpszPath[0] = '"'; 990 lpszPath[iLen] = '"'; 991 lpszPath[iLen + 1] = '\0'; 992 } 993 } 994 } 995 996 /************************************************************************* 997 * PathUnquoteSpacesA [SHLWAPI.@] 998 * 999 * Remove quotes ("") from around a path, if present. 1000 * 1001 * PARAMS 1002 * lpszPath [I/O] Path to strip quotes from 1003 * 1004 * RETURNS 1005 * Nothing 1006 * 1007 * NOTES 1008 * If the path contains a single quote only, an empty string will result. 1009 * Otherwise quotes are only removed if they appear at the start and end 1010 * of the path. 1011 */ 1012 VOID WINAPI PathUnquoteSpacesA(LPSTR lpszPath) 1013 { 1014 TRACE("(%s)\n", debugstr_a(lpszPath)); 1015 1016 if (lpszPath && *lpszPath == '"') 1017 { 1018 DWORD dwLen = strlen(lpszPath) - 1; 1019 1020 if (lpszPath[dwLen] == '"') 1021 { 1022 lpszPath[dwLen] = '\0'; 1023 for (; *lpszPath; lpszPath++) 1024 *lpszPath = lpszPath[1]; 1025 } 1026 } 1027 } 1028 1029 /************************************************************************* 1030 * PathUnquoteSpacesW [SHLWAPI.@] 1031 * 1032 * See PathUnquoteSpacesA. 1033 */ 1034 VOID WINAPI PathUnquoteSpacesW(LPWSTR lpszPath) 1035 { 1036 TRACE("(%s)\n", debugstr_w(lpszPath)); 1037 1038 if (lpszPath && *lpszPath == '"') 1039 { 1040 DWORD dwLen = strlenW(lpszPath) - 1; 1041 1042 if (lpszPath[dwLen] == '"') 1043 { 1044 lpszPath[dwLen] = '\0'; 1045 for (; *lpszPath; lpszPath++) 1046 *lpszPath = lpszPath[1]; 1047 } 1048 } 1049 } 1050 1051 /************************************************************************* 1052 * PathParseIconLocationA [SHLWAPI.@] 1053 * 1054 * Parse the location of an icon from a path. 1055 * 1056 * PARAMS 1057 * lpszPath [I/O] The path to parse the icon location from. 1058 * 1059 * RETURNS 1060 * Success: The number of the icon 1061 * Failure: 0 if the path does not contain an icon location or is NULL 1062 * 1063 * NOTES 1064 * The path has surrounding quotes and spaces removed regardless 1065 * of whether the call succeeds or not. 1066 */ 1067 int WINAPI PathParseIconLocationA(LPSTR lpszPath) 1068 { 1069 int iRet = 0; 1070 LPSTR lpszComma; 1071 1072 TRACE("(%s)\n", debugstr_a(lpszPath)); 1073 1074 if (lpszPath) 1075 { 1076 if ((lpszComma = strchr(lpszPath, ','))) 1077 { 1078 *lpszComma++ = '\0'; 1079 iRet = StrToIntA(lpszComma); 1080 } 1081 PathUnquoteSpacesA(lpszPath); 1082 PathRemoveBlanksA(lpszPath); 1083 } 1084 return iRet; 1085 } 1086 1087 /************************************************************************* 1088 * PathParseIconLocationW [SHLWAPI.@] 1089 * 1090 * See PathParseIconLocationA. 1091 */ 1092 int WINAPI PathParseIconLocationW(LPWSTR lpszPath) 1093 { 1094 int iRet = 0; 1095 LPWSTR lpszComma; 1096 1097 TRACE("(%s)\n", debugstr_w(lpszPath)); 1098 1099 if (lpszPath) 1100 { 1101 if ((lpszComma = StrChrW(lpszPath, ','))) 1102 { 1103 *lpszComma++ = '\0'; 1104 iRet = StrToIntW(lpszComma); 1105 } 1106 PathUnquoteSpacesW(lpszPath); 1107 PathRemoveBlanksW(lpszPath); 1108 } 1109 return iRet; 1110 } 1111 1112 /************************************************************************* 1113 * @ [SHLWAPI.4] 1114 * 1115 * Unicode version of PathFileExistsDefExtA. 1116 */ 1117 BOOL WINAPI PathFileExistsDefExtW(LPWSTR lpszPath,DWORD dwWhich) 1118 { 1119 static const WCHAR pszExts[][5] = { { '.', 'p', 'i', 'f', 0}, 1120 { '.', 'c', 'o', 'm', 0}, 1121 { '.', 'e', 'x', 'e', 0}, 1122 { '.', 'b', 'a', 't', 0}, 1123 { '.', 'l', 'n', 'k', 0}, 1124 { '.', 'c', 'm', 'd', 0}, 1125 { 0, 0, 0, 0, 0} }; 1126 1127 TRACE("(%s,%d)\n", debugstr_w(lpszPath), dwWhich); 1128 1129 if (!lpszPath || PathIsUNCServerW(lpszPath) || PathIsUNCServerShareW(lpszPath)) 1130 return FALSE; 1131 1132 if (dwWhich) 1133 { 1134 LPCWSTR szExt = PathFindExtensionW(lpszPath); 1135 #ifndef __REACTOS__ 1136 if (!*szExt || dwWhich & 0x40) 1137 #else 1138 if (!*szExt || dwWhich & WHICH_OPTIONAL) 1139 #endif 1140 { 1141 size_t iChoose = 0; 1142 int iLen = lstrlenW(lpszPath); 1143 if (iLen > (MAX_PATH - 5)) 1144 return FALSE; 1145 #ifndef __REACTOS__ 1146 while ( (dwWhich & 0x1) && pszExts[iChoose][0] ) 1147 #else 1148 while (pszExts[iChoose][0]) 1149 #endif 1150 { 1151 #ifdef __REACTOS__ 1152 if (dwWhich & 0x1) 1153 { 1154 if (GetFileAttributes(lpszPath) != FILE_ATTRIBUTE_DIRECTORY) 1155 #endif 1156 lstrcpyW(lpszPath + iLen, pszExts[iChoose]); 1157 if (PathFileExistsW(lpszPath)) 1158 return TRUE; 1159 #ifdef __REACTOS__ 1160 } 1161 #endif 1162 iChoose++; 1163 dwWhich >>= 1; 1164 } 1165 *(lpszPath + iLen) = (WCHAR)'\0'; 1166 return FALSE; 1167 } 1168 } 1169 return PathFileExistsW(lpszPath); 1170 } 1171 1172 /************************************************************************* 1173 * @ [SHLWAPI.3] 1174 * 1175 * Determine if a file exists locally and is of an executable type. 1176 * 1177 * PARAMS 1178 * lpszPath [I/O] File to search for 1179 * dwWhich [I] Type of executable to search for 1180 * 1181 * RETURNS 1182 * TRUE If the file was found. lpszPath contains the file name. 1183 * FALSE Otherwise. 1184 * 1185 * NOTES 1186 * lpszPath is modified in place and must be at least MAX_PATH in length. 1187 * If the function returns FALSE, the path is modified to its original state. 1188 * If the given path contains an extension or dwWhich is 0, executable 1189 * extensions are not checked. 1190 * 1191 * Ordinals 3-6 are a classic case of MS exposing limited functionality to 1192 * users (here through PathFindOnPathA()) and keeping advanced functionality for 1193 * their own developers exclusive use. Monopoly, anyone? 1194 */ 1195 BOOL WINAPI PathFileExistsDefExtA(LPSTR lpszPath,DWORD dwWhich) 1196 { 1197 BOOL bRet = FALSE; 1198 1199 TRACE("(%s,%d)\n", debugstr_a(lpszPath), dwWhich); 1200 1201 if (lpszPath) 1202 { 1203 WCHAR szPath[MAX_PATH]; 1204 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 1205 bRet = PathFileExistsDefExtW(szPath, dwWhich); 1206 if (bRet) 1207 WideCharToMultiByte(CP_ACP,0,szPath,-1,lpszPath,MAX_PATH,0,0); 1208 } 1209 return bRet; 1210 } 1211 1212 /************************************************************************* 1213 * SHLWAPI_PathFindInOtherDirs 1214 * 1215 * Internal helper for SHLWAPI_PathFindOnPathExA/W. 1216 */ 1217 static BOOL SHLWAPI_PathFindInOtherDirs(LPWSTR lpszFile, DWORD dwWhich) 1218 { 1219 static const WCHAR szSystem[] = { 'S','y','s','t','e','m','\0'}; 1220 static const WCHAR szPath[] = { 'P','A','T','H','\0'}; 1221 DWORD dwLenPATH; 1222 LPCWSTR lpszCurr; 1223 WCHAR *lpszPATH; 1224 WCHAR buff[MAX_PATH]; 1225 1226 TRACE("(%s,%08x)\n", debugstr_w(lpszFile), dwWhich); 1227 1228 /* Try system directories */ 1229 GetSystemDirectoryW(buff, MAX_PATH); 1230 if (!PathAppendW(buff, lpszFile)) 1231 return FALSE; 1232 if (PathFileExistsDefExtW(buff, dwWhich)) 1233 { 1234 strcpyW(lpszFile, buff); 1235 return TRUE; 1236 } 1237 GetWindowsDirectoryW(buff, MAX_PATH); 1238 if (!PathAppendW(buff, szSystem ) || !PathAppendW(buff, lpszFile)) 1239 return FALSE; 1240 if (PathFileExistsDefExtW(buff, dwWhich)) 1241 { 1242 strcpyW(lpszFile, buff); 1243 return TRUE; 1244 } 1245 GetWindowsDirectoryW(buff, MAX_PATH); 1246 if (!PathAppendW(buff, lpszFile)) 1247 return FALSE; 1248 if (PathFileExistsDefExtW(buff, dwWhich)) 1249 { 1250 strcpyW(lpszFile, buff); 1251 return TRUE; 1252 } 1253 /* Try dirs listed in %PATH% */ 1254 dwLenPATH = GetEnvironmentVariableW(szPath, buff, MAX_PATH); 1255 1256 if (!dwLenPATH || !(lpszPATH = HeapAlloc(GetProcessHeap(), 0, (dwLenPATH + 1) * sizeof (WCHAR)))) 1257 return FALSE; 1258 1259 GetEnvironmentVariableW(szPath, lpszPATH, dwLenPATH + 1); 1260 lpszCurr = lpszPATH; 1261 while (lpszCurr) 1262 { 1263 LPCWSTR lpszEnd = lpszCurr; 1264 LPWSTR pBuff = buff; 1265 1266 while (*lpszEnd == ' ') 1267 lpszEnd++; 1268 while (*lpszEnd && *lpszEnd != ';') 1269 *pBuff++ = *lpszEnd++; 1270 *pBuff = '\0'; 1271 1272 if (*lpszEnd) 1273 lpszCurr = lpszEnd + 1; 1274 else 1275 lpszCurr = NULL; /* Last Path, terminate after this */ 1276 1277 if (!PathAppendW(buff, lpszFile)) 1278 { 1279 HeapFree(GetProcessHeap(), 0, lpszPATH); 1280 return FALSE; 1281 } 1282 if (PathFileExistsDefExtW(buff, dwWhich)) 1283 { 1284 strcpyW(lpszFile, buff); 1285 HeapFree(GetProcessHeap(), 0, lpszPATH); 1286 return TRUE; 1287 } 1288 } 1289 HeapFree(GetProcessHeap(), 0, lpszPATH); 1290 return FALSE; 1291 } 1292 1293 /************************************************************************* 1294 * @ [SHLWAPI.5] 1295 * 1296 * Search a range of paths for a specific type of executable. 1297 * 1298 * PARAMS 1299 * lpszFile [I/O] File to search for 1300 * lppszOtherDirs [I] Other directories to look in 1301 * dwWhich [I] Type of executable to search for 1302 * 1303 * RETURNS 1304 * Success: TRUE. The path to the executable is stored in lpszFile. 1305 * Failure: FALSE. The path to the executable is unchanged. 1306 */ 1307 BOOL WINAPI PathFindOnPathExA(LPSTR lpszFile,LPCSTR *lppszOtherDirs,DWORD dwWhich) 1308 { 1309 WCHAR szFile[MAX_PATH]; 1310 WCHAR buff[MAX_PATH]; 1311 1312 TRACE("(%s,%p,%08x)\n", debugstr_a(lpszFile), lppszOtherDirs, dwWhich); 1313 1314 if (!lpszFile || !PathIsFileSpecA(lpszFile)) 1315 return FALSE; 1316 1317 MultiByteToWideChar(CP_ACP,0,lpszFile,-1,szFile,MAX_PATH); 1318 1319 /* Search provided directories first */ 1320 if (lppszOtherDirs && *lppszOtherDirs) 1321 { 1322 WCHAR szOther[MAX_PATH]; 1323 LPCSTR *lpszOtherPath = lppszOtherDirs; 1324 1325 while (lpszOtherPath && *lpszOtherPath && (*lpszOtherPath)[0]) 1326 { 1327 MultiByteToWideChar(CP_ACP,0,*lpszOtherPath,-1,szOther,MAX_PATH); 1328 PathCombineW(buff, szOther, szFile); 1329 if (PathFileExistsDefExtW(buff, dwWhich)) 1330 { 1331 WideCharToMultiByte(CP_ACP,0,buff,-1,lpszFile,MAX_PATH,0,0); 1332 return TRUE; 1333 } 1334 lpszOtherPath++; 1335 } 1336 } 1337 /* Not found, try system and path dirs */ 1338 if (SHLWAPI_PathFindInOtherDirs(szFile, dwWhich)) 1339 { 1340 WideCharToMultiByte(CP_ACP,0,szFile,-1,lpszFile,MAX_PATH,0,0); 1341 return TRUE; 1342 } 1343 return FALSE; 1344 } 1345 1346 /************************************************************************* 1347 * @ [SHLWAPI.6] 1348 * 1349 * Unicode version of PathFindOnPathExA. 1350 */ 1351 BOOL WINAPI PathFindOnPathExW(LPWSTR lpszFile,LPCWSTR *lppszOtherDirs,DWORD dwWhich) 1352 { 1353 WCHAR buff[MAX_PATH]; 1354 1355 TRACE("(%s,%p,%08x)\n", debugstr_w(lpszFile), lppszOtherDirs, dwWhich); 1356 1357 if (!lpszFile || !PathIsFileSpecW(lpszFile)) 1358 return FALSE; 1359 1360 /* Search provided directories first */ 1361 if (lppszOtherDirs && *lppszOtherDirs) 1362 { 1363 LPCWSTR *lpszOtherPath = lppszOtherDirs; 1364 while (lpszOtherPath && *lpszOtherPath && (*lpszOtherPath)[0]) 1365 { 1366 PathCombineW(buff, *lpszOtherPath, lpszFile); 1367 if (PathFileExistsDefExtW(buff, dwWhich)) 1368 { 1369 strcpyW(lpszFile, buff); 1370 return TRUE; 1371 } 1372 lpszOtherPath++; 1373 } 1374 } 1375 /* Not found, try system and path dirs */ 1376 return SHLWAPI_PathFindInOtherDirs(lpszFile, dwWhich); 1377 } 1378 1379 /************************************************************************* 1380 * PathFindOnPathA [SHLWAPI.@] 1381 * 1382 * Search a range of paths for an executable. 1383 * 1384 * PARAMS 1385 * lpszFile [I/O] File to search for 1386 * lppszOtherDirs [I] Other directories to look in 1387 * 1388 * RETURNS 1389 * Success: TRUE. The path to the executable is stored in lpszFile. 1390 * Failure: FALSE. The path to the executable is unchanged. 1391 */ 1392 BOOL WINAPI PathFindOnPathA(LPSTR lpszFile, LPCSTR *lppszOtherDirs) 1393 { 1394 TRACE("(%s,%p)\n", debugstr_a(lpszFile), lppszOtherDirs); 1395 return PathFindOnPathExA(lpszFile, lppszOtherDirs, 0); 1396 } 1397 1398 /************************************************************************* 1399 * PathFindOnPathW [SHLWAPI.@] 1400 * 1401 * See PathFindOnPathA. 1402 */ 1403 BOOL WINAPI PathFindOnPathW(LPWSTR lpszFile, LPCWSTR *lppszOtherDirs) 1404 { 1405 TRACE("(%s,%p)\n", debugstr_w(lpszFile), lppszOtherDirs); 1406 return PathFindOnPathExW(lpszFile,lppszOtherDirs, 0); 1407 } 1408 1409 /************************************************************************* 1410 * PathCompactPathExA [SHLWAPI.@] 1411 * 1412 * Compact a path into a given number of characters. 1413 * 1414 * PARAMS 1415 * lpszDest [O] Destination for compacted path 1416 * lpszPath [I] Source path 1417 * cchMax [I] Maximum size of compacted path 1418 * dwFlags [I] Reserved 1419 * 1420 * RETURNS 1421 * Success: TRUE. The compacted path is written to lpszDest. 1422 * Failure: FALSE. lpszPath is undefined. 1423 * 1424 * NOTES 1425 * If cchMax is given as 0, lpszDest will still be NUL terminated. 1426 * 1427 * The Win32 version of this function contains a bug: When cchMax == 7, 1428 * 8 bytes will be written to lpszDest. This bug is fixed in the Wine 1429 * implementation. 1430 * 1431 * Some relative paths will be different when cchMax == 5 or 6. This occurs 1432 * because Win32 will insert a "\" in lpszDest, even if one is 1433 * not present in the original path. 1434 */ 1435 BOOL WINAPI PathCompactPathExA(LPSTR lpszDest, LPCSTR lpszPath, 1436 UINT cchMax, DWORD dwFlags) 1437 { 1438 BOOL bRet = FALSE; 1439 1440 TRACE("(%p,%s,%d,0x%08x)\n", lpszDest, debugstr_a(lpszPath), cchMax, dwFlags); 1441 1442 if (lpszPath && lpszDest) 1443 { 1444 WCHAR szPath[MAX_PATH]; 1445 WCHAR szDest[MAX_PATH]; 1446 1447 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 1448 szDest[0] = '\0'; 1449 bRet = PathCompactPathExW(szDest, szPath, cchMax, dwFlags); 1450 WideCharToMultiByte(CP_ACP,0,szDest,-1,lpszDest,MAX_PATH,0,0); 1451 } 1452 return bRet; 1453 } 1454 1455 /************************************************************************* 1456 * PathCompactPathExW [SHLWAPI.@] 1457 * 1458 * See PathCompactPathExA. 1459 */ 1460 BOOL WINAPI PathCompactPathExW(LPWSTR lpszDest, LPCWSTR lpszPath, 1461 UINT cchMax, DWORD dwFlags) 1462 { 1463 static const WCHAR szEllipses[] = { '.', '.', '.', '\0' }; 1464 LPCWSTR lpszFile; 1465 DWORD dwLen, dwFileLen = 0; 1466 1467 TRACE("(%p,%s,%d,0x%08x)\n", lpszDest, debugstr_w(lpszPath), cchMax, dwFlags); 1468 1469 if (!lpszPath) 1470 return FALSE; 1471 1472 if (!lpszDest) 1473 { 1474 WARN("Invalid lpszDest would crash under Win32!\n"); 1475 return FALSE; 1476 } 1477 1478 *lpszDest = '\0'; 1479 1480 if (cchMax < 2) 1481 return TRUE; 1482 1483 dwLen = strlenW(lpszPath) + 1; 1484 1485 if (dwLen < cchMax) 1486 { 1487 /* Don't need to compact */ 1488 memcpy(lpszDest, lpszPath, dwLen * sizeof(WCHAR)); 1489 return TRUE; 1490 } 1491 1492 /* Path must be compacted to fit into lpszDest */ 1493 lpszFile = PathFindFileNameW(lpszPath); 1494 dwFileLen = lpszPath + dwLen - lpszFile; 1495 1496 if (dwFileLen == dwLen) 1497 { 1498 /* No root in psth */ 1499 if (cchMax <= 4) 1500 { 1501 while (--cchMax > 0) /* No room left for anything but ellipses */ 1502 *lpszDest++ = '.'; 1503 *lpszDest = '\0'; 1504 return TRUE; 1505 } 1506 /* Compact the file name with ellipses at the end */ 1507 cchMax -= 4; 1508 memcpy(lpszDest, lpszFile, cchMax * sizeof(WCHAR)); 1509 strcpyW(lpszDest + cchMax, szEllipses); 1510 return TRUE; 1511 } 1512 /* We have a root in the path */ 1513 lpszFile--; /* Start compacted filename with the path separator */ 1514 dwFileLen++; 1515 1516 if (dwFileLen + 3 > cchMax) 1517 { 1518 /* Compact the file name */ 1519 if (cchMax <= 4) 1520 { 1521 while (--cchMax > 0) /* No room left for anything but ellipses */ 1522 *lpszDest++ = '.'; 1523 *lpszDest = '\0'; 1524 return TRUE; 1525 } 1526 strcpyW(lpszDest, szEllipses); 1527 lpszDest += 3; 1528 cchMax -= 4; 1529 *lpszDest++ = *lpszFile++; 1530 if (cchMax <= 4) 1531 { 1532 while (--cchMax > 0) /* No room left for anything but ellipses */ 1533 *lpszDest++ = '.'; 1534 *lpszDest = '\0'; 1535 return TRUE; 1536 } 1537 cchMax -= 4; 1538 memcpy(lpszDest, lpszFile, cchMax * sizeof(WCHAR)); 1539 strcpyW(lpszDest + cchMax, szEllipses); 1540 return TRUE; 1541 } 1542 1543 /* Only the root needs to be Compacted */ 1544 dwLen = cchMax - dwFileLen - 3; 1545 memcpy(lpszDest, lpszPath, dwLen * sizeof(WCHAR)); 1546 strcpyW(lpszDest + dwLen, szEllipses); 1547 strcpyW(lpszDest + dwLen + 3, lpszFile); 1548 return TRUE; 1549 } 1550 1551 /************************************************************************* 1552 * PathIsRelativeA [SHLWAPI.@] 1553 * 1554 * Determine if a path is a relative path. 1555 * 1556 * PARAMS 1557 * lpszPath [I] Path to check 1558 * 1559 * RETURNS 1560 * TRUE: The path is relative, or is invalid. 1561 * FALSE: The path is not relative. 1562 */ 1563 BOOL WINAPI PathIsRelativeA (LPCSTR lpszPath) 1564 { 1565 TRACE("(%s)\n",debugstr_a(lpszPath)); 1566 1567 if (!lpszPath || !*lpszPath || IsDBCSLeadByte(*lpszPath)) 1568 return TRUE; 1569 if (*lpszPath == '\\' || (*lpszPath && lpszPath[1] == ':')) 1570 return FALSE; 1571 return TRUE; 1572 } 1573 1574 /************************************************************************* 1575 * PathIsRelativeW [SHLWAPI.@] 1576 * 1577 * See PathIsRelativeA. 1578 */ 1579 BOOL WINAPI PathIsRelativeW (LPCWSTR lpszPath) 1580 { 1581 TRACE("(%s)\n",debugstr_w(lpszPath)); 1582 1583 if (!lpszPath || !*lpszPath) 1584 return TRUE; 1585 if (*lpszPath == '\\' || (*lpszPath && lpszPath[1] == ':')) 1586 return FALSE; 1587 return TRUE; 1588 } 1589 1590 /************************************************************************* 1591 * PathIsRootA [SHLWAPI.@] 1592 * 1593 * Determine if a path is a root path. 1594 * 1595 * PARAMS 1596 * lpszPath [I] Path to check 1597 * 1598 * RETURNS 1599 * TRUE If lpszPath is valid and a root path, 1600 * FALSE Otherwise 1601 */ 1602 BOOL WINAPI PathIsRootA(LPCSTR lpszPath) 1603 { 1604 TRACE("(%s)\n", debugstr_a(lpszPath)); 1605 1606 if (lpszPath && *lpszPath) 1607 { 1608 if (*lpszPath == '\\') 1609 { 1610 if (!lpszPath[1]) 1611 return TRUE; /* \ */ 1612 else if (lpszPath[1]=='\\') 1613 { 1614 BOOL bSeenSlash = FALSE; 1615 lpszPath += 2; 1616 1617 /* Check for UNC root path */ 1618 while (*lpszPath) 1619 { 1620 if (*lpszPath == '\\') 1621 { 1622 if (bSeenSlash) 1623 return FALSE; 1624 bSeenSlash = TRUE; 1625 } 1626 lpszPath = CharNextA(lpszPath); 1627 } 1628 return TRUE; 1629 } 1630 } 1631 else if (lpszPath[1] == ':' && lpszPath[2] == '\\' && lpszPath[3] == '\0') 1632 return TRUE; /* X:\ */ 1633 } 1634 return FALSE; 1635 } 1636 1637 /************************************************************************* 1638 * PathIsRootW [SHLWAPI.@] 1639 * 1640 * See PathIsRootA. 1641 */ 1642 BOOL WINAPI PathIsRootW(LPCWSTR lpszPath) 1643 { 1644 TRACE("(%s)\n", debugstr_w(lpszPath)); 1645 1646 if (lpszPath && *lpszPath) 1647 { 1648 if (*lpszPath == '\\') 1649 { 1650 if (!lpszPath[1]) 1651 return TRUE; /* \ */ 1652 else if (lpszPath[1]=='\\') 1653 { 1654 BOOL bSeenSlash = FALSE; 1655 lpszPath += 2; 1656 1657 /* Check for UNC root path */ 1658 while (*lpszPath) 1659 { 1660 if (*lpszPath == '\\') 1661 { 1662 if (bSeenSlash) 1663 return FALSE; 1664 bSeenSlash = TRUE; 1665 } 1666 lpszPath++; 1667 } 1668 return TRUE; 1669 } 1670 } 1671 else if (lpszPath[1] == ':' && lpszPath[2] == '\\' && lpszPath[3] == '\0') 1672 return TRUE; /* X:\ */ 1673 } 1674 return FALSE; 1675 } 1676 1677 /************************************************************************* 1678 * PathIsDirectoryA [SHLWAPI.@] 1679 * 1680 * Determine if a path is a valid directory 1681 * 1682 * PARAMS 1683 * lpszPath [I] Path to check. 1684 * 1685 * RETURNS 1686 * FILE_ATTRIBUTE_DIRECTORY if lpszPath exists and can be read (See Notes) 1687 * FALSE if lpszPath is invalid or not a directory. 1688 * 1689 * NOTES 1690 * Although this function is prototyped as returning a BOOL, it returns 1691 * FILE_ATTRIBUTE_DIRECTORY for success. This means that code such as: 1692 * 1693 *| if (PathIsDirectoryA("c:\\windows\\") == TRUE) 1694 *| ... 1695 * 1696 * will always fail. 1697 */ 1698 BOOL WINAPI PathIsDirectoryA(LPCSTR lpszPath) 1699 { 1700 DWORD dwAttr; 1701 1702 TRACE("(%s)\n", debugstr_a(lpszPath)); 1703 1704 if (!lpszPath || PathIsUNCServerA(lpszPath)) 1705 return FALSE; 1706 1707 if (PathIsUNCServerShareA(lpszPath)) 1708 { 1709 FIXME("UNC Server Share not yet supported - FAILING\n"); 1710 return FALSE; 1711 } 1712 1713 if ((dwAttr = GetFileAttributesA(lpszPath)) == INVALID_FILE_ATTRIBUTES) 1714 return FALSE; 1715 return dwAttr & FILE_ATTRIBUTE_DIRECTORY; 1716 } 1717 1718 /************************************************************************* 1719 * PathIsDirectoryW [SHLWAPI.@] 1720 * 1721 * See PathIsDirectoryA. 1722 */ 1723 BOOL WINAPI PathIsDirectoryW(LPCWSTR lpszPath) 1724 { 1725 DWORD dwAttr; 1726 1727 TRACE("(%s)\n", debugstr_w(lpszPath)); 1728 1729 if (!lpszPath || PathIsUNCServerW(lpszPath)) 1730 return FALSE; 1731 1732 if (PathIsUNCServerShareW(lpszPath)) 1733 { 1734 FIXME("UNC Server Share not yet supported - FAILING\n"); 1735 return FALSE; 1736 } 1737 1738 if ((dwAttr = GetFileAttributesW(lpszPath)) == INVALID_FILE_ATTRIBUTES) 1739 return FALSE; 1740 return dwAttr & FILE_ATTRIBUTE_DIRECTORY; 1741 } 1742 1743 /************************************************************************* 1744 * PathFileExistsA [SHLWAPI.@] 1745 * 1746 * Determine if a file exists. 1747 * 1748 * PARAMS 1749 * lpszPath [I] Path to check 1750 * 1751 * RETURNS 1752 * TRUE If the file exists and is readable 1753 * FALSE Otherwise 1754 */ 1755 BOOL WINAPI PathFileExistsA(LPCSTR lpszPath) 1756 { 1757 UINT iPrevErrMode; 1758 DWORD dwAttr; 1759 1760 TRACE("(%s)\n",debugstr_a(lpszPath)); 1761 1762 if (!lpszPath) 1763 return FALSE; 1764 1765 /* Prevent a dialog box if path is on a disk that has been ejected. */ 1766 iPrevErrMode = SetErrorMode(SEM_FAILCRITICALERRORS); 1767 dwAttr = GetFileAttributesA(lpszPath); 1768 SetErrorMode(iPrevErrMode); 1769 return dwAttr != INVALID_FILE_ATTRIBUTES; 1770 } 1771 1772 /************************************************************************* 1773 * PathFileExistsW [SHLWAPI.@] 1774 * 1775 * See PathFileExistsA. 1776 */ 1777 BOOL WINAPI PathFileExistsW(LPCWSTR lpszPath) 1778 { 1779 UINT iPrevErrMode; 1780 DWORD dwAttr; 1781 1782 TRACE("(%s)\n",debugstr_w(lpszPath)); 1783 1784 if (!lpszPath) 1785 return FALSE; 1786 1787 iPrevErrMode = SetErrorMode(SEM_FAILCRITICALERRORS); 1788 dwAttr = GetFileAttributesW(lpszPath); 1789 SetErrorMode(iPrevErrMode); 1790 return dwAttr != INVALID_FILE_ATTRIBUTES; 1791 } 1792 1793 /************************************************************************* 1794 * PathFileExistsAndAttributesA [SHLWAPI.445] 1795 * 1796 * Determine if a file exists. 1797 * 1798 * PARAMS 1799 * lpszPath [I] Path to check 1800 * dwAttr [O] attributes of file 1801 * 1802 * RETURNS 1803 * TRUE If the file exists and is readable 1804 * FALSE Otherwise 1805 */ 1806 BOOL WINAPI PathFileExistsAndAttributesA(LPCSTR lpszPath, DWORD *dwAttr) 1807 { 1808 UINT iPrevErrMode; 1809 DWORD dwVal = 0; 1810 1811 TRACE("(%s %p)\n", debugstr_a(lpszPath), dwAttr); 1812 1813 if (dwAttr) 1814 *dwAttr = INVALID_FILE_ATTRIBUTES; 1815 1816 if (!lpszPath) 1817 return FALSE; 1818 1819 iPrevErrMode = SetErrorMode(SEM_FAILCRITICALERRORS); 1820 dwVal = GetFileAttributesA(lpszPath); 1821 SetErrorMode(iPrevErrMode); 1822 if (dwAttr) 1823 *dwAttr = dwVal; 1824 return (dwVal != INVALID_FILE_ATTRIBUTES); 1825 } 1826 1827 /************************************************************************* 1828 * PathFileExistsAndAttributesW [SHLWAPI.446] 1829 * 1830 * See PathFileExistsA. 1831 */ 1832 BOOL WINAPI PathFileExistsAndAttributesW(LPCWSTR lpszPath, DWORD *dwAttr) 1833 { 1834 UINT iPrevErrMode; 1835 DWORD dwVal; 1836 1837 TRACE("(%s %p)\n", debugstr_w(lpszPath), dwAttr); 1838 1839 if (!lpszPath) 1840 return FALSE; 1841 1842 iPrevErrMode = SetErrorMode(SEM_FAILCRITICALERRORS); 1843 dwVal = GetFileAttributesW(lpszPath); 1844 SetErrorMode(iPrevErrMode); 1845 if (dwAttr) 1846 *dwAttr = dwVal; 1847 return (dwVal != INVALID_FILE_ATTRIBUTES); 1848 } 1849 1850 /************************************************************************* 1851 * PathMatchSingleMaskA [internal] 1852 */ 1853 static BOOL PathMatchSingleMaskA(LPCSTR name, LPCSTR mask) 1854 { 1855 while (*name && *mask && *mask!=';') 1856 { 1857 if (*mask == '*') 1858 { 1859 do 1860 { 1861 if (PathMatchSingleMaskA(name,mask+1)) 1862 return TRUE; /* try substrings */ 1863 } while (*name++); 1864 return FALSE; 1865 } 1866 1867 if (toupper(*mask) != toupper(*name) && *mask != '?') 1868 return FALSE; 1869 1870 name = CharNextA(name); 1871 mask = CharNextA(mask); 1872 } 1873 1874 if (!*name) 1875 { 1876 while (*mask == '*') 1877 mask++; 1878 if (!*mask || *mask == ';') 1879 return TRUE; 1880 } 1881 return FALSE; 1882 } 1883 1884 /************************************************************************* 1885 * PathMatchSingleMaskW [internal] 1886 */ 1887 static BOOL PathMatchSingleMaskW(LPCWSTR name, LPCWSTR mask) 1888 { 1889 while (*name && *mask && *mask != ';') 1890 { 1891 if (*mask == '*') 1892 { 1893 do 1894 { 1895 if (PathMatchSingleMaskW(name,mask+1)) 1896 return TRUE; /* try substrings */ 1897 } while (*name++); 1898 return FALSE; 1899 } 1900 1901 if (toupperW(*mask) != toupperW(*name) && *mask != '?') 1902 return FALSE; 1903 1904 name++; 1905 mask++; 1906 } 1907 if (!*name) 1908 { 1909 while (*mask == '*') 1910 mask++; 1911 if (!*mask || *mask == ';') 1912 return TRUE; 1913 } 1914 return FALSE; 1915 } 1916 1917 /************************************************************************* 1918 * PathMatchSpecA [SHLWAPI.@] 1919 * 1920 * Determine if a path matches one or more search masks. 1921 * 1922 * PARAMS 1923 * lpszPath [I] Path to check 1924 * lpszMask [I] Search mask(s) 1925 * 1926 * RETURNS 1927 * TRUE If lpszPath is valid and is matched 1928 * FALSE Otherwise 1929 * 1930 * NOTES 1931 * Multiple search masks may be given if they are separated by ";". The 1932 * pattern "*.*" is treated specially in that it matches all paths (for 1933 * backwards compatibility with DOS). 1934 */ 1935 BOOL WINAPI PathMatchSpecA(LPCSTR lpszPath, LPCSTR lpszMask) 1936 { 1937 TRACE("(%s,%s)\n", lpszPath, lpszMask); 1938 1939 if (!lstrcmpA(lpszMask, "*.*")) 1940 return TRUE; /* Matches every path */ 1941 1942 while (*lpszMask) 1943 { 1944 while (*lpszMask == ' ') 1945 lpszMask++; /* Eat leading spaces */ 1946 1947 if (PathMatchSingleMaskA(lpszPath, lpszMask)) 1948 return TRUE; /* Matches the current mask */ 1949 1950 while (*lpszMask && *lpszMask != ';') 1951 lpszMask = CharNextA(lpszMask); /* masks separated by ';' */ 1952 1953 if (*lpszMask == ';') 1954 lpszMask++; 1955 } 1956 return FALSE; 1957 } 1958 1959 /************************************************************************* 1960 * PathMatchSpecW [SHLWAPI.@] 1961 * 1962 * See PathMatchSpecA. 1963 */ 1964 BOOL WINAPI PathMatchSpecW(LPCWSTR lpszPath, LPCWSTR lpszMask) 1965 { 1966 static const WCHAR szStarDotStar[] = { '*', '.', '*', '\0' }; 1967 1968 TRACE("(%s,%s)\n", debugstr_w(lpszPath), debugstr_w(lpszMask)); 1969 1970 if (!lstrcmpW(lpszMask, szStarDotStar)) 1971 return TRUE; /* Matches every path */ 1972 1973 while (*lpszMask) 1974 { 1975 while (*lpszMask == ' ') 1976 lpszMask++; /* Eat leading spaces */ 1977 1978 if (PathMatchSingleMaskW(lpszPath, lpszMask)) 1979 return TRUE; /* Matches the current path */ 1980 1981 while (*lpszMask && *lpszMask != ';') 1982 lpszMask++; /* masks separated by ';' */ 1983 1984 if (*lpszMask == ';') 1985 lpszMask++; 1986 } 1987 return FALSE; 1988 } 1989 1990 /************************************************************************* 1991 * PathIsSameRootA [SHLWAPI.@] 1992 * 1993 * Determine if two paths share the same root. 1994 * 1995 * PARAMS 1996 * lpszPath1 [I] Source path 1997 * lpszPath2 [I] Path to compare with 1998 * 1999 * RETURNS 2000 * TRUE If both paths are valid and share the same root. 2001 * FALSE If either path is invalid or the paths do not share the same root. 2002 */ 2003 BOOL WINAPI PathIsSameRootA(LPCSTR lpszPath1, LPCSTR lpszPath2) 2004 { 2005 LPCSTR lpszStart; 2006 int dwLen; 2007 2008 TRACE("(%s,%s)\n", debugstr_a(lpszPath1), debugstr_a(lpszPath2)); 2009 2010 if (!lpszPath1 || !lpszPath2 || !(lpszStart = PathSkipRootA(lpszPath1))) 2011 return FALSE; 2012 2013 dwLen = PathCommonPrefixA(lpszPath1, lpszPath2, NULL) + 1; 2014 if (lpszStart - lpszPath1 > dwLen) 2015 return FALSE; /* Paths not common up to length of the root */ 2016 return TRUE; 2017 } 2018 2019 /************************************************************************* 2020 * PathIsSameRootW [SHLWAPI.@] 2021 * 2022 * See PathIsSameRootA. 2023 */ 2024 BOOL WINAPI PathIsSameRootW(LPCWSTR lpszPath1, LPCWSTR lpszPath2) 2025 { 2026 LPCWSTR lpszStart; 2027 int dwLen; 2028 2029 TRACE("(%s,%s)\n", debugstr_w(lpszPath1), debugstr_w(lpszPath2)); 2030 2031 if (!lpszPath1 || !lpszPath2 || !(lpszStart = PathSkipRootW(lpszPath1))) 2032 return FALSE; 2033 2034 dwLen = PathCommonPrefixW(lpszPath1, lpszPath2, NULL) + 1; 2035 if (lpszStart - lpszPath1 > dwLen) 2036 return FALSE; /* Paths not common up to length of the root */ 2037 return TRUE; 2038 } 2039 2040 /************************************************************************* 2041 * PathIsContentTypeA [SHLWAPI.@] 2042 * 2043 * Determine if a file is of a given registered content type. 2044 * 2045 * PARAMS 2046 * lpszPath [I] File to check 2047 * lpszContentType [I] Content type to check for 2048 * 2049 * RETURNS 2050 * TRUE If lpszPath is a given registered content type, 2051 * FALSE Otherwise. 2052 * 2053 * NOTES 2054 * This function looks up the registered content type for lpszPath. If 2055 * a content type is registered, it is compared (case insensitively) to 2056 * lpszContentType. Only if this matches does the function succeed. 2057 */ 2058 BOOL WINAPI PathIsContentTypeA(LPCSTR lpszPath, LPCSTR lpszContentType) 2059 { 2060 LPCSTR szExt; 2061 DWORD dwDummy; 2062 char szBuff[MAX_PATH]; 2063 2064 TRACE("(%s,%s)\n", debugstr_a(lpszPath), debugstr_a(lpszContentType)); 2065 2066 if (lpszPath && (szExt = PathFindExtensionA(lpszPath)) && *szExt && 2067 !SHGetValueA(HKEY_CLASSES_ROOT, szExt, "Content Type", 2068 REG_NONE, szBuff, &dwDummy) && 2069 !strcasecmp(lpszContentType, szBuff)) 2070 { 2071 return TRUE; 2072 } 2073 return FALSE; 2074 } 2075 2076 /************************************************************************* 2077 * PathIsContentTypeW [SHLWAPI.@] 2078 * 2079 * See PathIsContentTypeA. 2080 */ 2081 BOOL WINAPI PathIsContentTypeW(LPCWSTR lpszPath, LPCWSTR lpszContentType) 2082 { 2083 static const WCHAR szContentType[] = { 'C','o','n','t','e','n','t',' ','T','y','p','e','\0' }; 2084 LPCWSTR szExt; 2085 DWORD dwDummy; 2086 WCHAR szBuff[MAX_PATH]; 2087 2088 TRACE("(%s,%s)\n", debugstr_w(lpszPath), debugstr_w(lpszContentType)); 2089 2090 if (lpszPath && (szExt = PathFindExtensionW(lpszPath)) && *szExt && 2091 !SHGetValueW(HKEY_CLASSES_ROOT, szExt, szContentType, 2092 REG_NONE, szBuff, &dwDummy) && 2093 !strcmpiW(lpszContentType, szBuff)) 2094 { 2095 return TRUE; 2096 } 2097 return FALSE; 2098 } 2099 2100 /************************************************************************* 2101 * PathIsFileSpecA [SHLWAPI.@] 2102 * 2103 * Determine if a path is a file specification. 2104 * 2105 * PARAMS 2106 * lpszPath [I] Path to check 2107 * 2108 * RETURNS 2109 * TRUE If lpszPath is a file specification (i.e. Contains no directories). 2110 * FALSE Otherwise. 2111 */ 2112 BOOL WINAPI PathIsFileSpecA(LPCSTR lpszPath) 2113 { 2114 TRACE("(%s)\n", debugstr_a(lpszPath)); 2115 2116 if (!lpszPath) 2117 return FALSE; 2118 2119 while (*lpszPath) 2120 { 2121 if (*lpszPath == '\\' || *lpszPath == ':') 2122 return FALSE; 2123 lpszPath = CharNextA(lpszPath); 2124 } 2125 return TRUE; 2126 } 2127 2128 /************************************************************************* 2129 * PathIsFileSpecW [SHLWAPI.@] 2130 * 2131 * See PathIsFileSpecA. 2132 */ 2133 BOOL WINAPI PathIsFileSpecW(LPCWSTR lpszPath) 2134 { 2135 TRACE("(%s)\n", debugstr_w(lpszPath)); 2136 2137 if (!lpszPath) 2138 return FALSE; 2139 2140 while (*lpszPath) 2141 { 2142 if (*lpszPath == '\\' || *lpszPath == ':') 2143 return FALSE; 2144 lpszPath++; 2145 } 2146 return TRUE; 2147 } 2148 2149 /************************************************************************* 2150 * PathIsPrefixA [SHLWAPI.@] 2151 * 2152 * Determine if a path is a prefix of another. 2153 * 2154 * PARAMS 2155 * lpszPrefix [I] Prefix 2156 * lpszPath [I] Path to check 2157 * 2158 * RETURNS 2159 * TRUE If lpszPath has lpszPrefix as its prefix, 2160 * FALSE If either path is NULL or lpszPrefix is not a prefix 2161 */ 2162 BOOL WINAPI PathIsPrefixA (LPCSTR lpszPrefix, LPCSTR lpszPath) 2163 { 2164 TRACE("(%s,%s)\n", debugstr_a(lpszPrefix), debugstr_a(lpszPath)); 2165 2166 if (lpszPrefix && lpszPath && 2167 PathCommonPrefixA(lpszPath, lpszPrefix, NULL) == (int)strlen(lpszPrefix)) 2168 return TRUE; 2169 return FALSE; 2170 } 2171 2172 /************************************************************************* 2173 * PathIsPrefixW [SHLWAPI.@] 2174 * 2175 * See PathIsPrefixA. 2176 */ 2177 BOOL WINAPI PathIsPrefixW(LPCWSTR lpszPrefix, LPCWSTR lpszPath) 2178 { 2179 TRACE("(%s,%s)\n", debugstr_w(lpszPrefix), debugstr_w(lpszPath)); 2180 2181 if (lpszPrefix && lpszPath && 2182 PathCommonPrefixW(lpszPath, lpszPrefix, NULL) == (int)strlenW(lpszPrefix)) 2183 return TRUE; 2184 return FALSE; 2185 } 2186 2187 /************************************************************************* 2188 * PathIsSystemFolderA [SHLWAPI.@] 2189 * 2190 * Determine if a path or file attributes are a system folder. 2191 * 2192 * PARAMS 2193 * lpszPath [I] Path to check. 2194 * dwAttrib [I] Attributes to check, if lpszPath is NULL. 2195 * 2196 * RETURNS 2197 * TRUE If lpszPath or dwAttrib are a system folder. 2198 * FALSE If GetFileAttributesA() fails or neither parameter is a system folder. 2199 */ 2200 BOOL WINAPI PathIsSystemFolderA(LPCSTR lpszPath, DWORD dwAttrib) 2201 { 2202 TRACE("(%s,0x%08x)\n", debugstr_a(lpszPath), dwAttrib); 2203 2204 if (lpszPath && *lpszPath) 2205 dwAttrib = GetFileAttributesA(lpszPath); 2206 2207 if (dwAttrib == INVALID_FILE_ATTRIBUTES || !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY) || 2208 !(dwAttrib & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_READONLY))) 2209 return FALSE; 2210 return TRUE; 2211 } 2212 2213 /************************************************************************* 2214 * PathIsSystemFolderW [SHLWAPI.@] 2215 * 2216 * See PathIsSystemFolderA. 2217 */ 2218 BOOL WINAPI PathIsSystemFolderW(LPCWSTR lpszPath, DWORD dwAttrib) 2219 { 2220 TRACE("(%s,0x%08x)\n", debugstr_w(lpszPath), dwAttrib); 2221 2222 if (lpszPath && *lpszPath) 2223 dwAttrib = GetFileAttributesW(lpszPath); 2224 2225 if (dwAttrib == INVALID_FILE_ATTRIBUTES || !(dwAttrib & FILE_ATTRIBUTE_DIRECTORY) || 2226 !(dwAttrib & (FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_READONLY))) 2227 return FALSE; 2228 return TRUE; 2229 } 2230 2231 /************************************************************************* 2232 * PathIsUNCA [SHLWAPI.@] 2233 * 2234 * Determine if a path is in UNC format. 2235 * 2236 * PARAMS 2237 * lpszPath [I] Path to check 2238 * 2239 * RETURNS 2240 * TRUE: The path is UNC. 2241 * FALSE: The path is not UNC or is NULL. 2242 */ 2243 BOOL WINAPI PathIsUNCA(LPCSTR lpszPath) 2244 { 2245 TRACE("(%s)\n",debugstr_a(lpszPath)); 2246 2247 /* 2248 * On Windows 2003, tests show that strings starting with "\\?" are 2249 * considered UNC, while on Windows Vista+ this is not the case anymore. 2250 */ 2251 // #ifdef __REACTOS__ 2252 #if (_WIN32_WINNT >= _WIN32_WINNT_VISTA) 2253 if (lpszPath && (lpszPath[0]=='\\') && (lpszPath[1]=='\\') && (lpszPath[2]!='?')) 2254 #else 2255 if (lpszPath && (lpszPath[0]=='\\') && (lpszPath[1]=='\\')) 2256 #endif 2257 return TRUE; 2258 return FALSE; 2259 } 2260 2261 /************************************************************************* 2262 * PathIsUNCW [SHLWAPI.@] 2263 * 2264 * See PathIsUNCA. 2265 */ 2266 BOOL WINAPI PathIsUNCW(LPCWSTR lpszPath) 2267 { 2268 TRACE("(%s)\n",debugstr_w(lpszPath)); 2269 2270 /* 2271 * On Windows 2003, tests show that strings starting with "\\?" are 2272 * considered UNC, while on Windows Vista+ this is not the case anymore. 2273 */ 2274 // #ifdef __REACTOS__ 2275 #if (_WIN32_WINNT >= _WIN32_WINNT_VISTA) 2276 if (lpszPath && (lpszPath[0]=='\\') && (lpszPath[1]=='\\') && (lpszPath[2]!='?')) 2277 #else 2278 if (lpszPath && (lpszPath[0]=='\\') && (lpszPath[1]=='\\')) 2279 #endif 2280 return TRUE; 2281 return FALSE; 2282 } 2283 2284 /************************************************************************* 2285 * PathIsUNCServerA [SHLWAPI.@] 2286 * 2287 * Determine if a path is a UNC server name ("\\SHARENAME"). 2288 * 2289 * PARAMS 2290 * lpszPath [I] Path to check. 2291 * 2292 * RETURNS 2293 * TRUE If lpszPath is a valid UNC server name. 2294 * FALSE Otherwise. 2295 * 2296 * NOTES 2297 * This routine is bug compatible with Win32: Server names with a 2298 * trailing backslash (e.g. "\\FOO\"), return FALSE incorrectly. 2299 * Fixing this bug may break other shlwapi functions! 2300 */ 2301 BOOL WINAPI PathIsUNCServerA(LPCSTR lpszPath) 2302 { 2303 TRACE("(%s)\n", debugstr_a(lpszPath)); 2304 2305 if (lpszPath && *lpszPath++ == '\\' && *lpszPath++ == '\\') 2306 { 2307 while (*lpszPath) 2308 { 2309 if (*lpszPath == '\\') 2310 return FALSE; 2311 lpszPath = CharNextA(lpszPath); 2312 } 2313 return TRUE; 2314 } 2315 return FALSE; 2316 } 2317 2318 /************************************************************************* 2319 * PathIsUNCServerW [SHLWAPI.@] 2320 * 2321 * See PathIsUNCServerA. 2322 */ 2323 BOOL WINAPI PathIsUNCServerW(LPCWSTR lpszPath) 2324 { 2325 TRACE("(%s)\n", debugstr_w(lpszPath)); 2326 2327 if (lpszPath && lpszPath[0] == '\\' && lpszPath[1] == '\\') 2328 { 2329 return !strchrW( lpszPath + 2, '\\' ); 2330 } 2331 return FALSE; 2332 } 2333 2334 /************************************************************************* 2335 * PathIsUNCServerShareA [SHLWAPI.@] 2336 * 2337 * Determine if a path is a UNC server share ("\\SHARENAME\SHARE"). 2338 * 2339 * PARAMS 2340 * lpszPath [I] Path to check. 2341 * 2342 * RETURNS 2343 * TRUE If lpszPath is a valid UNC server share. 2344 * FALSE Otherwise. 2345 * 2346 * NOTES 2347 * This routine is bug compatible with Win32: Server shares with a 2348 * trailing backslash (e.g. "\\FOO\BAR\"), return FALSE incorrectly. 2349 * Fixing this bug may break other shlwapi functions! 2350 */ 2351 BOOL WINAPI PathIsUNCServerShareA(LPCSTR lpszPath) 2352 { 2353 TRACE("(%s)\n", debugstr_a(lpszPath)); 2354 2355 if (lpszPath && *lpszPath++ == '\\' && *lpszPath++ == '\\') 2356 { 2357 BOOL bSeenSlash = FALSE; 2358 while (*lpszPath) 2359 { 2360 if (*lpszPath == '\\') 2361 { 2362 if (bSeenSlash) 2363 return FALSE; 2364 bSeenSlash = TRUE; 2365 } 2366 lpszPath = CharNextA(lpszPath); 2367 } 2368 return bSeenSlash; 2369 } 2370 return FALSE; 2371 } 2372 2373 /************************************************************************* 2374 * PathIsUNCServerShareW [SHLWAPI.@] 2375 * 2376 * See PathIsUNCServerShareA. 2377 */ 2378 BOOL WINAPI PathIsUNCServerShareW(LPCWSTR lpszPath) 2379 { 2380 TRACE("(%s)\n", debugstr_w(lpszPath)); 2381 2382 if (lpszPath && *lpszPath++ == '\\' && *lpszPath++ == '\\') 2383 { 2384 BOOL bSeenSlash = FALSE; 2385 while (*lpszPath) 2386 { 2387 if (*lpszPath == '\\') 2388 { 2389 if (bSeenSlash) 2390 return FALSE; 2391 bSeenSlash = TRUE; 2392 } 2393 lpszPath++; 2394 } 2395 return bSeenSlash; 2396 } 2397 return FALSE; 2398 } 2399 2400 /************************************************************************* 2401 * PathCanonicalizeA [SHLWAPI.@] 2402 * 2403 * Convert a path to its canonical form. 2404 * 2405 * PARAMS 2406 * lpszBuf [O] Output path 2407 * lpszPath [I] Path to canonicalize 2408 * 2409 * RETURNS 2410 * Success: TRUE. lpszBuf contains the output path, 2411 * Failure: FALSE, If input path is invalid. lpszBuf is undefined 2412 */ 2413 BOOL WINAPI PathCanonicalizeA(LPSTR lpszBuf, LPCSTR lpszPath) 2414 { 2415 BOOL bRet = FALSE; 2416 2417 TRACE("(%p,%s)\n", lpszBuf, debugstr_a(lpszPath)); 2418 2419 if (lpszBuf) 2420 *lpszBuf = '\0'; 2421 2422 if (!lpszBuf || !lpszPath) 2423 SetLastError(ERROR_INVALID_PARAMETER); 2424 else 2425 { 2426 WCHAR szPath[MAX_PATH]; 2427 WCHAR szBuff[MAX_PATH]; 2428 int ret = MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 2429 2430 if (!ret) { 2431 WARN("Failed to convert string to widechar (too long?), LE %d.\n", GetLastError()); 2432 return FALSE; 2433 } 2434 bRet = PathCanonicalizeW(szBuff, szPath); 2435 WideCharToMultiByte(CP_ACP,0,szBuff,-1,lpszBuf,MAX_PATH,0,0); 2436 } 2437 return bRet; 2438 } 2439 2440 2441 /************************************************************************* 2442 * PathCanonicalizeW [SHLWAPI.@] 2443 * 2444 * See PathCanonicalizeA. 2445 */ 2446 BOOL WINAPI PathCanonicalizeW(LPWSTR lpszBuf, LPCWSTR lpszPath) 2447 { 2448 LPWSTR lpszDst = lpszBuf; 2449 LPCWSTR lpszSrc = lpszPath; 2450 2451 TRACE("(%p,%s)\n", lpszBuf, debugstr_w(lpszPath)); 2452 2453 if (lpszBuf) 2454 *lpszDst = '\0'; 2455 2456 if (!lpszBuf || !lpszPath) 2457 { 2458 SetLastError(ERROR_INVALID_PARAMETER); 2459 return FALSE; 2460 } 2461 2462 if (!*lpszPath) 2463 { 2464 *lpszBuf++ = '\\'; 2465 *lpszBuf = '\0'; 2466 return TRUE; 2467 } 2468 2469 /* Copy path root */ 2470 if (*lpszSrc == '\\') 2471 { 2472 *lpszDst++ = *lpszSrc++; 2473 } 2474 else if (*lpszSrc && lpszSrc[1] == ':') 2475 { 2476 /* X:\ */ 2477 *lpszDst++ = *lpszSrc++; 2478 *lpszDst++ = *lpszSrc++; 2479 if (*lpszSrc == '\\') 2480 *lpszDst++ = *lpszSrc++; 2481 } 2482 2483 /* Canonicalize the rest of the path */ 2484 while (*lpszSrc) 2485 { 2486 if (*lpszSrc == '.') 2487 { 2488 if (lpszSrc[1] == '\\' && (lpszSrc == lpszPath || lpszSrc[-1] == '\\' || lpszSrc[-1] == ':')) 2489 { 2490 lpszSrc += 2; /* Skip .\ */ 2491 } 2492 else if (lpszSrc[1] == '.' && (lpszDst == lpszBuf || lpszDst[-1] == '\\')) 2493 { 2494 /* \.. backs up a directory, over the root if it has no \ following X:. 2495 * .. is ignored if it would remove a UNC server name or initial \\ 2496 */ 2497 if (lpszDst != lpszBuf) 2498 { 2499 *lpszDst = '\0'; /* Allow PathIsUNCServerShareA test on lpszBuf */ 2500 if (lpszDst > lpszBuf+1 && lpszDst[-1] == '\\' && 2501 (lpszDst[-2] != '\\' || lpszDst > lpszBuf+2)) 2502 { 2503 if (lpszDst[-2] == ':' && (lpszDst > lpszBuf+3 || lpszDst[-3] == ':')) 2504 { 2505 lpszDst -= 2; 2506 while (lpszDst > lpszBuf && *lpszDst != '\\') 2507 lpszDst--; 2508 if (*lpszDst == '\\') 2509 lpszDst++; /* Reset to last '\' */ 2510 else 2511 lpszDst = lpszBuf; /* Start path again from new root */ 2512 } 2513 else if (lpszDst[-2] != ':' && !PathIsUNCServerShareW(lpszBuf)) 2514 lpszDst -= 2; 2515 } 2516 while (lpszDst > lpszBuf && *lpszDst != '\\') 2517 lpszDst--; 2518 if (lpszDst == lpszBuf) 2519 { 2520 *lpszDst++ = '\\'; 2521 lpszSrc++; 2522 } 2523 } 2524 lpszSrc += 2; /* Skip .. in src path */ 2525 } 2526 else 2527 *lpszDst++ = *lpszSrc++; 2528 } 2529 else 2530 *lpszDst++ = *lpszSrc++; 2531 } 2532 /* Append \ to naked drive specs */ 2533 if (lpszDst - lpszBuf == 2 && lpszDst[-1] == ':') 2534 *lpszDst++ = '\\'; 2535 *lpszDst++ = '\0'; 2536 return TRUE; 2537 } 2538 2539 /************************************************************************* 2540 * PathFindNextComponentA [SHLWAPI.@] 2541 * 2542 * Find the next component in a path. 2543 * 2544 * PARAMS 2545 * lpszPath [I] Path to find next component in 2546 * 2547 * RETURNS 2548 * Success: A pointer to the next component, or the end of the string. 2549 * Failure: NULL, If lpszPath is invalid 2550 * 2551 * NOTES 2552 * A 'component' is either a backslash character (\) or UNC marker (\\). 2553 * Because of this, relative paths (e.g "c:foo") are regarded as having 2554 * only one component. 2555 */ 2556 LPSTR WINAPI PathFindNextComponentA(LPCSTR lpszPath) 2557 { 2558 LPSTR lpszSlash; 2559 2560 TRACE("(%s)\n", debugstr_a(lpszPath)); 2561 2562 if(!lpszPath || !*lpszPath) 2563 return NULL; 2564 2565 if ((lpszSlash = StrChrA(lpszPath, '\\'))) 2566 { 2567 if (lpszSlash[1] == '\\') 2568 lpszSlash++; 2569 return lpszSlash + 1; 2570 } 2571 return (LPSTR)lpszPath + strlen(lpszPath); 2572 } 2573 2574 /************************************************************************* 2575 * PathFindNextComponentW [SHLWAPI.@] 2576 * 2577 * See PathFindNextComponentA. 2578 */ 2579 LPWSTR WINAPI PathFindNextComponentW(LPCWSTR lpszPath) 2580 { 2581 LPWSTR lpszSlash; 2582 2583 TRACE("(%s)\n", debugstr_w(lpszPath)); 2584 2585 if(!lpszPath || !*lpszPath) 2586 return NULL; 2587 2588 if ((lpszSlash = StrChrW(lpszPath, '\\'))) 2589 { 2590 if (lpszSlash[1] == '\\') 2591 lpszSlash++; 2592 return lpszSlash + 1; 2593 } 2594 return (LPWSTR)lpszPath + strlenW(lpszPath); 2595 } 2596 2597 /************************************************************************* 2598 * PathAddExtensionA [SHLWAPI.@] 2599 * 2600 * Add a file extension to a path 2601 * 2602 * PARAMS 2603 * lpszPath [I/O] Path to add extension to 2604 * lpszExtension [I] Extension to add to lpszPath 2605 * 2606 * RETURNS 2607 * TRUE If the path was modified, 2608 * FALSE If lpszPath or lpszExtension are invalid, lpszPath has an 2609 * extension already, or the new path length is too big. 2610 * 2611 * FIXME 2612 * What version of shlwapi.dll adds "exe" if lpszExtension is NULL? Win2k 2613 * does not do this, so the behaviour was removed. 2614 */ 2615 BOOL WINAPI PathAddExtensionA(LPSTR lpszPath, LPCSTR lpszExtension) 2616 { 2617 size_t dwLen; 2618 2619 TRACE("(%s,%s)\n", debugstr_a(lpszPath), debugstr_a(lpszExtension)); 2620 2621 if (!lpszPath || !lpszExtension || *(PathFindExtensionA(lpszPath))) 2622 return FALSE; 2623 2624 dwLen = strlen(lpszPath); 2625 2626 if (dwLen + strlen(lpszExtension) >= MAX_PATH) 2627 return FALSE; 2628 2629 strcpy(lpszPath + dwLen, lpszExtension); 2630 return TRUE; 2631 } 2632 2633 /************************************************************************* 2634 * PathAddExtensionW [SHLWAPI.@] 2635 * 2636 * See PathAddExtensionA. 2637 */ 2638 BOOL WINAPI PathAddExtensionW(LPWSTR lpszPath, LPCWSTR lpszExtension) 2639 { 2640 size_t dwLen; 2641 2642 TRACE("(%s,%s)\n", debugstr_w(lpszPath), debugstr_w(lpszExtension)); 2643 2644 if (!lpszPath || !lpszExtension || *(PathFindExtensionW(lpszPath))) 2645 return FALSE; 2646 2647 dwLen = strlenW(lpszPath); 2648 2649 if (dwLen + strlenW(lpszExtension) >= MAX_PATH) 2650 return FALSE; 2651 2652 strcpyW(lpszPath + dwLen, lpszExtension); 2653 return TRUE; 2654 } 2655 2656 /************************************************************************* 2657 * PathMakePrettyA [SHLWAPI.@] 2658 * 2659 * Convert an uppercase DOS filename into lowercase. 2660 * 2661 * PARAMS 2662 * lpszPath [I/O] Path to convert. 2663 * 2664 * RETURNS 2665 * TRUE If the path was an uppercase DOS path and was converted, 2666 * FALSE Otherwise. 2667 */ 2668 BOOL WINAPI PathMakePrettyA(LPSTR lpszPath) 2669 { 2670 LPSTR pszIter = lpszPath; 2671 2672 TRACE("(%s)\n", debugstr_a(lpszPath)); 2673 2674 if (!pszIter) 2675 return FALSE; 2676 2677 if (*pszIter) 2678 { 2679 do 2680 { 2681 if (islower(*pszIter) || IsDBCSLeadByte(*pszIter)) 2682 return FALSE; /* Not DOS path */ 2683 pszIter++; 2684 } while (*pszIter); 2685 pszIter = lpszPath + 1; 2686 while (*pszIter) 2687 { 2688 *pszIter = tolower(*pszIter); 2689 pszIter++; 2690 } 2691 } 2692 return TRUE; 2693 } 2694 2695 /************************************************************************* 2696 * PathMakePrettyW [SHLWAPI.@] 2697 * 2698 * See PathMakePrettyA. 2699 */ 2700 BOOL WINAPI PathMakePrettyW(LPWSTR lpszPath) 2701 { 2702 LPWSTR pszIter = lpszPath; 2703 2704 TRACE("(%s)\n", debugstr_w(lpszPath)); 2705 2706 if (!pszIter) 2707 return FALSE; 2708 2709 if (*pszIter) 2710 { 2711 do 2712 { 2713 if (islowerW(*pszIter)) 2714 return FALSE; /* Not DOS path */ 2715 pszIter++; 2716 } while (*pszIter); 2717 pszIter = lpszPath + 1; 2718 while (*pszIter) 2719 { 2720 *pszIter = tolowerW(*pszIter); 2721 pszIter++; 2722 } 2723 } 2724 return TRUE; 2725 } 2726 2727 /************************************************************************* 2728 * PathCommonPrefixA [SHLWAPI.@] 2729 * 2730 * Determine the length of the common prefix between two paths. 2731 * 2732 * PARAMS 2733 * lpszFile1 [I] First path for comparison 2734 * lpszFile2 [I] Second path for comparison 2735 * achPath [O] Destination for common prefix string 2736 * 2737 * RETURNS 2738 * The length of the common prefix. This is 0 if there is no common 2739 * prefix between the paths or if any parameters are invalid. If the prefix 2740 * is non-zero and achPath is not NULL, achPath is filled with the common 2741 * part of the prefix and NUL terminated. 2742 * 2743 * NOTES 2744 * A common prefix of 2 is always returned as 3. It is thus possible for 2745 * the length returned to be invalid (i.e. Longer than one or both of the 2746 * strings given as parameters). This Win32 behaviour has been implemented 2747 * here, and cannot be changed (fixed?) without breaking other SHLWAPI calls. 2748 * To work around this when using this function, always check that the byte 2749 * at [common_prefix_len-1] is not a NUL. If it is, deduct 1 from the prefix. 2750 */ 2751 int WINAPI PathCommonPrefixA(LPCSTR lpszFile1, LPCSTR lpszFile2, LPSTR achPath) 2752 { 2753 size_t iLen = 0; 2754 LPCSTR lpszIter1 = lpszFile1; 2755 LPCSTR lpszIter2 = lpszFile2; 2756 2757 TRACE("(%s,%s,%p)\n", debugstr_a(lpszFile1), debugstr_a(lpszFile2), achPath); 2758 2759 if (achPath) 2760 *achPath = '\0'; 2761 2762 if (!lpszFile1 || !lpszFile2) 2763 return 0; 2764 2765 /* Handle roots first */ 2766 if (PathIsUNCA(lpszFile1)) 2767 { 2768 if (!PathIsUNCA(lpszFile2)) 2769 return 0; 2770 lpszIter1 += 2; 2771 lpszIter2 += 2; 2772 } 2773 else if (PathIsUNCA(lpszFile2)) 2774 return 0; /* Know already lpszFile1 is not UNC */ 2775 2776 do 2777 { 2778 /* Update len */ 2779 if ((!*lpszIter1 || *lpszIter1 == '\\') && 2780 (!*lpszIter2 || *lpszIter2 == '\\')) 2781 iLen = lpszIter1 - lpszFile1; /* Common to this point */ 2782 2783 if (!*lpszIter1 || (tolower(*lpszIter1) != tolower(*lpszIter2))) 2784 break; /* Strings differ at this point */ 2785 2786 lpszIter1++; 2787 lpszIter2++; 2788 } while (1); 2789 2790 if (iLen == 2) 2791 iLen++; /* Feature/Bug compatible with Win32 */ 2792 2793 if (iLen && achPath) 2794 { 2795 memcpy(achPath,lpszFile1,iLen); 2796 achPath[iLen] = '\0'; 2797 } 2798 return iLen; 2799 } 2800 2801 /************************************************************************* 2802 * PathCommonPrefixW [SHLWAPI.@] 2803 * 2804 * See PathCommonPrefixA. 2805 */ 2806 int WINAPI PathCommonPrefixW(LPCWSTR lpszFile1, LPCWSTR lpszFile2, LPWSTR achPath) 2807 { 2808 size_t iLen = 0; 2809 LPCWSTR lpszIter1 = lpszFile1; 2810 LPCWSTR lpszIter2 = lpszFile2; 2811 2812 TRACE("(%s,%s,%p)\n", debugstr_w(lpszFile1), debugstr_w(lpszFile2), achPath); 2813 2814 if (achPath) 2815 *achPath = '\0'; 2816 2817 if (!lpszFile1 || !lpszFile2) 2818 return 0; 2819 2820 /* Handle roots first */ 2821 if (PathIsUNCW(lpszFile1)) 2822 { 2823 if (!PathIsUNCW(lpszFile2)) 2824 return 0; 2825 lpszIter1 += 2; 2826 lpszIter2 += 2; 2827 } 2828 else if (PathIsUNCW(lpszFile2)) 2829 return 0; /* Know already lpszFile1 is not UNC */ 2830 2831 do 2832 { 2833 /* Update len */ 2834 if ((!*lpszIter1 || *lpszIter1 == '\\') && 2835 (!*lpszIter2 || *lpszIter2 == '\\')) 2836 iLen = lpszIter1 - lpszFile1; /* Common to this point */ 2837 2838 if (!*lpszIter1 || (tolowerW(*lpszIter1) != tolowerW(*lpszIter2))) 2839 break; /* Strings differ at this point */ 2840 2841 lpszIter1++; 2842 lpszIter2++; 2843 } while (1); 2844 2845 if (iLen == 2) 2846 iLen++; /* Feature/Bug compatible with Win32 */ 2847 2848 if (iLen && achPath) 2849 { 2850 memcpy(achPath,lpszFile1,iLen * sizeof(WCHAR)); 2851 achPath[iLen] = '\0'; 2852 } 2853 return iLen; 2854 } 2855 2856 /************************************************************************* 2857 * PathCompactPathA [SHLWAPI.@] 2858 * 2859 * Make a path fit into a given width when printed to a DC. 2860 * 2861 * PARAMS 2862 * hDc [I] Destination DC 2863 * lpszPath [I/O] Path to be printed to hDc 2864 * dx [I] Desired width 2865 * 2866 * RETURNS 2867 * TRUE If the path was modified/went well. 2868 * FALSE Otherwise. 2869 */ 2870 BOOL WINAPI PathCompactPathA(HDC hDC, LPSTR lpszPath, UINT dx) 2871 { 2872 BOOL bRet = FALSE; 2873 2874 TRACE("(%p,%s,%d)\n", hDC, debugstr_a(lpszPath), dx); 2875 2876 if (lpszPath) 2877 { 2878 WCHAR szPath[MAX_PATH]; 2879 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 2880 bRet = PathCompactPathW(hDC, szPath, dx); 2881 WideCharToMultiByte(CP_ACP,0,szPath,-1,lpszPath,MAX_PATH,0,0); 2882 } 2883 return bRet; 2884 } 2885 2886 /************************************************************************* 2887 * PathCompactPathW [SHLWAPI.@] 2888 * 2889 * See PathCompactPathA. 2890 */ 2891 BOOL WINAPI PathCompactPathW(HDC hDC, LPWSTR lpszPath, UINT dx) 2892 { 2893 static const WCHAR szEllipses[] = { '.', '.', '.', '\0' }; 2894 BOOL bRet = TRUE; 2895 HDC hdc = 0; 2896 WCHAR buff[MAX_PATH]; 2897 SIZE size; 2898 DWORD dwLen; 2899 2900 TRACE("(%p,%s,%d)\n", hDC, debugstr_w(lpszPath), dx); 2901 2902 if (!lpszPath) 2903 return FALSE; 2904 2905 if (!hDC) 2906 hdc = hDC = GetDC(0); 2907 2908 /* Get the length of the whole path */ 2909 dwLen = strlenW(lpszPath); 2910 GetTextExtentPointW(hDC, lpszPath, dwLen, &size); 2911 2912 if ((UINT)size.cx > dx) 2913 { 2914 /* Path too big, must reduce it */ 2915 LPWSTR sFile; 2916 DWORD dwEllipsesLen = 0, dwPathLen = 0; 2917 2918 sFile = PathFindFileNameW(lpszPath); 2919 if (sFile != lpszPath) sFile--; 2920 2921 /* Get the size of ellipses */ 2922 GetTextExtentPointW(hDC, szEllipses, 3, &size); 2923 dwEllipsesLen = size.cx; 2924 /* Get the size of the file name */ 2925 GetTextExtentPointW(hDC, sFile, strlenW(sFile), &size); 2926 dwPathLen = size.cx; 2927 2928 if (sFile != lpszPath) 2929 { 2930 LPWSTR sPath = sFile; 2931 BOOL bEllipses = FALSE; 2932 2933 /* The path includes a file name. Include as much of the path prior to 2934 * the file name as possible, allowing for the ellipses, e.g: 2935 * c:\some very long path\filename ==> c:\some v...\filename 2936 */ 2937 lstrcpynW(buff, sFile, MAX_PATH); 2938 2939 do 2940 { 2941 DWORD dwTotalLen = bEllipses? dwPathLen + dwEllipsesLen : dwPathLen; 2942 2943 GetTextExtentPointW(hDC, lpszPath, sPath - lpszPath, &size); 2944 dwTotalLen += size.cx; 2945 if (dwTotalLen <= dx) 2946 break; 2947 sPath--; 2948 if (!bEllipses) 2949 { 2950 bEllipses = TRUE; 2951 sPath -= 2; 2952 } 2953 } while (sPath > lpszPath); 2954 2955 if (sPath > lpszPath) 2956 { 2957 if (bEllipses) 2958 { 2959 strcpyW(sPath, szEllipses); 2960 strcpyW(sPath+3, buff); 2961 } 2962 bRet = TRUE; 2963 goto end; 2964 } 2965 strcpyW(lpszPath, szEllipses); 2966 strcpyW(lpszPath+3, buff); 2967 bRet = FALSE; 2968 goto end; 2969 } 2970 2971 /* Trim the path by adding ellipses to the end, e.g: 2972 * A very long file name.txt ==> A very... 2973 */ 2974 dwLen = strlenW(lpszPath); 2975 2976 if (dwLen > MAX_PATH - 3) 2977 dwLen = MAX_PATH - 3; 2978 lstrcpynW(buff, sFile, dwLen); 2979 2980 do { 2981 dwLen--; 2982 GetTextExtentPointW(hDC, buff, dwLen, &size); 2983 } while (dwLen && size.cx + dwEllipsesLen > dx); 2984 2985 if (!dwLen) 2986 { 2987 DWORD dwWritten = 0; 2988 2989 dwEllipsesLen /= 3; /* Size of a single '.' */ 2990 2991 /* Write as much of the Ellipses string as possible */ 2992 while (dwWritten + dwEllipsesLen < dx && dwLen < 3) 2993 { 2994 *lpszPath++ = '.'; 2995 dwWritten += dwEllipsesLen; 2996 dwLen++; 2997 } 2998 *lpszPath = '\0'; 2999 bRet = FALSE; 3000 } 3001 else 3002 { 3003 strcpyW(buff + dwLen, szEllipses); 3004 strcpyW(lpszPath, buff); 3005 } 3006 } 3007 3008 end: 3009 if (hdc) 3010 ReleaseDC(0, hdc); 3011 3012 return bRet; 3013 } 3014 3015 /************************************************************************* 3016 * PathGetCharTypeA [SHLWAPI.@] 3017 * 3018 * Categorise a character from a file path. 3019 * 3020 * PARAMS 3021 * ch [I] Character to get the type of 3022 * 3023 * RETURNS 3024 * A set of GCT_ bit flags (from "shlwapi.h") indicating the character type. 3025 */ 3026 UINT WINAPI PathGetCharTypeA(UCHAR ch) 3027 { 3028 return PathGetCharTypeW(ch); 3029 } 3030 3031 /************************************************************************* 3032 * PathGetCharTypeW [SHLWAPI.@] 3033 * 3034 * See PathGetCharTypeA. 3035 */ 3036 UINT WINAPI PathGetCharTypeW(WCHAR ch) 3037 { 3038 UINT flags = 0; 3039 3040 TRACE("(%d)\n", ch); 3041 3042 if (!ch || ch < ' ' || ch == '<' || ch == '>' || 3043 ch == '"' || ch == '|' || ch == '/') 3044 flags = GCT_INVALID; /* Invalid */ 3045 else if (ch == '*' || ch=='?') 3046 flags = GCT_WILD; /* Wildchars */ 3047 else if ((ch == '\\') || (ch == ':')) 3048 return GCT_SEPARATOR; /* Path separators */ 3049 else 3050 { 3051 if (ch < 126) 3052 { 3053 if (((ch & 0x1) && ch != ';') || !ch || isalnum(ch) || ch == '$' || ch == '&' || ch == '(' || 3054 ch == '.' || ch == '@' || ch == '^' || 3055 ch == '\'' || ch == 130 || ch == '`') 3056 flags |= GCT_SHORTCHAR; /* All these are valid for DOS */ 3057 } 3058 else 3059 flags |= GCT_SHORTCHAR; /* Bug compatible with win32 */ 3060 flags |= GCT_LFNCHAR; /* Valid for long file names */ 3061 } 3062 return flags; 3063 } 3064 3065 /************************************************************************* 3066 * SHLWAPI_UseSystemForSystemFolders 3067 * 3068 * Internal helper for PathMakeSystemFolderW. 3069 */ 3070 static BOOL SHLWAPI_UseSystemForSystemFolders(void) 3071 { 3072 static BOOL bCheckedReg = FALSE; 3073 static BOOL bUseSystemForSystemFolders = FALSE; 3074 3075 if (!bCheckedReg) 3076 { 3077 bCheckedReg = TRUE; 3078 3079 /* Key tells Win what file attributes to use on system folders */ 3080 if (SHGetValueA(HKEY_LOCAL_MACHINE, 3081 "Software\\Microsoft\\Windows\\CurrentVersion\\Explorer", 3082 "UseSystemForSystemFolders", 0, 0, 0)) 3083 bUseSystemForSystemFolders = TRUE; 3084 } 3085 return bUseSystemForSystemFolders; 3086 } 3087 3088 /************************************************************************* 3089 * PathMakeSystemFolderA [SHLWAPI.@] 3090 * 3091 * Set system folder attribute for a path. 3092 * 3093 * PARAMS 3094 * lpszPath [I] The path to turn into a system folder 3095 * 3096 * RETURNS 3097 * TRUE If the path was changed to/already was a system folder 3098 * FALSE If the path is invalid or SetFileAttributesA() fails 3099 */ 3100 BOOL WINAPI PathMakeSystemFolderA(LPCSTR lpszPath) 3101 { 3102 BOOL bRet = FALSE; 3103 3104 TRACE("(%s)\n", debugstr_a(lpszPath)); 3105 3106 if (lpszPath && *lpszPath) 3107 { 3108 WCHAR szPath[MAX_PATH]; 3109 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 3110 bRet = PathMakeSystemFolderW(szPath); 3111 } 3112 return bRet; 3113 } 3114 3115 /************************************************************************* 3116 * PathMakeSystemFolderW [SHLWAPI.@] 3117 * 3118 * See PathMakeSystemFolderA. 3119 */ 3120 BOOL WINAPI PathMakeSystemFolderW(LPCWSTR lpszPath) 3121 { 3122 DWORD dwDefaultAttr = FILE_ATTRIBUTE_READONLY, dwAttr; 3123 WCHAR buff[MAX_PATH]; 3124 3125 TRACE("(%s)\n", debugstr_w(lpszPath)); 3126 3127 if (!lpszPath || !*lpszPath) 3128 return FALSE; 3129 3130 /* If the directory is already a system directory, don't do anything */ 3131 GetSystemDirectoryW(buff, MAX_PATH); 3132 if (!strcmpW(buff, lpszPath)) 3133 return TRUE; 3134 3135 GetWindowsDirectoryW(buff, MAX_PATH); 3136 if (!strcmpW(buff, lpszPath)) 3137 return TRUE; 3138 3139 /* "UseSystemForSystemFolders" Tells Win what attributes to use */ 3140 if (SHLWAPI_UseSystemForSystemFolders()) 3141 dwDefaultAttr = FILE_ATTRIBUTE_SYSTEM; 3142 3143 if ((dwAttr = GetFileAttributesW(lpszPath)) == INVALID_FILE_ATTRIBUTES) 3144 return FALSE; 3145 3146 /* Change file attributes to system attributes */ 3147 dwAttr &= ~(FILE_ATTRIBUTE_SYSTEM|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_READONLY); 3148 return SetFileAttributesW(lpszPath, dwAttr | dwDefaultAttr); 3149 } 3150 3151 /************************************************************************* 3152 * PathRenameExtensionA [SHLWAPI.@] 3153 * 3154 * Swap the file extension in a path with another extension. 3155 * 3156 * PARAMS 3157 * lpszPath [I/O] Path to swap the extension in 3158 * lpszExt [I] The new extension 3159 * 3160 * RETURNS 3161 * TRUE if lpszPath was modified, 3162 * FALSE if lpszPath or lpszExt is NULL, or the new path is too long 3163 */ 3164 BOOL WINAPI PathRenameExtensionA(LPSTR lpszPath, LPCSTR lpszExt) 3165 { 3166 LPSTR lpszExtension; 3167 3168 TRACE("(%s,%s)\n", debugstr_a(lpszPath), debugstr_a(lpszExt)); 3169 3170 lpszExtension = PathFindExtensionA(lpszPath); 3171 3172 if (!lpszExtension || (lpszExtension - lpszPath + strlen(lpszExt) >= MAX_PATH)) 3173 return FALSE; 3174 3175 strcpy(lpszExtension, lpszExt); 3176 return TRUE; 3177 } 3178 3179 /************************************************************************* 3180 * PathRenameExtensionW [SHLWAPI.@] 3181 * 3182 * See PathRenameExtensionA. 3183 */ 3184 BOOL WINAPI PathRenameExtensionW(LPWSTR lpszPath, LPCWSTR lpszExt) 3185 { 3186 LPWSTR lpszExtension; 3187 3188 TRACE("(%s,%s)\n", debugstr_w(lpszPath), debugstr_w(lpszExt)); 3189 3190 lpszExtension = PathFindExtensionW(lpszPath); 3191 3192 if (!lpszExtension || (lpszExtension - lpszPath + strlenW(lpszExt) >= MAX_PATH)) 3193 return FALSE; 3194 3195 strcpyW(lpszExtension, lpszExt); 3196 return TRUE; 3197 } 3198 3199 /************************************************************************* 3200 * PathSearchAndQualifyA [SHLWAPI.@] 3201 * 3202 * Determine if a given path is correct and fully qualified. 3203 * 3204 * PARAMS 3205 * lpszPath [I] Path to check 3206 * lpszBuf [O] Output for correct path 3207 * cchBuf [I] Size of lpszBuf 3208 * 3209 * RETURNS 3210 * Unknown. 3211 */ 3212 BOOL WINAPI PathSearchAndQualifyA(LPCSTR lpszPath, LPSTR lpszBuf, UINT cchBuf) 3213 { 3214 TRACE("(%s,%p,0x%08x)\n", debugstr_a(lpszPath), lpszBuf, cchBuf); 3215 3216 if(SearchPathA(NULL, lpszPath, NULL, cchBuf, lpszBuf, NULL)) 3217 return TRUE; 3218 return !!GetFullPathNameA(lpszPath, cchBuf, lpszBuf, NULL); 3219 } 3220 3221 /************************************************************************* 3222 * PathSearchAndQualifyW [SHLWAPI.@] 3223 * 3224 * See PathSearchAndQualifyA. 3225 */ 3226 BOOL WINAPI PathSearchAndQualifyW(LPCWSTR lpszPath, LPWSTR lpszBuf, UINT cchBuf) 3227 { 3228 TRACE("(%s,%p,0x%08x)\n", debugstr_w(lpszPath), lpszBuf, cchBuf); 3229 3230 if(SearchPathW(NULL, lpszPath, NULL, cchBuf, lpszBuf, NULL)) 3231 return TRUE; 3232 return !!GetFullPathNameW(lpszPath, cchBuf, lpszBuf, NULL); 3233 } 3234 3235 /************************************************************************* 3236 * PathSkipRootA [SHLWAPI.@] 3237 * 3238 * Return the portion of a path following the drive letter or mount point. 3239 * 3240 * PARAMS 3241 * lpszPath [I] The path to skip on 3242 * 3243 * RETURNS 3244 * Success: A pointer to the next character after the root. 3245 * Failure: NULL, if lpszPath is invalid, has no root or is a multibyte string. 3246 */ 3247 LPSTR WINAPI PathSkipRootA(LPCSTR lpszPath) 3248 { 3249 TRACE("(%s)\n", debugstr_a(lpszPath)); 3250 3251 if (!lpszPath || !*lpszPath) 3252 return NULL; 3253 3254 if (*lpszPath == '\\' && lpszPath[1] == '\\') 3255 { 3256 /* Network share: skip share server and mount point */ 3257 lpszPath += 2; 3258 if ((lpszPath = StrChrA(lpszPath, '\\')) && 3259 (lpszPath = StrChrA(lpszPath + 1, '\\'))) 3260 lpszPath++; 3261 return (LPSTR)lpszPath; 3262 } 3263 3264 if (IsDBCSLeadByte(*lpszPath)) 3265 return NULL; 3266 3267 /* Check x:\ */ 3268 if (lpszPath[0] && lpszPath[1] == ':' && lpszPath[2] == '\\') 3269 return (LPSTR)lpszPath + 3; 3270 return NULL; 3271 } 3272 3273 /************************************************************************* 3274 * PathSkipRootW [SHLWAPI.@] 3275 * 3276 * See PathSkipRootA. 3277 */ 3278 LPWSTR WINAPI PathSkipRootW(LPCWSTR lpszPath) 3279 { 3280 TRACE("(%s)\n", debugstr_w(lpszPath)); 3281 3282 if (!lpszPath || !*lpszPath) 3283 return NULL; 3284 3285 if (*lpszPath == '\\' && lpszPath[1] == '\\') 3286 { 3287 /* Network share: skip share server and mount point */ 3288 lpszPath += 2; 3289 if ((lpszPath = StrChrW(lpszPath, '\\')) && 3290 (lpszPath = StrChrW(lpszPath + 1, '\\'))) 3291 lpszPath++; 3292 return (LPWSTR)lpszPath; 3293 } 3294 3295 /* Check x:\ */ 3296 if (lpszPath[0] && lpszPath[1] == ':' && lpszPath[2] == '\\') 3297 return (LPWSTR)lpszPath + 3; 3298 return NULL; 3299 } 3300 3301 /************************************************************************* 3302 * PathCreateFromUrlA [SHLWAPI.@] 3303 * 3304 * See PathCreateFromUrlW 3305 */ 3306 HRESULT WINAPI PathCreateFromUrlA(LPCSTR pszUrl, LPSTR pszPath, 3307 LPDWORD pcchPath, DWORD dwReserved) 3308 { 3309 WCHAR bufW[MAX_PATH]; 3310 WCHAR *pathW = bufW; 3311 UNICODE_STRING urlW; 3312 HRESULT ret; 3313 DWORD lenW = sizeof(bufW)/sizeof(WCHAR), lenA; 3314 3315 if (!pszUrl || !pszPath || !pcchPath || !*pcchPath) 3316 return E_INVALIDARG; 3317 3318 if(!RtlCreateUnicodeStringFromAsciiz(&urlW, pszUrl)) 3319 return E_INVALIDARG; 3320 if((ret = PathCreateFromUrlW(urlW.Buffer, pathW, &lenW, dwReserved)) == E_POINTER) { 3321 pathW = HeapAlloc(GetProcessHeap(), 0, lenW * sizeof(WCHAR)); 3322 ret = PathCreateFromUrlW(urlW.Buffer, pathW, &lenW, dwReserved); 3323 } 3324 if(ret == S_OK) { 3325 RtlUnicodeToMultiByteSize(&lenA, pathW, lenW * sizeof(WCHAR)); 3326 if(*pcchPath > lenA) { 3327 RtlUnicodeToMultiByteN(pszPath, *pcchPath - 1, &lenA, pathW, lenW * sizeof(WCHAR)); 3328 pszPath[lenA] = 0; 3329 *pcchPath = lenA; 3330 } else { 3331 *pcchPath = lenA + 1; 3332 ret = E_POINTER; 3333 } 3334 } 3335 if(pathW != bufW) HeapFree(GetProcessHeap(), 0, pathW); 3336 RtlFreeUnicodeString(&urlW); 3337 return ret; 3338 } 3339 3340 /************************************************************************* 3341 * PathCreateFromUrlW [SHLWAPI.@] 3342 * 3343 * Create a path from a URL 3344 * 3345 * PARAMS 3346 * lpszUrl [I] URL to convert into a path 3347 * lpszPath [O] Output buffer for the resulting Path 3348 * pcchPath [I] Length of lpszPath 3349 * dwFlags [I] Flags controlling the conversion 3350 * 3351 * RETURNS 3352 * Success: S_OK. lpszPath contains the URL in path format, 3353 * Failure: An HRESULT error code such as E_INVALIDARG. 3354 */ 3355 HRESULT WINAPI PathCreateFromUrlW(LPCWSTR pszUrl, LPWSTR pszPath, 3356 LPDWORD pcchPath, DWORD dwReserved) 3357 { 3358 static const WCHAR file_colon[] = { 'f','i','l','e',':',0 }; 3359 static const WCHAR localhost[] = { 'l','o','c','a','l','h','o','s','t',0 }; 3360 DWORD nslashes, unescape, len; 3361 const WCHAR *src; 3362 WCHAR *tpath, *dst; 3363 HRESULT ret; 3364 3365 TRACE("(%s,%p,%p,0x%08x)\n", debugstr_w(pszUrl), pszPath, pcchPath, dwReserved); 3366 3367 if (!pszUrl || !pszPath || !pcchPath || !*pcchPath) 3368 return E_INVALIDARG; 3369 3370 if (lstrlenW(pszUrl) < 5) 3371 return E_INVALIDARG; 3372 3373 if (CompareStringW(LOCALE_INVARIANT, NORM_IGNORECASE, pszUrl, 5, 3374 file_colon, 5) != CSTR_EQUAL) 3375 return E_INVALIDARG; 3376 pszUrl += 5; 3377 ret = S_OK; 3378 3379 src = pszUrl; 3380 nslashes = 0; 3381 while (*src == '/' || *src == '\\') { 3382 nslashes++; 3383 src++; 3384 } 3385 3386 /* We need a temporary buffer so we can compute what size to ask for. 3387 * We know that the final string won't be longer than the current pszUrl 3388 * plus at most two backslashes. All the other transformations make it 3389 * shorter. 3390 */ 3391 len = 2 + lstrlenW(pszUrl) + 1; 3392 if (*pcchPath < len) 3393 tpath = HeapAlloc(GetProcessHeap(), 0, len * sizeof(WCHAR)); 3394 else 3395 tpath = pszPath; 3396 3397 len = 0; 3398 dst = tpath; 3399 unescape = 1; 3400 switch (nslashes) 3401 { 3402 case 0: 3403 /* 'file:' + escaped DOS path */ 3404 break; 3405 case 1: 3406 /* 'file:/' + escaped DOS path */ 3407 /* fall through */ 3408 case 3: 3409 /* 'file:///' (implied localhost) + escaped DOS path */ 3410 if (!isalphaW(*src) || (src[1] != ':' && src[1] != '|')) 3411 src -= 1; 3412 break; 3413 case 2: 3414 if (lstrlenW(src) >= 10 && CompareStringW(LOCALE_INVARIANT, NORM_IGNORECASE, 3415 src, 9, localhost, 9) == CSTR_EQUAL && (src[9] == '/' || src[9] == '\\')) 3416 { 3417 /* 'file://localhost/' + escaped DOS path */ 3418 src += 10; 3419 } 3420 else if (isalphaW(*src) && (src[1] == ':' || src[1] == '|')) 3421 { 3422 /* 'file://' + unescaped DOS path */ 3423 unescape = 0; 3424 } 3425 else 3426 { 3427 /* 'file://hostname:port/path' (where path is escaped) 3428 * or 'file:' + escaped UNC path (\\server\share\path) 3429 * The second form is clearly specific to Windows and it might 3430 * even be doing a network lookup to try to figure it out. 3431 */ 3432 while (*src && *src != '/' && *src != '\\') 3433 src++; 3434 len = src - pszUrl; 3435 StrCpyNW(dst, pszUrl, len + 1); 3436 dst += len; 3437 if (*src && isalphaW(src[1]) && (src[2] == ':' || src[2] == '|')) 3438 { 3439 /* 'Forget' to add a trailing '/', just like Windows */ 3440 src++; 3441 } 3442 } 3443 break; 3444 case 4: 3445 /* 'file://' + unescaped UNC path (\\server\share\path) */ 3446 unescape = 0; 3447 if (isalphaW(*src) && (src[1] == ':' || src[1] == '|')) 3448 break; 3449 /* fall through */ 3450 default: 3451 /* 'file:/...' + escaped UNC path (\\server\share\path) */ 3452 src -= 2; 3453 } 3454 3455 /* Copy the remainder of the path */ 3456 len += lstrlenW(src); 3457 StrCpyW(dst, src); 3458 3459 /* First do the Windows-specific path conversions */ 3460 for (dst = tpath; *dst; dst++) 3461 if (*dst == '/') *dst = '\\'; 3462 if (isalphaW(*tpath) && tpath[1] == '|') 3463 tpath[1] = ':'; /* c| -> c: */ 3464 3465 /* And only then unescape the path (i.e. escaped slashes are left as is) */ 3466 if (unescape) 3467 { 3468 ret = UrlUnescapeW(tpath, NULL, &len, URL_UNESCAPE_INPLACE); 3469 if (ret == S_OK) 3470 { 3471 /* When working in-place UrlUnescapeW() does not set len */ 3472 len = lstrlenW(tpath); 3473 } 3474 } 3475 3476 if (*pcchPath < len + 1) 3477 { 3478 ret = E_POINTER; 3479 *pcchPath = len + 1; 3480 } 3481 else 3482 { 3483 *pcchPath = len; 3484 if (tpath != pszPath) 3485 StrCpyW(pszPath, tpath); 3486 } 3487 if (tpath != pszPath) 3488 HeapFree(GetProcessHeap(), 0, tpath); 3489 3490 TRACE("Returning (%u) %s\n", *pcchPath, debugstr_w(pszPath)); 3491 return ret; 3492 } 3493 3494 /************************************************************************* 3495 * PathCreateFromUrlAlloc [SHLWAPI.@] 3496 */ 3497 HRESULT WINAPI PathCreateFromUrlAlloc(LPCWSTR pszUrl, LPWSTR *pszPath, 3498 DWORD dwReserved) 3499 { 3500 WCHAR pathW[MAX_PATH]; 3501 DWORD size; 3502 HRESULT hr; 3503 3504 size = MAX_PATH; 3505 hr = PathCreateFromUrlW(pszUrl, pathW, &size, dwReserved); 3506 if (SUCCEEDED(hr)) 3507 { 3508 /* Yes, this is supposed to crash if pszPath is NULL */ 3509 *pszPath = StrDupW(pathW); 3510 } 3511 return hr; 3512 } 3513 3514 /************************************************************************* 3515 * PathRelativePathToA [SHLWAPI.@] 3516 * 3517 * Create a relative path from one path to another. 3518 * 3519 * PARAMS 3520 * lpszPath [O] Destination for relative path 3521 * lpszFrom [I] Source path 3522 * dwAttrFrom [I] File attribute of source path 3523 * lpszTo [I] Destination path 3524 * dwAttrTo [I] File attributes of destination path 3525 * 3526 * RETURNS 3527 * TRUE If a relative path can be formed. lpszPath contains the new path 3528 * FALSE If the paths are not relative or any parameters are invalid 3529 * 3530 * NOTES 3531 * lpszTo should be at least MAX_PATH in length. 3532 * 3533 * Calling this function with relative paths for lpszFrom or lpszTo may 3534 * give erroneous results. 3535 * 3536 * The Win32 version of this function contains a bug where the lpszTo string 3537 * may be referenced 1 byte beyond the end of the string. As a result random 3538 * garbage may be written to the output path, depending on what lies beyond 3539 * the last byte of the string. This bug occurs because of the behaviour of 3540 * PathCommonPrefix() (see notes for that function), and no workaround seems 3541 * possible with Win32. 3542 * 3543 * This bug has been fixed here, so for example the relative path from "\\" 3544 * to "\\" is correctly determined as "." in this implementation. 3545 */ 3546 BOOL WINAPI PathRelativePathToA(LPSTR lpszPath, LPCSTR lpszFrom, DWORD dwAttrFrom, 3547 LPCSTR lpszTo, DWORD dwAttrTo) 3548 { 3549 BOOL bRet = FALSE; 3550 3551 TRACE("(%p,%s,0x%08x,%s,0x%08x)\n", lpszPath, debugstr_a(lpszFrom), 3552 dwAttrFrom, debugstr_a(lpszTo), dwAttrTo); 3553 3554 if(lpszPath && lpszFrom && lpszTo) 3555 { 3556 WCHAR szPath[MAX_PATH]; 3557 WCHAR szFrom[MAX_PATH]; 3558 WCHAR szTo[MAX_PATH]; 3559 MultiByteToWideChar(CP_ACP,0,lpszFrom,-1,szFrom,MAX_PATH); 3560 MultiByteToWideChar(CP_ACP,0,lpszTo,-1,szTo,MAX_PATH); 3561 bRet = PathRelativePathToW(szPath,szFrom,dwAttrFrom,szTo,dwAttrTo); 3562 WideCharToMultiByte(CP_ACP,0,szPath,-1,lpszPath,MAX_PATH,0,0); 3563 } 3564 return bRet; 3565 } 3566 3567 /************************************************************************* 3568 * PathRelativePathToW [SHLWAPI.@] 3569 * 3570 * See PathRelativePathToA. 3571 */ 3572 BOOL WINAPI PathRelativePathToW(LPWSTR lpszPath, LPCWSTR lpszFrom, DWORD dwAttrFrom, 3573 LPCWSTR lpszTo, DWORD dwAttrTo) 3574 { 3575 static const WCHAR szPrevDirSlash[] = { '.', '.', '\\', '\0' }; 3576 static const WCHAR szPrevDir[] = { '.', '.', '\0' }; 3577 WCHAR szFrom[MAX_PATH]; 3578 WCHAR szTo[MAX_PATH]; 3579 DWORD dwLen; 3580 3581 TRACE("(%p,%s,0x%08x,%s,0x%08x)\n", lpszPath, debugstr_w(lpszFrom), 3582 dwAttrFrom, debugstr_w(lpszTo), dwAttrTo); 3583 3584 if(!lpszPath || !lpszFrom || !lpszTo) 3585 return FALSE; 3586 3587 *lpszPath = '\0'; 3588 lstrcpynW(szFrom, lpszFrom, MAX_PATH); 3589 lstrcpynW(szTo, lpszTo, MAX_PATH); 3590 3591 if(!(dwAttrFrom & FILE_ATTRIBUTE_DIRECTORY)) 3592 PathRemoveFileSpecW(szFrom); 3593 if(!(dwAttrTo & FILE_ATTRIBUTE_DIRECTORY)) 3594 PathRemoveFileSpecW(szTo); 3595 3596 /* Paths can only be relative if they have a common root */ 3597 if(!(dwLen = PathCommonPrefixW(szFrom, szTo, 0))) 3598 return FALSE; 3599 3600 /* Strip off lpszFrom components to the root, by adding "..\" */ 3601 lpszFrom = szFrom + dwLen; 3602 if (!*lpszFrom) 3603 { 3604 lpszPath[0] = '.'; 3605 lpszPath[1] = '\0'; 3606 } 3607 if (*lpszFrom == '\\') 3608 lpszFrom++; 3609 3610 while (*lpszFrom) 3611 { 3612 lpszFrom = PathFindNextComponentW(lpszFrom); 3613 strcatW(lpszPath, *lpszFrom ? szPrevDirSlash : szPrevDir); 3614 } 3615 3616 /* From the root add the components of lpszTo */ 3617 lpszTo += dwLen; 3618 /* We check lpszTo[-1] to avoid skipping end of string. See the notes for 3619 * this function. 3620 */ 3621 if (*lpszTo && lpszTo[-1]) 3622 { 3623 if (*lpszTo != '\\') 3624 lpszTo--; 3625 dwLen = strlenW(lpszPath); 3626 if (dwLen + strlenW(lpszTo) >= MAX_PATH) 3627 { 3628 *lpszPath = '\0'; 3629 return FALSE; 3630 } 3631 strcpyW(lpszPath + dwLen, lpszTo); 3632 } 3633 return TRUE; 3634 } 3635 3636 /************************************************************************* 3637 * PathUnmakeSystemFolderA [SHLWAPI.@] 3638 * 3639 * Remove the system folder attributes from a path. 3640 * 3641 * PARAMS 3642 * lpszPath [I] The path to remove attributes from 3643 * 3644 * RETURNS 3645 * Success: TRUE. 3646 * Failure: FALSE, if lpszPath is NULL, empty, not a directory, or calling 3647 * SetFileAttributesA() fails. 3648 */ 3649 BOOL WINAPI PathUnmakeSystemFolderA(LPCSTR lpszPath) 3650 { 3651 DWORD dwAttr; 3652 3653 TRACE("(%s)\n", debugstr_a(lpszPath)); 3654 3655 if (!lpszPath || !*lpszPath || (dwAttr = GetFileAttributesA(lpszPath)) == INVALID_FILE_ATTRIBUTES || 3656 !(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) 3657 return FALSE; 3658 3659 dwAttr &= ~(FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM); 3660 return SetFileAttributesA(lpszPath, dwAttr); 3661 } 3662 3663 /************************************************************************* 3664 * PathUnmakeSystemFolderW [SHLWAPI.@] 3665 * 3666 * See PathUnmakeSystemFolderA. 3667 */ 3668 BOOL WINAPI PathUnmakeSystemFolderW(LPCWSTR lpszPath) 3669 { 3670 DWORD dwAttr; 3671 3672 TRACE("(%s)\n", debugstr_w(lpszPath)); 3673 3674 if (!lpszPath || !*lpszPath || (dwAttr = GetFileAttributesW(lpszPath)) == INVALID_FILE_ATTRIBUTES || 3675 !(dwAttr & FILE_ATTRIBUTE_DIRECTORY)) 3676 return FALSE; 3677 3678 dwAttr &= ~(FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM); 3679 return SetFileAttributesW(lpszPath, dwAttr); 3680 } 3681 3682 3683 /************************************************************************* 3684 * PathSetDlgItemPathA [SHLWAPI.@] 3685 * 3686 * Set the text of a dialog item to a path, shrinking the path to fit 3687 * if it is too big for the item. 3688 * 3689 * PARAMS 3690 * hDlg [I] Dialog handle 3691 * id [I] ID of item in the dialog 3692 * lpszPath [I] Path to set as the items text 3693 * 3694 * RETURNS 3695 * Nothing. 3696 * 3697 * NOTES 3698 * If lpszPath is NULL, a blank string ("") is set (i.e. The previous 3699 * window text is erased). 3700 */ 3701 VOID WINAPI PathSetDlgItemPathA(HWND hDlg, int id, LPCSTR lpszPath) 3702 { 3703 WCHAR szPath[MAX_PATH]; 3704 3705 TRACE("(%p,%8x,%s)\n",hDlg, id, debugstr_a(lpszPath)); 3706 3707 if (lpszPath) 3708 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 3709 else 3710 szPath[0] = '\0'; 3711 PathSetDlgItemPathW(hDlg, id, szPath); 3712 } 3713 3714 /************************************************************************* 3715 * PathSetDlgItemPathW [SHLWAPI.@] 3716 * 3717 * See PathSetDlgItemPathA. 3718 */ 3719 VOID WINAPI PathSetDlgItemPathW(HWND hDlg, int id, LPCWSTR lpszPath) 3720 { 3721 WCHAR path[MAX_PATH + 1]; 3722 HWND hwItem; 3723 RECT rect; 3724 HDC hdc; 3725 HGDIOBJ hPrevObj; 3726 3727 TRACE("(%p,%8x,%s)\n",hDlg, id, debugstr_w(lpszPath)); 3728 3729 if (!(hwItem = GetDlgItem(hDlg, id))) 3730 return; 3731 3732 if (lpszPath) 3733 lstrcpynW(path, lpszPath, sizeof(path) / sizeof(WCHAR)); 3734 else 3735 path[0] = '\0'; 3736 3737 GetClientRect(hwItem, &rect); 3738 hdc = GetDC(hDlg); 3739 hPrevObj = SelectObject(hdc, (HGDIOBJ)SendMessageW(hwItem,WM_GETFONT,0,0)); 3740 3741 if (hPrevObj) 3742 { 3743 PathCompactPathW(hdc, path, rect.right); 3744 SelectObject(hdc, hPrevObj); 3745 } 3746 3747 ReleaseDC(hDlg, hdc); 3748 SetWindowTextW(hwItem, path); 3749 } 3750 3751 /************************************************************************* 3752 * PathIsNetworkPathA [SHLWAPI.@] 3753 * 3754 * Determine if the given path is a network path. 3755 * 3756 * PARAMS 3757 * lpszPath [I] Path to check 3758 * 3759 * RETURNS 3760 * TRUE If lpszPath is a UNC share or mapped network drive, or 3761 * FALSE If lpszPath is a local drive or cannot be determined 3762 */ 3763 BOOL WINAPI PathIsNetworkPathA(LPCSTR lpszPath) 3764 { 3765 int dwDriveNum; 3766 3767 TRACE("(%s)\n",debugstr_a(lpszPath)); 3768 3769 if (!lpszPath) 3770 return FALSE; 3771 if (*lpszPath == '\\' && lpszPath[1] == '\\') 3772 return TRUE; 3773 dwDriveNum = PathGetDriveNumberA(lpszPath); 3774 if (dwDriveNum == -1) 3775 return FALSE; 3776 #ifdef __REACTOS__ 3777 return IsNetDrive(dwDriveNum); 3778 #else 3779 GET_FUNC(pIsNetDrive, shell32, (LPCSTR)66, FALSE); /* ord 66 = shell32.IsNetDrive */ 3780 return pIsNetDrive(dwDriveNum); 3781 #endif 3782 } 3783 3784 /************************************************************************* 3785 * PathIsNetworkPathW [SHLWAPI.@] 3786 * 3787 * See PathIsNetworkPathA. 3788 */ 3789 BOOL WINAPI PathIsNetworkPathW(LPCWSTR lpszPath) 3790 { 3791 int dwDriveNum; 3792 3793 TRACE("(%s)\n", debugstr_w(lpszPath)); 3794 3795 if (!lpszPath) 3796 return FALSE; 3797 if (*lpszPath == '\\' && lpszPath[1] == '\\') 3798 return TRUE; 3799 dwDriveNum = PathGetDriveNumberW(lpszPath); 3800 if (dwDriveNum == -1) 3801 return FALSE; 3802 #ifdef __REACTOS__ 3803 return IsNetDrive(dwDriveNum); 3804 #else 3805 GET_FUNC(pIsNetDrive, shell32, (LPCSTR)66, FALSE); /* ord 66 = shell32.IsNetDrive */ 3806 return pIsNetDrive(dwDriveNum); 3807 #endif 3808 } 3809 3810 /************************************************************************* 3811 * PathIsLFNFileSpecA [SHLWAPI.@] 3812 * 3813 * Determine if the given path is a long file name 3814 * 3815 * PARAMS 3816 * lpszPath [I] Path to check 3817 * 3818 * RETURNS 3819 * TRUE If path is a long file name, 3820 * FALSE If path is a valid DOS short file name 3821 */ 3822 BOOL WINAPI PathIsLFNFileSpecA(LPCSTR lpszPath) 3823 { 3824 DWORD dwNameLen = 0, dwExtLen = 0; 3825 3826 TRACE("(%s)\n",debugstr_a(lpszPath)); 3827 3828 if (!lpszPath) 3829 return FALSE; 3830 3831 while (*lpszPath) 3832 { 3833 if (*lpszPath == ' ') 3834 return TRUE; /* DOS names cannot have spaces */ 3835 if (*lpszPath == '.') 3836 { 3837 if (dwExtLen) 3838 return TRUE; /* DOS names have only one dot */ 3839 dwExtLen = 1; 3840 } 3841 else if (dwExtLen) 3842 { 3843 dwExtLen++; 3844 if (dwExtLen > 4) 3845 return TRUE; /* DOS extensions are <= 3 chars*/ 3846 } 3847 else 3848 { 3849 dwNameLen++; 3850 if (dwNameLen > 8) 3851 return TRUE; /* DOS names are <= 8 chars */ 3852 } 3853 lpszPath += IsDBCSLeadByte(*lpszPath) ? 2 : 1; 3854 } 3855 return FALSE; /* Valid DOS path */ 3856 } 3857 3858 /************************************************************************* 3859 * PathIsLFNFileSpecW [SHLWAPI.@] 3860 * 3861 * See PathIsLFNFileSpecA. 3862 */ 3863 BOOL WINAPI PathIsLFNFileSpecW(LPCWSTR lpszPath) 3864 { 3865 DWORD dwNameLen = 0, dwExtLen = 0; 3866 3867 TRACE("(%s)\n",debugstr_w(lpszPath)); 3868 3869 if (!lpszPath) 3870 return FALSE; 3871 3872 while (*lpszPath) 3873 { 3874 if (*lpszPath == ' ') 3875 return TRUE; /* DOS names cannot have spaces */ 3876 if (*lpszPath == '.') 3877 { 3878 if (dwExtLen) 3879 return TRUE; /* DOS names have only one dot */ 3880 dwExtLen = 1; 3881 } 3882 else if (dwExtLen) 3883 { 3884 dwExtLen++; 3885 if (dwExtLen > 4) 3886 return TRUE; /* DOS extensions are <= 3 chars*/ 3887 } 3888 else 3889 { 3890 dwNameLen++; 3891 if (dwNameLen > 8) 3892 return TRUE; /* DOS names are <= 8 chars */ 3893 } 3894 lpszPath++; 3895 } 3896 return FALSE; /* Valid DOS path */ 3897 } 3898 3899 /************************************************************************* 3900 * PathIsDirectoryEmptyA [SHLWAPI.@] 3901 * 3902 * Determine if a given directory is empty. 3903 * 3904 * PARAMS 3905 * lpszPath [I] Directory to check 3906 * 3907 * RETURNS 3908 * TRUE If the directory exists and contains no files, 3909 * FALSE Otherwise 3910 */ 3911 BOOL WINAPI PathIsDirectoryEmptyA(LPCSTR lpszPath) 3912 { 3913 BOOL bRet = FALSE; 3914 3915 TRACE("(%s)\n",debugstr_a(lpszPath)); 3916 3917 if (lpszPath) 3918 { 3919 WCHAR szPath[MAX_PATH]; 3920 MultiByteToWideChar(CP_ACP,0,lpszPath,-1,szPath,MAX_PATH); 3921 bRet = PathIsDirectoryEmptyW(szPath); 3922 } 3923 return bRet; 3924 } 3925 3926 /************************************************************************* 3927 * PathIsDirectoryEmptyW [SHLWAPI.@] 3928 * 3929 * See PathIsDirectoryEmptyA. 3930 */ 3931 BOOL WINAPI PathIsDirectoryEmptyW(LPCWSTR lpszPath) 3932 { 3933 static const WCHAR szAllFiles[] = { '*', '.', '*', '\0' }; 3934 WCHAR szSearch[MAX_PATH]; 3935 DWORD dwLen; 3936 HANDLE hfind; 3937 BOOL retVal = TRUE; 3938 WIN32_FIND_DATAW find_data; 3939 3940 TRACE("(%s)\n",debugstr_w(lpszPath)); 3941 3942 if (!lpszPath || !PathIsDirectoryW(lpszPath)) 3943 return FALSE; 3944 3945 lstrcpynW(szSearch, lpszPath, MAX_PATH); 3946 PathAddBackslashW(szSearch); 3947 dwLen = strlenW(szSearch); 3948 if (dwLen > MAX_PATH - 4) 3949 return FALSE; 3950 3951 strcpyW(szSearch + dwLen, szAllFiles); 3952 hfind = FindFirstFileW(szSearch, &find_data); 3953 if (hfind == INVALID_HANDLE_VALUE) 3954 return FALSE; 3955 3956 do 3957 { 3958 if (find_data.cFileName[0] == '.') 3959 { 3960 if (find_data.cFileName[1] == '\0') continue; 3961 if (find_data.cFileName[1] == '.' && find_data.cFileName[2] == '\0') continue; 3962 } 3963 3964 retVal = FALSE; 3965 break; 3966 } 3967 while (FindNextFileW(hfind, &find_data)); 3968 3969 FindClose(hfind); 3970 return retVal; 3971 } 3972 3973 3974 /************************************************************************* 3975 * PathFindSuffixArrayA [SHLWAPI.@] 3976 * 3977 * Find a suffix string in an array of suffix strings 3978 * 3979 * PARAMS 3980 * lpszSuffix [I] Suffix string to search for 3981 * lppszArray [I] Array of suffix strings to search 3982 * dwCount [I] Number of elements in lppszArray 3983 * 3984 * RETURNS 3985 * Success: The index of the position of lpszSuffix in lppszArray 3986 * Failure: 0, if any parameters are invalid or lpszSuffix is not found 3987 * 3988 * NOTES 3989 * The search is case sensitive. 3990 * The match is made against the end of the suffix string, so for example: 3991 * lpszSuffix="fooBAR" matches "BAR", but lpszSuffix="fooBARfoo" does not. 3992 */ 3993 LPCSTR WINAPI PathFindSuffixArrayA(LPCSTR lpszSuffix, LPCSTR *lppszArray, int dwCount) 3994 { 3995 size_t dwLen; 3996 int dwRet = 0; 3997 3998 TRACE("(%s,%p,%d)\n",debugstr_a(lpszSuffix), lppszArray, dwCount); 3999 4000 if (lpszSuffix && lppszArray && dwCount > 0) 4001 { 4002 dwLen = strlen(lpszSuffix); 4003 4004 while (dwRet < dwCount) 4005 { 4006 size_t dwCompareLen = strlen(*lppszArray); 4007 if (dwCompareLen < dwLen) 4008 { 4009 if (!strcmp(lpszSuffix + dwLen - dwCompareLen, *lppszArray)) 4010 return *lppszArray; /* Found */ 4011 } 4012 dwRet++; 4013 lppszArray++; 4014 } 4015 } 4016 return NULL; 4017 } 4018 4019 /************************************************************************* 4020 * PathFindSuffixArrayW [SHLWAPI.@] 4021 * 4022 * See PathFindSuffixArrayA. 4023 */ 4024 LPCWSTR WINAPI PathFindSuffixArrayW(LPCWSTR lpszSuffix, LPCWSTR *lppszArray, int dwCount) 4025 { 4026 size_t dwLen; 4027 int dwRet = 0; 4028 4029 TRACE("(%s,%p,%d)\n",debugstr_w(lpszSuffix), lppszArray, dwCount); 4030 4031 if (lpszSuffix && lppszArray && dwCount > 0) 4032 { 4033 dwLen = strlenW(lpszSuffix); 4034 4035 while (dwRet < dwCount) 4036 { 4037 size_t dwCompareLen = strlenW(*lppszArray); 4038 if (dwCompareLen < dwLen) 4039 { 4040 if (!strcmpW(lpszSuffix + dwLen - dwCompareLen, *lppszArray)) 4041 return *lppszArray; /* Found */ 4042 } 4043 dwRet++; 4044 lppszArray++; 4045 } 4046 } 4047 return NULL; 4048 } 4049 4050 /************************************************************************* 4051 * PathUndecorateA [SHLWAPI.@] 4052 * 4053 * Undecorate a file path 4054 * 4055 * PARAMS 4056 * lpszPath [I/O] Path to remove any decoration from 4057 * 4058 * RETURNS 4059 * Nothing 4060 * 4061 * NOTES 4062 * A decorations form is "path[n].ext" where "n" is an optional decimal number. 4063 */ 4064 VOID WINAPI PathUndecorateA(LPSTR lpszPath) 4065 { 4066 TRACE("(%s)\n",debugstr_a(lpszPath)); 4067 4068 if (lpszPath) 4069 { 4070 LPSTR lpszExt = PathFindExtensionA(lpszPath); 4071 if (lpszExt > lpszPath && lpszExt[-1] == ']') 4072 { 4073 LPSTR lpszSkip = lpszExt - 2; 4074 if (*lpszSkip == '[') 4075 lpszSkip++; /* [] (no number) */ 4076 else 4077 while (lpszSkip > lpszPath && isdigit(lpszSkip[-1])) 4078 lpszSkip--; 4079 if (lpszSkip > lpszPath && lpszSkip[-1] == '[' && lpszSkip[-2] != '\\') 4080 { 4081 /* remove the [n] */ 4082 lpszSkip--; 4083 while (*lpszExt) 4084 *lpszSkip++ = *lpszExt++; 4085 *lpszSkip = '\0'; 4086 } 4087 } 4088 } 4089 } 4090 4091 /************************************************************************* 4092 * PathUndecorateW [SHLWAPI.@] 4093 * 4094 * See PathUndecorateA. 4095 */ 4096 VOID WINAPI PathUndecorateW(LPWSTR lpszPath) 4097 { 4098 TRACE("(%s)\n",debugstr_w(lpszPath)); 4099 4100 if (lpszPath) 4101 { 4102 LPWSTR lpszExt = PathFindExtensionW(lpszPath); 4103 if (lpszExt > lpszPath && lpszExt[-1] == ']') 4104 { 4105 LPWSTR lpszSkip = lpszExt - 2; 4106 if (*lpszSkip == '[') 4107 lpszSkip++; /* [] (no number) */ 4108 else 4109 while (lpszSkip > lpszPath && isdigitW(lpszSkip[-1])) 4110 lpszSkip--; 4111 if (lpszSkip > lpszPath && lpszSkip[-1] == '[' && lpszSkip[-2] != '\\') 4112 { 4113 /* remove the [n] */ 4114 lpszSkip--; 4115 while (*lpszExt) 4116 *lpszSkip++ = *lpszExt++; 4117 *lpszSkip = '\0'; 4118 } 4119 } 4120 } 4121 } 4122 4123 /************************************************************************* 4124 * PathUnExpandEnvStringsA [SHLWAPI.@] 4125 * 4126 * Substitute folder names in a path with their corresponding environment 4127 * strings. 4128 * 4129 * PARAMS 4130 * path [I] Buffer containing the path to unexpand. 4131 * buffer [O] Buffer to receive the unexpanded path. 4132 * buf_len [I] Size of pszBuf in characters. 4133 * 4134 * RETURNS 4135 * Success: TRUE 4136 * Failure: FALSE 4137 */ 4138 BOOL WINAPI PathUnExpandEnvStringsA(LPCSTR path, LPSTR buffer, UINT buf_len) 4139 { 4140 WCHAR bufferW[MAX_PATH], *pathW; 4141 DWORD len; 4142 BOOL ret; 4143 4144 TRACE("(%s, %p, %d)\n", debugstr_a(path), buffer, buf_len); 4145 4146 pathW = heap_strdupAtoW(path); 4147 if (!pathW) return FALSE; 4148 4149 ret = PathUnExpandEnvStringsW(pathW, bufferW, MAX_PATH); 4150 HeapFree(GetProcessHeap(), 0, pathW); 4151 if (!ret) return FALSE; 4152 4153 len = WideCharToMultiByte(CP_ACP, 0, bufferW, -1, NULL, 0, NULL, NULL); 4154 if (buf_len < len + 1) return FALSE; 4155 4156 WideCharToMultiByte(CP_ACP, 0, bufferW, -1, buffer, buf_len, NULL, NULL); 4157 return TRUE; 4158 } 4159 4160 static const WCHAR allusersprofileW[] = {'%','A','L','L','U','S','E','R','S','P','R','O','F','I','L','E','%',0}; 4161 static const WCHAR appdataW[] = {'%','A','P','P','D','A','T','A','%',0}; 4162 static const WCHAR programfilesW[] = {'%','P','r','o','g','r','a','m','F','i','l','e','s','%',0}; 4163 static const WCHAR systemrootW[] = {'%','S','y','s','t','e','m','R','o','o','t','%',0}; 4164 static const WCHAR systemdriveW[] = {'%','S','y','s','t','e','m','D','r','i','v','e','%',0}; 4165 static const WCHAR userprofileW[] = {'%','U','S','E','R','P','R','O','F','I','L','E','%',0}; 4166 4167 struct envvars_map 4168 { 4169 const WCHAR *var; 4170 UINT varlen; 4171 WCHAR path[MAX_PATH]; 4172 DWORD len; 4173 }; 4174 4175 static void init_envvars_map(struct envvars_map *map) 4176 { 4177 while (map->var) 4178 { 4179 map->len = ExpandEnvironmentStringsW(map->var, map->path, sizeof(map->path)/sizeof(WCHAR)); 4180 /* exclude null from length */ 4181 if (map->len) map->len--; 4182 map++; 4183 } 4184 } 4185 4186 /************************************************************************* 4187 * PathUnExpandEnvStringsW [SHLWAPI.@] 4188 * 4189 * Unicode version of PathUnExpandEnvStringsA. 4190 */ 4191 BOOL WINAPI PathUnExpandEnvStringsW(LPCWSTR path, LPWSTR buffer, UINT buf_len) 4192 { 4193 static struct envvars_map null_var = {NULL, 0, {0}, 0}; 4194 struct envvars_map *match = &null_var, *cur; 4195 struct envvars_map envvars[] = { 4196 { allusersprofileW, sizeof(allusersprofileW)/sizeof(WCHAR) }, 4197 { appdataW, sizeof(appdataW)/sizeof(WCHAR) }, 4198 { programfilesW, sizeof(programfilesW)/sizeof(WCHAR) }, 4199 { systemrootW, sizeof(systemrootW)/sizeof(WCHAR) }, 4200 { systemdriveW, sizeof(systemdriveW)/sizeof(WCHAR) }, 4201 { userprofileW, sizeof(userprofileW)/sizeof(WCHAR) }, 4202 { NULL } 4203 }; 4204 DWORD pathlen; 4205 UINT needed; 4206 4207 TRACE("(%s, %p, %d)\n", debugstr_w(path), buffer, buf_len); 4208 4209 pathlen = strlenW(path); 4210 init_envvars_map(envvars); 4211 cur = envvars; 4212 while (cur->var) 4213 { 4214 /* path can't contain expanded value or value wasn't retrieved */ 4215 if (cur->len == 0 || cur->len > pathlen || strncmpiW(cur->path, path, cur->len)) 4216 { 4217 cur++; 4218 continue; 4219 } 4220 4221 if (cur->len > match->len) 4222 match = cur; 4223 cur++; 4224 } 4225 4226 /* 'varlen' includes NULL termination char */ 4227 needed = match->varlen + pathlen - match->len; 4228 if (match->len == 0 || needed > buf_len) return FALSE; 4229 4230 strcpyW(buffer, match->var); 4231 strcatW(buffer, &path[match->len]); 4232 TRACE("ret %s\n", debugstr_w(buffer)); 4233 4234 return TRUE; 4235 } 4236 4237 /************************************************************************* 4238 * @ [SHLWAPI.440] 4239 * 4240 * Find localised or default web content in "%WINDOWS%\web\". 4241 * 4242 * PARAMS 4243 * lpszFile [I] File name containing content to look for 4244 * lpszPath [O] Buffer to contain the full path to the file 4245 * dwPathLen [I] Length of lpszPath 4246 * 4247 * RETURNS 4248 * Success: S_OK. lpszPath contains the full path to the content. 4249 * Failure: E_FAIL. The content does not exist or lpszPath is too short. 4250 */ 4251 HRESULT WINAPI SHGetWebFolderFilePathA(LPCSTR lpszFile, LPSTR lpszPath, DWORD dwPathLen) 4252 { 4253 WCHAR szFile[MAX_PATH], szPath[MAX_PATH]; 4254 HRESULT hRet; 4255 4256 TRACE("(%s,%p,%d)\n", lpszFile, lpszPath, dwPathLen); 4257 4258 MultiByteToWideChar(CP_ACP, 0, lpszFile, -1, szFile, MAX_PATH); 4259 szPath[0] = '\0'; 4260 hRet = SHGetWebFolderFilePathW(szFile, szPath, dwPathLen); 4261 WideCharToMultiByte(CP_ACP, 0, szPath, -1, lpszPath, dwPathLen, 0, 0); 4262 return hRet; 4263 } 4264 4265 /************************************************************************* 4266 * @ [SHLWAPI.441] 4267 * 4268 * Unicode version of SHGetWebFolderFilePathA. 4269 */ 4270 HRESULT WINAPI SHGetWebFolderFilePathW(LPCWSTR lpszFile, LPWSTR lpszPath, DWORD dwPathLen) 4271 { 4272 static const WCHAR szWeb[] = {'\\','W','e','b','\\','\0'}; 4273 static const WCHAR szWebMui[] = {'m','u','i','\\','%','0','4','x','\\','\0'}; 4274 #define szWebLen (sizeof(szWeb)/sizeof(WCHAR)) 4275 #define szWebMuiLen ((sizeof(szWebMui)+1)/sizeof(WCHAR)) 4276 DWORD dwLen, dwFileLen; 4277 LANGID lidSystem, lidUser; 4278 4279 TRACE("(%s,%p,%d)\n", debugstr_w(lpszFile), lpszPath, dwPathLen); 4280 4281 /* Get base directory for web content */ 4282 dwLen = GetSystemWindowsDirectoryW(lpszPath, dwPathLen); 4283 if (dwLen > 0 && lpszPath[dwLen-1] == '\\') 4284 dwLen--; 4285 4286 dwFileLen = strlenW(lpszFile); 4287 4288 if (dwLen + dwFileLen + szWebLen >= dwPathLen) 4289 return E_FAIL; /* lpszPath too short */ 4290 4291 strcpyW(lpszPath+dwLen, szWeb); 4292 dwLen += szWebLen; 4293 dwPathLen = dwPathLen - dwLen; /* Remaining space */ 4294 4295 lidSystem = GetSystemDefaultUILanguage(); 4296 lidUser = GetUserDefaultUILanguage(); 4297 4298 if (lidSystem != lidUser) 4299 { 4300 if (dwFileLen + szWebMuiLen < dwPathLen) 4301 { 4302 /* Use localised content in the users UI language if present */ 4303 wsprintfW(lpszPath + dwLen, szWebMui, lidUser); 4304 strcpyW(lpszPath + dwLen + szWebMuiLen, lpszFile); 4305 if (PathFileExistsW(lpszPath)) 4306 return S_OK; 4307 } 4308 } 4309 4310 /* Fall back to OS default installed content */ 4311 strcpyW(lpszPath + dwLen, lpszFile); 4312 if (PathFileExistsW(lpszPath)) 4313 return S_OK; 4314 return E_FAIL; 4315 } 4316 4317 #define PATH_CHAR_CLASS_LETTER 0x00000001 4318 #define PATH_CHAR_CLASS_ASTERIX 0x00000002 4319 #define PATH_CHAR_CLASS_DOT 0x00000004 4320 #define PATH_CHAR_CLASS_BACKSLASH 0x00000008 4321 #define PATH_CHAR_CLASS_COLON 0x00000010 4322 #define PATH_CHAR_CLASS_SEMICOLON 0x00000020 4323 #define PATH_CHAR_CLASS_COMMA 0x00000040 4324 #define PATH_CHAR_CLASS_SPACE 0x00000080 4325 #define PATH_CHAR_CLASS_OTHER_VALID 0x00000100 4326 #define PATH_CHAR_CLASS_DOUBLEQUOTE 0x00000200 4327 4328 #define PATH_CHAR_CLASS_INVALID 0x00000000 4329 #define PATH_CHAR_CLASS_ANY 0xffffffff 4330 4331 static const DWORD SHELL_charclass[] = 4332 { 4333 /* 0x00 */ PATH_CHAR_CLASS_INVALID, /* 0x01 */ PATH_CHAR_CLASS_INVALID, 4334 /* 0x02 */ PATH_CHAR_CLASS_INVALID, /* 0x03 */ PATH_CHAR_CLASS_INVALID, 4335 /* 0x04 */ PATH_CHAR_CLASS_INVALID, /* 0x05 */ PATH_CHAR_CLASS_INVALID, 4336 /* 0x06 */ PATH_CHAR_CLASS_INVALID, /* 0x07 */ PATH_CHAR_CLASS_INVALID, 4337 /* 0x08 */ PATH_CHAR_CLASS_INVALID, /* 0x09 */ PATH_CHAR_CLASS_INVALID, 4338 /* 0x0a */ PATH_CHAR_CLASS_INVALID, /* 0x0b */ PATH_CHAR_CLASS_INVALID, 4339 /* 0x0c */ PATH_CHAR_CLASS_INVALID, /* 0x0d */ PATH_CHAR_CLASS_INVALID, 4340 /* 0x0e */ PATH_CHAR_CLASS_INVALID, /* 0x0f */ PATH_CHAR_CLASS_INVALID, 4341 /* 0x10 */ PATH_CHAR_CLASS_INVALID, /* 0x11 */ PATH_CHAR_CLASS_INVALID, 4342 /* 0x12 */ PATH_CHAR_CLASS_INVALID, /* 0x13 */ PATH_CHAR_CLASS_INVALID, 4343 /* 0x14 */ PATH_CHAR_CLASS_INVALID, /* 0x15 */ PATH_CHAR_CLASS_INVALID, 4344 /* 0x16 */ PATH_CHAR_CLASS_INVALID, /* 0x17 */ PATH_CHAR_CLASS_INVALID, 4345 /* 0x18 */ PATH_CHAR_CLASS_INVALID, /* 0x19 */ PATH_CHAR_CLASS_INVALID, 4346 /* 0x1a */ PATH_CHAR_CLASS_INVALID, /* 0x1b */ PATH_CHAR_CLASS_INVALID, 4347 /* 0x1c */ PATH_CHAR_CLASS_INVALID, /* 0x1d */ PATH_CHAR_CLASS_INVALID, 4348 /* 0x1e */ PATH_CHAR_CLASS_INVALID, /* 0x1f */ PATH_CHAR_CLASS_INVALID, 4349 /* ' ' */ PATH_CHAR_CLASS_SPACE, /* '!' */ PATH_CHAR_CLASS_OTHER_VALID, 4350 /* '"' */ PATH_CHAR_CLASS_DOUBLEQUOTE, /* '#' */ PATH_CHAR_CLASS_OTHER_VALID, 4351 /* '$' */ PATH_CHAR_CLASS_OTHER_VALID, /* '%' */ PATH_CHAR_CLASS_OTHER_VALID, 4352 /* '&' */ PATH_CHAR_CLASS_OTHER_VALID, /* '\'' */ PATH_CHAR_CLASS_OTHER_VALID, 4353 /* '(' */ PATH_CHAR_CLASS_OTHER_VALID, /* ')' */ PATH_CHAR_CLASS_OTHER_VALID, 4354 /* '*' */ PATH_CHAR_CLASS_ASTERIX, /* '+' */ PATH_CHAR_CLASS_OTHER_VALID, 4355 /* ',' */ PATH_CHAR_CLASS_COMMA, /* '-' */ PATH_CHAR_CLASS_OTHER_VALID, 4356 /* '.' */ PATH_CHAR_CLASS_DOT, /* '/' */ PATH_CHAR_CLASS_INVALID, 4357 /* '0' */ PATH_CHAR_CLASS_OTHER_VALID, /* '1' */ PATH_CHAR_CLASS_OTHER_VALID, 4358 /* '2' */ PATH_CHAR_CLASS_OTHER_VALID, /* '3' */ PATH_CHAR_CLASS_OTHER_VALID, 4359 /* '4' */ PATH_CHAR_CLASS_OTHER_VALID, /* '5' */ PATH_CHAR_CLASS_OTHER_VALID, 4360 /* '6' */ PATH_CHAR_CLASS_OTHER_VALID, /* '7' */ PATH_CHAR_CLASS_OTHER_VALID, 4361 /* '8' */ PATH_CHAR_CLASS_OTHER_VALID, /* '9' */ PATH_CHAR_CLASS_OTHER_VALID, 4362 /* ':' */ PATH_CHAR_CLASS_COLON, /* ';' */ PATH_CHAR_CLASS_SEMICOLON, 4363 /* '<' */ PATH_CHAR_CLASS_INVALID, /* '=' */ PATH_CHAR_CLASS_OTHER_VALID, 4364 /* '>' */ PATH_CHAR_CLASS_INVALID, /* '?' */ PATH_CHAR_CLASS_LETTER, 4365 /* '@' */ PATH_CHAR_CLASS_OTHER_VALID, /* 'A' */ PATH_CHAR_CLASS_ANY, 4366 /* 'B' */ PATH_CHAR_CLASS_ANY, /* 'C' */ PATH_CHAR_CLASS_ANY, 4367 /* 'D' */ PATH_CHAR_CLASS_ANY, /* 'E' */ PATH_CHAR_CLASS_ANY, 4368 /* 'F' */ PATH_CHAR_CLASS_ANY, /* 'G' */ PATH_CHAR_CLASS_ANY, 4369 /* 'H' */ PATH_CHAR_CLASS_ANY, /* 'I' */ PATH_CHAR_CLASS_ANY, 4370 /* 'J' */ PATH_CHAR_CLASS_ANY, /* 'K' */ PATH_CHAR_CLASS_ANY, 4371 /* 'L' */ PATH_CHAR_CLASS_ANY, /* 'M' */ PATH_CHAR_CLASS_ANY, 4372 /* 'N' */ PATH_CHAR_CLASS_ANY, /* 'O' */ PATH_CHAR_CLASS_ANY, 4373 /* 'P' */ PATH_CHAR_CLASS_ANY, /* 'Q' */ PATH_CHAR_CLASS_ANY, 4374 /* 'R' */ PATH_CHAR_CLASS_ANY, /* 'S' */ PATH_CHAR_CLASS_ANY, 4375 /* 'T' */ PATH_CHAR_CLASS_ANY, /* 'U' */ PATH_CHAR_CLASS_ANY, 4376 /* 'V' */ PATH_CHAR_CLASS_ANY, /* 'W' */ PATH_CHAR_CLASS_ANY, 4377 /* 'X' */ PATH_CHAR_CLASS_ANY, /* 'Y' */ PATH_CHAR_CLASS_ANY, 4378 /* 'Z' */ PATH_CHAR_CLASS_ANY, /* '[' */ PATH_CHAR_CLASS_OTHER_VALID, 4379 /* '\\' */ PATH_CHAR_CLASS_BACKSLASH, /* ']' */ PATH_CHAR_CLASS_OTHER_VALID, 4380 /* '^' */ PATH_CHAR_CLASS_OTHER_VALID, /* '_' */ PATH_CHAR_CLASS_OTHER_VALID, 4381 /* '`' */ PATH_CHAR_CLASS_OTHER_VALID, /* 'a' */ PATH_CHAR_CLASS_ANY, 4382 /* 'b' */ PATH_CHAR_CLASS_ANY, /* 'c' */ PATH_CHAR_CLASS_ANY, 4383 /* 'd' */ PATH_CHAR_CLASS_ANY, /* 'e' */ PATH_CHAR_CLASS_ANY, 4384 /* 'f' */ PATH_CHAR_CLASS_ANY, /* 'g' */ PATH_CHAR_CLASS_ANY, 4385 /* 'h' */ PATH_CHAR_CLASS_ANY, /* 'i' */ PATH_CHAR_CLASS_ANY, 4386 /* 'j' */ PATH_CHAR_CLASS_ANY, /* 'k' */ PATH_CHAR_CLASS_ANY, 4387 /* 'l' */ PATH_CHAR_CLASS_ANY, /* 'm' */ PATH_CHAR_CLASS_ANY, 4388 /* 'n' */ PATH_CHAR_CLASS_ANY, /* 'o' */ PATH_CHAR_CLASS_ANY, 4389 /* 'p' */ PATH_CHAR_CLASS_ANY, /* 'q' */ PATH_CHAR_CLASS_ANY, 4390 /* 'r' */ PATH_CHAR_CLASS_ANY, /* 's' */ PATH_CHAR_CLASS_ANY, 4391 /* 't' */ PATH_CHAR_CLASS_ANY, /* 'u' */ PATH_CHAR_CLASS_ANY, 4392 /* 'v' */ PATH_CHAR_CLASS_ANY, /* 'w' */ PATH_CHAR_CLASS_ANY, 4393 /* 'x' */ PATH_CHAR_CLASS_ANY, /* 'y' */ PATH_CHAR_CLASS_ANY, 4394 /* 'z' */ PATH_CHAR_CLASS_ANY, /* '{' */ PATH_CHAR_CLASS_OTHER_VALID, 4395 /* '|' */ PATH_CHAR_CLASS_INVALID, /* '}' */ PATH_CHAR_CLASS_OTHER_VALID, 4396 /* '~' */ PATH_CHAR_CLASS_OTHER_VALID 4397 }; 4398 4399 /************************************************************************* 4400 * @ [SHLWAPI.455] 4401 * 4402 * Check if an ASCII char is of a certain class 4403 */ 4404 BOOL WINAPI PathIsValidCharA( char c, DWORD class ) 4405 { 4406 if ((unsigned)c > 0x7e) 4407 return class & PATH_CHAR_CLASS_OTHER_VALID; 4408 4409 return class & SHELL_charclass[(unsigned)c]; 4410 } 4411 4412 /************************************************************************* 4413 * @ [SHLWAPI.456] 4414 * 4415 * Check if a Unicode char is of a certain class 4416 */ 4417 BOOL WINAPI PathIsValidCharW( WCHAR c, DWORD class ) 4418 { 4419 if (c > 0x7e) 4420 return class & PATH_CHAR_CLASS_OTHER_VALID; 4421 4422 return class & SHELL_charclass[c]; 4423 } 4424