1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     GTK toolbar 
   4 // Author:      Robert Roebling 
   5 // Modified:    13.12.99 by VZ to derive from wxToolBarBase 
   7 // Copyright:   (c) Robert Roebling 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  11 // ============================================================================ 
  13 // ============================================================================ 
  15 // ---------------------------------------------------------------------------- 
  17 // ---------------------------------------------------------------------------- 
  20     #pragma implementation "tbargtk.h" 
  23 #include "wx/toolbar.h" 
  25 #if wxUSE_TOOLBAR_NATIVE 
  30 #include "wx/gtk/private.h" 
  32 extern GdkFont 
*GtkGetDefaultGuiFont(); 
  34 // ---------------------------------------------------------------------------- 
  36 // ---------------------------------------------------------------------------- 
  39 extern void wxapp_install_idle_handler(); 
  43 extern bool       g_blockEventsOnDrag
; 
  44 extern wxCursor   g_globalCursor
; 
  46 // ---------------------------------------------------------------------------- 
  48 // ---------------------------------------------------------------------------- 
  50 // translate wxWindows toolbar style flags to GTK orientation and style 
  51 static void GetGtkStyle(long style
, 
  52                         GtkOrientation 
*orient
, GtkToolbarStyle 
*gtkStyle
) 
  54     *orient 
= style 
& wxTB_VERTICAL 
? GTK_ORIENTATION_VERTICAL
 
  55                                     : GTK_ORIENTATION_HORIZONTAL
; 
  58     if ( style 
& wxTB_TEXT 
) 
  60         *gtkStyle 
= style 
& wxTB_NOICONS 
? GTK_TOOLBAR_TEXT 
: GTK_TOOLBAR_BOTH
; 
  62     else // no text, hence we must have the icons or what would we show? 
  64         *gtkStyle 
= GTK_TOOLBAR_ICONS
; 
  68 // ---------------------------------------------------------------------------- 
  70 // ---------------------------------------------------------------------------- 
  72 class wxToolBarTool 
: public wxToolBarToolBase
 
  75     wxToolBarTool(wxToolBar 
*tbar
, 
  77                   const wxString
& label
, 
  78                   const wxBitmap
& bitmap1
, 
  79                   const wxBitmap
& bitmap2
, 
  82                   const wxString
& shortHelpString
, 
  83                   const wxString
& longHelpString
) 
  84         : wxToolBarToolBase(tbar
, id
, label
, bitmap1
, bitmap2
, kind
, 
  85                             clientData
, shortHelpString
, longHelpString
) 
  90     wxToolBarTool(wxToolBar 
*tbar
, wxControl 
*control
) 
  91         : wxToolBarToolBase(tbar
, control
) 
  96     // is this a radio button? 
  98     // unlike GetKind(), can be called for any kind of tools, not just buttons 
  99     bool IsRadio() const { return IsButton() && GetKind() == wxITEM_RADIO
; } 
 101     // this is only called for the normal buttons, i.e. not separators nor 
 103     GtkToolbarChildType 
GetGtkChildType() const 
 108                 return GTK_TOOLBAR_CHILD_TOGGLEBUTTON
; 
 111                 return GTK_TOOLBAR_CHILD_RADIOBUTTON
; 
 114                 wxFAIL_MSG( _T("unknown toolbar child type") ); 
 118                 return GTK_TOOLBAR_CHILD_BUTTON
; 
 129 // ---------------------------------------------------------------------------- 
 131 // ---------------------------------------------------------------------------- 
 133 IMPLEMENT_DYNAMIC_CLASS(wxToolBar
, wxToolBarBase
) 
 135 // ============================================================================ 
 137 // ============================================================================ 
 139 //----------------------------------------------------------------------------- 
 140 // "clicked" (internal from gtk_toolbar) 
 141 //----------------------------------------------------------------------------- 
 143 static void gtk_toolbar_callback( GtkWidget 
*WXUNUSED(widget
), 
 144                                   wxToolBarTool 
*tool 
) 
 147         wxapp_install_idle_handler(); 
 149     wxToolBar 
