/* Desktop management for the Midnight Commander
 *
 * Copyright (C) 1998-1999 The Free Software Foundation
 *
 * Authors: Federico Mena <federico@nuclecu.unam.mx>
 *          Miguel de Icaza <miguel@nuclecu.unam.mx>
 */

#include <config.h>
#include "global.h"
#include <gdk/gdkx.h>
#include <gtk/gtkinvisible.h>
#include <gnome.h>
#include "dialog.h"
#define DIR_H_INCLUDE_HANDLE_DIRENT /* bleah */
#include "dir.h"
#include "ext.h"
#include "file.h"
#include "fileopctx.h"
#include "gconf.h"
#include "gdesktop.h"
#include "gdesktop-icon.h"
#include "gdesktop-init.h"
#include "gicon.h"
#include "gmain.h"
#include "gmetadata.h"
#include "gcmd.h"
#include "gdnd.h"
#include "gpopup.h"
#include "gprint.h"
#include "gscreen.h"
#include "../vfs/vfs.h"
#include "main.h"
#include "gmount.h"


struct layout_slot {
	int num_icons;		      /* Number of icons in this slot */
	GList *icons;		      /* The list of icons in this slot */
};


/* Configuration options for the desktop */


int desktop_use_shaped_icons = TRUE;
int desktop_auto_placement = FALSE;
int desktop_snap_icons = FALSE;
int desktop_arr_r2l = FALSE;
int desktop_arr_b2t = FALSE;
int desktop_arr_rows = FALSE;

/*
 * Possible values for this one:
 *
 * 0 -- Enable shaped text only if window manager is GNOME compliant
 * 1 -- Enable shaped text always
 * 2 -- Disable shaped text
 *
 * This might seem a bit strange, but it is like that for compatibility reasons.
 */

int desktop_use_shaped_text = 0;

/* The computed name of the user's desktop directory */
char *desktop_directory;

/* Time of the last directory reload */
time_t desktop_dir_modify_time;

/* Layout information:  number of rows/columns for the layout slots, and the
 * array of slots.  Each slot is an integer that specifies the number of icons
 * that belong to that slot.
 */
static int layout_spacer_width;
static int layout_spacer_height;
static int layout_screen_width;
static int layout_screen_height;
static int layout_cols;
static int layout_rows;
static struct layout_slot *layout_slots;


int desktop_wm_is_gnome_compliant = -1;

#define l_slots(u, v) (layout_slots[(u) * layout_rows + (v)])

/* The last icon to be selected */
static DesktopIconInfo *last_selected_icon;

/* Drag and drop sources and targets */

static GtkTargetEntry dnd_icon_sources[] = {
	{ TARGET_MC_DESKTOP_ICON_TYPE, 0, TARGET_MC_DESKTOP_ICON },
	{ TARGET_URI_LIST_TYPE, 0, TARGET_URI_LIST },
	{ TARGET_TEXT_PLAIN_TYPE, 0, TARGET_TEXT_PLAIN },
	{ TARGET_URL_TYPE, 0, TARGET_URL }
};

static GtkTargetEntry dnd_icon_targets[] = {
	{ TARGET_MC_DESKTOP_ICON_TYPE, 0, TARGET_MC_DESKTOP_ICON },
	{ TARGET_URI_LIST_TYPE, 0, TARGET_URI_LIST },
	{ TARGET_URL_TYPE, 0, TARGET_URL }
};

static GtkTargetEntry dnd_desktop_targets[] = {
	{ TARGET_MC_DESKTOP_ICON_TYPE, 0, TARGET_MC_DESKTOP_ICON },
	{ TARGET_URI_LIST_TYPE, 0, TARGET_URI_LIST },
	{ TARGET_URL_TYPE, 0, TARGET_URL }
};

static int dnd_icon_nsources = sizeof (dnd_icon_sources) / sizeof (dnd_icon_sources[0]);
static int dnd_icon_ntargets = sizeof (dnd_icon_targets) / sizeof (dnd_icon_targets[0]);
static int dnd_desktop_ntargets = sizeof (dnd_desktop_targets) / sizeof (dnd_desktop_targets[0]);

/* Proxy window for DnD and clicks on the desktop */
static GtkWidget *proxy_invisible;

/* Offsets for the DnD cursor hotspot */
static int dnd_press_x, dnd_press_y;

/* Whether a call to select_icon() is pending because the initial click on an
 * icon needs to be resolved at button release time.  Also, we store the
 * event->state.
 */
static int dnd_select_icon_pending;
static guint dnd_select_icon_pending_state;

/* Whether the button release signal on a desktop icon should be stopped due to
 * the icon being selected by clicking on the text item.
 */
static int icon_select_on_text;

/* Proxy window for clicks on the root window */
static GdkWindow *click_proxy_gdk_window;

/* GC for drawing the rubberband rectangle */
static GdkGC *click_gc;

/* Starting click position and event state for rubberbanding on the desktop */
static int click_start_x;
static int click_start_y;
static int click_start_state;

/* Current mouse position for rubberbanding on the desktop */
static int click_current_x;
static int click_current_y;

static int click_dragging;


static DesktopIconInfo *desktop_icon_info_new (char *filename, char *url, char *caption,
					       int xpos, int ypos);

static GHashTable *icon_hash;

/* Convenience function to figure out the (x, y) position from a slot */
static void
get_pos_from_slot (int u, int v, int *x, int *y)
{
        int xx, yy;

        xx = (layout_spacer_width / 2) +
             (u * (DESKTOP_SNAP_X + layout_spacer_width));

	yy = (layout_spacer_height / 2) +
             (v * (DESKTOP_SNAP_Y + layout_spacer_height));

	*x = CLAMP (xx, 0, layout_screen_width);
	*y = CLAMP (yy, 0, layout_screen_height);

}

/* Convenience function to figure out the slot corresponding to an (x, y) position */
static void
get_slot_from_pos (int x, int y, int *u, int *v)
{
	int uu, vv;

	uu = x / (DESKTOP_SNAP_X + layout_spacer_width);
	vv = y / (DESKTOP_SNAP_Y + layout_spacer_height);

	*u = CLAMP (uu, 0, layout_cols - 1);
	*v = CLAMP (vv, 0, layout_rows - 1);
}

/* Swaps two integer variables */
static void
swap (int *a, int *b)
{
	int tmp;

	tmp = *a;
	*a = *b;
	*b = tmp;
}

/* Looks for a free slot in the layout_slots array and returns the coordinates
 * that coorespond to it.  "Free" means it either has zero icons in it, or it
 * has the minimum number of icons of all the slots.  Returns the number of
 * icons in the sought spot (ideally 0).
 */
static int
auto_pos (int sx, int ex, int sy, int ey, int *slot)
{
	int min, min_slot;
	int x, y;
	int val = 0;
	int xinc, yinc;
	int r, b;

	min = l_slots (sx, sy).num_icons;
	min_slot = sx * layout_rows + sy;
	xinc = yinc = 1;

	if (desktop_arr_rows) {
		swap (&ex, &ey);
		swap (&sx, &sy);
	}

#if 0
	if (desktop_arr_r2l)
		swap (&sx, &ex);

	if (desktop_arr_b2t)
		swap (&sy, &ey);
#endif
	if (desktop_arr_r2l)
		xinc = -1;

	if (desktop_arr_b2t)
		yinc = -1;

	if (desktop_arr_rows)
		swap (&xinc, &yinc);

	r = desktop_arr_r2l;
	b = desktop_arr_b2t;

	if (desktop_arr_rows)
		swap (&r, &b);

	for (x = sx; (r ? (x >= ex) : (x <= ex)); x += xinc) {
		for (y = sy; (b ? (y >= ey) : (y <= ey)); y += yinc) {
			if (desktop_arr_rows)
				val = l_slots (y, x).num_icons;
			else
				val = l_slots (x, y).num_icons;

			if (val < min || val == 0) {
				min = val;
				if (desktop_arr_rows)
					min_slot = (y * layout_rows + x);
				else
					min_slot = (x * layout_rows + y);
				if (val == 0)
					break;
			}
		}

		if (val == 0)
			break;
	}

	*slot = min_slot;
	return min;
}

/* Looks for free space in the icon grid, scanning it in column-wise order */
static void
get_icon_auto_pos (int *x, int *y)
{
	int val1;
	int slot1;
	int slot;
	int sx, sy, ex, ey;
	int u, v;
	int xx, yy;

#if 0
	get_slot_from_pos (*x, *y, &sx, &sy);
#endif
	/* FIXME funky stuff going on with the efficient positioning thingy */
	if (desktop_arr_r2l)
		sx = layout_cols - 1;
	else
		sx = 0;

	if (desktop_arr_b2t)
		sy = layout_rows - 1;
	else
		sy = 0;

	if (desktop_arr_r2l)
		ex = 0;
	else
		ex = layout_cols - 1;

	if (desktop_arr_b2t)
		ey = 0;
	else
		ey = layout_rows - 1;

	/* Look forwards until the end of the grid.  If we could not find an
	 * empty spot, find the second best.
	 */

	val1 = auto_pos (sx, ex, sy, ey, &slot1);

	slot = slot1;

#if 0
	/* to be used at a later date */
	if (val1 == 0)
		slot = slot1;
	else {
		if (desktop_arr_r2l)
			sx = layout_cols - 1;
		else
			sx = 0;

		if (desktop_arr_b2t)
			sy = layout_rows - 1;
		else
			sy = 0;

		val2 = auto_pos (sx, ex, sy, ey, &slot2);
		if (val2 < val1)
			slot = slot2;
	}
#endif
	u = slot / layout_rows;
	v = slot % layout_rows;

	get_pos_from_slot(u, v, &xx, &yy);

	*x = xx;
	*y = yy;

}

/* Snaps the specified position to the icon grid.  It looks for the closest free spot on the grid,
 * or the closest one that has the least number of icons in it.
 */
static void
get_icon_snap_pos (int *x, int *y)
{
	int min, min_x, min_y;
	int min_dist;
	int u, v;
	int val, dist;
	int dx, dy;
	int ux, vy;
	int this_icon_u, this_icon_v;

	min = l_slots (0, 0).num_icons;
	min_x = min_y = 0;
	min_dist = INT_MAX;

	get_slot_from_pos(*x, *y, &this_icon_u, &this_icon_v);

	for (u = 0; u < layout_cols; u++)
		for (v = 0; v < layout_rows; v++) {
			val = l_slots (u, v).num_icons;

			if (u==this_icon_u && v==this_icon_v)
			        val--;
			
			get_pos_from_slot(u, v, &ux, &vy);
			dx = *x - ux;
			dy = *y - vy;
			dist = dx * dx + dy * dy;

			if ((val == min && dist < min_dist) || (val < min)) {
				min = val;
				min_dist = dist;
				min_x = ux;
				min_y = vy;
			}
		}

	*x = min_x;
	*y = min_y;

}

/* Removes an icon from the slot it is in, if any */
static void
remove_from_slot (DesktopIconInfo *dii)
{
	if (dii->slot == -1)
		return;

	g_assert (layout_slots[dii->slot].num_icons >= 1);
	g_assert (layout_slots[dii->slot].icons != NULL);

	layout_slots[dii->slot].num_icons--;
	layout_slots[dii->slot].icons = g_list_remove (layout_slots[dii->slot].icons, dii);

	dii->slot = -1;
	dii->x = 0;
	dii->y = 0;
}

/* Places a desktop icon on the specified position */
static void
desktop_icon_info_place (DesktopIconInfo *dii, int xpos, int ypos)
{
	int u, v;
	char *filename;

	remove_from_slot (dii);

	if (xpos < (layout_spacer_width / 2))
		xpos = layout_spacer_width / 2;
	else if (xpos > layout_screen_width - (DESKTOP_SNAP_X + (layout_spacer_width / 2)))
		xpos = layout_screen_width - (DESKTOP_SNAP_X + (layout_spacer_width / 2));

	if (ypos < (layout_spacer_height / 2))
	        ypos = layout_spacer_height / 2;
	else if (ypos > layout_screen_height - (DESKTOP_SNAP_Y + (layout_spacer_height / 2)))
		ypos = layout_screen_height - (DESKTOP_SNAP_Y + (layout_spacer_height / 2));


	/* Increase the number of icons in the corresponding slot */

	get_slot_from_pos (xpos, ypos, &u, &v);

	dii->slot = u * layout_rows + v;
	layout_slots[dii->slot].num_icons++;
	layout_slots[dii->slot].icons = g_list_append (layout_slots[dii->slot].icons, dii);

	/* Move the icon */

	dii->x = xpos;
	dii->y = ypos;
	gtk_widget_set_uposition (dii->dicon, xpos, ypos);

	/* Save the information */

	filename = g_concat_dir_and_file (desktop_directory, dii->filename);
	gmeta_set_icon_pos (filename, dii->x, dii->y);
	g_free (filename);
}

