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"
36 /* #define DEBUG_MODE */
38 #define DEBUG(x) g_debug (x);
43 #define GTK_FILE_SYSTEM_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GTK_TYPE_FILE_SYSTEM, GtkFileSystemPrivate))
44 #define GTK_FOLDER_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GTK_TYPE_FOLDER, GtkFolderPrivate))
45 #define FILES_PER_QUERY 100
47 /* The pointers we return for a GtkFileSystemVolume are opaque tokens; they are
48 * really pointers to GDrive, GVolume or GMount objects. We need an extra
49 * token for the fake "File System" volume. So, we'll return a pointer to
50 * this particular string.
52 static const gchar *root_volume_token = N_("File System");
53 #define IS_ROOT_VOLUME(volume) ((gpointer) (volume) == (gpointer) root_volume_token)
77 static guint fs_signals [FS_LAST_SIGNAL] = { 0, };
78 static guint folder_signals [FOLDER_LAST_SIGNAL] = { 0, };
80 typedef struct GtkFileSystemPrivate GtkFileSystemPrivate;
81 typedef struct GtkFolderPrivate GtkFolderPrivate;
82 typedef struct AsyncFuncData AsyncFuncData;
84 struct GtkFileSystemPrivate
86 GVolumeMonitor *volume_monitor;
88 /* This list contains elements that can be
89 * of type GDrive, GVolume and GMount
93 /* This list contains GtkFileSystemBookmark structs */
96 GFileMonitor *bookmarks_monitor;
99 struct GtkFolderPrivate
102 GHashTable *children;
103 GFileMonitor *directory_monitor;
104 GFileEnumerator *enumerator;
105 GCancellable *cancellable;
108 guint finished_loading : 1;
113 GtkFileSystem *file_system;
116 GCancellable *cancellable;
123 struct GtkFileSystemBookmark
129 G_DEFINE_TYPE (GtkFileSystem, _gtk_file_system, G_TYPE_OBJECT)
131 G_DEFINE_TYPE (GtkFolder, _gtk_folder, G_TYPE_OBJECT)
134 static void gtk_folder_set_finished_loading (GtkFolder *folder,
135 gboolean finished_loading);
136 static void gtk_folder_add_file (GtkFolder *folder,
141 /* GtkFileSystemBookmark methods */
143 _gtk_file_system_bookmark_free (GtkFileSystemBookmark *bookmark)
145 g_object_unref (bookmark->file);
146 g_free (bookmark->label);
147 g_slice_free (GtkFileSystemBookmark, bookmark);
150 /* GtkFileSystem methods */
152 volumes_changed (GVolumeMonitor *volume_monitor,
156 GtkFileSystem *file_system;
158 gdk_threads_enter ();
160 file_system = GTK_FILE_SYSTEM (user_data);
161 g_signal_emit (file_system, fs_signals[VOLUMES_CHANGED], 0, volume);
162 gdk_threads_leave ();
166 gtk_file_system_dispose (GObject *object)
168 GtkFileSystemPrivate *priv;
172 priv = GTK_FILE_SYSTEM_GET_PRIVATE (object);
176 g_slist_foreach (priv->volumes, (GFunc) g_object_unref, NULL);
177 g_slist_free (priv->volumes);
178 priv->volumes = NULL;
181 if (priv->volume_monitor)
183 g_signal_handlers_disconnect_by_func (priv->volume_monitor, volumes_changed, object);
184 g_object_unref (priv->volume_monitor);
185 priv->volume_monitor = NULL;
188 G_OBJECT_CLASS (_gtk_file_system_parent_class)->dispose (object);
192 gtk_file_system_finalize (GObject *object)
194 GtkFileSystemPrivate *priv;
198 priv = GTK_FILE_SYSTEM_GET_PRIVATE (object);
200 if (priv->bookmarks_monitor)
201 g_object_unref (priv->bookmarks_monitor);
205 g_slist_foreach (priv->bookmarks, (GFunc) _gtk_file_system_bookmark_free, NULL);
206 g_slist_free (priv->bookmarks);
209 G_OBJECT_CLASS (_gtk_file_system_parent_class)->finalize (object);
213 _gtk_file_system_class_init (GtkFileSystemClass *class)
215 GObjectClass *object_class = G_OBJECT_CLASS (class);
217 object_class->dispose = gtk_file_system_dispose;
218 object_class->finalize = gtk_file_system_finalize;
220 fs_signals[BOOKMARKS_CHANGED] =
221 g_signal_new ("bookmarks-changed",
222 G_TYPE_FROM_CLASS (object_class),
224 G_STRUCT_OFFSET (GtkFileSystemClass, bookmarks_changed),
226 g_cclosure_marshal_VOID__VOID,
229 fs_signals[VOLUMES_CHANGED] =
230 g_signal_new ("volumes-changed",
231 G_TYPE_FROM_CLASS (object_class),
233 G_STRUCT_OFFSET (GtkFileSystemClass, volumes_changed),
235 g_cclosure_marshal_VOID__VOID,
238 g_type_class_add_private (object_class, sizeof (GtkFileSystemPrivate));
242 get_bookmarks_file (void)
247 filename = g_build_filename (g_get_home_dir (), ".gtk-bookmarks", NULL);
248 file = g_file_new_for_path (filename);
255 read_bookmarks (GFile *file)
258 gchar **lines, *space;
259 GSList *bookmarks = NULL;
262 if (!g_file_load_contents (file, NULL, &contents,
266 lines = g_strsplit (contents, "\n", -1);
268 for (i = 0; lines[i]; i++)
270 GtkFileSystemBookmark *bookmark;
275 bookmark = g_slice_new0 (GtkFileSystemBookmark);
277 if ((space = strchr (lines[i], ' ')) != NULL)
280 bookmark->label = g_strdup (space + 1);
283 bookmark->file = g_file_new_for_uri (lines[i]);
284 bookmarks = g_slist_prepend (bookmarks, bookmark);
287 bookmarks = g_slist_reverse (bookmarks);
295 save_bookmarks (GFile *bookmarks_file,
298 GError *error = NULL;
301 contents = g_string_new ("");
305 GtkFileSystemBookmark *bookmark;
308 bookmark = bookmarks->data;
309 uri = g_file_get_uri (bookmark->file);
310 g_string_append (contents, uri);
313 g_string_append_printf (contents, " %s", bookmark->label);
315 g_string_append_c (contents, '\n');
316 bookmarks = bookmarks->next;
320 if (!g_file_replace_contents (bookmarks_file,
322 strlen (contents->str),
323 NULL, FALSE, 0, NULL,
326 g_critical ("%s", error->message);
327 g_error_free (error);
330 g_string_free (contents, TRUE);
334 bookmarks_file_changed (GFileMonitor *monitor,
337 GFileMonitorEvent event,
340 GtkFileSystemPrivate *priv;
342 priv = GTK_FILE_SYSTEM_GET_PRIVATE (data);
346 case G_FILE_MONITOR_EVENT_CHANGED:
347 case G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT:
348 case G_FILE_MONITOR_EVENT_CREATED:
349 case G_FILE_MONITOR_EVENT_DELETED:
350 g_slist_foreach (priv->bookmarks, (GFunc) _gtk_file_system_bookmark_free, NULL);
351 g_slist_free (priv->bookmarks);
353 priv->bookmarks = read_bookmarks (file);
355 gdk_threads_enter ();
356 g_signal_emit (data, fs_signals[BOOKMARKS_CHANGED], 0);
357 gdk_threads_leave ();
360 /* ignore at the moment */
366 mount_referenced_by_volume_activation_root (GList *volumes, GMount *mount)
374 mount_root = g_mount_get_root (mount);
376 for (l = volumes; l != NULL; l = l->next)
378 GVolume *volume = G_VOLUME (l->data);
379 GFile *volume_activation_root;
381 volume_activation_root = g_volume_get_activation_root (volume);
382 if (volume_activation_root != NULL)
384 if (g_file_has_prefix (volume_activation_root, mount_root))
387 g_object_unref (volume_activation_root);
390 g_object_unref (volume_activation_root);
394 g_object_unref (mount_root);
399 get_volumes_list (GtkFileSystem *file_system)
401 GtkFileSystemPrivate *priv;
410 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
414 g_slist_foreach (priv->volumes, (GFunc) g_object_unref, NULL);
415 g_slist_free (priv->volumes);
416 priv->volumes = NULL;
419 /* first go through all connected drives */
420 drives = g_volume_monitor_get_connected_drives (priv->volume_monitor);
422 for (l = drives; l != NULL; l = l->next)
425 volumes = g_drive_get_volumes (drive);
429 for (ll = volumes; ll != NULL; ll = ll->next)
432 mount = g_volume_get_mount (volume);
436 /* Show mounted volume */
437 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
438 g_object_unref (mount);
442 /* Do show the unmounted volumes in the sidebar;
443 * this is so the user can mount it (in case automounting
446 * Also, even if automounting is enabled, this gives a visual
447 * cue that the user should remember to yank out the media if
448 * he just unmounted it.
450 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
453 g_object_unref (volume);
456 g_list_free (volumes);
458 else if (g_drive_is_media_removable (drive) && !g_drive_is_media_check_automatic (drive))
460 /* If the drive has no mountable volumes and we cannot detect media change.. we
461 * display the drive in the sidebar so the user can manually poll the drive by
462 * right clicking and selecting "Rescan..."
464 * This is mainly for drives like floppies where media detection doesn't
465 * work.. but it's also for human beings who like to turn off media detection
466 * in the OS to save battery juice.
469 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (drive));
473 g_list_free (drives);
475 /* add all volumes that is not associated with a drive */
476 volumes = g_volume_monitor_get_volumes (priv->volume_monitor);
478 for (l = volumes; l != NULL; l = l->next)
481 drive = g_volume_get_drive (volume);
485 g_object_unref (drive);
489 mount = g_volume_get_mount (volume);
493 /* show this mount */
494 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
495 g_object_unref (mount);
499 /* see comment above in why we add an icon for a volume */
500 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
504 /* add mounts that has no volume (/etc/mtab mounts, ftp, sftp,...) */
505 mounts = g_volume_monitor_get_mounts (priv->volume_monitor);
507 for (l = mounts; l != NULL; l = l->next)
510 volume = g_mount_get_volume (mount);
514 g_object_unref (volume);
518 /* if there's exists one or more volumes with an activation root inside the mount,
519 * don't display the mount
521 if (mount_referenced_by_volume_activation_root (volumes, mount))
526 /* show this mount */
527 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
530 g_list_free (volumes);
532 g_list_free (mounts);
536 _gtk_file_system_init (GtkFileSystem *file_system)
538 GtkFileSystemPrivate *priv;
539 GFile *bookmarks_file;
540 GError *error = NULL;
544 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
547 priv->volume_monitor = g_volume_monitor_get ();
549 g_signal_connect (priv->volume_monitor, "mount-added",
550 G_CALLBACK (volumes_changed), file_system);
551 g_signal_connect (priv->volume_monitor, "mount-removed",
552 G_CALLBACK (volumes_changed), file_system);
553 g_signal_connect (priv->volume_monitor, "mount-changed",
554 G_CALLBACK (volumes_changed), file_system);
555 g_signal_connect (priv->volume_monitor, "volume-added",
556 G_CALLBACK (volumes_changed), file_system);
557 g_signal_connect (priv->volume_monitor, "volume-removed",
558 G_CALLBACK (volumes_changed), file_system);
559 g_signal_connect (priv->volume_monitor, "volume-changed",
560 G_CALLBACK (volumes_changed), file_system);
561 g_signal_connect (priv->volume_monitor, "drive-connected",
562 G_CALLBACK (volumes_changed), file_system);
563 g_signal_connect (priv->volume_monitor, "drive-disconnected",
564 G_CALLBACK (volumes_changed), file_system);
565 g_signal_connect (priv->volume_monitor, "drive-changed",
566 G_CALLBACK (volumes_changed), file_system);
569 bookmarks_file = get_bookmarks_file ();
570 priv->bookmarks = read_bookmarks (bookmarks_file);
571 priv->bookmarks_monitor = g_file_monitor_file (bookmarks_file,
576 g_warning ("%s", error->message);
577 g_error_free (error);
580 g_signal_connect (priv->bookmarks_monitor, "changed",
581 G_CALLBACK (bookmarks_file_changed), file_system);
583 g_object_unref (bookmarks_file);
586 /* GtkFileSystem public methods */
588 _gtk_file_system_new (void)
590 return g_object_new (GTK_TYPE_FILE_SYSTEM, NULL);
594 _gtk_file_system_list_volumes (GtkFileSystem *file_system)
596 GtkFileSystemPrivate *priv;
599 DEBUG ("list_volumes");
601 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
603 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
604 get_volumes_list (GTK_FILE_SYSTEM (file_system));
606 list = g_slist_copy (priv->volumes);
609 /* Prepend root volume */
610 list = g_slist_prepend (list, (gpointer) root_volume_token);
617 _gtk_file_system_list_bookmarks (GtkFileSystem *file_system)
619 GtkFileSystemPrivate *priv;
620 GSList *bookmarks, *files = NULL;
622 DEBUG ("list_bookmarks");
624 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
625 bookmarks = priv->bookmarks;
629 GtkFileSystemBookmark *bookmark;
631 bookmark = bookmarks->data;
632 bookmarks = bookmarks->next;
634 files = g_slist_prepend (files, g_object_ref (bookmark->file));
637 return g_slist_reverse (files);
641 _gtk_file_system_parse (GtkFileSystem *file_system,
649 gboolean result = FALSE;
650 gboolean is_dir = FALSE;
651 gchar *last_slash = NULL;
656 is_dir = (str [strlen (str) - 1] == G_DIR_SEPARATOR);
658 last_slash = strrchr (str, G_DIR_SEPARATOR);
661 file = g_file_parse_name (str);
663 file = g_file_resolve_relative_path (base_file, str);
665 if (g_file_equal (base_file, file))
667 /* this is when user types '.', could be the
668 * beginning of a hidden file, ./ or ../
670 *folder = g_object_ref (file);
671 *file_part = g_strdup (str);
676 /* it's a dir, or at least it ends with the dir separator */
677 *folder = g_object_ref (file);
678 *file_part = g_strdup ("");
685 parent_file = g_file_get_parent (file);
690 GTK_FILE_CHOOSER_ERROR,
691 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
692 "Could not get parent file");
698 *folder = parent_file;
702 *file_part = g_strdup (last_slash + 1);
704 *file_part = g_strdup (str);
708 g_object_unref (file);
714 free_async_data (AsyncFuncData *async_data)
716 g_object_unref (async_data->file_system);
717 g_object_unref (async_data->file);
718 g_object_unref (async_data->cancellable);
720 if (async_data->folder)
721 g_object_unref (async_data->folder);
723 g_free (async_data->attributes);
728 enumerate_children_callback (GObject *source_object,
729 GAsyncResult *result,
732 GFileEnumerator *enumerator;
733 AsyncFuncData *async_data;
734 GtkFolder *folder = NULL;
736 GError *error = NULL;
738 file = G_FILE (source_object);
739 async_data = (AsyncFuncData *) user_data;
740 enumerator = g_file_enumerate_children_finish (file, result, &error);
744 folder = g_object_new (GTK_TYPE_FOLDER,
745 "file", source_object,
746 "enumerator", enumerator,
747 "attributes", async_data->attributes,
749 g_object_unref (enumerator);
752 gdk_threads_enter ();
753 ((GtkFileSystemGetFolderCallback) async_data->callback) (async_data->cancellable,
754 folder, error, async_data->data);
755 gdk_threads_leave ();
757 free_async_data (async_data);
760 g_error_free (error);
764 _gtk_file_system_get_folder (GtkFileSystem *file_system,
766 const gchar *attributes,
767 GtkFileSystemGetFolderCallback callback,
770 GCancellable *cancellable;
771 AsyncFuncData *async_data;
773 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
774 g_return_val_if_fail (G_IS_FILE (file), NULL);
776 cancellable = g_cancellable_new ();
778 async_data = g_new0 (AsyncFuncData, 1);
779 async_data->file_system = g_object_ref (file_system);
780 async_data->file = g_object_ref (file);
781 async_data->cancellable = g_object_ref (cancellable);
782 async_data->attributes = g_strdup (attributes);
784 async_data->callback = callback;
785 async_data->data = data;
787 g_file_enumerate_children_async (file,
789 G_FILE_QUERY_INFO_NONE,
792 enumerate_children_callback,
798 query_info_callback (GObject *source_object,
799 GAsyncResult *result,
802 AsyncFuncData *async_data;
803 GError *error = NULL;
804 GFileInfo *file_info;
807 DEBUG ("query_info_callback");
809 file = G_FILE (source_object);
810 async_data = (AsyncFuncData *) user_data;
811 file_info = g_file_query_info_finish (file, result, &error);
813 if (async_data->callback)
815 gdk_threads_enter ();
816 ((GtkFileSystemGetInfoCallback) async_data->callback) (async_data->cancellable,
817 file_info, error, async_data->data);
818 gdk_threads_leave ();
822 g_object_unref (file_info);
825 g_error_free (error);
827 free_async_data (async_data);
831 _gtk_file_system_get_info (GtkFileSystem *file_system,
833 const gchar *attributes,
834 GtkFileSystemGetInfoCallback callback,
837 GCancellable *cancellable;
838 AsyncFuncData *async_data;
840 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
841 g_return_val_if_fail (G_IS_FILE (file), NULL);
843 cancellable = g_cancellable_new ();
845 async_data = g_new0 (AsyncFuncData, 1);
846 async_data->file_system = g_object_ref (file_system);
847 async_data->file = g_object_ref (file);
848 async_data->cancellable = g_object_ref (cancellable);
850 async_data->callback = callback;
851 async_data->data = data;
853 g_file_query_info_async (file,
855 G_FILE_QUERY_INFO_NONE,
865 drive_poll_for_media_cb (GObject *source_object,
866 GAsyncResult *result,
869 AsyncFuncData *async_data;
870 GError *error = NULL;
872 g_drive_poll_for_media_finish (G_DRIVE (source_object), result, &error);
873 async_data = (AsyncFuncData *) user_data;
875 gdk_threads_enter ();
876 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
877 (GtkFileSystemVolume *) source_object,
878 error, async_data->data);
879 gdk_threads_leave ();
882 g_error_free (error);
886 volume_mount_cb (GObject *source_object,
887 GAsyncResult *result,
890 AsyncFuncData *async_data;
891 GError *error = NULL;
893 g_volume_mount_finish (G_VOLUME (source_object), result, &error);
894 async_data = (AsyncFuncData *) user_data;
896 gdk_threads_enter ();
897 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
898 (GtkFileSystemVolume *) source_object,
899 error, async_data->data);
900 gdk_threads_leave ();
903 g_error_free (error);
907 _gtk_file_system_mount_volume (GtkFileSystem *file_system,
908 GtkFileSystemVolume *volume,
909 GMountOperation *mount_operation,
910 GtkFileSystemVolumeMountCallback callback,
913 GCancellable *cancellable;
914 AsyncFuncData *async_data;
915 gboolean handled = FALSE;
917 DEBUG ("volume_mount");
919 cancellable = g_cancellable_new ();
921 async_data = g_new0 (AsyncFuncData, 1);
922 async_data->file_system = g_object_ref (file_system);
923 async_data->cancellable = g_object_ref (cancellable);
925 async_data->callback = callback;
926 async_data->data = data;
928 if (G_IS_DRIVE (volume))
930 /* this path happens for drives that are not polled by the OS and where the last media
931 * check indicated that no media was available. So the thing to do here is to
932 * invoke poll_for_media() on the drive
934 g_drive_poll_for_media (G_DRIVE (volume), cancellable, drive_poll_for_media_cb, async_data);
937 else if (G_IS_VOLUME (volume))
939 g_volume_mount (G_VOLUME (volume), G_MOUNT_MOUNT_NONE, mount_operation, cancellable, volume_mount_cb, async_data);
944 free_async_data (async_data);
950 enclosing_volume_mount_cb (GObject *source_object,
951 GAsyncResult *result,
954 GtkFileSystemVolume *volume;
955 AsyncFuncData *async_data;
956 GError *error = NULL;
958 async_data = (AsyncFuncData *) user_data;
959 g_file_mount_enclosing_volume_finish (G_FILE (source_object), result, &error);
960 volume = _gtk_file_system_get_volume_for_file (async_data->file_system, G_FILE (source_object));
962 /* Silently drop G_IO_ERROR_ALREADY_MOUNTED error for gvfs backends without visible mounts. */
963 /* Better than doing query_info with additional I/O every time. */
964 if (error && g_error_matches (error, G_IO_ERROR, G_IO_ERROR_ALREADY_MOUNTED))
965 g_clear_error (&error);
967 gdk_threads_enter ();
968 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable, volume,
969 error, async_data->data);
970 gdk_threads_leave ();
973 g_error_free (error);
977 _gtk_file_system_mount_enclosing_volume (GtkFileSystem *file_system,
979 GMountOperation *mount_operation,
980 GtkFileSystemVolumeMountCallback callback,
983 GCancellable *cancellable;
984 AsyncFuncData *async_data;
986 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
987 g_return_val_if_fail (G_IS_FILE (file), NULL);
989 DEBUG ("mount_enclosing_volume");
991 cancellable = g_cancellable_new ();
993 async_data = g_new0 (AsyncFuncData, 1);
994 async_data->file_system = g_object_ref (file_system);
995 async_data->file = g_object_ref (file);
996 async_data->cancellable = g_object_ref (cancellable);
998 async_data->callback = callback;
999 async_data->data = data;
1001 g_file_mount_enclosing_volume (file,
1005 enclosing_volume_mount_cb,
1011 _gtk_file_system_insert_bookmark (GtkFileSystem *file_system,
1016 GtkFileSystemPrivate *priv;
1018 GtkFileSystemBookmark *bookmark;
1019 gboolean result = TRUE;
1020 GFile *bookmarks_file;
1022 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1023 bookmarks = priv->bookmarks;
1027 bookmark = bookmarks->data;
1028 bookmarks = bookmarks->next;
1030 if (g_file_equal (bookmark->file, file))
1032 /* File is already in bookmarks */
1040 gchar *uri = g_file_get_uri (file);
1043 GTK_FILE_CHOOSER_ERROR,
1044 GTK_FILE_CHOOSER_ERROR_ALREADY_EXISTS,
1045 "%s already exists in the bookmarks list",
1053 bookmark = g_slice_new0 (GtkFileSystemBookmark);
1054 bookmark->file = g_object_ref (file);
1056 priv->bookmarks = g_slist_insert (priv->bookmarks, bookmark, position);
1058 bookmarks_file = get_bookmarks_file ();
1059 save_bookmarks (bookmarks_file, priv->bookmarks);
1060 g_object_unref (bookmarks_file);
1062 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1068 _gtk_file_system_remove_bookmark (GtkFileSystem *file_system,
1072 GtkFileSystemPrivate *priv;
1073 GtkFileSystemBookmark *bookmark;
1075 gboolean result = FALSE;
1076 GFile *bookmarks_file;
1078 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1080 if (!priv->bookmarks)
1083 bookmarks = priv->bookmarks;
1087 bookmark = bookmarks->data;
1089 if (g_file_equal (bookmark->file, file))
1092 priv->bookmarks = g_slist_remove_link (priv->bookmarks, bookmarks);
1093 _gtk_file_system_bookmark_free (bookmark);
1094 g_slist_free_1 (bookmarks);
1098 bookmarks = bookmarks->next;
1103 gchar *uri = g_file_get_uri (file);
1106 GTK_FILE_CHOOSER_ERROR,
1107 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
1108 "%s does not exist in the bookmarks list",
1116 bookmarks_file = get_bookmarks_file ();
1117 save_bookmarks (bookmarks_file, priv->bookmarks);
1118 g_object_unref (bookmarks_file);
1120 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1126 _gtk_file_system_get_bookmark_label (GtkFileSystem *file_system,
1129 GtkFileSystemPrivate *priv;
1131 gchar *label = NULL;
1133 DEBUG ("get_bookmark_label");
1135 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1136 bookmarks = priv->bookmarks;
1140 GtkFileSystemBookmark *bookmark;
1142 bookmark = bookmarks->data;
1143 bookmarks = bookmarks->next;
1145 if (g_file_equal (file, bookmark->file))
1147 label = g_strdup (bookmark->label);
1156 _gtk_file_system_set_bookmark_label (GtkFileSystem *file_system,
1160 GtkFileSystemPrivate *priv;
1161 gboolean changed = FALSE;
1162 GFile *bookmarks_file;
1165 DEBUG ("set_bookmark_label");
1167 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1168 bookmarks = priv->bookmarks;
1172 GtkFileSystemBookmark *bookmark;
1174 bookmark = bookmarks->data;
1175 bookmarks = bookmarks->next;
1177 if (g_file_equal (file, bookmark->file))
1179 g_free (bookmark->label);
1180 bookmark->label = g_strdup (label);
1186 bookmarks_file = get_bookmarks_file ();
1187 save_bookmarks (bookmarks_file, priv->bookmarks);
1188 g_object_unref (bookmarks_file);
1191 g_signal_emit_by_name (file_system, "bookmarks-changed", 0);
1194 GtkFileSystemVolume *
1195 _gtk_file_system_get_volume_for_file (GtkFileSystem *file_system,
1198 GtkFileSystemPrivate *priv;
1201 DEBUG ("get_volume_for_file");
1203 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1204 mount = g_file_find_enclosing_mount (file, NULL, NULL);
1206 if (!mount && g_file_is_native (file))
1207 return (GtkFileSystemVolume *) root_volume_token;
1209 return (GtkFileSystemVolume *) mount;
1212 /* GtkFolder methods */
1214 gtk_folder_set_property (GObject *object,
1216 const GValue *value,
1219 GtkFolderPrivate *priv;
1221 priv = GTK_FOLDER_GET_PRIVATE (object);
1226 priv->folder_file = g_value_dup_object (value);
1228 case PROP_ENUMERATOR:
1229 priv->enumerator = g_value_dup_object (value);
1231 case PROP_ATTRIBUTES:
1232 priv->attributes = g_value_dup_string (value);
1235 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1241 gtk_folder_get_property (GObject *object,
1246 GtkFolderPrivate *priv;
1248 priv = GTK_FOLDER_GET_PRIVATE (object);
1253 g_value_set_object (value, priv->folder_file);
1255 case PROP_ENUMERATOR:
1256 g_value_set_object (value, priv->enumerator);
1258 case PROP_ATTRIBUTES:
1259 g_value_set_string (value, priv->attributes);
1262 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1268 query_created_file_info_callback (GObject *source_object,
1269 GAsyncResult *result,
1272 GFile *file = G_FILE (source_object);
1273 GError *error = NULL;
1278 info = g_file_query_info_finish (file, result, &error);
1282 g_error_free (error);
1286 folder = GTK_FOLDER (user_data);
1287 gtk_folder_add_file (folder, file, info);
1289 files = g_slist_prepend (NULL, file);
1290 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1291 g_slist_free (files);
1293 g_object_unref (info);
1297 directory_monitor_changed (GFileMonitor *monitor,
1300 GFileMonitorEvent event,
1303 GtkFolderPrivate *priv;
1307 folder = GTK_FOLDER (data);
1308 priv = GTK_FOLDER_GET_PRIVATE (folder);
1309 files = g_slist_prepend (NULL, file);
1311 gdk_threads_enter ();
1315 case G_FILE_MONITOR_EVENT_CREATED:
1316 g_file_query_info_async (file,
1318 G_FILE_QUERY_INFO_NONE,
1321 query_created_file_info_callback,
1324 case G_FILE_MONITOR_EVENT_DELETED:
1325 if (g_file_equal (file, priv->folder_file))
1326 g_signal_emit (folder, folder_signals[DELETED], 0);
1328 g_signal_emit (folder, folder_signals[FILES_REMOVED], 0, files);
1334 gdk_threads_leave ();
1336 g_slist_free (files);
1340 enumerator_files_callback (GObject *source_object,
1341 GAsyncResult *result,
1344 GFileEnumerator *enumerator;
1345 GtkFolderPrivate *priv;
1347 GError *error = NULL;
1348 GSList *files = NULL;
1349 GList *file_infos, *f;
1351 enumerator = G_FILE_ENUMERATOR (source_object);
1352 file_infos = g_file_enumerator_next_files_finish (enumerator, result, &error);
1356 if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
1357 g_warning ("%s", error->message);
1359 g_error_free (error);
1363 folder = GTK_FOLDER (user_data);
1364 priv = GTK_FOLDER_GET_PRIVATE (folder);
1368 g_file_enumerator_close_async (enumerator,
1372 gtk_folder_set_finished_loading (folder, TRUE);
1376 g_file_enumerator_next_files_async (enumerator, FILES_PER_QUERY,
1379 enumerator_files_callback,
1382 for (f = file_infos; f; f = f->next)
1388 child_file = g_file_get_child (priv->folder_file, g_file_info_get_name (info));
1389 gtk_folder_add_file (folder, child_file, info);
1390 files = g_slist_prepend (files, child_file);
1393 gdk_threads_enter ();
1394 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1395 gdk_threads_leave ();
1397 g_list_foreach (file_infos, (GFunc) g_object_unref, NULL);
1398 g_list_free (file_infos);
1400 g_slist_foreach (files, (GFunc) g_object_unref, NULL);
1401 g_slist_free (files);
1405 gtk_folder_constructed (GObject *object)
1407 GtkFolderPrivate *priv;
1408 GError *error = NULL;
1410 priv = GTK_FOLDER_GET_PRIVATE (object);
1411 priv->directory_monitor = g_file_monitor_directory (priv->folder_file, G_FILE_MONITOR_NONE, NULL, &error);
1415 g_warning ("%s", error->message);
1416 g_error_free (error);
1419 g_signal_connect (priv->directory_monitor, "changed",
1420 G_CALLBACK (directory_monitor_changed), object);
1422 g_file_enumerator_next_files_async (priv->enumerator,
1426 enumerator_files_callback,
1428 /* This isn't needed anymore */
1429 g_object_unref (priv->enumerator);
1430 priv->enumerator = NULL;
1434 gtk_folder_finalize (GObject *object)
1436 GtkFolderPrivate *priv;
1438 priv = GTK_FOLDER_GET_PRIVATE (object);
1440 g_hash_table_unref (priv->children);
1442 if (priv->folder_file)
1443 g_object_unref (priv->folder_file);
1445 if (priv->directory_monitor)
1446 g_object_unref (priv->directory_monitor);
1448 g_cancellable_cancel (priv->cancellable);
1449 g_object_unref (priv->cancellable);
1450 g_free (priv->attributes);
1452 G_OBJECT_CLASS (_gtk_folder_parent_class)->finalize (object);
1456 _gtk_folder_class_init (GtkFolderClass *class)
1458 GObjectClass *object_class = G_OBJECT_CLASS (class);
1460 object_class->set_property = gtk_folder_set_property;
1461 object_class->get_property = gtk_folder_get_property;
1462 object_class->constructed = gtk_folder_constructed;
1463 object_class->finalize = gtk_folder_finalize;
1465 g_object_class_install_property (object_class,
1467 g_param_spec_object ("file",
1469 "GFile for the folder",
1471 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1472 g_object_class_install_property (object_class,
1474 g_param_spec_object ("enumerator",
1476 "GFileEnumerator to list files",
1477 G_TYPE_FILE_ENUMERATOR,
1478 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1479 g_object_class_install_property (object_class,
1481 g_param_spec_string ("attributes",
1483 "Attributes to query for",
1485 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1486 folder_signals[FILES_ADDED] =
1487 g_signal_new ("files-added",
1488 G_TYPE_FROM_CLASS (object_class),
1490 G_STRUCT_OFFSET (GtkFolderClass, files_added),
1492 g_cclosure_marshal_VOID__POINTER,
1493 G_TYPE_NONE, 1, G_TYPE_POINTER);
1494 folder_signals[FILES_REMOVED] =
1495 g_signal_new ("files-removed",
1496 G_TYPE_FROM_CLASS (object_class),
1498 G_STRUCT_OFFSET (GtkFolderClass, files_removed),
1500 g_cclosure_marshal_VOID__POINTER,
1501 G_TYPE_NONE, 1, G_TYPE_POINTER);
1502 folder_signals[FILES_CHANGED] =
1503 g_signal_new ("files-changed",
1504 G_TYPE_FROM_CLASS (object_class),
1506 G_STRUCT_OFFSET (GtkFolderClass, files_changed),
1508 g_cclosure_marshal_VOID__POINTER,
1509 G_TYPE_NONE, 1, G_TYPE_POINTER);
1510 folder_signals[FINISHED_LOADING] =
1511 g_signal_new ("finished-loading",
1512 G_TYPE_FROM_CLASS (object_class),
1514 G_STRUCT_OFFSET (GtkFolderClass, finished_loading),
1516 g_cclosure_marshal_VOID__VOID,
1518 folder_signals[DELETED] =
1519 g_signal_new ("deleted",
1520 G_TYPE_FROM_CLASS (object_class),
1522 G_STRUCT_OFFSET (GtkFolderClass, deleted),
1524 g_cclosure_marshal_VOID__VOID,
1527 g_type_class_add_private (object_class, sizeof (GtkFolderPrivate));
1531 _gtk_folder_init (GtkFolder *folder)
1533 GtkFolderPrivate *priv;
1535 priv = GTK_FOLDER_GET_PRIVATE (folder);
1537 priv->children = g_hash_table_new_full (g_file_hash,
1538 (GEqualFunc) g_file_equal,
1539 (GDestroyNotify) g_object_unref,
1540 (GDestroyNotify) g_object_unref);
1541 priv->cancellable = g_cancellable_new ();
1545 gtk_folder_set_finished_loading (GtkFolder *folder,
1546 gboolean finished_loading)
1548 GtkFolderPrivate *priv;
1550 priv = GTK_FOLDER_GET_PRIVATE (folder);
1551 priv->finished_loading = (finished_loading == TRUE);
1553 gdk_threads_enter ();
1554 g_signal_emit (folder, folder_signals[FINISHED_LOADING], 0);
1555 gdk_threads_leave ();
1559 gtk_folder_add_file (GtkFolder *folder,
1563 GtkFolderPrivate *priv;
1565 priv = GTK_FOLDER_GET_PRIVATE (folder);
1567 g_hash_table_insert (priv->children,
1568 g_object_ref (file),
1569 g_object_ref (info));
1573 _gtk_folder_list_children (GtkFolder *folder)
1575 GtkFolderPrivate *priv;
1576 GList *files, *elem;
1577 GSList *children = NULL;
1579 priv = GTK_FOLDER_GET_PRIVATE (folder);
1580 files = g_hash_table_get_keys (priv->children);
1583 for (elem = files; elem; elem = elem->next)
1584 children = g_slist_prepend (children, g_object_ref (elem->data));
1586 g_list_free (files);
1592 _gtk_folder_get_info (GtkFolder *folder,
1595 GtkFolderPrivate *priv;
1598 priv = GTK_FOLDER_GET_PRIVATE (folder);
1599 info = g_hash_table_lookup (priv->children, file);
1604 return g_object_ref (info);
1608 _gtk_folder_is_finished_loading (GtkFolder *folder)
1610 GtkFolderPrivate *priv;
1612 priv = GTK_FOLDER_GET_PRIVATE (folder);
1614 return priv->finished_loading;
1617 /* GtkFileSystemVolume public methods */
1619 _gtk_file_system_volume_get_display_name (GtkFileSystemVolume *volume)
1621 DEBUG ("volume_get_display_name");
1623 if (IS_ROOT_VOLUME (volume))
1624 return g_strdup (_(root_volume_token));
1625 if (G_IS_DRIVE (volume))
1626 return g_drive_get_name (G_DRIVE (volume));
1627 else if (G_IS_MOUNT (volume))
1628 return g_mount_get_name (G_MOUNT (volume));
1629 else if (G_IS_VOLUME (volume))
1630 return g_volume_get_name (G_VOLUME (volume));
1636 _gtk_file_system_volume_is_mounted (GtkFileSystemVolume *volume)
1640 DEBUG ("volume_is_mounted");
1642 if (IS_ROOT_VOLUME (volume))
1647 if (G_IS_MOUNT (volume))
1649 else if (G_IS_VOLUME (volume))
1653 mount = g_volume_get_mount (G_VOLUME (volume));
1658 g_object_unref (mount);
1666 _gtk_file_system_volume_get_root (GtkFileSystemVolume *volume)
1670 DEBUG ("volume_get_base");
1672 if (IS_ROOT_VOLUME (volume))
1673 return g_file_new_for_uri ("file:///");
1675 if (G_IS_MOUNT (volume))
1676 file = g_mount_get_root (G_MOUNT (volume));
1677 else if (G_IS_VOLUME (volume))
1681 mount = g_volume_get_mount (G_VOLUME (volume));
1685 file = g_mount_get_root (mount);
1686 g_object_unref (mount);
1694 get_pixbuf_from_gicon (GIcon *icon,
1700 GtkIconTheme *icon_theme;
1701 GtkIconInfo *icon_info;
1704 screen = gtk_widget_get_screen (GTK_WIDGET (widget));
1705 icon_theme = gtk_icon_theme_get_for_screen (screen);
1707 icon_info = gtk_icon_theme_lookup_by_gicon (icon_theme,
1710 GTK_ICON_LOOKUP_USE_BUILTIN);
1715 pixbuf = gtk_icon_info_load_icon (icon_info, error);
1716 gtk_icon_info_free (icon_info);
1722 _gtk_file_system_volume_render_icon (GtkFileSystemVolume *volume,
1730 DEBUG ("volume_get_icon_name");
1732 if (IS_ROOT_VOLUME (volume))
1733 icon = g_themed_icon_new ("drive-harddisk");
1734 else if (G_IS_DRIVE (volume))
1735 icon = g_drive_get_icon (G_DRIVE (volume));
1736 else if (G_IS_VOLUME (volume))
1737 icon = g_volume_get_icon (G_VOLUME (volume));
1738 else if (G_IS_MOUNT (volume))
1739 icon = g_mount_get_icon (G_MOUNT (volume));
1744 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, error);
1746 g_object_unref (icon);
1752 _gtk_file_system_volume_free (GtkFileSystemVolume *volume)
1754 /* Root volume doesn't need to be freed */
1755 if (IS_ROOT_VOLUME (volume))
1758 if (G_IS_MOUNT (volume) ||
1759 G_IS_VOLUME (volume) ||
1760 G_IS_DRIVE (volume))
1761 g_object_unref (volume);
1764 /* GFileInfo helper functions */
1766 _gtk_file_info_render_icon (GFileInfo *info,
1771 GdkPixbuf *pixbuf = NULL;
1772 const gchar *thumbnail_path;
1774 thumbnail_path = g_file_info_get_attribute_byte_string (info, G_FILE_ATTRIBUTE_THUMBNAIL_PATH);
1777 pixbuf = gdk_pixbuf_new_from_file_at_size (thumbnail_path,
1778 icon_size, icon_size,
1783 icon = g_file_info_get_icon (info);
1786 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1790 /* Use general fallback for all files without icon */
1791 icon = g_themed_icon_new ("text-x-generic");
1792 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1793 g_object_unref (icon);
1801 _gtk_file_info_consider_as_directory (GFileInfo *info)
1803 GFileType type = g_file_info_get_file_type (info);
1805 return (type == G_FILE_TYPE_DIRECTORY ||
1806 type == G_FILE_TYPE_MOUNTABLE);