1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/common/filename.cpp 
   3 // Purpose:     wxFileName - encapsulates a file path 
   4 // Author:      Robert Roebling, Vadim Zeitlin 
   8 // Copyright:   (c) 2000 Robert Roebling 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13    Here are brief descriptions of the filename formats supported by this class: 
  15    wxPATH_UNIX: standard Unix format, used under Darwin as well, absolute file 
  17                 /dir1/dir2/.../dirN/filename, "." and ".." stand for the 
  18                 current and parent directory respectively, "~" is parsed as the 
  19                 user HOME and "~username" as the HOME of that user 
  21    wxPATH_DOS:  DOS/Windows format, absolute file names have the form: 
  22                 drive:\dir1\dir2\...\dirN\filename.ext where drive is a single 
  23                 letter. "." and ".." as for Unix but no "~". 
  25                 There are also UNC names of the form \\share\fullpath 
  27    wxPATH_MAC:  Mac OS 8/9 and Mac OS X under CodeWarrior 7 format, absolute file 
  29                     volume:dir1:...:dirN:filename 
  30                 and the relative file names are either 
  31                     :dir1:...:dirN:filename 
  34                 (although :filename works as well). 
  35                 Since the volume is just part of the file path, it is not 
  36                 treated like a separate entity as it is done under DOS and 
  37                 VMS, it is just treated as another dir. 
  39    wxPATH_VMS:  VMS native format, absolute file names have the form 
  40                     <device>:[dir1.dir2.dir3]file.txt 
  42                     <device>:[000000.dir1.dir2.dir3]file.txt 
  44                 the <device> is the physical device (i.e. disk). 000000 is the 
  45                 root directory on the device which can be omitted. 
  47                 Note that VMS uses different separators unlike Unix: 
  48                  : always after the device. If the path does not contain : than 
  49                    the default (the device of the current directory) is assumed. 
  50                  [ start of directory specification 
  51                  . separator between directory and subdirectory 
  52                  ] between directory and file 
  55 // ============================================================================ 
  57 // ============================================================================ 
  59 // ---------------------------------------------------------------------------- 
  61 // ---------------------------------------------------------------------------- 
  63 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  64 #pragma implementation "filename.h" 
  67 // For compilers that support precompilation, includes "wx.h". 
  68 #include "wx/wxprec.h" 
  80 #include "wx/filename.h" 
  81 #include "wx/tokenzr.h" 
  82 #include "wx/config.h"          // for wxExpandEnvVars 
  85 #include "wx/dynlib.h" 
  87 // For GetShort/LongPathName 
  89 #include "wx/msw/wrapwin.h" 
  90 #if defined(__MINGW32__) 
  91 #include "wx/msw/gccpriv.h" 
  96 #include "wx/msw/private.h" 
  99 #if defined(__WXMAC__) 
 100   #include  "wx/mac/private.h"  // includes mac headers 
 103 // utime() is POSIX so should normally be available on all Unices 
 105 #include <sys/types.h> 
 107 #include <sys/stat.h> 
 117 #include <sys/types.h> 
 119 #include <sys/stat.h> 
 130 #include <sys/utime.h> 
 131 #include <sys/stat.h> 
 142 #define MAX_PATH _MAX_PATH 
 145 // ---------------------------------------------------------------------------- 
 147 // ---------------------------------------------------------------------------- 
 149 // small helper class which opens and closes the file - we use it just to get 
 150 // a file handle for the given file name to pass it to some Win32 API function 
 151 #if defined(__WIN32__) && !defined(__WXMICROWIN__) 
 162     wxFileHandle(const wxString
& filename
, OpenMode mode
) 
 164         m_hFile 
= ::CreateFile
 
 167                      mode 
== Read 
? GENERIC_READ    
// access mask 
 169                      FILE_SHARE_READ 
|              // sharing mode 
 170                      FILE_SHARE_WRITE
,              // (allow everything) 
 171                      NULL
,                          // no secutity attr 
 172                      OPEN_EXISTING
,                 // creation disposition 
 174                      NULL                           
// no template file 
 177         if ( m_hFile 
== INVALID_HANDLE_VALUE 
) 
 179             wxLogSysError(_("Failed to open '%s' for %s"), 
 181                           mode 
== Read 
? _("reading") : _("writing")); 
 187         if ( m_hFile 
!= INVALID_HANDLE_VALUE 
) 
 189             if ( !::CloseHandle(m_hFile
) ) 
 191                 wxLogSysError(_("Failed to close file handle")); 
 196     // return true only if the file could be opened successfully 
 197     bool IsOk() const { return m_hFile 
!= INVALID_HANDLE_VALUE
; } 
 200     operator HANDLE() const { return m_hFile
; } 
 208 // ---------------------------------------------------------------------------- 
 210 // ---------------------------------------------------------------------------- 
 212 #if wxUSE_DATETIME && defined(__WIN32__) && !defined(__WXMICROWIN__) 
 214 // convert between wxDateTime and FILETIME which is a 64-bit value representing 
 215 // the number of 100-nanosecond intervals since January 1, 1601. 
 217 static void ConvertFileTimeToWx(wxDateTime 
*dt
, const FILETIME 
&ft
) 
 219     FILETIME ftcopy 
= ft
; 
 221     if ( !::FileTimeToLocalFileTime(&ftcopy
, &ftLocal
) ) 
 223         wxLogLastError(_T("FileTimeToLocalFileTime")); 
 227     if ( !::FileTimeToSystemTime(&ftLocal
, &st
) ) 
 229         wxLogLastError(_T("FileTimeToSystemTime")); 
 232     dt
->Set(st
.wDay
, wxDateTime::Month(st
.wMonth 
- 1), st
.wYear
, 
 233             st
.wHour
, st
.wMinute
, st
.wSecond
, st
.wMilliseconds
); 
 236 static void ConvertWxToFileTime(FILETIME 
*ft
, const wxDateTime
& dt
) 
 239     st
.wDay 
= dt
.GetDay(); 
 240     st
.wMonth 
= (WORD
)(dt
.GetMonth() + 1); 
 241     st
.wYear 
= (WORD
)dt
.GetYear(); 
 242     st
.wHour 
= dt
.GetHour(); 
 243     st
.wMinute 
= dt
.GetMinute(); 
 244     st
.wSecond 
= dt
.GetSecond(); 
 245     st
.wMilliseconds 
= dt
.GetMillisecond(); 
 248     if ( !::SystemTimeToFileTime(&st
, &ftLocal
) ) 
 250         wxLogLastError(_T("SystemTimeToFileTime")); 
 253     if ( !::LocalFileTimeToFileTime(&ftLocal
, ft
) ) 
 255         wxLogLastError(_T("LocalFileTimeToFileTime")); 
 259 #endif // wxUSE_DATETIME && __WIN32__ 
 261 // return a string with the volume par 
 262 static wxString 
wxGetVolumeString(const wxString
& volume
, wxPathFormat format
) 
 266     if ( !volume
.empty() ) 
 268         format 
= wxFileName::GetFormat(format
); 
 270         // Special Windows UNC paths hack, part 2: undo what we did in 
 271         // SplitPath() and make an UNC path if we have a drive which is not a 
 272         // single letter (hopefully the network shares can't be one letter only 
 273         // although I didn't find any authoritative docs on this) 
 274         if ( format 
== wxPATH_DOS 
&& volume
.length() > 1 ) 
 276             path 
<< wxFILE_SEP_PATH_DOS 
<< wxFILE_SEP_PATH_DOS 
<< volume
; 
 278         else if  ( format 
== wxPATH_DOS 
|| format 
== wxPATH_VMS 
) 
 280             path 
