1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/control.cpp 
   3 // Purpose:     wxControl implementation for wxGTK 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling, Julian Smart and Vadim Zeitlin 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  15 #include "wx/control.h" 
  19     #include "wx/settings.h" 
  22 #include "wx/fontutil.h" 
  23 #include "wx/gtk/private.h" 
  25 // ============================================================================ 
  26 // wxControl implementation 
  27 // ============================================================================ 
  29 // ---------------------------------------------------------------------------- 
  31 // ---------------------------------------------------------------------------- 
  33 IMPLEMENT_DYNAMIC_CLASS(wxControl
, wxWindow
) 
  35 wxControl::wxControl() 
  40 bool wxControl::Create( wxWindow 
*parent
, 
  45                       const wxValidator
& validator
, 
  46                       const wxString 
&name 
) 
  48     bool ret 
= wxWindow::Create(parent
, id
, pos
, size
, style
, name
); 
  51     SetValidator(validator
); 
  57 wxSize 
wxControl::DoGetBestSize() const 
  59     // Do not return any arbitrary default value... 
  60     wxASSERT_MSG( m_widget
, wxT("DoGetBestSize called before creation") ); 
  65     (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) )->size_request 
) 
  68     wxSize 
best(req
.width
, req
.height
); 
  73 void wxControl::PostCreation(const wxSize
& size
) 
  75     wxWindow::PostCreation(); 
  77     // NB: GetBestSize needs to know the style, otherwise it will assume 
  78     //     default font and if the user uses a different font, determined 
  79     //     best size will be different (typically, smaller) than the desired 
  80     //     size. This call ensure that a style is available at the time 
  81     //     GetBestSize is called. 
  82     gtk_widget_ensure_style(m_widget
); 
  88 // ---------------------------------------------------------------------------- 
  89 // wxControl dealing with labels 
  90 // ---------------------------------------------------------------------------- 
  92 void wxControl::SetLabel( const wxString 
&label 
) 
  94     // keep the original string internally to be able to return it later (for 
  95     // consistency with the other ports) 
 101 wxString 
wxControl::GetLabel() const 
 106 void wxControl::GTKSetLabelForLabel(GtkLabel 
*w
, const wxString
& label
) 
 108     // don't call the virtual function which might call this one back again 
 109     wxControl::SetLabel(label
); 
 111     const wxString labelGTK 
= GTKConvertMnemonics(label
); 
 113     gtk_label_set_text_with_mnemonic(w
, wxGTK_CONV(labelGTK
)); 
 116 // ---------------------------------------------------------------------------- 
 119 // GtkFrames do in fact support mnemonics in GTK2+ but not through 
 120 // gtk_frame_set_label, rather you need to use a custom label widget 
 121 // instead (idea gleaned from the native gtk font dialog code in GTK) 
 122 // ---------------------------------------------------------------------------- 
 124 GtkWidget
* wxControl::GTKCreateFrame(const wxString
& label
) 
 126     const wxString labelGTK 
= GTKConvertMnemonics(label
); 
 127     GtkWidget
* labelwidget 
= gtk_label_new_with_mnemonic(wxGTK_CONV(labelGTK
)); 
 128     gtk_widget_show(labelwidget
); // without this it won't show... 
 130     GtkWidget
* framewidget 
= gtk_frame_new(NULL
); 
 131     gtk_frame_set_label_widget(GTK_FRAME(framewidget
), labelwidget
); 
 133     return framewidget
; //note that the label is already set so you'll 
 134                         //only need to call wxControl::SetLabel afterwards 
 137 void wxControl::GTKSetLabelForFrame(GtkFrame 
*w
, const wxString
& label
) 
 139     GtkLabel
* labelwidget 
= GTK_LABEL(gtk_frame_get_label_widget(w
)); 
 140     GTKSetLabelForLabel(labelwidget
, label
); 
 143 void wxControl::GTKFrameApplyWidgetStyle(GtkFrame
* w
, GtkRcStyle
* style
) 
 145     gtk_widget_modify_style(GTK_WIDGET(w
), style
); 
 146     gtk_widget_modify_style(gtk_frame_get_label_widget (w
), style
); 
 149 void wxControl::GTKFrameSetMnemonicWidget(GtkFrame
* w
, GtkWidget
* widget
) 
 151     GtkLabel
* labelwidget 
= GTK_LABEL(gtk_frame_get_label_widget(w
)); 
 153     gtk_label_set_mnemonic_widget(labelwidget
, widget
); 
 156 // ---------------------------------------------------------------------------- 
 157 // worker function implementing both GTKConvert/RemoveMnemonics() 
 159 // notice that under GTK+ 1 we only really need to support MNEMONICS_REMOVE as 
 160 // it doesn't support mnemonics anyhow but this would make the code so ugly 
 161 // that we do the same thing for GKT+ 1 and 2 
 162 // ---------------------------------------------------------------------------- 
 170 static wxString 
GTKProcessMnemonics(const wxString
& label
, MnemonicsFlag flag
) 
 172     const size_t len 
= label
.length(); 
 174     labelGTK
.reserve(len
); 
 175     for ( size_t i 
= 0; i 
< len
; i
++ ) 
 177         wxChar ch 
= label
[i
]; 
 184                     // "&" at the end of string is an error 
 185                     wxLogDebug(wxT("Invalid label \"%s\"."), label
.c_str()); 
 189                 ch 
= label
[++i
]; // skip '&' itself 
 193                         // special case: "&&" is not a mnemonic at all but just 
 195                         labelGTK 
