1 ///////////////////////////////////////////////////////////////////////////// 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling, Vadim Zeitlin 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  11 #pragma implementation "notebook.h" 
  14 #include "wx/notebook.h" 
  20 #include "wx/imaglist.h" 
  26 #include "wx/gtk/win_gtk.h" 
  27 #include <gdk/gdkkeysyms.h> 
  29 //----------------------------------------------------------------------------- 
  31 //----------------------------------------------------------------------------- 
  33 extern void wxapp_install_idle_handler(); 
  36 //----------------------------------------------------------------------------- 
  38 //----------------------------------------------------------------------------- 
  40 extern bool   g_blockEventsOnDrag
; 
  42 //----------------------------------------------------------------------------- 
  44 //----------------------------------------------------------------------------- 
  48 extern void debug_focus_in( GtkWidget
* widget
, const wxChar
* name
, const wxChar 
*window 
); 
  52 //----------------------------------------------------------------------------- 
  54 //----------------------------------------------------------------------------- 
  56 class wxNotebookPage
: public wxObject
 
  63     m_page 
= (GtkNotebookPage 
*) NULL
; 
  64     m_client 
= (wxWindow 
*) NULL
; 
  65     m_box 
= (GtkWidget 
*) NULL
; 
  70   GtkNotebookPage   
*m_page
; 
  73   GtkWidget         
*m_box
;     // in which the label and image are packed 
  76 //----------------------------------------------------------------------------- 
  78 //----------------------------------------------------------------------------- 
  80 static void gtk_notebook_page_change_callback(GtkNotebook 
*WXUNUSED(widget
), 
  81                                               GtkNotebookPage 
*WXUNUSED(page
), 
  83                                               wxNotebook 
*notebook 
) 
  86         wxapp_install_idle_handler(); 
  88     int old 
= notebook
->GetSelection(); 
  90     wxNotebookEvent 
event1( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGING
, 
  91                             notebook
->GetId(), page
, old 
); 
  92     event1
