1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/dialog.cpp
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 // For compilers that support precompilation, includes "wx.h".
11 #include "wx/wxprec.h"
13 #include "wx/dialog.h"
16 #include "wx/cursor.h"
19 #include "wx/evtloop.h"
21 #include "wx/scopedptr.h"
22 #include "wx/testing.h"
25 #include "wx/gtk/private/gtk2-compat.h"
26 #include "wx/gtk/private/dialogcount.h"
28 wxDEFINE_TIED_SCOPED_PTR_TYPE(wxGUIEventLoop
)
31 //-----------------------------------------------------------------------------
33 //-----------------------------------------------------------------------------
39 m_modalShowing
= false;
40 m_themeEnabled
= true;
43 wxDialog::wxDialog( wxWindow
*parent
,
44 wxWindowID id
, const wxString
&title
,
45 const wxPoint
&pos
, const wxSize
&size
,
46 long style
, const wxString
&name
)
50 (void)Create( parent
, id
, title
, pos
, size
, style
, name
);
53 bool wxDialog::Create( wxWindow
*parent
,
54 wxWindowID id
, const wxString
&title
,
55 const wxPoint
&pos
, const wxSize
&size
,
56 long style
, const wxString
&name
)
58 SetExtraStyle(GetExtraStyle() | wxTOPLEVEL_EX_DIALOG
);
60 // all dialogs should have tab traversal enabled
61 style
|= wxTAB_TRAVERSAL
;
63 return wxTopLevelWindow::Create(parent
, id
, title
, pos
, size
, style
, name
);
66 bool wxDialog::Show( bool show
)
68 if (!show
&& IsModal())
70 EndModal( wxID_CANCEL
);
73 if (show
&& CanDoLayoutAdaptation())
76 bool ret
= wxDialogBase::Show(show
);
86 // if the dialog is modal, this will end its event loop
88 EndModal(wxID_CANCEL
);
91 bool wxDialog::IsModal() const
93 return m_modalShowing
;
96 // Workaround for Ubuntu overlay scrollbar, which adds our GtkWindow to a
97 // private window group in a GtkScrollbar realize handler. This breaks the grab
98 // done by gtk_window_set_modal(), and allows menus and toolbars in the parent
99 // frame to remain active. So, we install an emission hook on the "realize"
100 // signal while showing a modal dialog. For any realize on a GtkScrollbar,
101 // we check the top level parent to see if it has an explicitly set window
102 // group that is not the same as its transient parent. If we find this, we
103 // put the top level back in the same window group as its transient parent, and
105 // Ubuntu 12.04 and 12.10 are known to have this problem.
107 // need 2.10 for gtk_window_get_group()
108 #if GTK_CHECK_VERSION(2,10,0)
111 realize_hook(GSignalInvocationHint
*, unsigned, const GValue
* param_values
, void*)
113 void* p
= g_value_peek_pointer(param_values
);
114 if (GTK_IS_SCROLLBAR(p
))
116 GtkWindow
* toplevel
= GTK_WINDOW(gtk_widget_get_toplevel(GTK_WIDGET(p
)));
117 GtkWindow
* transient_parent
= gtk_window_get_transient_for(toplevel
);
118 if (transient_parent
&& gtk_window_has_group(toplevel
))
120 GtkWindowGroup
* group
= gtk_window_get_group(toplevel
);
121 GtkWindowGroup
* group_parent
= gtk_window_get_group(transient_parent
);
122 if (group
!= group_parent
)
124 gtk_window_group_add_window(group_parent
, toplevel
);
125 gtk_grab_add(GTK_WIDGET(toplevel
));
134 int wxDialog::ShowModal()
136 WX_TESTING_SHOW_MODAL_HOOK();
138 wxASSERT_MSG( !IsModal(), "ShowModal() can't be called twice" );
140 // release the mouse if it's currently captured as the window having it
141 // will be disabled when this dialog is shown -- but will still keep the
142 // capture making it impossible to do anything in the modal dialog itself
143 wxWindow
* const win
= wxWindow::GetCapture();
145 win
->GTKReleaseMouseAndNotify();
147 wxWindow
* const parent
= GetParentForModalDialog();
150 gtk_window_set_transient_for( GTK_WINDOW(m_widget
),
151 GTK_WINDOW(parent
->m_widget
) );
154 wxBusyCursorSuspender cs
; // temporarily suppress the busy cursor
156 #if GTK_CHECK_VERSION(2,10,0)
160 // Ubuntu overlay scrollbar uses at least GTK 2.24
161 if (gtk_check_version(2,24,0) == NULL
)
164 sigId
= g_signal_lookup("realize", GTK_TYPE_WIDGET
);
165 hookId
= g_signal_add_emission_hook(sigId
, 0, realize_hook
, NULL
, NULL
);
171 m_modalShowing
= true;
173 wxOpenModalDialogLocker modalLock
;
175 // NOTE: gtk_window_set_modal internally calls gtk_grab_add() !
176 gtk_window_set_modal(GTK_WINDOW(m_widget
), TRUE
);
178 // Run modal dialog event loop.
180 wxGUIEventLoopTiedPtr
modal(&m_modalLoop
, new wxGUIEventLoop());
184 #if GTK_CHECK_VERSION(2,10,0)
186 g_signal_remove_emission_hook(sigId
, hookId
);
189 gtk_window_set_modal(GTK_WINDOW(m_widget
), FALSE
);
191 return GetReturnCode();
194 void wxDialog::EndModal( int retCode
)
196 SetReturnCode( retCode
);
200 wxFAIL_MSG( "either wxDialog:EndModal called twice or ShowModal wasn't called" );
204 m_modalShowing
= false;
206 // Ensure Exit() is only called once. The dialog's event loop may be terminated
207 // externally due to an uncaught exception.
208 if (m_modalLoop
&& m_modalLoop
->IsRunning())