aboutsummaryrefslogtreecommitdiffstats
path: root/shell/e-storage-set-view.c
diff options
context:
space:
mode:
Diffstat (limited to 'shell/e-storage-set-view.c')
-rw-r--r--shell/e-storage-set-view.c194
1 files changed, 193 insertions, 1 deletions
diff --git a/shell/e-storage-set-view.c b/shell/e-storage-set-view.c
index 8dcc988234..dc299cb787 100644
--- a/shell/e-storage-set-view.c
+++ b/shell/e-storage-set-view.c
@@ -45,6 +45,10 @@ static char *list [] = {
};
#endif
+#define DRAG_RESISTANCE 3 /* FIXME hardcoded in ETable to this value as
+ * well, and there is no way for us to use the
+ * same value as it's not exported. */
+
#define ETABLE_SPEC "<ETableSpecification no-headers=\"true\" selection-mode=\"single\" cursor-mode=\"line\" draw-grid=\"true\" horizontal-scrolling=\"true\"> \
<ETableColumn model_col=\"0\" _title=\"Folder\" expansion=\"1.0\" minimum_width=\"20\" resizable=\"true\" cell=\"render_tree\" compare=\"string\"/> \
<ETableState> \
@@ -71,6 +75,9 @@ struct _EStorageSetViewPrivate {
const char *selected_row_path;
gboolean show_folders;
+
+ int drag_x, drag_y;
+ int drag_column, drag_row;
};
@@ -254,6 +261,100 @@ marshal_NONE__GDKDRAGCONTEXT_STRING_STRING_STRING (GtkObject *object,
/* DnD selection setup stuff. */
+/* This will create an array of GtkTargetEntries from the specified list of DND
+ types. The type name will *not* be allocated in the list, as this is
+ supposed to be used only temporarily to set up the cell as a drag source. */
+static GtkTargetEntry *
+create_target_entries_from_dnd_type_list (GList *dnd_types,
+ int *num_entries_return)
+{
+ GtkTargetEntry *entries;
+ GList *p;
+ int num_entries;
+ int i;
+
+ if (dnd_types == NULL)
+ return NULL;
+
+ num_entries = g_list_length (dnd_types);
+ if (num_entries == 0)
+ return NULL;
+
+ entries = g_new (GtkTargetEntry, num_entries);
+
+ for (p = dnd_types, i = 0; p != NULL; p = p->next, i++) {
+ const char *dnd_type;
+
+ g_assert (i < num_entries);
+
+ dnd_type = (const char *) p->data;
+
+ entries[i].target = (char *) dnd_type;
+ entries[i].flags = 0;
+ entries[i].info = i;
+ }
+
+ *num_entries_return = num_entries;
+ return entries;
+}
+
+static void
+free_target_entries (GtkTargetEntry *entries)
+{
+ g_assert (entries != NULL);
+
+ /* The target names are not strdup()ed so a simple free will do. */
+ g_free (entries);
+}
+
+static GtkTargetList *
+create_target_list_for_row (EStorageSetView *storage_set_view,
+ int row)
+{
+ EStorageSetViewPrivate *priv;
+ GtkTargetList *target_list;
+ EFolderTypeRegistry *folder_type_registry;
+ ETreePath *folder_node_path;
+ EFolder *folder;
+ const char *folder_path;
+ const char *folder_type;
+ GList *exported_dnd_types;
+ GtkTargetEntry *target_entries;
+ int num_target_entries;
+
+ priv = storage_set_view->priv;
+
+ target_list = gtk_target_list_new (NULL, 0);
+
+ folder_type_registry = e_storage_set_get_folder_type_registry (priv->storage_set);
+
+ folder_node_path = e_tree_model_node_at_row (priv->etree_model, row);
+ g_assert (folder_node_path != NULL);
+
+ folder_path = e_tree_model_node_get_data (priv->etree_model, folder_node_path);
+ g_assert (folder_path != NULL);
+
+ folder = e_storage_set_get_folder (priv->storage_set, folder_path);
+ g_assert (folder != NULL);
+
+ folder_type = e_folder_get_type_string (folder);
+
+ exported_dnd_types = e_folder_type_registry_get_exported_dnd_types_for_type (folder_type_registry,
+ folder_type);
+ if (exported_dnd_types == NULL)
+ return NULL;
+
+ target_entries = create_target_entries_from_dnd_type_list (exported_dnd_types,
+ &num_target_entries);
+ g_assert (target_entries != NULL);
+
+ target_list = gtk_target_list_new (target_entries, num_target_entries);
+
+ free_target_entries (target_entries);
+
+ return NULL;
+}
+
static void
set_uri_list_selection (EStorageSetView *storage_set_view,
GtkSelectionData *selection_data)
@@ -439,7 +540,87 @@ destroy (GtkObject *object)
}
-/* ETable methods */
+/* GtkWidget methods. */
+
+static int
+button_press_event (GtkWidget *widget,
+ GdkEventButton *event)
+{
+ EStorageSetView *storage_set_view;
+ EStorageSetViewPrivate *priv;
+ ETable *table;
+ int row, column;
+
+ storage_set_view = E_STORAGE_SET_VIEW (widget);
+ priv = storage_set_view->priv;
+
+ table = E_TABLE (widget);
+
+ /* FIXME correct? */
+ if (GTK_WIDGET_CLASS (parent_class)->button_press_event != NULL)
+ (* GTK_WIDGET_CLASS (parent_class)->button_press_event) (widget, event);
+
+ if (event->button != 1)
+ return FALSE;
+
+ e_table_get_cell_at (table, event->x, event->y, &row, &column);
+
+ g_print ("e-storage-set-view.c::button_press_event() -- row %d column %d\n", row, column);
+
+ priv->drag_x = event->x;
+ priv->drag_y = event->y;
+ priv->drag_column = column;
+ priv->drag_row = row;
+
+ /* FIXME correct? */
+ return TRUE;
+}
+
+static int
+motion_notify_event (GtkWidget *widget,
+ GdkEventMotion *event)
+{
+ EStorageSetView *storage_set_view;
+ EStorageSetViewPrivate *priv;
+ ETable *table;
+ GtkTargetList *target_list;
+ GdkDragAction actions;
+ GdkDragContext *context;
+
+ storage_set_view = E_STORAGE_SET_VIEW (widget);
+ priv = storage_set_view->priv;
+
+ table = E_TABLE (widget);
+
+ /* FIXME correct? */
+ if (GTK_WIDGET_CLASS (parent_class)->motion_notify_event != NULL)
+ (* GTK_WIDGET_CLASS (parent_class)->motion_notify_event) (widget, event);
+
+ /* FIXME correct? */
+ if (! (event->state & GDK_BUTTON1_MASK))
+ return FALSE;
+
+ if (ABS (priv->drag_x - event->x) < DRAG_RESISTANCE
+ && ABS (priv->drag_y - event->y) < DRAG_RESISTANCE)
+ return FALSE;
+
+ target_list = create_target_list_for_row (storage_set_view, priv->drag_row);
+ if (target_list == NULL)
+ return FALSE;
+
+ actions = GDK_ACTION_MOVE | GDK_ACTION_COPY;
+
+ context = e_table_drag_begin (table,
+ priv->drag_row, priv->drag_column,
+ target_list, actions,
+ 1, (GdkEvent *) event);
+ gtk_drag_set_icon_default (context);
+
+ return FALSE;
+}
+
+
+/* ETable methods. */
static void
table_drag_begin (ETable *etable,
@@ -870,6 +1051,7 @@ static void
class_init (EStorageSetViewClass *klass)
{
GtkObjectClass *object_class;
+ GtkWidgetClass *widget_class;
ETableClass *etable_class;
parent_class = gtk_type_class (e_table_get_type ());
@@ -877,6 +1059,10 @@ class_init (EStorageSetViewClass *klass)
object_class = GTK_OBJECT_CLASS (klass);
object_class->destroy = destroy;
+ widget_class = GTK_WIDGET_CLASS (klass);
+ widget_class->button_press_event = button_press_event;
+ widget_class->motion_notify_event = motion_notify_event;
+
etable_class = E_TABLE_CLASS (klass);
etable_class->right_click = right_click;
etable_class->cursor_activated = cursor_activated;
@@ -936,6 +1122,10 @@ init (EStorageSetView *storage_set_view)
priv->type_name_to_pixbuf = g_hash_table_new (g_str_hash, g_str_equal);
priv->selected_row_path = NULL;
priv->show_folders = TRUE;
+ priv->drag_x = 0;
+ priv->drag_y = 0;
+ priv->drag_column = 0;
+ priv->drag_row = 0;
storage_set_view->priv = priv;
}
@@ -1117,6 +1307,7 @@ e_storage_set_view_construct (EStorageSetView *storage_set_view,
ETABLE_SPEC, NULL);
gtk_object_unref (GTK_OBJECT (extras));
+#if 0
e_table_drag_source_set (E_TABLE (storage_set_view), GDK_BUTTON1_MASK,
source_drag_types, num_source_drag_types,
GDK_ACTION_MOVE | GDK_ACTION_COPY);
@@ -1124,6 +1315,7 @@ e_storage_set_view_construct (EStorageSetView *storage_set_view,
e_table_drag_dest_set (E_TABLE (storage_set_view), GTK_DEST_DEFAULT_ALL,
source_drag_types, num_source_drag_types,
GDK_ACTION_MOVE | GDK_ACTION_COPY);
+#endif
gtk_object_ref (GTK_OBJECT (storage_set));
priv->storage_set = storage_set;