1
1

1999-02-03 Federico Mena Quintero <federico@nuclecu.unam.mx>

* gdnd.c (gdnd_validate_action): Added an "on_desktop" argument;
	this specifies that the user is dragging over the desktop.  Added
	cases for drags of desktop icons and URLs.
	(gdnd_perform_drop): New public function to handle drops.  This is
	now the master function to handle drops -- use this to make all of
	MC consistent.
	(drop_on_directory): Made static and renamed from
	gdnd_drop_on_directory().  Made it fit for use by
	gdnd_perform_drop().
	(drop_url_on_directory): New function to drop an url on a
	directory, moved over from gdesktop.c.
	(drop_uri_list_on_directory): New function to handle drops of URI
	lists on a directory.
	(drop_on_file): New function to handle drops on a non-directory
	file entry.  This is moved over from gdesktop.c,
	desktop_icon_drop_uri_list().  This is yet to be finished.

	* gdesktop.c (desktop_icon_drop_uri_list): Fix memory leak.
	(desktop_drag_motion): Use gdnd_validate_action().
	(dnd_icon_targets): Allow dropping URLs on icons.
	(icon_drag_motion): Use gdnd_validate_action().
	(desktop_drag_data_received): Use gdnd_perform_drop().
	(icon_drag_data_received): Use gdnd_perform_drop().

	* gscreen.c (panel_icon_list_drag_data_received): Use
	gdnd_perform_drop().
	(panel_clist_drag_data_received): Use gdnd_perform_drop().
	(panel_tree_drag_data_received): Use gdnd_perform_drop().

	* gcmd.c (gnome_about_cmd): Fixed Foundation typo, and updated the
	copyright date.
Этот коммит содержится в:
Miguel de Icaza 1999-02-03 11:46:01 +00:00
родитель 89204b8f1c
Коммит 04c4abd818
5 изменённых файлов: 370 добавлений и 302 удалений

Просмотреть файл

@ -1,3 +1,37 @@
1999-02-03 Federico Mena Quintero <federico@nuclecu.unam.mx>
* gdnd.c (gdnd_validate_action): Added an "on_desktop" argument;
this specifies that the user is dragging over the desktop. Added
cases for drags of desktop icons and URLs.
(gdnd_perform_drop): New public function to handle drops. This is
now the master function to handle drops -- use this to make all of
MC consistent.
(drop_on_directory): Made static and renamed from
gdnd_drop_on_directory(). Made it fit for use by
gdnd_perform_drop().
(drop_url_on_directory): New function to drop an url on a
directory, moved over from gdesktop.c.
(drop_uri_list_on_directory): New function to handle drops of URI
lists on a directory.
(drop_on_file): New function to handle drops on a non-directory
file entry. This is moved over from gdesktop.c,
desktop_icon_drop_uri_list(). This is yet to be finished.
* gdesktop.c (desktop_icon_drop_uri_list): Fix memory leak.
(desktop_drag_motion): Use gdnd_validate_action().
(dnd_icon_targets): Allow dropping URLs on icons.
(icon_drag_motion): Use gdnd_validate_action().
(desktop_drag_data_received): Use gdnd_perform_drop().
(icon_drag_data_received): Use gdnd_perform_drop().
* gscreen.c (panel_icon_list_drag_data_received): Use
gdnd_perform_drop().
(panel_clist_drag_data_received): Use gdnd_perform_drop().
(panel_tree_drag_data_received): Use gdnd_perform_drop().
* gcmd.c (gnome_about_cmd): Fixed Foundation typo, and updated the
copyright date.
1999-02-02 Miguel de Icaza <miguel@nuclecu.unam.mx> 1999-02-02 Miguel de Icaza <miguel@nuclecu.unam.mx>
* gnome-file-property-dialog.c (create_general_properties): Use * gnome-file-property-dialog.c (create_general_properties): Use

Просмотреть файл

