1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/tbargtk.cpp 
   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 // ---------------------------------------------------------------------------- 
  19 // For compilers that support precompilation, includes "wx.h". 
  20 #include "wx/wxprec.h" 
  22 #if wxUSE_TOOLBAR_NATIVE 
  24 #include "wx/toolbar.h" 
  30 // FIXME: Use GtkImage instead of GtkPixmap. Use the new toolbar API for when gtk runtime is new enough? 
  31 // Beware that the new and old toolbar API may not be mixed in usage. 
  32 #include <gtk/gtkversion.h> 
  33 #ifdef GTK_DISABLE_DEPRECATED 
  34 #undef GTK_DISABLE_DEPRECATED 
  37 #include "wx/gtk/private.h" 
  39 // ---------------------------------------------------------------------------- 
  41 // ---------------------------------------------------------------------------- 
  44 extern bool       g_blockEventsOnDrag
; 
  45 extern wxCursor   g_globalCursor
; 
  47 // ---------------------------------------------------------------------------- 
  49 // ---------------------------------------------------------------------------- 
  51 // translate wxWidgets toolbar style flags to GTK orientation and style 
  52 static void GetGtkStyle(long style
, 
  53                         GtkOrientation 
*orient
, GtkToolbarStyle 
*gtkStyle
) 
  55     *orient 
= ( style 
& wxTB_LEFT 
|| style 
& wxTB_RIGHT 
) ? GTK_ORIENTATION_VERTICAL 
: GTK_ORIENTATION_HORIZONTAL
; 
  58     if ( style 
& wxTB_TEXT 
) 
  60         *gtkStyle 
= style 
& wxTB_NOICONS
 
  63                           style 
& wxTB_HORZ_LAYOUT 
? GTK_TOOLBAR_BOTH_HORIZ 
: 
  66     else // no text, hence we must have the icons or what would we show? 
  68         *gtkStyle 
= GTK_TOOLBAR_ICONS
; 
  72 // ---------------------------------------------------------------------------- 
  74 // ---------------------------------------------------------------------------- 
  76 class wxToolBarTool 
: public wxToolBarToolBase
 
  79     wxToolBarTool(wxToolBar 
*tbar
, 
  81                   const wxString
& label
, 
  82                   const wxBitmap
& bitmap1
, 
  83                   const wxBitmap
& bitmap2
, 
  86                   const wxString
& shortHelpString
, 
  87                   const wxString
& longHelpString
) 
  88         : wxToolBarToolBase(tbar
, id
, label
, bitmap1
, bitmap2
, kind
, 
  89                             clientData
, shortHelpString
, longHelpString
) 
  94     wxToolBarTool(wxToolBar 
*tbar
, wxControl 
*control
) 
  95         : wxToolBarToolBase(tbar
, control
) 
 100     // is this a radio button? 
 102     // unlike GetKind(), can be called for any kind of tools, not just buttons 
 103     bool IsRadio() const { return IsButton() && GetKind() == wxITEM_RADIO
; } 
 105     // this is only called for the normal buttons, i.e. not separators nor 
 107     GtkToolbarChildType 
GetGtkChildType() const 
 112                 return GTK_TOOLBAR_CHILD_TOGGLEBUTTON
; 
 115                 return GTK_TOOLBAR_CHILD_RADIOBUTTON
; 
 118                 wxFAIL_MSG( _T("unknown toolbar child type") ); 
 122                 return GTK_TOOLBAR_CHILD_BUTTON
; 
 126     void SetImage(const wxBitmap
& bitmap
) 
 130             // setting from pixmap doesn't seem to work right, but pixbuf works well 
 131             gtk_image_set_from_pixbuf((GtkImage
*)m_image
, bitmap
.GetPixbuf()); 
 142 // ---------------------------------------------------------------------------- 
 144 // ---------------------------------------------------------------------------- 
 146 IMPLEMENT_DYNAMIC_CLASS(wxToolBar
, wxControl
) 
 148 // ============================================================================ 
 150 // ============================================================================ 
 152 //----------------------------------------------------------------------------- 
 153 // "clicked" (internal from gtk_toolbar) 
 154 //----------------------------------------------------------------------------- 
 157 static void gtk_toolbar_callback( GtkWidget 
*WXUNUSED(widget
), 
 158                                   wxToolBarTool 
*tool 
) 
 161         wxapp_install_idle_handler(); 
 163     wxToolBar 
