// Returns true if SYSTRAY protocol is supported by the desktop
bool IsProtocolSupported();
+
+ wxEvtHandler *m_invokingWindow;
+
+#if wxUSE_MENUS_NATIVE
+ virtual bool DoPopupMenu( wxMenu *menu, int x, int y );
+#endif // wxUSE_MENUS_NATIVE
};
#endif // _WX_TASKBARPRIV_H_
// Returns true if SYSTRAY protocol is supported by the desktop
bool IsProtocolSupported();
+
+ wxEvtHandler *m_invokingWindow;
+
+#if wxUSE_MENUS_NATIVE
+ virtual bool DoPopupMenu( wxMenu *menu, int x, int y );
+#endif // wxUSE_MENUS_NATIVE
};
#endif // _WX_TASKBARPRIV_H_
#include "wx/gtk/taskbarpriv.h"
#include "wx/log.h"
#include "wx/frame.h"
+#include "wx/menu.h"
#include <gdk/gdkx.h>
#include <gtk/gtkversion.h>
#if GTK_CHECK_VERSION(2, 1, 0)
+#include "gtk/gtk.h"
+
#include "eggtrayicon.h"
wxTaskBarIconAreaBase::wxTaskBarIconAreaBase()
wxDEFAULT_FRAME_STYLE | wxFRAME_NO_TASKBAR | wxSIMPLE_BORDER |
wxFRAME_SHAPED,
wxEmptyString /*eggtray doesn't like setting wmclass*/);
+
+ m_invokingWindow = NULL;
}
bool wxTaskBarIconAreaBase::IsProtocolSupported()
return (bool)s_supported;
}
+//-----------------------------------------------------------------------------
+// Pop-up menu stuff
+//-----------------------------------------------------------------------------
+
+extern "C" void gtk_pop_hide_callback( GtkWidget *widget, bool* is_waiting );
+
+extern void SetInvokingWindow( wxMenu *menu, wxWindow* win );
+
+extern "C" void wxPopupMenuPositionCallback( GtkMenu *menu,
+ gint *x, gint *y,
+ gboolean * WXUNUSED(whatever),
+ gpointer user_data );
+
+bool wxTaskBarIconAreaBase::DoPopupMenu( wxMenu *menu, int x, int y )
+{
+ wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
+
+ wxCHECK_MSG( menu != NULL, false, wxT("invalid popup-menu") );
+
+ // NOTE: if you change this code, you need to update
+ // the same code in window.cpp as well. This
+ // is ugly code duplication, I know,
+
+ SetInvokingWindow( menu, this );
+
+ menu->UpdateUI( m_invokingWindow );
+
+ bool is_waiting = true;
+
+ gulong handler = gtk_signal_connect( GTK_OBJECT(menu->m_menu),
+ "hide",
+ GTK_SIGNAL_FUNC(gtk_pop_hide_callback),
+ (gpointer)&is_waiting );
+
+ wxPoint pos;
+ gpointer userdata;
+ GtkMenuPositionFunc posfunc;
+ if ( x == -1 && y == -1 )
+ {
+ // use GTK's default positioning algorithm
+ userdata = NULL;
+ posfunc = NULL;
+ }
+ else
+ {
+ pos = ClientToScreen(wxPoint(x, y));
+ userdata = &pos;
+ posfunc = wxPopupMenuPositionCallback;
+ }
+
+ gtk_menu_popup(
+ GTK_MENU(menu->m_menu),
+ (GtkWidget *) NULL, // parent menu shell
+ (GtkWidget *) NULL, // parent menu item
+ posfunc, // function to position it
+ userdata, // client data
+ 0, // button used to activate it
+ gtk_get_current_event_time()
+ );
+
+ while (is_waiting)
+ {
+ gtk_main_iteration();
+ }
+
+ gtk_signal_disconnect(GTK_OBJECT(menu->m_menu), handler);
+
+ return true;
+}
+
#endif // __WXGTK20__
#endif // GTK_CHECK_VERSION(2, 1, 0)
*is_waiting = FALSE;
}
-static void SetInvokingWindow( wxMenu *menu, wxWindowGTK *win )
+void SetInvokingWindow( wxMenu *menu, wxWindow* win )
{
menu->SetInvokingWindow( win );
+
wxMenuItemList::compatibility_iterator node = menu->GetMenuItems().GetFirst();
while (node)
{
wxCHECK_MSG( menu != NULL, false, wxT("invalid popup-menu") );
+ // NOTE: if you change this code, you need to update
+ // the same code in taskbar.cpp as well. This
+ // is ugly code duplication, I know,
+
SetInvokingWindow( menu, this );
menu->UpdateUI();
#include "wx/gtk/taskbarpriv.h"
#include "wx/log.h"
#include "wx/frame.h"
+#include "wx/menu.h"
#include <gdk/gdkx.h>
#include <gtk/gtkversion.h>
#if GTK_CHECK_VERSION(2, 1, 0)
+#include "gtk/gtk.h"
+
#include "eggtrayicon.h"
wxTaskBarIconAreaBase::wxTaskBarIconAreaBase()
wxDEFAULT_FRAME_STYLE | wxFRAME_NO_TASKBAR | wxSIMPLE_BORDER |
wxFRAME_SHAPED,
wxEmptyString /*eggtray doesn't like setting wmclass*/);
+
+ m_invokingWindow = NULL;
}
bool wxTaskBarIconAreaBase::IsProtocolSupported()
return (bool)s_supported;
}
+//-----------------------------------------------------------------------------
+// Pop-up menu stuff
+//-----------------------------------------------------------------------------
+
+extern "C" void gtk_pop_hide_callback( GtkWidget *widget, bool* is_waiting );
+
+extern void SetInvokingWindow( wxMenu *menu, wxWindow* win );
+
+extern "C" void wxPopupMenuPositionCallback( GtkMenu *menu,
+ gint *x, gint *y,
+ gboolean * WXUNUSED(whatever),
+ gpointer user_data );
+
+bool wxTaskBarIconAreaBase::DoPopupMenu( wxMenu *menu, int x, int y )
+{
+ wxCHECK_MSG( m_widget != NULL, false, wxT("invalid window") );
+
+ wxCHECK_MSG( menu != NULL, false, wxT("invalid popup-menu") );
+
+ // NOTE: if you change this code, you need to update
+ // the same code in window.cpp as well. This
+ // is ugly code duplication, I know,
+
+ SetInvokingWindow( menu, this );
+
+ menu->UpdateUI( m_invokingWindow );
+
+ bool is_waiting = true;
+
+ gulong handler = gtk_signal_connect( GTK_OBJECT(menu->m_menu),
+ "hide",
+ GTK_SIGNAL_FUNC(gtk_pop_hide_callback),
+ (gpointer)&is_waiting );
+
+ wxPoint pos;
+ gpointer userdata;
+ GtkMenuPositionFunc posfunc;
+ if ( x == -1 && y == -1 )
+ {
+ // use GTK's default positioning algorithm
+ userdata = NULL;
+ posfunc = NULL;
+ }
+ else
+ {
+ pos = ClientToScreen(wxPoint(x, y));
+ userdata = &pos;
+ posfunc = wxPopupMenuPositionCallback;
+ }
+
+ gtk_menu_popup(
+ GTK_MENU(menu->m_menu),
+ (GtkWidget *) NULL, // parent menu shell
+ (GtkWidget *) NULL, // parent menu item
+ posfunc, // function to position it
+ userdata, // client data
+ 0, // button used to activate it
+ gtk_get_current_event_time()
+ );
+
+ while (is_waiting)
+ {
+ gtk_main_iteration();
+ }
+
+ gtk_signal_disconnect(GTK_OBJECT(menu->m_menu), handler);
+
+ return true;
+}
+
#endif // __WXGTK20__
#endif // GTK_CHECK_VERSION(2, 1, 0)
*is_waiting = FALSE;
}
-static void SetInvokingWindow( wxMenu *menu, wxWindowGTK *win )
+void SetInvokingWindow( wxMenu *menu, wxWindow* win )
{
menu->SetInvokingWindow( win );
+
wxMenuItemList::compatibility_iterator node = menu->GetMenuItems().GetFirst();
while (node)
{
wxCHECK_MSG( menu != NULL, false, wxT("invalid popup-menu") );
+ // NOTE: if you change this code, you need to update
+ // the same code in taskbar.cpp as well. This
+ // is ugly code duplication, I know,
+
SetInvokingWindow( menu, this );
menu->UpdateUI();
_T("using legacy KDE1,2 and GNOME 1.2 methods"));
SetLegacyWMProperties();
}
+
+#ifdef __WXGTK20__
+ m_invokingWindow = icon;
+#endif
// Set initial size to bitmap size (tray manager may and often will
// change it):