<< volume 
<< wxFileName::GetVolumeSeparator(format
); 
 288 // ============================================================================ 
 290 // ============================================================================ 
 292 // ---------------------------------------------------------------------------- 
 293 // wxFileName construction 
 294 // ---------------------------------------------------------------------------- 
 296 void wxFileName::Assign( const wxFileName 
&filepath 
) 
 298     m_volume 
= filepath
.GetVolume(); 
 299     m_dirs 
= filepath
.GetDirs(); 
 300     m_name 
= filepath
.GetName(); 
 301     m_ext 
= filepath
.GetExt(); 
 302     m_relative 
= filepath
.m_relative
; 
 303     m_hasExt 
= filepath
.m_hasExt
; 
 306 void wxFileName::Assign(const wxString
& volume
, 
 307                         const wxString
& path
, 
 308                         const wxString
& name
, 
 311                         wxPathFormat format 
) 
 313     SetPath( path
, format 
); 
 322 void wxFileName::SetPath( const wxString
& pathOrig
, wxPathFormat format 
) 
 326     if ( pathOrig
.empty() ) 
 334     format 
= GetFormat( format 
); 
 336     // 0) deal with possible volume part first 
 339     SplitVolume(pathOrig
, &volume
, &path
, format
); 
 340     if ( !volume
.empty() ) 
 347     // 1) Determine if the path is relative or absolute. 
 348     wxChar leadingChar 
= path
[0u]; 
 353             m_relative 
= leadingChar 
== wxT(':'); 
 355             // We then remove a leading ":". The reason is in our 
 356             // storage form for relative paths: 
 357             // ":dir:file.txt" actually means "./dir/file.txt" in 
 358             // DOS notation and should get stored as 
 359             // (relative) (dir) (file.txt) 
 360             // "::dir:file.txt" actually means "../dir/file.txt" 
 361             // stored as (relative) (..) (dir) (file.txt) 
 362             // This is important only for the Mac as an empty dir 
 363             // actually means <UP>, whereas under DOS, double 
 364             // slashes can be ignored: "\\\\" is the same as "\\". 
 370             // TODO: what is the relative path format here? 
 375             wxFAIL_MSG( _T("Unknown path format") ); 
 376             // !! Fall through !! 
 379             // the paths of the form "~" or "~username" are absolute 
 380             m_relative 
= leadingChar 
!= wxT('/') && leadingChar 
!= _T('~'); 
 384             m_relative 
= !IsPathSeparator(leadingChar
, format
); 
 389     // 2) Break up the path into its members. If the original path 
 390     //    was just "/" or "\\", m_dirs will be empty. We know from 
 391     //    the m_relative field, if this means "nothing" or "root dir". 
 393     wxStringTokenizer 
tn( path
, GetPathSeparators(format
) ); 
 395     while ( tn
.HasMoreTokens() ) 
 397         wxString token 
= tn
.GetNextToken(); 
 399         // Remove empty token under DOS and Unix, interpret them 
 403             if (format 
== wxPATH_MAC
) 
 404                 m_dirs
