/* Eye Of Mate - Image Store * * Copyright (C) 2006-2008 The Free Software Foundation * * Author: Claudio Saavedra <csaavedra@gnome.org> * * Based on code by: Jens Finke <jens@triq.net> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "eom-list-store.h" #include "eom-thumbnail.h" #include "eom-image.h" #include "eom-job-queue.h" #include "eom-jobs.h" #include <string.h> #define EOM_LIST_STORE_GET_PRIVATE(object) \ (G_TYPE_INSTANCE_GET_PRIVATE ((object), EOM_TYPE_LIST_STORE, EomListStorePrivate)) G_DEFINE_TYPE (EomListStore, eom_list_store, GTK_TYPE_LIST_STORE); struct _EomListStorePrivate { GList *monitors; /* Monitors for the directories */ gint initial_image; /* The image that should be selected firstly by the view. */ GdkPixbuf *busy_image; /* Loading image icon */ GdkPixbuf *missing_image; /* Missing image icon */ GMutex mutex; /* Mutex for saving the jobs in the model */ }; static void eom_list_store_finalize (GObject *object) { EomListStore *store = EOM_LIST_STORE (object); if (store->priv != NULL) { g_free (store->priv); store->priv = NULL; } G_OBJECT_CLASS (eom_list_store_parent_class)->finalize (object); } static void foreach_monitors_free (gpointer data, gpointer user_data) { g_file_monitor_cancel (G_FILE_MONITOR (data)); } static void eom_list_store_dispose (GObject *object) { EomListStore *store = EOM_LIST_STORE (object); g_list_foreach (store->priv->monitors, foreach_monitors_free, NULL); g_list_free (store->priv->monitors); store->priv->monitors = NULL; if(store->priv->busy_image != NULL) { g_object_unref (store->priv->busy_image); store->priv->busy_image = NULL; } if(store->priv->missing_image != NULL) { g_object_unref (store->priv->missing_image); store->priv->missing_image = NULL; } g_mutex_clear (&store->priv->mutex); G_OBJECT_CLASS (eom_list_store_parent_class)->dispose (object); } static void eom_list_store_class_init (EomListStoreClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = eom_list_store_finalize; object_class->dispose = eom_list_store_dispose; g_type_class_add_private (object_class, sizeof (EomListStorePrivate)); } /* Sorting functions */ static gint eom_list_store_compare_func (GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data) { gint r_value; EomImage *image_a, *image_b; gtk_tree_model_get (model, a, EOM_LIST_STORE_EOM_IMAGE, &image_a, -1); gtk_tree_model_get (model, b, EOM_LIST_STORE_EOM_IMAGE, &image_b, -1); r_value = strcmp (eom_image_get_collate_key (image_a), eom_image_get_collate_key (image_b)); g_object_unref (G_OBJECT (image_a)); g_object_unref (G_OBJECT (image_b)); return r_value; } static GdkPixbuf * eom_list_store_get_icon (const gchar *icon_name) { GError *error = NULL; GtkIconTheme *icon_theme; GdkPixbuf *pixbuf; icon_theme = gtk_icon_theme_get_default (); pixbuf = gtk_icon_theme_load_icon (icon_theme, icon_name, EOM_LIST_STORE_THUMB_SIZE, 0, &error); if (!pixbuf) { g_warning ("Couldn't load icon: %s", error->message); g_error_free (error); } return pixbuf; } static void eom_list_store_init (EomListStore *self) { GType types[EOM_LIST_STORE_NUM_COLUMNS]; types[EOM_LIST_STORE_THUMBNAIL] = GDK_TYPE_PIXBUF; types[EOM_LIST_STORE_EOM_IMAGE] = G_TYPE_OBJECT; types[EOM_LIST_STORE_THUMB_SET] = G_TYPE_BOOLEAN; types[EOM_LIST_STORE_EOM_JOB] = G_TYPE_POINTER; gtk_list_store_set_column_types (GTK_LIST_STORE (self), EOM_LIST_STORE_NUM_COLUMNS, types); self->priv = EOM_LIST_STORE_GET_PRIVATE (self); self->priv->monitors = NULL; self->priv->initial_image = -1; self->priv->busy_image = eom_list_store_get_icon ("image-loading"); self->priv->missing_image = eom_list_store_get_icon ("image-missing"); g_mutex_init (&self->priv->mutex); gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (self), eom_list_store_compare_func, NULL, NULL); gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (self), GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID, GTK_SORT_ASCENDING); } /** * eom_list_store_new: * * Creates a new and empty #EomListStore. * * Returns: a newly created #EomListStore. **/ GtkListStore* eom_list_store_new (void) { return g_object_new (EOM_TYPE_LIST_STORE, NULL); } /* Searchs for a file in the store. If found and @iter_found is not NULL, then sets @iter_found to a #GtkTreeIter pointing to the file. */ static gboolean is_file_in_list_store (EomListStore *store, const gchar *info_uri, GtkTreeIter *iter_found) { gboolean found = FALSE; EomImage *image; GFile *file; gchar *str; GtkTreeIter iter; if (!gtk_tree_model_get_iter_first (GTK_TREE_MODEL (store), &iter)) { return FALSE; } do { gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); if (!image) continue; file = eom_image_get_file (image); str = g_file_get_uri (file); found = (strcmp (str, info_uri) == 0)? TRUE : FALSE; g_object_unref (file); g_free (str); g_object_unref (G_OBJECT (image)); } while (!found && gtk_tree_model_iter_next (GTK_TREE_MODEL (store), &iter)); if (found && iter_found != NULL) { *iter_found = iter; } return found; } static gboolean is_file_in_list_store_file (EomListStore *store, GFile *file, GtkTreeIter *iter_found) { gchar *uri_str; gboolean result; uri_str = g_file_get_uri (file); result = is_file_in_list_store (store, uri_str, iter_found); g_free (uri_str); return result; } static void eom_job_thumbnail_cb (EomJobThumbnail *job, gpointer data) { EomListStore *store; GtkTreeIter iter; EomImage *image; GdkPixbuf *thumbnail; GFile *file; g_return_if_fail (EOM_IS_LIST_STORE (data)); store = EOM_LIST_STORE (data); file = eom_image_get_file (job->image); if (is_file_in_list_store_file (store, file, &iter)) { gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); if (job->thumbnail) { eom_image_set_thumbnail (image, job->thumbnail); /* Getting the thumbnail, in case it needed * transformations */ thumbnail = eom_image_get_thumbnail (image); } else { thumbnail = g_object_ref (store->priv->missing_image); } gtk_list_store_set (GTK_LIST_STORE (store), &iter, EOM_LIST_STORE_THUMBNAIL, thumbnail, EOM_LIST_STORE_THUMB_SET, TRUE, EOM_LIST_STORE_EOM_JOB, NULL, -1); g_object_unref (thumbnail); } g_object_unref (file); } static void on_image_changed (EomImage *image, EomListStore *store) { GtkTreePath *path; GtkTreeIter iter; gint pos; pos = eom_list_store_get_pos_by_image (store, image); path = gtk_tree_path_new_from_indices (pos, -1); gtk_tree_model_get_iter (GTK_TREE_MODEL (store), &iter, path); eom_list_store_thumbnail_refresh (store, &iter); gtk_tree_path_free (path); } /** * eom_list_store_remove: * @store: An #EomListStore. * @iter: A #GtkTreeIter. * * Removes the image pointed by @iter from @store. **/ static void eom_list_store_remove (EomListStore *store, GtkTreeIter *iter) { EomImage *image; gtk_tree_model_get (GTK_TREE_MODEL (store), iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); g_signal_handlers_disconnect_by_func (image, on_image_changed, store); g_object_unref (image); gtk_list_store_remove (GTK_LIST_STORE (store), iter); } /** * eom_list_store_append_image: * @store: An #EomListStore. * @image: An #EomImage. * * Adds an #EomImage to @store. The thumbnail of the image is not * loaded and will only be loaded if the thumbnail is made visible * or eom_list_store_set_thumbnail() is called. * **/ void eom_list_store_append_image (EomListStore *store, EomImage *image) { GtkTreeIter iter; g_signal_connect (image, "changed", G_CALLBACK (on_image_changed), store); gtk_list_store_append (GTK_LIST_STORE (store), &iter); gtk_list_store_set (GTK_LIST_STORE (store), &iter, EOM_LIST_STORE_EOM_IMAGE, image, EOM_LIST_STORE_THUMBNAIL, store->priv->busy_image, EOM_LIST_STORE_THUMB_SET, FALSE, -1); } static void eom_list_store_append_image_from_file (EomListStore *store, GFile *file) { EomImage *image; g_return_if_fail (EOM_IS_LIST_STORE (store)); image = eom_image_new_file (file); eom_list_store_append_image (store, image); } static void file_monitor_changed_cb (GFileMonitor *monitor, GFile *file, GFile *other_file, GFileMonitorEvent event, EomListStore *store) { const char *mimetype; GFileInfo *file_info; GtkTreeIter iter; EomImage *image; switch (event) { case G_FILE_MONITOR_EVENT_CHANGES_DONE_HINT: file_info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, 0, NULL, NULL); if (file_info == NULL) { break; } mimetype = g_file_info_get_content_type (file_info); if (is_file_in_list_store_file (store, file, &iter)) { if (eom_image_is_supported_mime_type (mimetype)) { gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); eom_image_file_changed (image); g_object_unref (image); eom_list_store_thumbnail_refresh (store, &iter); } else { eom_list_store_remove (store, &iter); } } else { if (eom_image_is_supported_mime_type (mimetype)) { eom_list_store_append_image_from_file (store, file); } } g_object_unref (file_info); break; case G_FILE_MONITOR_EVENT_DELETED: if (is_file_in_list_store_file (store, file, &iter)) { EomImage *image; gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); eom_list_store_remove (store, &iter); } break; case G_FILE_MONITOR_EVENT_CREATED: if (!is_file_in_list_store_file (store, file, NULL)) { file_info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, 0, NULL, NULL); if (file_info == NULL) { break; } mimetype = g_file_info_get_content_type (file_info); if (eom_image_is_supported_mime_type (mimetype)) { eom_list_store_append_image_from_file (store, file); } g_object_unref (file_info); } break; case G_FILE_MONITOR_EVENT_ATTRIBUTE_CHANGED: file_info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, 0, NULL, NULL); if (file_info == NULL) { break; } mimetype = g_file_info_get_content_type (file_info); if (is_file_in_list_store_file (store, file, &iter) && eom_image_is_supported_mime_type (mimetype)) { eom_list_store_thumbnail_refresh (store, &iter); } g_object_unref (file_info); break; default: break; } } /* * Called for each file in a directory. Checks if the file is some * sort of image. If so, it creates an image object and adds it to the * list. */ static void directory_visit (GFile *directory, GFileInfo *children_info, EomListStore *store) { GFile *child; gboolean load_uri = FALSE; const char *mime_type, *name; mime_type = g_file_info_get_content_type (children_info); name = g_file_info_get_name (children_info); if (!g_str_has_prefix (name, ".")) { if (eom_image_is_supported_mime_type (mime_type)) { load_uri = TRUE; } } if (load_uri) { child = g_file_get_child (directory, name); eom_list_store_append_image_from_file (store, child); } } static void eom_list_store_append_directory (EomListStore *store, GFile *file, GFileType file_type) { GFileMonitor *file_monitor; GFileEnumerator *file_enumerator; GFileInfo *file_info; g_return_if_fail (file_type == G_FILE_TYPE_DIRECTORY); file_monitor = g_file_monitor_directory (file, 0, NULL, NULL); if (file_monitor != NULL) { g_signal_connect (file_monitor, "changed", G_CALLBACK (file_monitor_changed_cb), store); /* prepend seems more efficient to me, we don't need this list to be sorted */ store->priv->monitors = g_list_prepend (store->priv->monitors, file_monitor); } file_enumerator = g_file_enumerate_children (file, G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE "," G_FILE_ATTRIBUTE_STANDARD_NAME, 0, NULL, NULL); file_info = g_file_enumerator_next_file (file_enumerator, NULL, NULL); while (file_info != NULL) { directory_visit (file, file_info, store); g_object_unref (file_info); file_info = g_file_enumerator_next_file (file_enumerator, NULL, NULL); } g_object_unref (file_enumerator); } /** * eom_list_store_add_files: * @store: An #EomListStore. * @file_list: (element-type GFile): A %NULL-terminated list of #GFile's. * * Adds a list of #GFile's to @store. The given list * must be %NULL-terminated. * * If any of the #GFile's in @file_list is a directory, all the images * in that directory will be added to @store. If the list of files contains * only one file and this is a regular file, then all the images in the same * directory will be added as well to @store. * **/ void eom_list_store_add_files (EomListStore *store, GList *file_list) { GList *it; GFileInfo *file_info; GFileType file_type; GFile *initial_file = NULL; GtkTreeIter iter; if (file_list == NULL) { return; } gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store), GTK_TREE_SORTABLE_UNSORTED_SORT_COLUMN_ID, GTK_SORT_ASCENDING); for (it = file_list; it != NULL; it = it->next) { GFile *file = (GFile *) it->data; file_info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_TYPE"," G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE, 0, NULL, NULL); if (file_info == NULL) { continue; } file_type = g_file_info_get_file_type (file_info); /* Workaround for gvfs backends that don't set the GFileType. */ if (G_UNLIKELY (file_type == G_FILE_TYPE_UNKNOWN)) { const gchar *ctype; ctype = g_file_info_get_content_type (file_info); /* If the content type is supported adjust file_type */ if (eom_image_is_supported_mime_type (ctype)) file_type = G_FILE_TYPE_REGULAR; } g_object_unref (file_info); if (file_type == G_FILE_TYPE_DIRECTORY) { eom_list_store_append_directory (store, file, file_type); } else if (file_type == G_FILE_TYPE_REGULAR && g_list_length (file_list) == 1) { initial_file = g_file_dup (file); file = g_file_get_parent (file); file_info = g_file_query_info (file, G_FILE_ATTRIBUTE_STANDARD_TYPE, 0, NULL, NULL); /* If we can't get a file_info, file_type will stay as G_FILE_TYPE_REGULAR */ if (file_info != NULL) { file_type = g_file_info_get_file_type (file_info); g_object_unref (file_info); } if (file_type == G_FILE_TYPE_DIRECTORY) { eom_list_store_append_directory (store, file, file_type); if (!is_file_in_list_store_file (store, initial_file, &iter)) { eom_list_store_append_image_from_file (store, initial_file); } } else { eom_list_store_append_image_from_file (store, initial_file); } g_object_unref (file); } else if (file_type == G_FILE_TYPE_REGULAR && g_list_length (file_list) > 1) { eom_list_store_append_image_from_file (store, file); } } gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store), GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID, GTK_SORT_ASCENDING); if (initial_file && is_file_in_list_store_file (store, initial_file, &iter)) { store->priv->initial_image = eom_list_store_get_pos_by_iter (store, &iter); g_object_unref (initial_file); } else { store->priv->initial_image = 0; } } /** * eom_list_store_remove_image: * @store: An #EomListStore. * @image: An #EomImage. * * Removes @image from @store. **/ void eom_list_store_remove_image (EomListStore *store, EomImage *image) { GtkTreeIter iter; GFile *file; g_return_if_fail (EOM_IS_LIST_STORE (store)); g_return_if_fail (EOM_IS_IMAGE (image)); file = eom_image_get_file (image); if (is_file_in_list_store_file (store, file, &iter)) { eom_list_store_remove (store, &iter); } g_object_unref (file); } /** * eom_list_store_new_from_glist: * @list: (element-type EomImage): a %NULL-terminated list of #EomImage's. * * Creates a new #EomListStore from a list of #EomImage's. * The given list must be %NULL-terminated. * * Returns: a new #EomListStore. **/ GtkListStore * eom_list_store_new_from_glist (GList *list) { GList *it; GtkListStore *store = eom_list_store_new (); for (it = list; it != NULL; it = it->next) { eom_list_store_append_image (EOM_LIST_STORE (store), EOM_IMAGE (it->data)); } return store; } /** * eom_list_store_get_pos_by_image: * @store: An #EomListStore. * @image: An #EomImage. * * Gets the position where @image is stored in @store. If @image * is not stored in @store, -1 is returned. * * Returns: the position of @image in @store or -1 if not found. **/ gint eom_list_store_get_pos_by_image (EomListStore *store, EomImage *image) { GtkTreeIter iter; gint pos = -1; GFile *file; g_return_val_if_fail (EOM_IS_LIST_STORE (store), -1); g_return_val_if_fail (EOM_IS_IMAGE (image), -1); file = eom_image_get_file (image); if (is_file_in_list_store_file (store, file, &iter)) { pos = eom_list_store_get_pos_by_iter (store, &iter); } g_object_unref (file); return pos; } /** * eom_list_store_get_image_by_pos: * @store: An #EomListStore. * @pos: the position of the required #EomImage. * * Gets the #EomImage in the position @pos of @store. If there is * no image at position @pos, %NULL is returned. * * Returns: (transfer full): the #EomImage in position @pos or %NULL. * **/ EomImage * eom_list_store_get_image_by_pos (EomListStore *store, gint pos) { EomImage *image = NULL; GtkTreeIter iter; g_return_val_if_fail (EOM_IS_LIST_STORE (store), NULL); if (gtk_tree_model_iter_nth_child (GTK_TREE_MODEL (store), &iter, NULL, pos)) { gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); } return image; } /** * eom_list_store_get_pos_by_iter: * @store: An #EomListStore. * @iter: A #GtkTreeIter pointing to an image in @store. * * Gets the position of the image pointed by @iter. * * Returns: The position of the image pointed by @iter. **/ gint eom_list_store_get_pos_by_iter (EomListStore *store, GtkTreeIter *iter) { gint *indices; GtkTreePath *path; gint pos; path = gtk_tree_model_get_path (GTK_TREE_MODEL (store), iter); indices = gtk_tree_path_get_indices (path); pos = indices [0]; gtk_tree_path_free (path); return pos; } /** * eom_list_store_length: * @store: An #EomListStore. * * Returns the number of images in the store. * * Returns: The number of images in @store. **/ gint eom_list_store_length (EomListStore *store) { g_return_val_if_fail (EOM_IS_LIST_STORE (store), -1); return gtk_tree_model_iter_n_children (GTK_TREE_MODEL (store), NULL); } /** * eom_list_store_get_initial_pos: * @store: An #EomListStore. * * Gets the position of the #EomImage that should be loaded first. * If not set, it returns -1. * * Returns: the position of the image to be loaded first or -1. * **/ gint eom_list_store_get_initial_pos (EomListStore *store) { g_return_val_if_fail (EOM_IS_LIST_STORE (store), -1); return store->priv->initial_image; } static void eom_list_store_remove_thumbnail_job (EomListStore *store, GtkTreeIter *iter) { EomJob *job; gtk_tree_model_get (GTK_TREE_MODEL (store), iter, EOM_LIST_STORE_EOM_JOB, &job, -1); if (job != NULL) { g_mutex_lock (&store->priv->mutex); eom_job_queue_remove_job (job); gtk_list_store_set (GTK_LIST_STORE (store), iter, EOM_LIST_STORE_EOM_JOB, NULL, -1); g_mutex_unlock (&store->priv->mutex); } } static void eom_list_store_add_thumbnail_job (EomListStore *store, GtkTreeIter *iter) { EomImage *image; EomJob *job; gtk_tree_model_get (GTK_TREE_MODEL (store), iter, EOM_LIST_STORE_EOM_IMAGE, &image, EOM_LIST_STORE_EOM_JOB, &job, -1); if (job != NULL) { g_object_unref (image); return; } job = eom_job_thumbnail_new (image); g_signal_connect (job, "finished", G_CALLBACK (eom_job_thumbnail_cb), store); g_mutex_lock (&store->priv->mutex); gtk_list_store_set (GTK_LIST_STORE (store), iter, EOM_LIST_STORE_EOM_JOB, job, -1); eom_job_queue_add_job (job); g_mutex_unlock (&store->priv->mutex); g_object_unref (job); g_object_unref (image); } /** * eom_list_store_thumbnail_set: * @store: An #EomListStore. * @iter: A #GtkTreeIter pointing to an image in @store. * * Sets the thumbnail for the image pointed by @iter. * **/ void eom_list_store_thumbnail_set (EomListStore *store, GtkTreeIter *iter) { gboolean thumb_set = FALSE; gtk_tree_model_get (GTK_TREE_MODEL (store), iter, EOM_LIST_STORE_THUMB_SET, &thumb_set, -1); if (thumb_set) { return; } eom_list_store_add_thumbnail_job (store, iter); } /** * eom_list_store_thumbnail_unset: * @store: An #EomListStore. * @iter: A #GtkTreeIter pointing to an image in @store. * * Unsets the thumbnail for the image pointed by @iter, changing * it to a "busy" icon. * **/ void eom_list_store_thumbnail_unset (EomListStore *store, GtkTreeIter *iter) { EomImage *image; eom_list_store_remove_thumbnail_job (store, iter); gtk_tree_model_get (GTK_TREE_MODEL (store), iter, EOM_LIST_STORE_EOM_IMAGE, &image, -1); eom_image_set_thumbnail (image, NULL); g_object_unref (image); gtk_list_store_set (GTK_LIST_STORE (store), iter, EOM_LIST_STORE_THUMBNAIL, store->priv->busy_image, EOM_LIST_STORE_THUMB_SET, FALSE, -1); } /** * eom_list_store_thumbnail_refresh: * @store: An #EomListStore. * @iter: A #GtkTreeIter pointing to an image in @store. * * Refreshes the thumbnail for the image pointed by @iter. * **/ void eom_list_store_thumbnail_refresh (EomListStore *store, GtkTreeIter *iter) { eom_list_store_remove_thumbnail_job (store, iter); eom_list_store_add_thumbnail_job (store, iter); }