*tbar 
= (wxToolBar 
*)tool
->GetToolBar(); 
 151     if (tbar
->m_blockEvent
) return; 
 153     if (g_blockEventsOnDrag
) return; 
 154     if (!tool
->IsEnabled()) return; 
 156     if (tool
->CanBeToggled()) 
 160         wxBitmap bitmap 
= tool
->GetBitmap(); 
 163             GtkPixmap 
*pixmap 
= GTK_PIXMAP( tool
->m_pixmap 
); 
 165             GdkBitmap 
*mask 
= bitmap
.GetMask() ? bitmap
.GetMask()->GetBitmap() 
 168             gtk_pixmap_set( pixmap
, bitmap
.GetPixmap(), mask 
); 
 171         if ( tool
->IsRadio() && !tool
->IsToggled() ) 
 173             // radio button went up, don't report this as a wxWin event 
 178     tbar
->OnLeftClick( tool
->GetId(), tool
->IsToggled() ); 
 181 //----------------------------------------------------------------------------- 
 182 // "enter_notify_event" / "leave_notify_event" 
 183 //----------------------------------------------------------------------------- 
 185 static gint 
gtk_toolbar_tool_callback( GtkWidget 
*WXUNUSED(widget
),  
 186                                        GdkEventCrossing 
*gdk_event
, 
 187                                        wxToolBarTool 
*tool 
) 
 189     if (g_isIdle
) wxapp_install_idle_handler(); 
 191     if (g_blockEventsOnDrag
) return TRUE
; 
 193     wxToolBar 
*tb 
= (wxToolBar 
*)tool
->GetToolBar(); 
 196     if( gdk_event
->type 
== GDK_ENTER_NOTIFY 
) 
 197         tb
->OnMouseEnter( tool
->GetId() ); 
 199         tb
->OnMouseEnter( -1 ); 
 204 //----------------------------------------------------------------------------- 
 205 // InsertChild callback for wxToolBar 
 206 //----------------------------------------------------------------------------- 
 208 static void wxInsertChildInToolBar( wxToolBar
* WXUNUSED(parent
), 
 209                                     wxWindow
* WXUNUSED(child
) ) 
 211     // we don't do anything here 
 214 // ---------------------------------------------------------------------------- 
 216 // ---------------------------------------------------------------------------- 
 218 void wxToolBarTool::Init() 
 221     m_pixmap 
= (GtkWidget 
*)NULL
; 
 224 wxToolBarToolBase 
*wxToolBar::CreateTool(int id
, 
 225                                          const wxString
& text
, 
 226                                          const wxBitmap
& bitmap1
, 
 227                                          const wxBitmap
& bitmap2
, 
 229                                          wxObject 
*clientData
, 
 230                                          const wxString
& shortHelpString
, 
 231                                          const wxString
& longHelpString
) 
 233     return new wxToolBarTool(this, id
, text
, bitmap1
, bitmap2
, kind
, 
 234                              clientData
, shortHelpString
, longHelpString
); 
 237 wxToolBarToolBase 
*wxToolBar::CreateTool(wxControl 
*control
) 
 239     return new wxToolBarTool(this, control
); 
 242 //----------------------------------------------------------------------------- 
 243 // wxToolBar construction 
 244 //----------------------------------------------------------------------------- 
 246 void wxToolBar::Init() 
 249     m_bg 
= (GdkColor 
*)NULL
; 
 250     m_toolbar 
= (GtkToolbar 
*)NULL
; 
 251     m_blockEvent 
= FALSE
; 
 254 wxToolBar::~wxToolBar() 
 260 bool wxToolBar::Create( wxWindow 
*parent
, 
 265                         const wxString
& name 
) 
 268     m_insertCallback 