/* Destroys the specified desktop icon */
static void
desktop_icon_info_destroy (DesktopIconInfo *dii)
{
	if (last_selected_icon == dii)
		last_selected_icon = NULL;

	gtk_widget_destroy (dii->dicon);
	remove_from_slot (dii);

	g_hash_table_remove (icon_hash, dii->filename);

	g_free (dii->url);
	g_free (dii->filename);
	g_free (dii);
}

/* Destroys all the current desktop icons */
static void
destroy_desktop_icons (void)
{
	int i;
	GList *l;
	DesktopIconInfo *dii;

	for (i = 0; i < (layout_cols * layout_rows); i++) {
		l = layout_slots[i].icons;

		while (l) {
			dii = l->data;
			l = l->next;

			desktop_icon_info_destroy (dii);
		}
	}
}

/* Returns a list with all of the icons on the desktop */
static GList *
get_all_icons (void)
{
	GList *l, *res;
	int i;

	res = NULL;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots [i].icons; l; l = l->next)
			res = g_list_prepend (res, l->data);

	return res;
}

/* Returns the node in the list of DesktopIconInfo structures that contains the
 * icon for the specified filename.  If no icon is found, then returns NULL.
 */
static GList *
icon_exists_in_list (GList *list, char *filename)
{
	GList *l;
	DesktopIconInfo *dii;

	for (l = list; l; l = l->next) {
		dii = l->data;

		if (strcmp (filename, dii->filename) == 0)
			return l;
	}

	return NULL;
}

/* Loads from the metadata updated versions of the caption and the url */
static void
update_url (DesktopIconInfo *dii)
{
	char *fullname = g_concat_dir_and_file (desktop_directory, dii->filename);
	char *caption = NULL;
	char *url = NULL;
	int size;

	gnome_metadata_get (fullname, "icon-caption", &size, &caption);
	if (caption){
		desktop_icon_set_text (DESKTOP_ICON (dii->dicon), caption);
		g_free (caption);
	}

	gnome_metadata_get (fullname, "desktop-url", &size, &url);
	if (url){
		if (dii->url)
			g_free (dii->url);
		dii->url = url;
	}

	g_free (fullname);
}

typedef struct {
	char *filename;
	char *url;
	char *caption;
} file_and_url_t;

/* get_drop_grid generates a compact organization for icons that are dragged out
 * of applications on to the desktop.
 */
GSList *
get_drop_grid (gint xpos, gint ypos, gint need_position_list_length, GSList *grid_list)
{
        gint i = 1;
	gint my_bool = 0;
	gint j;
	gint k;
	gint new_xpos;
	gint new_ypos;

	/* Determine size of a square grid in which to put the dropped icons */
	while (!my_bool) {
	        if (need_position_list_length <= i * i)
		        my_bool = 1;
		else
		        i++;
	}

	/* Make sure no icons are off the side of the screen */
	if ((xpos + (i * DESKTOP_SNAP_X) + ((i - 1) * layout_spacer_width) +
	    (layout_spacer_width / 2)) > layout_screen_width) {
	        xpos = layout_screen_width - (layout_spacer_width / 2) -
		       (i * DESKTOP_SNAP_X) - ((i - 1) * layout_spacer_width);
	}
	if ((ypos + (i * DESKTOP_SNAP_Y) + ((i - 1) * layout_spacer_height) +
	     (layout_spacer_height / 2)) > layout_screen_height) {
		ypos = layout_screen_height - (layout_spacer_height / 2) -
		       (i * DESKTOP_SNAP_Y) - ((i - 1) * layout_spacer_height);
	}
	if (xpos < layout_spacer_width / 2) {
	        xpos = layout_spacer_width / 2;
	}
	if (ypos < layout_spacer_height / 2) {
        	ypos = layout_spacer_height / 2;
	}

	/* Now write the (x, y) coordinates of the dropped icons */
	for (j = 0; j < i; j++) {
	        for (k = 0; k < i; k++) {
		        new_xpos = xpos + k * (DESKTOP_SNAP_X + layout_spacer_width);
			new_ypos = ypos + j * (DESKTOP_SNAP_Y + layout_spacer_height);

			if (desktop_snap_icons) {
			        get_icon_snap_pos (&new_xpos, &new_ypos);
			}
	                
			grid_list = g_slist_append (grid_list, (void *) new_xpos);
			grid_list = g_slist_append (grid_list, (void *) new_ypos);
	        }
	}

	return grid_list;

}

#define NAUTILUS_KEY "<?xml version=\"1.0\"?>\n<nautilus_object nautilus_link=\""
#define NAUTILUS_MOUNT_KEY "Mount Link\""
#define NAUTILUS_GENERIC_KEY "Generic Link\""
#define NAUTILUS_HOME_KEY "Home Link\""
#define NAUTILUS_TRASH_KEY "Trash Link\""

static gboolean
should_skip_nautilus_file (char *name)
{
	char buffer [256], *full, *p;
	FILE *f;
	int n;

	if (strcmp (name, ".nautilus-metafile.xml") == 0)
		return TRUE;

	if (strcmp (name, "Trash") == 0)
		return TRUE;

	full = g_concat_dir_and_file (desktop_directory, name);
	f = fopen (full, "r");
	g_free (full);
	
	if (f == NULL)
		return FALSE;
	n = fread (buffer, sizeof (buffer), 1, f);
	fclose (f);

	if (n < 0)
		return FALSE;

	if (strncmp (buffer, NAUTILUS_KEY, strlen (NAUTILUS_KEY)) != 0)
		return FALSE;

	p = buffer + strlen (NAUTILUS_KEY);
	
	if (strncmp (p, NAUTILUS_HOME_KEY, strlen (NAUTILUS_HOME_KEY)) == 0)
		return TRUE;

	if (strncmp (p, NAUTILUS_TRASH_KEY, strlen (NAUTILUS_TRASH_KEY)) == 0)
		return TRUE;
	
	if (strncmp (p, NAUTILUS_MOUNT_KEY, strlen (NAUTILUS_MOUNT_KEY)) == 0)
		return TRUE;

	return FALSE;
}

/* Reloads the desktop icons efficiently.  If there are "new" files for which no
 * icons have been created, then icons for them will be created started at the
 * specified position if user_pos is TRUE.  If it is FALSE, the icons will be
 * auto-placed.
 */
void
desktop_reload_icons (int user_pos, int xpos, int ypos)
{
	struct dirent *dirent;
	DIR *dir;
	char *full_name;
	int have_pos, x, y, size;
	DesktopIconInfo *dii;
	GSList *need_position_list, *sl, *drop_grid_list = NULL, *drop_gl_copy = NULL;
	GList *all_icons, *l;
	char *desktop_url, *caption;
	const char *mime;
	int orig_xpos, orig_ypos;
	guint need_position_list_length;
	static int first_reload = TRUE;

	dir = mc_opendir (desktop_directory);
	if (!dir) {
		message (FALSE,
			 _("Warning"),
			 _("Could not open %s; will not have desktop icons"),
			 desktop_directory);
		return;
	}

	/*
	 * Save the last time we scanned the directory
	 */
	{
		struct stat buf;
		
		if (stat (desktop_directory, &buf) != -1)
			desktop_dir_modify_time = buf.st_ctime;
		else
			desktop_dir_modify_time = time (NULL);
	}
	
	gnome_metadata_lock ();

	/* Read the directory.  For each file for which we do have an existing
	 * icon, do nothing.  Otherwise, store it in a list
	 * of new icons for which positioning is pending. If we have metadata for
	 * the file but no icon, delete the metadata and treat the file as new.
	 */

	need_position_list = NULL;
	all_icons = get_all_icons ();

	while ((dirent = mc_readdir (dir)) != NULL) {
		/* Skip . and .. */

		if ((dirent->d_name[0] == '.' && dirent->d_name[1] == 0)
		    || (dirent->d_name[0] == '.' && dirent->d_name[1] == '.'
			&& dirent->d_name[2] == 0))
			continue;

		if (should_skip_nautilus_file (dirent->d_name))
			continue;
		    
		l = icon_exists_in_list (all_icons, dirent->d_name);
		if (l) {
			GdkImlibImage *im;
			file_entry *fe;

			/* Reload the icon for this file, as it may have changed */

			full_name = g_concat_dir_and_file (desktop_directory, dirent->d_name);
			fe = file_entry_from_file (full_name);
			if (!fe){
				g_free (full_name);
				continue;
			}
			im = gicon_get_icon_for_file (desktop_directory, fe, FALSE);
			file_entry_free (fe);
			g_free (full_name);

			dii = l->data;
			desktop_icon_set_icon (DESKTOP_ICON (dii->dicon), im);

			update_url (dii);

			/* Leave the icon in the desktop by removing it from the list */

			all_icons = g_list_remove_link (all_icons, l);
			continue;
		}

		full_name = g_concat_dir_and_file (desktop_directory, dirent->d_name);

		have_pos = gmeta_get_icon_pos (full_name, &x, &y);

		if (gnome_metadata_get (full_name, "desktop-url", &size, &desktop_url) != 0)
			desktop_url = NULL;

		caption = NULL;
		gnome_metadata_get (full_name, "icon-caption", &size, &caption);

		if (first_reload && have_pos) {
		        dii = desktop_icon_info_new (dirent->d_name, desktop_url, caption, x, y);
		        gtk_widget_show (dii->dicon);
		} else {
		        file_and_url_t *fau;
		        fau = g_new0 (file_and_url_t, 1);
		        fau->filename = g_strdup (dirent->d_name);

		        if (desktop_url)
		                fau->url = g_strdup (desktop_url);

		        if (caption)
		                fau->caption  = g_strdup (caption);

		        need_position_list = g_slist_prepend (need_position_list, fau);
		}
		
		g_free (full_name);

		if (desktop_url)
			g_free (desktop_url);

		if (caption)
			g_free (caption);
	}

	if (first_reload)
	        first_reload = FALSE;

	mc_closedir (dir);

	/* all_icons now contains a list of all of the icons that were not found
	 * in the ~/desktop directory, remove them.  To be paranoically tidy,
	 * also delete the icon position information -- someone may have deleted
	 * a file under us.
	 */

	for (l = all_icons; l; l = l->next) {
		dii = l->data;
		full_name = g_concat_dir_and_file (desktop_directory, dii->filename);
		gnome_metadata_delete (full_name);
		g_free (full_name);
		desktop_icon_info_destroy (dii);
	}

	g_list_free (all_icons);

	/* Now create the icons for all the files that did not have their
	 * position set.  This makes auto-placement work correctly without
	 * overlapping icons.
	 */

	need_position_list = g_slist_reverse (need_position_list);

	need_position_list_length = g_slist_length (need_position_list);

	/* Get a position list for the dragged icons, to nicely place multiple icon drops */
	if (user_pos && (need_position_list_length != 0)) {
		drop_grid_list = NULL;
	        drop_grid_list = get_drop_grid (xpos, ypos, need_position_list_length, drop_grid_list);
		drop_gl_copy = drop_grid_list;
	}

	orig_xpos = orig_ypos = 0;

	for (sl = need_position_list; sl; sl = sl->next) {
		file_and_url_t *fau;

		fau = sl->data;

		if (user_pos) {
			if (desktop_auto_placement) {
				xpos = ypos = 0;
				get_icon_auto_pos (&xpos, &ypos);
			} else { /* Place the icons according to the position list obtained above */
			        xpos = (int) drop_gl_copy->data;
				drop_gl_copy = drop_gl_copy->next;
				ypos = (int) drop_gl_copy->data;
				drop_gl_copy = drop_gl_copy->next;
			}
		} else {
			xpos = orig_xpos;
			ypos = orig_ypos;
			get_icon_auto_pos (&xpos, &ypos);
		}

		/* If the file dropped was a .desktop file, pull the suggested
		 * title and icon from there
		 */
		if (use_magic)
		{
			char *full_name;
			full_name = g_concat_dir_and_file (desktop_directory, fau->filename);
			mime = gnome_mime_type_or_default_of_file (full_name, NULL);
			g_free (full_name);
		}
		else
			mime = gnome_mime_type_or_default (fau->filename, NULL);
		if (mime && strcmp (mime, "application/x-gnome-app-info") == 0) {
			GnomeDesktopEntry *entry;
			char *fullname;

			fullname = g_concat_dir_and_file (desktop_directory, fau->filename);
			entry = gnome_desktop_entry_load (fullname);
			if (entry) {
				if (entry->name) {
					if (fau->caption)
						g_free (fau->caption);

					fau->caption = g_strdup (entry->name);
					gnome_metadata_set (fullname, "icon-caption",
							    strlen (fau->caption) + 1,
							    fau->caption);
				}

				if (entry->icon)
					gnome_metadata_set (fullname, "icon-filename",
							    strlen (entry->icon) + 1,
							    entry->icon);

				gnome_desktop_entry_free (entry);
			}
			g_free (fullname);
		}

		dii = desktop_icon_info_new (fau->filename, fau->url, fau->caption, xpos, ypos);
		gtk_widget_show (dii->dicon);

		if (fau->url)
			g_free (fau->url);

		if (fau->caption)
			g_free (fau->caption);

		g_free (fau->filename);
		g_free (fau);
	}

	if ((need_position_list_length != 0) && user_pos) {
	        g_slist_free (drop_grid_list);
	}

	g_slist_free (need_position_list);

	gnome_metadata_unlock ();
}

