X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/28e88942bc2c9db6ae9a8ef59545d446eefea5fb..a7132f4b4b2bbef6d0f0c41c690909178d2c168f:/src/gtk/notebook.cpp diff --git a/src/gtk/notebook.cpp b/src/gtk/notebook.cpp index 14ce6bfac3..13bc0dae9d 100644 --- a/src/gtk/notebook.cpp +++ b/src/gtk/notebook.cpp @@ -26,22 +26,6 @@ #include "wx/fontutil.h" #include "wx/gtk/private.h" -#include "wx/gtk/win_gtk.h" - -#include - -// ---------------------------------------------------------------------------- -// events -// ---------------------------------------------------------------------------- - -DEFINE_EVENT_TYPE(wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGED) -DEFINE_EVENT_TYPE(wxEVT_COMMAND_NOTEBOOK_PAGE_CHANGING) - -//----------------------------------------------------------------------------- -// data -//----------------------------------------------------------------------------- - -extern bool g_blockEventsOnDrag; //----------------------------------------------------------------------------- // wxGtkNotebookPage @@ -57,108 +41,62 @@ extern bool g_blockEventsOnDrag; class wxGtkNotebookPage: public wxObject { public: - wxGtkNotebookPage() - { - m_image = -1; - m_page = (GtkNotebookPage *) NULL; - m_box = (GtkWidget *) NULL; - } - - wxString m_text; - int m_image; - GtkNotebookPage *m_page; - GtkLabel *m_label; - GtkWidget *m_box; // in which the label and image are packed + GtkWidget* m_box; + GtkWidget* m_label; + GtkWidget* m_image; + int m_imageIndex; }; #include "wx/listimpl.cpp" WX_DEFINE_LIST(wxGtkNotebookPagesList) +extern "C" { +static void event_after(GtkNotebook*, GdkEvent*, wxNotebook*); +} //----------------------------------------------------------------------------- // "switch_page" //----------------------------------------------------------------------------- extern "C" { -static void gtk_notebook_page_changing_callback( GtkNotebook *widget, - GtkNotebookPage *WXUNUSED(gpage), - guint page, - wxNotebook *notebook ) -{ - int old = gtk_notebook_get_current_page( widget ); - - if ( !notebook->SendPageChangingEvent(page) ) - { - // program doesn't allow the page change - g_signal_stop_emission_by_name(notebook->m_widget, "switch_page"); - } - else - { - // the page change event also reports the old page - notebook->m_oldSelection = old; - } -} -} - -extern "C" { -static void gtk_notebook_page_changed_callback( GtkNotebook *widget, - GtkNotebookPage *WXUNUSED(gpage), - guint page, - wxNotebook *notebook ) +static void +switch_page_after(GtkWidget* widget, GtkNotebookPage*, guint, wxNotebook* win) { - int old = notebook->m_oldSelection; - notebook->SendPageChangedEvent( old ); + g_signal_handlers_block_by_func(widget, (void*)switch_page_after, win); + win->SendPageChangedEvent(win->m_oldSelection); } } -//----------------------------------------------------------------------------- -// "size_allocate" -//----------------------------------------------------------------------------- - extern "C" { -static void gtk_page_size_callback( GtkWidget *WXUNUSED(widget), GtkAllocation* alloc, wxWindow *win ) +static void +switch_page(GtkNotebook* widget, GtkNotebookPage*, int page, wxNotebook* win) { - if (g_isIdle) - wxapp_install_idle_handler(); + win->m_oldSelection = gtk_notebook_get_current_page(widget); - if ((win->m_x == alloc->x) && - (win->m_y == alloc->y) && - (win->m_width == alloc->width) && - (win->m_height == alloc->height)) - { - return; - } - - win->SetSize( alloc->x, alloc->y, alloc->width, alloc->height ); - - /* GTK 1.2 up to version 1.2.5 is broken so that we have to call allocate - here in order to make repositioning after resizing to take effect. */ - if ((gtk_major_version == 1) && - (gtk_minor_version == 2) && - (gtk_micro_version < 6) && - (win->m_wxwindow) && - (GTK_WIDGET_REALIZED(win->m_wxwindow))) - { - gtk_widget_size_allocate( win->m_wxwindow, alloc ); - } + if (win->SendPageChangingEvent(page)) + // allow change, unblock handler for changed event + g_signal_handlers_unblock_by_func(widget, (void*)switch_page_after, win); + else + // change vetoed, unblock handler to set selection back + g_signal_handlers_unblock_by_func(widget, (void*)event_after, win); } } //----------------------------------------------------------------------------- -// "realize" from m_widget +// "event_after" from m_widget //----------------------------------------------------------------------------- extern "C" { -static void -gtk_notebook_realized_callback( GtkWidget * WXUNUSED(widget), wxWindow *win ) +static void event_after(GtkNotebook* widget, GdkEvent*, wxNotebook* win) { - if (g_isIdle) - wxapp_install_idle_handler(); + g_signal_handlers_block_by_func(widget, (void*)event_after, win); + g_signal_handlers_block_by_func(widget, (void*)switch_page, win); - /* GTK 1.2 up to version 1.2.5 is broken so that we have to call a queue_resize - here in order to make repositioning before showing to take effect. */ - gtk_widget_queue_resize( win->m_widget ); + // restore previous selection + gtk_notebook_set_current_page(widget, win->m_oldSelection); + + g_signal_handlers_unblock_by_func(widget, (void*)switch_page, win); } } @@ -166,7 +104,7 @@ gtk_notebook_realized_callback( GtkWidget * WXUNUSED(widget), wxWindow *win ) // InsertChild callback for wxNotebook //----------------------------------------------------------------------------- -static void wxInsertChildInNotebook( wxNotebook* parent, wxWindow* child ) +void wxNotebook::AddChildGTK(wxWindowGTK* child) { // Hack Alert! (Part I): This sets the notebook as the parent of the child // widget, and takes care of some details such as updating the state and @@ -176,7 +114,7 @@ static void wxInsertChildInNotebook( wxNotebook* parent, wxWindow* child ) // incorrect sizes since the widget's style context is not fully known. // See bug #901694 for details // (http://sourceforge.net/tracker/?func=detail&aid=901694&group_id=9863&atid=109863) - gtk_widget_set_parent(child->m_widget, parent->m_widget); + gtk_widget_set_parent(child->m_widget, m_widget); // NOTE: This should be considered a temporary workaround until we can // work out the details and implement delaying the setting of the initial @@ -187,17 +125,15 @@ static void wxInsertChildInNotebook( wxNotebook* parent, wxWindow* child ) // wxNotebook //----------------------------------------------------------------------------- -IMPLEMENT_DYNAMIC_CLASS(wxNotebook,wxControl) +IMPLEMENT_DYNAMIC_CLASS(wxNotebook,wxBookCtrlBase) -BEGIN_EVENT_TABLE(wxNotebook, wxControl) +BEGIN_EVENT_TABLE(wxNotebook, wxBookCtrlBase) EVT_NAVIGATION_KEY(wxNotebook::OnNavigationKey) END_EVENT_TABLE() void wxNotebook::Init() { m_padding = 0; - - m_imageList = (wxImageList *) NULL; m_oldSelection = -1; m_themeEnabled = true; } @@ -224,9 +160,6 @@ bool wxNotebook::Create(wxWindow *parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name ) { - m_needParent = true; - m_insertCallback = (wxInsertChildFunction)wxInsertChildInNotebook; - if ( (style & wxBK_ALIGN_MASK) == wxBK_DEFAULT ) style |= wxBK_TOP; @@ -239,14 +172,19 @@ bool wxNotebook::Create(wxWindow *parent, wxWindowID id, m_widget = gtk_notebook_new(); + g_object_ref(m_widget); gtk_notebook_set_scrollable( GTK_NOTEBOOK(m_widget), 1 ); g_signal_connect (m_widget, "switch_page", - G_CALLBACK (gtk_notebook_page_changing_callback), this); + G_CALLBACK(switch_page), this); g_signal_connect_after (m_widget, "switch_page", - G_CALLBACK (gtk_notebook_page_changed_callback), this); + G_CALLBACK(switch_page_after), this); + g_signal_handlers_block_by_func(m_widget, (void*)switch_page_after, this); + + g_signal_connect(m_widget, "event_after", G_CALLBACK(event_after), this); + g_signal_handlers_block_by_func(m_widget, (void*)event_after, this); m_parent->DoAddChild( this ); @@ -259,9 +197,6 @@ bool wxNotebook::Create(wxWindow *parent, wxWindowID id, PostCreation(size); - g_signal_connect (m_widget, "realize", - G_CALLBACK (gtk_notebook_realized_callback), this); - return true; } @@ -274,65 +209,40 @@ int wxNotebook::GetSelection() const wxString wxNotebook::GetPageText( size_t page ) const { - wxCHECK_MSG( m_widget != NULL, wxEmptyString, wxT("invalid notebook") ); + wxCHECK_MSG(page < GetPageCount(), wxEmptyString, "invalid notebook index"); - wxGtkNotebookPage* nb_page = GetNotebookPage(page); - if (nb_page) - return nb_page->m_text; - else - return wxEmptyString; + GtkLabel* label = GTK_LABEL(GetNotebookPage(page)->m_label); + return wxGTK_CONV_BACK(gtk_label_get_text(label)); } int wxNotebook::GetPageImage( size_t page ) const { - wxCHECK_MSG( m_widget != NULL, -1, wxT("invalid notebook") ); + wxCHECK_MSG(page < GetPageCount(), -1, "invalid notebook index"); - wxGtkNotebookPage* nb_page = GetNotebookPage(page); - if (nb_page) - return nb_page->m_image; - else - return -1; + return GetNotebookPage(page)->m_imageIndex; } wxGtkNotebookPage* wxNotebook::GetNotebookPage( int page ) const { - wxCHECK_MSG( m_widget != NULL, (wxGtkNotebookPage*) NULL, wxT("invalid notebook") ); - - wxCHECK_MSG( page < (int)m_pagesData.GetCount(), (wxGtkNotebookPage*) NULL, wxT("invalid notebook index") ); - return m_pagesData.Item(page)->GetData(); } int wxNotebook::DoSetSelection( size_t page, int flags ) { - wxCHECK_MSG( m_widget != NULL, -1, wxT("invalid notebook") ); - - wxCHECK_MSG( page < m_pagesData.GetCount(), -1, wxT("invalid notebook index") ); + wxCHECK_MSG(page < GetPageCount(), -1, "invalid notebook index"); int selOld = GetSelection(); if ( !(flags & SetSelection_SendEvent) ) { - g_signal_handlers_disconnect_by_func (m_widget, - (gpointer) gtk_notebook_page_changing_callback, - this); - - g_signal_handlers_disconnect_by_func (m_widget, - (gpointer) gtk_notebook_page_changed_callback, - this); + g_signal_handlers_block_by_func(m_widget, (void*)switch_page, this); } gtk_notebook_set_current_page( GTK_NOTEBOOK(m_widget), page ); if ( !(flags & SetSelection_SendEvent) ) { - // reconnect to signals - - g_signal_connect (m_widget, "switch_page", - G_CALLBACK (gtk_notebook_page_changing_callback), this); - - g_signal_connect_after (m_widget, "switch_page", - G_CALLBACK (gtk_notebook_page_changed_callback), this); + g_signal_handlers_unblock_by_func(m_widget, (void*)switch_page, this); } wxNotebookPage *client = GetPage(page); @@ -344,94 +254,44 @@ int wxNotebook::DoSetSelection( size_t page, int flags ) bool wxNotebook::SetPageText( size_t page, const wxString &text ) { - wxCHECK_MSG( m_widget != NULL, false, wxT("invalid notebook") ); + wxCHECK_MSG(page < GetPageCount(), false, "invalid notebook index"); - wxGtkNotebookPage* nb_page = GetNotebookPage(page); - - wxCHECK_MSG( nb_page, false, wxT("SetPageText: invalid page index") ); - - nb_page->m_text = text; - - gtk_label_set_text( nb_page->m_label, wxGTK_CONV( nb_page->m_text ) ); + GtkLabel* label = GTK_LABEL(GetNotebookPage(page)->m_label); + gtk_label_set_text(label, wxGTK_CONV(text)); return true; } bool wxNotebook::SetPageImage( size_t page, int image ) { - /* HvdH 28-12-98: now it works, but it's a bit of a kludge */ - - wxGtkNotebookPage* nb_page = GetNotebookPage(page); - - if (!nb_page) return false; - - /* Optimization posibility: return immediately if image unchanged. - * Not enabled because it may break existing (stupid) code that - * manipulates the imagelist to cycle images */ - - /* if (image == nb_page->m_image) return true; */ + wxCHECK_MSG(page < GetPageCount(), false, "invalid notebook index"); - /* For different cases: - 1) no image -> no image - 2) image -> no image - 3) no image -> image - 4) image -> image */ - - if (image == -1 && nb_page->m_image == -1) - return true; /* Case 1): Nothing to do. */ - - GtkWidget *pixmapwid = (GtkWidget*) NULL; - - if (nb_page->m_image != -1) + wxGtkNotebookPage* pageData = GetNotebookPage(page); + if (image >= 0) { - /* Case 2) or 4). There is already an image in the gtkhbox. Let's find it */ - - GList *child = gtk_container_get_children(GTK_CONTAINER(nb_page->m_box)); - while (child) + wxCHECK_MSG(m_imageList, false, "invalid notebook imagelist"); + const wxBitmap* bitmap = m_imageList->GetBitmapPtr(image); + if (bitmap == NULL) + return false; + if (pageData->m_image) { - if (GTK_IS_IMAGE(child->data)) - { - pixmapwid = GTK_WIDGET(child->data); - break; - } - child = child->next; + gtk_image_set_from_pixbuf( + GTK_IMAGE(pageData->m_image), bitmap->GetPixbuf()); } - - /* We should have the pixmap widget now */ - wxASSERT(pixmapwid != NULL); - - if (image == -1) + else { - /* If there's no new widget, just remove the old from the box */ - gtk_container_remove(GTK_CONTAINER(nb_page->m_box), pixmapwid); - nb_page->m_image = -1; - - return true; /* Case 2) */ + pageData->m_image = gtk_image_new_from_pixbuf(bitmap->GetPixbuf()); + gtk_widget_show(pageData->m_image); + gtk_box_pack_start(GTK_BOX(pageData->m_box), + pageData->m_image, false, false, m_padding); } } - - /* Only cases 3) and 4) left */ - wxASSERT( m_imageList != NULL ); /* Just in case */ - - /* Construct the new pixmap */ - const wxBitmap *bmp = m_imageList->GetBitmapPtr(image); - - if (pixmapwid == NULL) + else if (pageData->m_image) { - /* Case 3) No old pixmap. Create a new one and prepend it to the hbox */ - pixmapwid = gtk_image_new_from_pixbuf(bmp->GetPixbuf()); - - /* CHECKME: Are these pack flags okay? */ - gtk_box_pack_start(GTK_BOX(nb_page->m_box), pixmapwid, FALSE, FALSE, m_padding); - gtk_widget_show(pixmapwid); - } - else - { - /* Case 4) Simply replace the pixmap */ - gtk_image_set_from_pixbuf((GtkImage*)pixmapwid, bmp->GetPixbuf()); + gtk_widget_destroy(pageData->m_image); + pageData->m_image = NULL; } - - nb_page->m_image = image; + pageData->m_imageIndex = image; return true; } @@ -447,24 +307,16 @@ void wxNotebook::SetPadding( const wxSize &padding ) m_padding = padding.GetWidth(); - int i; - for (i=0; im_image != -1) + wxGtkNotebookPage* pageData = GetNotebookPage(i); + if (pageData->m_image) { - // gtk_box_set_child_packing sets padding on BOTH sides - // icon provides left padding, label provides center and right - int image = nb_page->m_image; - SetPageImage(i,-1); - SetPageImage(i,image); + gtk_box_set_child_packing(GTK_BOX(pageData->m_box), + pageData->m_image, false, false, m_padding, GTK_PACK_START); } - wxASSERT(nb_page->m_label); - gtk_box_set_child_packing(GTK_BOX(nb_page->m_box), - GTK_WIDGET(nb_page->m_label), - FALSE, FALSE, m_padding, GTK_PACK_END); + gtk_box_set_child_packing(GTK_BOX(pageData->m_box), + pageData->m_label, false, false, m_padding, GTK_PACK_END); } } @@ -475,35 +327,37 @@ void wxNotebook::SetTabSize(const wxSize& WXUNUSED(sz)) bool wxNotebook::DeleteAllPages() { - wxCHECK_MSG( m_widget != NULL, false, wxT("invalid notebook") ); + for (size_t i = GetPageCount(); i--;) + DeletePage(i); - while (m_pagesData.GetCount() > 0) - DeletePage( m_pagesData.GetCount()-1 ); - - wxASSERT_MSG( GetPageCount() == 0, _T("all pages must have been deleted") ); - - InvalidateBestSize(); return wxNotebookBase::DeleteAllPages(); } wxNotebookPage *wxNotebook::DoRemovePage( size_t page ) { - wxNotebookPage *client = wxNotebookBase::DoRemovePage(page); + // We cannot remove the page yet, as GTK sends the "switch_page" + // signal before it has removed the notebook-page from its + // corresponding list. Thus, if we were to remove the page from + // m_pages at this point, the two lists of pages would be out + // of sync during the PAGE_CHANGING/PAGE_CHANGED events. + wxNotebookPage *client = GetPage(page); if ( !client ) return NULL; - gtk_widget_ref( client->m_widget ); gtk_widget_unrealize( client->m_widget ); // we don't need to unparent the client->m_widget; GTK+ will do // that for us (and will throw a warning if we do it!) - gtk_notebook_remove_page( GTK_NOTEBOOK(m_widget), page ); + // It's safe to remove the page now. + wxASSERT_MSG(GetPage(page) == client, wxT("pages changed during delete")); + wxNotebookBase::DoRemovePage(page); + wxGtkNotebookPage* p = GetNotebookPage(page); m_pagesData.DeleteObject(p); delete p; - + return client; } @@ -521,70 +375,61 @@ bool wxNotebook::InsertPage( size_t position, wxCHECK_MSG( position <= GetPageCount(), false, _T("invalid page index in wxNotebookPage::InsertPage()") ); - // Hack Alert! (Part II): See above in wxInsertChildInNotebook callback - // why this has to be done. NOTE: using gtk_widget_unparent here does not - // work as it seems to undo too much and will cause errors in the - // gtk_notebook_insert_page below, so instead just clear the parent by - // hand here. - win->m_widget->parent = NULL; + // Hack Alert! (Part II): See above in wxNotebook::AddChildGTK + // why this has to be done. + gtk_widget_unparent(win->m_widget); if (m_themeEnabled) win->SetThemeEnabled(true); GtkNotebook *notebook = GTK_NOTEBOOK(m_widget); - wxGtkNotebookPage *nb_page = new wxGtkNotebookPage(); - - if ( position == GetPageCount() ) - m_pagesData.Append( nb_page ); - else - m_pagesData.Insert( position, nb_page ); + wxGtkNotebookPage* pageData = new wxGtkNotebookPage; m_pages.Insert(win, position); + m_pagesData.Insert(position, pageData); - nb_page->m_box = gtk_hbox_new( FALSE, 1 ); - gtk_container_set_border_width((GtkContainer*)nb_page->m_box, 2); - - g_signal_connect (win->m_widget, "size_allocate", - G_CALLBACK (gtk_page_size_callback), win); + // set the label image and text + // this must be done before adding the page, as GetPageText + // and GetPageImage will otherwise return wrong values in + // the page-changed event that results from inserting the + // first page. + pageData->m_imageIndex = imageId; - gtk_notebook_insert_page( notebook, win->m_widget, nb_page->m_box, position ); - - nb_page->m_page = (GtkNotebookPage*) g_list_last(notebook->children)->data; - - /* set the label image */ - nb_page->m_image = imageId; + pageData->m_box = gtk_hbox_new(false, 1); + gtk_container_set_border_width(GTK_CONTAINER(pageData->m_box), 2); + pageData->m_image = NULL; if (imageId != -1) { - wxASSERT( m_imageList != NULL ); - - const wxBitmap *bmp = m_imageList->GetBitmapPtr(imageId); - GtkWidget* pixmapwid = gtk_image_new_from_pixbuf(bmp->GetPixbuf()); - gtk_box_pack_start(GTK_BOX(nb_page->m_box), pixmapwid, FALSE, FALSE, m_padding); - gtk_widget_show(pixmapwid); + if (m_imageList) + { + const wxBitmap* bitmap = m_imageList->GetBitmapPtr(imageId); + pageData->m_image = gtk_image_new_from_pixbuf(bitmap->GetPixbuf()); + gtk_box_pack_start(GTK_BOX(pageData->m_box), + pageData->m_image, false, false, m_padding); + } + else + wxFAIL_MSG("invalid notebook imagelist"); } /* set the label text */ + pageData->m_label = gtk_label_new(wxGTK_CONV(wxStripMenuCodes(text))); + gtk_box_pack_end(GTK_BOX(pageData->m_box), + pageData->m_label, false, false, m_padding); - nb_page->m_text = wxStripMenuCodes(text); - if (nb_page->m_text.empty()) nb_page->m_text = wxEmptyString; - - nb_page->m_label = GTK_LABEL( gtk_label_new(wxGTK_CONV(nb_page->m_text)) ); - gtk_box_pack_end( GTK_BOX(nb_page->m_box), GTK_WIDGET(nb_page->m_label), FALSE, FALSE, m_padding ); + gtk_widget_show_all(pageData->m_box); + gtk_notebook_insert_page(notebook, win->m_widget, pageData->m_box, position); /* apply current style */ GtkRcStyle *style = CreateWidgetStyle(); if ( style ) { - gtk_widget_modify_style(GTK_WIDGET(nb_page->m_label), style); + gtk_widget_modify_style(pageData->m_label, style); gtk_rc_style_unref(style); } - /* show the label */ - gtk_widget_show( GTK_WIDGET(nb_page->m_label) ); - - if (select && (m_pagesData.GetCount() > 1)) + if (select && GetPageCount() > 1) { SetSelection( position ); } @@ -621,8 +466,8 @@ int wxNotebook::HitTest(const wxPoint& pt, long *flags) const for ( ; i < count; i++ ) { - wxGtkNotebookPage* nb_page = GetNotebookPage(i); - GtkWidget *box = nb_page->m_box; + wxGtkNotebookPage* pageData = GetNotebookPage(i); + GtkWidget* box = pageData->m_box; const gint border = gtk_container_get_border_width(GTK_CONTAINER(box)); @@ -631,26 +476,11 @@ int wxNotebook::HitTest(const wxPoint& pt, long *flags) const // ok, we're inside this tab -- now find out where, if needed if ( flags ) { - GtkWidget *pixmap = NULL; - - GList *children = gtk_container_get_children(GTK_CONTAINER(box)); - for ( GList *child = children; child; child = child->next ) - { - if (GTK_IS_IMAGE(child->data)) - { - pixmap = GTK_WIDGET(child->data); - break; - } - } - - if ( children ) - g_list_free(children); - - if ( pixmap && IsPointInsideWidget(pt, pixmap, x, y) ) + if (pageData->m_image && IsPointInsideWidget(pt, pageData->m_image, x, y)) { *flags = wxBK_HITTEST_ONICON; } - else if ( IsPointInsideWidget(pt, GTK_WIDGET(nb_page->m_label), x, y) ) + else if (IsPointInsideWidget(pt, pageData->m_label, x, y)) { *flags = wxBK_HITTEST_ONLABEL; } @@ -712,9 +542,8 @@ bool wxNotebook::DoPhase( int WXUNUSED(nPhase) ) void wxNotebook::DoApplyWidgetStyle(GtkRcStyle *style) { gtk_widget_modify_style(m_widget, style); - size_t cnt = m_pagesData.GetCount(); - for (size_t i = 0; i < cnt; i++) - gtk_widget_modify_style(GTK_WIDGET(GetNotebookPage(i)->m_label), style); + for (size_t i = GetPageCount(); i--;) + gtk_widget_modify_style(GetNotebookPage(i)->m_label, style); } GdkWindow *wxNotebook::GTKGetWindow(wxArrayGdkWindows& windows) const @@ -732,10 +561,4 @@ wxNotebook::GetClassDefaultAttributes(wxWindowVariant WXUNUSED(variant)) return GetDefaultAttributesFromGTKWidget(gtk_notebook_new); } -//----------------------------------------------------------------------------- -// wxNotebookEvent -//----------------------------------------------------------------------------- - -IMPLEMENT_DYNAMIC_CLASS(wxNotebookEvent, wxNotifyEvent) - #endif