/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright © 2003, 2004 Marco Pesenti Gritti * Copyright © 2003, 2004 Christian Persch * Copyright © 2008 Jan Alonzo * Copyright © 2009 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-navigation-action.h" #include "ephy-debug.h" #include "ephy-embed-container.h" #include "ephy-embed-shell.h" #include "ephy-embed-utils.h" #include "ephy-favicon-cache.h" #include "ephy-gui.h" #include "ephy-history.h" #include "ephy-link.h" #include "ephy-shell.h" #include "ephy-type-builtins.h" #include "ephy-window.h" #include #include #define HISTORY_ITEM_DATA_KEY "HistoryItem" #define URL_DATA_KEY "GoURL" #define EPHY_NAVIGATION_ACTION_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_NAVIGATION_ACTION, EphyNavigationActionPrivate)) typedef enum { WEBKIT_HISTORY_BACKWARD, WEBKIT_HISTORY_FORWARD } WebKitHistoryType; struct _EphyNavigationActionPrivate { EphyWindow *window; EphyNavigationDirection direction; char *arrow_tooltip; guint statusbar_cid; }; enum { PROP_0, PROP_ARROW_TOOLTIP, PROP_DIRECTION, PROP_WINDOW }; static void ephy_navigation_action_init (EphyNavigationAction *action); static void ephy_navigation_action_class_init (EphyNavigationActionClass *class); G_DEFINE_TYPE (EphyNavigationAction, ephy_navigation_action, EPHY_TYPE_LINK_ACTION) #define MAX_LABEL_LENGTH 48 static GtkWidget * new_history_menu_item (const char *origtext, const char *address) { EphyFaviconCache *cache; EphyHistory *history; GtkWidget *item, *image; GdkPixbuf *icon = NULL; GtkLabel *label; const char *icon_address; g_return_val_if_fail (address != NULL && origtext != NULL, NULL); item = gtk_image_menu_item_new_with_label (origtext); label = GTK_LABEL (gtk_bin_get_child (GTK_BIN (item))); gtk_label_set_ellipsize (label, PANGO_ELLIPSIZE_END); gtk_label_set_max_width_chars (label, MAX_LABEL_LENGTH); history = EPHY_HISTORY (ephy_embed_shell_get_global_history (embed_shell)); icon_address = ephy_history_get_icon (history, address); cache = EPHY_FAVICON_CACHE (ephy_embed_shell_get_favicon_cache (embed_shell)); icon = ephy_favicon_cache_get (cache, icon_address); if (icon != NULL) { image = gtk_image_new_from_pixbuf (icon); gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image); gtk_widget_show (image); g_object_unref (icon); } gtk_widget_show (item); return item; } static void activate_back_or_forward_menu_item_cb (GtkWidget *menuitem, EphyNavigationAction *action) { WebKitWebHistoryItem *item; EphyEmbed *embed; embed = ephy_embed_container_get_active_child (EPHY_EMBED_CONTAINER (action->priv->window)); g_return_if_fail (embed != NULL); if (ephy_gui_is_middle_click ()) { embed = ephy_link_open (EPHY_LINK (action), "about:blank", NULL, EPHY_LINK_NEW_TAB); g_return_if_fail (embed != NULL); } item = (WebKitWebHistoryItem*)g_object_get_data (G_OBJECT (menuitem), HISTORY_ITEM_DATA_KEY); g_return_if_fail (item != NULL); webkit_web_view_go_to_back_forward_item (EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed), item); } static void select_menu_item_cb (GtkWidget *menuitem, EphyNavigationAction *action) { const char *url; GtkWidget *statusbar; WebKitWebHistoryItem *item; item = (WebKitWebHistoryItem*)g_object_get_data (G_OBJECT (menuitem), HISTORY_ITEM_DATA_KEY); if (item) { url = webkit_web_history_item_get_uri (item); } else { url = g_object_get_data (G_OBJECT (menuitem), URL_DATA_KEY); g_return_if_fail (url != NULL); } statusbar = ephy_window_get_statusbar (action->priv->window); gtk_statusbar_push (GTK_STATUSBAR (statusbar), action->priv->statusbar_cid, url); } static void deselect_menu_item_cb (GtkWidget *menuitem, EphyNavigationAction *action) { GtkWidget *statusbar; statusbar = ephy_window_get_statusbar (action->priv->window); gtk_statusbar_pop (GTK_STATUSBAR (statusbar), action->priv->statusbar_cid); } static void activate_up_menu_item_cb (GtkWidget *menuitem, EphyNavigationAction *action) { EphyEmbed *embed; char *url; embed = ephy_embed_container_get_active_child (EPHY_EMBED_CONTAINER (action->priv->window)); g_return_if_fail (embed != NULL); url = g_object_get_data (G_OBJECT (menuitem), URL_DATA_KEY); g_return_if_fail (url != NULL); ephy_link_open (EPHY_LINK (action), url, NULL, ephy_gui_is_middle_click () ? EPHY_LINK_NEW_TAB : 0); } static GList* webkit_construct_history_list (WebKitWebView *web_view, WebKitHistoryType hist_type) { WebKitWebBackForwardList *web_back_forward_list; GList *webkit_items; web_back_forward_list = webkit_web_view_get_back_forward_list (web_view); if (hist_type == WEBKIT_HISTORY_FORWARD) webkit_items = g_list_reverse (webkit_web_back_forward_list_get_forward_list_with_limit (web_back_forward_list, EPHY_WEBKIT_BACK_FORWARD_LIMIT)); else webkit_items = webkit_web_back_forward_list_get_back_list_with_limit (web_back_forward_list, EPHY_WEBKIT_BACK_FORWARD_LIMIT); return webkit_items; } static GtkWidget * build_back_or_forward_menu (EphyNavigationAction *action) { EphyWindow *window = action->priv->window; GtkMenuShell *menu; EphyEmbed *embed; GList *list, *l; WebKitWebView *web_view; embed = ephy_embed_container_get_active_child (EPHY_EMBED_CONTAINER (window)); g_return_val_if_fail (embed != NULL, NULL); web_view = EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed); g_return_val_if_fail (web_view != NULL, NULL); if (action->priv->direction == EPHY_NAVIGATION_DIRECTION_BACK) list = webkit_construct_history_list (web_view, WEBKIT_HISTORY_BACKWARD); else list = webkit_construct_history_list (web_view, WEBKIT_HISTORY_FORWARD); menu = GTK_MENU_SHELL (gtk_menu_new ()); l = list; for (l = list; l != NULL; l = l->next) { GtkWidget *item; WebKitWebHistoryItem *hitem; const char *title, *url; hitem = (WebKitWebHistoryItem*)l->data; url = webkit_web_history_item_get_uri (hitem); title = webkit_web_history_item_get_title (hitem); item = new_history_menu_item (title ? title : url, url); g_object_set_data_full (G_OBJECT (item), HISTORY_ITEM_DATA_KEY, g_object_ref (hitem), g_object_unref); g_signal_connect (item, "activate", G_CALLBACK (activate_back_or_forward_menu_item_cb), action); g_signal_connect (item, "select", G_CALLBACK (select_menu_item_cb), action); g_signal_connect (item, "deselect", G_CALLBACK (deselect_menu_item_cb), action); gtk_menu_shell_append (menu, item); gtk_widget_show_all (item); } g_list_free (list); return GTK_WIDGET (menu); } static GtkWidget * build_up_menu (EphyNavigationAction *action) { EphyWindow *window = action->priv->window; EphyEmbed *embed; EphyHistory *history; GtkMenuShell *menu; GtkWidget *item; GSList *list, *l; char *url; embed = ephy_embed_container_get_active_child (EPHY_EMBED_CONTAINER (window)); g_return_val_if_fail (embed != NULL, NULL); menu = GTK_MENU_SHELL (gtk_menu_new ()); history = EPHY_HISTORY (ephy_embed_shell_get_global_history (embed_shell)); list = ephy_web_view_get_go_up_list (EPHY_GET_EPHY_WEB_VIEW_FROM_EMBED (embed)); for (l = list; l != NULL; l = l->next) { EphyNode *node; const char *title = NULL; url = l->data; if (url == NULL) continue; node = ephy_history_get_page (history, url); if (node != NULL) { title = ephy_node_get_property_string (node, EPHY_NODE_PAGE_PROP_TITLE); } item = new_history_menu_item (title ? title : url, url); g_object_set_data_full (G_OBJECT (item), URL_DATA_KEY, url, (GDestroyNotify) g_free); g_signal_connect (item, "activate", G_CALLBACK (activate_up_menu_item_cb), action); g_signal_connect (item, "select", G_CALLBACK (select_menu_item_cb), action); g_signal_connect (item, "deselect", G_CALLBACK (deselect_menu_item_cb), action); gtk_menu_shell_append (menu, item); gtk_widget_show (item); } /* the list data has been consumed */ g_slist_foreach (list, (GFunc) g_free, NULL); g_slist_free (list); return GTK_WIDGET (menu); } static void menu_activated_cb (GtkMenuToolButton *button, EphyNavigationAction *action) { GtkWidget *menu = NULL; LOG ("menu_activated_cb dir %d", action->priv->direction); switch (action->priv->direction) { case EPHY_NAVIGATION_DIRECTION_UP: menu = build_up_menu (action); break; case EPHY_NAVIGATION_DIRECTION_FORWARD: case EPHY_NAVIGATION_DIRECTION_BACK: menu = build_back_or_forward_menu (action); break; default: g_assert_not_reached (); break; } gtk_menu_tool_button_set_menu (button, menu); } static void connect_proxy (GtkAction *gaction, GtkWidget *proxy) { LOG ("Connect navigation action proxy"); if (GTK_IS_MENU_TOOL_BUTTON (proxy)) { EphyNavigationAction *action = EPHY_NAVIGATION_ACTION (gaction); EphyNavigationActionPrivate *priv = action->priv; GtkMenuToolButton *button = GTK_MENU_TOOL_BUTTON (proxy); GtkWidget *menu; /* set dummy menu so the arrow gets sensitive */ menu = gtk_menu_new (); gtk_menu_tool_button_set_menu (button, menu); gtk_menu_tool_button_set_arrow_tooltip_text (button, priv->arrow_tooltip); g_signal_connect (proxy, "show-menu", G_CALLBACK (menu_activated_cb), gaction); } GTK_ACTION_CLASS (ephy_navigation_action_parent_class)->connect_proxy (gaction, proxy); } static void ephy_navigation_action_activate (GtkAction *gtk_action) { EphyNavigationAction *action = EPHY_NAVIGATION_ACTION (gtk_action); EphyWindow *window = action->priv->window; EphyEmbed *embed; WebKitWebView *web_view; embed = ephy_embed_container_get_active_child (EPHY_EMBED_CONTAINER (window)); g_return_if_fail (embed != NULL); web_view = EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed); if (action->priv->direction == EPHY_NAVIGATION_DIRECTION_BACK) { if (ephy_gui_is_middle_click ()) { embed = ephy_shell_new_tab (ephy_shell_get_default (), EPHY_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (embed))), embed, NULL, EPHY_NEW_TAB_IN_EXISTING_WINDOW); web_view = EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed); } webkit_web_view_go_back (web_view); } else if (action->priv->direction == EPHY_NAVIGATION_DIRECTION_FORWARD) { if (ephy_gui_is_middle_click ()) { const char *forward_uri; WebKitWebHistoryItem *forward_item; WebKitWebBackForwardList *history; /* Forward history is not copied when opening a new tab, so get the forward URI manually and load it */ history = webkit_web_view_get_back_forward_list (EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed)); forward_item = webkit_web_back_forward_list_get_forward_item (history); forward_uri = webkit_web_history_item_get_original_uri (forward_item); embed = ephy_shell_new_tab (ephy_shell_get_default (), EPHY_WINDOW (gtk_widget_get_toplevel (GTK_WIDGET (embed))), embed, NULL, EPHY_NEW_TAB_IN_EXISTING_WINDOW); web_view = EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed); webkit_web_view_load_uri (web_view, forward_uri); } else webkit_web_view_go_forward (EPHY_GET_WEBKIT_WEB_VIEW_FROM_EMBED (embed)); } else if (action->priv->direction == EPHY_NAVIGATION_DIRECTION_UP) { GSList *up_list; up_list = ephy_web_view_get_go_up_list (EPHY_GET_EPHY_WEB_VIEW_FROM_EMBED (embed)); ephy_link_open (EPHY_LINK (action), up_list->data, NULL, ephy_gui_is_middle_click () ? EPHY_LINK_NEW_TAB : 0); g_slist_foreach (up_list, (GFunc) g_free, NULL); g_slist_free (up_list); } } static void ephy_navigation_action_init (EphyNavigationAction *action) { action->priv = EPHY_NAVIGATION_ACTION_GET_PRIVATE (action); } static void ephy_navigation_action_finalize (GObject *object) { EphyNavigationAction *action = EPHY_NAVIGATION_ACTION (object); g_free (action->priv->arrow_tooltip); G_OBJECT_CLASS (ephy_navigation_action_parent_class)->finalize (object); } static void ephy_navigation_action_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { EphyNavigationAction *nav = EPHY_NAVIGATION_ACTION (object); switch (prop_id) { case PROP_ARROW_TOOLTIP: nav->priv->arrow_tooltip = g_value_dup_string (value); g_object_notify (object, "tooltip"); break; case PROP_DIRECTION: nav->priv->direction = g_value_get_int (value); break; case PROP_WINDOW: { GtkWidget *statusbar; nav->priv->window = EPHY_WINDOW (g_value_get_object (value)); /* statusbar context to display current selected item */ statusbar = ephy_window_get_statusbar (nav->priv->window); nav->priv->statusbar_cid = gtk_statusbar_get_context_id ( GTK_STATUSBAR (statusbar), "navigation_message"); } break; } } static void ephy_navigation_action_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { EphyNavigationAction *nav = EPHY_NAVIGATION_ACTION (object); switch (prop_id) { case PROP_ARROW_TOOLTIP: g_value_set_string (value, nav->priv->arrow_tooltip); break; case PROP_DIRECTION: g_value_set_int (value, nav->priv->direction); break; case PROP_WINDOW: g_value_set_object (value, nav->priv->window); break; } } static void ephy_navigation_action_class_init (EphyNavigationActionClass *class) { GObjectClass *object_class = G_OBJECT_CLASS (class); GtkActionClass *action_class = GTK_ACTION_CLASS (class); object_class->finalize = ephy_navigation_action_finalize; object_class->set_property = ephy_navigation_action_set_property; object_class->get_property = ephy_navigation_action_get_property; action_class->toolbar_item_type = GTK_TYPE_MENU_TOOL_BUTTON; action_class->connect_proxy = connect_proxy; action_class->activate = ephy_navigation_action_activate; g_object_class_install_property (object_class, PROP_ARROW_TOOLTIP, g_param_spec_string ("arrow-tooltip", NULL, NULL, NULL, G_PARAM_READWRITE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB)); g_object_class_install_property (object_class, PROP_DIRECTION, g_param_spec_int ("direction", NULL, NULL, 0, G_MAXINT, 0, G_PARAM_READWRITE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB)); g_object_class_install_property (object_class, PROP_WINDOW, g_param_spec_object ("window", NULL, NULL, G_TYPE_OBJECT, G_PARAM_READWRITE | G_PARAM_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB)); g_type_class_add_private (object_class, sizeof (EphyNavigationActionPrivate)); }