= (wxInsertChildFunction
)wxInsertChildInToolBar
; 
 270     if ( !PreCreation( parent
, pos
, size 
) || 
 271          !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
 273         wxFAIL_MSG( wxT("wxToolBar creation failed") ); 
 279     m_toolbar 
= GTK_TOOLBAR( gtk_toolbar_new() ); 
 282     GtkOrientation orient
; 
 283     GtkToolbarStyle gtkStyle
; 
 284     GetGtkStyle(style
, &orient
, >kStyle
); 
 286     m_toolbar 
= GTK_TOOLBAR( gtk_toolbar_new(orient
, gtkStyle
) ); 
 289     SetToolSeparation(7); 
 291     if (style 
& wxTB_DOCKABLE
) 
 293         m_widget 
= gtk_handle_box_new(); 
 294         gtk_container_add( GTK_CONTAINER(m_widget
), GTK_WIDGET(m_toolbar
) ); 
 295         gtk_widget_show( GTK_WIDGET(m_toolbar
) ); 
 297         if (style 
& wxTB_FLAT
) 
 298             gtk_handle_box_set_shadow_type( GTK_HANDLE_BOX(m_widget
), GTK_SHADOW_NONE 
); 
 302         m_widget 
= GTK_WIDGET(m_toolbar
); 
 305     gtk_toolbar_set_tooltips( GTK_TOOLBAR(m_toolbar
), TRUE 
); 
 307     // FIXME: there is no such function for toolbars in 2.0 
 309     if (style 
& wxTB_FLAT
) 
 310         gtk_toolbar_set_button_relief( GTK_TOOLBAR(m_toolbar
), GTK_RELIEF_NONE 
); 
 319     fg
.CalcPixel( gtk_widget_get_colormap( GTK_WIDGET(m_toolbar
) ) ); 
 320     m_fg
->pixel 
= fg
.GetPixel(); 
 326     wxColour 
bg(255,255,196); 
 327     bg
.CalcPixel( gtk_widget_get_colormap( GTK_WIDGET(m_toolbar
) ) ); 
 328     m_bg
->pixel 
= bg
.GetPixel(); 
 330     gtk_tooltips_force_window( GTK_TOOLBAR(m_toolbar
)->tooltips 
); 
 334                 gtk_widget_get_style(  
 335                     GTK_TOOLBAR(m_toolbar
)->tooltips
->tip_window 
) ); 
 337     g_style
->bg
[GTK_STATE_NORMAL
] = *m_bg
; 
 339     SET_STYLE_FONT(g_style
, GtkGetDefaultGuiFont()); 
 341     gtk_widget_set_style( GTK_TOOLBAR(m_toolbar
)->tooltips
->tip_window
, g_style 
); 
 343     m_parent
->DoAddChild( this ); 
 352 void wxToolBar::GtkSetStyle() 
 354     GtkOrientation orient
; 
 355     GtkToolbarStyle style
; 
 356     GetGtkStyle(GetWindowStyle(), &orient
, &style
); 
 358     gtk_toolbar_set_orientation(m_toolbar
, orient
); 
 359     gtk_toolbar_set_style(m_toolbar
, style
); 
 362 void wxToolBar::SetWindowStyleFlag( long style 
) 
 364     wxToolBarBase::SetWindowStyleFlag(style
); 
 370 bool wxToolBar::DoInsertTool(size_t pos
, wxToolBarToolBase 
*toolBase
) 
 372     wxToolBarTool 
*tool 
= (wxToolBarTool 
*)toolBase
; 
 374     // we have inserted a space before all the tools 
 375     if (m_xMargin 
> 1) pos
++; 
 377     if ( tool
->IsButton() ) 
 379         wxBitmap bitmap 
= tool
->GetNormalBitmap(); 
 381         wxCHECK_MSG( bitmap
.Ok(), FALSE
, 
 382                      wxT("invalid bitmap for wxToolBar icon") ); 
 384         wxCHECK_MSG( bitmap
.GetBitmap() == NULL
, FALSE
, 
 385                      wxT("wxToolBar doesn't support GdkBitmap") ); 
 387         wxCHECK_MSG( bitmap
.GetPixmap() != NULL
, FALSE
, 
 388                      wxT("wxToolBar::Add needs a wxBitmap") ); 
 390         GtkWidget 
*tool_pixmap 
= (GtkWidget 
*)NULL
; 
 392         GdkPixmap 
*pixmap 
= bitmap
.GetPixmap(); 
 394         GdkBitmap 
*mask 
= (GdkBitmap 
*)NULL
; 
 395         if ( bitmap
.GetMask() ) 
 396           mask 
= bitmap
.GetMask()->GetBitmap(); 
 398         tool_pixmap 
