Commit | Line | Data |
---|---|---|
ed8efd46 VZ |
1 | /////////////////////////////////////////////////////////////////////////////// |
2 | // Name: src/gtk/infobar.cpp | |
3 | // Purpose: wxInfoBar implementation for GTK | |
4 | // Author: Vadim Zeitlin | |
5 | // Created: 2009-09-27 | |
ed8efd46 VZ |
6 | // Copyright: (c) 2009 Vadim Zeitlin <vadim@wxwidgets.org> |
7 | // Licence: wxWindows licence | |
8 | /////////////////////////////////////////////////////////////////////////////// | |
9 | ||
10 | // ============================================================================ | |
11 | // declarations | |
12 | // ============================================================================ | |
13 | ||
14 | // ---------------------------------------------------------------------------- | |
15 | // headers | |
16 | // ---------------------------------------------------------------------------- | |
17 | ||
18 | // for compilers that support precompilation, includes "wx.h". | |
19 | #include "wx/wxprec.h" | |
20 | ||
21 | #ifdef __BORLANDC__ | |
22 | #pragma hdrstop | |
23 | #endif | |
24 | ||
25 | #include "wx/infobar.h" | |
26 | ||
27 | #if wxUSE_INFOBAR && defined(wxHAS_NATIVE_INFOBAR) | |
28 | ||
29 | #ifndef WX_PRECOMP | |
30 | #endif // WX_PRECOMP | |
31 | ||
e6b2aae1 | 32 | #include "wx/vector.h" |
ff7485d8 | 33 | #include "wx/stockitem.h" |
e6b2aae1 | 34 | |
ed8efd46 VZ |
35 | #include "wx/gtk/private.h" |
36 | #include "wx/gtk/private/messagetype.h" | |
37 | ||
e6b2aae1 VZ |
38 | // ---------------------------------------------------------------------------- |
39 | // local classes | |
40 | // ---------------------------------------------------------------------------- | |
41 | ||
42 | class wxInfoBarGTKImpl | |
43 | { | |
44 | public: | |
45 | wxInfoBarGTKImpl() | |
46 | { | |
47 | m_label = NULL; | |
6a3f8b4f | 48 | m_close = NULL; |
e6b2aae1 VZ |
49 | } |
50 | ||
6a3f8b4f | 51 | // label for the text shown in the bar |
e6b2aae1 VZ |
52 | GtkWidget *m_label; |
53 | ||
6a3f8b4f VZ |
54 | // the default close button, NULL if not needed (m_buttons is not empty) or |
55 | // not created yet | |
56 | GtkWidget *m_close; | |
57 | ||
58 | // information about the buttons added using AddButton() | |
e6b2aae1 VZ |
59 | struct Button |
60 | { | |
61 | Button(GtkWidget *button_, int id_) | |
62 | : button(button_), | |
63 | id(id_) | |
64 | { | |
65 | } | |
66 | ||
67 | GtkWidget *button; | |
68 | int id; | |
69 | }; | |
70 | typedef wxVector<Button> Buttons; | |
71 | ||
72 | Buttons m_buttons; | |
73 | }; | |
74 | ||
ed8efd46 VZ |
75 | // ---------------------------------------------------------------------------- |
76 | // local functions | |
77 | // ---------------------------------------------------------------------------- | |
78 | ||
79 | namespace | |
80 | { | |
81 | ||
82 | inline bool UseNative() | |
83 | { | |
9dc44eff PC |
84 | #ifdef __WXGTK3__ |
85 | return true; | |
86 | #else | |
ed8efd46 VZ |
87 | // native GtkInfoBar widget is only available in GTK+ 2.18 and later |
88 | return gtk_check_version(2, 18, 0) == 0; | |
9dc44eff | 89 | #endif |
ed8efd46 VZ |
90 | } |
91 | ||
92 | } // anonymous namespace | |
93 | ||
94 | extern "C" | |
95 | { | |
96 | ||
97 | static void wxgtk_infobar_response(GtkInfoBar * WXUNUSED(infobar), | |
98 | gint btnid, | |
99 | wxInfoBar *win) | |
100 | { | |
101 | win->GTKResponse(btnid); | |
102 | } | |
103 | ||
104 | static void wxgtk_infobar_close(GtkInfoBar * WXUNUSED(infobar), | |
105 | wxInfoBar *win) | |
106 | { | |
107 | win->GTKResponse(wxID_CANCEL); | |
108 | } | |
109 | ||
110 | } // extern "C" section with GTK+ callbacks | |
111 | ||
112 | // ============================================================================ | |
113 | // wxInfoBar implementation | |
114 | // ============================================================================ | |
115 | ||
116 | bool wxInfoBar::Create(wxWindow *parent, wxWindowID winid) | |
117 | { | |
118 | if ( !UseNative() ) | |
119 | return wxInfoBarGeneric::Create(parent, winid); | |
120 | ||
e6b2aae1 VZ |
121 | m_impl = new wxInfoBarGTKImpl; |
122 | ||
ed8efd46 VZ |
123 | // this control is created initially hidden |
124 | Hide(); | |
125 | if ( !CreateBase(parent, winid) ) | |
126 | return false; | |
127 | ||
128 | // create the info bar widget itself | |
129 | m_widget = gtk_info_bar_new(); | |
130 | wxCHECK_MSG( m_widget, false, "failed to create GtkInfoBar" ); | |
131 | g_object_ref(m_widget); | |
132 | ||
133 | // also create a label which will be used to show our message | |
e6b2aae1 VZ |
134 | m_impl->m_label = gtk_label_new(""); |
135 | gtk_widget_show(m_impl->m_label); | |
ed8efd46 VZ |
136 | |
137 | GtkWidget * const | |
138 | contentArea = gtk_info_bar_get_content_area(GTK_INFO_BAR(m_widget)); | |
139 | wxCHECK_MSG( contentArea, false, "failed to get GtkInfoBar content area" ); | |
e6b2aae1 | 140 | gtk_container_add(GTK_CONTAINER(contentArea), m_impl->m_label); |
ed8efd46 VZ |
141 | |
142 | // finish creation and connect to all the signals we're interested in | |
143 | m_parent->DoAddChild(this); | |
144 | ||
145 | PostCreation(wxDefaultSize); | |
146 | ||
147 | GTKConnectWidget("response", G_CALLBACK(wxgtk_infobar_response)); | |
148 | GTKConnectWidget("close", G_CALLBACK(wxgtk_infobar_close)); | |
149 | ||
ff7485d8 | 150 | return true; |
ed8efd46 VZ |
151 | } |
152 | ||
e6b2aae1 VZ |
153 | wxInfoBar::~wxInfoBar() |
154 | { | |
155 | delete m_impl; | |
156 | } | |
157 | ||
ed8efd46 VZ |
158 | void wxInfoBar::ShowMessage(const wxString& msg, int flags) |
159 | { | |
160 | if ( !UseNative() ) | |
161 | { | |
162 | wxInfoBarGeneric::ShowMessage(msg, flags); | |
163 | return; | |
164 | } | |
165 | ||
6a3f8b4f VZ |
166 | // if we don't have any buttons, create a standard close one to give the |
167 | // user at least some way to close the bar | |
168 | if ( m_impl->m_buttons.empty() && !m_impl->m_close ) | |
169 | { | |
170 | m_impl->m_close = GTKAddButton(wxID_CLOSE); | |
171 | } | |
172 | ||
ed8efd46 VZ |
173 | GtkMessageType type; |
174 | if ( wxGTKImpl::ConvertMessageTypeFromWX(flags, &type) ) | |
175 | gtk_info_bar_set_message_type(GTK_INFO_BAR(m_widget), type); | |
e6b2aae1 | 176 | gtk_label_set_text(GTK_LABEL(m_impl->m_label), wxGTK_CONV(msg)); |
ed8efd46 VZ |
177 | |
178 | if ( !IsShown() ) | |
179 | Show(); | |
180 | ||
181 | UpdateParent(); | |
182 | } | |
183 | ||
0b1add25 VZ |
184 | void wxInfoBar::Dismiss() |
185 | { | |
186 | if ( !UseNative() ) | |
187 | { | |
188 | wxInfoBarGeneric::Dismiss(); | |
189 | return; | |
190 | } | |
191 | ||
192 | Hide(); | |
193 | ||
194 | UpdateParent(); | |
195 | } | |
196 | ||
c0945eb2 | 197 | void wxInfoBar::GTKResponse(int btnid) |
ed8efd46 | 198 | { |
ce7fe42e | 199 | wxCommandEvent event(wxEVT_BUTTON, btnid); |
c0945eb2 | 200 | event.SetEventObject(this); |
ed8efd46 | 201 | |
c0945eb2 | 202 | if ( !HandleWindowEvent(event) ) |
0b1add25 | 203 | Dismiss(); |
ed8efd46 VZ |
204 | } |
205 | ||
6a3f8b4f | 206 | GtkWidget *wxInfoBar::GTKAddButton(wxWindowID btnid, const wxString& label) |
ed8efd46 | 207 | { |
6a3f8b4f VZ |
208 | // as GTK+ lays out the buttons vertically, adding another button changes |
209 | // our best size (at least in vertical direction) | |
210 | InvalidateBestSize(); | |
ed8efd46 | 211 | |
e6b2aae1 VZ |
212 | GtkWidget *button = gtk_info_bar_add_button |
213 | ( | |
214 | GTK_INFO_BAR(m_widget), | |
c7193f28 | 215 | (label.empty() |
e6b2aae1 | 216 | ? GTKConvertMnemonics(wxGetStockGtkID(btnid)) |
c7193f28 | 217 | : label).utf8_str(), |
e6b2aae1 VZ |
218 | btnid |
219 | ); | |
e6b2aae1 | 220 | |
6a3f8b4f VZ |
221 | wxASSERT_MSG( button, "unexpectedly failed to add button to info bar" ); |
222 | ||
223 | return button; | |
224 | } | |
225 | ||
226 | void wxInfoBar::AddButton(wxWindowID btnid, const wxString& label) | |
227 | { | |
228 | if ( !UseNative() ) | |
229 | { | |
230 | wxInfoBarGeneric::AddButton(btnid, label); | |
231 | return; | |
232 | } | |
233 | ||
234 | // if we had created the default close button before, remove it now that we | |
235 | // have some user-defined button | |
236 | if ( m_impl->m_close ) | |
237 | { | |
238 | gtk_widget_destroy(m_impl->m_close); | |
239 | m_impl->m_close = NULL; | |
240 | } | |
241 | ||
242 | GtkWidget * const button = GTKAddButton(btnid, label); | |
243 | if ( button ) | |
244 | m_impl->m_buttons.push_back(wxInfoBarGTKImpl::Button(button, btnid)); | |
e6b2aae1 VZ |
245 | } |
246 | ||
247 | void wxInfoBar::RemoveButton(wxWindowID btnid) | |
248 | { | |
249 | if ( !UseNative() ) | |
250 | { | |
251 | wxInfoBarGeneric::RemoveButton(btnid); | |
252 | return; | |
253 | } | |
254 | ||
255 | // as in the generic version, look for the button starting from the end | |
256 | wxInfoBarGTKImpl::Buttons& buttons = m_impl->m_buttons; | |
257 | for ( wxInfoBarGTKImpl::Buttons::reverse_iterator i = buttons.rbegin(); | |
258 | i != buttons.rend(); | |
259 | ++i ) | |
260 | { | |
ff7485d8 PC |
261 | if (i->id == btnid) |
262 | { | |
263 | gtk_widget_destroy(i->button); | |
264 | buttons.erase(i.base()); | |
6a3f8b4f | 265 | |
ff7485d8 PC |
266 | // see comment in GTKAddButton() |
267 | InvalidateBestSize(); | |
6a3f8b4f | 268 | |
ff7485d8 PC |
269 | return; |
270 | } | |
e6b2aae1 VZ |
271 | } |
272 | ||
273 | wxFAIL_MSG( wxString::Format("button with id %d not found", btnid) ); | |
ed8efd46 VZ |
274 | } |
275 | ||
4026acf1 VZ |
276 | void wxInfoBar::DoApplyWidgetStyle(GtkRcStyle *style) |
277 | { | |
278 | wxInfoBarGeneric::DoApplyWidgetStyle(style); | |
279 | ||
280 | if ( UseNative() ) | |
9dc44eff | 281 | GTKApplyStyle(m_impl->m_label, style); |
4026acf1 VZ |
282 | } |
283 | ||
ed8efd46 | 284 | #endif // wxUSE_INFOBAR |