static WPanel *create_panel_from_desktop(); /* Fwd decl */
static void free_panel_from_desktop(WPanel *panel);

/* Run all of the desktop icons through snap-to-grid */
void
desktop_tidy_icons (void)
{
        WPanel *panel;
        DesktopIconInfo *dii;
        int i;
        dir_list dir;
        int xpos, ypos;
    
        panel = create_panel_from_desktop ();
        if (panel->count == 0) {
	        free_panel_from_desktop (panel);
	        return;
        }
    
        dir = panel->dir;
        g_assert (dir.list != NULL);
    
        for (i = 0; i < dir.size; i++) {
	        dii = desktop_icon_info_get_by_filename (dir.list[i].fname);
	        xpos = dii->x;
	        ypos = dii->y;
	        get_icon_snap_pos (&xpos, &ypos);
	        desktop_icon_info_place (dii, xpos, ypos);
        }
    
        free_panel_from_desktop (panel);
}


/* Perform automatic arrangement of the desktop icons */
void
desktop_arrange_icons (SortType type)
{
	WPanel *panel;
	sortfn *sfn;
	DesktopIconInfo *dii;
	int i;
	dir_list dir;
	int xpos, ypos;

	panel = create_panel_from_desktop ();
	if (panel->count == 0) {
		free_panel_from_desktop (panel);
		return;
	}

	sfn = sort_get_func_from_type (type);
	g_assert (sfn != NULL);

	do_sort (&panel->dir, sfn, panel->count - 1, FALSE, FALSE);
	dir = panel->dir;
	g_assert (dir.list != NULL);

	for (i = 0; i < dir.size; i++)
	        remove_from_slot (desktop_icon_info_get_by_filename (dir.list[i].fname));

	for (i = 0; i < dir.size; i++) {
		dii = desktop_icon_info_get_by_filename (dir.list[i].fname);
		xpos = ypos = 0;
		get_icon_auto_pos (&xpos, &ypos);
		desktop_icon_info_place (dii, xpos, ypos);
	}

	free_panel_from_desktop (panel);
}

/* Unselects all the desktop icons except the one in exclude */
static void
unselect_all (DesktopIconInfo *exclude)
{
	int i;
	GList *l;
	DesktopIconInfo *dii;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			if (dii->selected && dii != exclude) {
				desktop_icon_select (DESKTOP_ICON (dii->dicon), FALSE);
				dii->selected = FALSE;
			}
		}
}

/* Sets the selection state of a range to the specified value.  The range starts at the
 * last_selected_icon and ends at the specified icon.
 */
static void
select_range (DesktopIconInfo *dii, int sel)
{
	int du, dv, lu, lv;
	int min_u, min_v;
	int max_u, max_v;
	int u, v;
	GList *l;
	DesktopIconInfo *ldii;

	/* Find out the selection range */

	if (!last_selected_icon)
		last_selected_icon = dii;

	du = dii->slot / layout_rows;
	dv = dii->slot % layout_rows;
	lu = last_selected_icon->slot / layout_rows;
	lv = last_selected_icon->slot % layout_rows;

	if (du < lu) {
		min_u = du;
		max_u = lu;
	} else {
		min_u = lu;
		max_u = du;
	}

	if (dv < lv) {
		min_v = dv;
		max_v = lv;
	} else {
		min_v = lv;
		max_v = dv;
	}

	/* Select all the icons in the rectangle */

	for (u = min_u; u <= max_u; u++)
		for (v = min_v; v <= max_v; v++)
			for (l = l_slots (u, v).icons; l; l = l->next) {
				ldii = l->data;

				desktop_icon_select (DESKTOP_ICON (ldii->dicon), sel);
				ldii->selected = sel;
			}
}

/*
 * Handles icon selection and unselection due to button presses.  The
 * event_state is the state field of the event.
 */
static void
select_icon (DesktopIconInfo *dii, int event_state)
{
	int range;
	int additive;

	range = ((event_state & GDK_SHIFT_MASK) != 0);
	additive = ((event_state & GDK_CONTROL_MASK) != 0);

	if (!additive)
		unselect_all (NULL);

	if (!range) {
		if (additive) {
			desktop_icon_select (DESKTOP_ICON (dii->dicon), !dii->selected);
			dii->selected = !dii->selected;
		} else if (!dii->selected) {
			desktop_icon_select (DESKTOP_ICON (dii->dicon), TRUE);
			dii->selected = TRUE;
		}

		last_selected_icon = dii;

		if (dii->selected)
			gdk_window_raise (dii->dicon->window);
	} else
		select_range (dii, TRUE);
}

/* Convenience function to fill a file entry */
static void
file_entry_fill (file_entry *fe, struct stat *s, char *filename)
{
	fe->fname = g_strdup (x_basename (filename));
	fe->fnamelen = strlen (fe->fname);
	fe->buf = *s;
	fe->f.marked = FALSE;
	fe->f.link_to_dir = FALSE;
	fe->f.stalled_link = FALSE;
	fe->f.dir_size_computed = FALSE;

	if (S_ISLNK (s->st_mode)) {
		struct stat s2;

		if (mc_stat (filename, &s2) == 0)
			fe->f.link_to_dir = S_ISDIR (s2.st_mode) != 0;
		else
			fe->f.stalled_link = TRUE;
	}
}

/* Creates a file entry structure and fills it with information appropriate to the specified file.  */
file_entry *
file_entry_from_file (char *filename)
{
	file_entry *fe;
	struct stat s;

	if (mc_lstat (filename, &s) == -1) {
		g_warning ("Could not stat %s, bad things will happen", filename);
		return NULL;
	}

	fe = g_new (file_entry, 1);
	file_entry_fill (fe, &s, filename);
	return fe;
}

/* Frees a file entry structure */
void
file_entry_free (file_entry *fe)
{
	if (fe->fname)
		g_free (fe->fname);

	g_free (fe);
}

/* Sets the wmclass and name of a desktop icon to an unique value */
static void
set_icon_wmclass (DesktopIconInfo *dii)
{
	XClassHint *h;

	g_assert (GTK_WIDGET_REALIZED (dii->dicon));

	h = XAllocClassHint ();
	if (!h) {
		g_warning ("XAllocClassHint() failed!");
		return; /* eek */
	}

	h->res_name = dii->filename;
	h->res_class = "gmc-desktop-icon";

	XSetClassHint (GDK_DISPLAY (), GDK_WINDOW_XWINDOW (GTK_WIDGET (dii->dicon)->window), h);
	XFree (h);
}

/* Removes the Gtk and Gdk grabs that are present when editing a desktop icon */
static void
remove_editing_grab (DesktopIconInfo *dii)
{
	gtk_grab_remove (dii->dicon);
	gdk_pointer_ungrab (GDK_CURRENT_TIME);
	gdk_keyboard_ungrab (GDK_CURRENT_TIME);
}

/* Callback used when an icon's text changes.  We must validate the
 * rename and return the appropriate value.  The desktop icon info
 * structure is passed in the user data.
 */
static int
text_changed (GnomeIconTextItem *iti, gpointer data)
{
	DesktopIconInfo *dii;
	char *new_name;
	char *source;
	char *dest;
	char *buf;
	int size;
	int retval;

	dii = data;
	remove_editing_grab (dii);

	source = g_concat_dir_and_file (desktop_directory, dii->filename);
	new_name = gnome_icon_text_item_get_text (iti);

	if (gnome_metadata_get (source, "icon-caption", &size, &buf) != 0) {
		if (strcmp (new_name, dii->filename) == 0) {
			g_free (source);
			return TRUE;
		}

		/* No icon caption metadata, so rename the file */

		dest = g_concat_dir_and_file (desktop_directory, new_name);

		if (rename_file_with_context (source, dest) == FILE_CONT) {
			GList *icons;
			GList *l;

			/* See if there was an icon for the new name.  If so,
			 * destroy it first; desktop_reload_icons() will not be
			 * happy if two icons have the same filename.
			 */

			icons = get_all_icons ();
			l = icon_exists_in_list (icons, new_name);
			if (l)
				desktop_icon_info_destroy (l->data);

			g_list_free (icons);

			/* Set the new name */

			g_hash_table_remove (icon_hash, dii->filename);
			g_free (dii->filename);

			dii->filename = g_strdup (new_name);
			g_hash_table_insert (icon_hash, dii->filename, dii);
			set_icon_wmclass (dii);

			desktop_reload_icons (FALSE, 0, 0);
			retval = TRUE;
		} else
			retval = FALSE; /* FIXME: maybe pop up a warning/query dialog? */

		g_free (dest);
	} else {
		/* The icon has the icon-caption metadata, so change that instead */

		g_free (buf);

		buf = gnome_icon_text_item_get_text (iti);
		if (*buf) {
			gnome_metadata_set (source, "icon-caption", strlen (buf) + 1, buf);
			desktop_reload_icons (FALSE, 0, 0);
			retval = TRUE;
		} else
			retval = FALSE;
	}

	g_free (source);
	return retval;
}

/* Sets up the mouse grab for when a desktop icon is being edited */
static void
setup_editing_grab (DesktopIconInfo *dii)
{
	GdkCursor *ibeam;

	ibeam = gdk_cursor_new (GDK_XTERM);
	gdk_pointer_grab (dii->dicon->window,
			  TRUE,
			  (GDK_BUTTON_PRESS_MASK
			   | GDK_BUTTON_RELEASE_MASK
			   | GDK_POINTER_MOTION_MASK
			   | GDK_ENTER_NOTIFY_MASK
			   | GDK_LEAVE_NOTIFY_MASK),
			  NULL,
			  ibeam,
			  GDK_CURRENT_TIME);
	gdk_cursor_destroy (ibeam);
}

/*
 * Callback used when the user begins editing the icon text item in a
 * desktop icon.  It installs the mouse and keyboard grabs that are
 * required while an icon is being edited.
 */
static void
editing_started (GnomeIconTextItem *iti, gpointer data)
{
	DesktopIconInfo *dii;

	dii = data;

	/* Disable drags from this icon until editing is finished */

	gtk_drag_source_unset (DESKTOP_ICON (dii->dicon)->canvas);

	/* Unselect all icons but this one */
	unselect_all (dii);

	gtk_grab_add (dii->dicon);
	setup_editing_grab (dii);
	gdk_keyboard_grab (GTK_LAYOUT (DESKTOP_ICON (dii->dicon)->canvas)->bin_window,
			   FALSE, GDK_CURRENT_TIME);
}

/* Sets up the specified icon as a drag source, but does not connect the signals */
static void
setup_icon_dnd_actions (DesktopIconInfo *dii)
{
	gtk_drag_source_set (DESKTOP_ICON (dii->dicon)->canvas,
			     GDK_BUTTON1_MASK | GDK_BUTTON2_MASK,
			     dnd_icon_sources,
			     dnd_icon_nsources,
			     GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK | GDK_ACTION_ASK);
}

/*
 * Callback used when the user finishes editing the icon text item in
 * a desktop icon.  It removes the mouse and keyboard grabs.
 */
static void
editing_stopped (GnomeIconTextItem *iti, gpointer data)
{
	DesktopIconInfo *dii;

	dii = data;
	remove_editing_grab (dii);

	/* Re-enable drags from this icon */

	setup_icon_dnd_actions (dii);
}

/* Callback used when the user stops selecting text in a desktop icon.  This function
 * restores the mouse grab that we had set up initially (the selection process changes
 * the grab and then removes it, so we need to restore the initial grab).
 */
static void
selection_stopped (GnomeIconTextItem *iti, gpointer data)
{
	DesktopIconInfo *dii;

	dii = data;
	setup_editing_grab (dii);
}

static char *mount_known_locations [] = {
	"/sbin/mount", "/bin/mount", "/etc/mount",
	"/usr/sbin/mount", "/usr/etc/mount", "/usr/bin/mount",
	NULL
};

