1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/button.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"
16 #include "wx/button.h"
19 #include "wx/stockitem.h"
21 #include "wx/gtk/private.h"
23 //-----------------------------------------------------------------------------
25 //-----------------------------------------------------------------------------
27 extern bool g_blockEventsOnDrag
;
29 //-----------------------------------------------------------------------------
31 //-----------------------------------------------------------------------------
34 static void gtk_button_clicked_callback( GtkWidget
*WXUNUSED(widget
), wxButton
*button
)
36 if (!button
->m_hasVMT
) return;
37 if (g_blockEventsOnDrag
) return;
39 wxCommandEvent
event(wxEVT_COMMAND_BUTTON_CLICKED
, button
->GetId());
40 event
.SetEventObject(button
);
41 button
->HandleWindowEvent(event
);
45 //-----------------------------------------------------------------------------
46 // "style_set" from m_widget
47 //-----------------------------------------------------------------------------
51 gtk_button_style_set_callback(GtkWidget
* widget
, GtkStyle
*, wxButton
* win
)
53 /* the default button has a border around it */
54 wxWindow
* parent
= win
->GetParent();
55 if (parent
&& parent
->m_wxwindow
&& GTK_WIDGET_CAN_DEFAULT(widget
))
57 GtkBorder
* border
= NULL
;
58 gtk_widget_style_get(widget
, "default_border", &border
, NULL
);
62 win
->m_x
- border
->left
,
63 win
->m_y
- border
->top
,
64 win
->m_width
+ border
->left
+ border
->right
,
65 win
->m_height
+ border
->top
+ border
->bottom
);
66 gtk_border_free(border
);
72 //-----------------------------------------------------------------------------
74 //-----------------------------------------------------------------------------
76 IMPLEMENT_DYNAMIC_CLASS(wxButton
,wxControl
)
86 bool wxButton::Create(wxWindow
*parent
,
88 const wxString
&label
,
92 const wxValidator
& validator
,
95 if (!PreCreation( parent
, pos
, size
) ||
96 !CreateBase( parent
, id
, pos
, size
, style
, validator
, name
))
98 wxFAIL_MSG( wxT("wxButton creation failed") );
102 m_widget
= gtk_button_new_with_mnemonic("");
104 float x_alignment
= 0.5;
105 if (HasFlag(wxBU_LEFT
))
107 else if (HasFlag(wxBU_RIGHT
))
110 float y_alignment
= 0.5;
111 if (HasFlag(wxBU_TOP
))
113 else if (HasFlag(wxBU_BOTTOM
))
116 gtk_button_set_alignment(GTK_BUTTON(m_widget
), x_alignment
, y_alignment
);
120 if (style
& wxNO_BORDER
)
121 gtk_button_set_relief( GTK_BUTTON(m_widget
), GTK_RELIEF_NONE
);
123 g_signal_connect_after (m_widget
, "clicked",
124 G_CALLBACK (gtk_button_clicked_callback
),
127 g_signal_connect_after (m_widget
, "style_set",
128 G_CALLBACK (gtk_button_style_set_callback
),
131 m_parent
->DoAddChild( this );
139 wxWindow
*wxButton::SetDefault()
141 wxWindow
*oldDefault
= wxButtonBase::SetDefault();
143 GTK_WIDGET_SET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
144 gtk_widget_grab_default( m_widget
);
146 // resize for default border
147 gtk_button_style_set_callback( m_widget
, NULL
, this );
153 wxSize
wxButtonBase::GetDefaultSize()
155 static wxSize size
= wxDefaultSize
;
156 if (size
== wxDefaultSize
)
158 // NB: Default size of buttons should be same as size of stock
159 // buttons as used in most GTK+ apps. Unfortunately it's a little
160 // tricky to obtain this size: stock button's size may be smaller
161 // than size of button in GtkButtonBox and vice versa,
162 // GtkButtonBox's minimal button size may be smaller than stock
163 // button's size. We have to retrieve both values and combine them.
165 GtkWidget
*wnd
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
166 GtkWidget
*box
= gtk_hbutton_box_new();
167 GtkWidget
*btn
= gtk_button_new_from_stock(GTK_STOCK_CANCEL
);
168 gtk_container_add(GTK_CONTAINER(box
), btn
);
169 gtk_container_add(GTK_CONTAINER(wnd
), box
);
171 gtk_widget_size_request(btn
, &req
);
173 gint minwidth
, minheight
;
174 gtk_widget_style_get(box
,
175 "child-min-width", &minwidth
,
176 "child-min-height", &minheight
,
179 size
.x
= wxMax(minwidth
, req
.width
);
180 size
.y
= wxMax(minheight
, req
.height
);
182 gtk_widget_destroy(wnd
);
187 void wxButton::SetLabel( const wxString
&lbl
)
189 wxCHECK_RET( m_widget
!= NULL
, wxT("invalid button") );
193 if (label
.empty() && wxIsStockID(m_windowId
))
194 label
= wxGetStockLabel(m_windowId
);
196 wxControl::SetLabel(label
);
198 if (wxIsStockID(m_windowId
) && wxIsStockLabel(m_windowId
, label
))
200 const char *stock
= wxGetStockGtkID(m_windowId
);
203 gtk_button_set_label(GTK_BUTTON(m_widget
), stock
);
204 gtk_button_set_use_stock(GTK_BUTTON(m_widget
), TRUE
);
209 const wxString labelGTK
= GTKConvertMnemonics(label
);
210 gtk_button_set_label(GTK_BUTTON(m_widget
), wxGTK_CONV(labelGTK
));
211 gtk_button_set_use_stock(GTK_BUTTON(m_widget
), FALSE
);
213 ApplyWidgetStyle( false );
216 bool wxButton::Enable( bool enable
)
218 if ( !wxControl::Enable( enable
) )
221 gtk_widget_set_sensitive(GTK_BIN(m_widget
)->child
, enable
);
226 GdkWindow
*wxButton::GTKGetWindow(wxArrayGdkWindows
& WXUNUSED(windows
)) const
228 return GTK_BUTTON(m_widget
)->event_window
;
231 void wxButton::DoApplyWidgetStyle(GtkRcStyle
*style
)
233 gtk_widget_modify_style(m_widget
, style
);
234 GtkWidget
*child
= GTK_BIN(m_widget
)->child
;
235 gtk_widget_modify_style(child
, style
);
237 // for buttons with images, the path to the label is (at least in 2.12)
238 // GtkButton -> GtkAlignment -> GtkHBox -> GtkLabel
239 if ( GTK_IS_ALIGNMENT(child
) )
241 GtkWidget
*box
= GTK_BIN(child
)->child
;
242 if ( GTK_IS_BOX(box
) )
244 GList
*items
= gtk_container_get_children(GTK_CONTAINER(box
));
245 for ( GList
*item
= items
; item
; item
= item
->next
)
246 gtk_widget_modify_style(GTK_WIDGET(item
->data
), style
);
251 wxSize
wxButton::DoGetBestSize() const
253 // the default button in wxGTK is bigger than the other ones because of an
254 // extra border around it, but we don't want to take it into account in
255 // our size calculations (otherwsie the result is visually ugly), so
256 // always return the size of non default button from here
257 const bool isDefault
= GTK_WIDGET_HAS_DEFAULT(m_widget
);
260 // temporarily unset default flag
261 GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
264 wxSize
ret( wxControl::DoGetBestSize() );
269 GTK_WIDGET_SET_FLAGS( m_widget
, GTK_CAN_DEFAULT
);
272 if (!HasFlag(wxBU_EXACTFIT
))
274 wxSize defaultSize
= GetDefaultSize();
275 if (ret
.x
< defaultSize
.x
) ret
.x
= defaultSize
.x
;
276 if (ret
.y
< defaultSize
.y
) ret
.y
= defaultSize
.y
;
285 wxButton::GetClassDefaultAttributes(wxWindowVariant
WXUNUSED(variant
))
287 return GetDefaultAttributesFromGTKWidget(gtk_button_new
);
290 #endif // wxUSE_BUTTON