*tbar 
= (wxToolBar 
*)tool
->GetToolBar(); 
 165     if (tbar
->m_blockEvent
) return; 
 167     if (g_blockEventsOnDrag
) return; 
 168     if (!tool
->IsEnabled()) return; 
 170     if (tool
->CanBeToggled()) 
 174         tool
->SetImage(tool
->GetBitmap()); 
 176         if ( tool
->IsRadio() && !tool
->IsToggled() ) 
 178             // radio button went up, don't report this as a wxWin event 
 183     if( !tbar
->OnLeftClick( tool
->GetId(), tool
->IsToggled() ) && tool
->CanBeToggled() ) 
 188         tool
->SetImage(tool
->GetBitmap()); 
 193 //----------------------------------------------------------------------------- 
 194 // "enter_notify_event" / "leave_notify_event" 
 195 //----------------------------------------------------------------------------- 
 198 static gint 
gtk_toolbar_tool_callback( GtkWidget 
*WXUNUSED(widget
), 
 199                                        GdkEventCrossing 
*gdk_event
, 
 200                                        wxToolBarTool 
*tool 
) 
 202     // don't need to install idle handler, its done from "event" signal 
 204     if (g_blockEventsOnDrag
) return TRUE
; 
 206     wxToolBar 
*tb 
= (wxToolBar 
*)tool
->GetToolBar(); 
 209     if( gdk_event
->type 
== GDK_ENTER_NOTIFY 
) 
 210         tb
->OnMouseEnter( tool
->GetId() ); 
 212         tb