static char *umount_known_locations [] = {
	"/sbin/umount", "/bin/umount", "/etc/umount",
	"/usr/sbin/umount", "/usr/etc/umount", "/usr/bin/umount",
	NULL
};

/*
 * Returns the full path to the mount command
 */
static char *
find_command (char **known_locations)
{
	int i;

	for (i = 0; known_locations [i]; i++){
		if (g_file_exists (known_locations [i]))
			return known_locations [i];
	}
	return NULL;
}

gboolean
is_mountable (char *filename, file_entry *fe, int *is_mounted, char **point)
{
	char buffer [128], *p;
	int len;

	if (point)
		*point = NULL;

	if (!S_ISLNK (fe->buf.st_mode))
		return FALSE;

	len = readlink (filename, buffer, sizeof (buffer));
	if (len == -1)
		return FALSE;
	buffer [len] = 0;

	p = is_block_device_mountable (buffer);
	if (!p)
		return FALSE;

	if (point)
		*point = p;
	else
		g_free (point);

	*is_mounted = is_block_device_mounted (buffer);

	return TRUE;
}

gboolean
do_mount_umount (char *filename, gboolean is_mount)
{
	static char *mount_command;
	static char *umount_command;
	char *op;
	char *buffer;

	if (is_mount){
		if (!mount_command)
			mount_command = find_command (mount_known_locations);
		op = mount_command;
	} else {
		if (!umount_command)
			umount_command = find_command (umount_known_locations);
		op = umount_command;
	}

	buffer = g_readlink (filename);
	if (buffer == NULL)
		return FALSE;

	if (op){
		gboolean success = TRUE;
		char *command;
		FILE *f;

		command = g_strconcat (op, " ", buffer, NULL);
		open_error_pipe ();
		f = popen (command, "r");
		if (f == NULL){
			success = !close_error_pipe (1, _("While running the mount/umount command"));
		} else
			success = !close_error_pipe (0, 0);
		pclose (f);

		g_free (buffer);
		return success;
	}
	g_free (buffer);
	return FALSE;
}

static char *eject_known_locations [] = {
	"/usr/bin/eject",
	"/sbin/eject",
	"/bin/eject",
	NULL
};

/*
 * Returns whether the device is ejectable
 *
 * Right now the test only checks if this system has the eject
 * command
 */
gboolean
is_ejectable (char *filename)
{
	char *buf;
	int size, retval;

	if (gnome_metadata_get (filename, "device-is-ejectable", &size, &buf) == 0){
		if (*buf)
			retval = TRUE;
		else
			retval = FALSE;
		g_free (buf);

		if (retval)
			return TRUE;
	}

	if (find_command (eject_known_locations))
		return TRUE;
	else
		return FALSE;
}

/*
 * Ejects the device pointed by filename
 */
gboolean
do_eject (char *filename)
{
	char *eject_command = find_command (eject_known_locations);
	char *command, *device;
	FILE *f;

	if (!eject_command)
		return FALSE;

	device = mount_point_to_device (filename);
	if (!device)
		return FALSE;

	command = g_strconcat (eject_command, " ", device, NULL);
	open_error_pipe ();
	f = popen (command, "r");
	if (f == NULL)
		close_error_pipe (1, _("While running the eject command"));
	else
		close_error_pipe (0, 0);

	pclose (f);

	return TRUE;
}

static gboolean
try_to_mount (char *filename, file_entry *fe)
{
	int x;

	if (!is_mountable (filename, fe, &x, NULL))
		return FALSE;

	return do_mount_umount (filename, TRUE);
}

/* This is a HORRIBLE HACK.  It creates a temporary panel structure for gpopup's
 * perusal.  Once gmc is rewritten, all file lists including panels will be a
 * single data structure, and the world will be happy again.
 */
static WPanel *
create_panel_from_desktop (void)
{
	WPanel *panel;
	int nicons, count;
	int marked_count, dir_marked_count;
	long total;
	int selected_index;
	int i;
	file_entry *fe;
	GList *l;
	struct stat s;

	panel = g_new0 (WPanel, 1);

	/* Count the number of desktop icons */

	nicons = 0;
	for (i = 0; i < layout_cols * layout_rows; i++)
		nicons += layout_slots[i].num_icons;

	/* Create the file entry list */

	panel->dir.size = nicons;

	count = 0;
	marked_count = 0;
	dir_marked_count = 0;
	total = 0;
	selected_index = -1;

	if (nicons != 0) {
		panel->dir.list = g_new (file_entry, nicons);

		fe = panel->dir.list;

		for (i = 0; i < layout_cols * layout_rows; i++)
			for (l = layout_slots[i].icons; l; l = l->next) {
				DesktopIconInfo *dii;
				char *full_name;

				dii = l->data;
				full_name = g_concat_dir_and_file (desktop_directory, dii->filename);
				if (mc_lstat (full_name, &s) == -1) {
					g_warning ("Could not stat %s, bad things will happen",
						   full_name);
					continue;
				}

				file_entry_fill (fe, &s, full_name);
				if (dii->selected) {
					marked_count++;
					fe->f.marked = TRUE;

					if (dii == last_selected_icon)
						selected_index = count;

					if (S_ISDIR (fe->buf.st_mode)) {
						dir_marked_count++;
						if (fe->f.dir_size_computed)
							total += fe->buf.st_size;
					} else
						total += fe->buf.st_size;
				}

				g_free (full_name);
				fe++;
				count++;
			}
	}

	/* Fill the rest of the panel structure */

	panel->list_type = list_icons;
	strncpy (panel->cwd, desktop_directory, sizeof (panel->cwd));
	panel->count = count; /* the actual number of lstat()ed files */
	panel->marked = marked_count;
	panel->dirs_marked = dir_marked_count;
	panel->total = total;
	panel->selected = selected_index;
	panel->is_a_desktop_panel = TRUE;
	panel->id = -1;

	return panel;
}

/* Pushes our hacked-up desktop panel into the containers list */
WPanel *
push_desktop_panel_hack (void)
{
	WPanel *panel;
	PanelContainer *container;

	panel = create_panel_from_desktop ();
	container = g_new (PanelContainer, 1);
	container->splitted = FALSE;
	container->panel = panel;

	containers = g_list_append (containers, container);

	if (!current_panel_ptr)
		current_panel_ptr = container;
	else if (!other_panel_ptr)
		other_panel_ptr = container;

	/* Set it as the current panel and invoke the menu */

	set_current_panel (panel);
	mc_chdir (desktop_directory);
	return panel;
}

/* Frees our hacked-up panel created in the function above */
static void
free_panel_from_desktop (WPanel *panel)
{
	int i;

	for (i = 0; i < panel->count; i++)
		g_free (panel->dir.list[i].fname);

	if (panel->dir.list)
		g_free (panel->dir.list);

	g_free (panel);
}

/**
 * desktop_icon_info_open:
 * @dii: The desktop icon to open.
 *
 * Opens the specified desktop icon when the user double-clicks on it.
 **/
void
desktop_icon_info_open (DesktopIconInfo *dii)
{
	char *filename;
	file_entry *fe;
	int is_mounted;
	char *point;
	int launch;

	filename = NULL;
	fe = NULL;
	launch = FALSE;

	/* Set the cursor */

	desktop_icon_set_busy (dii, TRUE);

	/* Open the icon */

	if (dii->url) {
		gnome_url_show (dii->url);
		goto out;
	}

	filename = g_concat_dir_and_file (desktop_directory, dii->filename);
	fe = file_entry_from_file (filename);
	if (!fe){
		message (1, _("Error"), "I could not fetch the information from the file");
		goto out;
	}

	if (is_mountable (filename, fe, &is_mounted, &point)){
		if (!is_mounted){
			if (try_to_mount (filename, fe))
				launch = TRUE;
			else
				launch = FALSE;
		} else
			launch = TRUE;

		if (launch)
			new_panel_at (point);
		g_free (point);
	} else {
		WPanel *panel;
		panel = push_desktop_panel_hack ();
		do_enter_on_file_entry (fe);
		layout_panel_gone (panel);
		free_panel_from_desktop (panel);
	}

 out:

	if (fe)
		file_entry_free (fe);

	if (filename)
		g_free (filename);

	/* Reset the cursor */

	desktop_icon_set_busy (dii, FALSE);
}

void
desktop_icon_info_delete (DesktopIconInfo *dii)
{
	char *full_name;
	struct stat s;
	FileOpContext *ctx;
	long progress_count = 0;
	double progress_bytes = 0;

	/* 1. Delete the file */
	ctx = file_op_context_new ();
	file_op_context_create_ui (ctx, OP_DELETE, TRUE);
	x_flush_events ();

	full_name = g_concat_dir_and_file (desktop_directory, dii->filename);

	if (lstat (full_name, &s) != -1) {
		if (S_ISLNK (s.st_mode))
			erase_file (ctx, full_name, &progress_count, &progress_bytes, TRUE);
		else {
			if (S_ISDIR (s.st_mode))
				erase_dir (ctx, full_name, &progress_count, &progress_bytes);
			else
				erase_file (ctx, full_name, &progress_count, &progress_bytes, TRUE);
		}

		gnome_metadata_delete (full_name);
	}

	g_free (full_name);
	file_op_context_destroy (ctx);

	/* 2. Destroy the dicon */
	desktop_icon_info_destroy (dii);
}

/**
 * desktop_icon_set_busy:
 * @dii: A desktop icon
 * @busy: TRUE to set a watch cursor, FALSE to reset the normal arrow cursor
 *
 * Sets a wait/normal cursor for a desktop icon.
 **/
void
desktop_icon_set_busy (DesktopIconInfo *dii, int busy)
{
	GdkCursor *cursor;

	g_return_if_fail (dii != NULL);

	if (!GTK_WIDGET_REALIZED (dii->dicon))
		return;

	cursor = gdk_cursor_new (busy ? GDK_WATCH : GDK_LEFT_PTR);
	gdk_window_set_cursor (dii->dicon->window, cursor);
	gdk_cursor_destroy (cursor);
	gdk_flush ();
}

/**
 * desktop_icon_info_get_by_filename:
 * @filename: A filename relative to the desktop directory
 *
 * Returns the desktop icon structure that corresponds to the specified filename,
 * which should be relative to the desktop directory.
 *
 * Return value: The sought desktop icon, or NULL if it is not found.
 **/
DesktopIconInfo *
desktop_icon_info_get_by_filename (char *filename)
{
	g_return_val_if_fail (filename != NULL, NULL);

	return g_hash_table_lookup (icon_hash, filename);
}

/* Used to execute the popup menu for desktop icons */
static void
do_popup_menu (DesktopIconInfo *dii, GdkEventButton *event)
{
	char *filename;
	WPanel *panel;
	DesktopIconInfo *dii_temp;

	/* Create the panel and the container structure */
	panel = push_desktop_panel_hack ();
	dii_temp = NULL;
	if (panel->marked == 1)
		dii_temp = dii;

	filename = g_concat_dir_and_file (desktop_directory, dii->filename);

	if (gpopup_do_popup2 (event, panel, dii_temp) != -1)
		desktop_reload_icons (FALSE, 0, 0);

	g_free (filename);
	layout_panel_gone (panel);
	free_panel_from_desktop (panel);
}

/* Idle handler that opens a desktop icon.  See below for information on why we
 * do things this way.
 */
static gint
idle_open_icon (gpointer data)
{
	desktop_icon_info_open (data);
	return FALSE;
}

/* Event handler for desktop icons.  Button events are ignored when the icon is
 * being edited; these will be handled either by the icon's text item or the
 * icon_event_after() fallback.
 */
