/*
* Copyright (C) 2003, 2004 Marco Pesenti Gritti
* Copyright (C) 2003, 2004 Christian Persch
*
* 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* $Id$
*/
#include "config.h"
#include "ephy-topic-action.h"
#include "ephy-node-common.h"
#include "ephy-bookmarks.h"
#include "ephy-bookmarksbar.h"
#include "ephy-link.h"
#include "ephy-favicon-cache.h"
#include "ephy-shell.h"
#include "ephy-dnd.h"
#include "ephy-gui.h"
#include "ephy-debug.h"
#include <glib/gi18n.h>
#include <gtk/gtkwidget.h>
#include <gtk/gtkarrow.h>
#include <gtk/gtkhbox.h>
#include <gtk/gtklabel.h>
#include <gtk/gtkbutton.h>
#include <gtk/gtktogglebutton.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkimage.h>
#include <gtk/gtkmenuitem.h>
#include <gtk/gtkimagemenuitem.h>
#include <gtk/gtkseparatormenuitem.h>
#include <gtk/gtkmain.h>
#include <libgnomevfs/gnome-vfs-uri.h>
#include <string.h>
#define TOOLITEM_WIDTH_CHARS 24
#define MENUITEM_WIDTH_CHARS 32
#define LABEL_WIDTH_CHARS 32
#define EPHY_TOPIC_ACTION_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_TOPIC_ACTION, EphyTopicActionPrivate))
static GtkTargetEntry drag_targets[] =
{
{ EPHY_DND_TOPIC_TYPE, 0, 0 }
};
static int n_drag_targets = G_N_ELEMENTS (drag_targets);
struct _EphyTopicActionPrivate
{
EphyNode *topic_node;
guint motion_handler;
guint release_handler;
gint drag_x;
gint drag_y;
};
enum
{
PROP_0,
PROP_TOPIC
};
static void ephy_topic_action_class_init (EphyTopicActionClass *class);
static void ephy_topic_action_init (EphyTopicAction *action);
static GObjectClass *parent_class = NULL;
GType
ephy_topic_action_get_type (void)
{
static GType type = 0;
if (G_UNLIKELY (type == 0))
{
static const GTypeInfo type_info =
{
sizeof (EphyTopicActionClass),
(GBaseInitFunc) NULL,
(GBaseFinalizeFunc) NULL,
(GClassInitFunc) ephy_topic_action_class_init,
(GClassFinalizeFunc) NULL,
NULL,
sizeof (EphyTopicAction),
0, /* n_preallocs */
(GInstanceInitFunc) ephy_topic_action_init,
};
type = g_type_register_static (EPHY_TYPE_LINK_ACTION,
"EphyTopicAction",
&type_info, 0);
}
return type;
}
static GtkWidget *
create_tool_item (GtkAction *action)
{
GtkWidget *item;
GtkWidget *button;
GtkWidget *arrow;
GtkWidget *hbox;
GtkWidget *label;
item = (* GTK_ACTION_CLASS (parent_class)->create_tool_item) (action);
hbox = gtk_hbox_new (FALSE, 0);
gtk_widget_show (hbox);
gtk_container_add (GTK_CONTAINER (item), hbox);
button = gtk_toggle_button_new ();
gtk_widget_add_events (GTK_WIDGET (button), GDK_BUTTON1_MOTION_MASK);
gtk_button_set_relief (GTK_BUTTON (button), GTK_RELIEF_NONE);
gtk_button_set_focus_on_click (GTK_BUTTON (button), FALSE);
gtk_widget_show (button);
gtk_container_add (GTK_CONTAINER (hbox), button);
g_object_set_data (G_OBJECT (item), "button", button);
arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_NONE);
gtk_widget_show (arrow);
hbox = gtk_hbox_new (FALSE, 3);
gtk_widget_show (hbox);
gtk_container_add (GTK_CONTAINER (button), hbox);
label = gtk_label_new (NULL);
gtk_label_set_ellipsize (GTK_LABEL (label), PANGO_ELLIPSIZE_END);
gtk_label_set_max_width_chars (GTK_LABEL (label), TOOLITEM_WIDTH_CHARS);
gtk_widget_show (label);
gtk_box_pack_start (GTK_BOX (hbox), label, TRUE, TRUE, 0);
gtk_box_pack_start (GTK_BOX (hbox), arrow, TRUE, TRUE, 0);
g_object_set_data (G_OBJECT (item), "label", label);
return item;
}
static void
menu_deactivate_cb (GtkMenuShell *ms, GtkWidget *button)
{
g_object_set_data (G_OBJECT (button), "popup", NULL);
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), FALSE);
gtk_button_released (GTK_BUTTON (button));
}
static void
menu_activate_cb (GtkWidget *item, GtkAction *action)
{
EphyNode *node;
const char *location;
node = g_object_get_data (G_OBJECT (item), "node");
location = ephy_node_get_property_string
(node, EPHY_NODE_BMK_PROP_LOCATION);
ephy_link_open (EPHY_LINK (action), location, NULL,
ephy_gui_is_middle_click () ? EPHY_LINK_NEW_TAB : 0);
}
static void
ephy_topic_action_sync_label (GtkAction *gaction,
GParamSpec *pspec,
GtkWidget *proxy)
{
EphyTopicAction *action = EPHY_TOPIC_ACTION (gaction);
g_return_if_fail (EPHY_IS_NODE (action->priv->topic_node));
/* note that we cannot use ellipsizing label with defined width,
* since that makes the label exactly that wide, even if the
* text takes less space. So we have to shorten the string.
*/
if (GTK_IS_TOOL_ITEM (proxy))
{
GtkWidget *label = NULL;
char *title;
label = g_object_get_data (G_OBJECT (proxy), "label");
g_return_if_fail (label != NULL);
g_object_get (G_OBJECT (action), "label", &title, NULL);
gtk_label_set_label (GTK_LABEL (label), title);
g_free (title);
}
}
static int
sort_bookmarks (gconstpointer a, gconstpointer b)
{
EphyNode *node_a = (EphyNode *)a;
EphyNode *node_b = (EphyNode *)b;
const char *title1, *title2;
int retval;
title1 = ephy_node_get_property_string (node_a, EPHY_NODE_BMK_PROP_TITLE);
title2 = ephy_node_get_property_string (node_b, EPHY_NODE_BMK_PROP_TITLE);
if (title1 == NULL)
{
retval = -1;
}
else if (title2 == NULL)
{
retval = 1;
}
else
{
char *str_a, *str_b;
str_a = g_utf8_casefold (title1, -1);
str_b = g_utf8_casefold (title2, -1);
retval = g_utf8_collate (str_a, str_b);
g_free (str_a);
g_free (str_b);
}
return retval;
}
static gboolean
can_open_in_tabs (EphyNode *node)
{
GPtrArray *children;
int priority;
priority = ephy_node_get_property_int (node, EPHY_NODE_KEYWORD_PROP_PRIORITY);
children = ephy_node_get_children (node);
return (priority != EPHY_NODE_ALL_PRIORITY) && (children->len > 1);
}
static void
append_bookmarks_menu (EphyTopicAction *action, GtkWidget *menu, EphyNode *node, gboolean show_empty)
{
EphyFaviconCache *cache;
GtkWidget *item;
GtkLabel *label;
GPtrArray *children;
cache = EPHY_FAVICON_CACHE
(ephy_embed_shell_get_favicon_cache (EPHY_EMBED_SHELL (ephy_shell)));
children = ephy_node_get_children (node);
if (children->len < 1 && show_empty)
{
/* This is the adjective, not the verb */
item = gtk_menu_item_new_with_label (_("Empty"));
gtk_widget_set_sensitive (item, FALSE);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
return;
}
else
{
GList *node_list = NULL, *l;
int i;
for (i = 0; i < children->len; ++i)
{
node_list = g_list_prepend (node_list,
g_ptr_array_index (children, i));
}
node_list = g_list_sort (node_list, (GCompareFunc)sort_bookmarks);
for (l = node_list; l != NULL; l = l->next)
{
EphyNode *kid = (EphyNode*) l->data;
const char *icon_location;
const char *title;
icon_location = ephy_node_get_property_string
(kid, EPHY_NODE_BMK_PROP_ICON);
title = ephy_node_get_property_string
(kid, EPHY_NODE_BMK_PROP_TITLE);
if (title == NULL) continue;
item = gtk_image_menu_item_new_with_label (title);
label = (GtkLabel *) ((GtkBin *) item)->child;
gtk_label_set_max_width_chars (label, LABEL_WIDTH_CHARS);
gtk_label_set_ellipsize (label, PANGO_ELLIPSIZE_END);
gtk_label_set_use_underline (label, FALSE);
if (icon_location)
{
GdkPixbuf *icon;
GtkWidget *image;
icon = ephy_favicon_cache_get (cache, icon_location);
if (icon != NULL)
{
image = gtk_image_new_from_pixbuf (icon);
gtk_widget_show (image);
gtk_image_menu_item_set_image
(GTK_IMAGE_MENU_ITEM (item), image);
g_object_unref (icon);
}
}
g_object_set_data (G_OBJECT (item), "node", kid);
g_signal_connect (item, "activate",
G_CALLBACK (menu_activate_cb), action);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
}
g_list_free (node_list);
}
}
#define TOPIC_NODE_DATA_KEY "TopicNode"
static void
open_in_tabs_activate_cb (GtkWidget *item, EphyTopicAction *action)
{
EphyNode *node;
GPtrArray *children;
EphyTab *tab = NULL;
GList *node_list = NULL, *l;
int i;
node = g_object_get_data (G_OBJECT (item), TOPIC_NODE_DATA_KEY);
g_return_if_fail (node != NULL);
children = ephy_node_get_children (node);
for (i = 0; i < children->len; ++i)
{
node_list = g_list_prepend (node_list,
g_ptr_array_index (children, i));
}
node_list = g_list_sort (node_list, (GCompareFunc) sort_bookmarks);
for (l = node_list; l != NULL; l = l->next)
{
EphyNode *child = (EphyNode *) l->data;
const char *address;
address = ephy_node_get_property_string
(child, EPHY_NODE_BMK_PROP_LOCATION);
tab = ephy_link_open (EPHY_LINK (action), address, tab,
tab ? EPHY_LINK_NEW_TAB : EPHY_LINK_NEW_WINDOW);
}
g_list_free (node_list);
}
static int
get_item_position (GtkWidget *widget, gboolean *last)
{
GtkWidget *item, *toolbar;
int index;
item = gtk_widget_get_ancestor (widget, GTK_TYPE_TOOL_ITEM);
g_return_val_if_fail (item != NULL, -1);
toolbar = gtk_widget_get_ancestor (widget, GTK_TYPE_TOOLBAR);
g_return_val_if_fail (toolbar != NULL, -1);
index = gtk_toolbar_get_item_index (GTK_TOOLBAR (toolbar),
GTK_TOOL_ITEM (item));
if (last)
{
int n_items;
n_items = gtk_toolbar_get_n_items (GTK_TOOLBAR (toolbar));
*last = (index == n_items - 1);
}
return index;
}
static void
remove_from_model (GtkWidget *widget)
{
EphyBookmarks *bookmarks;
EggToolbarsModel *model;
int pos;
pos = get_item_position (widget, NULL);
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
model = EGG_TOOLBARS_MODEL (ephy_bookmarks_get_toolbars_model (bookmarks));
egg_toolbars_model_remove_item (model, 0, pos);
}
static void
move_in_model (GtkWidget *widget, int direction)
{
EphyBookmarks *bookmarks;
EggToolbarsModel *model;
int pos, new_pos;
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
model = EGG_TOOLBARS_MODEL (ephy_bookmarks_get_toolbars_model (bookmarks));
pos = get_item_position (widget, NULL);
new_pos = MAX (0, pos + direction);
egg_toolbars_model_move_item (model, 0, pos, 0, new_pos);
}
static void
remove_activate_cb (GtkWidget *menu, GtkWidget *proxy)
{
remove_from_model (proxy);
}
static void
move_left_activate_cb (GtkWidget *menu, GtkWidget *proxy)
{
move_in_model (proxy, -1);
}
static void
move_right_activate_cb (GtkWidget *menu, GtkWidget *proxy)
{
move_in_model (proxy, +1);
}
static GtkWidget *
add_open_in_tabs_menu (EphyTopicAction *action,
GtkWidget *menu,
EphyNode *node)
{
GtkWidget *item;
const char *label;
label = ngettext ("Open in New _Tab",
"Open in New _Tabs",
ephy_node_get_n_children (node));
item = gtk_menu_item_new_with_mnemonic (label);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set_data (G_OBJECT (item), TOPIC_NODE_DATA_KEY, node);
g_signal_connect (item, "activate",
G_CALLBACK (open_in_tabs_activate_cb), action);
return item;
}
static GtkWidget *
build_bookmarks_menu (EphyTopicAction *action, EphyNode *node)
{
GtkWidget *menu;
menu = gtk_menu_new ();
append_bookmarks_menu (action, menu, node, TRUE);
if (can_open_in_tabs (node))
{
GtkWidget *item;
item = gtk_separator_menu_item_new ();
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
add_open_in_tabs_menu (action, menu, node);
}
return menu;
}
static int
sort_topics (gconstpointer a, gconstpointer b)
{
EphyNode *node_a = (EphyNode *)a;
EphyNode *node_b = (EphyNode *)b;
const char *title1, *title2;
int retval;
title1 = ephy_node_get_property_string (node_a, EPHY_NODE_KEYWORD_PROP_NAME);
title2 = ephy_node_get_property_string (node_b, EPHY_NODE_KEYWORD_PROP_NAME);
if (title1 == NULL)
{
retval = -1;
}
else if (title2 == NULL)
{
retval = 1;
}
else
{
char *str_a, *str_b;
str_a = g_utf8_casefold (title1, -1);
str_b = g_utf8_casefold (title2, -1);
retval = g_utf8_collate (str_a, str_b);
g_free (str_a);
g_free (str_b);
}
return retval;
}
static GtkWidget *
build_topics_menu (EphyTopicAction *action)
{
GtkWidget *menu, *item;
GPtrArray *children;
int i;
EphyBookmarks *bookmarks;
EphyNode *all, *uncategorized, *node;
EphyNodePriority priority;
GList *node_list = NULL, *l = NULL;
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
all = ephy_bookmarks_get_bookmarks (bookmarks);
node = ephy_bookmarks_get_keywords (bookmarks);
menu = gtk_menu_new ();
children = ephy_node_get_children (node);
for (i = 0; i < children->len; ++i)
{
EphyNode *kid;
kid = g_ptr_array_index (children, i);
priority = ephy_node_get_property_int
(kid, EPHY_NODE_KEYWORD_PROP_PRIORITY);
if (priority == EPHY_NODE_NORMAL_PRIORITY)
{
node_list = g_list_prepend (node_list, kid);
}
}
node_list = g_list_sort (node_list, (GCompareFunc)sort_topics);
for (l = g_list_first (node_list); l != NULL; l = g_list_next (l))
{
EphyNode *kid;
const char *title;
GtkWidget *bmk_menu;
GtkLabel *label;
kid = (EphyNode*)l->data;
if (kid == all) continue;
title = ephy_node_get_property_string
(kid, EPHY_NODE_KEYWORD_PROP_NAME);
item = gtk_image_menu_item_new_with_label (title);
label = (GtkLabel *) ((GtkBin *) item)->child;
gtk_label_set_max_width_chars (label, MENUITEM_WIDTH_CHARS);
gtk_label_set_ellipsize (label, PANGO_ELLIPSIZE_END);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
bmk_menu = build_bookmarks_menu (action, kid);
gtk_widget_show (bmk_menu);
gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), bmk_menu);
}
g_list_free (node_list);
uncategorized = ephy_bookmarks_get_not_categorized (bookmarks);
append_bookmarks_menu (action, menu, uncategorized, FALSE);
return menu;
}
static GtkWidget *
build_menu (EphyTopicAction *action)
{
if (ephy_node_get_id (action->priv->topic_node) == BOOKMARKS_NODE_ID)
{
return build_topics_menu (action);
}
else
{
return build_bookmarks_menu (action, action->priv->topic_node);
}
}
static void
drag_data_get_cb (GtkWidget *widget, GdkDragContext *context,
GtkSelectionData *selection_data, guint info,
guint32 time, EphyTopicAction *action)
{
EphyBookmarks *bookmarks;
char *uri;
bookmarks = ephy_shell_get_bookmarks (ephy_shell);
g_return_if_fail (bookmarks != NULL);
uri = ephy_bookmarks_get_topic_uri (bookmarks, action->priv->topic_node);
g_return_if_fail (uri != NULL);
gtk_selection_data_set (selection_data, selection_data->target, 8,
(unsigned char *) uri, strlen (uri));
g_free (uri);
}
static void
drag_data_delete_cb (GtkWidget *widget, GdkDragContext *context,
EphyTopicAction *action)
{
remove_from_model (widget);
}
static void
stop_drag_check (EphyTopicAction *action, GtkWidget *widget)
{
if (action->priv->motion_handler)
{
g_signal_handler_disconnect (widget, action->priv->motion_handler);
action->priv->motion_handler = 0;
g_signal_handler_disconnect (widget, action->priv->release_handler);
action->priv->release_handler = 0;
}
}
static gboolean
check_horizontal_threshold (GtkWidget *widget, gint start_x, gint start_y,
gint current_x, gint current_y)
{
gint drag_threshold;
g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE);
g_object_get (gtk_widget_get_settings (widget),
"gtk-dnd-drag-threshold", &drag_threshold,
NULL);
return (ABS (current_x - start_x) > drag_threshold &&
ABS (current_y - start_y) < drag_threshold);
}
static gboolean
drag_motion_cb (GtkWidget *widget, GdkEventMotion *event, EphyTopicAction *action)
{
GtkWidget *button, *event_widget;
event_widget = gtk_get_event_widget ((GdkEvent*) event);
button = GTK_WIDGET (g_object_get_data (G_OBJECT (widget), "button"));
if (!gtk_widget_is_ancestor (event_widget, widget) &&
check_horizontal_threshold (widget, action->priv->drag_x,
action->priv->drag_y, event->x, event->y))
{
GtkTargetList *target_list;
target_list = gtk_target_list_new (drag_targets, n_drag_targets);
stop_drag_check (action, widget);
gtk_menu_popdown (GTK_MENU (widget));
gtk_drag_begin (button, target_list, GDK_ACTION_MOVE |
GDK_ACTION_COPY, 1, (GdkEvent*)event);
gtk_target_list_unref (target_list);
}
return TRUE;
}
static void
button_toggled_cb (GtkWidget *button,
EphyTopicAction *action)
{
if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button)))
{
GtkWidget *menu;
GdkEvent *event;
guint32 event_time = 0;
guint event_button = 0;
menu = build_menu (action);
g_signal_connect_object (menu, "deactivate",
G_CALLBACK (menu_deactivate_cb), button, 0);
event = gtk_get_current_event ();
if (event != NULL)
{
if (event->type == GDK_BUTTON_PRESS)
{
event_button = ((GdkEventButton *) event)->button;
event_time = ((GdkEventButton *) event)->time;
}
gdk_event_free (event);
}
if (event_button == 0)
{
gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
ephy_gui_menu_position_under_widget,
button, 0 , gtk_get_current_event_time ());
gtk_menu_shell_select_first (GTK_MENU_SHELL (menu), FALSE);
}
else
{
gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
ephy_gui_menu_position_under_widget,
button, event_button, event_time);
}
g_object_set_data (G_OBJECT (button), "popup", menu);
}
}
static GtkWidget *
create_menu_item (GtkAction *action)
{
GtkWidget *menu, *menu_item;
GValue value = { 0, };
const char *title;
g_value_init (&value, G_TYPE_STRING);
g_object_get_property (G_OBJECT (action), "label", &value);
title = g_value_get_string (&value);
menu_item = gtk_menu_item_new_with_label (title);
g_value_unset (&value);
menu = build_menu (EPHY_TOPIC_ACTION (action));
gtk_widget_show (menu);
gtk_menu_item_set_submenu (GTK_MENU_ITEM (menu_item), menu);
return menu_item;
}
static void
show_context_menu (EphyTopicAction *action,
GtkWidget *proxy,
GdkEventButton *event,
GtkMenuPositionFunc func)
{
GtkWidget *menu, *item, *image;
gboolean last;
menu = gtk_menu_new ();
item = add_open_in_tabs_menu (action, menu, action->priv->topic_node);
gtk_widget_set_sensitive (item, can_open_in_tabs (action->priv->topic_node));
item = gtk_separator_menu_item_new ();
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
item = gtk_image_menu_item_new_with_mnemonic (_("_Remove from Toolbar"));
gtk_widget_show (item);
image = gtk_image_new_from_stock (GTK_STOCK_REMOVE, GTK_ICON_SIZE_MENU);
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
gtk_widget_show (image);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_signal_connect (item, "activate",
G_CALLBACK (remove_activate_cb), proxy);
item = gtk_separator_menu_item_new ();
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
item = gtk_menu_item_new_with_mnemonic (_("Move _Left"));
gtk_widget_set_sensitive (item, get_item_position (proxy, NULL) > 0);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_signal_connect (item, "activate",
G_CALLBACK (move_left_activate_cb), proxy);
item = gtk_menu_item_new_with_mnemonic (_("Move Ri_ght"));
get_item_position (proxy, &last);
gtk_widget_set_sensitive (item, !last);
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_signal_connect (item, "activate",
G_CALLBACK (move_right_activate_cb), proxy);
if (event != NULL)
{
gtk_menu_popup (GTK_MENU (menu), NULL, NULL, func, proxy,
event->button, event->time);
}
else
{
gtk_menu_popup (GTK_MENU (menu), NULL, NULL, func, proxy, 0,
gtk_get_current_event_time ());
gtk_menu_shell_select_first (GTK_MENU_SHELL (menu), FALSE);
}
}
static gboolean
popup_menu_cb (GtkWidget *widget, EphyTopicAction *action)
{
if (gtk_widget_get_ancestor (widget, EPHY_TYPE_BOOKMARKSBAR))
{
show_context_menu (action, widget, NULL,
ephy_gui_menu_position_under_widget);
return TRUE;
}
return FALSE;
}
static gboolean
button_release_cb (GtkWidget *widget,
GdkEventButton *event,
EphyTopicAction *action)
{
if (event->button == 1)
{
stop_drag_check (action, widget);
if (GTK_IS_TOGGLE_BUTTON (widget))
{
gtk_toggle_button_set_active
(GTK_TOGGLE_BUTTON (widget), FALSE);
}
}
return FALSE;
}
static gboolean
button_press_cb (GtkWidget *widget,
GdkEventButton *event,
EphyTopicAction *action)
{
if (event->button == 1 &&
gtk_widget_get_ancestor (widget, EPHY_TYPE_BOOKMARKSBAR))
{
if (!gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (widget)))
{
GtkWidget *menu;
gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (widget), TRUE);
menu = g_object_get_data (G_OBJECT (widget), "popup");
g_return_val_if_fail (menu != NULL, FALSE);
g_object_set_data (G_OBJECT (menu), "button", widget);
action->priv->drag_x = event->x;
action->priv->drag_y = event->y;
action->priv->motion_handler = g_signal_connect
(menu, "motion_notify_event",
G_CALLBACK (drag_motion_cb), action);
action->priv->release_handler = g_signal_connect
(menu, "button_release_event",
G_CALLBACK (button_release_cb), action);
return TRUE;
}
}
else if (event->button == 3 &&
gtk_widget_get_ancestor (widget, EPHY_TYPE_BOOKMARKSBAR))
{
show_context_menu (action, widget, event, NULL);
return TRUE;
}
return FALSE;
}
static void
connect_proxy (GtkAction *action, GtkWidget *proxy)
{
GtkWidget *button;
LOG ("connect_proxy action %p, proxy %p", action, proxy);
(* GTK_ACTION_CLASS (parent_class)->connect_proxy) (action, proxy);
if (GTK_IS_TOOL_ITEM (proxy))
{
ephy_topic_action_sync_label (action, NULL, proxy);
g_signal_connect_object (action, "notify::label",
G_CALLBACK (ephy_topic_action_sync_label), proxy, 0);
button = GTK_WIDGET (g_object_get_data (G_OBJECT (proxy), "button"));
g_signal_connect (button, "toggled",
G_CALLBACK (button_toggled_cb), action);
g_signal_connect (button, "popup_menu",
G_CALLBACK (popup_menu_cb), action);
g_signal_connect (button, "button-press-event",
G_CALLBACK (button_press_cb), action);
g_signal_connect (button, "button-release-event",
G_CALLBACK (button_release_cb), action);
g_signal_connect (button, "drag_data_get",
G_CALLBACK (drag_data_get_cb), action);
g_signal_connect (button, "drag_data_delete",
G_CALLBACK (drag_data_delete_cb), action);
}
else if (GTK_IS_MENU_ITEM (proxy))
{
GtkLabel *label;
label = (GtkLabel *) ((GtkBin *) proxy)->child;
gtk_label_set_use_underline (label, FALSE);
gtk_label_set_ellipsize (label, PANGO_ELLIPSIZE_END);
gtk_label_set_max_width_chars (label, LABEL_WIDTH_CHARS);
}
}
static void
topic_changed_cb (EphyNode *node,
guint property_id,
EphyTopicAction *action)
{
if (property_id == EPHY_NODE_KEYWORD_PROP_NAME)
{
GValue value = { 0, };
const char *title;
int priority;
priority = ephy_node_get_property_int
(node, EPHY_NODE_KEYWORD_PROP_PRIORITY);
if (priority == EPHY_NODE_ALL_PRIORITY)
{
title = _("Bookmarks");
}
else
{
title = ephy_node_get_property_string
(node, EPHY_NODE_KEYWORD_PROP_NAME);
}
g_value_init(&value, G_TYPE_STRING);
g_value_set_static_string (&value, title);
g_object_set_property (G_OBJECT (action), "label", &value);
g_value_unset (&value);
}
}
static void
ephy_topic_action_set_topic (EphyTopicAction *action,
EphyNode *node)
{
action->priv->topic_node = node;
topic_changed_cb (node, EPHY_NODE_KEYWORD_PROP_NAME, action);
ephy_node_signal_connect_object (node, EPHY_NODE_CHANGED,
(EphyNodeCallback) topic_changed_cb,
G_OBJECT (action));
}
static void
ephy_topic_action_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
EphyTopicAction *topic;
topic = EPHY_TOPIC_ACTION (object);
switch (prop_id)
{
case PROP_TOPIC:
ephy_topic_action_set_topic (topic, g_value_get_pointer (value));
break;
}
}
static void
ephy_topic_action_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
/* no readable properties */
g_return_if_reached ();
}
static void
ephy_topic_action_class_init (EphyTopicActionClass *class)
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
GtkActionClass *action_class = GTK_ACTION_CLASS (class);
parent_class = g_type_class_peek_parent (class);
action_class->toolbar_item_type = GTK_TYPE_TOOL_ITEM;
action_class->create_tool_item = create_tool_item;
action_class->create_menu_item = create_menu_item;
action_class->connect_proxy = connect_proxy;
object_class->set_property = ephy_topic_action_set_property;
object_class->get_property = ephy_topic_action_get_property;
g_object_class_install_property
(object_class,
PROP_TOPIC,
g_param_spec_pointer ("topic",
"Topic",
"Topic",
G_PARAM_WRITABLE |
G_PARAM_CONSTRUCT_ONLY));
g_type_class_add_private (object_class, sizeof(EphyTopicActionPrivate));
}
static void
ephy_topic_action_init (EphyTopicAction *action)
{
action->priv = EPHY_TOPIC_ACTION_GET_PRIVATE (action);
}
GtkAction *
ephy_topic_action_new (const char *name,
EphyNode *node)
{
return g_object_new (EPHY_TYPE_TOPIC_ACTION,
"name", name,
"topic", node,
NULL);
}