1 /* GTK - The GIMP Toolkit
2 * gtkfilesystem.c: Filesystem abstraction functions.
3 * Copyright (C) 2003, Red Hat, Inc.
4 * Copyright (C) 2007-2008 Carlos Garnacho
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU Lesser General Public License as
8 * published by the Free Software Foundation; either version 2 of the
9 * License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with this program; if not, write to the Free Software
18 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307, USA.
20 * Authors: Carlos Garnacho <carlos@imendio.com>
27 #include <glib/gi18n-lib.h>
29 #include "gtkfilechooser.h"
30 #include "gtkfilesystem.h"
31 #include "gtkicontheme.h"
32 #include "gtkprivate.h"
34 /* #define DEBUG_MODE */
36 #define DEBUG(x) g_debug (x);
41 #define FILES_PER_QUERY 100
43 /* The pointers we return for a GtkFileSystemVolume are opaque tokens; they are
44 * really pointers to GDrive, GVolume or GMount objects. We need an extra
45 * token for the fake "File System" volume. So, we'll return a pointer to
46 * this particular string.
48 static const gchar *root_volume_token = N_("File System");
49 #define IS_ROOT_VOLUME(volume) ((gpointer) (volume) == (gpointer) root_volume_token)
73 static guint fs_signals [FS_LAST_SIGNAL] = { 0, };
74 static guint folder_signals [FOLDER_LAST_SIGNAL] = { 0, };
76 typedef struct AsyncFuncData AsyncFuncData;
78 struct GtkFileSystemPrivate
80 GVolumeMonitor *volume_monitor;
82 /* This list contains elements that can be
83 * of type GDrive, GVolume and GMount
87 /* This list contains GtkFileSystemBookmark structs */
90 GFileMonitor *bookmarks_monitor;
93 struct GtkFolderPrivate
97 GFileMonitor *directory_monitor;
98 GFileEnumerator *enumerator;
99 GCancellable *cancellable;
102 guint finished_loading : 1;
107 GtkFileSystem *file_system;
110 GCancellable *cancellable;
117 struct GtkFileSystemBookmark
123 G_DEFINE_TYPE (GtkFileSystem, _gtk_file_system, G_TYPE_OBJECT)
125 G_DEFINE_TYPE (GtkFolder, _gtk_folder, G_TYPE_OBJECT)
128 static void gtk_folder_set_finished_loading (GtkFolder *folder,
129 gboolean finished_loading);
130 static void gtk_folder_add_file (GtkFolder *folder,
135 /* GtkFileSystemBookmark methods */
137 _gtk_file_system_bookmark_free (GtkFileSystemBookmark *bookmark)
139 g_object_unref (bookmark->file);
140 g_free (bookmark->label);
141 g_slice_free (GtkFileSystemBookmark, bookmark);
144 /* GtkFileSystem methods */
146 volumes_changed (GVolumeMonitor *volume_monitor,
150 GtkFileSystem *file_system;
152 gdk_threads_enter ();
154 file_system = GTK_FILE_SYSTEM (user_data);
155 g_signal_emit (file_system, fs_signals[VOLUMES_CHANGED], 0, volume);
156 gdk_threads_leave ();
160 gtk_file_system_dispose (GObject *object)
162 GtkFileSystem *file_system = GTK_FILE_SYSTEM (object);
163 GtkFileSystemPrivate *priv = file_system->priv;
169 g_slist_foreach (priv->volumes, (GFunc) g_object_unref, NULL);
170 g_slist_free (priv->volumes);
171 priv->volumes = NULL;
174 if (priv->volume_monitor)
176 g_signal_handlers_disconnect_by_func (priv->volume_monitor, volumes_changed, object);
177 g_object_unref (priv->volume_monitor);
178 priv->volume_monitor = NULL;
181 G_OBJECT_CLASS (_gtk_file_system_parent_class)->dispose (object);
185 gtk_file_system_finalize (GObject *object)
187 GtkFileSystem *file_system = GTK_FILE_SYSTEM (object);
188 GtkFileSystemPrivate *priv = file_system->priv;
192 if (priv->bookmarks_monitor)
193 g_object_unref (priv->bookmarks_monitor);
197 g_slist_foreach (priv->bookmarks, (GFunc) _gtk_file_system_bookmark_free, NULL);
198 g_slist_free (priv->bookmarks);
201 G_OBJECT_CLASS (_gtk_file_system_parent_class)->finalize (object);
205 _gtk_file_system_class_init (GtkFileSystemClass *class)
207 GObjectClass *object_class = G_OBJECT_CLASS (class);
209 object_class->dispose = gtk_file_system_dispose;
210 object_class->finalize = gtk_file_system_finalize;
212 fs_signals[BOOKMARKS_CHANGED] =
213 g_signal_new ("bookmarks-changed",
214 G_TYPE_FROM_CLASS (object_class),
216 G_STRUCT_OFFSET (GtkFileSystemClass, bookmarks_changed),
218 g_cclosure_marshal_VOID__VOID,
221 fs_signals[VOLUMES_CHANGED] =
222 g_signal_new ("volumes-changed",
223 G_TYPE_FROM_CLASS (object_class),
225 G_STRUCT_OFFSET (GtkFileSystemClass, volumes_changed),
227 g_cclosure_marshal_VOID__VOID,
230 g_type_class_add_private (object_class, sizeof (GtkFileSystemPrivate));
234 get_bookmarks_file (void)
239 filename = g_build_filename (g_get_home_dir (), ".gtk-bookmarks", NULL);
240 file = g_file_new_for_path (filename);
247 read_bookmarks (GFile *file)
250 gchar **lines, *space;
251 GSList *bookmarks = NULL;
254 if (!g_file_load_contents (file, NULL, &contents,
258 lines = g_strsplit (contents, "\n", -1);
260 for (i = 0; lines[i]; i++)
262 GtkFileSystemBookmark *bookmark;
267 bookmark = g_slice_new0 (GtkFileSystemBookmark);
269 if ((space = strchr (lines[i], ' ')) != NULL)
272 bookmark->label = g_strdup (space + 1);
275 bookmark->file = g_file_new_for_uri (lines[i]);
276 bookmarks = g_slist_prepend (bookmarks, bookmark);
279 bookmarks = g_slist_reverse (bookmarks);
287 save_bookmarks (GFile *bookmarks_file,
290 GError *error = NULL;
293 contents = g_string_new ("");
297 GtkFileSystemBookmark *bookmark;
300 bookmark = bookmarks->data;
301 uri = g_file_get_uri (bookmark->file);
302 g_string_append (contents, uri);
305 g_string_append_printf (contents, " %s", bookmark->label);
307 g_string_append_c (contents, '\n');
308 bookmarks = bookmarks->next;
312 if (!g_file_replace_contents (bookmarks_file,
314 strlen (contents->str),
315 NULL, FALSE, 0, NULL,
318 g_critical ("%s", error->message);
319 g_error_free (error);
322 g_string_free (contents, TRUE);
326 bookmarks_file_changed (GFileMonitor *monitor,
329 GFileMonitorEvent event,
332 GtkFileSystem *file_system = GTK_FILE_SYSTEM (data);
333 GtkFileSystemPrivate *priv = file_system->priv;
337 case G_FILE_MONITOR_EVENT_CHANGED:
338 case G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT:
339 case G_FILE_MONITOR_EVENT_CREATED:
340 case G_FILE_MONITOR_EVENT_DELETED:
341 g_slist_foreach (priv->bookmarks, (GFunc) _gtk_file_system_bookmark_free, NULL);
342 g_slist_free (priv->bookmarks);
344 priv->bookmarks = read_bookmarks (file);
346 gdk_threads_enter ();
347 g_signal_emit (data, fs_signals[BOOKMARKS_CHANGED], 0);
348 gdk_threads_leave ();
351 /* ignore at the moment */
357 mount_referenced_by_volume_activation_root (GList *volumes, GMount *mount)
365 mount_root = g_mount_get_root (mount);
367 for (l = volumes; l != NULL; l = l->next)
369 GVolume *volume = G_VOLUME (l->data);
370 GFile *volume_activation_root;
372 volume_activation_root = g_volume_get_activation_root (volume);
373 if (volume_activation_root != NULL)
375 if (g_file_has_prefix (volume_activation_root, mount_root))
378 g_object_unref (volume_activation_root);
381 g_object_unref (volume_activation_root);
385 g_object_unref (mount_root);
390 get_volumes_list (GtkFileSystem *file_system)
392 GtkFileSystemPrivate *priv = file_system->priv;
403 g_slist_foreach (priv->volumes, (GFunc) g_object_unref, NULL);
404 g_slist_free (priv->volumes);
405 priv->volumes = NULL;
408 /* first go through all connected drives */
409 drives = g_volume_monitor_get_connected_drives (priv->volume_monitor);
411 for (l = drives; l != NULL; l = l->next)
414 volumes = g_drive_get_volumes (drive);
418 for (ll = volumes; ll != NULL; ll = ll->next)
421 mount = g_volume_get_mount (volume);
425 /* Show mounted volume */
426 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
427 g_object_unref (mount);
431 /* Do show the unmounted volumes in the sidebar;
432 * this is so the user can mount it (in case automounting
435 * Also, even if automounting is enabled, this gives a visual
436 * cue that the user should remember to yank out the media if
437 * he just unmounted it.
439 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
442 g_object_unref (volume);
445 g_list_free (volumes);
447 else if (g_drive_is_media_removable (drive) && !g_drive_is_media_check_automatic (drive))
449 /* If the drive has no mountable volumes and we cannot detect media change.. we
450 * display the drive in the sidebar so the user can manually poll the drive by
451 * right clicking and selecting "Rescan..."
453 * This is mainly for drives like floppies where media detection doesn't
454 * work.. but it's also for human beings who like to turn off media detection
455 * in the OS to save battery juice.
458 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (drive));
461 g_object_unref (drive);
464 g_list_free (drives);
466 /* add all volumes that is not associated with a drive */
467 volumes = g_volume_monitor_get_volumes (priv->volume_monitor);
469 for (l = volumes; l != NULL; l = l->next)
472 drive = g_volume_get_drive (volume);
476 g_object_unref (drive);
480 mount = g_volume_get_mount (volume);
484 /* show this mount */
485 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
486 g_object_unref (mount);
490 /* see comment above in why we add an icon for a volume */
491 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
494 g_object_unref (volume);
497 /* add mounts that has no volume (/etc/mtab mounts, ftp, sftp,...) */
498 mounts = g_volume_monitor_get_mounts (priv->volume_monitor);
500 for (l = mounts; l != NULL; l = l->next)
503 volume = g_mount_get_volume (mount);
507 g_object_unref (volume);
511 /* if there's exists one or more volumes with an activation root inside the mount,
512 * don't display the mount
514 if (mount_referenced_by_volume_activation_root (volumes, mount))
516 g_object_unref (mount);
520 /* show this mount */
521 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
522 g_object_unref (mount);
525 g_list_free (volumes);
527 g_list_free (mounts);
531 _gtk_file_system_init (GtkFileSystem *file_system)
533 GtkFileSystemPrivate *priv;
534 GFile *bookmarks_file;
535 GError *error = NULL;
539 file_system->priv = G_TYPE_INSTANCE_GET_PRIVATE (file_system,
540 GTK_TYPE_FILE_SYSTEM,
541 GtkFileSystemPrivate);
542 priv = file_system->priv;
545 priv->volume_monitor = g_volume_monitor_get ();
547 g_signal_connect (priv->volume_monitor, "mount-added",
548 G_CALLBACK (volumes_changed), file_system);
549 g_signal_connect (priv->volume_monitor, "mount-removed",
550 G_CALLBACK (volumes_changed), file_system);
551 g_signal_connect (priv->volume_monitor, "mount-changed",
552 G_CALLBACK (volumes_changed), file_system);
553 g_signal_connect (priv->volume_monitor, "volume-added",
554 G_CALLBACK (volumes_changed), file_system);
555 g_signal_connect (priv->volume_monitor, "volume-removed",
556 G_CALLBACK (volumes_changed), file_system);
557 g_signal_connect (priv->volume_monitor, "volume-changed",
558 G_CALLBACK (volumes_changed), file_system);
559 g_signal_connect (priv->volume_monitor, "drive-connected",
560 G_CALLBACK (volumes_changed), file_system);
561 g_signal_connect (priv->volume_monitor, "drive-disconnected",
562 G_CALLBACK (volumes_changed), file_system);
563 g_signal_connect (priv->volume_monitor, "drive-changed",
564 G_CALLBACK (volumes_changed), file_system);
567 bookmarks_file = get_bookmarks_file ();
568 priv->bookmarks = read_bookmarks (bookmarks_file);
569 priv->bookmarks_monitor = g_file_monitor_file (bookmarks_file,
574 g_warning ("%s", error->message);
575 g_error_free (error);
578 g_signal_connect (priv->bookmarks_monitor, "changed",
579 G_CALLBACK (bookmarks_file_changed), file_system);
581 g_object_unref (bookmarks_file);
584 /* GtkFileSystem public methods */
586 _gtk_file_system_new (void)
588 return g_object_new (GTK_TYPE_FILE_SYSTEM, NULL);
592 _gtk_file_system_list_volumes (GtkFileSystem *file_system)
594 GtkFileSystemPrivate *priv = file_system->priv;
597 DEBUG ("list_volumes");
599 get_volumes_list (file_system);
601 list = g_slist_copy (priv->volumes);
604 /* Prepend root volume */
605 list = g_slist_prepend (list, (gpointer) root_volume_token);
612 _gtk_file_system_list_bookmarks (GtkFileSystem *file_system)
614 GtkFileSystemPrivate *priv = file_system->priv;
615 GSList *bookmarks, *files = NULL;
617 DEBUG ("list_bookmarks");
619 bookmarks = priv->bookmarks;
623 GtkFileSystemBookmark *bookmark;
625 bookmark = bookmarks->data;
626 bookmarks = bookmarks->next;
628 files = g_slist_prepend (files, g_object_ref (bookmark->file));
631 return g_slist_reverse (files);
635 is_valid_scheme_character (char c)
637 return g_ascii_isalnum (c) || c == '+' || c == '-' || c == '.';
641 has_uri_scheme (const char *str)
647 if (!is_valid_scheme_character (*p))
652 while (is_valid_scheme_character (*p));
654 return (strncmp (p, "://", 3) == 0);
658 _gtk_file_system_parse (GtkFileSystem *file_system,
666 gboolean result = FALSE;
667 gboolean is_dir = FALSE;
668 gchar *last_slash = NULL;
674 is_dir = (str [strlen (str) - 1] == G_DIR_SEPARATOR);
676 last_slash = strrchr (str, G_DIR_SEPARATOR);
678 is_uri = has_uri_scheme (str);
683 const char *slash_after_hostname;
685 colon = strchr (str, ':');
686 g_assert (colon != NULL);
687 g_assert (strncmp (colon, "://", 3) == 0);
689 slash_after_hostname = strchr (colon + 3, '/');
691 if (slash_after_hostname == NULL)
693 /* We don't have a full hostname yet. So, don't switch the folder
694 * until we have seen a full hostname. Otherwise, completion will
695 * happen for every character the user types for the hostname.
701 GTK_FILE_CHOOSER_ERROR,
702 GTK_FILE_CHOOSER_ERROR_INCOMPLETE_HOSTNAME,
703 "Incomplete hostname");
708 if (str[0] == '~' || g_path_is_absolute (str) || is_uri)
709 file = g_file_parse_name (str);
713 file = g_file_resolve_relative_path (base_file, str);
719 GTK_FILE_CHOOSER_ERROR,
720 GTK_FILE_CHOOSER_ERROR_BAD_FILENAME,
726 if (base_file && g_file_equal (base_file, file))
728 /* this is when user types '.', could be the
729 * beginning of a hidden file, ./ or ../
731 *folder = g_object_ref (file);
732 *file_part = g_strdup (str);
737 /* it's a dir, or at least it ends with the dir separator */
738 *folder = g_object_ref (file);
739 *file_part = g_strdup ("");
746 parent_file = g_file_get_parent (file);
751 GTK_FILE_CHOOSER_ERROR,
752 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
753 "Could not get parent file");
759 *folder = parent_file;
763 *file_part = g_strdup (last_slash + 1);
765 *file_part = g_strdup (str);
769 g_object_unref (file);
775 free_async_data (AsyncFuncData *async_data)
777 g_object_unref (async_data->file_system);
778 g_object_unref (async_data->file);
779 g_object_unref (async_data->cancellable);
781 if (async_data->folder)
782 g_object_unref (async_data->folder);
784 g_free (async_data->attributes);
789 enumerate_children_callback (GObject *source_object,
790 GAsyncResult *result,
793 GFileEnumerator *enumerator;
794 AsyncFuncData *async_data;
795 GtkFolder *folder = NULL;
797 GError *error = NULL;
799 file = G_FILE (source_object);
800 async_data = (AsyncFuncData *) user_data;
801 enumerator = g_file_enumerate_children_finish (file, result, &error);
805 folder = g_object_new (GTK_TYPE_FOLDER,
806 "file", source_object,
807 "enumerator", enumerator,
808 "attributes", async_data->attributes,
810 g_object_unref (enumerator);
813 gdk_threads_enter ();
814 ((GtkFileSystemGetFolderCallback) async_data->callback) (async_data->cancellable,
815 folder, error, async_data->data);
816 gdk_threads_leave ();
818 free_async_data (async_data);
821 g_error_free (error);
825 _gtk_file_system_get_folder (GtkFileSystem *file_system,
827 const gchar *attributes,
828 GtkFileSystemGetFolderCallback callback,
831 GCancellable *cancellable;
832 AsyncFuncData *async_data;
834 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
835 g_return_val_if_fail (G_IS_FILE (file), NULL);
837 cancellable = g_cancellable_new ();
839 async_data = g_new0 (AsyncFuncData, 1);
840 async_data->file_system = g_object_ref (file_system);
841 async_data->file = g_object_ref (file);
842 async_data->cancellable = g_object_ref (cancellable);
843 async_data->attributes = g_strdup (attributes);
845 async_data->callback = callback;
846 async_data->data = data;
848 g_file_enumerate_children_async (file,
850 G_FILE_QUERY_INFO_NONE,
853 enumerate_children_callback,
859 query_info_callback (GObject *source_object,
860 GAsyncResult *result,
863 AsyncFuncData *async_data;
864 GError *error = NULL;
865 GFileInfo *file_info;
868 DEBUG ("query_info_callback");
870 file = G_FILE (source_object);
871 async_data = (AsyncFuncData *) user_data;
872 file_info = g_file_query_info_finish (file, result, &error);
874 if (async_data->callback)
876 gdk_threads_enter ();
877 ((GtkFileSystemGetInfoCallback) async_data->callback) (async_data->cancellable,
878 file_info, error, async_data->data);
879 gdk_threads_leave ();
883 g_object_unref (file_info);
886 g_error_free (error);
888 free_async_data (async_data);
892 _gtk_file_system_get_info (GtkFileSystem *file_system,
894 const gchar *attributes,
895 GtkFileSystemGetInfoCallback callback,
898 GCancellable *cancellable;
899 AsyncFuncData *async_data;
901 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
902 g_return_val_if_fail (G_IS_FILE (file), NULL);
904 cancellable = g_cancellable_new ();
906 async_data = g_new0 (AsyncFuncData, 1);
907 async_data->file_system = g_object_ref (file_system);
908 async_data->file = g_object_ref (file);
909 async_data->cancellable = g_object_ref (cancellable);
911 async_data->callback = callback;
912 async_data->data = data;
914 g_file_query_info_async (file,
916 G_FILE_QUERY_INFO_NONE,
926 drive_poll_for_media_cb (GObject *source_object,
927 GAsyncResult *result,
930 AsyncFuncData *async_data;
931 GError *error = NULL;
933 g_drive_poll_for_media_finish (G_DRIVE (source_object), result, &error);
934 async_data = (AsyncFuncData *) user_data;
936 gdk_threads_enter ();
937 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
938 (GtkFileSystemVolume *) source_object,
939 error, async_data->data);
940 gdk_threads_leave ();
943 g_error_free (error);
947 volume_mount_cb (GObject *source_object,
948 GAsyncResult *result,
951 AsyncFuncData *async_data;
952 GError *error = NULL;
954 g_volume_mount_finish (G_VOLUME (source_object), result, &error);
955 async_data = (AsyncFuncData *) user_data;
957 gdk_threads_enter ();
958 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
959 (GtkFileSystemVolume *) source_object,
960 error, async_data->data);
961 gdk_threads_leave ();
964 g_error_free (error);
968 _gtk_file_system_mount_volume (GtkFileSystem *file_system,
969 GtkFileSystemVolume *volume,
970 GMountOperation *mount_operation,
971 GtkFileSystemVolumeMountCallback callback,
974 GCancellable *cancellable;
975 AsyncFuncData *async_data;
976 gboolean handled = FALSE;
978 DEBUG ("volume_mount");
980 cancellable = g_cancellable_new ();
982 async_data = g_new0 (AsyncFuncData, 1);
983 async_data->file_system = g_object_ref (file_system);
984 async_data->cancellable = g_object_ref (cancellable);
986 async_data->callback = callback;
987 async_data->data = data;
989 if (G_IS_DRIVE (volume))
991 /* this path happens for drives that are not polled by the OS and where the last media
992 * check indicated that no media was available. So the thing to do here is to
993 * invoke poll_for_media() on the drive
995 g_drive_poll_for_media (G_DRIVE (volume), cancellable, drive_poll_for_media_cb, async_data);
998 else if (G_IS_VOLUME (volume))
1000 g_volume_mount (G_VOLUME (volume), G_MOUNT_MOUNT_NONE, mount_operation, cancellable, volume_mount_cb, async_data);
1005 free_async_data (async_data);
1011 enclosing_volume_mount_cb (GObject *source_object,
1012 GAsyncResult *result,
1015 GtkFileSystemVolume *volume;
1016 AsyncFuncData *async_data;
1017 GError *error = NULL;
1019 async_data = (AsyncFuncData *) user_data;
1020 g_file_mount_enclosing_volume_finish (G_FILE (source_object), result, &error);
1021 volume = _gtk_file_system_get_volume_for_file (async_data->file_system, G_FILE (source_object));
1023 /* Silently drop G_IO_ERROR_ALREADY_MOUNTED error for gvfs backends without visible mounts. */
1024 /* Better than doing query_info with additional I/O every time. */
1025 if (error && g_error_matches (error, G_IO_ERROR, G_IO_ERROR_ALREADY_MOUNTED))
1026 g_clear_error (&error);
1028 gdk_threads_enter ();
1029 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable, volume,
1030 error, async_data->data);
1031 gdk_threads_leave ();
1034 g_error_free (error);
1036 _gtk_file_system_volume_unref (volume);
1040 _gtk_file_system_mount_enclosing_volume (GtkFileSystem *file_system,
1042 GMountOperation *mount_operation,
1043 GtkFileSystemVolumeMountCallback callback,
1046 GCancellable *cancellable;
1047 AsyncFuncData *async_data;
1049 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
1050 g_return_val_if_fail (G_IS_FILE (file), NULL);
1052 DEBUG ("mount_enclosing_volume");
1054 cancellable = g_cancellable_new ();
1056 async_data = g_new0 (AsyncFuncData, 1);
1057 async_data->file_system = g_object_ref (file_system);
1058 async_data->file = g_object_ref (file);
1059 async_data->cancellable = g_object_ref (cancellable);
1061 async_data->callback = callback;
1062 async_data->data = data;
1064 g_file_mount_enclosing_volume (file,
1068 enclosing_volume_mount_cb,
1074 _gtk_file_system_insert_bookmark (GtkFileSystem *file_system,
1079 GtkFileSystemPrivate *priv = file_system->priv;
1081 GtkFileSystemBookmark *bookmark;
1082 gboolean result = TRUE;
1083 GFile *bookmarks_file;
1085 bookmarks = priv->bookmarks;
1089 bookmark = bookmarks->data;
1090 bookmarks = bookmarks->next;
1092 if (g_file_equal (bookmark->file, file))
1094 /* File is already in bookmarks */
1102 gchar *uri = g_file_get_uri (file);
1105 GTK_FILE_CHOOSER_ERROR,
1106 GTK_FILE_CHOOSER_ERROR_ALREADY_EXISTS,
1107 "%s already exists in the bookmarks list",
1115 bookmark = g_slice_new0 (GtkFileSystemBookmark);
1116 bookmark->file = g_object_ref (file);
1118 priv->bookmarks = g_slist_insert (priv->bookmarks, bookmark, position);
1120 bookmarks_file = get_bookmarks_file ();
1121 save_bookmarks (bookmarks_file, priv->bookmarks);
1122 g_object_unref (bookmarks_file);
1124 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1130 _gtk_file_system_remove_bookmark (GtkFileSystem *file_system,
1134 GtkFileSystemPrivate *priv = file_system->priv;
1135 GtkFileSystemBookmark *bookmark;
1137 gboolean result = FALSE;
1138 GFile *bookmarks_file;
1140 if (!priv->bookmarks)
1143 bookmarks = priv->bookmarks;
1147 bookmark = bookmarks->data;
1149 if (g_file_equal (bookmark->file, file))
1152 priv->bookmarks = g_slist_remove_link (priv->bookmarks, bookmarks);
1153 _gtk_file_system_bookmark_free (bookmark);
1154 g_slist_free_1 (bookmarks);
1158 bookmarks = bookmarks->next;
1163 gchar *uri = g_file_get_uri (file);
1166 GTK_FILE_CHOOSER_ERROR,
1167 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
1168 "%s does not exist in the bookmarks list",
1176 bookmarks_file = get_bookmarks_file ();
1177 save_bookmarks (bookmarks_file, priv->bookmarks);
1178 g_object_unref (bookmarks_file);
1180 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1186 _gtk_file_system_get_bookmark_label (GtkFileSystem *file_system,
1189 GtkFileSystemPrivate *priv = file_system->priv;
1191 gchar *label = NULL;
1193 DEBUG ("get_bookmark_label");
1195 bookmarks = priv->bookmarks;
1199 GtkFileSystemBookmark *bookmark;
1201 bookmark = bookmarks->data;
1202 bookmarks = bookmarks->next;
1204 if (g_file_equal (file, bookmark->file))
1206 label = g_strdup (bookmark->label);
1215 _gtk_file_system_set_bookmark_label (GtkFileSystem *file_system,
1219 GtkFileSystemPrivate *priv = file_system->priv;
1220 gboolean changed = FALSE;
1221 GFile *bookmarks_file;
1224 DEBUG ("set_bookmark_label");
1226 bookmarks = priv->bookmarks;
1230 GtkFileSystemBookmark *bookmark;
1232 bookmark = bookmarks->data;
1233 bookmarks = bookmarks->next;
1235 if (g_file_equal (file, bookmark->file))
1237 g_free (bookmark->label);
1238 bookmark->label = g_strdup (label);
1244 bookmarks_file = get_bookmarks_file ();
1245 save_bookmarks (bookmarks_file, priv->bookmarks);
1246 g_object_unref (bookmarks_file);
1249 g_signal_emit_by_name (file_system, "bookmarks-changed", 0);
1252 GtkFileSystemVolume *
1253 _gtk_file_system_get_volume_for_file (GtkFileSystem *file_system,
1258 DEBUG ("get_volume_for_file");
1260 mount = g_file_find_enclosing_mount (file, NULL, NULL);
1262 if (!mount && g_file_is_native (file))
1263 return (GtkFileSystemVolume *) root_volume_token;
1265 return (GtkFileSystemVolume *) mount;
1268 /* GtkFolder methods */
1270 gtk_folder_set_property (GObject *object,
1272 const GValue *value,
1275 GtkFolder *folder = GTK_FOLDER (object);
1276 GtkFolderPrivate *priv = folder->priv;
1281 priv->folder_file = g_value_dup_object (value);
1283 case PROP_ENUMERATOR:
1284 priv->enumerator = g_value_dup_object (value);
1286 case PROP_ATTRIBUTES:
1287 priv->attributes = g_value_dup_string (value);
1290 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1296 gtk_folder_get_property (GObject *object,
1301 GtkFolder *folder = GTK_FOLDER (object);
1302 GtkFolderPrivate *priv = folder->priv;
1307 g_value_set_object (value, priv->folder_file);
1309 case PROP_ENUMERATOR:
1310 g_value_set_object (value, priv->enumerator);
1312 case PROP_ATTRIBUTES:
1313 g_value_set_string (value, priv->attributes);
1316 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1322 query_created_file_info_callback (GObject *source_object,
1323 GAsyncResult *result,
1326 GFile *file = G_FILE (source_object);
1327 GError *error = NULL;
1332 info = g_file_query_info_finish (file, result, &error);
1336 g_error_free (error);
1340 gdk_threads_enter ();
1342 folder = GTK_FOLDER (user_data);
1343 gtk_folder_add_file (folder, file, info);
1345 files = g_slist_prepend (NULL, file);
1346 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1347 g_slist_free (files);
1349 g_object_unref (info);
1350 gdk_threads_leave ();
1354 directory_monitor_changed (GFileMonitor *monitor,
1357 GFileMonitorEvent event,
1360 GtkFolder *folder = GTK_FOLDER (data);
1361 GtkFolderPrivate *priv = folder->priv;
1364 files = g_slist_prepend (NULL, file);
1366 gdk_threads_enter ();
1370 case G_FILE_MONITOR_EVENT_CREATED:
1371 g_file_query_info_async (file,
1373 G_FILE_QUERY_INFO_NONE,
1376 query_created_file_info_callback,
1379 case G_FILE_MONITOR_EVENT_DELETED:
1380 if (g_file_equal (file, priv->folder_file))
1381 g_signal_emit (folder, folder_signals[DELETED], 0);
1383 g_signal_emit (folder, folder_signals[FILES_REMOVED], 0, files);
1389 gdk_threads_leave ();
1391 g_slist_free (files);
1395 enumerator_files_callback (GObject *source_object,
1396 GAsyncResult *result,
1399 GtkFolder *folder = GTK_FOLDER (user_data);
1400 GtkFolderPrivate *priv = folder->priv;
1401 GFileEnumerator *enumerator;
1402 GError *error = NULL;
1403 GSList *files = NULL;
1404 GList *file_infos, *f;
1406 enumerator = G_FILE_ENUMERATOR (source_object);
1407 file_infos = g_file_enumerator_next_files_finish (enumerator, result, &error);
1411 if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
1412 g_warning ("%s", error->message);
1414 g_error_free (error);
1420 g_file_enumerator_close_async (enumerator,
1424 gtk_folder_set_finished_loading (folder, TRUE);
1428 g_file_enumerator_next_files_async (enumerator, FILES_PER_QUERY,
1431 enumerator_files_callback,
1434 for (f = file_infos; f; f = f->next)
1440 child_file = g_file_get_child (priv->folder_file, g_file_info_get_name (info));
1441 gtk_folder_add_file (folder, child_file, info);
1442 files = g_slist_prepend (files, child_file);
1445 gdk_threads_enter ();
1446 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1447 gdk_threads_leave ();
1449 g_list_foreach (file_infos, (GFunc) g_object_unref, NULL);
1450 g_list_free (file_infos);
1452 g_slist_foreach (files, (GFunc) g_object_unref, NULL);
1453 g_slist_free (files);
1457 gtk_folder_constructed (GObject *object)
1459 GtkFolder *folder = GTK_FOLDER (object);
1460 GtkFolderPrivate *priv = folder->priv;
1461 GError *error = NULL;
1463 priv->directory_monitor = g_file_monitor_directory (priv->folder_file, G_FILE_MONITOR_NONE, NULL, &error);
1467 g_warning ("%s", error->message);
1468 g_error_free (error);
1471 g_signal_connect (priv->directory_monitor, "changed",
1472 G_CALLBACK (directory_monitor_changed), object);
1474 g_file_enumerator_next_files_async (priv->enumerator,
1478 enumerator_files_callback,
1480 /* This isn't needed anymore */
1481 g_object_unref (priv->enumerator);
1482 priv->enumerator = NULL;
1486 gtk_folder_finalize (GObject *object)
1488 GtkFolder *folder = GTK_FOLDER (object);
1489 GtkFolderPrivate *priv = folder->priv;
1491 g_hash_table_unref (priv->children);
1493 if (priv->folder_file)
1494 g_object_unref (priv->folder_file);
1496 if (priv->directory_monitor)
1497 g_object_unref (priv->directory_monitor);
1499 g_cancellable_cancel (priv->cancellable);
1500 g_object_unref (priv->cancellable);
1501 g_free (priv->attributes);
1503 G_OBJECT_CLASS (_gtk_folder_parent_class)->finalize (object);
1507 _gtk_folder_class_init (GtkFolderClass *class)
1509 GObjectClass *object_class = G_OBJECT_CLASS (class);
1511 object_class->set_property = gtk_folder_set_property;
1512 object_class->get_property = gtk_folder_get_property;
1513 object_class->constructed = gtk_folder_constructed;
1514 object_class->finalize = gtk_folder_finalize;
1516 g_object_class_install_property (object_class,
1518 g_param_spec_object ("file",
1520 "GFile for the folder",
1522 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1523 g_object_class_install_property (object_class,
1525 g_param_spec_object ("enumerator",
1527 "GFileEnumerator to list files",
1528 G_TYPE_FILE_ENUMERATOR,
1529 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1530 g_object_class_install_property (object_class,
1532 g_param_spec_string ("attributes",
1534 "Attributes to query for",
1536 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1537 folder_signals[FILES_ADDED] =
1538 g_signal_new ("files-added",
1539 G_TYPE_FROM_CLASS (object_class),
1541 G_STRUCT_OFFSET (GtkFolderClass, files_added),
1543 g_cclosure_marshal_VOID__POINTER,
1544 G_TYPE_NONE, 1, G_TYPE_POINTER);
1545 folder_signals[FILES_REMOVED] =
1546 g_signal_new ("files-removed",
1547 G_TYPE_FROM_CLASS (object_class),
1549 G_STRUCT_OFFSET (GtkFolderClass, files_removed),
1551 g_cclosure_marshal_VOID__POINTER,
1552 G_TYPE_NONE, 1, G_TYPE_POINTER);
1553 folder_signals[FILES_CHANGED] =
1554 g_signal_new ("files-changed",
1555 G_TYPE_FROM_CLASS (object_class),
1557 G_STRUCT_OFFSET (GtkFolderClass, files_changed),
1559 g_cclosure_marshal_VOID__POINTER,
1560 G_TYPE_NONE, 1, G_TYPE_POINTER);
1561 folder_signals[FINISHED_LOADING] =
1562 g_signal_new ("finished-loading",
1563 G_TYPE_FROM_CLASS (object_class),
1565 G_STRUCT_OFFSET (GtkFolderClass, finished_loading),
1567 g_cclosure_marshal_VOID__VOID,
1569 folder_signals[DELETED] =
1570 g_signal_new ("deleted",
1571 G_TYPE_FROM_CLASS (object_class),
1573 G_STRUCT_OFFSET (GtkFolderClass, deleted),
1575 g_cclosure_marshal_VOID__VOID,
1578 g_type_class_add_private (object_class, sizeof (GtkFolderPrivate));
1582 _gtk_folder_init (GtkFolder *folder)
1584 GtkFolderPrivate *priv;
1586 folder->priv = G_TYPE_INSTANCE_GET_PRIVATE (folder,
1589 priv = folder->priv;
1590 priv->children = g_hash_table_new_full (g_file_hash,
1591 (GEqualFunc) g_file_equal,
1592 (GDestroyNotify) g_object_unref,
1593 (GDestroyNotify) g_object_unref);
1594 priv->cancellable = g_cancellable_new ();
1598 gtk_folder_set_finished_loading (GtkFolder *folder,
1599 gboolean finished_loading)
1601 GtkFolderPrivate *priv = folder->priv;
1603 priv->finished_loading = (finished_loading == TRUE);
1605 gdk_threads_enter ();
1606 g_signal_emit (folder, folder_signals[FINISHED_LOADING], 0);
1607 gdk_threads_leave ();
1611 gtk_folder_add_file (GtkFolder *folder,
1615 GtkFolderPrivate *priv = folder->priv;
1617 g_hash_table_insert (priv->children,
1618 g_object_ref (file),
1619 g_object_ref (info));
1623 _gtk_folder_list_children (GtkFolder *folder)
1625 GtkFolderPrivate *priv = folder->priv;
1626 GList *files, *elem;
1627 GSList *children = NULL;
1629 files = g_hash_table_get_keys (priv->children);
1632 for (elem = files; elem; elem = elem->next)
1633 children = g_slist_prepend (children, g_object_ref (elem->data));
1635 g_list_free (files);
1641 _gtk_folder_get_info (GtkFolder *folder,
1644 GtkFolderPrivate *priv = folder->priv;
1647 info = g_hash_table_lookup (priv->children, file);
1652 return g_object_ref (info);
1656 _gtk_folder_is_finished_loading (GtkFolder *folder)
1658 return folder->priv->finished_loading;
1661 /* GtkFileSystemVolume public methods */
1663 _gtk_file_system_volume_get_display_name (GtkFileSystemVolume *volume)
1665 DEBUG ("volume_get_display_name");
1667 if (IS_ROOT_VOLUME (volume))
1668 return g_strdup (_(root_volume_token));
1669 if (G_IS_DRIVE (volume))
1670 return g_drive_get_name (G_DRIVE (volume));
1671 else if (G_IS_MOUNT (volume))
1672 return g_mount_get_name (G_MOUNT (volume));
1673 else if (G_IS_VOLUME (volume))
1674 return g_volume_get_name (G_VOLUME (volume));
1680 _gtk_file_system_volume_is_mounted (GtkFileSystemVolume *volume)
1684 DEBUG ("volume_is_mounted");
1686 if (IS_ROOT_VOLUME (volume))
1691 if (G_IS_MOUNT (volume))
1693 else if (G_IS_VOLUME (volume))
1697 mount = g_volume_get_mount (G_VOLUME (volume));
1702 g_object_unref (mount);
1710 _gtk_file_system_volume_get_root (GtkFileSystemVolume *volume)
1714 DEBUG ("volume_get_base");
1716 if (IS_ROOT_VOLUME (volume))
1717 return g_file_new_for_uri ("file:///");
1719 if (G_IS_MOUNT (volume))
1720 file = g_mount_get_root (G_MOUNT (volume));
1721 else if (G_IS_VOLUME (volume))
1725 mount = g_volume_get_mount (G_VOLUME (volume));
1729 file = g_mount_get_root (mount);
1730 g_object_unref (mount);
1738 get_pixbuf_from_gicon (GIcon *icon,
1744 GtkIconTheme *icon_theme;
1745 GtkIconInfo *icon_info;
1748 screen = gtk_widget_get_screen (GTK_WIDGET (widget));
1749 icon_theme = gtk_icon_theme_get_for_screen (screen);
1751 icon_info = gtk_icon_theme_lookup_by_gicon (icon_theme,
1754 GTK_ICON_LOOKUP_USE_BUILTIN);
1759 pixbuf = gtk_icon_info_load_icon (icon_info, error);
1760 gtk_icon_info_free (icon_info);
1766 _gtk_file_system_volume_render_icon (GtkFileSystemVolume *volume,
1774 DEBUG ("volume_get_icon_name");
1776 if (IS_ROOT_VOLUME (volume))
1777 icon = g_themed_icon_new ("drive-harddisk");
1778 else if (G_IS_DRIVE (volume))
1779 icon = g_drive_get_icon (G_DRIVE (volume));
1780 else if (G_IS_VOLUME (volume))
1781 icon = g_volume_get_icon (G_VOLUME (volume));
1782 else if (G_IS_MOUNT (volume))
1783 icon = g_mount_get_icon (G_MOUNT (volume));
1788 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, error);
1790 g_object_unref (icon);
1795 GtkFileSystemVolume *
1796 _gtk_file_system_volume_ref (GtkFileSystemVolume *volume)
1798 if (IS_ROOT_VOLUME (volume))
1801 if (G_IS_MOUNT (volume) ||
1802 G_IS_VOLUME (volume) ||
1803 G_IS_DRIVE (volume))
1804 g_object_ref (volume);
1810 _gtk_file_system_volume_unref (GtkFileSystemVolume *volume)
1812 /* Root volume doesn't need to be freed */
1813 if (IS_ROOT_VOLUME (volume))
1816 if (G_IS_MOUNT (volume) ||
1817 G_IS_VOLUME (volume) ||
1818 G_IS_DRIVE (volume))
1819 g_object_unref (volume);
1822 /* GFileInfo helper functions */
1824 _gtk_file_info_render_icon (GFileInfo *info,
1829 GdkPixbuf *pixbuf = NULL;
1830 const gchar *thumbnail_path;
1832 thumbnail_path = g_file_info_get_attribute_byte_string (info, G_FILE_ATTRIBUTE_THUMBNAIL_PATH);
1835 pixbuf = gdk_pixbuf_new_from_file_at_size (thumbnail_path,
1836 icon_size, icon_size,
1841 icon = g_file_info_get_icon (info);
1844 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1848 /* Use general fallback for all files without icon */
1849 icon = g_themed_icon_new ("text-x-generic");
1850 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1851 g_object_unref (icon);
1859 _gtk_file_info_consider_as_directory (GFileInfo *info)
1861 GFileType type = g_file_info_get_file_type (info);
1863 return (type == G_FILE_TYPE_DIRECTORY ||
1864 type == G_FILE_TYPE_MOUNTABLE ||
1865 type == G_FILE_TYPE_SHORTCUT);