@ -73,6 +73,7 @@ static GtkTargetEntry dnd_icon_sources[] = {
static GtkTargetEntry dnd_icon_targets[] = { static GtkTargetEntry dnd_icon_targets[] = {
{ TARGET_MC_DESKTOP_ICON_TYPE, 0, TARGET_MC_DESKTOP_ICON }, { TARGET_MC_DESKTOP_ICON_TYPE, 0, TARGET_MC_DESKTOP_ICON },
{ TARGET_URI_LIST_TYPE, 0, TARGET_URI_LIST }, { TARGET_URI_LIST_TYPE, 0, TARGET_URI_LIST },
{ TARGET_URL_TYPE, 0, TARGET_URL }
}; };
static GtkTargetEntry dnd_desktop_targets[] = { static GtkTargetEntry dnd_desktop_targets[] = {
@ -1163,8 +1164,7 @@ setup_icon_dnd_source (DesktopIconInfo *dii)
dii); dii);
} }
/* /* Callback used when we get a drag_motion event from a desktop icon. We have
* Callback used when we get a drag_motion event from a desktop icon. We have
* to decide which operation to perform based on the type of the data the user * to decide which operation to perform based on the type of the data the user
* is dragging. * is dragging.
*/ */
@ -1176,27 +1176,25 @@ icon_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x, gint y, gu
char *filename; char *filename;
file_entry *fe; file_entry *fe;
GdkDragAction action; GdkDragAction action;
GtkWidget *source_widget;
int is_desktop_icon;
dii = data; dii = data;
filename = g_concat_dir_and_file (desktop_directory, dii->filename); filename = g_concat_dir_and_file (desktop_directory, dii->filename);
fe = file_entry_from_file (filename); fe = file_entry_from_file (filename);
g_free (filename); g_free (filename);
if (!fe)
return 0; /* eeek */
action = 0; /* be pessimistic by defaulting to nothing */ gdnd_find_panel_by_drag_context (context, &source_widget);
is_desktop_icon = gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON);
if (dii->selected action = gdnd_validate_action (context,
&& gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON) TRUE,
&& (context->actions & GDK_ACTION_MOVE)) source_widget != NULL,
action = GDK_ACTION_MOVE; source_widget && is_desktop_icon,
else if (gdnd_drag_context_has_target (context, TARGET_URI_LIST)) { fe,
if (S_ISDIR (fe->buf.st_mode) || fe->f.link_to_dir) dii->selected);
action = context->suggested_action;
else if (is_exe (fe->buf.st_mode)
&& if_link_is_exe (fe)
&& (context->actions & GDK_ACTION_COPY))
action = GDK_ACTION_COPY;
}
gdk_drag_status (context, action, time); gdk_drag_status (context, action, time);
return TRUE; return TRUE;
@ -1245,11 +1243,6 @@ drop_desktop_icons (GdkDragContext *context, GtkSelectionData *data, int x, int
GList *l; GList *l;
GSList *sel_icons, *sl; GSList *sel_icons, *sl;
/*
* FIXME: this needs to do the right thing (what Windows does)
* when desktop_auto_placement is enabled.
*/
/* Find the icon that the user is dragging */ /* Find the icon that the user is dragging */
source_dii = find_icon_by_drag_context (context); source_dii = find_icon_by_drag_context (context);
@ -1292,94 +1285,6 @@ drop_desktop_icons (GdkDragContext *context, GtkSelectionData *data, int x, int
g_slist_free (sel_icons); g_slist_free (sel_icons);
} }
/**
* drop_on_file_entry
*/
static void
desktop_icon_drop_uri_list (DesktopIconInfo *dii, GdkDragContext *context, GtkSelectionData *data)
{
char *filename;
file_entry *fe;
int size;
char *buf;
const char *mime_type;
filename = g_concat_dir_and_file (desktop_directory, dii->filename);
fe = file_entry_from_file (filename);
if (!fe)
return; /* eek */
/* 1. If it is a directory, drop the files there */
if (fe->f.link_to_dir) {
gdnd_drop_on_directory (context, data, filename);
goto out;
}
/* 2. Try to use a metadata-based drop action */
if (gnome_metadata_get (filename, "drop-action", &size, &buf) == 0) {
/*action_drop (filename, buf, context, data);*/ /* Fixme: i'm undefined */
g_free (buf);
goto out;
}
/* 3. Try a drop action from the mime-type */
mime_type = gnome_mime_type_or_default (filename, NULL);
if (mime_type) {
char *action;
action = gnome_mime_get_value (mime_type, "drop-action");
if (action) {
/*action_drop (filename, action, context, data);*/ /* Fixme: i'm undefined */
goto out;
}
}
/* 4. Executable. Try metadata keys for "open" */
if (is_exe (fe->buf.st_mode) && if_link_is_exe (fe)) {
GList *names, *l;
int len, i;
char **drops;
/* Convert the list of filenames into an array of char */
names = gnome_uri_list_extract_uris (data->data);
len = g_list_length (names);
drops = (char **) g_malloc (sizeof (char *) * (len+1));
for (l = names, i = 0; i < len; i++, l = l->next) {
char *text = l->data;
if (strncmp (text, "file:", 5) == 0)
text += 5;
drops [i] = text;
}
drops [i] = NULL;
if (gnome_metadata_get (filename, "open", &size, &buf) == 0) {
exec_extension (filename, buf, drops, NULL, 0);
goto out2;
}
exec_extension (filename, "%f %q", drops, NULL, 0);
g_free (drops);
out2:
gnome_uri_list_free_strings (names);
g_free (buf);
}
out:
file_entry_free (fe);
}
static void static void
icon_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y, icon_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
GtkSelectionData *data, guint info, guint time, gpointer user_data) GtkSelectionData *data, guint info, guint time, gpointer user_data)
@ -1388,22 +1293,21 @@ icon_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gin
dii = user_data; dii = user_data;
switch (info) { if (gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON) && dii->selected)
case TARGET_MC_DESKTOP_ICON: drop_desktop_icons (context, data, x + dii->x, y + dii->y);
if (dii->selected) else {
drop_desktop_icons (context, data, x + dii->x, y + dii->y); char *full_name;
else file_entry *fe;
printf ("FIXME: what do drop?\n"); /* FIXME */
break; full_name = g_concat_dir_and_file (desktop_directory, dii->filename);
fe = file_entry_from_file (full_name);
if (!fe)
return; /* eeeek */
case TARGET_URI_LIST: if (gdnd_perform_drop (context, data, fe, full_name))
printf ("Wheeeeee!\n"); reload_desktop_icons (TRUE, x, y);
desktop_icon_drop_uri_list (dii, context, data);
break;
default: file_entry_free (fe);
break;
} }
} }
@ -1727,73 +1631,27 @@ desktop_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
{ {
GdkDragAction action; GdkDragAction action;
GtkWidget *source_widget; GtkWidget *source_widget;
int is_desktop_icon;
action = context->suggested_action; /* this is the default */ gdnd_find_panel_by_drag_context (context, &source_widget);
is_desktop_icon = gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON);
if (gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON)) action = gdnd_validate_action (context,
action = GDK_ACTION_MOVE; TRUE,
else if (gdnd_drag_context_has_target (context, TARGET_URI_LIST)) { source_widget != NULL,
source_widget = gtk_drag_get_source_widget (context); source_widget && is_desktop_icon,
NULL,
/* If it comes from ourselves, make move the default unless the FALSE);
* user is explicitly asking for ASK.
*/
if (source_widget
&& context->suggested_action != GDK_ACTION_ASK
&& (context->actions & GDK_ACTION_MOVE))
action = GDK_ACTION_MOVE;
} else if (gdnd_drag_context_has_target (context, TARGET_URL)) {
/* FIXME: right now we only allow links. We should see if we
* can move or copy stuff instead (for ftp instead of http
* sites, for example).
*/
if (context->actions & GDK_ACTION_LINK)
action = GDK_ACTION_LINK;
else
action = 0;
} else
action = 0; /* we cannot handle that type of data */
gdk_drag_status (context, action, time); gdk_drag_status (context, action, time);
return TRUE; return TRUE;
} }
/*
* drop_url: Invoked when we have received an URL from Netscape.
* Install the url on the ~/desktop directory
*/
static void
drop_url (GdkDragContext *context, GtkSelectionData *data, gint x, gint y)
{
char *template;
template = g_concat_dir_and_file (desktop_directory, "urlXXXXXX");
if (mktemp (template)) {
FILE *f;
f = fopen (template, "w");
if (f){
fprintf (f, "URL: %s\n", data->data);
fclose (f);
gnome_metadata_set (template, "desktop-url",
strlen (data->data) + 1,
data->data);
reload_desktop_icons (TRUE, x, y);
}
}
g_free (template);
}
/* Callback used when the root window receives a drop */ /* Callback used when the root window receives a drop */
static void static void
desktop_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y, desktop_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
GtkSelectionData *data, guint info, guint time, gpointer user_data) GtkSelectionData *data, guint info, guint time, gpointer user_data)
{ {
int retval;
gint dx, dy; gint dx, dy;
/* Fix the proxy window offsets */ /* Fix the proxy window offsets */
@ -1802,23 +1660,19 @@ desktop_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x,
x += dx; x += dx;
y += dy; y += dy;
switch (info) { if (gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON))
case TARGET_MC_DESKTOP_ICON:
drop_desktop_icons (context, data, x, y); drop_desktop_icons (context, data, x, y);
break; else {
file_entry *desktop_fe;
case TARGET_URI_LIST: desktop_fe = file_entry_from_file (desktop_directory);
retval = gdnd_drop_on_directory (context, data, desktop_directory); if (!desktop_fe)
if (retval) return; /* eeek */
if (gdnd_perform_drop (context, data, desktop_fe, desktop_directory))
reload_desktop_icons (TRUE, x, y); reload_desktop_icons (TRUE, x, y);
break;
case TARGET_URL: file_entry_free (desktop_fe);
drop_url (context, data, x, y);
break;
default:
break;
} }
} }
@ -1984,11 +1838,6 @@ icon_is_in_area (DesktopIconInfo *dii, int x1, int y1, int x2, int y2)
dicon = DESKTOP_ICON (dii->dicon); dicon = DESKTOP_ICON (dii->dicon);
/* FIXME: this only intersects the rectangle with the icon image's
* bounds. Doing the "hard" intersection with the actual shape of the
* image is left as an exercise to the reader.
*/
x1 -= dii->x; x1 -= dii->x;
y1 -= dii->y; y1 -= dii->y;
x2 -= dii->x; x2 -= dii->x;