static gint
icon_event (GnomeCanvasItem *item, GdkEvent *event, gpointer data)
{
	DesktopIconInfo *dii;
	GnomeIconTextItem *iti;
	int on_text;
	int retval;

	dii = data;
	iti = GNOME_ICON_TEXT_ITEM (DESKTOP_ICON (dii->dicon)->text);

	on_text = item == GNOME_CANVAS_ITEM (iti);
	retval = FALSE;

	switch (event->type) {
	case GDK_BUTTON_PRESS:
		if (event->button.button == 1 || event->button.button == 2) {
			/* If se are editing, do not handle the event ourselves
			 * -- either let the text item handle it, or wait until
			 * we fall back to the icon_event_after() callback.
			 */
			if (iti->editing)
				break;

			/* Save the mouse position for DnD */

			dnd_press_x = event->button.x;
			dnd_press_y = event->button.y;

			/* Handle icon selection if we are not on the text item
			 * or if the icon is not selected in the first place.
			 * Otherwise, if there are modifier keys pressed, handle
			 * icon selection instead of starting editing.
			 */
			if (!on_text
			    || !dii->selected
			    || (event->button.state & (GDK_CONTROL_MASK | GDK_SHIFT_MASK))
			    || event->button.button == 2) {
				/* If click on text, and the icon was not
				 * selected in the first place or shift is down,
				 * save this flag.
				 */
				if (on_text
				    && (!dii->selected || (event->button.state & GDK_SHIFT_MASK)))
					icon_select_on_text = TRUE;

				if ((dii->selected
				     && !(event->button.state & (GDK_CONTROL_MASK | GDK_SHIFT_MASK)))
				    || ((event->button.state & GDK_CONTROL_MASK)
					&& !(event->button.state & GDK_SHIFT_MASK))) {
					dnd_select_icon_pending = TRUE;
					dnd_select_icon_pending_state = event->button.state;
				} else
					select_icon (dii, event->button.state);

				retval = TRUE;
			}
		} else if (event->button.button == 3) {
			if (!dii->selected)
				select_icon (dii, event->button.state);

			do_popup_menu (dii, (GdkEventButton *) event);
			retval = TRUE;
		}
		break;

	case GDK_2BUTTON_PRESS:
		if (event->button.button != 1 || iti->editing)
			break;

		/* We have an interesting race condition here.  If we open the
		 * desktop icon here instead of doing it in the idle handler,
		 * the icon thinks it must begin editing itself, even when the
		 * icon_select_on_text flag tries to prevent it.  I have no idea
		 * why this happens :-( - Federico
		 */
		gtk_idle_add (idle_open_icon, dii);
/*  		desktop_icon_info_open (dii); */
		icon_select_on_text = TRUE;
		retval = TRUE;
		break;

	case GDK_BUTTON_RELEASE:
		if (!(event->button.button == 1 || event->button.button != 2))
			break;

		if (on_text && icon_select_on_text) {
			icon_select_on_text = FALSE;
			retval = TRUE;
		}

		if (dnd_select_icon_pending) {
			select_icon (dii, dnd_select_icon_pending_state);
			dnd_select_icon_pending = FALSE;
			dnd_select_icon_pending_state = 0;
			retval = TRUE;
		}
		break;

	default:
		break;
	}

	/* If the click was on the text and we actually did something, then we
	 * need to stop the text item's event handler from executing.
	 */
	if (on_text && retval)
		gtk_signal_emit_stop_by_name (GTK_OBJECT (iti), "event");

	return retval;
}

/* Fallback handler for when the icon is being edited and the user clicks
 * outside the icon's text item.  This indicates that editing should be accepted
 * and terminated.
 */
static gint
icon_event_after (GtkWidget *widget, GdkEventButton *event, gpointer data)
{
	DesktopIconInfo *dii;
	GnomeIconTextItem *iti;

	dii = data;
	iti = GNOME_ICON_TEXT_ITEM (DESKTOP_ICON (dii->dicon)->text);

	if (event->type != GDK_BUTTON_PRESS)
		return FALSE;

	g_return_val_if_fail (iti->editing, FALSE); /* sanity check for dropped events */

	gnome_icon_text_item_stop_editing (iti, TRUE);
	return TRUE;
}

/* Callback used when a drag from the desktop icons is started.  We set the drag icon to the proper
 * pixmap.
 */
static void
drag_begin (GtkWidget *widget, GdkDragContext *context, gpointer data)
{
	DesktopIconInfo *dii;
	DesktopIcon *dicon;
	GtkArg args[3];
	GdkImlibImage *im;
	GdkPixmap *pixmap;
	GdkBitmap *mask;
	int x, y;

	dii = data;
	dicon = DESKTOP_ICON (dii->dicon);

	/* See if the icon was pending to be selected */

	if (dnd_select_icon_pending) {
		if (!dii->selected)
			select_icon (dii, dnd_select_icon_pending_state);

		dnd_select_icon_pending = FALSE;
		dnd_select_icon_pending_state = 0;
	}

	/* FIXME: see if it is more than one icon and if so, use a multiple-files icon. */

	args[0].name = "image";
	args[1].name = "x";
	args[2].name = "y";
	gtk_object_getv (GTK_OBJECT (dicon->icon), 3, args);
	im = GTK_VALUE_BOXED (args[0]);
	x = GTK_VALUE_DOUBLE (args[1]);
	y = GTK_VALUE_DOUBLE (args[2]);

	gdk_imlib_render (im, im->rgb_width, im->rgb_height);
	pixmap = gdk_imlib_copy_image (im);
	mask = gdk_imlib_copy_mask (im);

	gtk_drag_set_icon_pixmap (context,
				  gtk_widget_get_colormap (dicon->canvas),
				  pixmap,
				  mask,
				  dnd_press_x - x,
				  dnd_press_y - y);

	gdk_pixmap_unref (pixmap);
	gdk_bitmap_unref (mask);
}

/* Builds a string with the URI-list of the selected desktop icons */
static char *
build_selected_icons_uri_list (int *len)
{
	int i;
	GList *l;
	DesktopIconInfo *dii;
	char *filelist, *p;
	int desktop_dir_len;

	/* First, count the number of selected icons and add up their filename lengths */

	*len = 0;
	desktop_dir_len = strlen (desktop_directory);

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			/* "file:" + desktop_directory + "/" + dii->filename + "\r\n" */

			if (dii->selected)
				*len += 5 + desktop_dir_len + 1 + strlen (dii->filename) + 2;
		}

	/* Second, create the file list string */

	filelist = g_new (char, *len + 1); /* plus 1 for null terminator */
	p = filelist;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			if (dii->selected) {
				strcpy (p, "file:");
				p += 5;

				strcpy (p, desktop_directory);
				p += desktop_dir_len;

				*p++ = '/';

				strcpy (p, dii->filename);
				p += strlen (dii->filename);

				strcpy (p, "\r\n");
				p += 2;
			}
		}

	*p = 0;

	return filelist;
}

/* Callback used to get the drag data from the desktop icons */
static void
drag_data_get (GtkWidget *widget, GdkDragContext *context, GtkSelectionData *selection_data,
	       guint info, guint32 time, gpointer data)
{
	DesktopIconInfo *dii;
	char *filelist;
	int len;
	GList *files;

	dii = data;
	filelist = build_selected_icons_uri_list (&len);

	switch (info) {
	case TARGET_MC_DESKTOP_ICON:
	case TARGET_URI_LIST:
	case TARGET_TEXT_PLAIN:
		gtk_selection_data_set (selection_data,
					selection_data->target,
					8,
					filelist,
					len);
		break;

	case TARGET_URL:
		if (dii->url)
			gtk_selection_data_set (selection_data,
						selection_data->target,
						8,
						dii->url,
						strlen (dii->url));
		else {
			files = gnome_uri_list_extract_uris (filelist);
			if (files)
				gtk_selection_data_set (selection_data,
							selection_data->target,
							8,
							files->data,
							strlen (files->data));

			gnome_uri_list_free_strings (files);
		}

		break;

	default:
		g_assert_not_reached ();
	}

	g_free (filelist);
}

/* Callback used when a drag from the desktop is finished.  We need to reload
 * the desktop.
 */
static void
drag_end (GtkWidget *widget, GdkDragContext *context, gpointer data)
{
	desktop_reload_icons (FALSE, 0, 0);
}

/* Set up a desktop icon as a DnD source */
static void
setup_icon_dnd_source (DesktopIconInfo *dii)
{
	setup_icon_dnd_actions (dii);

	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_begin",
			    (GtkSignalFunc) drag_begin,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_data_get",
			    (GtkSignalFunc) drag_data_get,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_end",
			    (GtkSignalFunc) drag_end,
			    dii);
}

/* 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
 * is dragging.
 */
static gboolean
icon_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x, gint y, guint time,
		  gpointer data)
{
	DesktopIconInfo *dii;
	char *filename;
	file_entry *fe;
	GdkDragAction action;
	GtkWidget *source_widget;
	int is_desktop_icon;

	dii = data;
	filename = g_concat_dir_and_file (desktop_directory, dii->filename);
	fe = file_entry_from_file (filename);
	g_free (filename);
	if (!fe)
		return 0; /* eeek */

	gdnd_find_panel_by_drag_context (context, &source_widget);
	is_desktop_icon = gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON);

	action = gdnd_validate_action (context,
				       TRUE,
				       source_widget != NULL,
				       source_widget && is_desktop_icon,
				       desktop_directory,
				       fe,
				       dii->selected);

	gdk_drag_status (context, action, time);
	file_entry_free (fe);
	return TRUE;
}

/*
 * Returns the desktop icon that started the drag from the specified
 * context
 */
static DesktopIconInfo *
find_icon_by_drag_context (GdkDragContext *context)
{
	GtkWidget *source;
	int i;
	GList *l;
	DesktopIconInfo *dii;

	source = gtk_drag_get_source_widget (context);
	if (!source)
		return NULL;

	source = gtk_widget_get_toplevel (source);

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			if (dii->dicon == source)
				return dii;
		}

	return NULL;
}

/*
 * Performs a drop of desktop icons onto the desktop.  It basically
 * moves the icons from their original position to the new
 * coordinates.
 */
static void
drop_desktop_icons (GdkDragContext *context, GtkSelectionData *data, int x, int y)
{
	DesktopIconInfo *source_dii, *dii;
	int dx, dy, sl_x, sl_y, sl_max_x, sl_max_y, sl_min_x, sl_min_y;
	int i;
	GList *l;
	GSList *sel_icons, *sl, *sl2, *sl2_p;

	/* Find the icon that the user is dragging */

	source_dii = find_icon_by_drag_context (context);

	if (!source_dii) {
		g_warning ("Eeeeek, could not find the icon that started the drag!");
		return;
	}

	x -= dnd_press_x;
	y -= dnd_press_y;

	/* Compute the distance to move icons */

	if (desktop_snap_icons)
		get_icon_snap_pos (&x, &y);

	dx = x - source_dii->x;
	dy = y - source_dii->y;


	/* Build a list of selected icons */

	sel_icons = NULL;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;
			if (dii->selected)
				sel_icons = g_slist_prepend (sel_icons, l->data);
		}

	/* Move the icons.  FIXME: handle auto-placement by reinserting the
	 * icons in the proper place.
	 */

	if (!desktop_auto_placement) {
	        sl2 = NULL;
		sl2_p = NULL;
	        for (sl = sel_icons; sl; sl = sl->next) {
	                dii = sl->data;
	                sl2 = g_slist_prepend (sl2, (void *) (dii->x + dx));
	                sl2 = g_slist_prepend (sl2, (void *) (dii->y + dy));
	        }

	        sl_max_x = 0;
	        sl_max_y = 0;
		sl_min_x = INT_MAX;
		sl_min_y = INT_MAX;
		sl2_p = sl2;

		/* For multiple icon drags, get the min/max x and y coordinates */
	        for (sl = sel_icons; sl; sl = sl->next) {
	                sl_y = (int) sl2_p->data;
	                sl2_p = sl2_p->next;
	                sl_x = (int) sl2_p->data;
			sl2_p = sl2_p->next;

	                if (sl_x > sl_max_x) {
	                       sl_max_x = sl_x;
	                }
			if (sl_x < sl_min_x) {
			       sl_min_x = sl_x;
			}
	                if (sl_y > sl_max_y) {
	                       sl_max_y = sl_y;
	                }
			if (sl_x < sl_min_y) {
			       sl_min_y = sl_y;
			}
	        }

		g_slist_free (sl2);

		/* Make sure that no icons are placed off the screen, and keep their relative positions */
	        if (sl_max_x > layout_screen_width - (DESKTOP_SNAP_X + layout_spacer_width / 2)) {
	                dx -= (sl_max_x - (layout_screen_width -
			       (DESKTOP_SNAP_X + layout_spacer_width / 2)));
	        }
	        if (sl_max_y > layout_screen_height - (DESKTOP_SNAP_Y + layout_spacer_height / 2)) {
	                dy -= (sl_max_y - (layout_screen_height -
			       (DESKTOP_SNAP_Y + layout_spacer_width / 2)));
	        }
		if (sl_min_x < layout_spacer_width / 2) {
		        dx += (layout_spacer_width / 2 - sl_min_x);
		}
		if (sl_min_y < layout_spacer_height / 2) {
		        dy += (layout_spacer_height / 2 - sl_min_y);
		}

		/* Place the icons */
		for (sl = sel_icons; sl; sl = sl->next) {
			dii = sl->data;
			desktop_icon_info_place (dii, dii->x + dx, dii->y + dy);
		}
	}
	/* Clean up */

	g_slist_free (sel_icons);
}

