/* Desktop management for the Midnight Commander * * Copyright (C) 1998-1999 The Free Software Foundation * * Authors: Federico Mena * Miguel de Icaza */ #include #include "global.h" #include #include #include #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" #include "gprefs.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 "\nd_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) { char *desktop_symlink; 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 (); } /* Create a user-visible symlink at ~/Desktop */ desktop_symlink = g_concat_dir_and_file (gnome_user_home_dir, "Desktop"); if (! g_file_exists (desktop_symlink)) symlink (desktop_directory, desktop_symlink); g_free (desktop_symlink); 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); } }