nautilus/nautilus-drives-and-volumes-on-desktop.diff

1304 lines
43 KiB
Diff
Raw Normal View History

2007-05-24 Hans Petter Jansson <hpj@novell.com>
Fix https://bugzilla.novell.com/show_bug.cgi?id=276193, unable
to right-click after accessing files on CD via GUI. The problem
was that Nautilus was crashing under some circumstances when
unmounting/ejecting a CD. It was caused by some of the code
previously introduced by this patch.
The fix consists of ignoring the desktop link we are currently
repopulating when scanning the list of desktop links to uniquefy
the new filename for this link.
* libnautilus-private/nautilus-desktop-link-monitor.[ch]
(nautilus_desktop_link_monitor_make_filename_unique): Add a
"skip_link" argument that will make the algorithm skip the
provided desktop link when considering non-unique filenames.
(volume_file_name_used): Skip the link passed from caller,
if any.
2006-11-08 Federico Mena Quintero <federico@novell.com>
Fix https://bugzilla.novell.com/show_bug.cgi?id=215351, the icon
for a floppy drive disappears after unmounting the floppy. Also,
unmounting an NFS volume makes the icon disappear.
Fix the little bug where a recently-unmounted drive still
shows an "Unmount Volume" menu item.
* libnautilus-private/nautilus-desktop-link-monitor.c
(drive_has_volumes_apart_from): New function; checks whether a
drive has any mounted volumes apart from a given one. We need
this to check that corresponds to a volume that is being
unmounted, since within the volume_unmounted callback, the drive
still thinks it is mounted (as the volume in question has not been
removed from it yet).
(should_show_drive): Check whether the drive has any volumes left
over aside from the volume being unmounted.
(volume_unmounted_callback): Pass the volume to should_show_drive().
* libnautilus-private/nautilus-desktop-icon-file.c
(update_info_from_link): If the link acquires a volume, remove the
drive from the corresponding NautilusFile. And the converse: if
the link acquires a drive, remove the volume from the
corresponding NautilusFile. Files can have volumes XOR drives, or
none. This fixes the bug where you mount a drive, then umount the
corresponding volume, and the menu on the drive icon still shows
"unmount volume".
* libnautilus-private/nautilus-desktop-link.c
(nautilus_desktop_link_update_from_volume): Removed spurious
comment.
2006-11-07 Federico Mena Quintero <federico@novell.com>
Deal with a race condition when mounting drives. The callback
from gnome_vfs_drive_mount() can be called before the
GnomeVFSVolumeMonitor is notified that the drive is actually
mounted.
* src/file-manager/fm-directory-view.c (ActivateParameters):
Replace the "pending_mounts" counter with a list of
drives_being_mounted.
(DriveBeingMounted): New struct used as a closure for a drive
being mounted, or a drive waiting for its corresponding
NautilusFile to be notified of changes from the volume monitor.
(cancel_activate): Replace pending_mounts with
drives_being_mounted.
(activate_activation_uris_ready_callback): Don't use the
not_yet_mounted list. Instead, build the list of
parameters->drives_being_mounted out of DriveBeingMounted
structures. Mount the drives directly here (instead of using
g_list_foreach()) so that we can pass a DriveBeingMounted closure
to gnome_vfs_drive_mount().
(fm_directory_view_activate_files): Replace pending_mounts with
drives_being_mounted.
(try_to_finish_drive_activation): New function. Moved the code to
finish activation of drives from
activation_drive_mounted_callback() to here.
(activation_drive_mounted_callback): This is our "drive was
mounted" callback, but the GnomeVFSDrive itself may not think that
it is mounted yet. This is a race condition where the
GnomeVFSVolumeMonitor has not been notified yet. In this case,
wait until the corresponding NautilusFile changes; we'll check
again that the drive is mounted in the "changed" callback.
(activation_file_changed_after_drive_mounted): New callback for
NautilusFile::changed. This gets called when a NautilusFile
changes after a drive was reported as being mounted. Here, we try
to finish the drive activation.
2006-10-05 Federico Mena Quintero <federico@novell.com>
Support displaying drive icons in the desktop, even if the drives
are not mounted. Do this only for drives which are supposed to be
user-visible (i.e. those that do not support media checks and so
we cannot know when the user inserts media in them).
Also, support drives with multiple partitions, which will lead to
multiple volumes for the same drive.
To do this, we add several things:
1. The ability for NautilusDesktopLink to represent a drive as
well as a volume.
2. The ability to "transform" a NautilusDesktopLink which
represented a drive, into one that represents a volume. This
happens when a drive is mounted.
3. Keep the invariant that drives are only displayed if they are
unmounted and user visible. If they are mounted, they get
replaced with volume links.
* libnautilus-private/nautilus-desktop-link-monitor.c (struct
NautilusDesktopLinkMonitorDetails): Added "connected_id" and
"disconnected_id" fields for the signal IDs of GnomeVFSDrive.
(volume_delete_dialog): Added message about not being able to move
a drive to the trash. Support drives or volumes.
(should_show_drive): New utility function. Drives are shown if
they are user-visible, they are not mounted, and the preference to
show desktop volumes is turned on.
(should_show_volume): New utility function. Volumes are shown if
they are user visible and the preference to show desktop volumes
is turned on.
(create_drive_link): New utility function; creates a
NautilusDesktopLink for a GnomeVFSDrive.
(create_volume_link): Use should_show_volume().
(link_corresponds_to_drive): New function. A NautilusDesktopLink
corresponds to a drive if it represents the drive itself, or if it
represents a volume relative to that drive.
(find_unique_link_for_drive): New function. Returns the
NautilusDesktopLink in the desktop that corresponds uniquely to a
given drive; if there is more than one volume link for a drive
(e.g. a drive with multiple partitions), returns NULL.
(drive_connected_callback): New callback. When a drive is
connected, we create a link for it if appropriate.
(drive_disconnected_callback): New callback. When a drive is
disconnected, we remove all the links that correspond to it: a
single drive link for an unmounted drive, or one or more volume
links for a mounted drive.
(volume_mounted_callback): If the volume has a drive, and there is
an existing link for that drive, update the link to represent the
volume instead. Otherwise, create volume link as usual.
(find_link_for_volume): New utility function; returns the link on
the desktop which corresponds to a particular volume.
(volume_unmounted_callback): If unmounting a volume would yield a
drive that is user visible, transform the corresponding link from
representing the volume, into one that represents the drive.
Otherwise, remove the link as usual.
(refresh_volume_links): New function; does the initial population
of the links for drives and volumes.
(desktop_volumes_visible_changed): New callback; use
refresh_volume_links().
(nautilus_desktop_link_monitor_init): Use refresh_volume_links()
instead of doing things by hand here. Also, connect to
"drive_connected" and "drive_disconnected" on the volume monitor.
* libnautilus-private/nautilus-desktop-icon-file.c
(update_info_from_link): Accept a drive or a volume, not just
volumes.
* libnautilus-private/nautilus-desktop-link.h
(nautilus_desktop_link_new_from_drive_or_volume): New prototype.
Replaces nautilus_desktop_link_new_from_volume().
(nautilus_desktop_link_get_drive_or_volume): New prototype.
Replaces nautilus_desktop_link_get_volume().
(nautilus_desktop_link_update_from_volume): New prototype.
* libnautilus-private/nautilus-desktop-link.c (struct
NautilusDesktopLinkDetails): Replaced the "volume" field with
"drive_or_volume".
(reread_drive_or_volume): Refresh the link as appropriate from a
drive or a volume.
(desktop_link_finalize): Unref the drive or volume appropriately.
(nautilus_desktop_link_new_from_drive_or_volume): Replacement for
nautilus_desktop_link_new_from_volume().
(nautilus_desktop_link_get_drive_or_volume): Replacement for
nautilus_desktop_link_get_volume().
(nautilus_desktop_link_update_from_volume): New public function;
replaces nautilus_desktop_link_new_from_volume(). Updates the
link's information from a volume. This does the actual
transformation of a link from a drive to volume when an unmounted
drive gets first mounted, and vice-versa.
diff --git a/libnautilus-private/nautilus-desktop-icon-file.c b/libnautilus-private/nautilus-desktop-icon-file.c
index 733be41..9e00f5a 100644
--- a/libnautilus-private/nautilus-desktop-icon-file.c
+++ b/libnautilus-private/nautilus-desktop-icon-file.c
@@ -179,7 +179,7 @@ update_info_from_link (NautilusDesktopIconFile *icon_file)
NautilusFile *file;
GnomeVFSFileInfo *file_info;
NautilusDesktopLink *link;
- GnomeVFSVolume *volume;
+ GObject *drive_or_volume;
file = NAUTILUS_FILE (icon_file);
@@ -216,9 +216,25 @@ update_info_from_link (NautilusDesktopIconFile *icon_file)
GNOME_VFS_FILE_INFO_FIELDS_ACCESS |
GNOME_VFS_FILE_INFO_FIELDS_LINK_COUNT;
- volume = nautilus_desktop_link_get_volume (link);
- nautilus_file_set_volume (file, volume);
- gnome_vfs_volume_unref (volume);
+ drive_or_volume = nautilus_desktop_link_get_drive_or_volume (link);
+
+ if (GNOME_IS_VFS_VOLUME (drive_or_volume)) {
+ GnomeVFSVolume *volume;
+
+ volume = GNOME_VFS_VOLUME (drive_or_volume);
+ nautilus_file_set_volume (file, volume);
+ gnome_vfs_volume_unref (volume);
+
+ nautilus_file_set_drive (file, NULL);
+ } else {
+ GnomeVFSDrive *drive;
+
+ drive = GNOME_VFS_DRIVE (drive_or_volume);
+ nautilus_file_set_drive (file, drive);
+ gnome_vfs_drive_unref (drive);
+
+ nautilus_file_set_volume (file, NULL);
+ }
file->details->file_info_is_up_to_date = TRUE;
diff --git a/libnautilus-private/nautilus-desktop-link-monitor.c b/libnautilus-private/nautilus-desktop-link-monitor.c
index 6469c6a..c8e85d9 100644
--- a/libnautilus-private/nautilus-desktop-link-monitor.c
+++ b/libnautilus-private/nautilus-desktop-link-monitor.c
@@ -52,13 +52,14 @@ struct NautilusDesktopLinkMonitorDetails {
NautilusDesktopLink *trash_link;
NautilusDesktopLink *network_link;
+ gulong connected_id;
+ gulong disconnected_id;
gulong mount_id;
gulong unmount_id;
GList *volume_links;
};
-
static void nautilus_desktop_link_monitor_init (gpointer object,
gpointer klass);
static void nautilus_desktop_link_monitor_class_init (gpointer klass);
@@ -69,6 +70,8 @@ EEL_CLASS_BOILERPLATE (NautilusDesktopLinkMonitor,
static NautilusDesktopLinkMonitor *the_link_monitor = NULL;
+static void refresh_volume_links (NautilusDesktopLinkMonitor *monitor);
+
static void
destroy_desktop_link_monitor (void)
{
@@ -104,41 +107,48 @@ static void
volume_delete_dialog (GtkWidget *parent_view,
NautilusDesktopLink *link)
{
- GnomeVFSVolume *volume;
+ GObject *drive_or_volume;
char *dialog_str;
+ char *detail_str;
char *display_name;
- volume = nautilus_desktop_link_get_volume (link);
+ drive_or_volume = nautilus_desktop_link_get_drive_or_volume (link);
+ if (drive_or_volume == NULL)
+ return;
+
+ display_name = nautilus_desktop_link_get_display_name (link);
- if (volume != NULL) {
- display_name = nautilus_desktop_link_get_display_name (link);
+ if (GNOME_IS_VFS_VOLUME (drive_or_volume)) {
dialog_str = g_strdup_printf (_("You cannot move the volume \"%s\" to the trash."),
display_name);
- g_free (display_name);
-
- if (eject_for_type (gnome_vfs_volume_get_device_type (volume))) {
- eel_run_simple_dialog
- (parent_view,
- FALSE,
- GTK_MESSAGE_ERROR,
- dialog_str,
- _("If you want to eject the volume, please use \"Eject\" in the "
- "popup menu of the volume."),
- GTK_STOCK_OK, NULL);
+
+ if (eject_for_type (gnome_vfs_volume_get_device_type (GNOME_VFS_VOLUME (drive_or_volume)))) {
+ detail_str = _("If you want to eject the volume, please use \"Eject\" in the "
+ "popup menu of the volume.");
} else {
- eel_run_simple_dialog
- (parent_view,
- FALSE,
- GTK_MESSAGE_ERROR,
- dialog_str,
- _("If you want to unmount the volume, please use \"Unmount Volume\" in the "
- "popup menu of the volume."),
- GTK_STOCK_OK, NULL);
+ detail_str = _("If you want to unmount the volume, please use \"Unmount Volume\" in the "
+ "popup menu of the volume.");
}
+ } else {
+ dialog_str = g_strdup_printf (_("You cannot move the drive \"%s\" to the trash."),
+ display_name);
- gnome_vfs_volume_unref (volume);
- g_free (dialog_str);
+ detail_str = NULL;
+ }
+
+ eel_run_simple_dialog (parent_view, FALSE, GTK_MESSAGE_ERROR,
+ dialog_str,
+ detail_str,
+ NULL, GTK_STOCK_OK, NULL);
+
+ if (GNOME_IS_VFS_VOLUME (drive_or_volume)) {
+ gnome_vfs_volume_unref (GNOME_VFS_VOLUME (drive_or_volume));
+ } else {
+ gnome_vfs_drive_unref (GNOME_VFS_DRIVE (drive_or_volume));
}
+
+ g_free (display_name);
+ g_free (dialog_str);
}
void
@@ -161,6 +171,7 @@ nautilus_desktop_link_monitor_delete_link (NautilusDesktopLinkMonitor *monitor,
static gboolean
volume_file_name_used (NautilusDesktopLinkMonitor *monitor,
+ NautilusDesktopLink *skip_link,
const char *name)
{
GList *l;
@@ -168,6 +179,9 @@ volume_file_name_used (NautilusDesktopLinkMonitor *monitor,
gboolean same;
for (l = monitor->details->volume_links; l != NULL; l = l->next) {
+ if (l->data == skip_link)
+ continue;
+
other_name = nautilus_desktop_link_get_file_name (l->data);
same = strcmp (name, other_name) == 0;
g_free (other_name);
@@ -182,6 +196,7 @@ volume_file_name_used (NautilusDesktopLinkMonitor *monitor,
char *
nautilus_desktop_link_monitor_make_filename_unique (NautilusDesktopLinkMonitor *monitor,
+ NautilusDesktopLink *skip_link,
const char *filename)
{
char *unique_name;
@@ -189,13 +204,74 @@ nautilus_desktop_link_monitor_make_filename_unique (NautilusDesktopLinkMonitor *
i = 2;
unique_name = g_strdup (filename);
- while (volume_file_name_used (monitor, unique_name)) {
+ while (volume_file_name_used (monitor, skip_link, unique_name)) {
g_free (unique_name);
unique_name = g_strdup_printf ("%s.%d", filename, i++);
}
return unique_name;
}
+static gboolean
+drive_has_volumes_apart_from (GnomeVFSDrive *drive, GnomeVFSVolume *possibly_last_volume)
+{
+ GList *volumes;
+ GList *l;
+ gboolean has_other_volumes;
+
+ has_other_volumes = FALSE;
+
+ volumes = gnome_vfs_drive_get_mounted_volumes (drive);
+
+ for (l = volumes; l; l = l->next) {
+ GnomeVFSVolume *volume;
+
+ volume = GNOME_VFS_VOLUME (l->data);
+ if (volume != possibly_last_volume)
+ has_other_volumes = TRUE;
+
+ gnome_vfs_volume_unref (volume);
+ }
+
+ g_list_free (volumes);
+
+ return has_other_volumes;
+}
+
+static gboolean
+should_show_drive (GnomeVFSDrive *drive, GnomeVFSVolume *possibly_last_volume)
+{
+ gboolean should_show;
+
+ if (possibly_last_volume)
+ should_show = !drive_has_volumes_apart_from (drive, possibly_last_volume);
+ else
+ should_show = !gnome_vfs_drive_is_mounted (drive);
+
+ return (gnome_vfs_drive_is_user_visible (drive)
+ && should_show
+ && eel_preferences_get_boolean (NAUTILUS_PREFERENCES_DESKTOP_VOLUMES_VISIBLE));
+}
+
+static gboolean
+should_show_volume (GnomeVFSVolume *volume)
+{
+ return (gnome_vfs_volume_is_user_visible (volume)
+ && eel_preferences_get_boolean (NAUTILUS_PREFERENCES_DESKTOP_VOLUMES_VISIBLE));
+}
+
+static void
+create_drive_link (NautilusDesktopLinkMonitor *monitor,
+ GnomeVFSDrive *drive)
+{
+ NautilusDesktopLink *link;
+
+ if (!should_show_drive (drive, NULL))
+ return;
+
+ link = nautilus_desktop_link_new_from_drive_or_volume (G_OBJECT (drive));
+ monitor->details->volume_links = g_list_prepend (monitor->details->volume_links, link);
+}
+
static void
create_volume_link (NautilusDesktopLinkMonitor *monitor,
GnomeVFSVolume *volume)
@@ -204,24 +280,198 @@ create_volume_link (NautilusDesktopLinkMonitor *monitor,
link = NULL;
- if (!gnome_vfs_volume_is_user_visible (volume)) {
+ if (!should_show_volume (volume)) {
return;
}
- if (eel_preferences_get_boolean (NAUTILUS_PREFERENCES_DESKTOP_VOLUMES_VISIBLE)) {
- link = nautilus_desktop_link_new_from_volume (volume);
- monitor->details->volume_links = g_list_prepend (monitor->details->volume_links, link);
+ link = nautilus_desktop_link_new_from_drive_or_volume (G_OBJECT (volume));
+ monitor->details->volume_links = g_list_prepend (monitor->details->volume_links, link);
+}
+
+static gboolean
+link_corresponds_to_drive (NautilusDesktopLink *link,
+ GnomeVFSDrive *drive)
+{
+ GObject *drive_or_volume;
+ gboolean same;
+
+ drive_or_volume = nautilus_desktop_link_get_drive_or_volume (link);
+ same = FALSE;
+
+ if (GNOME_IS_VFS_DRIVE (drive_or_volume)) {
+ GnomeVFSDrive *link_drive;
+
+ link_drive = GNOME_VFS_DRIVE (drive_or_volume);
+
+ if (link_drive == drive) {
+ same = TRUE;
+ }
+
+ gnome_vfs_drive_unref (link_drive);
+ } else {
+ GnomeVFSVolume *link_volume;
+ GnomeVFSDrive *link_drive;
+
+ link_volume = GNOME_VFS_VOLUME (drive_or_volume);
+ link_drive = gnome_vfs_volume_get_drive (link_volume);
+
+ if (link_drive == drive) {
+ same = TRUE;
+ }
+
+ if (link_drive) {
+ gnome_vfs_drive_unref (link_drive);
+ }
+
+ gnome_vfs_volume_unref (link_volume);
}
+
+ return same;
}
+static NautilusDesktopLink *
+find_unique_link_for_drive (NautilusDesktopLinkMonitor *monitor,
+ GnomeVFSDrive *drive)
+{
+ GList *l;
+ NautilusDesktopLink *first_volume_link_for_drive;
+
+ first_volume_link_for_drive = NULL;
+ for (l = monitor->details->volume_links; l; l = l->next) {
+ NautilusDesktopLink *link;
+
+ link = NAUTILUS_DESKTOP_LINK (l->data);
+
+ if (link_corresponds_to_drive (link, drive)) {
+ if (first_volume_link_for_drive == NULL) {
+ first_volume_link_for_drive = link;
+ } else {
+ return NULL; /* We know that we have more than
+ * one link for volumes that belong
+ * to the same drive, so there is
+ * no unique link for the drive.
+ */
+ }
+ }
+ }
+
+ return first_volume_link_for_drive;
+}
+
+static void
+drive_connected_callback (GnomeVFSVolumeMonitor *volume_monitor,
+ GnomeVFSDrive *drive,
+ NautilusDesktopLinkMonitor *monitor)
+{
+ NautilusDesktopLink *link;
+
+ /* fprintf (stderr, "drive connected!\n"); */
+
+ link = find_unique_link_for_drive (monitor, drive);
+
+ if (link)
+ return; /* huh, we already have a link for that drive... */
+
+ create_drive_link (monitor, drive);
+}
+
+static void
+drive_disconnected_callback (GnomeVFSVolumeMonitor *volume_monitor,
+ GnomeVFSDrive *drive,
+ NautilusDesktopLinkMonitor *monitor)
+{
+ GList *l;
+
+ /* fprintf (stderr, "drive disconnected!\n"); */
+
+ /* Remove all the links that correspond to that drive, even if they have
+ * mounted volumes.
+ */
+
+ l = monitor->details->volume_links;
+
+ while (l) {
+ GList *next;
+ NautilusDesktopLink *link;
+
+ next = l->next;
+ link = NAUTILUS_DESKTOP_LINK (l->data);
+
+ if (link_corresponds_to_drive (link, drive)) {
+ g_object_unref (link);
+ monitor->details->volume_links = g_list_remove_link (monitor->details->volume_links, l);
+ g_list_free_1 (l);
+ }
+
+ l = next;
+ }
+}
static void
volume_mounted_callback (GnomeVFSVolumeMonitor *volume_monitor,
GnomeVFSVolume *volume,
NautilusDesktopLinkMonitor *monitor)
{
- create_volume_link (monitor, volume);
+ GnomeVFSDrive *drive;
+
+ /* fprintf (stderr, "volume mounted!\n"); */
+
+ drive = gnome_vfs_volume_get_drive (volume);
+
+ if (drive) {
+ NautilusDesktopLink *link;
+
+ /* We may have an existing link for the drive, which needs to be
+ * updated for the volume. Or we may have several volumes
+ * within the same drive; in this case, we need to create a
+ * completely new link.
+ */
+
+ link = find_unique_link_for_drive (monitor, drive);
+ gnome_vfs_drive_unref (drive);
+
+ if (link) {
+ /* fprintf (stderr, "updating desktop link from mounted volume\n"); */
+ nautilus_desktop_link_update_from_volume (link, volume);
+ } else {
+ /* fprintf (stderr, "creating desktop link\n"); */
+ create_volume_link (monitor, volume);
+ }
+ } else {
+ /* fprintf (stderr, "creating desktop link\n"); */
+ create_volume_link (monitor, volume);
+ }
+}
+
+static NautilusDesktopLink *
+find_link_for_volume (NautilusDesktopLinkMonitor *monitor,
+ GnomeVFSVolume *volume)
+{
+ GList *l;
+
+ for (l = monitor->details->volume_links; l != NULL; l = l->next) {
+ NautilusDesktopLink *link;
+ GObject *drive_or_volume;
+ gboolean same;
+
+ link = NAUTILUS_DESKTOP_LINK (l->data);
+ drive_or_volume = nautilus_desktop_link_get_drive_or_volume (link);
+
+ same = FALSE;
+
+ if (GNOME_IS_VFS_VOLUME (drive_or_volume)) {
+ same = (GNOME_VFS_VOLUME (drive_or_volume) == volume);
+ gnome_vfs_volume_unref (GNOME_VFS_VOLUME (drive_or_volume));
+ } else {
+ gnome_vfs_drive_unref (GNOME_VFS_DRIVE (drive_or_volume));
+ }
+
+ if (same)
+ return link;
+ }
+
+ return NULL;
}
@@ -230,22 +480,31 @@ volume_unmounted_callback (GnomeVFSVolumeMonitor *volume_monitor,
GnomeVFSVolume *volume,
NautilusDesktopLinkMonitor *monitor)
{
- GList *l;
NautilusDesktopLink *link;
- GnomeVFSVolume *other_volume;
+ GnomeVFSDrive *drive;
+ gboolean remove_link;
- link = NULL;
- for (l = monitor->details->volume_links; l != NULL; l = l->next) {
- other_volume = nautilus_desktop_link_get_volume (l->data);
- if (volume == other_volume) {
- gnome_vfs_volume_unref (other_volume);
- link = l->data;
- break;
+ /* fprintf (stderr, "volume unmounted!\n"); */
+
+ link = find_link_for_volume (monitor, volume);
+ if (!link) {
+ return;
+ }
+
+ remove_link = FALSE;
+
+ drive = gnome_vfs_volume_get_drive (volume);
+ if (drive) {
+ if (should_show_drive (drive, volume)) {
+ nautilus_desktop_link_update_from_volume (link, volume);
+ } else {
+ remove_link = TRUE;
}
- gnome_vfs_volume_unref (other_volume);
+ } else {
+ remove_link = TRUE;
}
- if (link) {
+ if (remove_link) {
monitor->details->volume_links = g_list_remove (monitor->details->volume_links, link);
g_object_unref (link);
}
@@ -322,32 +581,60 @@ desktop_network_visible_changed (gpointer callback_data)
}
static void
-desktop_volumes_visible_changed (gpointer callback_data)
+refresh_volume_links (NautilusDesktopLinkMonitor *monitor)
{
GnomeVFSVolumeMonitor *volume_monitor;
- NautilusDesktopLinkMonitor *monitor;
- GList *l, *volumes;
volume_monitor = gnome_vfs_get_volume_monitor ();
- monitor = NAUTILUS_DESKTOP_LINK_MONITOR (callback_data);
+
+ /* Free existing links */
+
+ g_list_foreach (monitor->details->volume_links, (GFunc)g_object_unref, NULL);
+ g_list_free (monitor->details->volume_links);
+ monitor->details->volume_links = NULL;
+
+ /* Scan the links again */
if (eel_preferences_get_boolean (NAUTILUS_PREFERENCES_DESKTOP_VOLUMES_VISIBLE)) {
- if (monitor->details->volume_links == NULL) {
- volumes = gnome_vfs_volume_monitor_get_mounted_volumes (volume_monitor);
- for (l = volumes; l != NULL; l = l->next) {
- create_volume_link (monitor, l->data);
- gnome_vfs_volume_unref (l->data);
- }
- g_list_free (volumes);
+ GList *l;
+ GList *volumes, *drives;
+
+ /* Unmounted drives */
+
+ drives = gnome_vfs_volume_monitor_get_connected_drives (volume_monitor);
+ for (l = drives; l != NULL; l = l->next) {
+ GnomeVFSDrive *drive;
+
+ drive = GNOME_VFS_DRIVE (l->data);
+ create_drive_link (monitor, drive);
+ gnome_vfs_drive_unref (drive);
}
- } else {
- g_list_foreach (monitor->details->volume_links, (GFunc)g_object_unref, NULL);
- g_list_free (monitor->details->volume_links);
- monitor->details->volume_links = NULL;
+ g_list_free (drives);
+
+ /* Volumes */
+
+ volumes = gnome_vfs_volume_monitor_get_mounted_volumes (volume_monitor);
+ for (l = volumes; l != NULL; l = l->next) {
+ GnomeVFSVolume *volume;
+
+ volume = GNOME_VFS_VOLUME (l->data);
+ create_volume_link (monitor, volume);
+ gnome_vfs_volume_unref (volume);
+ }
+ g_list_free (volumes);
}
}
static void
+desktop_volumes_visible_changed (gpointer callback_data)
+{
+ NautilusDesktopLinkMonitor *monitor;
+
+ monitor = NAUTILUS_DESKTOP_LINK_MONITOR (callback_data);
+ refresh_volume_links (monitor);
+}
+
+static void
create_link_and_add_preference (NautilusDesktopLink **link_ref,
NautilusDesktopLinkType link_type,
const char *preference_key,
@@ -365,8 +652,6 @@ static void
nautilus_desktop_link_monitor_init (gpointer object, gpointer klass)
{
NautilusDesktopLinkMonitor *monitor;
- GList *l, *volumes;
- GnomeVFSVolume *volume;
GnomeVFSVolumeMonitor *volume_monitor;
monitor = NAUTILUS_DESKTOP_LINK_MONITOR (object);
@@ -404,22 +689,22 @@ nautilus_desktop_link_monitor_init (gpointer object, gpointer klass)
desktop_network_visible_changed,
monitor);
- /* Volume links */
+ /* Drives and volumes */
- volume_monitor = gnome_vfs_get_volume_monitor ();
-
- volumes = gnome_vfs_volume_monitor_get_mounted_volumes (volume_monitor);
- for (l = volumes; l != NULL; l = l->next) {
- volume = l->data;
- create_volume_link (monitor, volume);
- gnome_vfs_volume_unref (volume);
- }
- g_list_free (volumes);
+ refresh_volume_links (monitor);
eel_preferences_add_callback (NAUTILUS_PREFERENCES_DESKTOP_VOLUMES_VISIBLE,
desktop_volumes_visible_changed,
monitor);
+ volume_monitor = gnome_vfs_get_volume_monitor ();
+
+ monitor->details->connected_id = g_signal_connect_object (volume_monitor, "drive_connected",
+ G_CALLBACK (drive_connected_callback),
+ monitor, 0);
+ monitor->details->disconnected_id = g_signal_connect_object (volume_monitor, "drive_disconnected",
+ G_CALLBACK (drive_disconnected_callback),
+ monitor, 0);
monitor->details->mount_id = g_signal_connect_object (volume_monitor, "volume_mounted",
G_CALLBACK (volume_mounted_callback), monitor, 0);
monitor->details->unmount_id = g_signal_connect_object (volume_monitor, "volume_unmounted",
diff --git a/libnautilus-private/nautilus-desktop-link-monitor.h b/libnautilus-private/nautilus-desktop-link-monitor.h
index 2c3c23d..1622ced 100644
--- a/libnautilus-private/nautilus-desktop-link-monitor.h
+++ b/libnautilus-private/nautilus-desktop-link-monitor.h
@@ -59,6 +59,7 @@ void nautilus_desktop_link_monitor_delete_link (NautilusDesktopLinkMonitor *moni
/* Used by nautilus-desktop-link.c */
char * nautilus_desktop_link_monitor_make_filename_unique (NautilusDesktopLinkMonitor *monitor,
+ NautilusDesktopLink *skip_link,
const char *filename);
#endif /* NAUTILUS_DESKTOP_LINK_MONITOR_H */
diff --git a/libnautilus-private/nautilus-desktop-link.c b/libnautilus-private/nautilus-desktop-link.c
index e27bac4..621245a 100644
--- a/libnautilus-private/nautilus-desktop-link.c
+++ b/libnautilus-private/nautilus-desktop-link.c
@@ -54,8 +54,8 @@ struct NautilusDesktopLinkDetails {
/* Just for trash icons: */
gulong trash_state_handler;
- /* Just for volume icons: */
- GnomeVFSVolume *volume;
+ /* Just for drive/volume icons */
+ GObject *drive_or_volume;
};
static void nautilus_desktop_link_init (gpointer object,
@@ -208,50 +208,105 @@ nautilus_desktop_link_new (NautilusDesktopLinkType type)
return link;
}
-NautilusDesktopLink *
-nautilus_desktop_link_new_from_volume (GnomeVFSVolume *volume)
+static void
+reread_drive_or_volume (NautilusDesktopLink *link)
{
- NautilusDesktopLink *link;
- GnomeVFSDrive *drive;
char *name, *filename;
+ char *display_name, *activation_uri, *icon;
- link = NAUTILUS_DESKTOP_LINK (g_object_new (NAUTILUS_TYPE_DESKTOP_LINK, NULL));
-
- link->details->type = NAUTILUS_DESKTOP_LINK_VOLUME;
+ g_assert (link->details->type == NAUTILUS_DESKTOP_LINK_VOLUME);
+ g_assert (link->details->drive_or_volume != NULL);
+
+ g_free (link->details->filename);
+ g_free (link->details->display_name);
+ g_free (link->details->activation_uri);
+ g_free (link->details->icon);
- link->details->volume = gnome_vfs_volume_ref (volume);
+ if (GNOME_IS_VFS_VOLUME (link->details->drive_or_volume)) {
+ GnomeVFSVolume *volume;
+ GnomeVFSDrive *drive;
- /* We try to use the drive name to get somewhat stable filenames
- for metadata */
- drive = gnome_vfs_volume_get_drive (volume);
- if (drive != NULL) {
- name = gnome_vfs_drive_get_display_name (drive);
+ volume = GNOME_VFS_VOLUME (link->details->drive_or_volume);
+
+ /* We try to use the drive name to get somewhat stable filenames
+ for metadata */
+ drive = gnome_vfs_volume_get_drive (volume);
+ if (drive != NULL) {
+ name = gnome_vfs_drive_get_display_name (drive);
+ } else {
+ name = gnome_vfs_volume_get_display_name (volume);
+ }
+ gnome_vfs_drive_unref (drive);
+
+ display_name = gnome_vfs_volume_get_display_name (volume);
+ activation_uri = gnome_vfs_volume_get_activation_uri (volume);
+ icon = gnome_vfs_volume_get_icon (volume);
} else {
- name = gnome_vfs_volume_get_display_name (volume);
+ GnomeVFSDrive *drive;
+
+ g_assert (GNOME_IS_VFS_DRIVE (link->details->drive_or_volume));
+
+ drive = GNOME_VFS_DRIVE (link->details->drive_or_volume);
+
+ name = gnome_vfs_drive_get_display_name (drive);
+
+ display_name = gnome_vfs_drive_get_display_name (drive);
+ activation_uri = NULL; /* We don't know the activation URI until the drive gets mounted */
+ icon = gnome_vfs_drive_get_icon (drive);
}
- gnome_vfs_drive_unref (drive);
filename = g_strconcat (name, ".volume", NULL);
link->details->filename =
nautilus_desktop_link_monitor_make_filename_unique (nautilus_desktop_link_monitor_get (),
+ link,
filename);
g_free (filename);
g_free (name);
+
+ link->details->display_name = display_name;
+ link->details->activation_uri = activation_uri;
+ link->details->icon = icon;
+}
+
+NautilusDesktopLink *
+nautilus_desktop_link_new_from_drive_or_volume (GObject *object)
+{
+ NautilusDesktopLink *link;
+ gboolean is_volume;
+
+ is_volume = GNOME_IS_VFS_VOLUME (object);
+
+ g_return_val_if_fail (GNOME_IS_VFS_DRIVE (object) || is_volume, NULL);
+
+ link = NAUTILUS_DESKTOP_LINK (g_object_new (NAUTILUS_TYPE_DESKTOP_LINK, NULL));
- link->details->display_name = gnome_vfs_volume_get_display_name (volume);
-
- link->details->activation_uri = gnome_vfs_volume_get_activation_uri (volume);
- link->details->icon = gnome_vfs_volume_get_icon (volume);
+ link->details->type = NAUTILUS_DESKTOP_LINK_VOLUME;
+
+ if (is_volume) {
+ GnomeVFSVolume *volume;
+
+ volume = gnome_vfs_volume_ref (GNOME_VFS_VOLUME (object));
+ link->details->drive_or_volume = G_OBJECT (volume);
+ } else {
+ GnomeVFSDrive *drive;
+ drive = gnome_vfs_drive_ref (GNOME_VFS_DRIVE (object));
+ link->details->drive_or_volume = G_OBJECT (drive);
+ }
+
+ reread_drive_or_volume (link);
create_icon_file (link);
return link;
}
-GnomeVFSVolume *
-nautilus_desktop_link_get_volume (NautilusDesktopLink *link)
+GObject *
+nautilus_desktop_link_get_drive_or_volume (NautilusDesktopLink *link)
{
- return gnome_vfs_volume_ref (link->details->volume);
+ if (GNOME_IS_VFS_VOLUME (link->details->drive_or_volume))
+ return G_OBJECT (gnome_vfs_volume_ref (GNOME_VFS_VOLUME (link->details->drive_or_volume)));
+ else
+ return G_OBJECT (gnome_vfs_drive_ref (GNOME_VFS_DRIVE (link->details->drive_or_volume)));
}
@@ -416,7 +471,11 @@ desktop_link_finalize (GObject *object)
}
if (link->details->type == NAUTILUS_DESKTOP_LINK_VOLUME) {
- gnome_vfs_volume_unref (link->details->volume);
+ if (GNOME_IS_VFS_VOLUME (link->details->drive_or_volume)) {
+ gnome_vfs_volume_unref (GNOME_VFS_VOLUME (link->details->drive_or_volume));
+ } else {
+ gnome_vfs_drive_unref (GNOME_VFS_DRIVE (link->details->drive_or_volume));
+ }
}
g_free (link->details->filename);
@@ -438,3 +497,61 @@ nautilus_desktop_link_class_init (gpointer klass)
object_class->finalize = desktop_link_finalize;
}
+
+void
+nautilus_desktop_link_update_from_volume (NautilusDesktopLink *link,
+ GnomeVFSVolume *volume)
+{
+ g_return_if_fail (NAUTILUS_IS_DESKTOP_LINK (link));
+ g_return_if_fail (GNOME_IS_VFS_VOLUME (volume));
+
+ g_assert (link->details->type == NAUTILUS_DESKTOP_LINK_VOLUME);
+ g_assert (link->details->drive_or_volume != NULL);
+
+ if (GNOME_IS_VFS_DRIVE (link->details->drive_or_volume)) {
+ GnomeVFSDrive *drive;
+
+ drive = gnome_vfs_volume_get_drive (volume);
+
+ g_assert (G_OBJECT (drive) == G_OBJECT (link->details->drive_or_volume));
+ gnome_vfs_drive_unref (drive);
+
+ /* The link is for a drive. If the new volume is mounted,
+ * replace the link's object with the volume. If the new volume
+ * is unmounted, just refresh the link (who knows why we didn't
+ * get the mount notification before).
+ */
+
+ if (gnome_vfs_volume_is_mounted (volume)) {
+ gnome_vfs_drive_unref (GNOME_VFS_DRIVE (link->details->drive_or_volume));
+
+ gnome_vfs_volume_ref (volume);
+ link->details->drive_or_volume = G_OBJECT (volume);
+ }
+
+ /* The link will get updated below */
+ } else {
+ g_assert (GNOME_IS_VFS_VOLUME (link->details->drive_or_volume));
+ g_assert (GNOME_VFS_VOLUME (link->details->drive_or_volume) == volume);
+
+ /* If the volume got unmounted, restore the link's object to the
+ * corresponding drive. Otherwise, we shouldn't need to be
+ * called, but just update the link in that case.
+ */
+
+ if (!gnome_vfs_volume_is_mounted (volume)) {
+ GnomeVFSDrive *drive;
+
+ drive = gnome_vfs_volume_get_drive (volume);
+
+ gnome_vfs_volume_unref (GNOME_VFS_VOLUME (link->details->drive_or_volume));
+
+ link->details->drive_or_volume = G_OBJECT (drive);
+ }
+
+ /* The link will get updated below */
+ }
+
+ reread_drive_or_volume (link);
+ nautilus_desktop_link_changed (link);
+}
diff --git a/libnautilus-private/nautilus-desktop-link.h b/libnautilus-private/nautilus-desktop-link.h
index 578e592..e127500 100644
--- a/libnautilus-private/nautilus-desktop-link.h
+++ b/libnautilus-private/nautilus-desktop-link.h
@@ -26,6 +26,7 @@
#define NAUTILUS_DESKTOP_LINK_H
#include <libnautilus-private/nautilus-file.h>
+#include <libgnomevfs/gnome-vfs-drive.h>
#include <libgnomevfs/gnome-vfs-volume.h>
#define NAUTILUS_TYPE_DESKTOP_LINK \
@@ -61,7 +62,7 @@ typedef enum {
GType nautilus_desktop_link_get_type (void);
NautilusDesktopLink * nautilus_desktop_link_new (NautilusDesktopLinkType type);
-NautilusDesktopLink * nautilus_desktop_link_new_from_volume (GnomeVFSVolume *volume);
+NautilusDesktopLink * nautilus_desktop_link_new_from_drive_or_volume (GObject *object);
NautilusDesktopLinkType nautilus_desktop_link_get_link_type (NautilusDesktopLink *link);
char * nautilus_desktop_link_get_file_name (NautilusDesktopLink *link);
char * nautilus_desktop_link_get_display_name (NautilusDesktopLink *link);
@@ -70,10 +71,12 @@ char * nautilus_desktop_link_get_activation_uri (NautilusDeskto
gboolean nautilus_desktop_link_get_date (NautilusDesktopLink *link,
NautilusDateType date_type,
time_t *date);
-GnomeVFSVolume * nautilus_desktop_link_get_volume (NautilusDesktopLink *link);
+GObject * nautilus_desktop_link_get_drive_or_volume (NautilusDesktopLink *link);
gboolean nautilus_desktop_link_can_rename (NautilusDesktopLink *link);
gboolean nautilus_desktop_link_rename (NautilusDesktopLink *link,
const char *name);
+void nautilus_desktop_link_update_from_volume (NautilusDesktopLink *link,
+ GnomeVFSVolume *volume);
#endif /* NAUTILUS_DESKTOP_LINK_H */
diff --git a/src/file-manager/fm-directory-view.c b/src/file-manager/fm-directory-view.c
index aa3b266..0a31995 100644
--- a/src/file-manager/fm-directory-view.c
+++ b/src/file-manager/fm-directory-view.c
@@ -301,11 +301,19 @@ typedef struct {
NautilusWindowOpenMode mode;
NautilusWindowOpenFlags flags;
gboolean mount_success;
- unsigned int pending_mounts;
+ GList *drives_being_mounted;
gboolean cancelled;
} ActivateParameters;
typedef struct {
+ ActivateParameters *parameters;
+
+ GnomeVFSDrive *drive;
+ NautilusFile *file;
+ guint file_changed_id;
+} DriveBeingMounted;
+
+typedef struct {
NautilusFile *file;
NautilusDirectory *directory;
} FileAndDirectory;
@@ -8272,7 +8280,7 @@ cancel_activate (gpointer callback_data)
parameters->cancelled = TRUE;
- if (parameters->pending_mounts == 0) {
+ if (parameters->drives_being_mounted == NULL) {
nautilus_file_list_cancel_call_when_ready (parameters->files_handle);
nautilus_file_list_free (parameters->files);
g_free (parameters);
@@ -8644,16 +8652,81 @@ activate_callback (GList *files, gpointer callback_data)
}
static void
+try_to_finish_drive_activation (ActivateParameters *parameters)
+{
+ if (parameters->drives_being_mounted != NULL) {
+ /* wait for other mounts to finish... */
+ return;
+ }
+
+ if (parameters->cancelled || !parameters->mount_success) {
+ stop_activate (parameters);
+
+ nautilus_file_list_free (parameters->files);
+ g_free (parameters);
+ return;
+ }
+
+ /* all drives were mounted successfully */
+ activate_activation_uris_ready_callback (parameters->files,
+ parameters);
+}
+
+static void
+remove_drive_being_mounted (DriveBeingMounted *drive_being_mounted)
+{
+ ActivateParameters *parameters;
+
+ parameters = drive_being_mounted->parameters;
+
+ if (drive_being_mounted->file_changed_id != 0) {
+ g_signal_handler_disconnect (drive_being_mounted->file, drive_being_mounted->file_changed_id);
+ }
+
+ nautilus_file_unref (drive_being_mounted->file);
+ gnome_vfs_drive_unref (drive_being_mounted->drive);
+
+ parameters->drives_being_mounted = g_list_remove (parameters->drives_being_mounted,
+ drive_being_mounted);
+ g_free (drive_being_mounted);
+}
+
+static void
+activation_file_changed_after_drive_mounted (NautilusFile *file,
+ gpointer data)
+{
+ DriveBeingMounted *drive_being_mounted;
+ ActivateParameters *parameters;
+
+ drive_being_mounted = data;
+ parameters = drive_being_mounted->parameters;
+
+ if (!gnome_vfs_drive_is_mounted (drive_being_mounted->drive)) {
+ /* Huh? The NautilusFile changed, but the drive is not mounted
+ * yet. So, wait some more until the file gets another change
+ * and the drive gets mounted.
+ */
+ return;
+ }
+
+ remove_drive_being_mounted (drive_being_mounted);
+
+ try_to_finish_drive_activation (parameters);
+}
+
+static void
activation_drive_mounted_callback (gboolean succeeded,
char *error,
char *detailed_error,
gpointer callback_data)
{
+ DriveBeingMounted *drive_being_mounted;
ActivateParameters *parameters;
- parameters = callback_data;
+ drive_being_mounted = callback_data;
+ parameters = drive_being_mounted->parameters;
- parameters->mount_success &= succeeded;
+ parameters->mount_success = parameters->mount_success && succeeded;
if (!succeeded && !parameters->cancelled) {
if (*error == 0 &&
@@ -8666,29 +8739,25 @@ activation_drive_mounted_callback (gboolean succeeded,
}
}
- if (--parameters->pending_mounts > 0) {
- /* wait for other mounts to finish... */
- return;
- }
-
- if (parameters->cancelled || !parameters->mount_success) {
- stop_activate (parameters);
+ if (gnome_vfs_drive_is_mounted (drive_being_mounted->drive)) {
+ remove_drive_being_mounted (drive_being_mounted);
+ } else {
+ /* Here, the drive is already mounted, but
+ * gnome-vfs-volume-monitor thinks that it is not. This is
+ * because gnome-vfs-daemon has not yet notified it. So we'll
+ * wait for the NautilusFile to change: this will happen
+ * when gnome-vfs-volume-monitor actually picks up and emits the
+ * volume_mounted notification, and the NautilusDesktopLink
+ * modifies the NautilusFile.
+ */
- nautilus_file_list_free (parameters->files);
- g_free (parameters);
- return;
+ drive_being_mounted->file_changed_id = g_signal_connect (
+ drive_being_mounted->file, "changed",
+ G_CALLBACK (activation_file_changed_after_drive_mounted),
+ drive_being_mounted);
}
- /* all drives were mounted successfully */
- activate_activation_uris_ready_callback (parameters->files,
- parameters);
-}
-
-static void
-mount_foreach (gpointer drive,
- gpointer callback_data)
-{
- gnome_vfs_drive_mount (drive, activation_drive_mounted_callback, callback_data);
+ try_to_finish_drive_activation (parameters);
}
static void
@@ -8696,7 +8765,6 @@ activate_activation_uris_ready_callback (GList *files_ignore,
gpointer callback_data)
{
ActivateParameters *parameters;
- GList *not_yet_mounted;
GList *l, *next;
NautilusFile *file;
NautilusFile *actual_file;
@@ -8705,7 +8773,6 @@ activate_activation_uris_ready_callback (GList *files_ignore,
char *uri;
parameters = callback_data;
- not_yet_mounted = NULL;
for (l = parameters->files; l != NULL; l = next) {
file = NAUTILUS_FILE (l->data);
@@ -8728,24 +8795,46 @@ activate_activation_uris_ready_callback (GList *files_ignore,
drive = nautilus_file_get_drive (file);
if (drive != NULL &&
!gnome_vfs_drive_is_mounted (drive)) {
- not_yet_mounted = g_list_prepend (not_yet_mounted, drive);
- parameters->pending_mounts++;
+ DriveBeingMounted *drive_being_mounted;
+
+ drive_being_mounted = g_new (DriveBeingMounted, 1);
+ drive_being_mounted->parameters = parameters;
+ drive_being_mounted->drive = gnome_vfs_drive_ref (drive);
+ drive_being_mounted->file = nautilus_file_ref (file);
+ drive_being_mounted->file_changed_id = 0;
+
+ parameters->drives_being_mounted = g_list_prepend (parameters->drives_being_mounted,
+ drive_being_mounted);
}
}
}
if (parameters->files == NULL) {
- g_assert (not_yet_mounted == NULL);
+ g_assert (parameters->drives_being_mounted == NULL);
stop_activate (parameters);
g_free (parameters);
return;
}
- if (not_yet_mounted != NULL) {
- not_yet_mounted = g_list_reverse (not_yet_mounted);
- g_list_foreach (not_yet_mounted, mount_foreach, callback_data);
- g_list_free (not_yet_mounted);
+ if (parameters->drives_being_mounted != NULL) {
+ GList *l;
+
+ parameters->drives_being_mounted = g_list_reverse (parameters->drives_being_mounted);
+
+ for (l = parameters->drives_being_mounted; l; l = l->next) {
+ DriveBeingMounted *drive_being_mounted;
+
+ drive_being_mounted = l->data;
+
+ /* @#$% gnome_vfs_drive_mount() doesn't tell the
+ * callback *which* drive was mounted. So, we pass the
+ * drive as part of the drive_being_mounted closure.
+ */
+ gnome_vfs_drive_mount (drive_being_mounted->drive,
+ activation_drive_mounted_callback,
+ drive_being_mounted);
+ }
/* activation_drive_mounted_callback will reveal whether all mounts were successful */
parameters->mount_success = TRUE;
@@ -8869,7 +8958,7 @@ fm_directory_view_activate_files (FMDirectoryView *view,
parameters->mode = mode;
parameters->flags = flags;
parameters->mount_success = FALSE;
- parameters->pending_mounts = 0;
+ parameters->drives_being_mounted = NULL;
parameters->cancelled = FALSE;
if (file_count == 1) {