->OnMouseEnter( -1 ); 
 220 void gtktoolwidget_size_callback( GtkWidget 
*widget
, 
 221                                   GtkAllocation 
*alloc
, 
 224     // this shouldn't happen... 
 225     if (win
->GetParent()->m_wxwindow
) return; 
 227     wxSize size 
= win
->GetEffectiveMinSize(); 
 228     if (size
.y 
!= alloc
->height
) 
 230         GtkAllocation alloc2
; 
 232         alloc2
.y 
= (alloc
->height 
- size
.y 
+ 3) / 2; 
 233         alloc2
.width 
= alloc
->width
; 
 234         alloc2
.height 
= size
.y
; 
 235         gtk_widget_size_allocate( widget
, &alloc2 
); 
 239 //----------------------------------------------------------------------------- 
 240 // InsertChild callback for wxToolBar 
 241 //----------------------------------------------------------------------------- 
 243 static void wxInsertChildInToolBar( wxToolBar
* WXUNUSED(parent
), 
 244                                     wxWindow
* WXUNUSED(child
) ) 
 246     // we don't do anything here 
 249 // ---------------------------------------------------------------------------- 
 251 // ---------------------------------------------------------------------------- 
 253 void wxToolBarTool::Init() 
 259 wxToolBarToolBase 
*wxToolBar::CreateTool(int id
, 
 260                                          const wxString
& text
, 
 261                                          const wxBitmap
& bitmap1
, 
 262                                          const wxBitmap
& bitmap2
, 
 264                                          wxObject 
*clientData
, 
 265                                          const wxString
& shortHelpString
, 
 266                                          const wxString
& longHelpString
) 
 268     return new wxToolBarTool(this, id
, text
, bitmap1
, bitmap2
, kind
, 
 269                              clientData
, shortHelpString
, longHelpString
); 
 272 wxToolBarToolBase 
*wxToolBar::CreateTool(wxControl 
*control
) 
 274     return new wxToolBarTool(this, control
); 
 277 //----------------------------------------------------------------------------- 
 278 // wxToolBar construction 
 279 //----------------------------------------------------------------------------- 
 281 void wxToolBar::Init() 
 283     m_toolbar 
= (GtkToolbar 
*)NULL
; 
 284     m_blockEvent 
= false; 
 286     m_defaultHeight 
= 32; 
 289 wxToolBar::~wxToolBar() 
 293 bool wxToolBar::Create( wxWindow 
*parent
, 
 298                         const wxString
& name 
) 
 301     m_insertCallback 
= (wxInsertChildFunction
)wxInsertChildInToolBar
; 
 303     if ( !PreCreation( parent
, pos
, size 
) || 
 304          !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
 306         wxFAIL_MSG( wxT("wxToolBar creation failed") ); 
 313     m_toolbar 
= GTK_TOOLBAR( gtk_toolbar_new() ); 
 316     // Doesn't work this way. 
 317     // GtkToolbarSpaceStyle space_style = GTK_TOOLBAR_SPACE_EMPTY; 
 318     // gtk_widget_style_set (GTK_WIDGET (m_toolbar), "space_style", &space_style, NULL); 
 320     SetToolSeparation(7); 
 322     if (style 
& wxTB_DOCKABLE
) 
 324         m_widget 
= gtk_handle_box_new(); 
 325         gtk_container_add( GTK_CONTAINER(m_widget
), GTK_WIDGET(m_toolbar
) ); 
 326         gtk_widget_show( GTK_WIDGET(m_toolbar
) ); 
 328         if (style 
& wxTB_FLAT
) 
 329             gtk_handle_box_set_shadow_type( GTK_HANDLE_BOX(m_widget
), GTK_SHADOW_NONE 
); 
 333         m_widget 
= gtk_event_box_new(); 
 334         gtk_container_add( GTK_CONTAINER(m_widget
), GTK_WIDGET(m_toolbar
) ); 
 335         ConnectWidget( m_widget 
); 
 336         gtk_widget_show(GTK_WIDGET(m_toolbar
)); 
 339     // FIXME: there is no such function for toolbars in 2.0 
 341     if (style 
& wxTB_FLAT
) 
 342         gtk_toolbar_set_button_relief( GTK_TOOLBAR(m_toolbar
), GTK_RELIEF_NONE 
); 
 345     m_parent
->DoAddChild( this ); 
 352 GdkWindow 
*wxToolBar::GTKGetWindow(wxArrayGdkWindows
& windows
) const 
 354     return GTK_WIDGET(m_toolbar
)->window
; 
 357 void wxToolBar::GtkSetStyle() 
 359     GtkOrientation orient
; 
 360     GtkToolbarStyle style
; 
 361     GetGtkStyle(GetWindowStyle(), &orient
, &style
); 
 363     gtk_toolbar_set_orientation(m_toolbar
, orient
); 
 364     gtk_toolbar_set_style(m_toolbar
, style
); 
 365     gtk_toolbar_set_tooltips(m_toolbar
, !(style 
& wxTB_NO_TOOLTIPS
)); 
 368 void wxToolBar::SetWindowStyleFlag( long style 
) 
 370     wxToolBarBase::SetWindowStyleFlag(style
); 
 376 bool wxToolBar::DoInsertTool(size_t pos
, wxToolBarToolBase 
*toolBase
) 
 378     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, toolBase
); 
 380     if ( tool
->IsButton() ) 
 382         if ( !HasFlag(wxTB_NOICONS
) ) 
 384             wxBitmap bitmap 
= tool
->GetNormalBitmap(); 
 386             wxCHECK_MSG( bitmap
.Ok(), false, 
 387                          wxT("invalid bitmap for wxToolBar icon") ); 
 389             tool
->m_image 
= gtk_image_new(); 
 390             tool
->SetImage(bitmap
); 
 392             gtk_misc_set_alignment((GtkMisc
*)tool
->m_image
, 0.5, 0.5); 
 396     const int posGtk 
= int(pos
); 
 398     switch ( tool
->GetStyle() ) 
 400         case wxTOOL_STYLE_BUTTON
: 
 401             // for a radio button we need the widget which starts the radio 
 402             // group it belongs to, i.e. the first radio button immediately 
 403             // preceding this one 
 405                 GtkWidget 
*widget 
= NULL
; 
 407                 if ( tool
->IsRadio() ) 
 409                     wxToolBarToolsList::compatibility_iterator node
 
 410                         = wxToolBarToolsList::compatibility_iterator(); 
 412                         node 
= m_tools
.Item(pos 
- 1); 
 416                         wxToolBarTool 
*toolNext 
= (wxToolBarTool 
*)node
->GetData(); 
 417                         if ( !toolNext
->IsRadio() ) 
 420                         widget 
= toolNext
->m_item
; 
 422                         node 
= node
->GetPrevious(); 
 427                         // this is the first button in the radio button group, 
 428                         // it will be toggled automatically by GTK so bring the 
 429                         // internal flag in sync 
 434                 tool
->m_item 
= gtk_toolbar_insert_element
 
 437                                   tool
->GetGtkChildType(), 
 439                                   tool
->GetLabel().empty() 
 441                                     : (const char*) wxGTK_CONV( tool
->GetLabel() ), 
 442                                   tool
->GetShortHelp().empty() 
 444                                     : (const char*) wxGTK_CONV( tool
->GetShortHelp() ), 
 445                                   "", // tooltip_private_text (?) 
 447                                   (GtkSignalFunc
)gtk_toolbar_callback
, 
 452                 wxCHECK_MSG(tool
->m_item 
!= NULL
, false, _T("gtk_toolbar_insert_element() failed")); 
 454                 g_signal_connect (tool
->m_item
, "enter_notify_event", 
 455                                   G_CALLBACK (gtk_toolbar_tool_callback
), 
 457                 g_signal_connect (tool
->m_item
, "leave_notify_event", 
 458                                   G_CALLBACK (gtk_toolbar_tool_callback
), 
 463         case wxTOOL_STYLE_SEPARATOR
: 
 464             gtk_toolbar_insert_space( m_toolbar
, posGtk 
); 
 469         case wxTOOL_STYLE_CONTROL
: 
 470             gtk_toolbar_insert_widget( 
 472                                        tool
->GetControl()->m_widget
, 
 478             // connect after in order to correct size_allocate events 
 479             g_signal_connect_after (tool
->GetControl()->m_widget
, "size_allocate", 
 480                           G_CALLBACK (gtktoolwidget_size_callback
), tool
->GetControl()); 
 486     (* GTK_WIDGET_CLASS( GTK_OBJECT_GET_CLASS(m_widget
) )->size_request 
) 
 488     m_width 
= req
.width 
+ m_xMargin
; 
 489     m_height 
= req
.height 
+ 2*m_yMargin
; 
 490     InvalidateBestSize(); 
 495 bool wxToolBar::DoDeleteTool(size_t pos
, wxToolBarToolBase 
*toolBase
) 
 497     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, toolBase
); 
 499     switch ( tool
->GetStyle() ) 
 501         case wxTOOL_STYLE_CONTROL
: 
 502             tool
->GetControl()->Destroy(); 
 505         case wxTOOL_STYLE_BUTTON
: 
 506             gtk_widget_destroy( tool
->m_item 
); 
 509         case wxTOOL_STYLE_SEPARATOR
: 
 510             gtk_toolbar_remove_space( m_toolbar
, pos 
); 
 514     InvalidateBestSize(); 
 518 // ---------------------------------------------------------------------------- 
 519 // wxToolBar tools state 
 520 // ---------------------------------------------------------------------------- 
 522 void wxToolBar::DoEnableTool(wxToolBarToolBase 
*toolBase
, bool enable
) 
 524     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, toolBase
); 
 528         gtk_widget_set_sensitive( tool
->m_item
, enable 
); 
 532 void wxToolBar::DoToggleTool( wxToolBarToolBase 
*toolBase
, bool toggle 
) 
 534     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, toolBase
); 
 536     GtkWidget 
*item 
= tool
->m_item
; 
 537     if ( item 
&& GTK_IS_TOGGLE_BUTTON(item
) ) 
 539         tool
->SetImage(tool
->GetBitmap()); 
 543         gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(item
), toggle 
); 
 545         m_blockEvent 