/* Handler for drag_data_received for desktop icons */
static void
icon_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
			 GtkSelectionData *data, guint info, guint time, gpointer user_data)
{
	DesktopIconInfo *dii;

	dii = user_data;

	if (gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON) && dii->selected)
		drop_desktop_icons (context, data, x + dii->x, y + dii->y);
	else {
		char *full_name;
		file_entry *fe;

		full_name = g_concat_dir_and_file (desktop_directory, dii->filename);
		fe = file_entry_from_file (full_name);
		if (!fe) {
			g_free (full_name);
			return;
		}

		if (gdnd_perform_drop (context, data, full_name, fe))
			desktop_reload_icons (FALSE, 0, 0);

		file_entry_free (fe);
		g_free (full_name);
	}
}

/* Set up a desktop icon as a DnD destination */
static void
setup_icon_dnd_dest (DesktopIconInfo *dii)
{
	gtk_drag_dest_set (DESKTOP_ICON (dii->dicon)->canvas,
			   GTK_DEST_DEFAULT_DROP,
			   dnd_icon_targets,
			   dnd_icon_ntargets,
			   GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK | GDK_ACTION_ASK);

	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_motion",
			    (GtkSignalFunc) (icon_drag_motion),
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_data_received",
			    (GtkSignalFunc) icon_drag_data_received,
			    dii);
}

/* Creates a new desktop icon.  The filename is the pruned filename inside the
 * desktop directory.  It does not show the icon.
 */
static DesktopIconInfo *
desktop_icon_info_new (char *filename, char *url, char *caption, int xpos, int ypos)
{
	DesktopIconInfo *dii;
	file_entry *fe;
	char *full_name;
	GdkImlibImage *icon_im;

	/* Create the icon structure */

	full_name = g_concat_dir_and_file (desktop_directory, filename);
	fe = file_entry_from_file (full_name);
	if (!fe)
		return NULL;

	dii = g_new (DesktopIconInfo, 1);
	dii->x = 0;
	dii->y = 0;
	dii->slot = -1;

	if (url) {
		dii->url = g_strdup (url);
		if (!caption)
			caption = url;
	} else {
		dii->url = NULL;
		if (caption == NULL)
			caption = filename;
	}

	icon_im = gicon_get_icon_for_file (desktop_directory, fe, FALSE);
	dii->dicon = desktop_icon_new (icon_im, caption);
	dii->filename = g_strdup (filename);
	dii->selected = FALSE;

	file_entry_free (fe);
	g_free (full_name);

	/* Connect to the icon's signals.  We connect to the image/stipple and
	 * text items separately so that the callback can distinguish between
	 * them.  This is not a hack.
	 */

	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->icon), "event",
			    (GtkSignalFunc) icon_event,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->stipple), "event",
			    (GtkSignalFunc) icon_event,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "event",
			    (GtkSignalFunc) icon_event,
			    dii);

	/* Connect_after to button presses on the icon's window.  This is a
	 * fallback for when the icon is being edited and a button press is not
	 * handled by the icon's text item -- this means the user has clicked
	 * outside the text item and wishes to accept and terminate editing.
	 */
	gtk_signal_connect_after (GTK_OBJECT (dii->dicon), "button_press_event",
				  (GtkSignalFunc) icon_event_after,
				  dii);

	/* Connect to the text item's signals */

	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "text_changed",
			    (GtkSignalFunc) text_changed, dii);

	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "editing_started",
			    (GtkSignalFunc) editing_started,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "editing_stopped",
			    (GtkSignalFunc) editing_stopped,
			    dii);
	gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "selection_stopped",
			    (GtkSignalFunc) selection_stopped,
			    dii);

	/* We must set the icon's wmclass and name.  It is already realized (it
	 * comes out realized from desktop_icon_new()).
	 */
	set_icon_wmclass (dii);

	/* Prepare the DnD functionality for this icon */

	setup_icon_dnd_source (dii);
	setup_icon_dnd_dest (dii);

	/* Place the icon and append it to the list */
	desktop_icon_info_place (dii, xpos, ypos);

	/* Put into icon hash table */
	g_hash_table_insert (icon_hash, dii->filename, dii);

	return dii;
}

/* Creates the layout information array */
static void
create_layout_info (void)
{
	layout_screen_width = gdk_screen_width ();
	layout_screen_height = gdk_screen_height ();
	layout_cols = layout_screen_width / DESKTOP_SNAP_X;
	layout_rows = layout_screen_height / DESKTOP_SNAP_Y;

	/* Adjust icon size to spread the icons evenly across the screen */
	layout_spacer_width = (layout_screen_width / layout_cols) - DESKTOP_SNAP_X;
	layout_spacer_height = (layout_screen_height / layout_rows) - DESKTOP_SNAP_Y;
	

	layout_slots = g_new0 (struct layout_slot, layout_cols * layout_rows);
}

/*
 * Rename old Trash directory if found
 */
static void
migrate_init (void)
{
	struct stat buf;
	
	char *old_trash_dir = g_concat_dir_and_file (desktop_directory, "Trash");
	if (stat (old_trash_dir, &buf) == 0 && S_ISDIR (buf.st_mode)){
		char *new_trash_dir = g_concat_dir_and_file (desktop_directory, "Trash.gmc");
		
		rename (old_trash_dir, new_trash_dir);
		gnome_metadata_rename (old_trash_dir, new_trash_dir);
		g_free (new_trash_dir);
	}
	g_free (old_trash_dir);
}

/*
 * Check that the user's desktop directory exists, and if not, create the
 * default desktop setup.
 */
static void
create_desktop_dir (void)
{
	if (getenv ("GNOME_DESKTOP_DIR") != NULL)
		desktop_directory = g_strdup (getenv ("GNOME_DESKTOP_DIR"));
	else
		desktop_directory = g_concat_dir_and_file (gnome_user_home_dir, DESKTOP_DIR_NAME);

	if (!g_file_exists (desktop_directory)) {
		/* Create the directory */
		mkdir (desktop_directory, 0777);

		/* Create the default set of icons */

		gdesktop_links_init ();
		gmount_setup_devices ();
		gprint_setup_devices ();
	}
	migrate_init ();
}

/* Property placed on target windows */
typedef struct {
  guint8 byte_order;
  guint8 protocol_version;
  guint8 protocol_style;
  guint8 pad;
  guint32 proxy_window;
  guint16 num_drop_sites;
  guint16 padding;
  guint32 total_size;
} MotifDragReceiverInfo;

/* Sets up a proxy window for DnD on the specified X window.  Courtesy of Owen Taylor */
static gboolean
setup_xdnd_proxy (guint32 xid, GdkWindow *proxy_window)
{
	GdkAtom xdnd_proxy_atom;
	Window proxy_xid;
	Atom type;
	int format;
	unsigned long nitems, after;
	Window *proxy_data;
	Window proxy;
	guint32 old_warnings;

	XGrabServer (GDK_DISPLAY ());

	xdnd_proxy_atom = gdk_atom_intern ("XdndProxy", FALSE);
	proxy_xid = GDK_WINDOW_XWINDOW (proxy_window);
	type = None;
	proxy = None;

	old_warnings = gdk_error_warnings;

	gdk_error_code = 0;
	gdk_error_warnings = 0;

	/* Check if somebody else already owns drops on the root window */

	XGetWindowProperty (GDK_DISPLAY (), xid,
			    xdnd_proxy_atom, 0,
			    1, False, AnyPropertyType,
			    &type, &format, &nitems, &after,
			    (guchar **) &proxy_data);

	if (type != None) {
		if (format == 32 && nitems == 1)
			proxy = *proxy_data;

		XFree (proxy_data);
	}

	/* The property was set, now check if the window it points to exists and
	 * has a XdndProxy property pointing to itself.
	 */
	if (proxy) {
		XGetWindowProperty (GDK_DISPLAY (), proxy,
				    xdnd_proxy_atom, 0,
				    1, False, AnyPropertyType,
				    &type, &format, &nitems, &after,
				    (guchar **) &proxy_data);

		if (!gdk_error_code && type != None) {
			if (format == 32 && nitems == 1)
				if (*proxy_data != proxy)
					proxy = GDK_NONE;

			XFree (proxy_data);
		} else
			proxy = GDK_NONE;
	}

	if (!proxy) {
		/* OK, we can set the property to point to us */

		XChangeProperty (GDK_DISPLAY (), xid,
				 xdnd_proxy_atom, gdk_atom_intern ("WINDOW", FALSE),
				 32, PropModeReplace,
				 (guchar *) &proxy_xid, 1);
	}

	gdk_error_code = 0;
	gdk_error_warnings = old_warnings;

	XUngrabServer (GDK_DISPLAY ());
	gdk_flush ();

	if (!proxy) {
		/* Mark our window as a valid proxy window with a XdndProxy
		 * property pointing recursively;
		 */
		XChangeProperty (GDK_DISPLAY (), proxy_xid,
				 xdnd_proxy_atom, gdk_atom_intern ("WINDOW", FALSE),
				 32, PropModeReplace,
				 (guchar *) &proxy_xid, 1);

		return TRUE;
	} else
		return FALSE;
}

/* Sets up a window as a Motif DnD proxy */
static void
setup_motif_dnd_proxy (guint32 xid, GdkWindow *proxy_window)
{
	Window proxy_xid;
	MotifDragReceiverInfo info;
	Atom receiver_info_atom;
	guint32 myint;

	myint = 0x01020304;
	proxy_xid = GDK_WINDOW_XWINDOW (proxy_window);
	receiver_info_atom = gdk_atom_intern ("_MOTIF_DRAG_RECEIVER_INFO", FALSE);

	info.byte_order = (*((gchar *) &myint) == 1) ? 'B' : 'l';
	info.protocol_version = 0;
	info.protocol_style = 5; /* XmDRAG_DYNAMIC */
	info.proxy_window = proxy_xid;
	info.num_drop_sites = 0;
	info.total_size = sizeof (info);

	XChangeProperty (gdk_display, xid,
			 receiver_info_atom,
			 receiver_info_atom,
			 8, PropModeReplace,
			 (guchar *)&info,
			 sizeof (info));
}

/* Callback used when we get a drag_motion event from the desktop.  We must
 * decide what kind of operation can be performed with what the user is
 * dragging.
 */
static gboolean
desktop_drag_motion (GtkWidget *widget, GdkDragContext *context, gint x, gint y, guint time,
		     gpointer data)
{
	GdkDragAction action;
	GtkWidget *source_widget;
	int is_desktop_icon;

	gdnd_find_panel_by_drag_context (context, &source_widget);
	is_desktop_icon = gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON);

	action = gdnd_validate_action (context,
				       TRUE,
				       source_widget != NULL,
				       source_widget && is_desktop_icon,
				       desktop_directory,
				       NULL,
				       FALSE);

	gdk_drag_status (context, action, time);
	return TRUE;
}

/* Callback used when the root window receives a drop */
static void
desktop_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
			    GtkSelectionData *data, guint info, guint time, gpointer user_data)
{
	gint dx, dy;

	/* Fix the proxy window offsets */

	gdk_window_get_position (widget->window, &dx, &dy);
	x += dx;
	y += dy;

	if (gdnd_drag_context_has_target (context, TARGET_MC_DESKTOP_ICON))
		drop_desktop_icons (context, data, x, y);
	else {
		file_entry *desktop_fe;

		desktop_fe = file_entry_from_file (desktop_directory);
		if (!desktop_fe)
			return; /* eeek */

		if (gdnd_perform_drop (context, data, desktop_directory, desktop_fe))
			desktop_reload_icons (TRUE, x, y);

		file_entry_free (desktop_fe);
	}
}

/* Sets up drag and drop to the desktop root window */
static void
setup_desktop_dnd (void)
{
	if (!setup_xdnd_proxy (GDK_ROOT_WINDOW (), proxy_invisible->window))
		g_warning ("There is already a process taking drops on the desktop!\n");

	setup_motif_dnd_proxy (GDK_ROOT_WINDOW (), proxy_invisible->window);

	gtk_drag_dest_set (proxy_invisible,
			   GTK_DEST_DEFAULT_DROP,
			   dnd_desktop_targets,
			   dnd_desktop_ntargets,
			   GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK | GDK_ACTION_ASK);

	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "drag_motion",
			    (GtkSignalFunc) desktop_drag_motion,
			    NULL);
	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "drag_data_received",
			    (GtkSignalFunc) desktop_drag_data_received,
			    NULL);
}

