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 get_volumes_list (GtkFileSystem *file_system)
368 GtkFileSystemPrivate *priv;
377 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
381 g_slist_foreach (priv->volumes, (GFunc) g_object_unref, NULL);
382 g_slist_free (priv->volumes);
383 priv->volumes = NULL;
386 /* first go through all connected drives */
387 drives = g_volume_monitor_get_connected_drives (priv->volume_monitor);
389 for (l = drives; l != NULL; l = l->next)
392 volumes = g_drive_get_volumes (drive);
396 for (ll = volumes; ll != NULL; ll = ll->next)
399 mount = g_volume_get_mount (volume);
403 /* Show mounted volume */
404 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
405 g_object_unref (mount);
409 /* Do show the unmounted volumes in the sidebar;
410 * this is so the user can mount it (in case automounting
413 * Also, even if automounting is enabled, this gives a visual
414 * cue that the user should remember to yank out the media if
415 * he just unmounted it.
417 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
420 g_object_unref (volume);
423 g_list_free (volumes);
425 else if (g_drive_is_media_removable (drive) && !g_drive_is_media_check_automatic (drive))
427 /* If the drive has no mountable volumes and we cannot detect media change.. we
428 * display the drive in the sidebar so the user can manually poll the drive by
429 * right clicking and selecting "Rescan..."
431 * This is mainly for drives like floppies where media detection doesn't
432 * work.. but it's also for human beings who like to turn off media detection
433 * in the OS to save battery juice.
436 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (drive));
440 g_list_free (drives);
442 /* add all volumes that is not associated with a drive */
443 volumes = g_volume_monitor_get_volumes (priv->volume_monitor);
445 for (l = volumes; l != NULL; l = l->next)
448 drive = g_volume_get_drive (volume);
452 g_object_unref (drive);
456 mount = g_volume_get_mount (volume);
460 /* show this mount */
461 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
462 g_object_unref (mount);
466 /* see comment above in why we add an icon for a volume */
467 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (volume));
471 g_list_free (volumes);
473 /* add mounts that has no volume (/etc/mtab mounts, ftp, sftp,...) */
474 mounts = g_volume_monitor_get_mounts (priv->volume_monitor);
476 for (l = mounts; l != NULL; l = l->next)
479 volume = g_mount_get_volume (mount);
483 g_object_unref (volume);
487 /* show this mount */
488 priv->volumes = g_slist_prepend (priv->volumes, g_object_ref (mount));
491 g_list_free (mounts);
495 _gtk_file_system_init (GtkFileSystem *file_system)
497 GtkFileSystemPrivate *priv;
498 GFile *bookmarks_file;
499 GError *error = NULL;
503 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
506 priv->volume_monitor = g_volume_monitor_get ();
508 g_signal_connect (priv->volume_monitor, "mount-added",
509 G_CALLBACK (volumes_changed), file_system);
510 g_signal_connect (priv->volume_monitor, "mount-removed",
511 G_CALLBACK (volumes_changed), file_system);
512 g_signal_connect (priv->volume_monitor, "mount-changed",
513 G_CALLBACK (volumes_changed), file_system);
514 g_signal_connect (priv->volume_monitor, "volume-added",
515 G_CALLBACK (volumes_changed), file_system);
516 g_signal_connect (priv->volume_monitor, "volume-removed",
517 G_CALLBACK (volumes_changed), file_system);
518 g_signal_connect (priv->volume_monitor, "volume-changed",
519 G_CALLBACK (volumes_changed), file_system);
520 g_signal_connect (priv->volume_monitor, "drive-connected",
521 G_CALLBACK (volumes_changed), file_system);
522 g_signal_connect (priv->volume_monitor, "drive-disconnected",
523 G_CALLBACK (volumes_changed), file_system);
524 g_signal_connect (priv->volume_monitor, "drive-changed",
525 G_CALLBACK (volumes_changed), file_system);
528 bookmarks_file = get_bookmarks_file ();
529 priv->bookmarks = read_bookmarks (bookmarks_file);
530 priv->bookmarks_monitor = g_file_monitor_file (bookmarks_file,
535 g_warning ("%s", error->message);
536 g_error_free (error);
539 g_signal_connect (priv->bookmarks_monitor, "changed",
540 G_CALLBACK (bookmarks_file_changed), file_system);
542 g_object_unref (bookmarks_file);
545 /* GtkFileSystem public methods */
547 _gtk_file_system_new (void)
549 return g_object_new (GTK_TYPE_FILE_SYSTEM, NULL);
553 _gtk_file_system_list_volumes (GtkFileSystem *file_system)
555 GtkFileSystemPrivate *priv;
558 DEBUG ("list_volumes");
560 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
562 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
563 get_volumes_list (GTK_FILE_SYSTEM (file_system));
565 list = g_slist_copy (priv->volumes);
568 /* Prepend root volume */
569 list = g_slist_prepend (list, (gpointer) root_volume_token);
576 _gtk_file_system_list_bookmarks (GtkFileSystem *file_system)
578 GtkFileSystemPrivate *priv;
579 GSList *bookmarks, *files = NULL;
581 DEBUG ("list_bookmarks");
583 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
584 bookmarks = priv->bookmarks;
588 GtkFileSystemBookmark *bookmark;
590 bookmark = bookmarks->data;
591 bookmarks = bookmarks->next;
593 files = g_slist_prepend (files, g_object_ref (bookmark->file));
596 return g_slist_reverse (files);
600 _gtk_file_system_parse (GtkFileSystem *file_system,
608 gboolean result = FALSE;
609 gboolean is_dir = FALSE;
610 gchar *last_slash = NULL;
615 is_dir = (str [strlen (str) - 1] == G_DIR_SEPARATOR);
617 last_slash = strrchr (str, G_DIR_SEPARATOR);
620 file = g_file_parse_name (str);
622 file = g_file_resolve_relative_path (base_file, str);
624 if (g_file_equal (base_file, file))
626 /* this is when user types '.', could be the
627 * beginning of a hidden file, ./ or ../
629 *folder = g_object_ref (file);
630 *file_part = g_strdup (str);
635 /* it's a dir, or at least it ends with the dir separator */
636 *folder = g_object_ref (file);
637 *file_part = g_strdup ("");
644 parent_file = g_file_get_parent (file);
649 GTK_FILE_CHOOSER_ERROR,
650 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
651 "Could not get parent file");
657 *folder = parent_file;
661 *file_part = g_strdup (last_slash + 1);
663 *file_part = g_strdup (str);
667 g_object_unref (file);
673 free_async_data (AsyncFuncData *async_data)
675 g_object_unref (async_data->file_system);
676 g_object_unref (async_data->file);
677 g_object_unref (async_data->cancellable);
679 if (async_data->folder)
680 g_object_unref (async_data->folder);
682 g_free (async_data->attributes);
687 enumerate_children_callback (GObject *source_object,
688 GAsyncResult *result,
691 GFileEnumerator *enumerator;
692 AsyncFuncData *async_data;
693 GtkFolder *folder = NULL;
695 GError *error = NULL;
697 file = G_FILE (source_object);
698 async_data = (AsyncFuncData *) user_data;
699 enumerator = g_file_enumerate_children_finish (file, result, &error);
703 folder = g_object_new (GTK_TYPE_FOLDER,
704 "file", source_object,
705 "enumerator", enumerator,
706 "attributes", async_data->attributes,
708 g_object_unref (enumerator);
711 gdk_threads_enter ();
712 ((GtkFileSystemGetFolderCallback) async_data->callback) (async_data->cancellable,
713 folder, error, async_data->data);
714 gdk_threads_leave ();
716 free_async_data (async_data);
719 g_error_free (error);
723 _gtk_file_system_get_folder (GtkFileSystem *file_system,
725 const gchar *attributes,
726 GtkFileSystemGetFolderCallback callback,
729 GCancellable *cancellable;
730 AsyncFuncData *async_data;
732 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
733 g_return_val_if_fail (G_IS_FILE (file), NULL);
735 cancellable = g_cancellable_new ();
737 async_data = g_new0 (AsyncFuncData, 1);
738 async_data->file_system = g_object_ref (file_system);
739 async_data->file = g_object_ref (file);
740 async_data->cancellable = g_object_ref (cancellable);
741 async_data->attributes = g_strdup (attributes);
743 async_data->callback = callback;
744 async_data->data = data;
746 g_file_enumerate_children_async (file,
748 G_FILE_QUERY_INFO_NONE,
751 enumerate_children_callback,
757 query_info_callback (GObject *source_object,
758 GAsyncResult *result,
761 AsyncFuncData *async_data;
762 GError *error = NULL;
763 GFileInfo *file_info;
766 DEBUG ("query_info_callback");
768 file = G_FILE (source_object);
769 async_data = (AsyncFuncData *) user_data;
770 file_info = g_file_query_info_finish (file, result, &error);
772 if (async_data->callback)
774 gdk_threads_enter ();
775 ((GtkFileSystemGetInfoCallback) async_data->callback) (async_data->cancellable,
776 file_info, error, async_data->data);
777 gdk_threads_leave ();
781 g_object_unref (file_info);
784 g_error_free (error);
786 free_async_data (async_data);
790 _gtk_file_system_get_info (GtkFileSystem *file_system,
792 const gchar *attributes,
793 GtkFileSystemGetInfoCallback callback,
796 GCancellable *cancellable;
797 AsyncFuncData *async_data;
799 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
800 g_return_val_if_fail (G_IS_FILE (file), NULL);
802 cancellable = g_cancellable_new ();
804 async_data = g_new0 (AsyncFuncData, 1);
805 async_data->file_system = g_object_ref (file_system);
806 async_data->file = g_object_ref (file);
807 async_data->cancellable = g_object_ref (cancellable);
809 async_data->callback = callback;
810 async_data->data = data;
812 g_file_query_info_async (file,
814 G_FILE_QUERY_INFO_NONE,
824 drive_poll_for_media_cb (GObject *source_object,
825 GAsyncResult *result,
828 AsyncFuncData *async_data;
829 GError *error = NULL;
831 g_drive_poll_for_media_finish (G_DRIVE (source_object), result, &error);
832 async_data = (AsyncFuncData *) user_data;
834 gdk_threads_enter ();
835 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
836 (GtkFileSystemVolume *) source_object,
837 error, async_data->data);
838 gdk_threads_leave ();
841 g_error_free (error);
845 volume_mount_cb (GObject *source_object,
846 GAsyncResult *result,
849 AsyncFuncData *async_data;
850 GError *error = NULL;
852 g_volume_mount_finish (G_VOLUME (source_object), result, &error);
853 async_data = (AsyncFuncData *) user_data;
855 gdk_threads_enter ();
856 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable,
857 (GtkFileSystemVolume *) source_object,
858 error, async_data->data);
859 gdk_threads_leave ();
862 g_error_free (error);
866 _gtk_file_system_mount_volume (GtkFileSystem *file_system,
867 GtkFileSystemVolume *volume,
868 GMountOperation *mount_operation,
869 GtkFileSystemVolumeMountCallback callback,
872 GCancellable *cancellable;
873 AsyncFuncData *async_data;
874 gboolean handled = FALSE;
876 DEBUG ("volume_mount");
878 cancellable = g_cancellable_new ();
880 async_data = g_new0 (AsyncFuncData, 1);
881 async_data->file_system = g_object_ref (file_system);
882 async_data->cancellable = g_object_ref (cancellable);
884 async_data->callback = callback;
885 async_data->data = data;
887 if (G_IS_DRIVE (volume))
889 /* this path happens for drives that are not polled by the OS and where the last media
890 * check indicated that no media was available. So the thing to do here is to
891 * invoke poll_for_media() on the drive
893 g_drive_poll_for_media (G_DRIVE (volume), cancellable, drive_poll_for_media_cb, async_data);
896 else if (G_IS_VOLUME (volume))
898 g_volume_mount (G_VOLUME (volume), G_MOUNT_MOUNT_NONE, mount_operation, cancellable, volume_mount_cb, async_data);
903 free_async_data (async_data);
909 enclosing_volume_mount_cb (GObject *source_object,
910 GAsyncResult *result,
913 GtkFileSystemVolume *volume;
914 AsyncFuncData *async_data;
915 GError *error = NULL;
917 async_data = (AsyncFuncData *) user_data;
918 g_file_mount_enclosing_volume_finish (G_FILE (source_object), result, &error);
919 volume = _gtk_file_system_get_volume_for_file (async_data->file_system, G_FILE (source_object));
921 gdk_threads_enter ();
922 ((GtkFileSystemVolumeMountCallback) async_data->callback) (async_data->cancellable, volume,
923 error, async_data->data);
924 gdk_threads_leave ();
927 g_error_free (error);
931 _gtk_file_system_mount_enclosing_volume (GtkFileSystem *file_system,
933 GMountOperation *mount_operation,
934 GtkFileSystemVolumeMountCallback callback,
937 GCancellable *cancellable;
938 AsyncFuncData *async_data;
940 g_return_val_if_fail (GTK_IS_FILE_SYSTEM (file_system), NULL);
941 g_return_val_if_fail (G_IS_FILE (file), NULL);
943 DEBUG ("mount_enclosing_volume");
945 cancellable = g_cancellable_new ();
947 async_data = g_new0 (AsyncFuncData, 1);
948 async_data->file_system = g_object_ref (file_system);
949 async_data->file = g_object_ref (file);
950 async_data->cancellable = g_object_ref (cancellable);
952 async_data->callback = callback;
953 async_data->data = data;
955 g_file_mount_enclosing_volume (file,
959 enclosing_volume_mount_cb,
965 _gtk_file_system_insert_bookmark (GtkFileSystem *file_system,
970 GtkFileSystemPrivate *priv;
972 GtkFileSystemBookmark *bookmark;
973 gboolean result = TRUE;
974 GFile *bookmarks_file;
976 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
977 bookmarks = priv->bookmarks;
981 bookmark = bookmarks->data;
982 bookmarks = bookmarks->next;
984 if (g_file_equal (bookmark->file, file))
986 /* File is already in bookmarks */
994 gchar *uri = g_file_get_uri (file);
997 GTK_FILE_CHOOSER_ERROR,
998 GTK_FILE_CHOOSER_ERROR_ALREADY_EXISTS,
999 "%s already exists in the bookmarks list",
1007 bookmark = g_slice_new0 (GtkFileSystemBookmark);
1008 bookmark->file = g_object_ref (file);
1010 priv->bookmarks = g_slist_insert (priv->bookmarks, bookmark, position);
1012 bookmarks_file = get_bookmarks_file ();
1013 save_bookmarks (bookmarks_file, priv->bookmarks);
1014 g_object_unref (bookmarks_file);
1016 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1022 _gtk_file_system_remove_bookmark (GtkFileSystem *file_system,
1026 GtkFileSystemPrivate *priv;
1027 GtkFileSystemBookmark *bookmark;
1029 gboolean result = FALSE;
1030 GFile *bookmarks_file;
1032 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1034 if (!priv->bookmarks)
1037 bookmarks = priv->bookmarks;
1041 bookmark = bookmarks->data;
1043 if (g_file_equal (bookmark->file, file))
1046 priv->bookmarks = g_slist_remove_link (priv->bookmarks, bookmarks);
1047 _gtk_file_system_bookmark_free (bookmark);
1048 g_slist_free_1 (bookmarks);
1052 bookmarks = bookmarks->next;
1057 gchar *uri = g_file_get_uri (file);
1060 GTK_FILE_CHOOSER_ERROR,
1061 GTK_FILE_CHOOSER_ERROR_NONEXISTENT,
1062 "%s does not exist in the bookmarks list",
1070 bookmarks_file = get_bookmarks_file ();
1071 save_bookmarks (bookmarks_file, priv->bookmarks);
1072 g_object_unref (bookmarks_file);
1074 g_signal_emit (file_system, fs_signals[BOOKMARKS_CHANGED], 0);
1080 _gtk_file_system_get_bookmark_label (GtkFileSystem *file_system,
1083 GtkFileSystemPrivate *priv;
1085 gchar *label = NULL;
1087 DEBUG ("get_bookmark_label");
1089 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1090 bookmarks = priv->bookmarks;
1094 GtkFileSystemBookmark *bookmark;
1096 bookmark = bookmarks->data;
1097 bookmarks = bookmarks->next;
1099 if (g_file_equal (file, bookmark->file))
1101 label = g_strdup (bookmark->label);
1110 _gtk_file_system_set_bookmark_label (GtkFileSystem *file_system,
1114 GtkFileSystemPrivate *priv;
1115 gboolean changed = FALSE;
1116 GFile *bookmarks_file;
1119 DEBUG ("set_bookmark_label");
1121 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1122 bookmarks = priv->bookmarks;
1126 GtkFileSystemBookmark *bookmark;
1128 bookmark = bookmarks->data;
1129 bookmarks = bookmarks->next;
1131 if (g_file_equal (file, bookmark->file))
1133 g_free (bookmark->label);
1134 bookmark->label = g_strdup (label);
1140 bookmarks_file = get_bookmarks_file ();
1141 save_bookmarks (bookmarks_file, priv->bookmarks);
1142 g_object_unref (bookmarks_file);
1145 g_signal_emit_by_name (file_system, "bookmarks-changed", 0);
1148 GtkFileSystemVolume *
1149 _gtk_file_system_get_volume_for_file (GtkFileSystem *file_system,
1152 GtkFileSystemPrivate *priv;
1155 DEBUG ("get_volume_for_file");
1157 priv = GTK_FILE_SYSTEM_GET_PRIVATE (file_system);
1158 mount = g_file_find_enclosing_mount (file, NULL, NULL);
1160 if (!mount && g_file_is_native (file))
1161 return (GtkFileSystemVolume *) root_volume_token;
1163 return (GtkFileSystemVolume *) mount;
1166 /* GtkFolder methods */
1168 gtk_folder_set_property (GObject *object,
1170 const GValue *value,
1173 GtkFolderPrivate *priv;
1175 priv = GTK_FOLDER_GET_PRIVATE (object);
1180 priv->folder_file = g_value_dup_object (value);
1182 case PROP_ENUMERATOR:
1183 priv->enumerator = g_value_dup_object (value);
1185 case PROP_ATTRIBUTES:
1186 priv->attributes = g_value_dup_string (value);
1189 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1195 gtk_folder_get_property (GObject *object,
1200 GtkFolderPrivate *priv;
1202 priv = GTK_FOLDER_GET_PRIVATE (object);
1207 g_value_set_object (value, priv->folder_file);
1209 case PROP_ENUMERATOR:
1210 g_value_set_object (value, priv->enumerator);
1212 case PROP_ATTRIBUTES:
1213 g_value_set_string (value, priv->attributes);
1216 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
1222 query_created_file_info_callback (GObject *source_object,
1223 GAsyncResult *result,
1226 GFile *file = G_FILE (source_object);
1227 GError *error = NULL;
1232 info = g_file_query_info_finish (file, result, &error);
1236 g_error_free (error);
1240 folder = GTK_FOLDER (user_data);
1241 gtk_folder_add_file (folder, file, info);
1243 files = g_slist_prepend (NULL, file);
1244 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1245 g_slist_free (files);
1247 g_object_unref (info);
1251 directory_monitor_changed (GFileMonitor *monitor,
1254 GFileMonitorEvent event,
1257 GtkFolderPrivate *priv;
1261 folder = GTK_FOLDER (data);
1262 priv = GTK_FOLDER_GET_PRIVATE (folder);
1263 files = g_slist_prepend (NULL, file);
1265 gdk_threads_enter ();
1269 case G_FILE_MONITOR_EVENT_CREATED:
1270 g_file_query_info_async (file,
1272 G_FILE_QUERY_INFO_NONE,
1275 query_created_file_info_callback,
1278 case G_FILE_MONITOR_EVENT_DELETED:
1279 if (g_file_equal (file, priv->folder_file))
1280 g_signal_emit (folder, folder_signals[DELETED], 0);
1282 g_signal_emit (folder, folder_signals[FILES_REMOVED], 0, files);
1288 gdk_threads_leave ();
1290 g_slist_free (files);
1294 enumerator_files_callback (GObject *source_object,
1295 GAsyncResult *result,
1298 GFileEnumerator *enumerator;
1299 GtkFolderPrivate *priv;
1301 GError *error = NULL;
1302 GSList *files = NULL;
1303 GList *file_infos, *f;
1305 enumerator = G_FILE_ENUMERATOR (source_object);
1306 file_infos = g_file_enumerator_next_files_finish (enumerator, result, &error);
1310 if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
1311 g_warning ("%s", error->message);
1313 g_error_free (error);
1317 folder = GTK_FOLDER (user_data);
1318 priv = GTK_FOLDER_GET_PRIVATE (folder);
1322 g_file_enumerator_close_async (enumerator,
1326 gtk_folder_set_finished_loading (folder, TRUE);
1330 g_file_enumerator_next_files_async (enumerator, FILES_PER_QUERY,
1333 enumerator_files_callback,
1336 for (f = file_infos; f; f = f->next)
1342 child_file = g_file_get_child (priv->folder_file, g_file_info_get_name (info));
1343 gtk_folder_add_file (folder, child_file, info);
1344 files = g_slist_prepend (files, child_file);
1347 gdk_threads_enter ();
1348 g_signal_emit (folder, folder_signals[FILES_ADDED], 0, files);
1349 gdk_threads_leave ();
1351 g_list_foreach (file_infos, (GFunc) g_object_unref, NULL);
1352 g_list_free (file_infos);
1354 g_slist_foreach (files, (GFunc) g_object_unref, NULL);
1355 g_slist_free (files);
1359 gtk_folder_constructed (GObject *object)
1361 GtkFolderPrivate *priv;
1362 GError *error = NULL;
1364 priv = GTK_FOLDER_GET_PRIVATE (object);
1365 priv->directory_monitor = g_file_monitor_directory (priv->folder_file, G_FILE_MONITOR_NONE, NULL, &error);
1369 g_warning ("%s", error->message);
1370 g_error_free (error);
1373 g_signal_connect (priv->directory_monitor, "changed",
1374 G_CALLBACK (directory_monitor_changed), object);
1376 g_file_enumerator_next_files_async (priv->enumerator,
1380 enumerator_files_callback,
1382 /* This isn't needed anymore */
1383 g_object_unref (priv->enumerator);
1384 priv->enumerator = NULL;
1388 gtk_folder_finalize (GObject *object)
1390 GtkFolderPrivate *priv;
1392 priv = GTK_FOLDER_GET_PRIVATE (object);
1394 g_hash_table_unref (priv->children);
1396 if (priv->folder_file)
1397 g_object_unref (priv->folder_file);
1399 if (priv->directory_monitor)
1400 g_object_unref (priv->directory_monitor);
1402 g_cancellable_cancel (priv->cancellable);
1403 g_object_unref (priv->cancellable);
1404 g_free (priv->attributes);
1406 G_OBJECT_CLASS (_gtk_folder_parent_class)->finalize (object);
1410 _gtk_folder_class_init (GtkFolderClass *class)
1412 GObjectClass *object_class = G_OBJECT_CLASS (class);
1414 object_class->set_property = gtk_folder_set_property;
1415 object_class->get_property = gtk_folder_get_property;
1416 object_class->constructed = gtk_folder_constructed;
1417 object_class->finalize = gtk_folder_finalize;
1419 g_object_class_install_property (object_class,
1421 g_param_spec_object ("file",
1423 "GFile for the folder",
1425 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1426 g_object_class_install_property (object_class,
1428 g_param_spec_object ("enumerator",
1430 "GFileEnumerator to list files",
1431 G_TYPE_FILE_ENUMERATOR,
1432 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1433 g_object_class_install_property (object_class,
1435 g_param_spec_string ("attributes",
1437 "Attributes to query for",
1439 GTK_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
1440 folder_signals[FILES_ADDED] =
1441 g_signal_new ("files-added",
1442 G_TYPE_FROM_CLASS (object_class),
1444 G_STRUCT_OFFSET (GtkFolderClass, files_added),
1446 g_cclosure_marshal_VOID__POINTER,
1447 G_TYPE_NONE, 1, G_TYPE_POINTER);
1448 folder_signals[FILES_REMOVED] =
1449 g_signal_new ("files-removed",
1450 G_TYPE_FROM_CLASS (object_class),
1452 G_STRUCT_OFFSET (GtkFolderClass, files_removed),
1454 g_cclosure_marshal_VOID__POINTER,
1455 G_TYPE_NONE, 1, G_TYPE_POINTER);
1456 folder_signals[FILES_CHANGED] =
1457 g_signal_new ("files-changed",
1458 G_TYPE_FROM_CLASS (object_class),
1460 G_STRUCT_OFFSET (GtkFolderClass, files_changed),
1462 g_cclosure_marshal_VOID__POINTER,
1463 G_TYPE_NONE, 1, G_TYPE_POINTER);
1464 folder_signals[FINISHED_LOADING] =
1465 g_signal_new ("finished-loading",
1466 G_TYPE_FROM_CLASS (object_class),
1468 G_STRUCT_OFFSET (GtkFolderClass, finished_loading),
1470 g_cclosure_marshal_VOID__VOID,
1472 folder_signals[DELETED] =
1473 g_signal_new ("deleted",
1474 G_TYPE_FROM_CLASS (object_class),
1476 G_STRUCT_OFFSET (GtkFolderClass, deleted),
1478 g_cclosure_marshal_VOID__VOID,
1481 g_type_class_add_private (object_class, sizeof (GtkFolderPrivate));
1485 _gtk_folder_init (GtkFolder *folder)
1487 GtkFolderPrivate *priv;
1489 priv = GTK_FOLDER_GET_PRIVATE (folder);
1491 priv->children = g_hash_table_new_full (g_file_hash,
1492 (GEqualFunc) g_file_equal,
1493 (GDestroyNotify) g_object_unref,
1494 (GDestroyNotify) g_object_unref);
1495 priv->cancellable = g_cancellable_new ();
1499 gtk_folder_set_finished_loading (GtkFolder *folder,
1500 gboolean finished_loading)
1502 GtkFolderPrivate *priv;
1504 priv = GTK_FOLDER_GET_PRIVATE (folder);
1505 priv->finished_loading = (finished_loading == TRUE);
1507 gdk_threads_enter ();
1508 g_signal_emit (folder, folder_signals[FINISHED_LOADING], 0);
1509 gdk_threads_leave ();
1513 gtk_folder_add_file (GtkFolder *folder,
1517 GtkFolderPrivate *priv;
1519 priv = GTK_FOLDER_GET_PRIVATE (folder);
1521 g_hash_table_insert (priv->children,
1522 g_object_ref (file),
1523 g_object_ref (info));
1527 _gtk_folder_list_children (GtkFolder *folder)
1529 GtkFolderPrivate *priv;
1530 GList *files, *elem;
1531 GSList *children = NULL;
1533 priv = GTK_FOLDER_GET_PRIVATE (folder);
1534 files = g_hash_table_get_keys (priv->children);
1537 for (elem = files; elem; elem = elem->next)
1538 children = g_slist_prepend (children, g_object_ref (elem->data));
1540 g_list_free (files);
1546 _gtk_folder_get_info (GtkFolder *folder,
1549 GtkFolderPrivate *priv;
1552 priv = GTK_FOLDER_GET_PRIVATE (folder);
1553 info = g_hash_table_lookup (priv->children, file);
1558 return g_object_ref (info);
1562 _gtk_folder_is_finished_loading (GtkFolder *folder)
1564 GtkFolderPrivate *priv;
1566 priv = GTK_FOLDER_GET_PRIVATE (folder);
1568 return priv->finished_loading;
1571 /* GtkFileSystemVolume public methods */
1573 _gtk_file_system_volume_get_display_name (GtkFileSystemVolume *volume)
1575 DEBUG ("volume_get_display_name");
1577 if (IS_ROOT_VOLUME (volume))
1578 return g_strdup (_(root_volume_token));
1579 if (G_IS_DRIVE (volume))
1580 return g_drive_get_name (G_DRIVE (volume));
1581 else if (G_IS_MOUNT (volume))
1582 return g_mount_get_name (G_MOUNT (volume));
1583 else if (G_IS_VOLUME (volume))
1584 return g_volume_get_name (G_VOLUME (volume));
1590 _gtk_file_system_volume_is_mounted (GtkFileSystemVolume *volume)
1594 DEBUG ("volume_is_mounted");
1596 if (IS_ROOT_VOLUME (volume))
1601 if (G_IS_MOUNT (volume))
1603 else if (G_IS_VOLUME (volume))
1607 mount = g_volume_get_mount (G_VOLUME (volume));
1612 g_object_unref (mount);
1620 _gtk_file_system_volume_get_root (GtkFileSystemVolume *volume)
1624 DEBUG ("volume_get_base");
1626 if (IS_ROOT_VOLUME (volume))
1627 return g_file_new_for_uri ("file:///");
1629 if (G_IS_MOUNT (volume))
1630 file = g_mount_get_root (G_MOUNT (volume));
1631 else if (G_IS_VOLUME (volume))
1635 mount = g_volume_get_mount (G_VOLUME (volume));
1639 file = g_mount_get_root (mount);
1640 g_object_unref (mount);
1648 get_pixbuf_from_gicon (GIcon *icon,
1654 GtkIconTheme *icon_theme;
1655 GtkIconInfo *icon_info;
1658 screen = gtk_widget_get_screen (GTK_WIDGET (widget));
1659 icon_theme = gtk_icon_theme_get_for_screen (screen);
1661 icon_info = gtk_icon_theme_lookup_by_gicon (icon_theme,
1664 GTK_ICON_LOOKUP_USE_BUILTIN);
1669 pixbuf = gtk_icon_info_load_icon (icon_info, error);
1670 gtk_icon_info_free (icon_info);
1676 _gtk_file_system_volume_render_icon (GtkFileSystemVolume *volume,
1684 DEBUG ("volume_get_icon_name");
1686 if (IS_ROOT_VOLUME (volume))
1687 icon = g_themed_icon_new ("drive-harddisk");
1688 else if (G_IS_DRIVE (volume))
1689 icon = g_drive_get_icon (G_DRIVE (volume));
1690 else if (G_IS_VOLUME (volume))
1691 icon = g_volume_get_icon (G_VOLUME (volume));
1692 else if (G_IS_MOUNT (volume))
1693 icon = g_mount_get_icon (G_MOUNT (volume));
1698 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, error);
1700 g_object_unref (icon);
1706 _gtk_file_system_volume_free (GtkFileSystemVolume *volume)
1708 /* Root volume doesn't need to be freed */
1709 if (IS_ROOT_VOLUME (volume))
1712 if (G_IS_MOUNT (volume) ||
1713 G_IS_VOLUME (volume) ||
1714 G_IS_DRIVE (volume))
1715 g_object_unref (volume);
1718 /* GFileInfo helper functions */
1720 _gtk_file_info_render_icon (GFileInfo *info,
1725 GdkPixbuf *pixbuf = NULL;
1726 const gchar *thumbnail_path;
1728 thumbnail_path = g_file_info_get_attribute_byte_string (info, G_FILE_ATTRIBUTE_THUMBNAIL_PATH);
1731 pixbuf = gdk_pixbuf_new_from_file_at_size (thumbnail_path,
1732 icon_size, icon_size,
1737 icon = g_file_info_get_icon (info);
1740 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1744 /* Use general fallback for all files without icon */
1745 icon = g_themed_icon_new ("text-x-generic");
1746 pixbuf = get_pixbuf_from_gicon (icon, widget, icon_size, NULL);
1747 g_object_unref (icon);