= false; 
 549 void wxToolBar::DoSetToggle(wxToolBarToolBase 
* WXUNUSED(tool
), 
 550                             bool WXUNUSED(toggle
)) 
 552     // VZ: absolutely no idea about how to do it 
 553     wxFAIL_MSG( _T("not implemented") ); 
 556 // ---------------------------------------------------------------------------- 
 557 // wxToolBar geometry 
 558 // ---------------------------------------------------------------------------- 
 560 wxToolBarToolBase 
*wxToolBar::FindToolForPosition(wxCoord 
WXUNUSED(x
), 
 561                                                   wxCoord 
WXUNUSED(y
)) const 
 563     // VZ: GTK+ doesn't seem to have such thing 
 564     wxFAIL_MSG( _T("wxToolBar::FindToolForPosition() not implemented") ); 
 566     return (wxToolBarToolBase 
*)NULL
; 
 569 void wxToolBar::SetMargins( int x
, int y 
) 
 571     wxCHECK_RET( GetToolsCount() == 0, 
 572                  wxT("wxToolBar::SetMargins must be called before adding tools.") ); 
 578 void wxToolBar::SetToolSeparation( int separation 
) 
 580     // FIXME: this function disappeared 
 582     gtk_toolbar_set_space_size( m_toolbar
, separation 
); 
 585     m_toolSeparation 
