1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
10 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA)
11 #pragma implementation "button.h"
14 // For compilers that support precompilation, includes "wx.h".
15 #include "wx/wxprec.h"
21 #include "wx/button.h"
22 #include "wx/stockitem.h"
24 #include "wx/gtk/private.h"
25 #include "wx/gtk/win_gtk.h"
27 //-----------------------------------------------------------------------------
29 //-----------------------------------------------------------------------------
33 //-----------------------------------------------------------------------------
35 //-----------------------------------------------------------------------------
37 extern void wxapp_install_idle_handler();
40 //-----------------------------------------------------------------------------
42 //-----------------------------------------------------------------------------
44 extern bool g_blockEventsOnDrag
;
46 //-----------------------------------------------------------------------------
48 //-----------------------------------------------------------------------------
50 static void gtk_button_clicked_callback( GtkWidget
*WXUNUSED(widget
), wxButton
*button
)
53 wxapp_install_idle_handler();
55 if (!button
->m_hasVMT
) return;
56 if (g_blockEventsOnDrag
) return;
58 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, button
->GetId());
59 event
.SetEventObject(button
);
60 button
->GetEventHandler()->ProcessEvent(event
);
63 //-----------------------------------------------------------------------------
64 // "style_set" from m_widget
65 //-----------------------------------------------------------------------------
68 gtk_button_style_set_callback( GtkWidget
*m_widget
, GtkStyle
*WXUNUSED(style
), wxButton
*win
)
71 wxapp_install_idle_handler();
76 int bottom_border
= 0;
78 /* the default button has a border around it */
79 if (GTK_WIDGET_CAN_DEFAULT(m_widget
))
82 GtkBorder
*default_border
= NULL
;
83 gtk_widget_style_get( m_widget
, "default_border", &default_border
, NULL
);
86 left_border
+= default_border
->left
;
87 right_border
+= default_border
->right
;
88 top_border
+= default_border
->top
;
89 bottom_border
+= default_border
->bottom
;
90 g_free( default_border
);
98 win
->DoMoveWindow( win
->m_x
-top_border
,
100 win
->m_width
+left_border
+right_border
,
101 win
->m_height
+top_border
+bottom_border
);
107 //-----------------------------------------------------------------------------
109 //-----------------------------------------------------------------------------
111 IMPLEMENT_DYNAMIC_CLASS(wxButton
,wxControl
)
117 wxButton::~wxButton()
121 bool wxButton::Create( wxWindow
*parent
, wxWindowID id
, const wxString
&label
,
122 const wxPoint
&pos
, const wxSize
&size
,
123 long style
, const wxValidator
& validator
, const wxString
&name
)
126 m_acceptsFocus
= TRUE
;
128 if (!PreCreation( parent
, pos
, size
) ||
129 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
131 wxFAIL_MSG( wxT("wxButton creation failed") );
136 wxString label2( label );
137 for (size_t i = 0; i < label2.Len(); i++)
139 if (label2.GetChar(i) == wxT('&'))
140 label2.SetChar(i,wxT('_'));
143 GtkWidget *accel_label = gtk_accel_label_new( label2.mb_str() );
144 gtk_widget_show( accel_label );
146 m_widget = gtk_button_new();
147 gtk_container_add( GTK_CONTAINER(m_widget), accel_label );
149 gtk_accel_label_set_accel_widget( GTK_ACCEL_LABEL(accel_label), m_widget );
151 guint accel_key = gtk_label_parse_uline (GTK_LABEL(accel_label), label2.mb_str() );
152 gtk_accel_label_refetch( GTK_ACCEL_LABEL(accel_label) );
154 wxControl::SetLabel( label );
158 m_widget
= gtk_button_new_with_mnemonic("");
160 m_widget
= gtk_button_new_with_label("");
163 float x_alignment
= 0.5;
164 if (HasFlag(wxBU_LEFT
))
166 else if (HasFlag(wxBU_RIGHT
))
169 float y_alignment
= 0.5;
170 if (HasFlag(wxBU_TOP
))
172 else if (HasFlag(wxBU_BOTTOM
))
176 if (!gtk_check_version(2,4,0))
178 gtk_button_set_alignment(GTK_BUTTON(m_widget
), x_alignment
, y_alignment
);
183 if (GTK_IS_MISC(BUTTON_CHILD(m_widget
)))
184 gtk_misc_set_alignment (GTK_MISC (BUTTON_CHILD (m_widget
)),
185 x_alignment
, y_alignment
);
190 if (style
& wxNO_BORDER
)
191 gtk_button_set_relief( GTK_BUTTON(m_widget
), GTK_RELIEF_NONE
);
193 gtk_signal_connect_after( GTK_OBJECT(m_widget
), "clicked",
194 GTK_SIGNAL_FUNC(gtk_button_clicked_callback
), (gpointer
*)this );
196 gtk_signal_connect_after( GTK_OBJECT(m_widget
), "style_set",
197 GTK_SIGNAL_FUNC(gtk_button_style_set_callback
), (gpointer
*) this );
199 m_parent
->DoAddChild( this );
207 void wxButton::SetDefault()
209 wxWindow
*parent
= GetParent();
210 wxCHECK_RET( parent
, _T("button without parent?") );
212 parent
->SetDefaultItem(this);
214 GTK_WIDGET_SET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
215 gtk_widget_grab_default( m_widget
);
217 // resize for default border
218 gtk_button_style_set_callback( m_widget
, NULL
, this );
222 wxSize
wxButtonBase::GetDefaultSize()
225 static wxSize size
= wxDefaultSize
;
226 if (size
== wxDefaultSize
)
228 // NB: Default size of buttons should be same as size of stock
229 // buttons as used in most GTK+ apps. Unfortunately it's a little
230 // tricky to obtain this size: stock button's size may be smaller
231 // than size of button in GtkButtonBox and vice versa,
232 // GtkButtonBox's minimal button size may be smaller than stock
233 // button's size. We have to retrieve both values and combine them.
235 GtkWidget
*wnd
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
236 GtkWidget
*box
= gtk_hbutton_box_new();
237 GtkWidget
*btn
= gtk_button_new_from_stock(GTK_STOCK_CANCEL
);
238 gtk_container_add(GTK_CONTAINER(box
), btn
);
239 gtk_container_add(GTK_CONTAINER(wnd
), box
);
241 gtk_widget_size_request(btn
, &req
);
243 gint minwidth
, minheight
;
244 gtk_widget_style_get(box
,
245 "child-min-width", &minwidth
,
246 "child-min-height", &minheight
,
249 size
.x
= wxMax(minwidth
, req
.width
);
250 size
.y
= wxMax(minheight
, req
.height
);
252 gtk_widget_destroy(wnd
);
256 return wxSize(80,26);
260 void wxButton::SetLabel( const wxString
&lbl
)
262 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid button") );
266 if (label
.empty() && wxIsStockID(m_windowId
))
267 label
= wxGetStockLabel(m_windowId
);
269 wxControl::SetLabel(label
);
272 if (wxIsStockID(m_windowId
) && wxIsStockLabel(m_windowId
, label
))
274 const char *stock
= wxGetStockGtkID(m_windowId
);
277 gtk_button_set_label(GTK_BUTTON(m_widget
), stock
);
278 gtk_button_set_use_stock(GTK_BUTTON(m_widget
), TRUE
);
283 wxString label2
= PrepareLabelMnemonics(label
);
284 gtk_button_set_label(GTK_BUTTON(m_widget
), wxGTK_CONV(label2
));
285 gtk_button_set_use_stock(GTK_BUTTON(m_widget
), FALSE
);
287 ApplyWidgetStyle( false );
290 gtk_label_set(GTK_LABEL(BUTTON_CHILD(m_widget
)), wxGTK_CONV(GetLabel()));
294 bool wxButton::Enable( bool enable
)
296 if ( !wxControl::Enable( enable
) )
299 gtk_widget_set_sensitive( BUTTON_CHILD(m_widget
), enable
);
304 bool wxButton::IsOwnGtkWindow( GdkWindow
*window
)
307 return GTK_BUTTON(m_widget
)->event_window
;
309 return (window
== m_widget
->window
);
313 void wxButton::DoApplyWidgetStyle(GtkRcStyle
*style
)
315 gtk_widget_modify_style(m_widget
, style
);
316 gtk_widget_modify_style(BUTTON_CHILD(m_widget
), style
);
319 wxSize
wxButton::DoGetBestSize() const
321 // the default button in wxGTK is bigger than the other ones because of an
322 // extra border around it, but we don't want to take it into account in
323 // our size calculations (otherwsie the result is visually ugly), so
324 // always return the size of non default button from here
325 const bool isDefault
= GTK_WIDGET_HAS_DEFAULT(m_widget
);
328 // temporarily unset default flag
329 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
332 wxSize
ret( wxControl::DoGetBestSize() );
337 GTK_WIDGET_SET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
341 ret
.x
+= 10; // add a few pixels for sloppy (but common) themes
344 if (!HasFlag(wxBU_EXACTFIT
))
346 wxSize defaultSize
= GetDefaultSize();
347 if (ret
.x
< defaultSize
.x
) ret
.x
= defaultSize
.x
;
348 if (ret
.y
< defaultSize
.y
) ret
.y
= defaultSize
.y
;
357 wxButton::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
359 return GetDefaultAttributesFromGTKWidget(gtk_button_new
);
362 #endif // wxUSE_BUTTON