.SetEventObject( notebook 
); 
  94     if ((notebook
->GetEventHandler()->ProcessEvent( event1 
)) && 
  97         /* program doesn't allow the page change */ 
  98         gtk_signal_emit_stop_by_name( GTK_OBJECT(notebook
->m_widget
), "switch_page" ); 
 102     wxNotebookEvent 
event2( wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED
, 
 103                             notebook
->GetId(), page
, old 
); 
 104     event2
.SetEventObject( notebook 
); 
 105     notebook
->GetEventHandler()->ProcessEvent( event2 
); 
 108 //----------------------------------------------------------------------------- 
 110 //----------------------------------------------------------------------------- 
 112 static void gtk_page_size_callback( GtkWidget 
*WXUNUSED(widget
), GtkAllocation
* alloc
, wxWindow 
*win 
) 
 115         wxapp_install_idle_handler(); 
 117     if ((win
->m_x 
== alloc
->x
) && 
 118         (win
->m_y 
== alloc
->y
) && 
 119         (win
->m_width 
== alloc
->width
) && 
 120         (win
->m_height 
== alloc
->height
)) 
 125     win
->SetSize( alloc
->x
, alloc
->y
, alloc
->width
, alloc
->height 
); 
 127     /* GTK 1.2 up to version 1.2.5 is broken so that we have to call allocate 
 128        here in order to make repositioning after resizing to take effect. */ 
 129     if ((gtk_major_version 
== 1) && 
 130         (gtk_minor_version 
== 2) && 
 131             (gtk_micro_version 
< 6) && 
 133             (GTK_WIDGET_REALIZED(win
->m_wxwindow
))) 
 135         gtk_widget_size_allocate( win
->m_wxwindow
, alloc 
); 
 139 //----------------------------------------------------------------------------- 
 140 // "realize" from m_widget 
 141 //----------------------------------------------------------------------------- 
 144 gtk_notebook_realized_callback( GtkWidget 
* WXUNUSED(widget
), wxWindow 
*win 
) 
 147         wxapp_install_idle_handler(); 
 149     /* GTK 1.2 up to version 1.2.5 is broken so that we have to call a queue_resize 
 150        here in order to make repositioning before showing to take effect. */ 
 151     gtk_widget_queue_resize( win
->m_widget 
); 
 156 //----------------------------------------------------------------------------- 
 157 // InsertChild callback for wxNotebook 
 158 //----------------------------------------------------------------------------- 
 160 static void wxInsertChildInNotebook( wxNotebook
* WXUNUSED(parent
), wxWindow
* WXUNUSED(child
) ) 
 162     /* we don't do anything here but pray */ 
 165 //----------------------------------------------------------------------------- 
 167 //----------------------------------------------------------------------------- 
 169 IMPLEMENT_DYNAMIC_CLASS(wxNotebook
,wxControl
) 
 171 BEGIN_EVENT_TABLE(wxNotebook
, wxControl
) 
 172     EVT_NAVIGATION_KEY(wxNotebook::OnNavigationKey
) 
 175 void wxNotebook::Init() 
 177     m_imageList 
= (wxImageList 
*) NULL
; 
 178     m_pages
.DeleteContents( TRUE 
); 
 179     m_lastSelection 
= -1; 
 182 wxNotebook::wxNotebook() 
 187 wxNotebook::wxNotebook( wxWindow 
*parent
, wxWindowID id
, 
 188       const wxPoint
& pos
, const wxSize
& size
, 
 189       long style
, const wxString
& name 
) 
 192     Create( parent
, id
, pos
, size
, style
, name 
); 
 195 wxNotebook::~wxNotebook() 
 197     /* don't generate change page events any more */ 
 198     gtk_signal_disconnect_by_func( GTK_OBJECT(m_widget
), 
 199       GTK_SIGNAL_FUNC(gtk_notebook_page_change_callback
), (gpointer
) this ); 
 204 bool wxNotebook::Create(wxWindow 
*parent
, wxWindowID id
, 
 205       const wxPoint
& pos
, const wxSize
& size
, 
 206       long style
, const wxString
& name 
) 
 209     m_acceptsFocus 
= TRUE
; 
 210     m_insertCallback 
= (wxInsertChildFunction
)wxInsertChildInNotebook
; 
 212     if (!PreCreation( parent
, pos
, size 
) || 
 213         !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
 215         wxFAIL_MSG( wxT("wxNoteBook creation failed") ); 
 220     m_widget 
= gtk_notebook_new(); 
 223     debug_focus_in( m_widget
, wxT("wxNotebook::m_widget"), name 
); 
 226     gtk_notebook_set_scrollable( GTK_NOTEBOOK(m_widget
), 1 ); 
 228     gtk_signal_connect( GTK_OBJECT(m_widget
), "switch_page", 
 229       GTK_SIGNAL_FUNC(gtk_notebook_page_change_callback
), (gpointer
)this ); 
 231     m_parent
->DoAddChild( this ); 
 233         if (m_windowStyle 
& wxNB_RIGHT
) 
 234                 gtk_notebook_set_tab_pos( GTK_NOTEBOOK(m_widget
), GTK_POS_RIGHT 
); 
 235         if (m_windowStyle 
& wxNB_LEFT
) 
 236                 gtk_notebook_set_tab_pos( GTK_NOTEBOOK(m_widget
), GTK_POS_LEFT 
); 
 237         if (m_windowStyle 
& wxNB_BOTTOM
) 
 238                 gtk_notebook_set_tab_pos( GTK_NOTEBOOK(m_widget
), GTK_POS_BOTTOM 
); 
 242     SetFont( parent
->GetFont() ); 
 244     gtk_signal_connect( GTK_OBJECT(m_widget
), "realize", 
 245                             GTK_SIGNAL_FUNC(gtk_notebook_realized_callback
), (gpointer
) this ); 
 252 void wxNotebook::SetFocus() 
 254     if (m_pages
.GetCount() == 0) return; 
 256     wxNode 
*node 
= m_pages
.Nth( GetSelection() ); 
 260     wxNotebookPage 
*page 
= (wxNotebookPage
*) node
->Data(); 
 262     page
->m_client
->SetFocus(); 
 265 int wxNotebook::GetSelection() const 
 267     wxCHECK_MSG( m_widget 
!= NULL
, -1, wxT("invalid notebook") ); 
 269     GList 
*pages 
= GTK_NOTEBOOK(m_widget
)->children
; 
 271     if (g_list_length(pages
) == 0) return -1; 
 273     GtkNotebook 
*notebook 
= GTK_NOTEBOOK(m_widget
); 
 275     if (notebook
->cur_page 
== NULL
) return m_lastSelection
; 
 277     return g_list_index( pages
, (gpointer
)(notebook
->cur_page
) ); 
 280 int wxNotebook::GetPageCount() const 
 282     return (int) g_list_length( GTK_NOTEBOOK(m_widget
)->children 
); 
 285 int wxNotebook::GetRowCount() const 
 290 wxString 
wxNotebook::GetPageText( int page 
) const 
 292     wxCHECK_MSG( m_widget 
!= NULL
, wxT(""), wxT("invalid notebook") ); 
 294     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 296         return nb_page
->m_text
; 
 301 int wxNotebook::GetPageImage( int page 
) const 
 303     wxCHECK_MSG( m_widget 
!= NULL
, -1, wxT("invalid notebook") ); 
 305     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 307         return nb_page
->m_image
; 
 312 wxNotebookPage
* wxNotebook::GetNotebookPage( int page 
) const 
 314     wxCHECK_MSG( m_widget 
!= NULL
, (wxNotebookPage
*) NULL
, wxT("invalid notebook") ); 
 316     wxCHECK_MSG( page 
< (int)m_pages
.GetCount(), (wxNotebookPage
*) NULL
, wxT("invalid notebook index") ); 
 318     wxNode 
*node 
= m_pages
.Nth( page 
); 
 320     return (wxNotebookPage 
*) node
->Data(); 
 323 int wxNotebook::SetSelection( int page 
) 
 325     wxCHECK_MSG( m_widget 
!= NULL
, -1, wxT("invalid notebook") ); 
 327     wxCHECK_MSG( page 
< (int)m_pages
.GetCount(), -1, wxT("invalid notebook index") ); 
 329     int selOld 
= GetSelection(); 
 331     gtk_notebook_set_page( GTK_NOTEBOOK(m_widget
), page 
); 
 336 void wxNotebook::AdvanceSelection( bool forward 
) 
 338     wxCHECK_RET( m_widget 
!= NULL
, wxT("invalid notebook") ); 
 340     int sel 
= GetSelection(); 
 341     int max 
= GetPageCount(); 
 344         SetSelection( sel 
== max 
? 0 : sel 
+ 1 ); 
 346         SetSelection( sel 
== 0 ? max
-1 : sel 
- 1 ); 
 349 void wxNotebook::SetImageList( wxImageList
* imageList 
) 
 351     m_imageList 
= imageList
; 
 354 bool wxNotebook::SetPageText( int page
, const wxString 
&text 
) 
 356     wxCHECK_MSG( m_widget 
!= NULL
, FALSE
, wxT("invalid notebook") ); 
 358     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 360     wxCHECK_MSG( nb_page
, FALSE
, wxT("SetPageText: invalid page index") ); 
 362     nb_page
->m_text 
= text
; 
 364     gtk_label_set( nb_page
->m_label
, nb_page
->m_text
.mbc_str() ); 
 369 bool wxNotebook::SetPageImage( int page
, int image 
) 
 371     /* HvdH 28-12-98: now it works, but it's a bit of a kludge */ 
 373     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 375     if (!nb_page
) return FALSE
; 
 377     /* Optimization posibility: return immediately if image unchanged. 
 378      * Not enabled because it may break existing (stupid) code that 
 379      * manipulates the imagelist to cycle images */ 
 381     /* if (image == nb_page->m_image) return TRUE; */ 
 383     /* For different cases: 
 384        1) no image -> no image 
 389     if (image 
== -1 && nb_page
->m_image 
== -1) 
 390         return TRUE
; /* Case 1): Nothing to do. */ 
 392     GtkWidget 
*pixmapwid 
= (GtkWidget
*) NULL
; 
 394     if (nb_page
->m_image 
!= -1) 
 396         /* Case 2) or 4). There is already an image in the gtkhbox. Let's find it */ 
 398         GList 
*child 
= gtk_container_children(GTK_CONTAINER(nb_page
->m_box
)); 
 401             if (GTK_IS_PIXMAP(child
->data
)) 
 403                 pixmapwid 
= GTK_WIDGET(child
->data
); 
 409         /* We should have the pixmap widget now */ 
 410         wxASSERT(pixmapwid 
!= NULL
); 
 414             /* If there's no new widget, just remove the old from the box */ 
 415             gtk_container_remove(GTK_CONTAINER(nb_page
->m_box
), pixmapwid
); 
 416             nb_page
->m_image 
= -1; 
 418             return TRUE
; /* Case 2) */ 
 422     /* Only cases 3) and 4) left */ 
 423     wxASSERT( m_imageList 
!= NULL 
); /* Just in case */ 
 425     /* Construct the new pixmap */ 
 426     const wxBitmap 
*bmp 
= m_imageList
->GetBitmap(image
); 
 427     GdkPixmap 
*pixmap 
= bmp
->GetPixmap(); 
 428     GdkBitmap 
*mask 
= (GdkBitmap
*) NULL
; 
 429     if ( bmp
->GetMask() ) 
 431         mask 
= bmp
->GetMask()->GetBitmap(); 
 434     if (pixmapwid 
== NULL
) 
 436         /* Case 3) No old pixmap. Create a new one and prepend it to the hbox */ 
 437         pixmapwid 
= gtk_pixmap_new (pixmap
, mask 
); 
 439         /* CHECKME: Are these pack flags okay? */ 
 440         gtk_box_pack_start(GTK_BOX(nb_page
->m_box
), pixmapwid
, FALSE
, FALSE
, 3); 
 441         gtk_widget_show(pixmapwid
); 
 445         /* Case 4) Simply replace the pixmap */ 
 446         gtk_pixmap_set(GTK_PIXMAP(pixmapwid
), pixmap
, mask
); 
 449     nb_page
->m_image 
= image
; 
 454 void wxNotebook::SetPageSize( const wxSize 
&WXUNUSED(size
) ) 
 456     wxFAIL_MSG( wxT("wxNotebook::SetPageSize not implemented") ); 
 459 void wxNotebook::SetPadding( const wxSize 
&WXUNUSED(padding
) ) 
 461     wxFAIL_MSG( wxT("wxNotebook::SetPadding not implemented") ); 
 464 void wxNotebook::SetTabSize(const wxSize
& WXUNUSED(sz
)) 
 466     wxFAIL_MSG( wxT("wxNotebook::SetTabSize not implemented") ); 
 469 bool wxNotebook::DeleteAllPages() 
 471     wxCHECK_MSG( m_widget 
!= NULL
, FALSE
, wxT("invalid notebook") ); 
 473     while (m_pages
.GetCount() > 0) 
 474         DeletePage( m_pages
.GetCount()-1 ); 
 479 bool wxNotebook::DeletePage( int page 
) 
 481     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 482     if (!nb_page
) return FALSE
; 
 484     /* GTK sets GtkNotebook.cur_page to NULL before sending 
 485        the switvh page event */ 
 486     m_lastSelection 
= GetSelection(); 
 488     nb_page
->m_client
->Destroy(); 
 489     m_pages
.DeleteObject( nb_page 
); 
 491     m_lastSelection 
= -1; 
 496 bool wxNotebook::RemovePage( int page 
) 
 498     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 500     if (!nb_page
) return FALSE
; 
 502     gtk_notebook_remove_page( GTK_NOTEBOOK(m_widget
), page 
); 
 504     m_pages
.DeleteObject( nb_page 
); 
 509 bool wxNotebook::InsertPage( int position
, wxWindow
* win
, const wxString
& text
, 
 510                              bool select
, int imageId 
) 
 512     wxCHECK_MSG( m_widget 
!= NULL
, FALSE
, wxT("invalid notebook") ); 
 514     wxCHECK_MSG( win
->GetParent() == this, FALSE
, 
 515                wxT("Can't add a page whose parent is not the notebook!") ); 
 517     /* don't receive switch page during addition */ 
 518     gtk_signal_disconnect_by_func( GTK_OBJECT(m_widget
), 
 519       GTK_SIGNAL_FUNC(gtk_notebook_page_change_callback
), (gpointer
) this ); 
 521     GtkNotebook 
*notebook 
= GTK_NOTEBOOK(m_widget
); 
 523     wxNotebookPage 
*page 
= new wxNotebookPage(); 
 526         m_pages
.Append( page 
); 
 528         m_pages
.Insert( m_pages
.Nth( position 
), page 
); 
 530     page
->m_client 
= win
; 
 532     page
->m_box 
= gtk_hbox_new( FALSE
, 0 ); 
 533     gtk_container_border_width( GTK_CONTAINER(page
->m_box
), 2 ); 
 535     gtk_signal_connect( GTK_OBJECT(win
->m_widget
), "size_allocate", 
 536       GTK_SIGNAL_FUNC(gtk_page_size_callback
), (gpointer
)win 
); 
 539         gtk_notebook_append_page( notebook
, win
->m_widget
, page
->m_box 
); 
 541         gtk_notebook_insert_page( notebook
, win
->m_widget
, page
->m_box
, position 
); 
 543     page
->m_page 
= (GtkNotebookPage
*) g_list_last(notebook
->children
)->data
; 
 545     /* set the label image */ 
 546     page
->m_image 
= imageId
; 
 550         wxASSERT( m_imageList 
!= NULL 
); 
 552         const wxBitmap 
*bmp 
= m_imageList
->GetBitmap(imageId
); 
 553         GdkPixmap 
*pixmap 
= bmp
->GetPixmap(); 
 554         GdkBitmap 
*mask 
= (GdkBitmap
*) NULL
; 
 555         if ( bmp
->GetMask() ) 
 557             mask 
= bmp
->GetMask()->GetBitmap(); 
 560         GtkWidget 
*pixmapwid 
= gtk_pixmap_new (pixmap
, mask 
); 
 562         gtk_box_pack_start(GTK_BOX(page
->m_box
), pixmapwid
, FALSE
, FALSE
, 3); 
 564         gtk_widget_show(pixmapwid
); 
 567     /* set the label text */ 
 569     if (page
->m_text
.IsEmpty()) page
->m_text 
= wxT(""); 
 571     page
->m_label 
= GTK_LABEL( gtk_label_new(page
->m_text
.mbc_str()) ); 
 572     gtk_box_pack_end( GTK_BOX(page
->m_box
), GTK_WIDGET(page
->m_label
), FALSE
, FALSE
, 3 ); 
 575     gtk_widget_show( GTK_WIDGET(page
->m_label
) ); 
 577     if (select 
&& (m_pages
.GetCount() > 1)) 
 580             SetSelection( GetPageCount()-1 ); 
 582             SetSelection( position 
); 
 585     gtk_signal_connect( GTK_OBJECT(m_widget
), "switch_page", 
 586       GTK_SIGNAL_FUNC(gtk_notebook_page_change_callback
), (gpointer
)this ); 
 591 bool wxNotebook::AddPage(wxWindow
* win
, const wxString
& text
, 
 592                          bool select
, int imageId
) 
 594     return InsertPage( -1, win
, text
, select
, imageId 
); 
 597 void wxNotebook::OnNavigationKey(wxNavigationKeyEvent
& event
) 
 599     if (event
.IsWindowChange()) 
 600         AdvanceSelection( event
.GetDirection() ); 
 605 wxWindow 
*wxNotebook::GetPage( int page 
) const 
 607     wxCHECK_MSG( m_widget 
!= NULL
, (wxWindow
*) NULL
, wxT("invalid notebook") ); 
 609     wxNotebookPage
* nb_page 
= GetNotebookPage(page
); 
 611         return (wxWindow 
*) NULL
; 
 613         return nb_page
->m_client
; 
 616 // override these 2 functions to do nothing: everything is done in OnSize 
 617 void wxNotebook::SetConstraintSizes( bool WXUNUSED(recurse
) ) 
 619     // don't set the sizes of the pages - their correct size is not yet known 
 620     wxControl::SetConstraintSizes(FALSE
); 
 623 bool wxNotebook::DoPhase( int WXUNUSED(nPhase
) ) 
 628 void wxNotebook::ApplyWidgetStyle() 
 630     // TODO, font for labels etc 
 633     gtk_widget_set_style( m_widget
, m_widgetStyle 
); 
 636 bool wxNotebook::IsOwnGtkWindow( GdkWindow 
*window 
) 
 638     return ((m_widget
->window 
== window
) || 
 639             (GTK_NOTEBOOK(m_widget
)->panel 
== window
)); 
 642 //----------------------------------------------------------------------------- 
 644 //----------------------------------------------------------------------------- 
 646 IMPLEMENT_DYNAMIC_CLASS(wxNotebookEvent
, wxNotifyEvent
)