= separation
; 
 588 void wxToolBar::SetToolShortHelp( int id
, const wxString
& helpString 
) 
 590     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, FindById(id
)); 
 594         (void)tool
->SetShortHelp(helpString
); 
 595         gtk_tooltips_set_tip(m_toolbar
->tooltips
, tool
->m_item
, 
 596                              wxGTK_CONV( helpString 
), ""); 
 600 void wxToolBar::SetToolNormalBitmap( int id
, const wxBitmap
& bitmap 
) 
 602     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, FindById(id
)); 
 605         wxCHECK_RET( tool
->IsButton(), wxT("Can only set bitmap on button tools.")); 
 607         tool
->SetNormalBitmap(bitmap
); 
 608         tool
->SetImage(tool
->GetBitmap()); 
 612 void wxToolBar::SetToolDisabledBitmap( int id
, const wxBitmap
& bitmap 
) 
 614     wxToolBarTool
* tool 
= wx_static_cast(wxToolBarTool
*, FindById(id
)); 
 617         wxCHECK_RET( tool
->IsButton(), wxT("Can only set bitmap on button tools.")); 
 619         tool
->SetDisabledBitmap(bitmap
); 
 620         tool
->SetImage(tool
->GetBitmap()); 
 624 // ---------------------------------------------------------------------------- 
 625 // wxToolBar idle handling 
 626 // ---------------------------------------------------------------------------- 
 628 void wxToolBar::OnInternalIdle() 
 630     // Check if we have to show window now 
 631     if (GtkShowFromOnIdle()) return; 
 633     wxCursor cursor 
= m_cursor
; 
 634     if (g_globalCursor
.Ok()) cursor 
= g_globalCursor
; 
 638         /* I now set the cursor the anew in every OnInternalIdle call 
 639            as setting the cursor in a parent window also effects the 
 640            windows above so that checking for the current cursor is 
 643         if (HasFlag(wxTB_DOCKABLE
) && (m_widget
->window
)) 
 645             /* if the toolbar is dockable, then m_widget stands for the 
 646                GtkHandleBox widget, which uses its own window so that we 
 647                can set the cursor for it. if the toolbar is not dockable, 
 648                m_widget comes from m_toolbar which uses its parent's 
 649                window ("windowless windows") and thus we cannot set the 
 651             gdk_window_set_cursor( m_widget
->window
, cursor
.GetCursor() ); 
 654         wxToolBarToolsList::compatibility_iterator node 
= m_tools
.GetFirst(); 
 657             wxToolBarTool 
*tool 
= (wxToolBarTool 
*)node
->GetData(); 
 658             node 
= node
->GetNext(); 
 660             GtkWidget 
*item 
= tool
->m_item
; 
 663                 GdkWindow 
*window 
= item
->window
; 
 667                     gdk_window_set_cursor( window
, cursor
.GetCursor() ); 
 673     if (wxUpdateUIEvent::CanUpdate(this)) 
 674         UpdateWindowUI(wxUPDATE_UI_FROMIDLE
); 
 678 // ---------------------------------------------------------------------------- 
 682 wxToolBar::GetClassDefaultAttributes(wxWindowVariant 
WXUNUSED(variant
)) 
 684     return GetDefaultAttributesFromGTKWidget(gtk_toolbar_new
); 
 687 #endif // wxUSE_TOOLBAR_NATIVE