+/*
+ Test for menu swapping bug (?) with GTK 2.1.13
+*/
+
+
+
/////////////////////////////////////////////////////////////////////////////
// Name: minimal.cpp
// Purpose: Minimal wxWindows sample
#include "wx/wx.h"
#endif
+
+// ----------------------------------------------------------------------------
+// Global variables
+// ----------------------------------------------------------------------------
+static wxMenuBar *bar[2];
+static int current_bar;
+
// ----------------------------------------------------------------------------
// ressources
// ----------------------------------------------------------------------------
// event handlers (these functions should _not_ be virtual)
void OnQuit(wxCommandEvent& event);
void OnAbout(wxCommandEvent& event);
+ void OnSwapMenus(wxCommandEvent& event);
+ void OnReplaceMenu(wxCommandEvent& event);
+ void OnRemoveInsertMenu(wxCommandEvent& event);
private:
// any class wishing to process wxWindows events must use this macro
{
// menu items
Minimal_Quit = 1,
- Minimal_About
+ Minimal_About,
+ Minimal_SwapMenus,
+ Minimal_ReplaceMenu,
+ Minimal_RemoveInsertMenu
};
// ----------------------------------------------------------------------------
BEGIN_EVENT_TABLE(MyFrame, wxFrame)
EVT_MENU(Minimal_Quit, MyFrame::OnQuit)
EVT_MENU(Minimal_About, MyFrame::OnAbout)
+ EVT_MENU(Minimal_SwapMenus, MyFrame::OnSwapMenus)
+ EVT_MENU(Minimal_ReplaceMenu, MyFrame::OnReplaceMenu)
+ EVT_MENU(Minimal_RemoveInsertMenu, MyFrame::OnRemoveInsertMenu)
END_EVENT_TABLE()
// Create a new application object: this macro will allow wxWindows to create
// implementation
// ============================================================================
+wxString IntTowxString(int number)
+{
+ return(wxString(IntToString(number)));
+}
+
+wxMenu *GetFileMenu(int menu_number)
+{
+ wxMenu *menuFile = new wxMenu("", wxMENU_TEAROFF);
+ menuFile->Append(Minimal_Quit, "E&xit" + IntTowxString(menu_number) +
+ "\tAlt-X", "Quit this program");
+ menuFile->Append(Minimal_SwapMenus, "&SwapMenus" + IntTowxString(menu_number)
+ + "\tAlt-S", "Swap Menus");
+ menuFile->Append(Minimal_ReplaceMenu, "&ReplaceMenu" +
+ IntTowxString(menu_number) + "\tAlt-R", "Replace Menu");
+ menuFile->Append(Minimal_RemoveInsertMenu, "&RemoveInsertMenu" +
+ IntTowxString(menu_number) + "\tAlt-I", "Remove Then Insert Menu");
+
+ return(menuFile);
+}
+
// ----------------------------------------------------------------------------
// the application class
// ----------------------------------------------------------------------------
// set the frame icon
SetIcon(wxICON(mondrian));
- // create a menu bar
- wxMenu *menuFile = new wxMenu("", wxMENU_TEAROFF);
+ bar[0] = new wxMenuBar();
+
+ wxMenu *menuFile0 = GetFileMenu(0);
+
+ wxMenu *helpMenu0 = new wxMenu;
+ helpMenu0->Append(Minimal_About, "&About0...\tCtrl-A",
+ "Show about dialog");
+
+ bar[0]->Append(menuFile0, "&File0");
+ bar[0]->Append(helpMenu0, "&Help0");
- // the "About" item should be in the help menu
- wxMenu *helpMenu = new wxMenu;
- helpMenu->Append(Minimal_About, "&About...\tCtrl-A", "Show about dialog");
- menuFile->Append(Minimal_Quit, "E&xit\tAlt-X", "Quit this program");
+ bar[1] = new wxMenuBar();
- // now append the freshly created menu to the menu bar...
- wxMenuBar *menuBar = new wxMenuBar();
- menuBar->Append(menuFile, "&File");
- menuBar->Append(helpMenu, "&Help");
+ wxMenu *menuFile1 = GetFileMenu(1);
+
+ wxMenu *helpMenu1 = new wxMenu;
+ helpMenu1->Append(Minimal_About, "&About1...\tCtrl-A",
+ "Show about dialog");
+
+ bar[1]->Append(menuFile1, "&File1");
+ bar[1]->Append(helpMenu1, "&Help1");
+
+ current_bar = 1;
// ... and attach this menu bar to the frame
- SetMenuBar(menuBar);
+ SetMenuBar(bar[current_bar]);
#if wxUSE_STATUSBAR
// create a status bar just for fun (by default with 1 pane only)
wxMessageBox(msg, "About Minimal", wxOK | wxICON_INFORMATION, this);
}
+
+void MyFrame::OnSwapMenus(wxCommandEvent& WXUNUSED(event))
+{
+ // Change the menu set around
+ current_bar = 1 - current_bar;
+ SetMenuBar(bar[current_bar]);
+}
+
+void MyFrame::OnReplaceMenu(wxCommandEvent& WXUNUSED(event))
+{
+ wxMenuBar *curr_bar = bar[current_bar];
+ wxMenu *menu = GetFileMenu(3);
+ wxString title = "&File3";
+
+ // Replace the first menu with the same thing
+ int pos = 1;
+ if (pos != wxNOT_FOUND)
+ {
+ curr_bar->Replace(pos, menu, title);
+// SetMenuBar(curr_bar);
+ }
+}
+
+
+void MyFrame::OnRemoveInsertMenu(wxCommandEvent& WXUNUSED(event))
+{
+ wxMenuBar *curr_bar = bar[current_bar];
+ wxMenu *menu = GetFileMenu(current_bar);
+ wxString title = "&File3";
+
+ // Remove the first menu then insert it back in
+ int pos = 1;
+ if (pos != wxNOT_FOUND)
+ {
+ curr_bar->Remove(pos);
+ if (curr_bar->GetMenuCount() != 0)
+ curr_bar->Insert(pos, menu, title);
+ else
+ curr_bar->Append(menu, title);
+
+ SetMenuBar(curr_bar);
+ }
+}
+
+
+
{
if ( pos == m_menus.GetCount() )
{
- return Append(menu, title);
+ return wxMenuBarBase::Append(menu, title);
}
else
{
wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
wxASSERT_MSG( (m_wxwindow != NULL), wxT("invalid frame") );
+ if (menuBar == m_frameMenuBar)
+ return;
+
+ if (m_frameMenuBar)
+ {
+ m_frameMenuBar->UnsetInvokingWindow( this );
+
+ if (m_frameMenuBar->GetWindowStyle() & wxMB_DOCKABLE)
+ {
+ gtk_signal_disconnect_by_func( GTK_OBJECT(m_frameMenuBar->m_widget),
+ GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
+
+ gtk_signal_disconnect_by_func( GTK_OBJECT(m_frameMenuBar->m_widget),
+ GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
+ }
+
+ gtk_container_remove( GTK_CONTAINER(m_mainWidget), m_frameMenuBar->m_widget );
+ gtk_widget_ref( m_frameMenuBar->m_widget );
+ gtk_widget_unparent( m_frameMenuBar->m_widget );
+ }
+
m_frameMenuBar = menuBar;
if (m_frameMenuBar)
{
m_frameMenuBar->SetInvokingWindow( this );
- if (m_frameMenuBar->GetParent() != this)
- {
- m_frameMenuBar->SetParent(this);
- gtk_pizza_put( GTK_PIZZA(m_mainWidget),
+ m_frameMenuBar->SetParent(this);
+ gtk_pizza_put( GTK_PIZZA(m_mainWidget),
m_frameMenuBar->m_widget,
m_frameMenuBar->m_x,
m_frameMenuBar->m_y,
m_frameMenuBar->m_width,
m_frameMenuBar->m_height );
- if (menuBar->GetWindowStyle() & wxMB_DOCKABLE)
- {
- gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_attached",
- GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
-
- gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_detached",
- GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
- }
+ if (menuBar->GetWindowStyle() & wxMB_DOCKABLE)
+ {
+ gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_attached",
+ GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
- m_frameMenuBar->Show( TRUE );
+ gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_detached",
+ GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
}
+
+ m_frameMenuBar->Show( TRUE );
}
/* resize window in OnInternalIdle */
if ( !GtkAppend(menu, title) )
return FALSE;
+ if (pos+1 >= m_menus.GetCount())
+ return TRUE;
+
GtkMenuShell *menu_shell = GTK_MENU_SHELL(m_factory->widget);
gpointer data = g_list_last(menu_shell->children)->data;
menu_shell->children = g_list_remove(menu_shell->children, data);
wxMenu *wxMenuBar::Replace(size_t pos, wxMenu *menu, const wxString& title)
{
- if ( !wxMenuBarBase::Replace(pos, menu, title) )
- return (wxMenu*) NULL;
-
// remove the old item and insert a new one
wxMenu *menuOld = Remove(pos);
if ( menuOld && !Insert(pos, menu, title) )
wxASSERT_MSG( (m_widget != NULL), wxT("invalid frame") );
wxASSERT_MSG( (m_wxwindow != NULL), wxT("invalid frame") );
+ if (menuBar == m_frameMenuBar)
+ return;
+
+ if (m_frameMenuBar)
+ {
+ m_frameMenuBar->UnsetInvokingWindow( this );
+
+ if (m_frameMenuBar->GetWindowStyle() & wxMB_DOCKABLE)
+ {
+ gtk_signal_disconnect_by_func( GTK_OBJECT(m_frameMenuBar->m_widget),
+ GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
+
+ gtk_signal_disconnect_by_func( GTK_OBJECT(m_frameMenuBar->m_widget),
+ GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
+ }
+
+ gtk_container_remove( GTK_CONTAINER(m_mainWidget), m_frameMenuBar->m_widget );
+ gtk_widget_ref( m_frameMenuBar->m_widget );
+ gtk_widget_unparent( m_frameMenuBar->m_widget );
+ }
+
m_frameMenuBar = menuBar;
if (m_frameMenuBar)
{
m_frameMenuBar->SetInvokingWindow( this );
- if (m_frameMenuBar->GetParent() != this)
- {
- m_frameMenuBar->SetParent(this);
- gtk_pizza_put( GTK_PIZZA(m_mainWidget),
+ m_frameMenuBar->SetParent(this);
+ gtk_pizza_put( GTK_PIZZA(m_mainWidget),
m_frameMenuBar->m_widget,
m_frameMenuBar->m_x,
m_frameMenuBar->m_y,
m_frameMenuBar->m_width,
m_frameMenuBar->m_height );
- if (menuBar->GetWindowStyle() & wxMB_DOCKABLE)
- {
- gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_attached",
- GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
-
- gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_detached",
- GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
- }
+ if (menuBar->GetWindowStyle() & wxMB_DOCKABLE)
+ {
+ gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_attached",
+ GTK_SIGNAL_FUNC(gtk_menu_attached_callback), (gpointer)this );
- m_frameMenuBar->Show( TRUE );
+ gtk_signal_connect( GTK_OBJECT(menuBar->m_widget), "child_detached",
+ GTK_SIGNAL_FUNC(gtk_menu_detached_callback), (gpointer)this );
}
+
+ m_frameMenuBar->Show( TRUE );
}
/* resize window in OnInternalIdle */
if ( !GtkAppend(menu, title) )
return FALSE;
+ if (pos+1 >= m_menus.GetCount())
+ return TRUE;
+
GtkMenuShell *menu_shell = GTK_MENU_SHELL(m_factory->widget);
gpointer data = g_list_last(menu_shell->children)->data;
menu_shell->children = g_list_remove(menu_shell->children, data);
wxMenu *wxMenuBar::Replace(size_t pos, wxMenu *menu, const wxString& title)
{
- if ( !wxMenuBarBase::Replace(pos, menu, title) )
- return (wxMenu*) NULL;
-
// remove the old item and insert a new one
wxMenu *menuOld = Remove(pos);
if ( menuOld && !Insert(pos, menu, title) )