1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/msw/filedlg.cpp 
   3 // Purpose:     wxFileDialog 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart and Markus Holzem 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // ============================================================================ 
  14 // ============================================================================ 
  16 // ---------------------------------------------------------------------------- 
  18 // ---------------------------------------------------------------------------- 
  21     #pragma implementation "filedlg.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  33     #include "wx/msgdlg.h" 
  34     #include "wx/dialog.h" 
  35     #include "wx/filedlg.h" 
  41 #include "wx/msw/private.h" 
  43 #if !defined(__WIN32__) || defined(__SALFORDC__) || defined(__WXWINE__) 
  51 #include "wx/tokenzr.h" 
  53 // ---------------------------------------------------------------------------- 
  55 // ---------------------------------------------------------------------------- 
  58 # define wxMAXPATH   4096 
  60 # define wxMAXPATH   1024 
  63 # define wxMAXFILE   1024 
  67 // ============================================================================ 
  69 // ============================================================================ 
  71 // ---------------------------------------------------------------------------- 
  73 // ---------------------------------------------------------------------------- 
  75 IMPLEMENT_CLASS(wxFileDialog
, wxDialog
) 
  77 // ---------------------------------------------------------------------------- 
  79 // ---------------------------------------------------------------------------- 
  81 wxString 
wxFileSelector(const wxChar 
*title
, 
  82                         const wxChar 
*defaultDir
, 
  83                         const wxChar 
*defaultFileName
, 
  84                         const wxChar 
*defaultExtension
, 
  90     // In the original implementation, defaultExtension is passed to the 
  91     // lpstrDefExt member of OPENFILENAME. This extension, if non-NULL, is 
  92     // appended to the filename if the user fails to type an extension. The new 
  93     // implementation (taken from wxFileSelectorEx) appends the extension 
  94     // automatically, by looking at the filter specification. In fact this 
  95     // should be better than the native Microsoft implementation because 
  96     // Windows only allows *one* default extension, whereas here we do the 
  97     // right thing depending on the filter the user has chosen. 
  99     // If there's a default extension specified but no filter, we create a 
 103     if ( defaultExtension 
&& !filter 
) 
 104         filter2 
= wxString(wxT("*.")) + defaultExtension
; 
 108     wxString defaultDirString
; 
 110         defaultDirString 
= defaultDir
; 
 112     wxString defaultFilenameString
; 
 114         defaultFilenameString 
= defaultFileName
; 
 116     wxFileDialog 
fileDialog(parent
, title
, defaultDirString
, 
 117                             defaultFilenameString
, filter2
, 
 118                             flags
, wxPoint(x
, y
)); 
 119     if( wxStrlen(defaultExtension
) != 0 ) 
 124         for( unsigned int i 
= 0; i 
< filter2
.Len(); i
++ ) 
 126             if( filter2
.GetChar(i
) == wxT('|') ) 
 128                 // save the start index of the new filter 
 129                 unsigned int is 
= i
++; 
 131                 // find the end of the filter 
 132                 for( ; i 
< filter2
.Len(); i
++ ) 
 134                     if(filter2
[i
] == wxT('|')) 
 138                 if( i
-is
-1 > 0 && is
+1 < filter2
.Len() ) 
 140                     if( filter2
.Mid(is
+1,i
-is
-1).Contains(defaultExtension
) ) 
 142                         filterFind 
= filterIndex
; 
 151         fileDialog
.SetFilterIndex(filterFind
); 
 155     if ( fileDialog
.ShowModal() == wxID_OK 
) 
 157         filename 
= fileDialog
.GetPath(); 
 164 wxString 
wxFileSelectorEx(const wxChar 
*title
, 
 165                        const wxChar 
*defaultDir
, 
 166                        const wxChar 
*defaultFileName
, 
 167                        int* defaultFilterIndex
, 
 168                        const wxChar 
*filter
, 
 175     wxFileDialog 
fileDialog(parent
, 
 176                             title 
? title 
: wxT(""), 
 177                             defaultDir 
? defaultDir 
: wxT(""), 
 178                             defaultFileName 
? defaultFileName 
: wxT(""), 
 179                             filter 
? filter 
: wxT(""), 
 180                             flags
, wxPoint(x
, y
)); 
 183     if ( fileDialog
.ShowModal() == wxID_OK 
) 
 185         if ( defaultFilterIndex 
) 
 186             *defaultFilterIndex 
= fileDialog
.GetFilterIndex(); 
 188         filename 
= fileDialog
.GetPath(); 
 194 wxFileDialog::wxFileDialog(wxWindow 
*parent
, const wxString
& message
, 
 195         const wxString
& defaultDir
, const wxString
& defaultFileName
, const wxString
& wildCard
, 
 196         long style
, const wxPoint
& pos
) 
 199     m_dialogStyle 
= style
; 
 200     if ( ( m_dialogStyle 
& wxMULTIPLE 
) && ( m_dialogStyle 
& wxSAVE 
) ) 
 201         m_dialogStyle 
&= ~wxMULTIPLE
; 
 204     m_fileName 
= defaultFileName
; 
 206     m_wildCard 
= wildCard
; 
 210 void wxFileDialog::GetPaths(wxArrayString
& paths
) const 
 215     if ( m_dir
.Last() != _T('\\') ) 
 218     size_t count 
= m_fileNames
.GetCount(); 
 219     for ( size_t n 
= 0; n 
< count
; n
++ ) 
 221         paths
.Add(dir 
+ m_fileNames
[n
]); 
 225 int wxFileDialog::ShowModal() 
 228     if (m_parent
) hWnd 
= (HWND
) m_parent
->GetHWND(); 
 229     if (!hWnd 
&& wxTheApp
->GetTopWindow()) 
 230         hWnd 
= (HWND
) wxTheApp
->GetTopWindow()->GetHWND(); 
 232     static wxChar fileNameBuffer 
[ wxMAXPATH 
];           // the file-name 
 233     wxChar        titleBuffer    
[ wxMAXFILE
+1+wxMAXEXT 
];  // the file-name, without path 
 235     *fileNameBuffer 
= wxT('\0'); 
 236     *titleBuffer    
= wxT('\0'); 
 239     if ( (m_dialogStyle 
& wxHIDE_READONLY
) || (m_dialogStyle 
& wxSAVE
) ) 
 240         msw_flags 
|= OFN_HIDEREADONLY
; 
 241     if ( m_dialogStyle 
& wxFILE_MUST_EXIST 
) 
 242         msw_flags 
|= OFN_PATHMUSTEXIST 
| OFN_FILEMUSTEXIST
; 
 243     if (m_dialogStyle 
& wxMULTIPLE 
) 
 245 #if defined(OFN_EXPLORER) 
 247 #endif // OFN_EXPLORER 
 248         OFN_ALLOWMULTISELECT
; 
 253     // the OPENFILENAME struct has been extended in newer version of 
 254     // comcdlg32.dll, but as we don't use the extended fields anyhow, set 
 255     // the struct size to the old value - otherwise, the programs compiled 
 256     // with new headers will not work with the old libraries 
 257 #if defined(_WIN32_WINNT) && (_WIN32_WINNT >= 0x0500) 
 258     of
.lStructSize       
= sizeof(OPENFILENAME
) - 
 259                            (sizeof(void *) + 2*sizeof(DWORD
)); 
 261     of
.lStructSize       
= sizeof(OPENFILENAME
); 
 265     of
.lpstrTitle        
= WXSTRINGCAST m_message
; 
 266     of
.lpstrFileTitle    
= titleBuffer
; 
 267     of
.nMaxFileTitle     
= wxMAXFILE 
+ 1 + wxMAXEXT
;    // Windows 3.0 and 3.1 
 269     // Convert forward slashes to backslashes (file selector doesn't like 
 272     size_t len 
= m_dir
.Length(); 
 273     for (i 
= 0; i 
< len
; i
++) 
 274         if (m_dir
[i
] == wxT('/')) 
 275             m_dir
[i
] = wxT('\\'); 
 277     of
.lpstrInitialDir   
= m_dir
.c_str(); 
 279     of
.Flags             
= msw_flags
; 
 282     //=== Like Alejandro Sierra's wildcard modification >>=================== 
 284        In wxFileSelector you can put, instead of a single wild_card, 
 285        pairs of strings separated by '|'. 
 286        The first string is a description, and the 
 287        second is the wild card. You can put any number of pairs. 
 289        eg.  "description1 (*.ex1)|*.ex1|description2 (*.ex2)|*.ex2" 
 291        If you put a single wild card, it works as before the modification. 
 293     //======================================================================= 
 296     if ( wxStrlen(m_wildCard
) == 0 ) 
 297         theFilter 
= wxString(wxT("*.*")); 
 299         theFilter 
= m_wildCard 
; 
 300     wxString filterBuffer
; 
 302     if ( !wxStrchr( theFilter
, wxT('|') ) ) {    // only one filter ==> default text 
 303         filterBuffer
.Printf(_("Files (%s)|%s"), 
 304                             theFilter
.c_str(), theFilter
.c_str()); 
 306     else {                                // more then one filter 
 307         filterBuffer 
= theFilter
; 
 311     filterBuffer 
+= wxT("|"); 
 313     for (i 
= 0; i 
< filterBuffer
.Len(); i
++ ) { 
 314         if ( filterBuffer
.GetChar(i
) == wxT('|') ) { 
 315             filterBuffer
[i
] = wxT('\0'); 
 319     of
.lpstrFilter  
= (LPTSTR
)(const wxChar 
*)filterBuffer
; 
 320     of
.nFilterIndex 
= m_filterIndex 
+ 1; 
 322     //=== Setting defaultFileName >>========================================= 
 324     wxStrncpy( fileNameBuffer
, (const wxChar 
*)m_fileName
, wxMAXPATH
-1 ); 
 325     fileNameBuffer
[ wxMAXPATH
-1 ] = wxT('\0'); 
 327     of
.lpstrFile 
= fileNameBuffer
;  // holds returned filename 
 328     of
.nMaxFile  
= wxMAXPATH
; 
 330     //== Execute FileDialog >>================================================= 
 332     bool success 
= (m_dialogStyle 
& wxSAVE 
? GetSaveFileName(&of
) 
 333                                            : GetOpenFileName(&of
)) != 0; 
 335     DWORD errCode 
= CommDlgExtendedError(); 
 338     if (!success 
&& (errCode 
== CDERR_STRUCTSIZE
)) 
 340         // The struct size has changed so try a smaller or bigger size 
 342         int oldStructSize 
= of
.lStructSize
; 
 343         of
.lStructSize       
= oldStructSize 
- (sizeof(void *) + 2*sizeof(DWORD
)); 
 344         success 
= (m_dialogStyle 
& wxSAVE
) ? (GetSaveFileName(&of
) != 0) 
 345                                             : (GetOpenFileName(&of
) != 0); 
 346         errCode 
= CommDlgExtendedError(); 
 348         if (!success 
&& (errCode 
== CDERR_STRUCTSIZE
)) 
 350             of
.lStructSize       
= oldStructSize 
+ (sizeof(void *) + 2*sizeof(DWORD
)); 
 351             success 
= (m_dialogStyle 
& wxSAVE
) ? (GetSaveFileName(&of
) != 0) 
 352                                             : (GetOpenFileName(&of
) != 0); 
 361         if ( ( m_dialogStyle 
& wxMULTIPLE 
) && 
 362 #if defined(OFN_EXPLORER) 
 363              ( fileNameBuffer
[of
.nFileOffset
-1] == wxT('\0') ) ) 
 365              ( fileNameBuffer
[of
.nFileOffset
-1] == wxT(' ') ) ) 
 366 #endif // OFN_EXPLORER 
 368 #if defined(OFN_EXPLORER) 
 369             m_dir 
= fileNameBuffer
; 
 371             m_fileName 
= &fileNameBuffer
[i
]; 
 372             m_fileNames
.Add(m_fileName
); 
 373             i 
+= m_fileName
.Len() + 1; 
 375             while (fileNameBuffer
[i
] != wxT('\0')) 
 377                 m_fileNames
.Add(&fileNameBuffer
[i
]); 
 378                 i 
+= wxStrlen(&fileNameBuffer
[i
]) + 1; 
 381             wxStringTokenizer 
toke(fileNameBuffer
, " \t\r\n"); 
 382             m_dir 
= toke
.GetNextToken(); 
 383             m_fileName 
= toke
.GetNextToken(); 
 384             m_fileNames
.Add(m_fileName
); 
 386             while (toke
.HasMoreTokens()) 
 387                 m_fileNames
.Add(toke
.GetNextToken()); 
 388 #endif // OFN_EXPLORER 
 391             if ( m_dir
.Last() != _T('\\') ) 
 395             m_path 
= dir 
+ m_fileName
; 
 399             const wxChar
* extension 
= NULL
; 
 401             //=== Adding the correct extension >>================================= 
 403             m_filterIndex 
= (int)of
.nFilterIndex 
- 1; 
 405             if ( !of
.nFileExtension 
|| (of
.nFileExtension 
&& fileNameBuffer
[ of
.nFileExtension
-1] != wxT('.')) ) 
 406             {                                    // user has typed an filename 
 407                 // without an extension: 
 409                 int   maxFilter 
= (int)(of
.nFilterIndex
*2L-1L); 
 410                 extension 
= filterBuffer
; 
 412                 for( int i 
= 0; i 
< maxFilter
; i
++ ) {          // get extension 
 413                     extension 
= extension 
+ wxStrlen( extension 
) +1; 
 416                 extension 
= wxStrrchr( extension
, wxT('.') ); 
 417                 if (  extension                                 
// != "blabla" 
 418                         && !wxStrrchr( extension
, wxT('*') )       // != "blabla.*" 
 419                         && !wxStrrchr( extension
, wxT('?') )       // != "blabla.?" 
 420                         && extension
[1]                           // != "blabla." 
 421                         && extension
[1] != wxT(' ') )              // != "blabla. " 
 423                     // now concat extension to the fileName: 
 424                     m_fileName 
= wxString(fileNameBuffer
) + extension
; 
 426                     int len 
= wxStrlen( fileNameBuffer 
); 
 427                     wxStrncpy( fileNameBuffer 
+ len
, extension
, wxMAXPATH 
- len 
); 
 428                     fileNameBuffer
[ wxMAXPATH 
-1 ] = wxT('\0'); 
 432             m_path 
= fileNameBuffer
; 
 433             m_fileName 
= wxFileNameFromPath(fileNameBuffer
); 
 434             m_fileNames
.Add(m_fileName
); 
 435             m_dir 
= wxPathOnly(fileNameBuffer
); 
 439         //=== Simulating the wxOVERWRITE_PROMPT >>============================ 
 441         if ( (m_dialogStyle 
& wxOVERWRITE_PROMPT
) && 
 442              ::wxFileExists( fileNameBuffer 
) ) 
 444             wxString messageText
; 
 445             messageText
.Printf(_("Replace file '%s'?"), fileNameBuffer
); 
 447             if ( wxMessageBox(messageText
, m_message
, wxYES_NO 
) != wxYES 
) 
 456         // common dialog failed - why? 
 458         DWORD dwErr 
= CommDlgExtendedError(); 
 461             // this msg is only for developers 
 462             wxLogError(wxT("Common dialog failed with error code %0lx."), 
 465         //else: it was just cancelled 
 469     return success 
? wxID_OK 
: wxID_CANCEL
; 
 473 // Generic file load/save dialog (for internal use only) 
 475 wxString 
wxDefaultFileSelector(bool load
, 
 477                                const wxChar 
*extension
, 
 478                                const wxChar 
*default_name
, 
 483   if (load
) str 
= _("Load %s file"); 
 484   else str 
= _("Save %s file"); 
 485   prompt
.Printf(str
, what
); 
 487   const wxChar 
*ext 
= extension
; 
 488   if (*ext 
== wxT('.')) 
 492   wild
.Printf(wxT("*.%s"), ext
); 
 494   return wxFileSelector (prompt
, NULL
, default_name
, ext
, wild
, 0, parent
); 
 497 // Generic file load dialog 
 498 WXDLLEXPORT wxString 
wxLoadFileSelector(const wxChar 
*what
, 
 499                                         const wxChar 
*extension
, 
 500                                         const wxChar 
*default_name
, 
 503     return wxDefaultFileSelector(TRUE
, what
, extension
, default_name
, parent
); 
 506 // Generic file save dialog 
 507 WXDLLEXPORT wxString 
wxSaveFileSelector(const wxChar 
*what
, 
 508                                         const wxChar 
*extension
, 
 509                                         const wxChar 
*default_name
, 
 512     return wxDefaultFileSelector(FALSE
, what
, extension
, default_name
, parent
);