aboutsummaryrefslogtreecommitdiffstats
path: root/lib
diff options
context:
space:
mode:
Diffstat (limited to 'lib')
-rw-r--r--lib/ephy-marshal.list1
-rw-r--r--lib/ephy-signal-accumulator.c11
-rw-r--r--lib/ephy-signal-accumulator.h4
-rw-r--r--lib/widgets/ephy-location-entry.c156
-rw-r--r--lib/widgets/ephy-location-entry.h11
5 files changed, 154 insertions, 29 deletions
diff --git a/lib/ephy-marshal.list b/lib/ephy-marshal.list
index 8d61a9a54..3bfa98df3 100644
--- a/lib/ephy-marshal.list
+++ b/lib/ephy-marshal.list
@@ -3,6 +3,7 @@ BOOLEAN:STRING,STRING
BOOLEAN:VOID
OBJECT:FLAGS
OBJECT:STRING,OBJECT,FLAGS
+STRING:VOID
VOID:POINTER,BOOLEAN
VOID:POINTER,POINTER
VOID:STRING,FLAGS
diff --git a/lib/ephy-signal-accumulator.c b/lib/ephy-signal-accumulator.c
index d34273721..eb6a05562 100644
--- a/lib/ephy-signal-accumulator.c
+++ b/lib/ephy-signal-accumulator.c
@@ -48,3 +48,14 @@ ephy_signal_accumulator_object (GSignalInvocationHint *ihint,
return TRUE;
}
+
+gboolean
+ephy_signal_accumulator_string (GSignalInvocationHint *ihint,
+ GValue *return_accu,
+ const GValue *handler_return,
+ gpointer accu_data)
+{
+ g_value_copy (handler_return, return_accu);
+
+ return g_value_get_string (handler_return) == NULL;
+}
diff --git a/lib/ephy-signal-accumulator.h b/lib/ephy-signal-accumulator.h
index 8991bcf12..b62e7d610 100644
--- a/lib/ephy-signal-accumulator.h
+++ b/lib/ephy-signal-accumulator.h
@@ -30,6 +30,10 @@ gboolean ephy_signal_accumulator_object (GSignalInvocationHint *ihint,
const GValue *handler_return,
gpointer accu_data);
+gboolean ephy_signal_accumulator_string (GSignalInvocationHint *ihint,
+ GValue *return_accu,
+ const GValue *handler_return,
+ gpointer accu_data);
G_END_DECLS
diff --git a/lib/widgets/ephy-location-entry.c b/lib/widgets/ephy-location-entry.c
index 0dbb7192e..889b23003 100644
--- a/lib/widgets/ephy-location-entry.c
+++ b/lib/widgets/ephy-location-entry.c
@@ -26,9 +26,12 @@
#include "ephy-tree-model-node.h"
#include "ephy-location-entry.h"
#include "ephy-marshal.h"
-#include "ephy-debug.h"
+#include "ephy-signal-accumulator.h"
+#include "ephy-dnd.h"
#include "egg-editable-toolbar.h"
+#include "ephy-debug.h"
+#include <glib/gi18n.h>
#include <gtk/gtktoolbar.h>
#include <gtk/gtkentry.h>
#include <gtk/gtkwindow.h>
@@ -37,9 +40,11 @@
#include <gtk/gtktreemodelsort.h>
#include <gtk/gtkstock.h>
#include <gtk/gtkimage.h>
+#include <gtk/gtkeventbox.h>
+#include <gtk/gtkbox.h>
+#include <gtk/gtkhbox.h>
#include <gtk/gtkimagemenuitem.h>
#include <gtk/gtkseparatormenuitem.h>
-#include <glib/gi18n.h>
#include <string.h>
@@ -48,6 +53,8 @@
struct _EphyLocationEntryPrivate
{
GtkWidget *entry;
+ GtkWidget *icon_ebox;
+ GtkWidget *icon;
char *before_completion;
gboolean user_changed;
@@ -71,17 +78,27 @@ web_prefixes [] =
{ "www.", 4 }
};
+static GtkTargetEntry url_drag_types [] =
+{
+ { EPHY_DND_URI_LIST_TYPE, 0, 0 },
+ { EPHY_DND_TEXT_TYPE, 0, 1 },
+ { EPHY_DND_URL_TYPE, 0, 2 }
+};
+static int n_url_drag_types = G_N_ELEMENTS (url_drag_types);
+
static void ephy_location_entry_class_init (EphyLocationEntryClass *klass);
static void ephy_location_entry_init (EphyLocationEntry *le);
static GObjectClass *parent_class = NULL;
-enum EphyLocationEntrySignalsEnum
+enum signalsEnum
{
USER_CHANGED,
+ GET_LOCATION,
+ GET_TITLE,
LAST_SIGNAL
};
-static gint EphyLocationEntrySignals[LAST_SIGNAL];
+static gint signals[LAST_SIGNAL];
enum
{
@@ -133,9 +150,9 @@ ephy_location_entry_set_tooltip (GtkToolItem *tool_item,
{
EphyLocationEntry *entry = EPHY_LOCATION_ENTRY (tool_item);
- g_return_val_if_fail (EPHY_IS_LOCATION_ENTRY (entry), FALSE);
-
gtk_tooltips_set_tip (tooltips, entry->priv->entry, tip_text, tip_private);
+ gtk_tooltips_set_tip (tooltips, entry->priv->icon_ebox,
+ _("Drag and drop this icon to create a link to this page"), NULL);
return TRUE;
}
@@ -150,9 +167,9 @@ ephy_location_entry_class_init (EphyLocationEntryClass *klass)
tool_item_class->set_tooltip = ephy_location_entry_set_tooltip;
- EphyLocationEntrySignals[USER_CHANGED] = g_signal_new (
+ signals[USER_CHANGED] = g_signal_new (
"user_changed", G_OBJECT_CLASS_TYPE (klass),
- G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST | G_SIGNAL_RUN_CLEANUP,
+ G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyLocationEntryClass, user_changed),
NULL, NULL,
g_cclosure_marshal_VOID__VOID,
@@ -160,6 +177,27 @@ ephy_location_entry_class_init (EphyLocationEntryClass *klass)
0,
G_TYPE_NONE);
+ signals[GET_LOCATION] = g_signal_new (
+ "get-location", G_OBJECT_CLASS_TYPE (klass),
+ G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (EphyLocationEntryClass, get_location),
+ ephy_signal_accumulator_string, NULL,
+ ephy_marshal_STRING__VOID,
+ G_TYPE_STRING,
+ 0,
+ G_TYPE_NONE);
+
+ signals[GET_TITLE] = g_signal_new (
+ "get-title", G_OBJECT_CLASS_TYPE (klass),
+ G_SIGNAL_RUN_FIRST | G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET (EphyLocationEntryClass, get_title),
+ ephy_signal_accumulator_string, NULL,
+ ephy_marshal_STRING__VOID,
+ G_TYPE_STRING,
+ 0,
+ G_TYPE_NONE);
+
+
g_type_class_add_private (object_class, sizeof (EphyLocationEntryPrivate));
}
@@ -170,7 +208,7 @@ editable_changed_cb (GtkEditable *editable, EphyLocationEntry *e)
if (p->user_changed)
{
- g_signal_emit (e, EphyLocationEntrySignals[USER_CHANGED], 0);
+ g_signal_emit (e, signals[USER_CHANGED], 0);
}
}
@@ -382,27 +420,77 @@ entry_populate_popup_cb (GtkEntry *entry,
}
static void
-ephy_location_entry_construct_contents (EphyLocationEntry *le)
+each_url_get_data_binder (EphyDragEachSelectedItemDataGet iteratee,
+ gpointer iterator_context,
+ gpointer return_data)
{
- EphyLocationEntryPrivate *p = le->priv;
+ EphyLocationEntry *entry = EPHY_LOCATION_ENTRY (iterator_context);
+ char *title = NULL, *address = NULL;
+
+ g_signal_emit (entry, signals[GET_LOCATION], 0, &address);
+ g_signal_emit (entry, signals[GET_TITLE], 0, &title);
+ g_return_if_fail (address != NULL && title != NULL);
+
+ iteratee (address, title, return_data);
- LOG ("EphyLocationEntry constructing contents %p", le)
-
- p->entry = gtk_entry_new ();
-
- gtk_container_add (GTK_CONTAINER (le), p->entry);
- gtk_widget_show (p->entry);
-
- g_signal_connect (p->entry, "populate_popup",
- G_CALLBACK (entry_populate_popup_cb), le);
- g_signal_connect (p->entry, "button_press_event",
- G_CALLBACK (entry_button_press_cb), le);
- g_signal_connect (p->entry, "changed",
- G_CALLBACK (editable_changed_cb), le);
- g_signal_connect (p->entry, "drag_motion",
- G_CALLBACK (entry_drag_motion_cb), le);
- g_signal_connect (p->entry, "drag_drop",
- G_CALLBACK (entry_drag_drop_cb), le);
+ g_free (address);
+ g_free (title);
+}
+
+static void
+favicon_drag_data_get_cb (GtkWidget *widget,
+ GdkDragContext *context,
+ GtkSelectionData *selection_data,
+ guint info,
+ guint32 time,
+ EphyLocationEntry *entry)
+{
+ g_assert (widget != NULL);
+ g_return_if_fail (context != NULL);
+
+ ephy_dnd_drag_data_get (widget, context, selection_data,
+ time, entry, each_url_get_data_binder);
+}
+
+static void
+ephy_location_entry_construct_contents (EphyLocationEntry *entry)
+{
+ EphyLocationEntryPrivate *priv = entry->priv;
+ GtkWidget *hbox;
+
+ LOG ("EphyLocationEntry constructing contents %p", entry)
+
+ hbox = gtk_hbox_new (FALSE, 3); /* FIXME themeable spacing? */
+ gtk_container_add (GTK_CONTAINER (entry), hbox);
+
+ priv->icon_ebox = gtk_event_box_new ();
+ gtk_container_set_border_width (GTK_CONTAINER (priv->icon_ebox), 2);
+ gtk_event_box_set_visible_window (GTK_EVENT_BOX (priv->icon_ebox), FALSE);
+ gtk_box_pack_start (GTK_BOX (hbox), priv->icon_ebox, FALSE, FALSE, 0);
+ gtk_drag_source_set (priv->icon_ebox, GDK_BUTTON1_MASK,
+ url_drag_types, n_url_drag_types,
+ GDK_ACTION_COPY);
+
+ priv->icon = gtk_image_new ();
+ gtk_container_add (GTK_CONTAINER (priv->icon_ebox), priv->icon);
+
+ priv->entry = gtk_entry_new ();
+ gtk_box_pack_start (GTK_BOX (hbox), priv->entry, TRUE, TRUE, 0);
+
+ gtk_widget_show_all (hbox);
+
+ g_signal_connect (priv->icon_ebox, "drag_data_get",
+ G_CALLBACK (favicon_drag_data_get_cb), entry);
+ g_signal_connect (priv->entry, "populate_popup",
+ G_CALLBACK (entry_populate_popup_cb), entry);
+ g_signal_connect (priv->entry, "button_press_event",
+ G_CALLBACK (entry_button_press_cb), entry);
+ g_signal_connect (priv->entry, "changed",
+ G_CALLBACK (editable_changed_cb), entry);
+ g_signal_connect (priv->entry, "drag_motion",
+ G_CALLBACK (entry_drag_motion_cb), entry);
+ g_signal_connect (priv->entry, "drag_drop",
+ G_CALLBACK (entry_drag_drop_cb), entry);
}
static void
@@ -518,3 +606,15 @@ ephy_location_entry_activate (EphyLocationEntry *le)
gtk_window_set_focus (GTK_WINDOW(toplevel),
le->priv->entry);
}
+
+GtkWidget *
+ephy_location_entry_get_entry (EphyLocationEntry *entry)
+{
+ return entry->priv->entry;
+}
+
+GtkWidget *
+ephy_location_entry_get_image (EphyLocationEntry *entry)
+{
+ return entry->priv->icon;
+}
diff --git a/lib/widgets/ephy-location-entry.h b/lib/widgets/ephy-location-entry.h
index 08446b0d5..c4a44addc 100644
--- a/lib/widgets/ephy-location-entry.h
+++ b/lib/widgets/ephy-location-entry.h
@@ -25,6 +25,7 @@
#include "ephy-node.h"
+#include <gtk/gtkwidget.h>
#include <gtk/gtktoolitem.h>
#include <gtk/gtktreemodel.h>
@@ -45,7 +46,11 @@ struct _EphyLocationEntryClass
{
GtkToolItemClass parent_class;
- void (*user_changed) (EphyLocationEntry *le);
+ /* Signals */
+ void (*user_changed) (EphyLocationEntry *entry);
+ /* for getting the drag data */
+ char * (* get_location) (EphyLocationEntry *entry);
+ char * (* get_title) (EphyLocationEntry *entry);
};
struct _EphyLocationEntry
@@ -74,6 +79,10 @@ const char *ephy_location_entry_get_location (EphyLocationEntry *le);
void ephy_location_entry_activate (EphyLocationEntry *le);
+GtkWidget *ephy_location_entry_get_entry (EphyLocationEntry *entry);
+
+GtkWidget *ephy_location_entry_get_image (EphyLocationEntry *entry);
+
G_END_DECLS
#endif