= gtk_pixmap_new( pixmap
, mask 
); 
 399 #if (GTK_MINOR_VERSION > 0) 
 400         gtk_pixmap_set_build_insensitive( GTK_PIXMAP(tool_pixmap
), TRUE 
); 
 403         gtk_misc_set_alignment( GTK_MISC(tool_pixmap
), 0.5, 0.5 ); 
 405         tool
->m_pixmap 
= tool_pixmap
; 
 408     switch ( tool
->GetStyle() ) 
 410         case wxTOOL_STYLE_BUTTON
: 
 411             // for a radio button we need the widget which starts the radio 
 412             // group it belongs to, i.e. the first radio button immediately 
 413             // preceding this one 
 415                 GtkWidget 
*widget 
= NULL
; 
 417                 if ( tool
->IsRadio() ) 
 419                     wxToolBarToolsList::Node 
*node 
= pos 
? m_tools
.Item(pos 
- 1) 
 423                         wxToolBarTool 
*tool 
= (wxToolBarTool 
*)node
->GetData(); 
 424                         if ( !tool
->IsRadio() ) 
 427                         widget 
= tool
->m_item
; 
 429                         node 
= node
->GetPrevious(); 
 434                         // this is the first button in the radio button group, 
 435                         // it will be toggled automatically by GTK so bring the 
 436                         // internal flag in sync 
 441                 tool
->m_item 
= gtk_toolbar_insert_element
 
 444                                   tool
->GetGtkChildType(), 
 446                                   tool
->GetLabel().empty() 
 448                                     : (const char*) wxGTK_CONV( tool
->GetLabel() ), 
 449                                   tool