Просмотреть файл

@ -13,6 +13,7 @@
#include "fileopctx.h" #include "fileopctx.h"
#include "main.h" #include "main.h"
#include "panel.h" #include "panel.h"
#include "ext.h"
#include "gscreen.h" #include "gscreen.h"
#include "../vfs/vfs.h" #include "../vfs/vfs.h"
#include <gdk/gdkprivate.h> #include <gdk/gdkprivate.h>
@ -96,21 +97,20 @@ get_action (GdkDragContext *context)
return action; return action;
} }
/* /* Performs a drop action on the specified panel. Only supports copy and move
* Performs a drop action on the specified panel. Only supports copy * operations. The files are moved or copied to the specified destination
* and move operations. The files are moved or copied to the * directory.
* specified destination directory.
*/ */
static void static void
perform_action_on_panel (WPanel *source_panel, GdkDragAction action, char *destdir, int ask) perform_action_on_panel (WPanel *source_panel, GdkDragAction action, char *destdir)
{ {
switch (action) { switch (action) {
case GDK_ACTION_COPY: case GDK_ACTION_COPY:
panel_operate (source_panel, OP_COPY, destdir, ask); panel_operate (source_panel, OP_COPY, destdir, FALSE);
break; break;
case GDK_ACTION_MOVE: case GDK_ACTION_MOVE:
panel_operate (source_panel, OP_MOVE, destdir, ask); panel_operate (source_panel, OP_MOVE, destdir, FALSE);
break; break;
default: default:
@ -146,11 +146,10 @@ perform_links (GList *names, char *destdir)
} }
} }
/* /* Performs a drop action manually, by going through the list of files to
* Performs a drop action manually, by going through the list of files * operate on. The files are copied or moved to the specified directory. This
* to operate on. The files are copied or moved to the specified * should also encompass symlinking when the file operations window supports
* directory. This should also encompass symlinking when the file * links.
* operations window supports links.
*/ */
static void static void
perform_action (GList *names, GdkDragAction action, char *destdir) perform_action (GList *names, GdkDragAction action, char *destdir)
@ -226,46 +225,24 @@ perform_action (GList *names, GdkDragAction action, char *destdir)
file_op_context_destroy (ctx); file_op_context_destroy (ctx);
} }
/** /* Drop a URI list on a directory. If the data comes from a panel, use the nice
* gdnd_drop_on_directory: * MC progress display; otherwise `do it by hand'.
* @context: The drag context received from the drag_data_received callback */
* @selection_data: The selection data from the drag_data_received callback static void
* @dirname: The name of the directory to drop onto drop_uri_list_on_directory (GdkDragContext *context, GtkSelectionData *selection_data,
* GdkDragAction action, char *destdir)
* Extracts an URI list from the selection data and drops all the files in the
* specified directory.
*
* Return Value: TRUE if the drop was sucessful, FALSE if it was not.
**/
int
gdnd_drop_on_directory (GdkDragContext *context, GtkSelectionData *selection_data, char *destdir)
{ {
GdkDragAction action;
WPanel *source_panel; WPanel *source_panel;
GtkWidget *source_widget; GtkWidget *source_widget;
GList *names; GList *names;
if (context->action == GDK_ACTION_ASK) {
action = get_action (context);
if (action == GDK_ACTION_ASK)
return FALSE;
} else
action = context->action;
/* If we are dragging from a file panel, we can display a nicer status
* display. But if the drag was from the tree, we cannot do this.
*/
source_panel = gdnd_find_panel_by_drag_context (context, &source_widget); source_panel = gdnd_find_panel_by_drag_context (context, &source_widget);
if (source_widget == source_panel->tree)
source_panel = NULL;
/* Symlinks do not use file.c */ /* We cannot use file.c if we are going to symlink or if we are dragging
* from a tree.
if (source_panel && action != GDK_ACTION_LINK) */
perform_action_on_panel (source_panel, action, destdir, if (source_panel && source_widget != source_panel->tree && action != GDK_ACTION_LINK)
context->action == GDK_ACTION_ASK); perform_action_on_panel (source_panel, action, destdir);
else { else {
names = gnome_uri_list_extract_uris (selection_data->data); names = gnome_uri_list_extract_uris (selection_data->data);
@ -276,10 +253,153 @@ gdnd_drop_on_directory (GdkDragContext *context, GtkSelectionData *selection_dat
gnome_uri_list_free_strings (names); gnome_uri_list_free_strings (names);
} }
}
/* Drop a Netscape URL in a directory */
static void
drop_url_on_directory (GdkDragContext *context, GtkSelectionData *selection_data, char *destdir)
{
char *template;
template = g_concat_dir_and_file (destdir, "urlXXXXXX");
if (mktemp (template)) {
FILE *f;
f = fopen (template, "w");
if (f) {
fprintf (f, "URL: %s\n", selection_data->data);
fclose (f);
gnome_metadata_set (template, "desktop-url",
strlen (selection_data->data) + 1,
selection_data->data);
}
}
g_free (template);
}
/* Drop stuff on a directory */
static int
drop_on_directory (GdkDragContext *context, GtkSelectionData *selection_data, char *destdir)
{
GdkDragAction action;
if (context->action == GDK_ACTION_ASK) {
action = get_action (context);
if (action == GDK_ACTION_ASK)
return FALSE;
} else
action = context->action;
if (gdnd_drag_context_has_target (context, TARGET_URI_LIST))
drop_uri_list_on_directory (context, selection_data, action, destdir);
else if (gdnd_drag_context_has_target (context, TARGET_URL))
drop_url_on_directory (context, selection_data, destdir);
else
return FALSE;
return TRUE; return TRUE;
} }
/* Drop stuff on a non-directory file. This uses metadata and MIME as well. */
static int
drop_on_file (GdkDragContext *context, GtkSelectionData *selection_data,
file_entry *dest_fe, char *dest_name)
{
int size;
char *buf;
const char *mime_type;
/* 1. Try to use a metadata-based drop action */
if (gnome_metadata_get (dest_name, "drop-action", &size, &buf) == 0) {
/*action_drop (dest_name, buf, context, selection_data);*/ /* Fixme: i'm undefined */
g_free (buf);
return TRUE;
}
/* 2. Try a drop action from the MIME-type */
mime_type = gnome_mime_type_or_default (dest_name, NULL);
if (mime_type) {
char *action;
action = gnome_mime_get_value (mime_type, "drop-action");
if (action) {
/* Fixme: i'm undefined */
/*action_drop (dest_name, action, context, selection_data);*/
return TRUE;
}
}
/* 3. If executable, try metadata keys for "open" */
if (is_exe (dest_fe->buf.st_mode) && if_link_is_exe (dest_fe)) {
GList *names, *l;
int len, i;
char **drops;
/* FIXME: handle the case for Netscape URLs */
/* Convert the data list into an array of strings */
names = gnome_uri_list_extract_uris (selection_data->data);
len = g_list_length (names);
drops = g_new (char *, len + 1);
for (l = names, i = 0; i < len; i++, l = l->next) {
char *text = l->data;
if (strncmp (text, "file:", 5) == 0)
text += 5;
drops[i] = text;
}
drops[i] = NULL;
if (gnome_metadata_get (dest_name, "open", &size, &buf) == 0)
exec_extension (dest_name, buf, drops, NULL, 0);
else
exec_extension (dest_name, "%f %q", drops, NULL, 0);
g_free (drops);
gnome_uri_list_free_strings (names);
g_free (buf);
return TRUE;
}
return FALSE; /* could not drop */
}
int
gdnd_perform_drop (GdkDragContext *context, GtkSelectionData *selection_data,
file_entry *dest_fe, char *dest_name)
{
GdkDragAction action;
g_return_val_if_fail (context != NULL, FALSE);
g_return_val_if_fail (selection_data != NULL, FALSE);
g_return_val_if_fail (dest_fe != NULL, FALSE);
/* Get action */
if (context->action == GDK_ACTION_ASK) {
action = get_action (context);
if (action == GDK_ACTION_ASK)
return FALSE;
} else
action = context->action;
if (S_ISDIR (dest_fe->buf.st_mode) || dest_fe->f.link_to_dir)
return drop_on_directory (context, selection_data, dest_name);
else
return drop_on_file (context, selection_data, dest_fe, dest_name);
}
/** /**
* gdnd_drag_context_has_target: * gdnd_drag_context_has_target:
* @context: The context to query for a target type * @context: The context to query for a target type
@ -348,6 +468,7 @@ gdnd_find_panel_by_drag_context (GdkDragContext *context, GtkWidget **source_wid
/** /**
* gdnd_validate_action: * gdnd_validate_action:
* @context: The drag context for this drag operation. * @context: The drag context for this drag operation.
* @on_desktop: Whether we are dragging onto the desktop or a desktop icon.
* @same_process: Whether the drag comes from the same process or not. * @same_process: Whether the drag comes from the same process or not.
* @same_source: If same_process, then whether the source and dest widgets are the same. * @same_source: If same_process, then whether the source and dest widgets are the same.
* @dest: The destination file entry, or NULL if dropping on empty space. * @dest: The destination file entry, or NULL if dropping on empty space.
@ -359,48 +480,70 @@ gdnd_find_panel_by_drag_context (GdkDragContext *context, GtkWidget **source_wid
* Return value: The computed action, meant to be passed to gdk_drag_action(). * Return value: The computed action, meant to be passed to gdk_drag_action().
**/ **/
GdkDragAction GdkDragAction
gdnd_validate_action (GdkDragContext *context, int same_process, int same_source, gdnd_validate_action (GdkDragContext *context,
int on_desktop, int same_process, int same_source,
file_entry *dest_fe, int dest_selected) file_entry *dest_fe, int dest_selected)
{ {
int on_directory; int on_directory;
int on_exe; int on_exe;
if (dest_fe) { g_return_val_if_fail (context != NULL, 0);
on_directory = dest_fe->f.link_to_dir || S_ISDIR (dest_fe->buf.st_mode);
on_exe = is_exe (dest_fe->buf.st_mode) && if_link_is_exe (dest_fe);
}
if (dest_fe) { /* If we are dragging a desktop icon onto the desktop or onto a selected
if (same_source && dest_selected) * desktop icon, unconditionally specify MOVE.
return 0; */
if (on_desktop
&& gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON)
&& (!dest_fe || dest_selected))
return GDK_ACTION_MOVE;
if (on_directory) { if (gdnd_drag_context_has_target (context, TARGET_URI_LIST)) {
if ((same_source || same_process) if (dest_fe) {
&& (context->actions & GDK_ACTION_MOVE) on_directory = S_ISDIR (dest_fe->buf.st_mode) || dest_fe->f.link_to_dir;
&& context->suggested_action != GDK_ACTION_ASK) on_exe = is_exe (dest_fe->buf.st_mode) && if_link_is_exe (dest_fe);
}
if (dest_fe) {
if (same_source && dest_selected)
return 0;
if (on_directory) {
if ((same_source || same_process)
&& (context->actions & GDK_ACTION_MOVE)
&& context->suggested_action != GDK_ACTION_ASK)
return GDK_ACTION_MOVE;
else
return context->suggested_action;
} else if (on_exe) {
if (context->actions & GDK_ACTION_COPY)
return GDK_ACTION_COPY;
} else if (same_source)
return 0;
else if (same_process
&& (context->actions & GDK_ACTION_MOVE)
&& context->suggested_action != GDK_ACTION_ASK)
return GDK_ACTION_MOVE; return GDK_ACTION_MOVE;
else else
return context->suggested_action; return context->suggested_action;
} else if (on_exe) { } else {
if (context->actions & GDK_ACTION_COPY) if (same_source)
return GDK_ACTION_COPY; return 0;
} else if (same_source) else if (same_process
return 0; && (context->actions & GDK_ACTION_MOVE)
else if (same_process && context->suggested_action != GDK_ACTION_ASK)
&& (context->actions & GDK_ACTION_MOVE) return GDK_ACTION_MOVE;
&& context->suggested_action != GDK_ACTION_ASK) else
return GDK_ACTION_MOVE; return context->suggested_action;
else }
return context->suggested_action; }
} else {
if (same_source) if (gdnd_drag_context_has_target (context, TARGET_URL)) {
return 0; /* FIXME: right now we only allow links. We should see if we
else if (same_process * can move or copy stuff instead (for ftp instead of http
&& (context->actions & GDK_ACTION_MOVE) * sites, for example).
&& context->suggested_action != GDK_ACTION_ASK) */
return GDK_ACTION_MOVE; if (context->actions & GDK_ACTION_LINK)
else return GDK_ACTION_LINK;
return context->suggested_action;
} }
return 0; return 0;

Просмотреть файл

@ -35,8 +35,12 @@ extern GdkAtom dnd_target_atoms[];
/* Initializes drag and drop by interning the target convenience atoms */ /* Initializes drag and drop by interning the target convenience atoms */
void gdnd_init (void); void gdnd_init (void);
/* Drop the list of URIs in the selection data to the specified directory */ /* Perform a drop on the specified file entry. This function takes care of
int gdnd_drop_on_directory (GdkDragContext *context, GtkSelectionData *selection_data, char *dirname); * determining how to drop the stuff epending on the type of the file entry.
* Returns TRUE if an action was performed, FALSE otherwise (i.e. invalid drop).
*/
int gdnd_perform_drop (GdkDragContext *context, GtkSelectionData *selection_data,
file_entry *dest_fe, char *dest_name);
/* Test whether the specified context has a certain target type */ /* Test whether the specified context has a certain target type */
int gdnd_drag_context_has_target (GdkDragContext *context, TargetType type); int gdnd_drag_context_has_target (GdkDragContext *context, TargetType type);
@ -47,7 +51,8 @@ WPanel *gdnd_find_panel_by_drag_context (GdkDragContext *context, GtkWidget **so
/* Computes the final drag action based on the suggested actions and the /* Computes the final drag action based on the suggested actions and the
* specified conditions. * specified conditions.
*/ */
GdkDragAction gdnd_validate_action (GdkDragContext *context, int same_process, int same_source, GdkDragAction gdnd_validate_action (GdkDragContext *context,
int on_desktop, int same_process, int same_source,
file_entry *dest_fe, int dest_selected); file_entry *dest_fe, int dest_selected);

Просмотреть файл

@ -776,25 +776,38 @@ panel_icon_list_drag_data_received (GtkWidget *widget,
WPanel *panel) WPanel *panel)
{ {
GnomeIconList *gil = GNOME_ICON_LIST (widget); GnomeIconList *gil = GNOME_ICON_LIST (widget);
char *dir; file_entry *fe;
char *file;
int free_file, free_fe;
int idx; int idx;
gboolean reload; gboolean reload;
idx = gnome_icon_list_get_icon_at (gil, x, y); idx = gnome_icon_list_get_icon_at (gil, x, y);
if (idx == -1) if (idx == -1) {
dir = g_strdup (panel->cwd); file = panel->cwd;
else { fe = file_entry_from_file (file);
if (panel->dir.list [idx].f.link_to_dir || if (!fe)
S_ISDIR (panel->dir.list [idx].buf.st_mode)) return; /* eeeek */
dir = concat_dir_and_file (panel->cwd, panel->dir.list [idx].fname);
else free_file = FALSE;
dir = g_strdup (panel->cwd); free_fe = TRUE;
} else {
file = g_concat_dir_and_file (panel->cwd, panel->dir.list[idx].fname);
fe = &panel->dir.list[idx];
free_file = TRUE;
free_fe = FALSE;
} }
reload = gdnd_drop_on_directory (context, selection_data, dir); reload = gdnd_perform_drop (context, selection_data, fe, file);
g_free (dir);
if (reload){ if (free_file)
g_free (file);
if (free_fe)
file_entry_free (fe);
if (reload) {
update_one_panel_widget (panel, 0, UP_KEEPSEL); update_one_panel_widget (panel, 0, UP_KEEPSEL);
panel_update_contents (panel); panel_update_contents (panel);
} }
@ -817,26 +830,42 @@ panel_clist_drag_data_received (GtkWidget *widget,
WPanel *panel) WPanel *panel)
{ {
GtkCList *clist = GTK_CLIST (widget); GtkCList *clist = GTK_CLIST (widget);
char *dir; file_entry *fe;
char *file;
int free_file, free_fe;
int row; int row;
int reload;
if (gtk_clist_get_selection_info (clist, x, y, &row, NULL) == 0) if (gtk_clist_get_selection_info (clist, x, y, &row, NULL) == 0) {
dir = g_strdup (panel->cwd); file = panel->cwd;
else { fe = file_entry_from_file (file);
if (!fe)
return; /* eeeek */
free_file = FALSE;
free_fe = TRUE;
} else {
g_assert (row < panel->count); g_assert (row < panel->count);
if (S_ISDIR (panel->dir.list [row].buf.st_mode) || file = g_concat_dir_and_file (panel->cwd, panel->dir.list[row].fname);
panel->dir.list [row].f.link_to_dir) fe = &panel->dir.list[row];
dir = concat_dir_and_file (panel->cwd, panel->dir.list [row].fname);
else free_file = TRUE;
dir = g_strdup (panel->cwd); free_fe = FALSE;
} }
gdnd_drop_on_directory (context, selection_data, dir); reload = gdnd_perform_drop (context, selection_data, fe, file);
g_free (dir);
update_one_panel_widget (panel, 0, UP_KEEPSEL); if (free_file)
panel_update_contents (panel); g_free (file);
if (free_fe)
file_entry_free (fe);
if (reload) {
update_one_panel_widget (panel, 0, UP_KEEPSEL);
panel_update_contents (panel);
}
} }
/** /**
@ -857,6 +886,7 @@ panel_tree_drag_data_received (GtkWidget *widget,
GtkDTree *dtree = GTK_DTREE (widget); GtkDTree *dtree = GTK_DTREE (widget);
GtkCTreeNode *node; GtkCTreeNode *node;
int row, col; int row, col;
file_entry *fe;
char *path; char *path;
if (!gtk_clist_get_selection_info (GTK_CLIST (dtree), x, y, &row, &col)) if (!gtk_clist_get_selection_info (GTK_CLIST (dtree), x, y, &row, &col))
@ -866,10 +896,15 @@ panel_tree_drag_data_received (GtkWidget *widget,
if (!node) if (!node)
return; return;
gtk_ctree_expand_recursive (GTK_CTREE (dtree), node); gtk_ctree_expand_recursive (GTK_CTREE (dtree), node);
path = gtk_dtree_get_row_path (dtree, node, 0); path = gtk_dtree_get_row_path (dtree, node, 0);
fe = file_entry_from_file (path);
if (!fe)
return; /* eeeek */
gdnd_drop_on_directory (context, selection_data, path); gdnd_perform_drop (context, selection_data, fe, path);
file_entry_free (fe);
g_free (path); g_free (path);
} }
@ -1073,6 +1108,7 @@ panel_clist_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x, gin
fe = &panel->dir.list[idx]; fe = &panel->dir.list[idx];
action = gdnd_validate_action (context, action = gdnd_validate_action (context,
FALSE,
source_widget != NULL, source_widget != NULL,
source_widget == widget, source_widget == widget,
fe, fe,
@ -1187,6 +1223,7 @@ panel_icon_list_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x,
fe = (idx == -1) ? NULL : &panel->dir.list[idx]; fe = (idx == -1) ? NULL : &panel->dir.list[idx];
action = gdnd_validate_action (context, action = gdnd_validate_action (context,
FALSE,
source_widget != NULL, source_widget != NULL,
source_widget == widget, source_widget == widget,
fe, fe,