1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/cocoa/filedlg.mm
3 // Purpose: wxFileDialog for wxCocoa
7 // RCS-ID: $Id: filedlg.mm 40007 2006-07-05 13:10:46Z SC $
8 // Copyright: (c) Ryan Norton
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
25 #include "wx/filedlg.h"
28 #include "wx/msgdlg.h"
32 #include "wx/filename.h"
33 #include "wx/tokenzr.h"
35 #include "wx/osx/private.h"
36 #include "wx/sysopt.h"
38 // ============================================================================
40 // ============================================================================
43 // - parameter support for descending into packages as directories (setTreatsFilePackagesAsDirectories)
44 // - as setAllowedFileTypes is only functional for NSOpenPanel on 10.6+, on earlier systems, the file
45 // type choice will not be shown, but all possible file items will be shown, if a popup must be working
46 // then the delegate method - (BOOL)panel:(id)sender shouldShowFilename:(NSString *)filename will have to
49 @interface wxOpenPanelDelegate : NSObject wxOSX_10_6_AND_LATER(<NSOpenSavePanelDelegate>)
51 wxFileDialog* _dialog;
54 - (wxFileDialog*) fileDialog;
55 - (void) setFileDialog:(wxFileDialog*) dialog;
57 - (BOOL)panel:(id)sender shouldShowFilename:(NSString *)filename;
61 @implementation wxOpenPanelDelegate
70 - (wxFileDialog*) fileDialog
75 - (void) setFileDialog:(wxFileDialog*) dialog
80 - (BOOL)panel:(id)sender shouldShowFilename:(NSString *)filename
82 BOOL showObject = YES;
84 NSString* resolvedLink = [[NSFileManager defaultManager] pathContentOfSymbolicLinkAtPath:filename];
85 if ( resolvedLink != nil )
86 filename = resolvedLink;
88 NSDictionary* fileAttribs = [[NSFileManager defaultManager]
89 fileAttributesAtPath:filename traverseLink:YES];
93 if ([NSFileTypeDirectory isEqualTo:[fileAttribs objectForKey:NSFileType]])
95 if ([[NSWorkspace sharedWorkspace] isFilePackageAtPath:filename] == NO)
96 showObject = YES; // it's a folder, OK to show
99 // it's a packaged directory, apply check
100 wxCFStringRef filecf([filename retain]);
101 showObject = _dialog->CheckFile(filecf.AsString());
106 // the code above only solves links, not aliases, do this here:
108 NSString* resolvedAlias = nil;
110 CFURLRef url = CFURLCreateWithFileSystemPath (kCFAllocatorDefault,
111 (CFStringRef)filename,
112 kCFURLPOSIXPathStyle,
117 if (CFURLGetFSRef(url, &fsRef))
119 Boolean targetIsFolder, wasAliased;
120 OSErr err = FSResolveAliasFile (&fsRef, true, &targetIsFolder, &wasAliased);
122 if ((err == noErr) && wasAliased)
124 CFURLRef resolvedUrl = CFURLCreateFromFSRef(kCFAllocatorDefault, &fsRef);
125 if (resolvedUrl != NULL)
127 resolvedAlias = (NSString*) CFURLCopyFileSystemPath(resolvedUrl,
128 kCFURLPOSIXPathStyle);
129 CFRelease(resolvedUrl);
136 if (resolvedAlias != nil)
139 [resolvedAlias autorelease];
140 showObject = [self panel:sender shouldShowFilename:resolvedAlias];
144 wxCFStringRef filecf([filename retain]);
145 showObject = _dialog->CheckFile(filecf.AsString());
155 IMPLEMENT_CLASS(wxFileDialog, wxFileDialogBase)
157 wxFileDialog::wxFileDialog(
158 wxWindow *parent, const wxString& message,
159 const wxString& defaultDir, const wxString& defaultFileName, const wxString& wildCard,
160 long style, const wxPoint& pos, const wxSize& sz, const wxString& name)
161 : wxFileDialogBase(parent, message, defaultDir, defaultFileName, wildCard, style, pos, sz, name)
165 bool wxFileDialog::SupportsExtraControl() const
170 NSArray* GetTypesFromExtension( const wxString extensiongroup, wxArrayString& extensions )
172 NSMutableArray* types = nil;
175 wxStringTokenizer tokenizer( extensiongroup, wxT(";") ) ;
176 while ( tokenizer.HasMoreTokens() )
178 wxString extension = tokenizer.GetNextToken() ;
179 // Remove leading '*'
180 if ( extension.length() && (extension.GetChar(0) == '*') )
181 extension = extension.Mid( 1 );
183 // Remove leading '.'
184 if ( extension.length() && (extension.GetChar(0) == '.') )
185 extension = extension.Mid( 1 );
187 // Remove leading '*', this is for handling *.*
188 if ( extension.length() && (extension.GetChar(0) == '*') )
189 extension = extension.Mid( 1 );
191 if ( extension.IsEmpty() )
200 types = [[NSMutableArray alloc] init];
202 extensions.Add(extension.Lower());
203 wxCFStringRef cfext(extension);
204 [types addObject: (NSString*)cfext.AsNSString() ];
206 // add support for classic fileType / creator here
207 wxUint32 fileType, creator;
208 // extension -> mactypes
215 NSArray* GetTypesFromFilter( const wxString& filter, wxArrayString& names, wxArrayString& extensiongroups )
217 NSMutableArray* types = nil;
218 bool allowAll = false;
221 extensiongroups.Clear();
223 if ( !filter.empty() )
225 wxStringTokenizer tokenizer( filter, wxT("|") );
226 int numtokens = (int)tokenizer.CountTokens();
229 // we allow for compatibility reason to have a single filter expression (like *.*) without
230 // an explanatory text, in that case the first part is name and extension at the same time
231 wxString extension = tokenizer.GetNextToken();
232 names.Add( extension );
233 extensiongroups.Add( extension );
237 int numextensions = numtokens / 2;
238 for(int i = 0; i < numextensions; i++)
240 wxString name = tokenizer.GetNextToken();
241 wxString extension = tokenizer.GetNextToken();
243 extensiongroups.Add( extension );
247 const size_t extCount = extensiongroups.GetCount();
248 wxArrayString extensions;
249 for ( size_t i = 0 ; i < extCount; i++ )
251 NSArray* exttypes = GetTypesFromExtension(extensiongroups[i], extensions);
252 if ( exttypes != nil )
254 if ( allowAll == false )
257 types = [[NSMutableArray alloc] init];
259 [types addObjectsFromArray:exttypes];
274 void wxFileDialog::ShowWindowModal()
276 wxCFStringRef cf( m_message );
277 wxCFStringRef dir( m_dir );
278 wxCFStringRef file( m_fileName );
280 wxNonOwnedWindow* parentWindow = NULL;
282 m_modality = wxDIALOG_MODALITY_WINDOW_MODAL;
285 parentWindow = dynamic_cast<wxNonOwnedWindow*>(wxGetTopLevelParent(GetParent()));
287 wxASSERT_MSG(parentWindow, "Window modal display requires parent.");
289 NSArray* types = GetTypesFromFilter( m_wildCard, m_filterNames, m_filterExtensions ) ;
290 if ( HasFlag(wxFD_SAVE) )
292 NSSavePanel* sPanel = [NSSavePanel savePanel];
294 SetupExtraControls(sPanel);
296 // makes things more convenient:
297 [sPanel setCanCreateDirectories:YES];
298 [sPanel setMessage:cf.AsNSString()];
299 // if we should be able to descend into pacakges we must somehow
300 // be able to pass this in
301 [sPanel setTreatsFilePackagesAsDirectories:NO];
302 [sPanel setCanSelectHiddenExtension:YES];
303 [sPanel setAllowedFileTypes:types];
304 [sPanel setAllowsOtherFileTypes:NO];
306 NSWindow* nativeParent = parentWindow->GetWXWindow();
307 ModalDialogDelegate* sheetDelegate = [[ModalDialogDelegate alloc] init];
308 [sheetDelegate setImplementation: this];
309 [sPanel beginSheetForDirectory:dir.AsNSString() file:file.AsNSString()
310 modalForWindow: nativeParent modalDelegate: sheetDelegate
311 didEndSelector: @selector(sheetDidEnd:returnCode:contextInfo:)
316 NSOpenPanel* oPanel = [NSOpenPanel openPanel];
318 SetupExtraControls(oPanel);
320 [oPanel setTreatsFilePackagesAsDirectories:NO];
321 [oPanel setCanChooseDirectories:NO];
322 [oPanel setResolvesAliases:YES];
323 [oPanel setCanChooseFiles:YES];
324 [oPanel setMessage:cf.AsNSString()];
325 [oPanel setAllowsMultipleSelection: (HasFlag(wxFD_MULTIPLE) ? YES : NO )];
327 NSWindow* nativeParent = parentWindow->GetWXWindow();
328 ModalDialogDelegate* sheetDelegate = [[ModalDialogDelegate alloc] init];
329 [sheetDelegate setImplementation: this];
330 [oPanel beginSheetForDirectory:dir.AsNSString() file:file.AsNSString()
331 types: types modalForWindow: nativeParent
332 modalDelegate: sheetDelegate
333 didEndSelector: @selector(sheetDidEnd:returnCode:contextInfo:)
338 // Create a panel with the file type drop down list
339 // If extra controls need to be added (see wxFileDialog::SetExtraControlCreator), add
340 // them to the panel as well
341 // Returns the newly created wxPanel
343 wxWindow* wxFileDialog::CreateFilterPanel(wxWindow *extracontrol)
345 wxPanel *extrapanel = new wxPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize);
346 wxBoxSizer *verticalSizer = new wxBoxSizer(wxVERTICAL);
347 extrapanel->SetSizer(verticalSizer);
349 // the file type control
351 wxBoxSizer *horizontalSizer = new wxBoxSizer(wxHORIZONTAL);
352 verticalSizer->Add(horizontalSizer, 0, wxEXPAND, 0);
353 wxStaticText *stattext = new wxStaticText( extrapanel, wxID_ANY, _("File type:") );
354 horizontalSizer->Add(stattext, 0, wxALIGN_CENTER_VERTICAL|wxALL, 5);
355 m_filterChoice = new wxChoice(extrapanel, wxID_ANY);
356 horizontalSizer->Add(m_filterChoice, 1, wxALIGN_CENTER_VERTICAL|wxALL, 5);
357 m_filterChoice->Append(m_filterNames);
358 if( m_filterNames.GetCount() > 0)
360 if ( m_firstFileTypeFilter >= 0 )
361 m_filterChoice->SetSelection(m_firstFileTypeFilter);
363 m_filterChoice->Connect(wxEVT_COMMAND_CHOICE_SELECTED, wxCommandEventHandler(wxFileDialog::OnFilterSelected), NULL, this);
368 wxBoxSizer *horizontalSizer = new wxBoxSizer(wxHORIZONTAL);
369 verticalSizer->Add(horizontalSizer, 0, wxEXPAND, 0);
371 extracontrol->Reparent(extrapanel);
372 horizontalSizer->Add(extracontrol);
375 verticalSizer->Layout();
376 verticalSizer->SetSizeHints(extrapanel);
380 // An item has been selected in the file filter wxChoice:
381 void wxFileDialog::OnFilterSelected( wxCommandEvent &WXUNUSED(event) )
383 int index = m_filterChoice->GetSelection();
385 NSArray* types = GetTypesFromExtension(m_filterExtensions[index],m_currentExtensions);
386 NSSavePanel* panel = (NSSavePanel*) GetWXWindow();
388 [panel validateVisibleColumns];
390 [panel setAllowedFileTypes:types];
393 bool wxFileDialog::CheckFile( const wxString& filename )
395 if ( m_currentExtensions.GetCount() == 0 )
398 wxString ext = filename.AfterLast('.').Lower();
400 for ( size_t i = 0; i < m_currentExtensions.GetCount(); ++i )
402 if ( ext == m_currentExtensions[i] )
408 void wxFileDialog::SetupExtraControls(WXWindow nativeWindow)
410 NSSavePanel* panel = (NSSavePanel*) nativeWindow;
412 wxNonOwnedWindow::Create( GetParent(), nativeWindow );
413 wxWindow* extracontrol = NULL;
414 if ( HasExtraControlCreator() )
416 CreateExtraControl();
417 extracontrol = GetExtraControl();
420 NSView* accView = nil;
423 if ( m_useFileTypeFilter )
425 m_filterPanel = CreateFilterPanel(extracontrol);
426 accView = m_filterPanel->GetHandle();
427 if( HasFlag(wxFD_OPEN) )
429 if ( 1 /* UMAGetSystemVersion() < 0x1060 */ )
431 wxOpenPanelDelegate* del = [[wxOpenPanelDelegate alloc]init];
432 [del setFileDialog:this];
433 [panel setDelegate:del];
440 m_filterPanel = NULL;
441 m_filterChoice = NULL;
442 if ( extracontrol != nil )
443 accView = extracontrol->GetHandle();
446 if ( accView != nil )
448 [accView removeFromSuperview];
449 [panel setAccessoryView:accView];
453 [panel setAccessoryView:nil];
457 int wxFileDialog::ShowModal()
459 wxCFStringRef cf( m_message );
461 wxCFStringRef dir( m_dir );
462 wxCFStringRef file( m_fileName );
464 m_path = wxEmptyString;
467 // since we don't support retrieving the matching filter
470 wxNonOwnedWindow* parentWindow = NULL;
475 parentWindow = dynamic_cast<wxNonOwnedWindow*>(wxGetTopLevelParent(GetParent()));
479 NSArray* types = GetTypesFromFilter( m_wildCard, m_filterNames, m_filterExtensions ) ;
481 m_useFileTypeFilter = m_filterExtensions.GetCount() > 1;
483 if( HasFlag(wxFD_OPEN) )
485 if ( !(wxSystemOptions::HasOption( wxOSX_FILEDIALOG_ALWAYS_SHOW_TYPES ) && (wxSystemOptions::GetOptionInt( wxOSX_FILEDIALOG_ALWAYS_SHOW_TYPES ) == 1)) )
486 m_useFileTypeFilter = false;
489 m_firstFileTypeFilter = -1;
491 if ( m_useFileTypeFilter )
494 bool useDefault = true;
495 for ( size_t i = 0; i < m_filterExtensions.GetCount(); ++i )
497 types = GetTypesFromExtension(m_filterExtensions[i], m_currentExtensions);
498 if ( m_currentExtensions.GetCount() == 0 )
501 m_firstFileTypeFilter = i;
505 for ( size_t j = 0; j < m_currentExtensions.GetCount(); ++j )
507 if ( m_fileName.EndsWith(m_currentExtensions[j]) )
509 m_firstFileTypeFilter = i;
519 types = GetTypesFromExtension(m_filterExtensions[0], m_currentExtensions);
520 m_firstFileTypeFilter = 0;
524 if ( HasFlag(wxFD_SAVE) )
526 NSSavePanel* sPanel = [NSSavePanel savePanel];
528 SetupExtraControls(sPanel);
530 // makes things more convenient:
531 [sPanel setCanCreateDirectories:YES];
532 [sPanel setMessage:cf.AsNSString()];
533 // if we should be able to descend into pacakges we must somehow
534 // be able to pass this in
535 [sPanel setTreatsFilePackagesAsDirectories:NO];
536 [sPanel setCanSelectHiddenExtension:YES];
537 [sPanel setAllowedFileTypes:types];
538 [sPanel setAllowsOtherFileTypes:NO];
540 if ( HasFlag(wxFD_OVERWRITE_PROMPT) )
544 returnCode = [sPanel runModalForDirectory:dir.AsNSString() file:file.AsNSString() ];
545 ModalFinishedCallback(sPanel, returnCode);
549 NSOpenPanel* oPanel = [NSOpenPanel openPanel];
551 SetupExtraControls(oPanel);
553 [oPanel setTreatsFilePackagesAsDirectories:NO];
554 [oPanel setCanChooseDirectories:NO];
555 [oPanel setResolvesAliases:YES];
556 [oPanel setCanChooseFiles:YES];
557 [oPanel setMessage:cf.AsNSString()];
558 [oPanel setAllowsMultipleSelection: (HasFlag(wxFD_MULTIPLE) ? YES : NO )];
560 if ( UMAGetSystemVersion() < 0x1060 )
562 returnCode = [oPanel runModalForDirectory:dir.AsNSString()
563 file:file.AsNSString() types:(m_delegate == nil ? types : nil)];
567 [oPanel setAllowedFileTypes: (m_delegate == nil ? types : nil)];
568 [oPanel setDirectoryURL:[NSURL fileURLWithPath:dir.AsNSString()
570 returnCode = [oPanel runModal];
573 ModalFinishedCallback(oPanel, returnCode);
576 return GetReturnCode();
579 void wxFileDialog::ModalFinishedCallback(void* panel, int returnCode)
581 int result = wxID_CANCEL;
582 if (HasFlag(wxFD_SAVE))
584 if (returnCode == NSOKButton )
586 NSSavePanel* sPanel = (NSSavePanel*)panel;
589 m_path = wxCFStringRef::AsString([sPanel filename]);
590 m_fileName = wxFileNameFromPath(m_path);
591 m_dir = wxPathOnly( m_path );
596 NSOpenPanel* oPanel = (NSOpenPanel*)panel;
597 if (returnCode == NSOKButton )
601 NSArray* filenames = [oPanel filenames];
602 for ( size_t i = 0 ; i < [filenames count] ; ++ i )
604 wxString fnstr = wxCFStringRef::AsString([filenames objectAtIndex:i]);
605 m_paths.Add( fnstr );
606 m_fileNames.Add( wxFileNameFromPath(fnstr) );
610 m_fileName = wxFileNameFromPath(fnstr);
611 m_dir = wxPathOnly( fnstr );
617 [oPanel setDelegate:nil];
618 [m_delegate release];
622 SetReturnCode(result);
624 if (GetModality() == wxDIALOG_MODALITY_WINDOW_MODAL)
625 SendWindowModalDialogEvent ( wxEVT_WINDOW_MODAL_DIALOG_CLOSED );
628 [(NSSavePanel*) panel setAccessoryView:nil];
631 #endif // wxUSE_FILEDLG