/* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */ /* vim: set sw=2 ts=2 sts=2 et: */ /* * 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-history-view.h" #include "ephy-gui.h" #include #include #define EPHY_HISTORY_VIEW_GET_PRIVATE(object) (G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_HISTORY_VIEW, EphyHistoryViewPrivate)) struct _EphyHistoryViewPrivate { GtkTreePath *pressed_path; }; G_DEFINE_TYPE (EphyHistoryView, ephy_history_view, GTK_TYPE_TREE_VIEW) enum { ROW_MIDDLE_CLICKED, LAST_SIGNAL }; static guint signals[LAST_SIGNAL]; static gboolean button_event_modifies_selection (GdkEventButton *event) { return (event->state & (GDK_CONTROL_MASK | GDK_SHIFT_MASK)) != 0; } static gboolean ephy_history_view_button_press (GtkWidget *treeview, GdkEventButton *event) { EphyHistoryView *view; GtkTreeSelection *selection; GtkTreePath *path = NULL; gboolean path_is_selected, call_parent = TRUE; if (event->window != gtk_tree_view_get_bin_window (GTK_TREE_VIEW (treeview))) return GTK_WIDGET_CLASS (ephy_history_view_parent_class)->button_press_event (treeview, event); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (treeview)); if (gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (treeview), event->x, event->y, &path, NULL, NULL, NULL)) { path_is_selected = gtk_tree_selection_path_is_selected (selection, path); if (!gtk_widget_is_focus (GTK_WIDGET (treeview))) gtk_widget_grab_focus (GTK_WIDGET (treeview)); if (event->button == 3 && path_is_selected) call_parent = FALSE; if (!button_event_modifies_selection (event) && event->button == 1 && path_is_selected && gtk_tree_selection_count_selected_rows (selection) > 1) call_parent = FALSE; if (call_parent) GTK_WIDGET_CLASS (ephy_history_view_parent_class)->button_press_event (treeview, event); if (event->button == 3) { gboolean retval; g_signal_emit_by_name (treeview, "popup_menu", &retval); } else if (event->button == 2) { view = EPHY_HISTORY_VIEW (treeview); if (view->priv->pressed_path) gtk_tree_path_free (view->priv->pressed_path); view->priv->pressed_path = gtk_tree_path_copy (path); } gtk_tree_path_free (path); } return FALSE; } static gboolean ephy_history_view_button_release (GtkWidget *treeview, GdkEventButton *event) { EphyHistoryView *view; GtkTreePath *path; view = EPHY_HISTORY_VIEW (treeview); if (view->priv->pressed_path && event->button == 2 && gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (treeview), event->x, event->y, &path, NULL, NULL, NULL)) { if (gtk_tree_path_compare (view->priv->pressed_path, path) == 0) g_signal_emit (view, signals[ROW_MIDDLE_CLICKED], 0, path); gtk_tree_path_free (path); } return FALSE; } static void ephy_history_view_finalize (GObject *object) { EphyHistoryView *view = EPHY_HISTORY_VIEW (object); if (view->priv->pressed_path) gtk_tree_path_free (view->priv->pressed_path); G_OBJECT_CLASS (ephy_history_view_parent_class)->finalize (object); } static void ephy_history_view_class_init (EphyHistoryViewClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); object_class->finalize = ephy_history_view_finalize; widget_class->button_press_event = ephy_history_view_button_press; widget_class->button_release_event = ephy_history_view_button_release; signals[ROW_MIDDLE_CLICKED] = g_signal_new ("row-middle-clicked", G_OBJECT_CLASS_TYPE (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EphyHistoryViewClass, row_middle_clicked), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); g_type_class_add_private (object_class, sizeof (EphyHistoryViewPrivate)); } static void ephy_history_view_init (EphyHistoryView *self) { GtkTreeSelection *selection; self->priv = EPHY_HISTORY_VIEW_GET_PRIVATE (self); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (self)); gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE); } void ephy_history_view_popup (EphyHistoryView *view, GtkWidget *menu) { GdkEvent *event; g_return_if_fail (EPHY_IS_HISTORY_VIEW (view)); event = gtk_get_current_event (); if (event) { if (event->type == GDK_KEY_PRESS) { GdkEventKey *key = (GdkEventKey *) event; gtk_menu_popup (GTK_MENU (menu), NULL, NULL, ephy_gui_menu_position_tree_selection, view, 0, key->time); gtk_menu_shell_select_first (GTK_MENU_SHELL (menu), FALSE); } else if (event->type == GDK_BUTTON_PRESS) { GdkEventButton *button = (GdkEventButton *) event; gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL, button->button, button->time); } gdk_event_free (event); } }