1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/dialog.cpp
4 // Author: Robert Roebling
5 // Copyright: (c) 1998 Robert Roebling
6 // Licence: wxWindows licence
7 /////////////////////////////////////////////////////////////////////////////
9 // For compilers that support precompilation, includes "wx.h".
10 #include "wx/wxprec.h"
12 #include "wx/dialog.h"
15 #include "wx/cursor.h"
18 #include "wx/evtloop.h"
20 #include "wx/scopedptr.h"
21 #include "wx/modalhook.h"
24 #include "wx/gtk/private/gtk2-compat.h"
25 #include "wx/gtk/private/dialogcount.h"
27 wxDEFINE_TIED_SCOPED_PTR_TYPE(wxGUIEventLoop
)
30 //-----------------------------------------------------------------------------
32 //-----------------------------------------------------------------------------
37 m_modalShowing
= false;
40 wxDialog::wxDialog( wxWindow
*parent
,
41 wxWindowID id
, const wxString
&title
,
42 const wxPoint
&pos
, const wxSize
&size
,
43 long style
, const wxString
&name
)
47 (void)Create( parent
, id
, title
, pos
, size
, style
, name
);
50 bool wxDialog::Create( wxWindow
*parent
,
51 wxWindowID id
, const wxString
&title
,
52 const wxPoint
&pos
, const wxSize
&size
,
53 long style
, const wxString
&name
)
55 SetExtraStyle(GetExtraStyle() | wxTOPLEVEL_EX_DIALOG
);
57 // all dialogs should have tab traversal enabled
58 style
|= wxTAB_TRAVERSAL
;
60 return wxTopLevelWindow::Create(parent
, id
, title
, pos
, size
, style
, name
);
63 bool wxDialog::Show( bool show
)
65 if (!show
&& IsModal())
67 EndModal( wxID_CANCEL
);
70 if (show
&& CanDoLayoutAdaptation())
73 bool ret
= wxDialogBase::Show(show
);
83 // if the dialog is modal, this will end its event loop
85 EndModal(wxID_CANCEL
);
88 bool wxDialog::IsModal() const
90 return m_modalShowing
;
93 // Workaround for Ubuntu overlay scrollbar, which adds our GtkWindow to a
94 // private window group in a GtkScrollbar realize handler. This breaks the grab
95 // done by gtk_window_set_modal(), and allows menus and toolbars in the parent
96 // frame to remain active. So, we install an emission hook on the "realize"
97 // signal while showing a modal dialog. For any realize on a GtkScrollbar,
98 // we check the top level parent to see if it has an explicitly set window
99 // group that is not the same as its transient parent. If we find this, we
100 // put the top level back in the same window group as its transient parent, and
102 // Ubuntu 12.04 and 12.10 are known to have this problem.
104 // need 2.10 for gtk_window_get_group()
105 #if GTK_CHECK_VERSION(2,10,0)
108 realize_hook(GSignalInvocationHint
*, unsigned, const GValue
* param_values
, void*)
110 void* p
= g_value_peek_pointer(param_values
);
111 if (GTK_IS_SCROLLBAR(p
))
113 GtkWindow
* toplevel
= GTK_WINDOW(gtk_widget_get_toplevel(GTK_WIDGET(p
)));
114 GtkWindow
* transient_parent
= gtk_window_get_transient_for(toplevel
);
115 if (transient_parent
&& gtk_window_has_group(toplevel
))
117 GtkWindowGroup
* group
= gtk_window_get_group(toplevel
);
118 GtkWindowGroup
* group_parent
= gtk_window_get_group(transient_parent
);
119 if (group
!= group_parent
)
121 gtk_window_group_add_window(group_parent
, toplevel
);
122 gtk_grab_add(GTK_WIDGET(toplevel
));
131 int wxDialog::ShowModal()
133 WX_HOOK_MODAL_DIALOG();
135 wxASSERT_MSG( !IsModal(), "ShowModal() can't be called twice" );
137 // release the mouse if it's currently captured as the window having it
138 // will be disabled when this dialog is shown -- but will still keep the
139 // capture making it impossible to do anything in the modal dialog itself
140 wxWindow
* const win
= wxWindow::GetCapture();
142 win
->GTKReleaseMouseAndNotify();
144 wxWindow
* const parent
= GetParentForModalDialog();
147 gtk_window_set_transient_for( GTK_WINDOW(m_widget
),
148 GTK_WINDOW(parent
->m_widget
) );
151 wxBusyCursorSuspender cs
; // temporarily suppress the busy cursor
153 #if GTK_CHECK_VERSION(2,10,0)
157 // Ubuntu overlay scrollbar uses at least GTK 2.24
158 if (gtk_check_version(2,24,0) == NULL
)
161 sigId
= g_signal_lookup("realize", GTK_TYPE_WIDGET
);
162 hookId
= g_signal_add_emission_hook(sigId
, 0, realize_hook
, NULL
, NULL
);
168 m_modalShowing
= true;
170 wxOpenModalDialogLocker modalLock
;
172 // NOTE: gtk_window_set_modal internally calls gtk_grab_add() !
173 gtk_window_set_modal(GTK_WINDOW(m_widget
), TRUE
);
175 // Run modal dialog event loop.
177 wxGUIEventLoopTiedPtr
modal(&m_modalLoop
, new wxGUIEventLoop());
181 #if GTK_CHECK_VERSION(2,10,0)
183 g_signal_remove_emission_hook(sigId
, hookId
);
186 gtk_window_set_modal(GTK_WINDOW(m_widget
), FALSE
);
188 return GetReturnCode();
191 void wxDialog::EndModal( int retCode
)
193 SetReturnCode( retCode
);
197 wxFAIL_MSG( "either wxDialog:EndModal called twice or ShowModal wasn't called" );
201 m_modalShowing
= false;
203 // Ensure Exit() is only called once. The dialog's event loop may be terminated
204 // externally due to an uncaught exception.
205 if (m_modalLoop
&& m_modalLoop
->IsRunning())