]> git.saurik.com Git - wxWidgets.git/blame - src/msw/filedlg.cpp
always return length of the string, *not* size of the buffer from wxMBConv_win32...
[wxWidgets.git] / src / msw / filedlg.cpp
CommitLineData
2bda0e17 1/////////////////////////////////////////////////////////////////////////////
f6bcfd97 2// Name: src/msw/filedlg.cpp
2bda0e17
KB
3// Purpose: wxFileDialog
4// Author: Julian Smart
5// Modified by:
6// Created: 01/02/97
7// RCS-ID: $Id$
6c9a19aa 8// Copyright: (c) Julian Smart
e15e548b 9// Licence: wxWindows licence
2bda0e17
KB
10/////////////////////////////////////////////////////////////////////////////
11
f6bcfd97
BP
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
14f355c2 20#if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
ba681060 21 #pragma implementation "filedlg.h"
2bda0e17
KB
22#endif
23
24// For compilers that support precompilation, includes "wx.h".
25#include "wx/wxprec.h"
26
27#ifdef __BORLANDC__
ba681060 28 #pragma hdrstop
2bda0e17
KB
29#endif
30
449110cd 31#if wxUSE_FILEDLG && !wxUSE_SMARTPHONE
1e6feb95 32
2bda0e17 33#ifndef WX_PRECOMP
ba681060
VZ
34 #include "wx/utils.h"
35 #include "wx/msgdlg.h"
ba681060 36 #include "wx/filedlg.h"
2b5f62a0 37 #include "wx/filefn.h"
ba681060 38 #include "wx/intl.h"
2662e49e 39 #include "wx/log.h"
f6bcfd97 40 #include "wx/app.h"
8f177c8e 41#endif
2bda0e17 42
f6bcfd97
BP
43#include "wx/msw/private.h"
44
4676948b 45#if !defined(__WIN32__) || defined(__WXWINCE__)
ba681060 46 #include <commdlg.h>
2bda0e17
KB
47#endif
48
2bda0e17
KB
49#include <math.h>
50#include <stdlib.h>
51#include <string.h>
52
8ad9ca97 53#include "wx/filename.h"
8f177c8e
VZ
54#include "wx/tokenzr.h"
55
6e8aa701
VZ
56#ifndef OFN_EXPLORER
57 #define OFN_EXPLORER 0x00080000
58#endif
59
f6bcfd97
BP
60// ----------------------------------------------------------------------------
61// constants
62// ----------------------------------------------------------------------------
63
64#ifdef __WIN32__
2b5f62a0 65# define wxMAXPATH 65534
f6bcfd97
BP
66#else
67# define wxMAXPATH 1024
68#endif
69
70# define wxMAXFILE 1024
71
72# define wxMAXEXT 5
73
74// ============================================================================
75// implementation
76// ============================================================================
77
f74172ab 78IMPLEMENT_CLASS(wxFileDialog, wxFileDialogBase)
2bda0e17 79
f6bcfd97 80// ----------------------------------------------------------------------------
b600ed13 81// wxFileDialog
f6bcfd97
BP
82// ----------------------------------------------------------------------------
83
2b5f62a0
VZ
84wxFileDialog::wxFileDialog(wxWindow *parent,
85 const wxString& message,
86 const wxString& defaultDir,
87 const wxString& defaultFileName,
88 const wxString& wildCard,
89 long style,
f74172ab
VZ
90 const wxPoint& pos)
91 :wxFileDialogBase(parent, message, defaultDir, defaultFileName, wildCard, style, pos)
92
2bda0e17 93{
c61f4f6d
VZ
94 if ( ( m_dialogStyle & wxMULTIPLE ) && ( m_dialogStyle & wxSAVE ) )
95 m_dialogStyle &= ~wxMULTIPLE;
2bda0e17
KB
96}
97
c61f4f6d
VZ
98void wxFileDialog::GetPaths(wxArrayString& paths) const
99{
100 paths.Empty();
101
102 wxString dir(m_dir);
103 if ( m_dir.Last() != _T('\\') )
104 dir += _T('\\');
105
106 size_t count = m_fileNames.GetCount();
107 for ( size_t n = 0; n < count; n++ )
108 {
8ad9ca97
JS
109 if (wxFileName(m_fileNames[n]).IsAbsolute())
110 paths.Add(m_fileNames[n]);
111 else
112 paths.Add(dir + m_fileNames[n]);
c61f4f6d
VZ
113 }
114}
115
89654c9a
VZ
116void wxFileDialog::GetFilenames(wxArrayString& files) const
117{
118 files = m_fileNames;
119}
120
2b5f62a0
VZ
121void wxFileDialog::SetPath(const wxString& path)
122{
123 wxString ext;
124 wxSplitPath(path, &m_dir, &m_fileName, &ext);
125 if ( !ext.empty() )
126 m_fileName << _T('.') << ext;
127}
128
c61f4f6d 129int wxFileDialog::ShowModal()
2bda0e17 130{
1f2f0331
VZ
131 HWND hWnd = 0;
132 if (m_parent) hWnd = (HWND) m_parent->GetHWND();
f6bcfd97
BP
133 if (!hWnd && wxTheApp->GetTopWindow())
134 hWnd = (HWND) wxTheApp->GetTopWindow()->GetHWND();
2bda0e17 135
f6bcfd97
BP
136 static wxChar fileNameBuffer [ wxMAXPATH ]; // the file-name
137 wxChar titleBuffer [ wxMAXFILE+1+wxMAXEXT ]; // the file-name, without path
2bda0e17 138
223d09f6
KB
139 *fileNameBuffer = wxT('\0');
140 *titleBuffer = wxT('\0');
2bda0e17 141
2bda0e17 142 long msw_flags = 0;
e15e548b 143 if ( (m_dialogStyle & wxHIDE_READONLY) || (m_dialogStyle & wxSAVE) )
1f2f0331 144 msw_flags |= OFN_HIDEREADONLY;
e15e548b 145 if ( m_dialogStyle & wxFILE_MUST_EXIST )
1f2f0331 146 msw_flags |= OFN_PATHMUSTEXIST | OFN_FILEMUSTEXIST;
6e8aa701 147
c61f4f6d 148 if (m_dialogStyle & wxMULTIPLE )
6e8aa701
VZ
149 {
150 // OFN_EXPLORER must always be specified with OFN_ALLOWMULTISELECT
151 msw_flags |= OFN_EXPLORER | OFN_ALLOWMULTISELECT;
152 }
153
99d1b93d
VZ
154 // if wxCHANGE_DIR flag is not given we shouldn't change the CWD which the
155 // standard dialog does by default
6e8aa701
VZ
156 if ( !(m_dialogStyle & wxCHANGE_DIR) )
157 {
158 msw_flags |= OFN_NOCHANGEDIR;
159 }
012a01fc
JS
160/* chris elliott for some reason this does not work usefully if no extension
161 is given, as it test for junk instead of junk.ext
99d1b93d
VZ
162 if ( m_dialogStyle & wxOVERWRITE_PROMPT )
163 {
164 msw_flags |= OFN_OVERWRITEPROMPT;
165 }
012a01fc 166*/
e15e548b 167 OPENFILENAME of;
f6bcfd97
BP
168 wxZeroMemory(of);
169
170 // the OPENFILENAME struct has been extended in newer version of
171 // comcdlg32.dll, but as we don't use the extended fields anyhow, set
172 // the struct size to the old value - otherwise, the programs compiled
173 // with new headers will not work with the old libraries
174#if defined(_WIN32_WINNT) && (_WIN32_WINNT >= 0x0500)
175 of.lStructSize = sizeof(OPENFILENAME) -
176 (sizeof(void *) + 2*sizeof(DWORD));
177#else // old headers
e15e548b 178 of.lStructSize = sizeof(OPENFILENAME);
f6bcfd97
BP
179#endif
180
e15e548b 181 of.hwndOwner = hWnd;
837e5743 182 of.lpstrTitle = WXSTRINGCAST m_message;
e15e548b 183 of.lpstrFileTitle = titleBuffer;
f6bcfd97 184 of.nMaxFileTitle = wxMAXFILE + 1 + wxMAXEXT; // Windows 3.0 and 3.1
2bda0e17 185
0bc9b25e 186 // Convert forward slashes to backslashes (file selector doesn't like
99d1b93d
VZ
187 // forward slashes) and also squeeze multiple consecutive slashes into one
188 // as it doesn't like two backslashes in a row neither
0627d091
RL
189
190 wxString dir;
191 size_t i, len = m_dir.length();
99d1b93d 192 dir.reserve(len);
0627d091 193 for ( i = 0; i < len; i++ )
99d1b93d
VZ
194 {
195 wxChar ch = m_dir[i];
196 switch ( ch )
197 {
198 case _T('/'):
199 // convert to backslash
200 ch = _T('\\');
201
202 // fall through
0bc9b25e 203
99d1b93d
VZ
204 case _T('\\'):
205 while ( i < len - 1 )
206 {
207 wxChar chNext = m_dir[i + 1];
208 if ( chNext != _T('\\') && chNext != _T('/') )
209 break;
210
04d93c3a
CE
211 // ignore the next one, unless it is at the start of a UNC path
212 if (i > 0)
213 i++;
214 else
215 break;
99d1b93d
VZ
216 }
217 // fall through
218
219 default:
220 // normal char
221 dir += ch;
222 }
223 }
224
225 of.lpstrInitialDir = dir.c_str();
2bda0e17 226
e15e548b 227 of.Flags = msw_flags;
2bda0e17
KB
228
229
2bda0e17
KB
230 //=== Like Alejandro Sierra's wildcard modification >>===================
231 /*
1f2f0331
VZ
232 In wxFileSelector you can put, instead of a single wild_card,
233 pairs of strings separated by '|'.
234 The first string is a description, and the
235 second is the wild card. You can put any number of pairs.
2bda0e17 236
1f2f0331 237 eg. "description1 (*.ex1)|*.ex1|description2 (*.ex2)|*.ex2"
2bda0e17 238
1f2f0331
VZ
239 If you put a single wild card, it works as before the modification.
240 */
2bda0e17
KB
241 //=======================================================================
242
4dba84be 243 wxString theFilter;
837e5743 244 if ( wxStrlen(m_wildCard) == 0 )
223d09f6 245 theFilter = wxString(wxT("*.*"));
4dba84be
JS
246 else
247 theFilter = m_wildCard ;
1f2f0331 248 wxString filterBuffer;
2bda0e17 249
223d09f6 250 if ( !wxStrchr( theFilter, wxT('|') ) ) { // only one filter ==> default text
1f2f0331
VZ
251 filterBuffer.Printf(_("Files (%s)|%s"),
252 theFilter.c_str(), theFilter.c_str());
e15e548b 253 }
1f2f0331
VZ
254 else { // more then one filter
255 filterBuffer = theFilter;
2bda0e17 256
574c0bbf
JS
257 }
258
223d09f6 259 filterBuffer += wxT("|");
574c0bbf 260 // Replace | with \0
0bc9b25e 261 for (i = 0; i < filterBuffer.Len(); i++ ) {
223d09f6
KB
262 if ( filterBuffer.GetChar(i) == wxT('|') ) {
263 filterBuffer[i] = wxT('\0');
e15e548b
VZ
264 }
265 }
2bda0e17 266
837e5743 267 of.lpstrFilter = (LPTSTR)(const wxChar *)filterBuffer;
cc42eb7a 268 of.nFilterIndex = m_filterIndex + 1;
2bda0e17
KB
269
270 //=== Setting defaultFileName >>=========================================
271
f6bcfd97
BP
272 wxStrncpy( fileNameBuffer, (const wxChar *)m_fileName, wxMAXPATH-1 );
273 fileNameBuffer[ wxMAXPATH-1 ] = wxT('\0');
2bda0e17 274
e15e548b 275 of.lpstrFile = fileNameBuffer; // holds returned filename
f6bcfd97 276 of.nMaxFile = wxMAXPATH;
2bda0e17
KB
277
278 //== Execute FileDialog >>=================================================
279
3f6638b8
VZ
280 bool success = (m_dialogStyle & wxSAVE ? GetSaveFileName(&of)
281 : GetOpenFileName(&of)) != 0;
2bda0e17 282
f6bcfd97
BP
283 DWORD errCode = CommDlgExtendedError();
284
285#ifdef __WIN32__
286 if (!success && (errCode == CDERR_STRUCTSIZE))
287 {
288 // The struct size has changed so try a smaller or bigger size
289
290 int oldStructSize = of.lStructSize;
291 of.lStructSize = oldStructSize - (sizeof(void *) + 2*sizeof(DWORD));
292 success = (m_dialogStyle & wxSAVE) ? (GetSaveFileName(&of) != 0)
293 : (GetOpenFileName(&of) != 0);
294 errCode = CommDlgExtendedError();
295
296 if (!success && (errCode == CDERR_STRUCTSIZE))
297 {
298 of.lStructSize = oldStructSize + (sizeof(void *) + 2*sizeof(DWORD));
299 success = (m_dialogStyle & wxSAVE) ? (GetSaveFileName(&of) != 0)
300 : (GetOpenFileName(&of) != 0);
301 }
302 }
c6603ac2 303#endif // __WIN32__
f6bcfd97 304
2bda0e17
KB
305 if ( success )
306 {
c61f4f6d
VZ
307 m_fileNames.Empty();
308
309 if ( ( m_dialogStyle & wxMULTIPLE ) &&
310#if defined(OFN_EXPLORER)
c39e82f0 311 ( fileNameBuffer[of.nFileOffset-1] == wxT('\0') )
c61f4f6d 312#else
c39e82f0 313 ( fileNameBuffer[of.nFileOffset-1] == wxT(' ') )
c61f4f6d 314#endif // OFN_EXPLORER
c39e82f0 315 )
c61f4f6d
VZ
316 {
317#if defined(OFN_EXPLORER)
318 m_dir = fileNameBuffer;
319 i = of.nFileOffset;
320 m_fileName = &fileNameBuffer[i];
321 m_fileNames.Add(m_fileName);
322 i += m_fileName.Len() + 1;
323
324 while (fileNameBuffer[i] != wxT('\0'))
325 {
326 m_fileNames.Add(&fileNameBuffer[i]);
327 i += wxStrlen(&fileNameBuffer[i]) + 1;
328 }
329#else
c6603ac2 330 wxStringTokenizer toke(fileNameBuffer, _T(" \t\r\n"));
c61f4f6d
VZ
331 m_dir = toke.GetNextToken();
332 m_fileName = toke.GetNextToken();
333 m_fileNames.Add(m_fileName);
334
335 while (toke.HasMoreTokens())
336 m_fileNames.Add(toke.GetNextToken());
337#endif // OFN_EXPLORER
338
339 wxString dir(m_dir);
340 if ( m_dir.Last() != _T('\\') )
341 dir += _T('\\');
342
c61f4f6d
VZ
343 m_path = dir + m_fileName;
344 }
345 else
346 {
c61f4f6d 347 //=== Adding the correct extension >>=================================
2bda0e17 348
cc42eb7a 349 m_filterIndex = (int)of.nFilterIndex - 1;
2bda0e17 350
c6603ac2
VS
351 if ( !of.nFileExtension ||
352 (of.nFileExtension && fileNameBuffer[of.nFileExtension] == wxT('\0')) )
353 {
354 // User has typed a filename without an extension:
f74172ab
VZ
355 const wxChar* extension = filterBuffer;
356 int maxFilter = (int)(of.nFilterIndex*2L) - 1;
2bda0e17 357
f74172ab
VZ
358 for( int i = 0; i < maxFilter; i++ ) // get extension
359 extension = extension + wxStrlen( extension ) + 1;
a039ccbf 360
f74172ab
VZ
361 m_fileName = AppendExtension(fileNameBuffer, extension);
362 wxStrncpy(fileNameBuffer, m_fileName.c_str(), wxMin(m_fileName.Len(), wxMAXPATH-1));
363 fileNameBuffer[wxMin(m_fileName.Len(), wxMAXPATH-1)] = wxT('\0');
2bda0e17 364 }
2bda0e17 365
c61f4f6d
VZ
366 m_path = fileNameBuffer;
367 m_fileName = wxFileNameFromPath(fileNameBuffer);
368 m_fileNames.Add(m_fileName);
369 m_dir = wxPathOnly(fileNameBuffer);
370 }
012a01fc
JS
371 //=== Simulating the wxOVERWRITE_PROMPT >>============================
372 //should we also test for file save style ??
373 if ( (m_dialogStyle & wxOVERWRITE_PROMPT) &&
374 ::wxFileExists( fileNameBuffer ) )
375 {
376 wxString messageText;
377 messageText.Printf(_("File '%s' already exists.\nDo you want to replace it?"), fileNameBuffer);
6f661de7 378 if ( wxMessageBox(messageText, _("Save File As"), wxYES_NO | wxICON_EXCLAMATION ) != wxYES )
012a01fc
JS
379 {
380 success = FALSE;
381 }
382 }
7cc98b3e
VZ
383 }
384 else
385 {
386 // common dialog failed - why?
387#ifdef __WXDEBUG__
388 DWORD dwErr = CommDlgExtendedError();
389 if ( dwErr != 0 )
390 {
391 // this msg is only for developers
223d09f6 392 wxLogError(wxT("Common dialog failed with error code %0lx."),
7cc98b3e
VZ
393 dwErr);
394 }
395 //else: it was just cancelled
396#endif
397 }
2bda0e17 398
7cc98b3e 399 return success ? wxID_OK : wxID_CANCEL;
2bda0e17
KB
400
401}
402
1e6feb95 403#endif // wxUSE_FILEDLG
c61f4f6d 404