.Add( wxT("..") ); 
 414 void wxFileName::Assign(const wxString
& fullpath
, 
 417     wxString volume
, path
, name
, ext
; 
 419     SplitPath(fullpath
, &volume
, &path
, &name
, &ext
, &hasExt
, format
); 
 421     Assign(volume
, path
, name
, ext
, hasExt
, format
); 
 424 void wxFileName::Assign(const wxString
& fullpathOrig
, 
 425                         const wxString
& fullname
, 
 428     // always recognize fullpath as directory, even if it doesn't end with a 
 430     wxString fullpath 
= fullpathOrig
; 
 431     if ( !wxEndsWithPathSeparator(fullpath
) ) 
 433         fullpath 
+= GetPathSeparator(format
); 
 436     wxString volume
, path
, name
, ext
; 
 439     // do some consistency checks in debug mode: the name should be really just 
 440     // the filename and the path should be really just a path 
 442     wxString volDummy
, pathDummy
, nameDummy
, extDummy
; 
 444     SplitPath(fullname
, &volDummy
, &pathDummy
, &name
, &ext
, &hasExt
, format
); 
 446     wxASSERT_MSG( volDummy
.empty() && pathDummy
.empty(), 
 447                   _T("the file name shouldn't contain the path") ); 
 449     SplitPath(fullpath
, &volume
, &path
, &nameDummy
, &extDummy
, format
); 
 451     wxASSERT_MSG( nameDummy
.empty() && extDummy
.empty(), 
 452                   _T("the path shouldn't contain file name nor extension") ); 
 454 #else // !__WXDEBUG__ 
 455     SplitPath(fullname
, NULL 
/* no volume */, NULL 
/* no path */, 
 456                         &name
, &ext
, &hasExt
, format
); 
 457     SplitPath(fullpath
, &volume
, &path
, NULL
, NULL
, format
); 
 458 #endif // __WXDEBUG__/!__WXDEBUG__ 
 460     Assign(volume
, path
, name
, ext
, hasExt
, format
); 
 463 void wxFileName::Assign(const wxString
& pathOrig
, 
 464                         const wxString
& name
, 
 470     SplitVolume(pathOrig
, &volume
, &path
, format
); 
 472     Assign(volume
, path
, name
, ext
, format
); 
 475 void wxFileName::AssignDir(const wxString
& dir
, wxPathFormat format
) 
 477     Assign(dir
, wxEmptyString
, format
); 
 480 void wxFileName::Clear() 
 486     m_ext 
= wxEmptyString
; 
 488     // we don't have any absolute path for now 
 496 wxFileName 
wxFileName::FileName(const wxString
& file
, wxPathFormat format
) 
 498     return wxFileName(file
, format
); 
 502 wxFileName 
wxFileName::DirName(const wxString
& dir
, wxPathFormat format
) 
 505     fn
.AssignDir(dir
, format
); 
 509 // ---------------------------------------------------------------------------- 
 511 // ---------------------------------------------------------------------------- 
 513 bool wxFileName::FileExists() const 
 515     return wxFileName::FileExists( GetFullPath() ); 
 518 bool wxFileName::FileExists( const wxString 
&file 
) 
 520     return ::wxFileExists( file 
); 
 523 bool wxFileName::DirExists() const 
 525     return wxFileName::DirExists( GetFullPath() ); 
 528 bool wxFileName::DirExists( const wxString 
&dir 
) 
 530     return ::wxDirExists( dir 
); 
 533 // ---------------------------------------------------------------------------- 
 534 // CWD and HOME stuff 
 535 // ---------------------------------------------------------------------------- 
 537 void wxFileName::AssignCwd(const wxString
& volume
) 
 539     AssignDir(wxFileName::GetCwd(volume
)); 
 543 wxString 
wxFileName::GetCwd(const wxString
& volume
) 
 545     // if we have the volume, we must get the current directory on this drive 
 546     // and to do this we have to chdir to this volume - at least under Windows, 
 547     // I don't know how to get the current drive on another volume elsewhere 
 550     if ( !volume
.empty() ) 
 553         SetCwd(volume 
+ GetVolumeSeparator()); 
 556     wxString cwd 
= ::wxGetCwd(); 
 558     if ( !volume
.empty() ) 
 566 bool wxFileName::SetCwd() 
 568     return wxFileName::SetCwd( GetFullPath() ); 
 571 bool wxFileName::SetCwd( const wxString 
&cwd 
) 
 573     return ::wxSetWorkingDirectory( cwd 
); 
 576 void wxFileName::AssignHomeDir() 
 578     AssignDir(wxFileName::GetHomeDir()); 
 581 wxString 
wxFileName::GetHomeDir() 
 583     return ::wxGetHomeDir(); 
 588 void wxFileName::AssignTempFileName(const wxString
& prefix
, wxFile 
*fileTemp
) 
 590     wxString tempname 
= CreateTempFileName(prefix
, fileTemp
); 
 591     if ( tempname
.empty() ) 
 593         // error, failed to get temp file name 
 604 wxFileName::CreateTempFileName(const wxString
& prefix
, wxFile 
*fileTemp
) 
 606     wxString path
, dir
, name
; 
 608     // use the directory specified by the prefix 
 609     SplitPath(prefix
, &dir
, &name
, NULL 
/* extension */); 
 611 #if defined(__WXWINCE__) 
 614         // FIXME. Create \temp dir? 
 617     path 
= dir 
+ wxT("\\") + prefix
; 
 619     while (FileExists(path
)) 
 621         path 
= dir 
+ wxT("\\") + prefix 
; 
 626 #elif defined(__WINDOWS__) && !defined(__WXMICROWIN__) 
 630         if ( !::GetTempPath(MAX_PATH
, wxStringBuffer(dir
, MAX_PATH 
+ 1)) ) 
 632             wxLogLastError(_T("GetTempPath")); 
 637             // GetTempFileName() fails if we pass it an empty string 
 641     else // we have a dir to create the file in 
 643         // ensure we use only the back slashes as GetTempFileName(), unlike all 
 644         // the other APIs, is picky and doesn't accept the forward ones 
 645         dir
.Replace(_T("/"), _T("\\")); 
 648     if ( !::GetTempFileName(dir
, name
, 0, wxStringBuffer(path
, MAX_PATH 
+ 1)) ) 
 650         wxLogLastError(_T("GetTempFileName")); 
 658 #if defined(__WXMAC__) && !defined(__DARWIN__) 
 659         dir 
= wxMacFindFolder(  (short) kOnSystemDisk
, kTemporaryFolderType
, kCreateFolder 
) ; 
 661         dir 
= wxGetenv(_T("TMP")); 
 664             dir 
= wxGetenv(_T("TEMP")); 
 670             #if defined(__DOS__) || defined(__OS2__) 
 681     if ( !wxEndsWithPathSeparator(dir
) && 
 682             (name
.empty() || !wxIsPathSeparator(name
[0u])) ) 
 684         path 
+= wxFILE_SEP_PATH
; 
 689 #if defined(HAVE_MKSTEMP) 
 690     // scratch space for mkstemp() 
 691     path 
+= _T("XXXXXX"); 
 693     // we need to copy the path to the buffer in which mkstemp() can modify it 
 694     wxCharBuffer 
buf( wxConvFile
.cWX2MB( path 
) ); 
 696     // cast is safe because the string length doesn't change 
 697     int fdTemp 
= mkstemp( (char*)(const char*) buf 
); 
 700         // this might be not necessary as mkstemp() on most systems should have 
 701         // already done it but it doesn't hurt neither... 
 704     else // mkstemp() succeeded 
 706         path 
= wxConvFile
.cMB2WX( (const char*) buf 
); 
 708         // avoid leaking the fd 
 711             fileTemp
->Attach(fdTemp
); 
 718 #else // !HAVE_MKSTEMP 
 722     path 
+= _T("XXXXXX"); 
 724     wxCharBuffer buf 
= wxConvFile
.cWX2MB( path 
); 
 725     if ( !mktemp( (const char*) buf 
) ) 
 731         path 
= wxConvFile
.cMB2WX( (const char*) buf 
); 
 733 #else // !HAVE_MKTEMP (includes __DOS__) 
 734     // generate the unique file name ourselves 
 735     #if !defined(__DOS__) && !defined(__PALMOS__) && (!defined(__MWERKS__) || defined(__DARWIN__) ) 
 736     path 
<< (unsigned int)getpid(); 
 741     static const size_t numTries 
= 1000; 
 742     for ( size_t n 
= 0; n 
< numTries
; n
++ ) 
 744         // 3 hex digits is enough for numTries == 1000 < 4096 
 745         pathTry 
= path 
+ wxString::Format(_T("%.03x"), (unsigned int) n
); 
 746         if ( !FileExists(pathTry
) ) 
 755 #endif // HAVE_MKTEMP/!HAVE_MKTEMP 
 760 #endif // HAVE_MKSTEMP/!HAVE_MKSTEMP 
 762 #endif // Windows/!Windows 
 766         wxLogSysError(_("Failed to create a temporary file name")); 
 768     else if ( fileTemp 
&& !fileTemp
->IsOpened() ) 
 770         // open the file - of course, there is a race condition here, this is 
 771         // why we always prefer using mkstemp()... 
 773         // NB: GetTempFileName() under Windows creates the file, so using 
 774         //     write_excl there would fail 
 775         if ( !fileTemp
->Open(path
, 
 776 #if defined(__WINDOWS__) && !defined(__WXMICROWIN__) 
 781                              wxS_IRUSR 
| wxS_IWUSR
) ) 
 783             // FIXME: If !ok here should we loop and try again with another 
 784             //        file name?  That is the standard recourse if open(O_EXCL) 
 785             //        fails, though of course it should be protected against 
 786             //        possible infinite looping too. 
 788             wxLogError(_("Failed to open temporary file.")); 
 799 // ---------------------------------------------------------------------------- 
 800 // directory operations 
 801 // ---------------------------------------------------------------------------- 
 803 bool wxFileName::Mkdir( int perm
, int flags 
) 
 805     return wxFileName::Mkdir( GetFullPath(), perm
, flags 
); 
 808 bool wxFileName::Mkdir( const wxString
& dir
, int perm
, int flags 
) 
 810     if ( flags 
& wxPATH_MKDIR_FULL 
) 
 812         // split the path in components 
 814         filename
.AssignDir(dir
); 
 817         if ( filename
.HasVolume()) 
 819             currPath 
<< wxGetVolumeString(filename
.GetVolume(), wxPATH_NATIVE
); 
 822         wxArrayString dirs 
= filename
.GetDirs(); 
 823         size_t count 
= dirs
.GetCount(); 
 824         for ( size_t i 
= 0; i 
< count
; i
++ ) 
 827 #if defined(__WXMAC__) && !defined(__DARWIN__) 
 828             // relative pathnames are exactely the other way round under mac... 
 829                 !filename
.IsAbsolute() 
 831                 filename
.IsAbsolute() 
 834                 currPath 
+= wxFILE_SEP_PATH
; 
 837             if (!DirExists(currPath
)) 
 839                 if (!wxMkdir(currPath
, perm
)) 
 841                     // no need to try creating further directories 
 851     return ::wxMkdir( dir
, perm 
); 
 854 bool wxFileName::Rmdir() 
 856     return wxFileName::Rmdir( GetFullPath() ); 
 859 bool wxFileName::Rmdir( const wxString 
&dir 
) 
 861     return ::wxRmdir( dir 
); 
 864 // ---------------------------------------------------------------------------- 
 865 // path normalization 
 866 // ---------------------------------------------------------------------------- 
 868 bool wxFileName::Normalize(int flags
, 
 872     // deal with env vars renaming first as this may seriously change the path 
 873     if ( flags 
& wxPATH_NORM_ENV_VARS 
) 
 875         wxString pathOrig 
= GetFullPath(format
); 
 876         wxString path 
= wxExpandEnvVars(pathOrig
); 
 877         if ( path 
!= pathOrig 
) 
 884     // the existing path components 
 885     wxArrayString dirs 
= GetDirs(); 
 887     // the path to prepend in front to make the path absolute 
 890     format 
= GetFormat(format
); 
 892     // make the path absolute 
 893     if ( (flags 
& wxPATH_NORM_ABSOLUTE
) && !IsAbsolute(format
) ) 
 897             curDir
.AssignCwd(GetVolume()); 
 901             curDir
.AssignDir(cwd
); 
 904         // the path may be not absolute because it doesn't have the volume name 
 905         // but in this case we shouldn't modify the directory components of it 
 906         // but just set the current volume 
 907         if ( !HasVolume() && curDir
.HasVolume() ) 
 909             SetVolume(curDir
.GetVolume()); 
 913                 // yes, it was the case - we don't need curDir then 
 919     // handle ~ stuff under Unix only 
 920     if ( (format 
== wxPATH_UNIX
) && (flags 
& wxPATH_NORM_TILDE
) ) 
 922         if ( !dirs
.IsEmpty() ) 
 924             wxString dir 
= dirs
[0u]; 
 925             if ( !dir
.empty() && dir
[0u] == _T('~') ) 
 927                 curDir
.AssignDir(wxGetUserHome(dir
.c_str() + 1)); 
 934     // transform relative path into abs one 
 937         wxArrayString dirsNew 
= curDir
.GetDirs(); 
 938         size_t count 
= dirs
.GetCount(); 
 939         for ( size_t n 
= 0; n 
< count
; n
++ ) 
 941             dirsNew
.Add(dirs
[n
]); 
 947     // now deal with ".", ".." and the rest 
 949     size_t count 
= dirs
.GetCount(); 
 950     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 952         wxString dir 
= dirs
[n
]; 
 954         if ( flags 
& wxPATH_NORM_DOTS 
) 
 956             if ( dir 
== wxT(".") ) 
 962             if ( dir 
== wxT("..") ) 
 964                 if ( m_dirs
.IsEmpty() ) 
 966                     wxLogError(_("The path '%s' contains too many \"..\"!"), 
 967                                GetFullPath().c_str()); 
 971                 m_dirs
.RemoveAt(m_dirs
.GetCount() - 1); 
 976         if ( (flags 
& wxPATH_NORM_CASE
) && !IsCaseSensitive(format
) ) 
 984 #if defined(__WIN32__) && !defined(__WXWINCE__) && wxUSE_OLE 
 985     if ( (flags 
& wxPATH_NORM_SHORTCUT
) ) 
 988         if (GetShortcutTarget(GetFullPath(format
), filename
)) 
 990             // Repeat this since we may now have a new path 
 991             if ( (flags 
& wxPATH_NORM_CASE
) && !IsCaseSensitive(format
) ) 
 993                 filename
.MakeLower(); 
1001     if ( (flags 
& wxPATH_NORM_CASE
) && !IsCaseSensitive(format
) ) 
1003         // VZ: expand env vars here too? 
1005         m_volume
.MakeLower(); 
1010     // we do have the path now 
1012     // NB: need to do this before (maybe) calling Assign() below 
1015 #if defined(__WIN32__) 
1016     if ( (flags 
& wxPATH_NORM_LONG
) && (format 
== wxPATH_DOS
) ) 
1018         Assign(GetLongPath()); 
1025 // ---------------------------------------------------------------------------- 
1026 // get the shortcut target 
1027 // ---------------------------------------------------------------------------- 
1029 // WinCE (3) doesn't have CLSID_ShellLink, IID_IShellLink definitions. 
1030 // The .lnk file is a plain text file so it should be easy to 
1031 // make it work. Hint from Google Groups: 
1032 // "If you open up a lnk file, you'll see a 
1033 // number, followed by a pound sign (#), followed by more text. The 
1034 // number is the number of characters that follows the pound sign. The 
1035 // characters after the pound sign are the command line (which _can_ 
1036 // include arguments) to be executed. Any path (e.g. \windows\program 
1037 // files\myapp.exe) that includes spaces needs to be enclosed in 
1038 // quotation marks." 
1040 #if defined(__WIN32__) && !defined(__WXWINCE__) && wxUSE_OLE 
1041 // The following lines are necessary under WinCE 
1042 // #include "wx/msw/private.h" 
1043 // #include <ole2.h> 
1045 #if defined(__WXWINCE__) 
1046 #include <shlguid.h> 
1049 bool wxFileName::GetShortcutTarget(const wxString
& shortcutPath
, wxString
& targetFilename
, wxString
* arguments
) 
1051     wxString path
, file
, ext
; 
1052     wxSplitPath(shortcutPath
, & path
, & file
, & ext
); 
1056     bool success 
= false; 
1058     // Assume it's not a shortcut if it doesn't end with lnk 
1059     if (ext
.CmpNoCase(wxT("lnk"))!=0) 
1062     // create a ShellLink object 
1063     hres 
= CoCreateInstance(CLSID_ShellLink
, NULL
, CLSCTX_INPROC_SERVER
, 
1064                             IID_IShellLink
, (LPVOID
*) &psl
); 
1066     if (SUCCEEDED(hres
)) 
1069         hres 
= psl
->QueryInterface( IID_IPersistFile
, (LPVOID 
*) &ppf
); 
1070         if (SUCCEEDED(hres
)) 
1072             WCHAR wsz
[MAX_PATH
]; 
1074             MultiByteToWideChar(CP_ACP
, MB_PRECOMPOSED
, shortcutPath
.mb_str(), -1, wsz
, 
1077             hres 
= ppf
->Load(wsz
, 0); 
1078             if (SUCCEEDED(hres
)) 
1081                 // Wrong prototype in early versions 
1082 #if defined(__MINGW32__) && !wxCHECK_W32API_VERSION(2, 2) 
1083                 psl
->GetPath((CHAR
*) buf
, 2048, NULL
, SLGP_UNCPRIORITY
); 
1085                 psl
->GetPath(buf
, 2048, NULL
, SLGP_UNCPRIORITY
); 
1087                 targetFilename 
= wxString(buf
); 
1088                 success 
= (shortcutPath 
!= targetFilename
); 
1090                 psl
->GetArguments(buf
, 2048); 
1092                 if (!args
.empty() && arguments
) 
1105 // ---------------------------------------------------------------------------- 
1106 // absolute/relative paths 
1107 // ---------------------------------------------------------------------------- 
1109 bool wxFileName::IsAbsolute(wxPathFormat format
) const 
1111     // if our path doesn't start with a path separator, it's not an absolute 
1116     if ( !GetVolumeSeparator(format
).empty() ) 
1118         // this format has volumes and an absolute path must have one, it's not 
1119         // enough to have the full path to bean absolute file under Windows 
1120         if ( GetVolume().empty() ) 
1127 bool wxFileName::MakeRelativeTo(const wxString
& pathBase
, wxPathFormat format
) 
1129     wxFileName fnBase 
= wxFileName::DirName(pathBase
, format
); 
1131     // get cwd only once - small time saving 
1132     wxString cwd 
= wxGetCwd(); 
1133     Normalize(wxPATH_NORM_ALL 
& ~wxPATH_NORM_CASE
, cwd
, format
); 
1134     fnBase
.Normalize(wxPATH_NORM_ALL 
& ~wxPATH_NORM_CASE
, cwd
, format
); 
1136     bool withCase 
= IsCaseSensitive(format
); 
1138     // we can't do anything if the files live on different volumes 
1139     if ( !GetVolume().IsSameAs(fnBase
.GetVolume(), withCase
) ) 
1145     // same drive, so we don't need our volume 
1148     // remove common directories starting at the top 
1149     while ( !m_dirs
.IsEmpty() && !fnBase
.m_dirs
.IsEmpty() && 
1150                 m_dirs
[0u].IsSameAs(fnBase
.m_dirs
[0u], withCase
) ) 
1153         fnBase
.m_dirs
.RemoveAt(0); 
1156     // add as many ".." as needed 
1157     size_t count 
= fnBase
.m_dirs
.GetCount(); 
1158     for ( size_t i 
= 0; i 
< count
; i
++ ) 
1160         m_dirs
.Insert(wxT(".."), 0u); 
1163     if ( format 
== wxPATH_UNIX 
|| format 
== wxPATH_DOS 
) 
1165         // a directory made relative with respect to itself is '.' under Unix 
1166         // and DOS, by definition (but we don't have to insert "./" for the 
1168         if ( m_dirs
.IsEmpty() && IsDir() ) 
1170             m_dirs
.Add(_T('.')); 
1180 // ---------------------------------------------------------------------------- 
1181 // filename kind tests 
1182 // ---------------------------------------------------------------------------- 
1184 bool wxFileName::SameAs(const wxFileName
& filepath
, wxPathFormat format
) const 
1186     wxFileName fn1 
= *this, 
1189     // get cwd only once - small time saving 
1190     wxString cwd 
= wxGetCwd(); 
1191     fn1
.Normalize(wxPATH_NORM_ALL 
| wxPATH_NORM_CASE
, cwd
, format
); 
1192     fn2
.Normalize(wxPATH_NORM_ALL 
| wxPATH_NORM_CASE
, cwd
, format
); 
1194     if ( fn1
.GetFullPath() == fn2
.GetFullPath() ) 
1197     // TODO: compare inodes for Unix, this works even when filenames are 
1198     //       different but files are the same (symlinks) (VZ) 
1204 bool wxFileName::IsCaseSensitive( wxPathFormat format 
) 
1206     // only Unix filenames are truely case-sensitive 
1207     return GetFormat(format
) == wxPATH_UNIX
; 
1211 wxString 
wxFileName::GetForbiddenChars(wxPathFormat format
) 
1213     // Inits to forbidden characters that are common to (almost) all platforms. 
1214     wxString strForbiddenChars 
= wxT("*?"); 
1216     // If asserts, wxPathFormat has been changed. In case of a new path format 
1217     // addition, the following code might have to be updated. 
1218     wxCOMPILE_TIME_ASSERT(wxPATH_MAX 
== 5, wxPathFormatChanged
); 
1219     switch ( GetFormat(format
) ) 
1222             wxFAIL_MSG( wxT("Unknown path format") ); 
1223             // !! Fall through !! 
1229             // On a Mac even names with * and ? are allowed (Tested with OS 
1230             // 9.2.1 and OS X 10.2.5) 
1231             strForbiddenChars 
= wxEmptyString
; 
1235             strForbiddenChars 
+= wxT("\\/:\"<>|"); 
1242     return strForbiddenChars
; 
1246 wxString 
wxFileName::GetVolumeSeparator(wxPathFormat format
) 
1250     if ( (GetFormat(format
) == wxPATH_DOS
) || 
1251          (GetFormat(format
) == wxPATH_VMS
) ) 
1253         sepVol 
= wxFILE_SEP_DSK
; 
1261 wxString 
wxFileName::GetPathSeparators(wxPathFormat format
) 
1264     switch ( GetFormat(format
) ) 
1267             // accept both as native APIs do but put the native one first as 
1268             // this is the one we use in GetFullPath() 
1269             seps 
<< wxFILE_SEP_PATH_DOS 
<< wxFILE_SEP_PATH_UNIX
; 
1273             wxFAIL_MSG( _T("Unknown wxPATH_XXX style") ); 
1277             seps 
= wxFILE_SEP_PATH_UNIX
; 
1281             seps 
= wxFILE_SEP_PATH_MAC
; 
1285             seps 
= wxFILE_SEP_PATH_VMS
; 
1293 wxString 
wxFileName::GetPathTerminators(wxPathFormat format
) 
1295     format 
= GetFormat(format
); 
1297     // under VMS the end of the path is ']', not the path separator used to 
1298     // separate the components 
1299     return format 
== wxPATH_VMS 
? wxString(_T(']')) : GetPathSeparators(format
); 
1303 bool wxFileName::IsPathSeparator(wxChar ch
, wxPathFormat format
) 
1305     // wxString::Find() doesn't work as expected with NUL - it will always find 
1306     // it, so test for it separately 
1307     return ch 
!= _T('\0') && GetPathSeparators(format
).Find(ch
) != wxNOT_FOUND
; 
1310 // ---------------------------------------------------------------------------- 
1311 // path components manipulation 
1312 // ---------------------------------------------------------------------------- 
1314 /* static */ bool wxFileName::IsValidDirComponent(const wxString
& dir
) 
1318         wxFAIL_MSG( _T("empty directory passed to wxFileName::InsertDir()") ); 
1323     const size_t len 
= dir
.length(); 
1324     for ( size_t n 
= 0; n 
< len
; n
++ ) 
1326         if ( dir
[n
] == GetVolumeSeparator() || IsPathSeparator(dir
[n
]) ) 
1328             wxFAIL_MSG( _T("invalid directory component in wxFileName") ); 
1337 void wxFileName::AppendDir( const wxString
& dir 
) 
1339     if ( IsValidDirComponent(dir
) ) 
1343 void wxFileName::PrependDir( const wxString
& dir 
) 
1348 void wxFileName::InsertDir(size_t before
, const wxString
& dir
) 
1350     if ( IsValidDirComponent(dir
) ) 
1351         m_dirs
.Insert(dir
, before
); 
1354 void wxFileName::RemoveDir(size_t pos
) 
1356     m_dirs
.RemoveAt(pos
); 
1359 // ---------------------------------------------------------------------------- 
1361 // ---------------------------------------------------------------------------- 
1363 void wxFileName::SetFullName(const wxString
& fullname
) 
1365     SplitPath(fullname
, NULL 
/* no volume */, NULL 
/* no path */, 
1366                         &m_name
, &m_ext
, &m_hasExt
); 
1369 wxString 
wxFileName::GetFullName() const 
1371     wxString fullname 
= m_name
; 
1374         fullname 
<< wxFILE_SEP_EXT 
<< m_ext
; 
1380 wxString 
wxFileName::GetPath( int flags
, wxPathFormat format 
) const 
1382     format 
= GetFormat( format 
); 
1386     // return the volume with the path as well if requested 
1387     if ( flags 
& wxPATH_GET_VOLUME 
) 
1389         fullpath 
+= wxGetVolumeString(GetVolume(), format
); 
1392     // the leading character 
1397                 fullpath 
+= wxFILE_SEP_PATH_MAC
; 
1402                 fullpath 
+= wxFILE_SEP_PATH_DOS
; 
1406             wxFAIL_MSG( wxT("Unknown path format") ); 
1412                 // normally the absolute file names start with a slash 
1413                 // with one exception: the ones like "~/foo.bar" don't 
1415                 if ( m_dirs
.IsEmpty() || m_dirs
[0u] != _T('~') ) 
1417                     fullpath 
+= wxFILE_SEP_PATH_UNIX
; 
1423             // no leading character here but use this place to unset 
1424             // wxPATH_GET_SEPARATOR flag: under VMS it doesn't make sense 
1425             // as, if I understand correctly, there should never be a dot 
1426             // before the closing bracket 
1427             flags 
&= ~wxPATH_GET_SEPARATOR
; 
1430     if ( m_dirs
.empty() ) 
1432         // there is nothing more 
1436     // then concatenate all the path components using the path separator 
1437     if ( format 
== wxPATH_VMS 
) 
1439         fullpath 
+= wxT('['); 
1442     const size_t dirCount 
= m_dirs
.GetCount(); 
1443     for ( size_t i 
= 0; i 
< dirCount
; i
++ ) 
1448                 if ( m_dirs
[i
] == wxT(".") ) 
1450                     // skip appending ':', this shouldn't be done in this 
1451                     // case as "::" is interpreted as ".." under Unix 
1455                 // convert back from ".." to nothing 
1456                 if ( !m_dirs
[i
].IsSameAs(wxT("..")) ) 
1457                      fullpath 
+= m_dirs
[i
]; 
1461                 wxFAIL_MSG( wxT("Unexpected path format") ); 
1462                 // still fall through 
1466                 fullpath 
+= m_dirs
[i
]; 
1470                 // TODO: What to do with ".." under VMS 
1472                 // convert back from ".." to nothing 
1473                 if ( !m_dirs
[i
].IsSameAs(wxT("..")) ) 
1474                     fullpath 
+= m_dirs
[i
]; 
1478         if ( (flags 
& wxPATH_GET_SEPARATOR
) || (i 
!= dirCount 
- 1) ) 
1479             fullpath 
+= GetPathSeparator(format
); 
1482     if ( format 
== wxPATH_VMS 
) 
1484         fullpath 
+= wxT(']'); 
1490 wxString 
wxFileName::GetFullPath( wxPathFormat format 
) const 
1492     // we already have a function to get the path 
1493     wxString fullpath 
= GetPath(wxPATH_GET_VOLUME 
| wxPATH_GET_SEPARATOR
, 
1496     // now just add the file name and extension to it 
1497     fullpath 
+= GetFullName(); 
1502 // Return the short form of the path (returns identity on non-Windows platforms) 
1503 wxString 
wxFileName::GetShortPath() const 
1505 #if defined(__WXMSW__) && defined(__WIN32__) && !defined(__WXMICROWIN__) && !defined(__WXWINCE__) 
1506     wxString 
path(GetFullPath()); 
1508     DWORD sz 
= ::GetShortPathName(path
, NULL
, 0); 
1512         ok 
= ::GetShortPathName
 
1515                 wxStringBuffer(pathOut
, sz
), 
1524     return GetFullPath(); 
1528 // Return the long form of the path (returns identity on non-Windows platforms) 
1529 wxString 
wxFileName::GetLongPath() const 
1532              path 
= GetFullPath(); 
1534 #if defined(__WIN32__) && !defined(__WXMICROWIN__) 
1535     bool success 
= false; 
1537 #if wxUSE_DYNAMIC_LOADER 
1538     typedef DWORD (WINAPI 
*GET_LONG_PATH_NAME
)(const wxChar 
*, wxChar 
*, DWORD
); 
1540     static bool s_triedToLoad 
= false; 
1542     if ( !s_triedToLoad 
) 
1544         // suppress the errors about missing GetLongPathName[AW] 
1547         s_triedToLoad 
= true; 
1548         wxDynamicLibrary 
dllKernel(_T("kernel32")); 
1549         if ( dllKernel
.IsLoaded() ) 
1551             // may succeed or fail depending on the Windows version 
1552             static GET_LONG_PATH_NAME s_pfnGetLongPathName 
= NULL
; 
1554             s_pfnGetLongPathName 
= (GET_LONG_PATH_NAME
) dllKernel
.GetSymbol(_T("GetLongPathNameW")); 
1556             s_pfnGetLongPathName 
= (GET_LONG_PATH_NAME
) dllKernel
.GetSymbol(_T("GetLongPathNameA")); 
1559             if ( s_pfnGetLongPathName 
) 
1561                 DWORD dwSize 
= (*s_pfnGetLongPathName
)(path
, NULL
, 0); 
1562                 bool ok 
= dwSize 
> 0; 
1566                     DWORD sz 
= (*s_pfnGetLongPathName
)(path
, NULL
, 0); 
1570                         ok 
= (*s_pfnGetLongPathName
) 
1573                                 wxStringBuffer(pathOut
, sz
), 
1585 #endif // wxUSE_DYNAMIC_LOADER 
1589         // The OS didn't support GetLongPathName, or some other error. 
1590         // We need to call FindFirstFile on each component in turn. 
1592         WIN32_FIND_DATA findFileData
; 
1596             pathOut 
= GetVolume() + 
1597                       GetVolumeSeparator(wxPATH_DOS
) + 
1598                       GetPathSeparator(wxPATH_DOS
); 
1600             pathOut 
= wxEmptyString
; 
1602         wxArrayString dirs 
= GetDirs(); 
1603         dirs
.Add(GetFullName()); 
1607         size_t count 
= dirs
.GetCount(); 
1608         for ( size_t i 
= 0; i 
< count
; i
++ ) 
1610             // We're using pathOut to collect the long-name path, but using a 
1611             // temporary for appending the last path component which may be 
1613             tmpPath 
= pathOut 
+ dirs
[i
]; 
1615             if ( tmpPath
.empty() ) 
1618             // can't see this being necessary? MF 
1619             if ( tmpPath
.Last() == GetVolumeSeparator(wxPATH_DOS
) ) 
1621                 // Can't pass a drive and root dir to FindFirstFile, 
1622                 // so continue to next dir 
1623                 tmpPath 
+= wxFILE_SEP_PATH
; 
1628             hFind 
= ::FindFirstFile(tmpPath
, &findFileData
); 
1629             if (hFind 
== INVALID_HANDLE_VALUE
) 
1631                 // Error: most likely reason is that path doesn't exist, so 
1632                 // append any unprocessed parts and return 
1633                 for ( i 
+= 1; i 
< count
; i
++ ) 
1634                     tmpPath 
+= wxFILE_SEP_PATH 
+ dirs
[i
]; 
1639             pathOut 
+= findFileData
.cFileName
; 
1640             if ( (i 
< (count
-1)) ) 
1641                 pathOut 
+= wxFILE_SEP_PATH
; 
1648 #endif // Win32/!Win32 
1653 wxPathFormat 
wxFileName::GetFormat( wxPathFormat format 
) 
1655     if (format 
== wxPATH_NATIVE
) 
1657 #if defined(__WXMSW__) || defined(__OS2__) || defined(__DOS__) 
1658         format 
= wxPATH_DOS
; 
1659 #elif defined(__WXMAC__) && !defined(__DARWIN__) 
1660         format 
= wxPATH_MAC
; 
1661 #elif defined(__VMS) 
1662         format 
= wxPATH_VMS
; 
1664         format 
= wxPATH_UNIX
; 
1670 // ---------------------------------------------------------------------------- 
1671 // path splitting function 
1672 // ---------------------------------------------------------------------------- 
1676 wxFileName::SplitVolume(const wxString
& fullpathWithVolume
, 
1677                         wxString 
*pstrVolume
, 
1679                         wxPathFormat format
) 
1681     format 
= GetFormat(format
); 
1683     wxString fullpath 
= fullpathWithVolume
; 
1685     // special Windows UNC paths hack: transform \\share\path into share:path 
1686     if ( format 
== wxPATH_DOS 
) 
1688         if ( fullpath
.length() >= 4 && 
1689                 fullpath
[0u] == wxFILE_SEP_PATH_DOS 
&& 
1690                     fullpath
[1u] == wxFILE_SEP_PATH_DOS 
) 
1692             fullpath
.erase(0, 2); 
1694             size_t posFirstSlash 
= 
1695                 fullpath
.find_first_of(GetPathTerminators(format
)); 
1696             if ( posFirstSlash 
!= wxString::npos 
) 
1698                 fullpath
[posFirstSlash
] = wxFILE_SEP_DSK
; 
1700                 // UNC paths are always absolute, right? (FIXME) 
1701                 fullpath
.insert(posFirstSlash 
+ 1, 1, wxFILE_SEP_PATH_DOS
); 
1706     // We separate the volume here 
1707     if ( format 
== wxPATH_DOS 
|| format 
== wxPATH_VMS 
) 
1709         wxString sepVol 
= GetVolumeSeparator(format
); 
1711         size_t posFirstColon 
= fullpath
.find_first_of(sepVol
); 
1712         if ( posFirstColon 
!= wxString::npos 
) 
1716                 *pstrVolume 
= fullpath
.Left(posFirstColon
); 
1719             // remove the volume name and the separator from the full path 
1720             fullpath
.erase(0, posFirstColon 
+ sepVol
.length()); 
1725         *pstrPath 
= fullpath
; 
1729 void wxFileName::SplitPath(const wxString
& fullpathWithVolume
, 
1730                            wxString 
*pstrVolume
, 
1735                            wxPathFormat format
) 
1737     format 
= GetFormat(format
); 
1740     SplitVolume(fullpathWithVolume
, pstrVolume
, &fullpath
, format
); 
1742     // find the positions of the last dot and last path separator in the path 
1743     size_t posLastDot 
= fullpath
.find_last_of(wxFILE_SEP_EXT
); 
1744     size_t posLastSlash 
= fullpath
.find_last_of(GetPathTerminators(format
)); 
1746     // check whether this dot occurs at the very beginning of a path component 
1747     if ( (posLastDot 
!= wxString::npos
) && 
1749             IsPathSeparator(fullpath
[posLastDot 
- 1]) || 
1750             (format 
== wxPATH_VMS 
&& fullpath
[posLastDot 
- 1] == _T(']'))) ) 
1752         // dot may be (and commonly -- at least under Unix -- is) the first 
1753         // character of the filename, don't treat the entire filename as 
1754         // extension in this case 
1755         posLastDot 
= wxString::npos
; 
1758     // if we do have a dot and a slash, check that the dot is in the name part 
1759     if ( (posLastDot 
!= wxString::npos
) && 
1760          (posLastSlash 
!= wxString::npos
) && 
1761          (posLastDot 
< posLastSlash
) ) 
1763         // the dot is part of the path, not the start of the extension 
1764         posLastDot 
= wxString::npos
; 
1767     // now fill in the variables provided by user 
1770         if ( posLastSlash 
== wxString::npos 
) 
1777             // take everything up to the path separator but take care to make 
1778             // the path equal to something like '/', not empty, for the files 
1779             // immediately under root directory 
1780             size_t len 
= posLastSlash
; 
1782             // this rule does not apply to mac since we do not start with colons (sep) 
1783             // except for relative paths 
1784             if ( !len 
&& format 
!= wxPATH_MAC
) 
1787             *pstrPath 
= fullpath
.Left(len
); 
1789             // special VMS hack: remove the initial bracket 
1790             if ( format 
== wxPATH_VMS 
) 
1792                 if ( (*pstrPath
)[0u] == _T('[') ) 
1793                     pstrPath
->erase(0, 1); 
1800         // take all characters starting from the one after the last slash and 
1801         // up to, but excluding, the last dot 
1802         size_t nStart 
= posLastSlash 
== wxString::npos 
? 0 : posLastSlash 
+ 1; 
1804         if ( posLastDot 
== wxString::npos 
) 
1806             // take all until the end 
1807             count 
= wxString::npos
; 
1809         else if ( posLastSlash 
== wxString::npos 
) 
1813         else // have both dot and slash 
1815             count 
= posLastDot 
- posLastSlash 
- 1; 
1818         *pstrName 
= fullpath
.Mid(nStart
, count
); 
1821     // finally deal with the extension here: we have an added complication that 
1822     // extension may be empty (but present) as in "foo." where trailing dot 
1823     // indicates the empty extension at the end -- and hence we must remember 
1824     // that we have it independently of pstrExt 
1825     if ( posLastDot 
== wxString::npos 
) 
1835         // take everything after the dot 
1837             *pstrExt 
= fullpath
.Mid(posLastDot 
+ 1); 
1844 void wxFileName::SplitPath(const wxString
& fullpath
, 
1848                            wxPathFormat format
) 
1851     SplitPath(fullpath
, &volume
, path
, name
, ext
, format
); 
1855         path
->Prepend(wxGetVolumeString(volume
, format
)); 
1859 // ---------------------------------------------------------------------------- 
1861 // ---------------------------------------------------------------------------- 
1865 bool wxFileName::SetTimes(const wxDateTime 
*dtAccess
, 
1866                           const wxDateTime 
*dtMod
, 
1867                           const wxDateTime 
*dtCreate
) 
1869 #if defined(__WIN32__) 
1872         // VZ: please let me know how to do this if you can 
1873         wxFAIL_MSG( _T("SetTimes() not implemented for the directories") ); 
1877         wxFileHandle 
fh(GetFullPath(), wxFileHandle::Write
); 
1880             FILETIME ftAccess
, ftCreate
, ftWrite
; 
1883                 ConvertWxToFileTime(&ftCreate
, *dtCreate
); 
1885                 ConvertWxToFileTime(&ftAccess
, *dtAccess
); 
1887                 ConvertWxToFileTime(&ftWrite
, *dtMod
); 
1889             if ( ::SetFileTime(fh
, 
1890                                dtCreate 
? &ftCreate 
: NULL
, 
1891                                dtAccess 
? &ftAccess 
: NULL
, 
1892                                dtMod 
? &ftWrite 
: NULL
) ) 
1898 #elif defined(__UNIX_LIKE__) || (defined(__DOS__) && defined(__WATCOMC__)) 
1899     if ( !dtAccess 
&& !dtMod 
) 
1901         // can't modify the creation time anyhow, don't try 
1905     // if dtAccess or dtMod is not specified, use the other one (which must be 
1906     // non NULL because of the test above) for both times 
1908     utm
.actime 
= dtAccess 
? dtAccess
->GetTicks() : dtMod
->GetTicks(); 
1909     utm
.modtime 
= dtMod 
? dtMod
->GetTicks() : dtAccess
->GetTicks(); 
1910     if ( utime(GetFullPath().fn_str(), &utm
) == 0 ) 
1914 #else // other platform 
1917     wxLogSysError(_("Failed to modify file times for '%s'"), 
1918                   GetFullPath().c_str()); 
1923 bool wxFileName::Touch() 
1925 #if defined(__UNIX_LIKE__) 
1926     // under Unix touching file is simple: just pass NULL to utime() 
1927     if ( utime(GetFullPath().fn_str(), NULL
) == 0 ) 
1932     wxLogSysError(_("Failed to touch the file '%s'"), GetFullPath().c_str()); 
1935 #else // other platform 
1936     wxDateTime dtNow 
= wxDateTime::Now(); 
1938     return SetTimes(&dtNow
, &dtNow
, NULL 
/* don't change create time */); 
1942 bool wxFileName::GetTimes(wxDateTime 
*dtAccess
, 
1944                           wxDateTime 
*dtCreate
) const 
1946 #if defined(__WIN32__) 
1947     // we must use different methods for the files and directories under 
1948     // Windows as CreateFile(GENERIC_READ) doesn't work for the directories and 
1949     // CreateFile(FILE_FLAG_BACKUP_SEMANTICS) works -- but only under NT and 
1952     FILETIME ftAccess
, ftCreate
, ftWrite
; 
1955         // implemented in msw/dir.cpp 
1956         extern bool wxGetDirectoryTimes(const wxString
& dirname
, 
1957                                         FILETIME 
*, FILETIME 
*, FILETIME 
*); 
1959         // we should pass the path without the trailing separator to 
1960         // wxGetDirectoryTimes() 
1961         ok 
= wxGetDirectoryTimes(GetPath(wxPATH_GET_VOLUME
), 
1962                                  &ftAccess
, &ftCreate
, &ftWrite
); 
1966         wxFileHandle 
fh(GetFullPath(), wxFileHandle::Read
); 
1969             ok 
= ::GetFileTime(fh
, 
1970                                dtCreate 
? &ftCreate 
: NULL
, 
1971                                dtAccess 
? &ftAccess 
: NULL
, 
1972                                dtMod 
? &ftWrite 
: NULL
) != 0; 
1983             ConvertFileTimeToWx(dtCreate
, ftCreate
); 
1985             ConvertFileTimeToWx(dtAccess
, ftAccess
); 
1987             ConvertFileTimeToWx(dtMod
, ftWrite
); 
1991 #elif defined(__UNIX_LIKE__) || defined(__WXMAC__) || (defined(__DOS__) && defined(__WATCOMC__)) 
1993     if ( wxStat( GetFullPath().c_str(), &stBuf
) == 0 ) 
1996             dtAccess
->Set(stBuf
.st_atime
); 
1998             dtMod
->Set(stBuf
.st_mtime
); 
2000             dtCreate
->Set(stBuf
.st_ctime
); 
2004 #else // other platform 
2007     wxLogSysError(_("Failed to retrieve file times for '%s'"), 
2008                   GetFullPath().c_str()); 
2013 #endif // wxUSE_DATETIME 
2017 const short kMacExtensionMaxLength 
= 16 ; 
2018 class MacDefaultExtensionRecord
 
2021   MacDefaultExtensionRecord() 
2024     m_type 
= m_creator 
= 0 ; 
2026   MacDefaultExtensionRecord( const MacDefaultExtensionRecord
& from 
) 
2028     wxStrcpy( m_ext 
, from
.m_ext 
) ; 
2029     m_type 
= from
.m_type 
; 
2030     m_creator 
= from
.m_creator 
; 
2032   MacDefaultExtensionRecord( const wxChar 
* extension 
, OSType type 
, OSType creator 
) 
2034     wxStrncpy( m_ext 
, extension 
, kMacExtensionMaxLength 
) ; 
2035     m_ext
[kMacExtensionMaxLength
] = 0 ; 
2037     m_creator 
= creator 
; 
2039   wxChar m_ext
[kMacExtensionMaxLength
] ; 
2044 #include "wx/dynarray.h" 
2045 WX_DECLARE_OBJARRAY(MacDefaultExtensionRecord
, MacDefaultExtensionArray
) ; 
2047 bool gMacDefaultExtensionsInited 
= false ; 
2049 #include "wx/arrimpl.cpp" 
2051 WX_DEFINE_EXPORTED_OBJARRAY(MacDefaultExtensionArray
) ; 
2053 MacDefaultExtensionArray gMacDefaultExtensions 
; 
2055 // load the default extensions 
2056 MacDefaultExtensionRecord gDefaults
[] = 
2058     MacDefaultExtensionRecord( wxT("txt") , 'TEXT' , 'ttxt' ) , 
2059     MacDefaultExtensionRecord( wxT("tif") , 'TIFF' , '****' ) , 
2060     MacDefaultExtensionRecord( wxT("jpg") , 'JPEG' , '****' ) , 
2063 static void MacEnsureDefaultExtensionsLoaded() 
2065     if ( !gMacDefaultExtensionsInited 
) 
2067         // we could load the pc exchange prefs here too 
2068         for ( size_t i 
= 0 ; i 
< WXSIZEOF( gDefaults 
) ; ++i 
) 
2070             gMacDefaultExtensions
.Add( gDefaults
[i
] ) ; 
2072         gMacDefaultExtensionsInited 
= true ; 
2076 bool wxFileName::MacSetTypeAndCreator( wxUint32 type 
, wxUint32 creator 
) 
2079     FSCatalogInfo catInfo
; 
2082     if ( wxMacPathToFSRef( GetFullPath() , &fsRef 
) == noErr 
) 
2084         if ( FSGetCatalogInfo (&fsRef
, kFSCatInfoFinderInfo
, &catInfo
, NULL
, NULL
, NULL
) == noErr 
) 
2086             finfo 
= (FileInfo
*)&catInfo
.finderInfo
; 
2087             finfo
->fileType 
= type 
; 
2088             finfo
->fileCreator 
= creator 
; 
2089             FSSetCatalogInfo( &fsRef
, kFSCatInfoFinderInfo
, &catInfo 
) ; 
2096 bool wxFileName::MacGetTypeAndCreator( wxUint32 
*type 
, wxUint32 
*creator 
) 
2099     FSCatalogInfo catInfo
; 
2102     if ( wxMacPathToFSRef( GetFullPath() , &fsRef 
) == noErr 
) 
2104         if ( FSGetCatalogInfo (&fsRef
, kFSCatInfoFinderInfo
, &catInfo
, NULL
, NULL
, NULL
) == noErr 
) 
2106             finfo 
= (FileInfo
*)&catInfo
.finderInfo
; 
2107             *type 
= finfo
->fileType 
; 
2108             *creator 
= finfo
->fileCreator 
; 
2115 bool wxFileName::MacSetDefaultTypeAndCreator() 
2117     wxUint32 type 
, creator 
; 
2118     if ( wxFileName::MacFindDefaultTypeAndCreator(GetExt() , &type 
, 
2121         return MacSetTypeAndCreator( type 
, creator 
) ; 
2126 bool wxFileName::MacFindDefaultTypeAndCreator( const wxString
& ext 
, wxUint32 
*type 
, wxUint32 
*creator 
) 
2128   MacEnsureDefaultExtensionsLoaded() ; 
2129   wxString extl 
= ext
.Lower() ; 
2130   for( int i 
= gMacDefaultExtensions
.Count() - 1 ; i 
>= 0 ; --i 
) 
2132     if ( gMacDefaultExtensions
.Item(i
).m_ext 
== extl 
) 
2134       *type 
= gMacDefaultExtensions
.Item(i
).m_type 
; 
2135       *creator 
= gMacDefaultExtensions
.Item(i
).m_creator 
; 
2142 void wxFileName::MacRegisterDefaultTypeAndCreator( const wxString
& ext 
, wxUint32 type 
, wxUint32 creator 
) 
2144   MacEnsureDefaultExtensionsLoaded() ; 
2145   MacDefaultExtensionRecord rec 
; 
2147   rec
.m_creator 
= creator 
; 
2148   wxStrncpy( rec
.m_ext 
, ext
.Lower().c_str() , kMacExtensionMaxLength 
) ; 
2149   gMacDefaultExtensions
.Add( rec 
) ;