1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/filepicker.cpp
3 // Purpose: implementation of wxFileButton and wxDirButton
4 // Author: Francesco Montorsi
7 // Copyright: (c) Francesco Montorsi
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
12 // ----------------------------------------------------------------------------
14 // ----------------------------------------------------------------------------
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
19 #if wxUSE_FILEPICKERCTRL
21 #include "wx/filepicker.h"
22 #include "wx/tooltip.h"
25 #include "wx/gtk/private.h"
27 // ============================================================================
29 // ============================================================================
31 //-----------------------------------------------------------------------------
33 //-----------------------------------------------------------------------------
35 IMPLEMENT_DYNAMIC_CLASS(wxFileButton
, wxButton
)
37 bool wxFileButton::Create( wxWindow
*parent
, wxWindowID id
,
38 const wxString
&label
, const wxString
&path
,
39 const wxString
&message
, const wxString
&wildcard
,
40 const wxPoint
&pos
, const wxSize
&size
,
41 long style
, const wxValidator
& validator
,
42 const wxString
&name
)
44 // we can't use the native button for wxFLP_SAVE pickers as it can only
45 // open existing files and there is no way to create a new file using it
46 if (!(style
& wxFLP_SAVE
) && !(style
& wxFLP_USE_TEXTCTRL
))
48 // VERY IMPORTANT: this code is identical to relative code in wxDirButton;
49 // if you find a problem here, fix it also in wxDirButton !
51 if (!PreCreation( parent
, pos
, size
) ||
52 !wxControl::CreateBase(parent
, id
, pos
, size
, style
& wxWINDOW_STYLE_MASK
,
55 wxFAIL_MSG( wxT("wxFileButton creation failed") );
59 // create the dialog associated with this button
60 // NB: unlike generic implementation, native GTK implementation needs to create
61 // the filedialog here as it needs to use gtk_file_chooser_button_new_with_dialog()
62 SetWindowStyle(style
);
65 m_wildcard
= wildcard
;
66 if ((m_dialog
= CreateDialog()) == NULL
)
69 // little trick used to avoid problems when there are other GTK windows 'grabbed':
70 // GtkFileChooserDialog won't be responsive to user events if there is another
71 // window which called gtk_grab_add (and this happens if e.g. a wxDialog is running
72 // in modal mode in the application - see wxDialogGTK::ShowModal).
73 // An idea could be to put the grab on the m_dialog->m_widget when the GtkFileChooserButton
74 // is clicked and then remove it as soon as the user closes the dialog itself.
75 // Unfortunately there's no way to hook in the 'clicked' event of the GtkFileChooserButton,
76 // thus we add grab on m_dialog->m_widget when it's shown and remove it when it's
77 // hidden simply using its "show" and "hide" events - clean & simple :)
78 g_signal_connect(m_dialog
->m_widget
, "show", G_CALLBACK(gtk_grab_add
), NULL
);
79 g_signal_connect(m_dialog
->m_widget
, "hide", G_CALLBACK(gtk_grab_remove
), NULL
);
81 // use as label the currently selected file
82 m_widget
= gtk_file_chooser_button_new_with_dialog( m_dialog
->m_widget
);
83 g_object_ref(m_widget
);
85 // we need to know when the dialog has been dismissed clicking OK...
86 // NOTE: the "clicked" signal is not available for a GtkFileChooserButton
87 // thus we are forced to use wxFileDialog's event
88 m_dialog
->Connect(wxEVT_BUTTON
,
89 wxCommandEventHandler(wxFileButton::OnDialogOK
),
92 m_parent
->DoAddChild( this );
98 return wxGenericFileButton::Create(parent
, id
, label
, path
, message
, wildcard
,
99 pos
, size
, style
, validator
, name
);
103 wxFileButton::~wxFileButton()
107 // We need to delete the C++ dialog object here but we shouldn't delete
108 // its widget which is used by our GtkFileChooserButton and will be
109 // deleted by it when it is itself destroyed in our base class dtor. So
110 // take the widget ownership away from the dialog to avoid GTK+ errors
111 // that would happen if GtkFileChooserButton tried to access the
112 // already destroyed dialog widget.
113 g_object_unref(m_dialog
->m_widget
);
114 m_dialog
->m_widget
= NULL
;
119 void wxFileButton::OnDialogOK(wxCommandEvent
& ev
)
121 // the wxFileDialog associated with the GtkFileChooserButton has been closed
122 // using the OK button, thus the selected file has changed...
123 if (ev
.GetId() == wxID_OK
)
125 // ...update our path
126 UpdatePathFromDialog(m_dialog
);
128 // ...and fire an event
129 wxFileDirPickerEvent
event(wxEVT_FILEPICKER_CHANGED
, this, GetId(), m_path
);
130 HandleWindowEvent(event
);
134 void wxFileButton::SetPath(const wxString
&str
)
139 UpdateDialogPath(m_dialog
);
142 void wxFileButton::SetInitialDirectory(const wxString
& dir
)
146 // Only change the directory if the default file name doesn't have any
147 // directory in it, otherwise it takes precedence.
148 if ( m_path
.find_first_of(wxFileName::GetPathSeparators()) ==
151 static_cast<wxFileDialog
*>(m_dialog
)->SetDirectory(dir
);
155 wxGenericFileButton::SetInitialDirectory(dir
);
158 #endif // wxUSE_FILEPICKERCTRL
160 #if wxUSE_DIRPICKERCTRL
163 #include <unistd.h> // chdir
166 //-----------------------------------------------------------------------------
167 // "current-folder-changed"
168 //-----------------------------------------------------------------------------
171 static void gtk_dirbutton_currentfolderchanged_callback(GtkFileChooserButton
*widget
,
174 // update the m_path member of the wxDirButtonGTK
175 // unless the path was changed by wxDirButton::SetPath()
176 if (p
->m_bIgnoreNextChange
)
178 p
->m_bIgnoreNextChange
=false;
183 // NB: it's important to use gtk_file_chooser_get_filename instead of
184 // gtk_file_chooser_get_current_folder (see GTK docs) !
185 wxGtkString
filename(gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(widget
)));
186 p
->GTKUpdatePath(filename
);
188 // since GtkFileChooserButton when used to pick directories also uses a combobox,
189 // maybe that the current folder has been changed but not through the GtkFileChooserDialog
190 // and thus the 'gtk_filedialog_ok_callback' could have not been called...
191 // thus we need to make sure the current working directory is updated if wxDIRP_CHANGE_DIR
193 if (p
->HasFlag(wxDIRP_CHANGE_DIR
))
196 // ...and fire an event
197 wxFileDirPickerEvent
event(wxEVT_DIRPICKER_CHANGED
, p
, p
->GetId(), p
->GetPath());
198 p
->HandleWindowEvent(event
);
203 //-----------------------------------------------------------------------------
205 //-----------------------------------------------------------------------------
207 IMPLEMENT_DYNAMIC_CLASS(wxDirButton
, wxButton
)
209 bool wxDirButton::Create( wxWindow
*parent
, wxWindowID id
,
210 const wxString
&label
, const wxString
&path
,
211 const wxString
&message
, const wxString
&wildcard
,
212 const wxPoint
&pos
, const wxSize
&size
,
213 long style
, const wxValidator
& validator
,
214 const wxString
&name
)
216 if (!(style
& wxDIRP_USE_TEXTCTRL
))
218 // VERY IMPORTANT: this code is identic to relative code in wxFileButton;
219 // if you find a problem here, fix it also in wxFileButton !
221 if (!PreCreation( parent
, pos
, size
) ||
222 !wxControl::CreateBase(parent
, id
, pos
, size
, style
& wxWINDOW_STYLE_MASK
,
225 wxFAIL_MSG( wxT("wxDirButtonGTK creation failed") );
229 // create the dialog associated with this button
230 SetWindowStyle(style
);
232 m_wildcard
= wildcard
;
233 if ((m_dialog
= CreateDialog()) == NULL
)
237 // little trick used to avoid problems when there are other GTK windows 'grabbed':
238 // GtkFileChooserDialog won't be responsive to user events if there is another
239 // window which called gtk_grab_add (and this happens if e.g. a wxDialog is running
240 // in modal mode in the application - see wxDialogGTK::ShowModal).
241 // An idea could be to put the grab on the m_dialog->m_widget when the GtkFileChooserButton
242 // is clicked and then remove it as soon as the user closes the dialog itself.
243 // Unfortunately there's no way to hook in the 'clicked' event of the GtkFileChooserButton,
244 // thus we add grab on m_dialog->m_widget when it's shown and remove it when it's
245 // hidden simply using its "show" and "hide" events - clean & simple :)
246 g_signal_connect(m_dialog
->m_widget
, "show", G_CALLBACK(gtk_grab_add
), NULL
);
247 g_signal_connect(m_dialog
->m_widget
, "hide", G_CALLBACK(gtk_grab_remove
), NULL
);
250 // NOTE: we deliberately ignore the given label as GtkFileChooserButton
251 // use as label the currently selected file
252 m_widget
= gtk_file_chooser_button_new_with_dialog( m_dialog
->m_widget
);
253 g_object_ref(m_widget
);
255 // GtkFileChooserButton signals
256 g_signal_connect(m_widget
, "current-folder-changed",
257 G_CALLBACK(gtk_dirbutton_currentfolderchanged_callback
), this);
259 m_parent
->DoAddChild( this );
262 SetInitialSize(size
);
265 return wxGenericDirButton::Create(parent
, id
, label
, path
, message
, wildcard
,
266 pos
, size
, style
, validator
, name
);
270 wxDirButton::~wxDirButton()
275 void wxDirButton::GTKUpdatePath(const char *gtkpath
)
277 m_path
= wxString::FromUTF8(gtkpath
);
279 void wxDirButton::SetPath(const wxString
& str
)
283 // don't do anything and especially don't set m_bIgnoreNextChange
289 // wxDirButton uses the "current-folder-changed" signal which is triggered also
290 // when we set the path on the dialog associated with this button; thus we need
291 // to set the following flag to avoid sending a wxFileDirPickerEvent from this
292 // function (which would be inconsistent with wxFileButton's behaviour and in
293 // general with all wxWidgets control-manipulation functions which do not send events).
294 m_bIgnoreNextChange
= true;
297 UpdateDialogPath(m_dialog
);
300 void wxDirButton::SetInitialDirectory(const wxString
& dir
)
305 static_cast<wxDirDialog
*>(m_dialog
)->SetPath(dir
);
308 wxGenericDirButton::SetInitialDirectory(dir
);
311 #endif // wxUSE_DIRPICKERCTRL