/* Looks for the proxy window to get root window clicks from the window manager */
static GdkWindow *
find_click_proxy_window (void)
{
	GdkAtom click_proxy_atom;
	Atom type;
	int format;
	unsigned long nitems, after;
	Window *proxy_data;
	Window proxy;
	guint32 old_warnings;
	GdkWindow *proxy_gdk_window;

	XGrabServer (GDK_DISPLAY ());

	click_proxy_atom = gdk_atom_intern ("_WIN_DESKTOP_BUTTON_PROXY", FALSE);
	type = None;
	proxy = None;

	old_warnings = gdk_error_warnings;

	gdk_error_code = 0;
	gdk_error_warnings = 0;

	/* Check if the proxy window exists */

	XGetWindowProperty (GDK_DISPLAY (), GDK_ROOT_WINDOW (),
			    click_proxy_atom, 0,
			    1, False, AnyPropertyType,
			    &type, &format, &nitems, &after,
			    (guchar **) &proxy_data);

	if (type != None) {
		if (format == 32 && nitems == 1)
			proxy = *proxy_data;

		XFree (proxy_data);
	}

	/* The property was set, now check if the window it points to exists and
	 * has a _WIN_DESKTOP_BUTTON_PROXY property pointing to itself.
	 */

	if (proxy) {
		XGetWindowProperty (GDK_DISPLAY (), proxy,
				    click_proxy_atom, 0,
				    1, False, AnyPropertyType,
				    &type, &format, &nitems, &after,
				    (guchar **) &proxy_data);

		if (!gdk_error_code && type != None) {
			if (format == 32 && nitems == 1)
				if (*proxy_data != proxy)
					proxy = GDK_NONE;

			XFree (proxy_data);
		} else
			proxy = GDK_NONE;
	}

	gdk_error_code = 0;
	gdk_error_warnings = old_warnings;

	XUngrabServer (GDK_DISPLAY ());
	gdk_flush ();

	if (proxy)
		proxy_gdk_window = gdk_window_foreign_new (proxy);
	else
		proxy_gdk_window = NULL;

	return proxy_gdk_window;
}

/* Callback for arranging icons by name */
static void
handle_arrange_icons_name (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_NAME);
}

/* Callback for arranging icons by file type */
static void
handle_arrange_icons_type (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_EXTENSION);
}

/* Callback for arranging icons by file size */
static void
handle_arrange_icons_size (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_SIZE);
}

/* Callback for arranging icons by access time */
static void
handle_arrange_icons_access (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_ACCESS);
}

/* Callback for arranging icons by modification time */
static void
handle_arrange_icons_mod (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_MODIFY);
}

/* Callback for arranging icons by change time */
static void
handle_arrange_icons_change (GtkWidget *widget, gpointer data)
{
	desktop_arrange_icons (SORT_CHANGE);
}

/* Callback for running all desktop icons through snap-to-grid */
static void
handle_tidy_icons (GtkWidget *widget, gpointer data)
{
        desktop_tidy_icons();
}


/* Callback for creating a new panel window */
static void
handle_new_window (GtkWidget *widget, gpointer data)
{
	new_panel_at (gnome_user_home_dir);
}

/* Callback for rescanning the desktop directory */
static void
handle_rescan_desktop (GtkWidget *widget, gpointer data)
{
	desktop_reload_icons (FALSE, 0, 0);
}

/* Rescans the mountable devices in the desktop and re-creates their icons */
void
desktop_rescan_devices (void)
{
	desktop_cleanup_devices ();
	gmount_setup_devices ();
	desktop_reload_icons (FALSE, 0, 0);
}

/* Callback for rescanning the mountable devices */
static void
handle_rescan_devices (GtkWidget *widget, gpointer data)
{
	desktop_rescan_devices ();
}

void
desktop_recreate_default_icons (void)
{
	gdesktop_links_init ();
	gprint_setup_devices ();
}

/* Callback for re-creating the default icons */
static void
handle_recreate_default_icons (GtkWidget *widget, gpointer data)
{
	desktop_recreate_default_icons ();
}

static void
set_background_image (GtkWidget *widget, gpointer data)
{
	gchar *bg_capplet;
	gchar *argv[1];
	GtkWidget *msg_box;

	bg_capplet = gnome_is_program_in_path ("background-properties-capplet");

	if (bg_capplet) {
		argv[0] = bg_capplet;
		gnome_execute_async (bg_capplet, 1, argv);
		g_free (bg_capplet);
	} else {
		msg_box = gnome_message_box_new (
			_("Unable to locate the file:\nbackground-properties-capplet\n"
			  "in your path.\n\nWe are unable to set the background."),
			GNOME_MESSAGE_BOX_WARNING,
			GNOME_STOCK_BUTTON_OK,
			NULL);
		gnome_dialog_run (GNOME_DIALOG (msg_box));
	}
}

static void
set_desktop_icons (GtkWidget *widget, gpointer data)
{
	gnome_configure_box_with_desktop (NULL, NULL, TRUE);
}

/* Callback from menus to create a terminal.  If the user creates a terminal
 * from the desktop, he usually wants the cwd to be his home directory, not the
 * desktop directory.
 */
static void
new_terminal (GtkWidget *widget, gpointer data)
{
	if (is_a_desktop_panel (cpanel))
		mc_chdir (home_dir);

	gnome_open_terminal ();
}

static GnomeUIInfo gnome_panel_new_menu [] = {
	GNOMEUIINFO_ITEM_NONE (N_("_Terminal"), N_("Launch a new terminal in the current directory"), new_terminal),
	/* If this ever changes, make sure you update create_new_menu accordingly. */
	GNOMEUIINFO_ITEM_NONE (N_("_Directory..."), N_("Creates a new directory"), gnome_mkdir_cmd),
	GNOMEUIINFO_ITEM_NONE (N_("URL L_ink..."),  N_("Creates a new URL link"), gnome_new_link),
	GNOMEUIINFO_ITEM_NONE (N_("_Launcher..."), N_("Creates a new launcher"), gnome_new_launcher),
	GNOMEUIINFO_END
};

/* Menu items for arranging the desktop icons */
GnomeUIInfo desktop_arrange_icons_items[] = {
	GNOMEUIINFO_ITEM_NONE (N_("By _Name"), NULL, handle_arrange_icons_name),
	GNOMEUIINFO_ITEM_NONE (N_("By File _Type"), NULL, handle_arrange_icons_type),
	GNOMEUIINFO_ITEM_NONE (N_("By _Size"), NULL, handle_arrange_icons_size),
	GNOMEUIINFO_ITEM_NONE (N_("By Time Last _Accessed"), NULL, handle_arrange_icons_access),
	GNOMEUIINFO_ITEM_NONE (N_("By Time Last _Modified"), NULL, handle_arrange_icons_mod),
	GNOMEUIINFO_ITEM_NONE (N_("By Time Last _Changed"), NULL, handle_arrange_icons_change),
	GNOMEUIINFO_END
};

/* The popup menu for the desktop */
GnomeUIInfo desktop_popup_items[] = {
	GNOMEUIINFO_MENU_NEW_SUBTREE (gnome_panel_new_menu),
	GNOMEUIINFO_SEPARATOR,
	GNOMEUIINFO_SUBTREE (N_("_Arrange Icons"), desktop_arrange_icons_items),
	GNOMEUIINFO_ITEM_NONE (N_("_Tidy Icons"), NULL, handle_tidy_icons),
	GNOMEUIINFO_ITEM_NONE (N_("Create _New Window"), NULL, handle_new_window),
	GNOMEUIINFO_SEPARATOR,
	GNOMEUIINFO_ITEM_NONE (N_("Rescan _Desktop Directory"), NULL, handle_rescan_desktop),
	GNOMEUIINFO_ITEM_NONE (N_("Rescan De_vices"), NULL, handle_rescan_devices),
	GNOMEUIINFO_ITEM_NONE (N_("Recreate Default _Icons"), NULL, handle_recreate_default_icons),
	GNOMEUIINFO_SEPARATOR,
	GNOMEUIINFO_ITEM_NONE (N_("Configure _Background Image"), NULL, set_background_image),
	GNOMEUIINFO_ITEM_NONE (N_("Des_ktop Properties"), NULL, set_desktop_icons),
	GNOMEUIINFO_END
};

static int
strip_tearoff_menu_item (GnomeUIInfo *infos)
{
	GtkWidget *shell;
	GList *child_list;
	int n;

	g_assert (infos != NULL);

	shell = infos[0].widget->parent;
	child_list = gtk_container_children (GTK_CONTAINER (shell));
	n = g_list_length (child_list);

	if (child_list && GTK_IS_TEAROFF_MENU_ITEM (child_list->data)) {
		n--;
		gtk_widget_destroy (GTK_WIDGET (child_list->data));
#if 0
		gtk_widget_hide (GTK_WIDGET (child_list->data));
#endif
	}

	return n;
}

/* Executes the popup menu for the desktop */
static void
desktop_popup (GdkEventButton *event)
{
	GtkWidget *shell;
	GtkWidget *popup;
	gchar *file, *file2;
	WPanel *panel;
	gint i;

	/* Create the menu and then strip the tearoff menu items, sigh... */

	if (nodesktop)
		return;

	popup = gnome_popup_menu_new (desktop_popup_items);

	strip_tearoff_menu_item (desktop_arrange_icons_items);
	i = strip_tearoff_menu_item (gnome_panel_new_menu);
	shell = gnome_panel_new_menu[0].widget->parent;
	file = gnome_unconditional_datadir_file ("mc/templates");
	i = create_new_menu_from (file, shell, i);
	file2 = gnome_datadir_file ("mc/templates");
	if (file2 != NULL){
		if (strcmp (file, file2) != 0)
			create_new_menu_from (file2, shell, i);
	}
	g_free (file);
	g_free (file2);

	panel = push_desktop_panel_hack ();

	/* Disable the "arrange icons" commands if there are no icons */

	if (panel->count == 0)
		for (i = 0; desktop_arrange_icons_items[i].type != GNOME_APP_UI_ENDOFINFO; i++)
			gtk_widget_set_sensitive (desktop_arrange_icons_items[i].widget, FALSE);

	gnome_popup_menu_do_popup_modal (popup, NULL, NULL, event, NULL);
	layout_panel_gone (panel);
	free_panel_from_desktop (panel);
	gtk_widget_destroy (popup);

	desktop_reload_icons (FALSE, 0, 0);
}

/* Draws the rubberband rectangle for selecting icons on the desktop */
static void
draw_rubberband (int x, int y)
{
       	int x1, y1, x2, y2;

	if (click_start_x < x) {
		x1 = click_start_x;
		x2 = x;
	} else {
		x1 = x;
		x2 = click_start_x;
	}

	if (click_start_y < y) {
		y1 = click_start_y;
		y2 = y;
	} else {
		y1 = y;
		y2 = click_start_y;
	}

	gdk_draw_rectangle (GDK_ROOT_PARENT (), click_gc, FALSE, x1, y1, x2 - x1, y2 - y1);
}

/*
 * Stores dii->selected into dii->tmp_selected to keep the original selection
 * around while the user is rubberbanding.
 */
static void
store_temp_selection (void)
{
	int i;
	GList *l;
	DesktopIconInfo *dii;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			dii->tmp_selected = dii->selected;
		}
}

/**
 * icon_is_in_area:
 * @dii: the desktop icon information
 *
 * Returns TRUE if the specified icon is at least partially inside the specified
 * area, or FALSE otherwise.
 */
static int
icon_is_in_area (DesktopIconInfo *dii, int x1, int y1, int x2, int y2)
{
	DesktopIcon *dicon;

	dicon = DESKTOP_ICON (dii->dicon);

	x1 -= dii->x;
	y1 -= dii->y;
	x2 -= dii->x;
	y2 -= dii->y;

	if (x1 == x2 && y1 == y2)
		return FALSE;

	if (x1 < dicon->icon_x + dicon->icon_w
	    && x2 >= dicon->icon_x
	    && y1 < dicon->icon_y + dicon->icon_h
	    && y2 >= dicon->icon_y)
		return TRUE;

	if (x1 < dicon->text_x + dicon->text_w
	    && x2 >= dicon->text_x
	    && y1 < dicon->text_y + dicon->text_h
	    && y2 >= dicon->text_y)
		return TRUE;

	return FALSE;
}

/* Update the selection being rubberbanded.  It selects or unselects the icons
 * as appropriate.
 */
