/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* * Copyright © 2012 Igalia S.L. * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * */ #include "config.h" #include "ephy-page-menu-action.h" G_DEFINE_TYPE (EphyPageMenuAction, ephy_page_menu_action, EPHY_TYPE_WINDOW_ACTION); #define EPHY_PAGE_MENU_ACTION_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), EPHY_TYPE_PAGE_MENU_ACTION, EphyPageMenuActionPrivate)) struct _EphyPageMenuActionPrivate { GtkWidget *menu; GtkWidget *button; }; static void menu_position_func (GtkMenu *menu, int *x, int *y, gboolean *push_in, GtkButton *button) { GtkAllocation allocation; GtkWidget *widget = GTK_WIDGET (button); GtkRequisition menu_req; GtkTextDirection direction; GdkWindow *window; GtkWidget *toplevel; toplevel = gtk_widget_get_toplevel (GTK_WIDGET (menu)); gtk_window_set_type_hint (GTK_WINDOW (toplevel), GDK_WINDOW_TYPE_HINT_DROPDOWN_MENU); gtk_widget_get_preferred_size (GTK_WIDGET (menu), &menu_req, NULL); direction = gtk_widget_get_direction (widget); window = gtk_widget_get_window (widget); gtk_widget_get_allocation (widget, &allocation); gdk_window_get_origin (window, x, y); *x += allocation.x; *y += allocation.y + allocation.height; if (direction == GTK_TEXT_DIR_LTR) *x += allocation.width - menu_req.width; *push_in = FALSE; } static void visible_cb (GtkWidget *menu, GParamSpec *pspec, GtkWidget *button) { if (gtk_widget_get_visible (menu)) gtk_style_context_add_class (gtk_widget_get_style_context (button), "active-menu"); else gtk_style_context_remove_class (gtk_widget_get_style_context (button), "active-menu"); } static void button_press_cb (GtkWidget *button, GdkEventButton *event, EphyPageMenuAction *action) { GtkWidget *menu; EphyWindow *window; GtkUIManager *manager; guint event_button = 1; guint32 event_time = 0; if (!button) { GSList *l = gtk_action_get_proxies (GTK_ACTION (action)); if (GTK_IS_BUTTON (l->data)) button = GTK_WIDGET (l->data); } g_return_if_fail (GTK_IS_BUTTON (button)); if (!action->priv->menu) { window = ephy_window_action_get_window (EPHY_WINDOW_ACTION (action)); manager = ephy_window_get_ui_manager (window); menu = gtk_ui_manager_get_widget (manager, "/ui/PagePopup"); g_signal_connect (menu, "notify::visible", G_CALLBACK (visible_cb), button); action->priv->menu = g_object_ref (menu); action->priv->button = g_object_ref (button); } if (event) { event_button = event->button; event_time = event->time; } gtk_menu_popup (GTK_MENU (action->priv->menu), NULL, NULL, (GtkMenuPositionFunc)menu_position_func, button, event_button, event_time); } static void ephy_page_menu_action_activate (GtkAction *action) { button_press_cb (NULL, NULL, EPHY_PAGE_MENU_ACTION (action)); } static void ephy_page_menu_action_connect_proxy (GtkAction *action, GtkWidget *proxy) { if (GTK_IS_BUTTON (proxy)) g_signal_connect (proxy, "button-press-event", G_CALLBACK (button_press_cb), action); GTK_ACTION_CLASS (ephy_page_menu_action_parent_class)->connect_proxy (action, proxy); } static void ephy_page_menu_action_disconnect_proxy (GtkAction *action, GtkWidget *proxy) { if (GTK_IS_BUTTON (proxy)) g_signal_handlers_disconnect_by_func (proxy, G_CALLBACK (button_press_cb), action); GTK_ACTION_CLASS (ephy_page_menu_action_parent_class)->disconnect_proxy (action, proxy); } static void ephy_page_menu_action_dispose (GObject *object) { EphyPageMenuActionPrivate *priv = EPHY_PAGE_MENU_ACTION (object)->priv; if (priv->menu) g_signal_handlers_disconnect_by_func (priv->menu, G_CALLBACK (visible_cb), priv->button); g_clear_object (&priv->menu); g_clear_object (&priv->button); G_OBJECT_CLASS (ephy_page_menu_action_parent_class)->dispose (object); } static void ephy_page_menu_action_class_init (EphyPageMenuActionClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GtkActionClass *action_class = GTK_ACTION_CLASS (klass); gobject_class->dispose = ephy_page_menu_action_dispose; action_class->activate = ephy_page_menu_action_activate; action_class->connect_proxy = ephy_page_menu_action_connect_proxy; action_class->disconnect_proxy = ephy_page_menu_action_disconnect_proxy; g_type_class_add_private (klass, sizeof (EphyPageMenuActionPrivate)); } static void ephy_page_menu_action_init (EphyPageMenuAction *self) { self->priv = EPHY_PAGE_MENU_ACTION_GET_PRIVATE (self); }