+= wxT('&'); 
 199                         if ( flag 
== MNEMONICS_CONVERT 
) 
 201                             // '_' can't be a GTK mnemonic apparently so 
 202                             // replace it with something similar 
 203                             labelGTK 
+= wxT("_-"); 
 209                         if ( flag 
== MNEMONICS_CONVERT 
) 
 210                             labelGTK 
+= wxT('_'); 
 216                 if ( flag 
== MNEMONICS_CONVERT 
) 
 218                     // escape any existing underlines in the string so that 
 219                     // they don't become mnemonics accidentally 
 220                     labelGTK 
+= wxT("__"); 
 234 wxString 
wxControl::GTKRemoveMnemonics(const wxString
& label
) 
 236     return GTKProcessMnemonics(label
, MNEMONICS_REMOVE
); 
 240 wxString 
wxControl::GTKConvertMnemonics(const wxString
& label
) 
 242     return GTKProcessMnemonics(label
, MNEMONICS_CONVERT
); 
 245 // ---------------------------------------------------------------------------- 
 246 // wxControl styles (a.k.a. attributes) 
 247 // ---------------------------------------------------------------------------- 
 249 wxVisualAttributes 
wxControl::GetDefaultAttributes() const 
 251     return GetDefaultAttributesFromGTKWidget(m_widget
, 
 256 #define SHIFT (8*(sizeof(short int)-sizeof(char))) 
 260 wxControl::GetDefaultAttributesFromGTKWidget(GtkWidget
* widget
, 
 265     wxVisualAttributes attr
; 
 267     style 
= gtk_rc_get_style(widget
); 
 269         style 
= gtk_widget_get_default_style(); 
 273         return wxWindow::GetClassDefaultAttributes(wxWINDOW_VARIANT_NORMAL
); 
 277         state 
= GTK_STATE_NORMAL
; 
 279     // get the style's colours 
 280     attr
.colFg 
= wxColour(style
->fg
[state
].red   
>> SHIFT
, 
 281                           style
->fg
[state
].green 
>> SHIFT
, 
 282                           style
->fg
[state
].blue  
>> SHIFT
); 
 284         attr
.colBg 
= wxColour(style
->base
[state
].red   
>> SHIFT
, 
 285                               style
->base
[state
].green 
>> SHIFT
, 
 286                               style
->base
[state
].blue  
>> SHIFT
); 
 288         attr
.colBg 
= wxColour(style
->bg
[state
].red   
>> SHIFT
, 
 289                               style
->bg
[state
].green 
>> SHIFT
, 
 290                               style
->bg
[state
].blue  
>> SHIFT
); 
 292     // get the style's font 
 293     if ( !style
->font_desc 
) 
 294         style 
= gtk_widget_get_default_style(); 
 295     if ( style 
&& style
->font_desc 
) 
 297         wxNativeFontInfo info
; 
 298         info
.description 
= pango_font_description_copy(style
->font_desc
); 
 299         attr
.font 
= wxFont(info
); 
 303         GtkSettings 
*settings 
= gtk_settings_get_default(); 
 304         gchar 
*font_name 
= NULL
; 
 305         g_object_get ( settings
, 
 310             attr
.font 
= wxSystemSettings::GetFont( wxSYS_DEFAULT_GUI_FONT 
); 
 312             attr
.font 
= wxFont(wxString::FromAscii(font_name
)); 
 322 wxControl::GetDefaultAttributesFromGTKWidget(wxGtkWidgetNew_t widget_new
, 
 326     wxVisualAttributes attr
; 
 327     // NB: we need toplevel window so that GTK+ can find the right style 
 328     GtkWidget 
*wnd 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 329     GtkWidget
* widget 
= widget_new(); 
 330     gtk_container_add(GTK_CONTAINER(wnd
), widget
); 
 331     attr 
= GetDefaultAttributesFromGTKWidget(widget
, useBase
, state
); 
 332     gtk_widget_destroy(wnd
); 
 338 wxControl::GetDefaultAttributesFromGTKWidget(wxGtkWidgetNewFromStr_t widget_new
, 
 342     wxVisualAttributes attr
; 
 343     // NB: we need toplevel window so that GTK+ can find the right style 
 344     GtkWidget 
*wnd 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 345     GtkWidget
* widget 
= widget_new(""); 
 346     gtk_container_add(GTK_CONTAINER(wnd
), widget
); 
 347     attr 
= GetDefaultAttributesFromGTKWidget(widget
, useBase
, state
); 
 348     gtk_widget_destroy(wnd
); 
 355 wxControl::GetDefaultAttributesFromGTKWidget(wxGtkWidgetNewFromAdj_t widget_new
, 
 359     wxVisualAttributes attr
; 
 360     // NB: we need toplevel window so that GTK+ can find the right style 
 361     GtkWidget 
*wnd 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 362     GtkWidget
* widget 
= widget_new(NULL
); 
 363     gtk_container_add(GTK_CONTAINER(wnd
), widget
); 
 364     attr 
= GetDefaultAttributesFromGTKWidget(widget
, useBase
, state
); 
 365     gtk_widget_destroy(wnd
); 
 369 // ---------------------------------------------------------------------------- 
 371 // ---------------------------------------------------------------------------- 
 373 void wxControl::OnInternalIdle() 
 375     if ( GtkShowFromOnIdle() ) 
 378     if ( GTK_WIDGET_REALIZED(m_widget
) ) 
 382         GTKSetDelayedFocusIfNeeded(); 
 385     if ( wxUpdateUIEvent::CanUpdate(this) ) 
 386         UpdateWindowUI(wxUPDATE_UI_FROMIDLE
); 
 389 #endif // wxUSE_CONTROLS