->GetShortHelp().empty() 
 451                                     : (const char*) wxGTK_CONV( tool
->GetShortHelp() ), 
 452                                   "", // tooltip_private_text (?) 
 454                                   (GtkSignalFunc
)gtk_toolbar_callback
, 
 461                     wxFAIL_MSG( _T("gtk_toolbar_insert_element() failed") ); 
 466                 gtk_signal_connect( GTK_OBJECT(tool
->m_item
), 
 467                                     "enter_notify_event",  
 468                                     GTK_SIGNAL_FUNC(gtk_toolbar_tool_callback
), 
 470                 gtk_signal_connect( GTK_OBJECT(tool
->m_item
), 
 471                                     "leave_notify_event",  
 472                                     GTK_SIGNAL_FUNC(gtk_toolbar_tool_callback
), 
 477         case wxTOOL_STYLE_SEPARATOR
: 
 478             gtk_toolbar_insert_space( m_toolbar
, pos 
); 
 483         case wxTOOL_STYLE_CONTROL
: 
 484             gtk_toolbar_insert_widget( 
 486                                        tool
->GetControl()->m_widget
, 
 495     (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) )->size_request 
) 
 497     m_width 
= req
.width 
+ m_xMargin
; 
 498     m_height 
= req
.height 
+ 2*m_yMargin
; 
 503 bool wxToolBar::DoDeleteTool(size_t WXUNUSED(pos
), wxToolBarToolBase 
*toolBase
) 
 505     wxToolBarTool 
*tool 
= (wxToolBarTool 
*)toolBase
; 
 507     switch ( tool
->GetStyle() ) 
 509         case wxTOOL_STYLE_CONTROL
: 
 510             tool
->GetControl()->Destroy(); 
 513         case wxTOOL_STYLE_BUTTON
: 
 514             gtk_widget_destroy( tool
->m_item 
); 
 517         //case wxTOOL_STYLE_SEPARATOR: -- nothing to do 
 523 // ---------------------------------------------------------------------------- 
 524 // wxToolBar tools state 
 525 // ---------------------------------------------------------------------------- 
 527 void wxToolBar::DoEnableTool(wxToolBarToolBase 
*toolBase
, bool enable
) 
 529     wxToolBarTool 
*tool 
= (wxToolBarTool 
*)toolBase
; 
 533         gtk_widget_set_sensitive( tool
->m_item
, enable 
); 
 537 void wxToolBar::DoToggleTool( wxToolBarToolBase 
*toolBase
, bool toggle 
)  
 539     wxToolBarTool 
*tool 
= (wxToolBarTool 
*)toolBase
; 
 541     GtkWidget 
*item 
= tool
->m_item
; 
 542     if ( item 
&& GTK_IS_TOGGLE_BUTTON(item
) ) 
 544         wxBitmap bitmap 
= tool
->GetBitmap(); 
 547             GtkPixmap 
*pixmap 
= GTK_PIXMAP( tool
->m_pixmap 
); 
 549             GdkBitmap 
*mask 
= bitmap
.GetMask() ? bitmap
.GetMask()->GetBitmap() 
 552             gtk_pixmap_set( pixmap
, bitmap
.GetPixmap(), mask 
); 
 557         gtk_toggle_button_set_state( GTK_TOGGLE_BUTTON(item
), toggle 
); 
 559         m_blockEvent 
= FALSE
; 
 563 void wxToolBar::DoSetToggle(wxToolBarToolBase 
* WXUNUSED(tool
), 
 564                             bool WXUNUSED(toggle
)) 
 566     // VZ: absolutely no idea about how to do it 
 567     wxFAIL_MSG( _T("not implemented") ); 
 570 // ---------------------------------------------------------------------------- 
 571 // wxToolBar geometry 
 572 // ---------------------------------------------------------------------------- 
 574 wxToolBarToolBase 
*wxToolBar::FindToolForPosition(wxCoord 
WXUNUSED(x
), 
 575                                                   wxCoord 
WXUNUSED(y
)) const 
 577     // VZ: GTK+ doesn't seem to have such thing 
 578     wxFAIL_MSG( _T("wxToolBar::FindToolForPosition() not implemented") ); 
 580     return (wxToolBarToolBase 
*)NULL
; 
 583 void wxToolBar::SetMargins( int x
, int y 
) 
 585     wxCHECK_RET( GetToolsCount() == 0, 
 586                  wxT("wxToolBar::SetMargins must be called before adding tools.") ); 
 588     if (x 
> 1) gtk_toolbar_append_space( m_toolbar 
);  // oh well 
 594 void wxToolBar::SetToolSeparation( int separation 
) 
 596     // FIXME: this function disappeared 
 598     gtk_toolbar_set_space_size( m_toolbar
, separation 
); 
 601     m_toolSeparation 
= separation
; 
 604 void wxToolBar::SetToolShortHelp( int id
, const wxString
& helpString 
) 
 606     wxToolBarTool 
*tool 
= (wxToolBarTool 
*)FindById(id
); 
 610         (void)tool
->SetShortHelp(helpString
); 
 611         gtk_tooltips_set_tip(m_toolbar
->tooltips
, tool
->m_item
, 
 612                              wxGTK_CONV( helpString 
), ""); 
 616 // ---------------------------------------------------------------------------- 
 617 // wxToolBar idle handling 
 618 // ---------------------------------------------------------------------------- 
 620 void wxToolBar::OnInternalIdle() 
 622     wxCursor cursor 
= m_cursor
; 
 623     if (g_globalCursor
.Ok()) cursor 
= g_globalCursor
; 
 627         /* I now set the cursor the anew in every OnInternalIdle call 
 628            as setting the cursor in a parent window also effects the 
 629            windows above so that checking for the current cursor is 
 632         if (HasFlag(wxTB_DOCKABLE
) && (m_widget
->window
)) 
 634             /* if the toolbar is dockable, then m_widget stands for the 
 635                GtkHandleBox widget, which uses its own window so that we 
 636                can set the cursor for it. if the toolbar is not dockable, 
 637                m_widget comes from m_toolbar which uses its parent's 
 638                window ("windowless windows") and thus we cannot set the 
 640             gdk_window_set_cursor( m_widget
->window
, cursor
.GetCursor() ); 
 643         wxToolBarToolsList::Node 
*node 
= m_tools
.GetFirst(); 
 646             wxToolBarTool 
*tool 
= (wxToolBarTool 
*)node
->GetData(); 
 647             node 
= node
->GetNext(); 
 649             GtkWidget 
*item 
= tool
->m_item
; 
 652                 GdkWindow 
*window 
= item
->window
; 
 656                     gdk_window_set_cursor( window
, cursor
.GetCursor() ); 
 665 #endif // wxUSE_TOOLBAR_NATIVE