static void
update_drag_selection (int x, int y)
{
	int x1, y1, x2, y2;
	int i;
	GList *l;
	DesktopIconInfo *dii;
	int additive, invert, in_area;

	if (click_start_x < x) {
		x1 = click_start_x;
		x2 = x;
	} else {
		x1 = x;
		x2 = click_start_x;
	}

	if (click_start_y < y) {
		y1 = click_start_y;
		y2 = y;
	} else {
		y1 = y;
		y2 = click_start_y;
	}

	/* Select or unselect icons as appropriate */

	additive = click_start_state & GDK_SHIFT_MASK;
	invert = click_start_state & GDK_CONTROL_MASK;

	for (i = 0; i < (layout_cols * layout_rows); i++)
		for (l = layout_slots[i].icons; l; l = l->next) {
			dii = l->data;

			in_area = icon_is_in_area (dii, x1, y1, x2, y2);

			if (in_area) {
				if (invert) {
					if (dii->selected == dii->tmp_selected) {
						desktop_icon_select (DESKTOP_ICON (dii->dicon),
								     !dii->selected);
						dii->selected = !dii->selected;
					}
				} else if (additive) {
					if (!dii->selected) {
						desktop_icon_select (DESKTOP_ICON (dii->dicon), TRUE);
						dii->selected = TRUE;
					}
				} else {
					if (!dii->selected) {
						desktop_icon_select (DESKTOP_ICON (dii->dicon), TRUE);
						dii->selected = TRUE;
					}
				}
			} else if (dii->selected != dii->tmp_selected) {
				desktop_icon_select (DESKTOP_ICON (dii->dicon), dii->tmp_selected);
				dii->selected = dii->tmp_selected;
			}
		}
}

/* Terminates rubberbanding when the button is released.  This is shared by the
 * button_release handler and the motion_notify handler.
 */
static void
perform_release (guint32 time)
{
	draw_rubberband (click_current_x, click_current_y);
	gdk_pointer_ungrab (time);
	click_dragging = FALSE;

	update_drag_selection (click_current_x, click_current_y);

	XUngrabServer (GDK_DISPLAY ());
	gdk_flush ();
}

/* Handles button presses on the root window via the click_proxy_gdk_window */
static gint
click_proxy_button_press (GtkWidget *widget, GdkEventButton *event, gpointer data)
{
	GdkCursor *cursor;

	/* maybe the user wants to click on the icon text */
	if (event->button == 1 && desktop_use_shaped_text) {
		int x = event->x;
		int y = event->y;
		GList *l, *icons = get_all_icons ();
		DesktopIconInfo *clicked = NULL;
		for (l = icons; l; l = l->next) {
			DesktopIconInfo *dii = l->data;
			DesktopIcon *di = DESKTOP_ICON (dii->dicon);
			int x1 = dii->x + di->text_x;
			int y1 = dii->y + di->text_y;
			int x2 = x1 + di->text_w;
			int y2 = y1 + di->text_h;
			if (x>=x1 && y>=y1 && x<=x2 && y<=y2)
				clicked = dii;
		}
		g_list_free (icons);
		if (clicked) {
			select_icon (clicked, event->state);
			return FALSE;
		}
	}

	if (event->button == 1) {
		click_start_x = event->x;
		click_start_y = event->y;
		click_start_state = event->state;

		XGrabServer (GDK_DISPLAY ());

		cursor = gdk_cursor_new (GDK_LEFT_PTR);
		gdk_pointer_grab (GDK_ROOT_PARENT (),
				  FALSE,
				  GDK_BUTTON_RELEASE_MASK | GDK_POINTER_MOTION_MASK,
				  NULL,
				  cursor,
				  event->time);
		gdk_cursor_destroy (cursor);

		/* If no modifiers are pressed, we unselect all the icons */

		if ((click_start_state & (GDK_SHIFT_MASK | GDK_CONTROL_MASK)) == 0)
			unselect_all (NULL);

		store_temp_selection (); /* Save the original selection */

		draw_rubberband (event->x, event->y);
		click_current_x = event->x;
		click_current_y = event->y;
		click_dragging = TRUE;

		return TRUE;
	} else if (event->button == 3) {
		if (click_dragging)
			perform_release (event->time);

		desktop_popup (event);
		return TRUE;
	}

	return FALSE;
}

/* Handles button releases on the root window via the click_proxy_gdk_window */
static gint
click_proxy_button_release (GtkWidget *widget, GdkEventButton *event, gpointer data)
{
	if (!click_dragging || event->button != 1)
		return FALSE;

	perform_release (event->time);
	return TRUE;
}

/* Handles motion events when dragging the icon-selection rubberband on the desktop */
static gint
click_proxy_motion (GtkWidget *widget, GdkEventMotion *event, gpointer data)
{
	if (!click_dragging)
		return FALSE;

	/* There exists the following race condition.  If in the button_press
	 * handler we manage to grab the server before the window manager can
	 * proxy the button release to us, then we wil not get the release
	 * event.  So we have to check the event mask and fake a release by
	 * hand.
	 */
	if (!(event->state & GDK_BUTTON1_MASK)) {
		perform_release (event->time);
		return TRUE;
	}

	draw_rubberband (click_current_x, click_current_y);
	draw_rubberband (event->x, event->y);
	update_drag_selection (event->x, event->y);
	click_current_x = event->x;
	click_current_y = event->y;

	return TRUE;
}

/*
 * Filter that translates proxied events from virtual root windows into normal
 * Gdk events for the proxy_invisible widget.
 */
static GdkFilterReturn
click_proxy_filter (GdkXEvent *xevent, GdkEvent *event, gpointer data)
{
	XEvent *xev;

	xev = xevent;

	switch (xev->type) {
	case ButtonPress:
	case ButtonRelease:
		/* Translate button events into events that come from the proxy
		 * window, so that we can catch them as a signal from the
		 * invisible widget.
		 */
		if (xev->type == ButtonPress)
			event->button.type = GDK_BUTTON_PRESS;
		else
			event->button.type = GDK_BUTTON_RELEASE;

		gdk_window_ref (click_proxy_gdk_window);

		event->button.window = click_proxy_gdk_window;
		event->button.send_event = xev->xbutton.send_event;
		event->button.time = xev->xbutton.time;
		event->button.x = xev->xbutton.x;
		event->button.y = xev->xbutton.y;
		event->button.state = xev->xbutton.state;
		event->button.button = xev->xbutton.button;

		return GDK_FILTER_TRANSLATE;

	case DestroyNotify:
		/* The proxy window was destroyed (i.e. the window manager
		 * died), so we have to cope with it
		 */
		if (((GdkEventAny *) event)->window == click_proxy_gdk_window) {
			gdk_window_destroy_notify (click_proxy_gdk_window);
			click_proxy_gdk_window = NULL;
		}

		return GDK_FILTER_REMOVE;

	default:
		break;
	}

	return GDK_FILTER_CONTINUE;
}

/*
 * Creates a proxy window to receive clicks from the root window and
 * sets up the necessary event filters.
 */
static void
setup_desktop_click_proxy_window (void)
{
	click_proxy_gdk_window = find_click_proxy_window ();
	if (!click_proxy_gdk_window) {
		desktop_wm_is_gnome_compliant = 0;
		g_warning ("Root window clicks will not work as no GNOME-compliant window manager could be found!");
		return;
	}
	desktop_wm_is_gnome_compliant = 1;

	/* Make the proxy window send events to the invisible proxy widget */
	gdk_window_set_user_data (click_proxy_gdk_window, proxy_invisible);

	/* Add our filter to get events */
	gdk_window_add_filter (click_proxy_gdk_window, click_proxy_filter, NULL);

	/*
	 * The proxy window for clicks sends us button press events with
	 * SubstructureNotifyMask.  We need StructureNotifyMask to receive
	 * DestroyNotify events, too.
	 */

	XSelectInput (GDK_DISPLAY (), GDK_WINDOW_XWINDOW (click_proxy_gdk_window),
		      SubstructureNotifyMask | StructureNotifyMask);
}

/*
 * Handler for PropertyNotify events from the root window; it must change the
 * proxy window to a new one.
 */
static gint
click_proxy_property_notify (GtkWidget *widget, GdkEventProperty *event, gpointer data)
{
	if (event->window != GDK_ROOT_PARENT ())
		return FALSE;

	if (event->atom != gdk_atom_intern ("_WIN_DESKTOP_BUTTON_PROXY", FALSE))
		return FALSE;

	/* If there already is a proxy window, destroy it */

	click_proxy_gdk_window = NULL;

	/* Get the new proxy window */

	setup_desktop_click_proxy_window ();

	return TRUE;
}

#define gray50_width 2
#define gray50_height 2
static char gray50_bits[] = {
  0x02, 0x01, };

/* Sets up the window manager proxy window to receive clicks on the desktop root window */
static void
setup_desktop_clicks (void)
{
	GdkColormap *cmap;
	GdkColor color;
	GdkBitmap *stipple;

	/* Make the root window send events to the invisible proxy widget */
	gdk_window_set_user_data (GDK_ROOT_PARENT (), proxy_invisible);

	/* Add our filter to get button press/release events (they are sent by
	 * the WM * with the window set to the root).  Our filter will translate
	 * them to a GdkEvent with the proxy window as its window field.
	 */
	gdk_window_add_filter (GDK_ROOT_PARENT (), click_proxy_filter, NULL);

	/* Select for PropertyNotify events from the root window */

	XSelectInput (GDK_DISPLAY (), GDK_ROOT_WINDOW (), PropertyChangeMask);

	/* Create the proxy window for clicks on the root window */
	setup_desktop_click_proxy_window ();

	/* Connect the signals */

	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "button_press_event",
			    (GtkSignalFunc) click_proxy_button_press,
			    NULL);
	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "button_release_event",
			    (GtkSignalFunc) click_proxy_button_release,
			    NULL);
	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "motion_notify_event",
			    (GtkSignalFunc) click_proxy_motion,
			    NULL);

	gtk_signal_connect (GTK_OBJECT (proxy_invisible), "property_notify_event",
			    (GtkSignalFunc) click_proxy_property_notify,
			    NULL);

	/* Create the GC to paint the rubberband rectangle */

	click_gc = gdk_gc_new (GDK_ROOT_PARENT ());

	cmap = gdk_window_get_colormap (GDK_ROOT_PARENT ());

	gdk_color_white (cmap, &color);
	if (color.pixel == 0)
		gdk_color_black (cmap, &color);

	gdk_gc_set_foreground (click_gc, &color);
	gdk_gc_set_function (click_gc, GDK_XOR);

	gdk_gc_set_fill (click_gc, GDK_STIPPLED);

	stipple = gdk_bitmap_create_from_data (NULL, gray50_bits, gray50_width, gray50_height);
	gdk_gc_set_stipple (click_gc, stipple);
	gdk_bitmap_unref (stipple);
}

static gboolean
do_rescan (gpointer data)
{
	struct stat buf;
	
	if (stat (desktop_directory, &buf) == -1)
		return TRUE;

	if (buf.st_ctime == desktop_dir_modify_time)
		return TRUE;

	desktop_reload_icons (FALSE, 0, 0);
	
	return TRUE;
}

#define RESCAN_TIMEOUT 4000

static void
setup_desktop_reload_monitor (void)
{
	gtk_timeout_add (RESCAN_TIMEOUT, do_rescan, NULL);
}

/**
 * desktop_init
 *
 * Initializes the desktop by setting up the default icons (if necessary), setting up drag and drop,
 * and other miscellaneous tasks.
 */
void
desktop_init (void)
{
	icon_hash = g_hash_table_new (g_str_hash, g_str_equal);

	if (!nodesktop) {
		gdnd_init ();
		gicon_init ();
		create_layout_info ();
		create_desktop_dir ();
		desktop_reload_icons (FALSE, 0, 0);
	}

	/* Create the proxy window and initialize all proxying stuff */

	proxy_invisible = gtk_invisible_new ();
	gtk_widget_show (proxy_invisible);

	setup_desktop_dnd ();
	setup_desktop_clicks ();

	/* Setup reloading */
	setup_desktop_reload_monitor ();
}

/**
 * desktop_destroy
 *
 * Shuts the desktop down by destroying the desktop icons.
 */
void
desktop_destroy (void)
{
	/* Destroy the desktop icons */

	destroy_desktop_icons ();

	/* Cleanup */

	g_free (layout_slots);
	layout_slots = NULL;
	layout_cols = 0;
	layout_rows = 0;

	g_free (desktop_directory);
	desktop_directory = NULL;

	/* Remove click-on-desktop crap */

	gdk_window_unref (click_proxy_gdk_window);

	/* Remove DnD crap */

	gtk_widget_destroy (proxy_invisible);
	XDeleteProperty (GDK_DISPLAY (), GDK_ROOT_WINDOW (), gdk_atom_intern ("XdndProxy", FALSE));

	g_hash_table_destroy (icon_hash);
}

void
desktop_create_url (const char *filename, const char *title, const char *url, const char *icon)
{
	FILE *f;

	f = fopen (filename, "w");
	if (f) {

		fprintf (f, "URL: %s\n", url);
		fclose (f);

		gnome_metadata_set (filename, "desktop-url",
				    strlen (url) + 1, url);
		gnome_metadata_set (filename, "icon-caption",
				    strlen (title) + 1, title);

		gnome_metadata_set (filename, "icon-filename", strlen (icon) + 1, icon);
	}
}