1998-10-30 00:47:54 +00:00
|
|
|
/* Desktop management for the Midnight Commander
|
|
|
|
*
|
|
|
|
* Copyright (C) 1998 The Free Software Foundation
|
1998-02-27 05:10:44 +00:00
|
|
|
*
|
1998-10-30 00:47:54 +00:00
|
|
|
* Authors: Federico Mena <federico@nuclecu.unam.mx>
|
|
|
|
* Miguel de Icaza <miguel@nuclecu.unam.mx>
|
1998-02-27 05:10:44 +00:00
|
|
|
*/
|
1998-11-07 01:19:53 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* TO-DO list for the desktop;
|
|
|
|
*
|
|
|
|
* - Put an InputOnly window over icons to be able to select them even if the user clicks on
|
|
|
|
* the transparent area.
|
|
|
|
*
|
|
|
|
* - DnD from file windows to icons.
|
|
|
|
*
|
|
|
|
* - DnD from icons to desktop (move icon).
|
|
|
|
*
|
|
|
|
* - DnD from icons to windows.
|
|
|
|
*
|
|
|
|
* - DnD from icons to icons (file->directory or file->executable).
|
|
|
|
*
|
|
|
|
* - Popup menus for icons.
|
|
|
|
*
|
|
|
|
* - Select icons with rubberband on the root window.
|
|
|
|
*
|
|
|
|
*/
|
1998-12-09 17:23:38 +00:00
|
|
|
|
1998-10-30 00:47:54 +00:00
|
|
|
#include <config.h>
|
1998-11-02 07:26:19 +00:00
|
|
|
#include "fs.h"
|
1998-12-02 23:37:27 +00:00
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <gtk/gtkinvisible.h>
|
1998-10-30 00:47:54 +00:00
|
|
|
#include <gnome.h>
|
1998-10-30 23:22:35 +00:00
|
|
|
#include "dialog.h"
|
1998-12-07 02:36:47 +00:00
|
|
|
#define DIR_H_INCLUDE_HANDLE_DIRENT /* bleah */
|
|
|
|
#include "dir.h"
|
1998-10-30 00:47:54 +00:00
|
|
|
#include "gdesktop.h"
|
1998-10-30 23:22:35 +00:00
|
|
|
#include "gdesktop-icon.h"
|
1998-12-09 17:23:38 +00:00
|
|
|
#include "gicon.h"
|
1998-12-07 02:36:47 +00:00
|
|
|
#include "gmain.h"
|
1998-10-30 17:08:03 +00:00
|
|
|
#include "gmetadata.h"
|
1998-12-02 23:37:27 +00:00
|
|
|
#include "gdnd.h"
|
1998-11-25 03:29:23 +00:00
|
|
|
#include "gpopup.h"
|
1998-10-30 00:47:54 +00:00
|
|
|
#include "../vfs/vfs.h"
|
|
|
|
|
|
|
|
|
1998-11-07 01:19:53 +00:00
|
|
|
/* Name of the user's desktop directory (i.e. ~/desktop) */
|
1998-10-30 00:47:54 +00:00
|
|
|
#define DESKTOP_DIR_NAME "desktop"
|
|
|
|
|
|
|
|
|
|
|
|
/* This structure defines the information carried by a desktop icon */
|
|
|
|
struct desktop_icon_info {
|
|
|
|
GtkWidget *dicon; /* The desktop icon widget */
|
|
|
|
int x, y; /* Position in the desktop */
|
1998-11-03 01:32:38 +00:00
|
|
|
int slot; /* Index of the slot the icon is in, or -1 for none */
|
1998-10-30 17:08:03 +00:00
|
|
|
char *filename; /* The file this icon refers to (relative to the desktop_directory) */
|
1998-10-30 00:47:54 +00:00
|
|
|
int selected : 1; /* Is the icon selected? */
|
|
|
|
};
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
struct layout_slot {
|
|
|
|
int num_icons; /* Number of icons in this slot */
|
|
|
|
GList *icons; /* The list of icons in this slot */
|
|
|
|
};
|
|
|
|
|
1998-10-30 00:47:54 +00:00
|
|
|
|
1998-10-30 17:08:03 +00:00
|
|
|
/* Configuration options for the desktop */
|
|
|
|
|
1998-10-30 00:47:54 +00:00
|
|
|
int desktop_use_shaped_icons = TRUE;
|
1998-10-30 17:08:03 +00:00
|
|
|
int desktop_auto_placement = FALSE;
|
1998-12-08 01:31:17 +00:00
|
|
|
int desktop_snap_icons = FALSE;
|
1998-10-30 00:47:54 +00:00
|
|
|
|
|
|
|
/* The computed name of the user's desktop directory */
|
|
|
|
static char *desktop_directory;
|
|
|
|
|
1998-10-30 23:22:35 +00:00
|
|
|
/* 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.
|
|
|
|
*/
|
1998-12-07 15:47:00 +00:00
|
|
|
static int layout_screen_width;
|
|
|
|
static int layout_screen_height;
|
1998-10-30 00:47:54 +00:00
|
|
|
static int layout_cols;
|
|
|
|
static int layout_rows;
|
1998-11-03 01:32:38 +00:00
|
|
|
static struct layout_slot *layout_slots;
|
1998-10-30 00:47:54 +00:00
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
#define l_slots(u, v) (layout_slots[(u) * layout_rows + (v)])
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
/* The last icon to be selected */
|
|
|
|
static struct desktop_icon_info *last_selected_icon;
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
/* Drag and drop sources and targets */
|
1998-12-05 01:01:13 +00:00
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
static GtkTargetEntry dnd_icon_sources[] = {
|
1998-12-05 01:01:13 +00:00
|
|
|
{ "application/mc-desktop-icon", 0, TARGET_MC_DESKTOP_ICON },
|
|
|
|
{ "text/uri-list", 0, TARGET_URI_LIST },
|
|
|
|
{ "text/plain", 0, TARGET_TEXT_PLAIN }
|
|
|
|
};
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
static GtkTargetEntry dnd_icon_targets[] = {
|
|
|
|
{ "text/uri-list", 0, TARGET_URI_LIST }
|
|
|
|
};
|
|
|
|
|
|
|
|
static GtkTargetEntry dnd_desktop_targets[] = {
|
1998-12-05 01:01:13 +00:00
|
|
|
{ "application/mc-desktop-icon", 0, TARGET_MC_DESKTOP_ICON },
|
1998-12-02 23:37:27 +00:00
|
|
|
{ "text/uri-list", 0, TARGET_URI_LIST }
|
|
|
|
};
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
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]);
|
1998-12-02 23:37:27 +00:00
|
|
|
|
1998-12-04 02:08:06 +00:00
|
|
|
/* Proxy window for DnD on the root window */
|
1998-12-02 23:37:27 +00:00
|
|
|
static GtkWidget *dnd_proxy_window;
|
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
/* Offsets for the DnD cursor hotspot */
|
|
|
|
static int dnd_press_x, dnd_press_y;
|
|
|
|
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-12-07 02:36:47 +00:00
|
|
|
static struct desktop_icon_info *desktop_icon_info_new (char *filename, int auto_pos, int xpos, int ypos);
|
|
|
|
static void desktop_icon_info_free (struct desktop_icon_info *dii);
|
|
|
|
|
|
|
|
|
1998-10-30 23:22:35 +00:00
|
|
|
/* 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.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
get_icon_auto_pos (int *x, int *y)
|
|
|
|
{
|
|
|
|
int min, min_x, min_y;
|
|
|
|
int u, v;
|
|
|
|
int val;
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
min = l_slots (0, 0).num_icons;
|
1998-10-30 23:22:35 +00:00
|
|
|
min_x = min_y = 0;
|
|
|
|
|
|
|
|
for (u = 0; u < layout_cols; u++)
|
|
|
|
for (v = 0; v < layout_rows; v++) {
|
1998-11-03 01:32:38 +00:00
|
|
|
val = l_slots (u, v).num_icons;
|
1998-10-30 23:22:35 +00:00
|
|
|
|
|
|
|
if (val == 0) {
|
|
|
|
/* Optimization: if it is zero, return immediately */
|
|
|
|
|
|
|
|
*x = u * DESKTOP_SNAP_X;
|
|
|
|
*y = v * DESKTOP_SNAP_Y;
|
|
|
|
return;
|
|
|
|
} else if (val < min) {
|
|
|
|
min = val;
|
|
|
|
min_x = u;
|
|
|
|
min_y = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*x = min_x * DESKTOP_SNAP_X;
|
|
|
|
*y = min_y * DESKTOP_SNAP_Y;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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;
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
min = l_slots (0, 0).num_icons;
|
1998-10-30 23:22:35 +00:00
|
|
|
min_x = min_y = 0;
|
|
|
|
min_dist = INT_MAX;
|
|
|
|
|
|
|
|
for (u = 0; u < layout_cols; u++)
|
|
|
|
for (v = 0; v < layout_rows; v++) {
|
1998-11-03 01:32:38 +00:00
|
|
|
val = l_slots (u, v).num_icons;
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-12-05 01:01:13 +00:00
|
|
|
dx = *x - u * DESKTOP_SNAP_X;
|
|
|
|
dy = *y - v * DESKTOP_SNAP_Y;
|
1998-10-30 23:22:35 +00:00
|
|
|
dist = dx * dx + dy * dy;
|
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
if ((val == min && dist < min_dist) || (val < min)) {
|
1998-12-05 01:01:13 +00:00
|
|
|
min = val;
|
1998-10-30 23:22:35 +00:00
|
|
|
min_dist = dist;
|
|
|
|
min_x = u;
|
|
|
|
min_y = v;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*x = min_x * DESKTOP_SNAP_X;
|
|
|
|
*y = min_y * DESKTOP_SNAP_Y;
|
|
|
|
}
|
1998-10-30 00:47:54 +00:00
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
/* Removes an icon from the slot it is in, if any */
|
|
|
|
static void
|
|
|
|
remove_from_slot (struct desktop_icon_info *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);
|
|
|
|
}
|
|
|
|
|
1998-10-30 17:08:03 +00:00
|
|
|
/* Places a desktop icon. If auto_pos is true, then the function will look for a place to position
|
|
|
|
* the icon automatically, else it will use the specified coordinates, snapped to the grid if the
|
|
|
|
* global desktop_snap_icons flag is set.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
desktop_icon_info_place (struct desktop_icon_info *dii, int auto_pos, int xpos, int ypos)
|
|
|
|
{
|
1998-10-30 23:22:35 +00:00
|
|
|
int u, v;
|
1998-11-24 03:58:05 +00:00
|
|
|
char *filename;
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-12-05 01:01:13 +00:00
|
|
|
if (auto_pos) {
|
|
|
|
if (desktop_auto_placement)
|
|
|
|
get_icon_auto_pos (&xpos, &ypos);
|
|
|
|
else if (desktop_snap_icons)
|
|
|
|
get_icon_snap_pos (&xpos, &ypos);
|
|
|
|
}
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
if (xpos < 0)
|
|
|
|
xpos = 0;
|
|
|
|
else if (xpos > layout_screen_width)
|
1998-12-07 15:47:00 +00:00
|
|
|
xpos = layout_screen_width - DESKTOP_SNAP_X;
|
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
if (ypos < 0)
|
|
|
|
ypos = 0;
|
|
|
|
else if (ypos > layout_screen_height)
|
1998-12-07 15:47:00 +00:00
|
|
|
ypos = layout_screen_height - DESKTOP_SNAP_Y;
|
|
|
|
|
1998-10-30 23:22:35 +00:00
|
|
|
/* Increase the number of icons in the corresponding slot */
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
remove_from_slot (dii);
|
|
|
|
|
1998-10-30 23:22:35 +00:00
|
|
|
u = xpos / DESKTOP_SNAP_X;
|
|
|
|
v = ypos / DESKTOP_SNAP_Y;
|
1998-11-03 01:32:38 +00:00
|
|
|
|
|
|
|
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);
|
1998-10-30 23:22:35 +00:00
|
|
|
|
|
|
|
/* Move the icon */
|
|
|
|
|
|
|
|
dii->x = xpos;
|
|
|
|
dii->y = ypos;
|
|
|
|
gtk_widget_set_uposition (dii->dicon, xpos, ypos);
|
1998-11-24 03:58:05 +00:00
|
|
|
|
|
|
|
/* 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);
|
1998-10-30 17:08:03 +00:00
|
|
|
}
|
|
|
|
|
1998-12-06 21:01:30 +00:00
|
|
|
/* Returns TRUE if there is already an icon in the desktop for the specified filename, FALSE otherwise. */
|
|
|
|
static int
|
|
|
|
icon_exists (char *filename)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
GList *l;
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
|
|
|
|
for (i = 0; i < (layout_cols * layout_rows); i++)
|
|
|
|
for (l = layout_slots[i].icons; l; l = l->next) {
|
|
|
|
dii = l->data;
|
|
|
|
if (strcmp (filename, dii->filename) == 0)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reads the ~/Desktop directory and creates the desktop icons. If incremental is TRUE, then an
|
|
|
|
* icon will not be created for a file if there is already an icon for it, and icons will be created
|
|
|
|
* starting at the specified position.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
load_desktop_icons (int incremental, int xpos, int ypos)
|
|
|
|
{
|
|
|
|
struct dirent *dirent;
|
|
|
|
DIR *dir;
|
|
|
|
char *full_name;
|
|
|
|
int have_pos, x, y;
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
GSList *need_position_list, *l;
|
|
|
|
|
|
|
|
dir = mc_opendir (desktop_directory);
|
|
|
|
if (!dir) {
|
|
|
|
message (FALSE,
|
|
|
|
_("Warning"),
|
|
|
|
_("Could not open %s; will not have initial desktop icons"),
|
|
|
|
desktop_directory);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* First create the icons for all the files that do have their icon position set. Build a
|
|
|
|
* list of the icons that do not have their position set.
|
|
|
|
*/
|
|
|
|
|
|
|
|
need_position_list = NULL;
|
|
|
|
|
|
|
|
while ((dirent = mc_readdir (dir)) != NULL) {
|
|
|
|
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 (incremental && icon_exists (dirent->d_name))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
full_name = g_concat_dir_and_file (desktop_directory, dirent->d_name);
|
|
|
|
|
|
|
|
have_pos = gmeta_get_icon_pos (full_name, &x, &y);
|
|
|
|
|
|
|
|
if (have_pos) {
|
|
|
|
dii = desktop_icon_info_new (dirent->d_name, FALSE, x, y);
|
|
|
|
gtk_widget_show (dii->dicon);
|
|
|
|
|
|
|
|
g_free (full_name);
|
|
|
|
} else
|
|
|
|
need_position_list = g_slist_prepend (need_position_list, g_strdup (dirent->d_name));
|
|
|
|
}
|
|
|
|
|
|
|
|
mc_closedir (dir);
|
|
|
|
|
|
|
|
/* 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);
|
|
|
|
|
|
|
|
for (l = need_position_list; l; l = l->next) {
|
|
|
|
dii = desktop_icon_info_new (l->data, TRUE, xpos, ypos);
|
|
|
|
gtk_widget_show (dii->dicon);
|
|
|
|
g_free (l->data);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_slist_free (need_position_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Destroys all the current desktop icons */
|
|
|
|
static void
|
|
|
|
destroy_desktop_icons (void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
GList *l;
|
|
|
|
struct desktop_icon_info *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_free (dii);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Reloads the desktop icons. If incremental is TRUE, then the existing icons will not be destroyed
|
|
|
|
* first, and the new icons will be put at the specified position.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
reload_desktop_icons (int incremental, int x, int y)
|
|
|
|
{
|
|
|
|
if (!incremental)
|
|
|
|
destroy_desktop_icons ();
|
|
|
|
|
|
|
|
load_desktop_icons (incremental, x, y);
|
|
|
|
}
|
|
|
|
|
1998-11-01 23:21:24 +00:00
|
|
|
/* Unselects all the desktop icons */
|
|
|
|
static void
|
|
|
|
unselect_all (void)
|
|
|
|
{
|
1998-11-03 01:32:38 +00:00
|
|
|
int i;
|
1998-11-01 23:21:24 +00:00
|
|
|
GList *l;
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
for (i = 0; i < (layout_cols * layout_rows); i++)
|
|
|
|
for (l = layout_slots[i].icons; l; l = l->next) {
|
|
|
|
dii = l->data;
|
1998-11-01 23:21:24 +00:00
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
if (dii->selected) {
|
|
|
|
desktop_icon_select (DESKTOP_ICON (dii->dicon), FALSE);
|
|
|
|
dii->selected = FALSE;
|
|
|
|
}
|
1998-11-01 23:21:24 +00:00
|
|
|
}
|
1998-11-03 01:32:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* 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 (struct desktop_icon_info *dii, int sel)
|
|
|
|
{
|
1998-11-03 16:29:47 +00:00
|
|
|
int du, dv, lu, lv;
|
|
|
|
int min_u, min_v;
|
|
|
|
int max_u, max_v;
|
|
|
|
int u, v;
|
1998-11-03 01:32:38 +00:00
|
|
|
GList *l;
|
1998-11-03 16:29:47 +00:00
|
|
|
struct desktop_icon_info *ldii;
|
|
|
|
struct desktop_icon_info *min_udii, *min_vdii;
|
|
|
|
struct desktop_icon_info *max_udii, *max_vdii;
|
1998-11-03 01:32:38 +00:00
|
|
|
|
|
|
|
/* Find out the selection range */
|
|
|
|
|
|
|
|
if (!last_selected_icon)
|
|
|
|
last_selected_icon = dii;
|
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
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;
|
|
|
|
min_udii = dii;
|
|
|
|
max_udii = last_selected_icon;
|
1998-11-03 01:32:38 +00:00
|
|
|
} else {
|
1998-11-03 16:29:47 +00:00
|
|
|
min_u = lu;
|
|
|
|
max_u = du;
|
|
|
|
min_udii = last_selected_icon;
|
|
|
|
max_udii = dii;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dv < lv) {
|
|
|
|
min_v = dv;
|
|
|
|
max_v = lv;
|
|
|
|
min_vdii = dii;
|
|
|
|
max_vdii = last_selected_icon;
|
|
|
|
} else {
|
|
|
|
min_v = lv;
|
|
|
|
max_v = dv;
|
|
|
|
min_vdii = last_selected_icon;
|
|
|
|
max_vdii = dii;
|
1998-11-01 23:21:24 +00:00
|
|
|
}
|
1998-11-03 01:32:38 +00:00
|
|
|
|
1998-11-04 16:51:57 +00:00
|
|
|
/* Select all the icons in the rectangle */
|
1998-11-03 01:32:38 +00:00
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
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;
|
1998-11-03 01:32:38 +00:00
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
if ((u == min_u && ldii->x < min_udii->x)
|
|
|
|
|| (v == min_v && ldii->y < min_vdii->y)
|
|
|
|
|| (u == max_u && ldii->x > max_udii->x)
|
|
|
|
|| (v == max_v && ldii->y > max_vdii->y))
|
|
|
|
continue;
|
1998-11-03 01:32:38 +00:00
|
|
|
|
1998-11-03 16:29:47 +00:00
|
|
|
desktop_icon_select (DESKTOP_ICON (ldii->dicon), sel);
|
|
|
|
ldii->selected = sel;
|
|
|
|
}
|
1998-11-01 23:21:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Handles icon selection and unselection due to button presses */
|
|
|
|
static void
|
|
|
|
select_icon (struct desktop_icon_info *dii, GdkEventButton *event)
|
|
|
|
{
|
1998-11-03 01:32:38 +00:00
|
|
|
int range;
|
|
|
|
int additive;
|
|
|
|
|
|
|
|
range = ((event->state & GDK_SHIFT_MASK) != 0);
|
|
|
|
additive = ((event->state & GDK_CONTROL_MASK) != 0);
|
|
|
|
|
|
|
|
if (!additive)
|
1998-11-01 23:21:24 +00:00
|
|
|
unselect_all ();
|
1998-11-03 01:32:38 +00:00
|
|
|
|
|
|
|
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;
|
1998-11-04 16:51:57 +00:00
|
|
|
|
|
|
|
if (dii->selected)
|
|
|
|
gdk_window_raise (dii->dicon->window);
|
1998-11-03 01:32:38 +00:00
|
|
|
} else
|
|
|
|
select_range (dii, TRUE);
|
1998-11-01 23:21:24 +00:00
|
|
|
}
|
|
|
|
|
1998-12-06 21:01:30 +00:00
|
|
|
/* Creates a file entry structure and fills it with information appropriate to the specified file. */
|
|
|
|
static file_entry *
|
|
|
|
file_entry_from_file (char *filename)
|
|
|
|
{
|
1998-12-07 02:36:47 +00:00
|
|
|
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);
|
|
|
|
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;
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fe;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Frees a file entry structure */
|
|
|
|
static void
|
|
|
|
file_entry_free (file_entry *fe)
|
|
|
|
{
|
|
|
|
if (fe->fname)
|
|
|
|
g_free (fe->fname);
|
1998-12-06 21:01:30 +00:00
|
|
|
|
1998-12-07 02:36:47 +00:00
|
|
|
g_free (fe);
|
1998-12-06 21:01:30 +00:00
|
|
|
}
|
|
|
|
|
1998-11-01 23:21:24 +00:00
|
|
|
/* Handler for events on desktop icons. The on_text flag specifies whether the event ocurred on the
|
|
|
|
* text item in the icon or not.
|
|
|
|
*/
|
|
|
|
static gint
|
|
|
|
desktop_icon_info_event (struct desktop_icon_info *dii, GdkEvent *event, int on_text)
|
|
|
|
{
|
|
|
|
int retval;
|
1998-11-25 03:29:23 +00:00
|
|
|
char *filename;
|
1998-12-07 02:36:47 +00:00
|
|
|
file_entry *fe;
|
1998-11-01 23:21:24 +00:00
|
|
|
|
|
|
|
retval = FALSE;
|
|
|
|
|
|
|
|
switch (event->type) {
|
|
|
|
case GDK_BUTTON_PRESS:
|
1998-12-10 00:46:14 +00:00
|
|
|
if ((event->button.button == 1) && (!dii->selected || (event->button.state & GDK_CONTROL_MASK))) {
|
1998-11-01 23:21:24 +00:00
|
|
|
select_icon (dii, (GdkEventButton *) event);
|
|
|
|
retval = TRUE;
|
1998-11-25 03:29:23 +00:00
|
|
|
} else if (event->button.button == 3) {
|
|
|
|
filename = g_concat_dir_and_file (desktop_directory, dii->filename);
|
1998-12-02 06:33:07 +00:00
|
|
|
|
|
|
|
if (gpopup_do_popup ((GdkEventButton *) event, NULL, 0, filename) != -1)
|
1998-12-07 02:36:47 +00:00
|
|
|
reload_desktop_icons (FALSE, 0, 0); /* bleah */
|
1998-12-02 06:33:07 +00:00
|
|
|
|
1998-12-01 06:23:50 +00:00
|
|
|
g_free (filename);
|
1998-11-25 03:29:23 +00:00
|
|
|
retval = TRUE;
|
|
|
|
}
|
1998-11-01 23:21:24 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_2BUTTON_PRESS:
|
|
|
|
if (event->button.button != 1)
|
|
|
|
break;
|
|
|
|
|
1998-12-06 21:01:30 +00:00
|
|
|
filename = g_concat_dir_and_file (desktop_directory, dii->filename);
|
|
|
|
|
1998-12-07 02:36:47 +00:00
|
|
|
fe = file_entry_from_file (filename);
|
1998-11-01 23:21:24 +00:00
|
|
|
|
1998-12-07 02:36:47 +00:00
|
|
|
if (S_ISDIR (fe->buf.st_mode) || link_isdir (fe))
|
|
|
|
new_panel_at (filename);
|
|
|
|
else
|
1998-12-06 21:01:30 +00:00
|
|
|
do_enter_on_file_entry (fe);
|
1998-12-07 02:36:47 +00:00
|
|
|
|
|
|
|
file_entry_free (fe);
|
1998-12-06 21:01:30 +00:00
|
|
|
|
|
|
|
retval = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GDK_BUTTON_RELEASE:
|
1998-12-10 00:46:14 +00:00
|
|
|
/* select_icon (dii, (GdkEventButton *) event); */
|
1998-11-01 23:21:24 +00:00
|
|
|
retval = TRUE;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we handled the event, do not pass it on to the icon text item */
|
|
|
|
|
|
|
|
if (on_text && retval)
|
|
|
|
gtk_signal_emit_stop_by_name (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text),
|
|
|
|
"event");
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handler for button presses on the images on desktop icons. The desktop icon info structure is
|
|
|
|
* passed in the user data.
|
|
|
|
*/
|
|
|
|
static gint
|
|
|
|
icon_event (GnomeCanvasItem *item, GdkEvent *event, gpointer data)
|
|
|
|
{
|
|
|
|
return desktop_icon_info_event (data, event, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Handler for button presses on the text on desktop icons. The desktop icon info structure is
|
|
|
|
* passed in the user data.
|
|
|
|
*/
|
|
|
|
static gint
|
|
|
|
text_event (GnomeCanvasItem *item, GdkEvent *event, gpointer data)
|
|
|
|
{
|
|
|
|
return desktop_icon_info_event (data, event, TRUE);
|
|
|
|
}
|
|
|
|
|
1998-11-05 23:24:47 +00:00
|
|
|
/* 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)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
1998-11-07 01:19:53 +00:00
|
|
|
char *new_name;
|
1998-11-05 23:24:47 +00:00
|
|
|
char *source;
|
|
|
|
char *dest;
|
|
|
|
int retval;
|
|
|
|
|
|
|
|
dii = data;
|
|
|
|
|
|
|
|
source = g_concat_dir_and_file (desktop_directory, dii->filename);
|
1998-11-07 01:19:53 +00:00
|
|
|
new_name = gnome_icon_text_item_get_text (iti);
|
|
|
|
dest = g_concat_dir_and_file (desktop_directory, new_name);
|
1998-11-05 23:24:47 +00:00
|
|
|
|
1998-11-07 01:19:53 +00:00
|
|
|
if (mc_rename (source, dest) == 0) {
|
|
|
|
g_free (dii->filename);
|
|
|
|
dii->filename = g_strdup (new_name);
|
1998-11-05 23:24:47 +00:00
|
|
|
retval = TRUE;
|
1998-11-07 01:19:53 +00:00
|
|
|
} else
|
1998-11-05 23:24:47 +00:00
|
|
|
retval = FALSE; /* FIXME: maybe pop up a warning/query dialog? */
|
|
|
|
|
|
|
|
g_free (source);
|
|
|
|
g_free (dest);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
1998-11-04 23:01:24 +00:00
|
|
|
/* 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)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
GdkCursor *ibeam;
|
|
|
|
|
|
|
|
dii = data;
|
|
|
|
|
|
|
|
ibeam = gdk_cursor_new (GDK_XTERM);
|
|
|
|
gdk_pointer_grab (GTK_LAYOUT (DESKTOP_ICON (dii->dicon)->canvas)->bin_window,
|
|
|
|
FALSE,
|
|
|
|
(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);
|
|
|
|
|
|
|
|
gdk_keyboard_grab (GTK_LAYOUT (DESKTOP_ICON (dii->dicon)->canvas)->bin_window, FALSE, GDK_CURRENT_TIME);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
{
|
|
|
|
gdk_pointer_ungrab (GDK_CURRENT_TIME);
|
|
|
|
gdk_keyboard_ungrab (GDK_CURRENT_TIME);
|
|
|
|
}
|
|
|
|
|
1998-12-05 01:01:13 +00:00
|
|
|
/* Callback used to store the button press position for the hot spot of the DnD cursor */
|
|
|
|
static gint
|
|
|
|
button_press (GtkWidget *widget, GdkEventButton *event, gpointer data)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
|
|
|
|
dii = data;
|
1998-12-07 21:07:27 +00:00
|
|
|
dnd_press_x = event->x;
|
|
|
|
dnd_press_y = event->y;
|
1998-12-05 01:01:13 +00:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* 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)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
DesktopIcon *dicon;
|
|
|
|
GtkArg args[3];
|
|
|
|
GdkImlibImage *im;
|
|
|
|
GdkPixmap *pixmap;
|
|
|
|
GdkBitmap *mask;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
dii = data;
|
|
|
|
dicon = DESKTOP_ICON (dii->dicon);
|
|
|
|
|
|
|
|
/* 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,
|
1998-12-07 21:07:27 +00:00
|
|
|
dnd_press_x - x,
|
|
|
|
dnd_press_y - y);
|
1998-12-05 01:01:13 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
struct desktop_icon_info *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)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
char *filelist;
|
|
|
|
int len;
|
|
|
|
|
|
|
|
dii = data;
|
|
|
|
|
|
|
|
switch (info) {
|
|
|
|
case TARGET_MC_DESKTOP_ICON:
|
|
|
|
case TARGET_URI_LIST:
|
|
|
|
case TARGET_TEXT_PLAIN:
|
|
|
|
filelist = build_selected_icons_uri_list (&len);
|
|
|
|
gtk_selection_data_set (selection_data,
|
|
|
|
selection_data->target,
|
|
|
|
8,
|
|
|
|
filelist,
|
|
|
|
len);
|
|
|
|
g_free (filelist);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
/* Set up a desktop icon as a DnD source */
|
|
|
|
static void
|
|
|
|
setup_icon_dnd_source (struct desktop_icon_info *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);
|
|
|
|
|
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "button_press_event",
|
|
|
|
(GtkSignalFunc) button_press,
|
|
|
|
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",
|
|
|
|
GTK_SIGNAL_FUNC (drag_data_get),
|
|
|
|
dii);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
icon_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
|
|
|
|
GtkSelectionData *data, guint info, guint time, gpointer user_data)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *dii;
|
|
|
|
char *filename;
|
|
|
|
file_entry *fe;
|
|
|
|
|
|
|
|
printf ("Here!\n");
|
|
|
|
|
|
|
|
dii = user_data;
|
|
|
|
filename = g_concat_dir_and_file (desktop_directory, dii->filename);
|
|
|
|
|
|
|
|
fe = file_entry_from_file (filename);
|
|
|
|
if (!fe)
|
|
|
|
return; /* eek */
|
|
|
|
|
|
|
|
switch (info) {
|
|
|
|
case TARGET_URI_LIST:
|
|
|
|
if (fe->f.link_to_dir)
|
1998-12-08 23:21:15 +00:00
|
|
|
gdnd_drop_on_directory (context, data, filename);
|
1998-12-08 01:31:17 +00:00
|
|
|
else if (is_exe (fe->buf.st_mode) && if_link_is_exe (fe))
|
1998-12-08 23:21:15 +00:00
|
|
|
printf ("Implement execution of desktop icons!\n"); /* FIXME: launch with the dropped files */
|
1998-12-08 01:31:17 +00:00
|
|
|
else
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up a desktop icon as a DnD destination */
|
|
|
|
static void
|
|
|
|
setup_icon_dnd_dest (struct desktop_icon_info *dii)
|
|
|
|
{
|
|
|
|
char *filename;
|
|
|
|
file_entry *fe;
|
|
|
|
int actions;
|
|
|
|
|
|
|
|
filename = g_concat_dir_and_file (desktop_directory, dii->filename);
|
|
|
|
fe = file_entry_from_file (filename);
|
|
|
|
g_free (filename);
|
|
|
|
|
|
|
|
if (!fe)
|
|
|
|
return; /* eek */
|
|
|
|
|
|
|
|
if (fe->f.link_to_dir)
|
|
|
|
actions = GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK | GDK_ACTION_ASK;
|
|
|
|
else if (is_exe (fe->buf.st_mode) && if_link_is_exe (fe))
|
|
|
|
actions = GDK_ACTION_COPY;
|
|
|
|
else
|
|
|
|
actions = 0;
|
|
|
|
|
|
|
|
file_entry_free (fe);
|
|
|
|
|
|
|
|
if (!actions)
|
|
|
|
return;
|
|
|
|
|
|
|
|
gtk_drag_dest_set (DESKTOP_ICON (dii->dicon)->canvas,
|
|
|
|
GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP,
|
|
|
|
dnd_icon_targets,
|
|
|
|
dnd_icon_ntargets,
|
|
|
|
actions);
|
|
|
|
|
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->canvas), "drag_data_received",
|
|
|
|
GTK_SIGNAL_FUNC (icon_drag_data_received),
|
|
|
|
dii);
|
|
|
|
}
|
|
|
|
|
1998-10-30 00:47:54 +00:00
|
|
|
/* Creates a new desktop icon. The filename is the pruned filename inside the desktop directory.
|
1998-12-05 01:01:13 +00:00
|
|
|
* If auto_pos is false, it will use the specified coordinates for the icon. Else, it will use
|
|
|
|
* auto- positioning trying to start at the specified coordinates. It does not show the icon.
|
1998-10-30 00:47:54 +00:00
|
|
|
*/
|
1998-10-30 23:22:35 +00:00
|
|
|
static struct desktop_icon_info *
|
1998-10-30 00:47:54 +00:00
|
|
|
desktop_icon_info_new (char *filename, int auto_pos, int xpos, int ypos)
|
|
|
|
{
|
1998-10-30 17:08:03 +00:00
|
|
|
struct desktop_icon_info *dii;
|
1998-12-09 17:23:38 +00:00
|
|
|
file_entry *fe;
|
1998-10-30 17:08:03 +00:00
|
|
|
char *full_name;
|
1998-12-09 17:23:38 +00:00
|
|
|
GdkImlibImage *icon_im;
|
1998-11-01 23:21:24 +00:00
|
|
|
|
|
|
|
/* Create the icon structure */
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
full_name = g_concat_dir_and_file (desktop_directory, filename);
|
1998-12-09 17:23:38 +00:00
|
|
|
fe = file_entry_from_file (full_name);
|
|
|
|
icon_im = gicon_get_icon_for_file (fe);
|
1998-10-30 17:08:03 +00:00
|
|
|
|
|
|
|
dii = g_new (struct desktop_icon_info, 1);
|
1998-12-09 17:23:38 +00:00
|
|
|
dii->dicon = desktop_icon_new (icon_im, filename);
|
1998-11-03 01:32:38 +00:00
|
|
|
dii->x = 0;
|
|
|
|
dii->y = 0;
|
|
|
|
dii->slot = -1;
|
1998-10-30 17:08:03 +00:00
|
|
|
dii->filename = g_strdup (filename);
|
|
|
|
dii->selected = FALSE;
|
|
|
|
|
1998-12-09 17:23:38 +00:00
|
|
|
file_entry_free (fe);
|
1998-10-30 17:08:03 +00:00
|
|
|
g_free (full_name);
|
|
|
|
|
1998-11-01 23:21:24 +00:00
|
|
|
/* Connect to the icon's signals */
|
|
|
|
|
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->icon), "event",
|
|
|
|
(GtkSignalFunc) icon_event,
|
|
|
|
dii);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "event",
|
|
|
|
(GtkSignalFunc) text_event,
|
|
|
|
dii);
|
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->stipple), "event",
|
|
|
|
(GtkSignalFunc) icon_event,
|
|
|
|
dii);
|
1998-10-30 23:22:35 +00:00
|
|
|
|
1998-11-04 23:01:24 +00:00
|
|
|
/* Connect to the text item's signals */
|
|
|
|
|
1998-11-05 23:24:47 +00:00
|
|
|
gtk_signal_connect (GTK_OBJECT (DESKTOP_ICON (dii->dicon)->text), "text_changed",
|
|
|
|
(GtkSignalFunc) text_changed,
|
|
|
|
dii);
|
1998-11-04 23:01:24 +00:00
|
|
|
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);
|
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
/* Prepare the DnD functionality for this icon */
|
1998-12-05 01:01:13 +00:00
|
|
|
|
1998-12-08 01:31:17 +00:00
|
|
|
setup_icon_dnd_source (dii);
|
|
|
|
setup_icon_dnd_dest (dii);
|
1998-12-05 01:01:13 +00:00
|
|
|
|
1998-11-01 23:21:24 +00:00
|
|
|
/* Place the icon and append it to the list */
|
|
|
|
|
|
|
|
desktop_icon_info_place (dii, auto_pos, xpos, ypos);
|
1998-10-30 23:22:35 +00:00
|
|
|
return dii;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Frees a desktop icon information structure, and destroy the icon widget. Does not remove the
|
|
|
|
* structure from the desktop_icons list!
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
desktop_icon_info_free (struct desktop_icon_info *dii)
|
|
|
|
{
|
|
|
|
gtk_widget_destroy (dii->dicon);
|
1998-11-03 01:32:38 +00:00
|
|
|
remove_from_slot (dii);
|
1998-10-30 23:22:35 +00:00
|
|
|
|
|
|
|
g_free (dii->filename);
|
|
|
|
g_free (dii);
|
1998-10-30 00:47:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Creates the layout information array */
|
|
|
|
static void
|
|
|
|
create_layout_info (void)
|
|
|
|
{
|
1998-12-07 15:47:00 +00:00
|
|
|
layout_screen_width = gdk_screen_width ();
|
|
|
|
layout_screen_height = gdk_screen_height ();
|
|
|
|
layout_cols = (layout_screen_width + DESKTOP_SNAP_X - 1) / DESKTOP_SNAP_X;
|
|
|
|
layout_rows = (layout_screen_height + DESKTOP_SNAP_Y - 1) / DESKTOP_SNAP_Y;
|
1998-11-03 01:32:38 +00:00
|
|
|
layout_slots = g_new0 (struct layout_slot, layout_cols * layout_rows);
|
1998-10-30 00:47:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Check that the user's desktop directory exists, and if not, create it with a symlink to the
|
|
|
|
* user's home directory so that an icon will be displayed.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
create_desktop_dir (void)
|
|
|
|
{
|
|
|
|
char *home_link_name;
|
|
|
|
|
|
|
|
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 link to the user's home directory so that he will have an icon */
|
|
|
|
|
|
|
|
home_link_name = g_concat_dir_and_file (desktop_directory, _("Home directory"));
|
|
|
|
|
|
|
|
if (mc_symlink (gnome_user_home_dir, home_link_name) != 0) {
|
|
|
|
message (FALSE,
|
|
|
|
_("Warning"),
|
|
|
|
_("Could not symlink %s to %s; will not have initial desktop icons."),
|
|
|
|
gnome_user_home_dir, home_link_name);
|
|
|
|
g_free (home_link_name);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (home_link_name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
1998-12-02 23:37:27 +00:00
|
|
|
/* 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;
|
|
|
|
guint32 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 ());
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
1998-12-07 15:47:00 +00:00
|
|
|
/* Returns the desktop icon that started the drag from the specified context */
|
|
|
|
struct desktop_icon_info *
|
|
|
|
find_icon_by_drag_context (GdkDragContext *context)
|
|
|
|
{
|
|
|
|
GtkWidget *source;
|
|
|
|
int i;
|
|
|
|
GList *l;
|
|
|
|
struct desktop_icon_info *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)
|
|
|
|
{
|
|
|
|
struct desktop_icon_info *source_dii, *dii;
|
|
|
|
int dx, dy;
|
|
|
|
int i;
|
|
|
|
GList *l;
|
|
|
|
GSList *sel_icons, *sl;
|
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
/* FIXME: this needs to do the right thing (what Windows does) when desktop_auto_placement
|
|
|
|
* is enabled.
|
|
|
|
*/
|
|
|
|
|
1998-12-07 15:47:00 +00:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compute the distance to move icons */
|
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
if (desktop_snap_icons)
|
|
|
|
get_icon_snap_pos (&x, &y);
|
|
|
|
|
|
|
|
dx = x - source_dii->x - dnd_press_x;
|
|
|
|
dy = y - source_dii->y - dnd_press_y;
|
1998-12-07 15:47:00 +00:00
|
|
|
|
|
|
|
/* 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 */
|
|
|
|
|
|
|
|
for (sl = sel_icons; sl; sl = sl->next) {
|
|
|
|
dii = sl->data;
|
|
|
|
desktop_icon_info_place (dii, FALSE, dii->x + dx, dii->y + dy);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clean up */
|
|
|
|
|
|
|
|
g_slist_free (sel_icons);
|
|
|
|
}
|
|
|
|
|
1998-12-02 23:37:27 +00:00
|
|
|
/* Callback used when the root window receives a drop */
|
1998-12-03 16:11:01 +00:00
|
|
|
static void
|
1998-12-08 01:31:17 +00:00
|
|
|
desktop_drag_data_received (GtkWidget *widget, GdkDragContext *context, gint x, gint y,
|
|
|
|
GtkSelectionData *data, guint info, guint time, gpointer user_data)
|
1998-12-02 23:37:27 +00:00
|
|
|
{
|
1998-12-04 02:08:06 +00:00
|
|
|
int retval;
|
|
|
|
gint dx, dy;
|
|
|
|
|
1998-12-07 21:07:27 +00:00
|
|
|
/* Fix the proxy window offsets */
|
|
|
|
|
|
|
|
gdk_window_get_position (widget->window, &dx, &dy);
|
|
|
|
x += dx;
|
|
|
|
y += dy;
|
|
|
|
|
1998-12-02 23:37:27 +00:00
|
|
|
switch (info) {
|
1998-12-05 01:01:13 +00:00
|
|
|
case TARGET_MC_DESKTOP_ICON:
|
1998-12-07 15:47:00 +00:00
|
|
|
drop_desktop_icons (context, data, x, y);
|
1998-12-05 01:01:13 +00:00
|
|
|
break;
|
|
|
|
|
1998-12-02 23:37:27 +00:00
|
|
|
case TARGET_URI_LIST:
|
1998-12-04 02:08:06 +00:00
|
|
|
retval = gdnd_drop_on_directory (context, data, desktop_directory);
|
|
|
|
if (retval)
|
|
|
|
reload_desktop_icons (TRUE, x, y);
|
1998-12-02 23:37:27 +00:00
|
|
|
|
1998-12-07 15:47:00 +00:00
|
|
|
break;
|
1998-12-02 23:37:27 +00:00
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Sets up drag and drop to the desktop root window */
|
|
|
|
static void
|
|
|
|
setup_desktop_dnd (void)
|
|
|
|
{
|
|
|
|
dnd_proxy_window = gtk_invisible_new ();
|
|
|
|
gtk_widget_show (dnd_proxy_window);
|
|
|
|
|
|
|
|
if (!setup_xdnd_proxy (GDK_ROOT_WINDOW (), dnd_proxy_window->window))
|
|
|
|
g_warning ("Eeeeek, some moron is already taking drops on the root window!");
|
|
|
|
|
|
|
|
gtk_drag_dest_set (dnd_proxy_window,
|
1998-12-04 02:08:06 +00:00
|
|
|
GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_DROP,
|
1998-12-08 01:31:17 +00:00
|
|
|
dnd_desktop_targets,
|
|
|
|
dnd_desktop_ntargets,
|
1998-12-04 02:08:06 +00:00
|
|
|
GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK | GDK_ACTION_ASK);
|
1998-12-02 23:37:27 +00:00
|
|
|
|
|
|
|
gtk_signal_connect (GTK_OBJECT (dnd_proxy_window), "drag_data_received",
|
1998-12-08 01:31:17 +00:00
|
|
|
GTK_SIGNAL_FUNC (desktop_drag_data_received),
|
|
|
|
NULL);
|
1998-12-02 23:37:27 +00:00
|
|
|
}
|
|
|
|
|
1998-10-30 00:47:54 +00:00
|
|
|
/**
|
|
|
|
* 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)
|
|
|
|
{
|
|
|
|
create_layout_info ();
|
|
|
|
create_desktop_dir ();
|
1998-12-04 02:08:06 +00:00
|
|
|
load_desktop_icons (FALSE, 0, 0);
|
1998-12-02 23:37:27 +00:00
|
|
|
setup_desktop_dnd ();
|
1998-10-30 00:47:54 +00:00
|
|
|
}
|
|
|
|
|
1998-11-03 01:32:38 +00:00
|
|
|
/**
|
|
|
|
* desktop_destroy
|
1998-10-30 00:47:54 +00:00
|
|
|
*
|
|
|
|
* Shuts the desktop down by destroying the desktop icons.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
desktop_destroy (void)
|
|
|
|
{
|
|
|
|
/* Destroy the desktop icons */
|
|
|
|
|
1998-12-02 23:37:27 +00:00
|
|
|
destroy_desktop_icons ();
|
1998-10-30 00:47:54 +00:00
|
|
|
|
|
|
|
/* Cleanup */
|
|
|
|
|
|
|
|
g_free (layout_slots);
|
|
|
|
layout_slots = NULL;
|
|
|
|
layout_cols = 0;
|
|
|
|
layout_rows = 0;
|
|
|
|
|
|
|
|
g_free (desktop_directory);
|
|
|
|
desktop_directory = NULL;
|
1998-12-02 23:37:27 +00:00
|
|
|
|
|
|
|
/* Remove DnD crap */
|
|
|
|
|
|
|
|
gtk_widget_destroy (dnd_proxy_window);
|
|
|
|
XDeleteProperty (GDK_DISPLAY (), GDK_ROOT_WINDOW (), gdk_atom_intern ("XdndProxy", FALSE));
|
1998-10-30 00:47:54 +00:00
|
|
|
}
|