diff options
Diffstat (limited to 'libnautilus-private/nautilus-tree-view-drag-dest.c')
-rw-r--r-- | libnautilus-private/nautilus-tree-view-drag-dest.c | 686 |
1 files changed, 686 insertions, 0 deletions
diff --git a/libnautilus-private/nautilus-tree-view-drag-dest.c b/libnautilus-private/nautilus-tree-view-drag-dest.c new file mode 100644 index 000000000..96ffb5cd0 --- /dev/null +++ b/libnautilus-private/nautilus-tree-view-drag-dest.c @@ -0,0 +1,686 @@ +/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ + +/* + * Nautilus + * + * Copyright (C) 2002 Sun Microsystems, Inc. + * + * Nautilus 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 of the + * License, or (at your option) any later version. + * + * Nautilus 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. + * + * Author: Dave Camp <dave@ximian.com> + */ + +/* nautilus-tree-view-drag-dest.c: Handles drag and drop for treeviews which + * contain a hierarchy of files + */ + +#include <config.h> +#include "nautilus-tree-view-drag-dest.h" + +#include <eel/eel-gtk-macros.h> +#include <gtk/gtkmain.h> +#include <gtk/gtksignal.h> +#include <libgnome/gnome-macros.h> +#include <libgnomevfs/gnome-vfs-utils.h> +#include "nautilus-file-dnd.h" +#include "nautilus-icon-dnd.h" +#include "nautilus-link.h" +#include "nautilus-marshal.h" + +#define AUTO_SCROLL_MARGIN 20 + +struct _NautilusTreeViewDragDestDetails { + GtkTreeView *tree_view; + + gboolean drop_occurred; + + gboolean have_drag_data; + guint drag_type; + GtkSelectionData *drag_data; + GList *drag_list; + + guint highlight_id; + guint scroll_id; +}; + +enum { + GET_ROOT_URI, + GET_FILE_FOR_PATH, + MOVE_COPY_ITEMS, + LAST_SIGNAL +}; + +static void nautilus_tree_view_drag_dest_instance_init (NautilusTreeViewDragDest *dest); +static void nautilus_tree_view_drag_dest_class_init (NautilusTreeViewDragDestClass *class); + +static guint signals[LAST_SIGNAL]; + +GNOME_CLASS_BOILERPLATE (NautilusTreeViewDragDest, + nautilus_tree_view_drag_dest, + GObject, G_TYPE_OBJECT); + +static GtkTargetEntry drag_types [] = { + { NAUTILUS_ICON_DND_GNOME_ICON_LIST_TYPE, 0, NAUTILUS_ICON_DND_GNOME_ICON_LIST }, + { NAUTILUS_ICON_DND_URI_LIST_TYPE, 0, NAUTILUS_ICON_DND_URI_LIST }, + { NAUTILUS_ICON_DND_URL_TYPE, 0, NAUTILUS_ICON_DND_URL } + /* FIXME: Should handle emblems once the list view supports them */ +}; + + +static void +gtk_tree_view_vertical_autoscroll (GtkTreeView *tree_view) +{ + GdkRectangle visible_rect; + GtkAdjustment *vadjustment; + GdkWindow *window; + int y; + int offset; + float value; + + window = gtk_tree_view_get_bin_window (tree_view); + vadjustment = gtk_tree_view_get_vadjustment (tree_view); + + gdk_window_get_pointer (window, NULL, &y, NULL); + + y += vadjustment->value; + + gtk_tree_view_get_visible_rect (tree_view, &visible_rect); + + offset = y - (visible_rect.y + 2 * AUTO_SCROLL_MARGIN); + if (offset > 0) { + offset = y - (visible_rect.y + visible_rect.height - 2 * AUTO_SCROLL_MARGIN); + if (offset < 0) { + return; + } + } + + value = CLAMP (vadjustment->value + offset, 0.0, + vadjustment->upper - vadjustment->page_size); + gtk_adjustment_set_value (vadjustment, value); +} + +static int +scroll_timeout (gpointer data) +{ + GtkTreeView *tree_view = GTK_TREE_VIEW (data); + + gtk_tree_view_vertical_autoscroll (tree_view); + + return TRUE; +} + +static void +remove_scroll_timeout (NautilusTreeViewDragDest *dest) +{ + if (dest->details->scroll_id) { + gtk_timeout_remove (dest->details->scroll_id); + dest->details->scroll_id = 0; + } +} + +static gboolean +highlight_expose (GtkWidget *widget, + GdkEventExpose *event, + gpointer data) +{ + GdkWindow *bin_window; + int width; + int height; + + if (GTK_WIDGET_DRAWABLE (widget)) { + bin_window = + gtk_tree_view_get_bin_window (GTK_TREE_VIEW (widget)); + + gdk_drawable_get_size (bin_window, &width, &height); + + gtk_paint_focus (widget->style, + bin_window, + GTK_WIDGET_STATE (widget), + NULL, + widget, + "treeview-drop-indicator", + 0, 0, width, height); + } + + return FALSE; +} + +static void +set_widget_highlight (NautilusTreeViewDragDest *dest, gboolean highlight) +{ + if (!highlight && dest->details->highlight_id) { + g_signal_handler_disconnect (dest->details->tree_view, + dest->details->highlight_id); + dest->details->highlight_id = 0; + } + + if (highlight && !dest->details->highlight_id) { + dest->details->highlight_id = + g_signal_connect_object (dest->details->tree_view, + "expose_event", + G_CALLBACK (highlight_expose), + dest, + G_CONNECT_AFTER); + } + gtk_widget_queue_draw (GTK_WIDGET (dest->details->tree_view)); +} + +static void +set_drag_dest_row (NautilusTreeViewDragDest *dest, + GtkTreePath *path) +{ + if (path) { + set_widget_highlight (dest, FALSE); + gtk_tree_view_set_drag_dest_row + (dest->details->tree_view, + path, + GTK_TREE_VIEW_DROP_INTO_OR_BEFORE); + } else { + set_widget_highlight (dest, TRUE); + gtk_tree_view_set_drag_dest_row (dest->details->tree_view, + NULL, + 0); + } +} + +static void +clear_drag_dest_row (NautilusTreeViewDragDest *dest) +{ + gtk_tree_view_set_drag_dest_row (dest->details->tree_view, NULL, 0); + set_widget_highlight (dest, FALSE); +} + +static void +get_drag_data (NautilusTreeViewDragDest *dest, + GdkDragContext *context, + guint32 time) +{ + GdkAtom target; + + target = gtk_drag_dest_find_target (GTK_WIDGET (dest->details->tree_view), + context, + NULL); + + gtk_drag_get_data (GTK_WIDGET (dest->details->tree_view), + context, target, time); +} + +static void +free_drag_data (NautilusTreeViewDragDest *dest) +{ + dest->details->have_drag_data = FALSE; + + if (dest->details->drag_data) { + gtk_selection_data_free (dest->details->drag_data); + dest->details->drag_data = NULL; + } + + if (dest->details->drag_list) { + nautilus_drag_destroy_selection_list (dest->details->drag_list); + dest->details->drag_list = NULL; + } +} + +static char * +get_root_uri (NautilusTreeViewDragDest *dest) +{ + char *uri; + + g_signal_emit (dest, signals[GET_ROOT_URI], 0, &uri); + + return uri; +} + +static NautilusFile * +file_for_path (NautilusTreeViewDragDest *dest, GtkTreePath *path) +{ + NautilusFile *file; + char *uri; + + if (path) { + g_signal_emit (dest, signals[GET_FILE_FOR_PATH], 0, path, &file); + } else { + uri = get_root_uri (dest); + + file = nautilus_file_get (uri); + + g_free (uri); + } + + return file; +} + +static GtkTreePath * +get_drop_path (NautilusTreeViewDragDest *dest, + GtkTreePath *path) +{ + NautilusFile *file; + GtkTreePath *ret; + + if (!path) { + return NULL; + } + + file = file_for_path (dest, path); + + ret = NULL; + + if (!file || !nautilus_drag_can_accept_items (file, dest->details->drag_list)){ + if (gtk_tree_path_get_depth (path) == 1) { + ret = NULL; + } else { + ret = gtk_tree_path_copy (path); + gtk_tree_path_up (ret); + } + } else { + ret = gtk_tree_path_copy (path); + } + + nautilus_file_unref (file); + + return ret; +} + +static char * +get_drop_target (NautilusTreeViewDragDest *dest, + GtkTreePath *path) +{ + NautilusFile *file; + char *target; + + file = file_for_path (dest, path); + target = nautilus_file_get_drop_target_uri (file); + nautilus_file_unref (file); + + return target; +} + +static guint +get_drop_action (NautilusTreeViewDragDest *dest, + GdkDragContext *context, + GtkTreePath *path) +{ + char *drop_target; + guint action; + + if (!dest->details->have_drag_data || !dest->details->drag_list) { + return 0; + } + + switch (dest->details->drag_type) { + case NAUTILUS_ICON_DND_GNOME_ICON_LIST : + drop_target = get_drop_target (dest, path); + + if (!drop_target) { + return 0; + } + + nautilus_drag_default_drop_action_for_icons + (context, + drop_target, + dest->details->drag_list, + &action); + + g_free (drop_target); + + return action; + case NAUTILUS_ICON_DND_URI_LIST : + case NAUTILUS_ICON_DND_URL : + return context->suggested_action; + } + + return 0; +} + +static gboolean +drag_motion_callback (GtkWidget *widget, + GdkDragContext *context, + int x, + int y, + guint32 time, + gpointer data) +{ + NautilusTreeViewDragDest *dest; + GtkTreePath *path; + GtkTreePath *drop_path; + GtkTreeViewDropPosition pos; + guint action; + + dest = NAUTILUS_TREE_VIEW_DRAG_DEST (data); + + gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget), + x, y, &path, &pos); + + + if (!dest->details->have_drag_data) { + get_drag_data (dest, context, time); + } + drop_path = get_drop_path (dest, path); + + action = get_drop_action (dest, context, drop_path); + + if (action) { + set_drag_dest_row (dest, drop_path); + } else { + clear_drag_dest_row (dest); + } + + if (path) { + gtk_tree_path_free (path); + } + + if (drop_path) { + gtk_tree_path_free (drop_path); + } + + if (dest->details->scroll_id == 0) { + dest->details->scroll_id = + gtk_timeout_add (150, + scroll_timeout, + dest->details->tree_view); + } + + gdk_drag_status (context, action, time); + + return TRUE; +} + +static void +drag_leave_callback (GtkWidget *widget, + GdkDragContext *context, + guint32 time, + gpointer data) +{ + NautilusTreeViewDragDest *dest; + + dest = NAUTILUS_TREE_VIEW_DRAG_DEST (data); + + clear_drag_dest_row (dest); + + free_drag_data (dest); + + remove_scroll_timeout (dest); +} + +static void +receive_uris (NautilusTreeViewDragDest *dest, + GdkDragContext *context, + GList *source_uris, + int x, int y) +{ + char *drop_target; + GtkTreePath *path; + GtkTreePath *drop_path; + GtkTreeViewDropPosition pos; + GdkDragAction action; + + gtk_tree_view_get_dest_row_at_pos (dest->details->tree_view, x, y, + &path, &pos); + + drop_path = get_drop_path (dest, path); + + drop_target = get_drop_target (dest, drop_path); + + if (context->action == GDK_ACTION_ASK) { + if (nautilus_drag_selection_includes_special_link (dest->details->drag_list)) { + /* We only want to move the trash */ + action = GDK_ACTION_MOVE; + } else { + action = GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_LINK; + } + context->action = nautilus_drag_drop_action_ask (action); + } + + /* We only want to copy external uris */ + if (dest->details->drag_type == NAUTILUS_ICON_DND_URI_LIST) { + action = GDK_ACTION_COPY; + } + + if (context->action > 0) { + g_signal_emit (dest, signals[MOVE_COPY_ITEMS], 0, + source_uris, + drop_target, + context->action, + x, y); + } + + if (path) { + gtk_tree_path_free (path); + } + + if (drop_path) { + gtk_tree_path_free (drop_path); + } + + g_free (drop_target); +} + +static void +receive_dropped_icons (NautilusTreeViewDragDest *dest, + GdkDragContext *context, + int x, int y) +{ + GList *source_uris; + GList *l; + + /* FIXME: ignore local only moves */ + + if (!dest->details->drag_list) { + return; + } + + source_uris = NULL; + for (l = dest->details->drag_list; l != NULL; l = l->next) { + source_uris = g_list_prepend (source_uris, + ((NautilusDragSelectionItem *)l->data)->uri); + } + + source_uris = g_list_reverse (source_uris); + + receive_uris (dest, context, source_uris, x, y); + + g_list_free (source_uris); +} + +static void +receive_dropped_uri_list (NautilusTreeViewDragDest *dest, + GdkDragContext *context, + int x, int y) +{ + GList *source_uris; + + if (!dest->details->drag_data) { + return; + } + + source_uris = nautilus_icon_dnd_uri_list_extract_uris ((char*)dest->details->drag_data->data); + + receive_uris (dest, context, source_uris, x, y); + + nautilus_icon_dnd_uri_list_free_strings (source_uris); +} + +static gboolean +drag_data_received_callback (GtkWidget *widget, + GdkDragContext *context, + int x, + int y, + GtkSelectionData *selection_data, + guint info, + guint32 time, + gpointer data) +{ + NautilusTreeViewDragDest *dest; + gboolean success; + + dest = NAUTILUS_TREE_VIEW_DRAG_DEST (data); + + if (!dest->details->have_drag_data) { + dest->details->have_drag_data = TRUE; + dest->details->drag_type = info; + dest->details->drag_data = + gtk_selection_data_copy (selection_data); + if (info == NAUTILUS_ICON_DND_GNOME_ICON_LIST) { + dest->details->drag_list = + nautilus_drag_build_selection_list (selection_data); + } + } + + if (dest->details->drop_occurred) { + success = FALSE; + switch (info) { + case NAUTILUS_ICON_DND_GNOME_ICON_LIST : + receive_dropped_icons (dest, context, x, y); + success = TRUE; + break; + case NAUTILUS_ICON_DND_URI_LIST : + case NAUTILUS_ICON_DND_URL : + receive_dropped_uri_list (dest, context, x, y); + success = TRUE; + break; + } + + dest->details->drop_occurred = FALSE; + free_drag_data (dest); + gtk_drag_finish (context, success, FALSE, time); + } + + /* appease GtkTreeView by preventing its drag_data_receive + * from being called */ + g_signal_stop_emission_by_name (dest->details->tree_view, + "drag_data_received"); + + return TRUE; +} + +static gboolean +drag_drop_callback (GtkWidget *widget, + GdkDragContext *context, + int x, + int y, + guint32 time, + gpointer data) +{ + NautilusTreeViewDragDest *dest; + + dest = NAUTILUS_TREE_VIEW_DRAG_DEST (data); + + dest->details->drop_occurred = TRUE; + + get_drag_data (dest, context, time); + remove_scroll_timeout (dest); + clear_drag_dest_row (dest); + + return TRUE; +} + +static void +nautilus_tree_view_drag_dest_finalize (GObject *object) +{ + NautilusTreeViewDragDest *dest; + + dest = NAUTILUS_TREE_VIEW_DRAG_DEST (object); + + free_drag_data (dest); + + g_free (dest->details); + + EEL_CALL_PARENT (G_OBJECT_CLASS, finalize, (object)); +} + +static void +nautilus_tree_view_drag_dest_instance_init (NautilusTreeViewDragDest *dest) +{ + dest->details = g_new0 (NautilusTreeViewDragDestDetails, 1); +} + +static void +nautilus_tree_view_drag_dest_class_init (NautilusTreeViewDragDestClass *class) +{ + GObjectClass *gobject_class; + + gobject_class = G_OBJECT_CLASS (class); + + gobject_class->finalize = nautilus_tree_view_drag_dest_finalize; + + signals[GET_ROOT_URI] = + g_signal_new ("get_root_uri", + G_TYPE_FROM_CLASS (class), + G_SIGNAL_RUN_LAST, + G_STRUCT_OFFSET (NautilusTreeViewDragDestClass, + get_root_uri), + NULL, NULL, + nautilus_marshal_STRING__VOID, + G_TYPE_STRING, 0); + signals[GET_FILE_FOR_PATH] = + g_signal_new ("get_file_for_path", + G_TYPE_FROM_CLASS (class), + G_SIGNAL_RUN_LAST, + G_STRUCT_OFFSET (NautilusTreeViewDragDestClass, + get_file_for_path), + NULL, NULL, + nautilus_marshal_OBJECT__BOXED, + NAUTILUS_TYPE_FILE, 1, + GTK_TYPE_TREE_PATH); + signals[MOVE_COPY_ITEMS] = + g_signal_new ("move_copy_items", + G_TYPE_FROM_CLASS (class), + G_SIGNAL_RUN_LAST, + G_STRUCT_OFFSET (NautilusTreeViewDragDestClass, + move_copy_items), + NULL, NULL, + + nautilus_marshal_VOID__POINTER_STRING_UINT_INT_INT, + G_TYPE_NONE, 5, + G_TYPE_POINTER, + G_TYPE_STRING, + G_TYPE_UINT, + G_TYPE_INT, + G_TYPE_INT); +} + +NautilusTreeViewDragDest * +nautilus_tree_view_drag_dest_new (GtkTreeView *tree_view) +{ + NautilusTreeViewDragDest *dest; + + dest = g_object_new (NAUTILUS_TYPE_TREE_VIEW_DRAG_DEST, NULL); + + dest->details->tree_view = tree_view; + + gtk_drag_dest_set (GTK_WIDGET (tree_view), + 0, drag_types, G_N_ELEMENTS (drag_types), + GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_LINK | GDK_ACTION_ASK); + + g_signal_connect_object (tree_view, + "drag_motion", + G_CALLBACK (drag_motion_callback), + dest, 0); + g_signal_connect_object (tree_view, + "drag_leave", + G_CALLBACK (drag_leave_callback), + dest, 0); + g_signal_connect_object (tree_view, + "drag_drop", + G_CALLBACK (drag_drop_callback), + dest, 0); + g_signal_connect_object (tree_view, + "drag_data_received", + G_CALLBACK (drag_data_received_callback), + dest, 0); + + return dest; +} |