/* bg-pictures-source.c */ /* * Copyright (C) 2010 Intel, Inc * * 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, see . * * Author: Thomas Wood * */ #include "bg-pictures-source.h" #include "cc-appearance-item.h" #include #include #include #include G_DEFINE_TYPE (BgPicturesSource, bg_pictures_source, BG_TYPE_SOURCE) #define PICTURES_SOURCE_PRIVATE(o) \ (G_TYPE_INSTANCE_GET_PRIVATE ((o), BG_TYPE_PICTURES_SOURCE, BgPicturesSourcePrivate)) #define ATTRIBUTES G_FILE_ATTRIBUTE_STANDARD_NAME "," \ G_FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE struct _BgPicturesSourcePrivate { GCancellable *cancellable; GnomeDesktopThumbnailFactory *thumb_factory; GHashTable *known_items; }; const char * const content_types[] = { "image/png", "image/jpeg", "image/bmp", "image/svg+xml", NULL }; static char *bg_pictures_source_get_unique_filename (const char *uri); static void bg_pictures_source_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { switch (property_id) { default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void bg_pictures_source_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { switch (property_id) { default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void bg_pictures_source_dispose (GObject *object) { BgPicturesSourcePrivate *priv = BG_PICTURES_SOURCE (object)->priv; if (priv->cancellable) { g_cancellable_cancel (priv->cancellable); g_object_unref (priv->cancellable); priv->cancellable = NULL; } if (priv->thumb_factory) { g_object_unref (priv->thumb_factory); priv->thumb_factory = NULL; } G_OBJECT_CLASS (bg_pictures_source_parent_class)->dispose (object); } static void bg_pictures_source_finalize (GObject *object) { BgPicturesSource *bg_source = BG_PICTURES_SOURCE (object); if (bg_source->priv->thumb_factory) { g_object_unref (bg_source->priv->thumb_factory); bg_source->priv->thumb_factory = NULL; } if (bg_source->priv->known_items) { g_hash_table_destroy (bg_source->priv->known_items); bg_source->priv->known_items = NULL; } G_OBJECT_CLASS (bg_pictures_source_parent_class)->finalize (object); } static void bg_pictures_source_class_init (BgPicturesSourceClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); g_type_class_add_private (klass, sizeof (BgPicturesSourcePrivate)); object_class->get_property = bg_pictures_source_get_property; object_class->set_property = bg_pictures_source_set_property; object_class->dispose = bg_pictures_source_dispose; object_class->finalize = bg_pictures_source_finalize; } static int sort_func (GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, BgPicturesSource *bg_source) { CcAppearanceItem *item_a; CcAppearanceItem *item_b; const char *name_a; const char *name_b; int retval; gtk_tree_model_get (model, a, 1, &item_a, -1); gtk_tree_model_get (model, b, 1, &item_b, -1); name_a = cc_appearance_item_get_name (item_a); name_b = cc_appearance_item_get_name (item_b); retval = g_utf8_collate (name_a, name_b); g_object_unref (item_a); g_object_unref (item_b); return retval; } static void picture_scaled (GObject *source_object, GAsyncResult *res, gpointer user_data) { BgPicturesSource *bg_source; CcAppearanceItem *item; GError *error = NULL; GdkPixbuf *pixbuf; const char *source_url; const char *software; GtkTreeIter iter; GtkListStore *store; pixbuf = gdk_pixbuf_new_from_stream_finish (res, &error); if (pixbuf == NULL) { if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) g_warning ("Failed to load image: %s", error->message); g_error_free (error); return; } /* since we were not cancelled, we can now cast user_data * back to BgPicturesSource. */ bg_source = BG_PICTURES_SOURCE (user_data); store = bg_source_get_liststore (BG_SOURCE (bg_source)); item = g_object_get_data (source_object, "item"); gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (store), 1, (GtkTreeIterCompareFunc)sort_func, bg_source, NULL); gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store), 1, GTK_SORT_ASCENDING); /* Ignore screenshots */ software = gdk_pixbuf_get_option (pixbuf, "tEXt::Software"); if (software != NULL && g_str_equal (software, "gnome-screenshot")) { g_debug ("Ignored URL '%s' as it's a screenshot from gnome-screenshot", cc_appearance_item_get_uri (item)); g_object_unref (pixbuf); g_object_unref (item); return; } cc_appearance_item_load (item, NULL); /* insert the item into the liststore */ gtk_list_store_insert_with_values (store, &iter, 0, 0, pixbuf, 1, item, -1); source_url = cc_appearance_item_get_source_url (item); if (source_url != NULL) { g_hash_table_insert (bg_source->priv->known_items, bg_pictures_source_get_unique_filename (source_url), GINT_TO_POINTER (TRUE)); } else { char *cache_path; GFile *file, *parent, *dir; cache_path = bg_pictures_source_get_cache_path (); dir = g_file_new_for_path (cache_path); g_free (cache_path); file = g_file_new_for_uri (cc_appearance_item_get_uri (item)); parent = g_file_get_parent (file); if (g_file_equal (parent, dir)) { char *basename; basename = g_file_get_basename (file); g_hash_table_insert (bg_source->priv->known_items, basename, GINT_TO_POINTER (TRUE)); } g_object_unref (file); g_object_unref (parent); } g_object_unref (pixbuf); } static void picture_opened_for_read (GObject *source_object, GAsyncResult *res, gpointer user_data) { BgPicturesSource *bg_source; CcAppearanceItem *item; GFileInputStream *stream; GError *error = NULL; item = g_object_get_data (source_object, "item"); stream = g_file_read_finish (G_FILE (source_object), res, &error); if (stream == NULL) { if (!g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED)) { char *filename = g_file_get_path (G_FILE (source_object)); g_warning ("Failed to load picture '%s': %s", filename, error->message); g_free (filename); } g_error_free (error); g_object_unref (item); return; } /* since we were not cancelled, we can now cast user_data * back to BgPicturesSource. */ bg_source = BG_PICTURES_SOURCE (user_data); g_object_set_data (G_OBJECT (stream), "item", item); gdk_pixbuf_new_from_stream_at_scale_async (G_INPUT_STREAM (stream), THUMBNAIL_WIDTH, THUMBNAIL_HEIGHT, TRUE, bg_source->priv->cancellable, picture_scaled, bg_source); g_object_unref (stream); } static gboolean in_content_types (const char *content_type) { guint i; for (i = 0; content_types[i]; i++) if (g_str_equal (content_types[i], content_type)) return TRUE; return FALSE; } static gboolean add_single_file (BgPicturesSource *bg_source, GFile *file, GFileInfo *info, const char *source_uri) { const gchar *content_type; CcAppearanceItem *item; char *uri; /* find png and jpeg files */ content_type = g_file_info_get_content_type (info); if (!content_type) return FALSE; if (!in_content_types (content_type)) return FALSE; /* create a new CcAppearanceItem */ uri = g_file_get_uri (file); item = cc_appearance_item_new (uri); g_free (uri); g_object_set (G_OBJECT (item), "flags", CC_APPEARANCE_ITEM_HAS_URI | CC_APPEARANCE_ITEM_HAS_SHADING, "shading", G_DESKTOP_BACKGROUND_SHADING_SOLID, "placement", G_DESKTOP_BACKGROUND_STYLE_ZOOM, NULL); if (source_uri != NULL && !g_file_is_native (file)) g_object_set (G_OBJECT (item), "source-url", source_uri, NULL); g_object_set_data (G_OBJECT (file), "item", item); g_file_read_async (file, G_PRIORITY_DEFAULT, bg_source->priv->cancellable, picture_opened_for_read, bg_source); g_object_unref (file); return TRUE; } gboolean bg_pictures_source_add (BgPicturesSource *bg_source, const char *uri) { GFile *file; GFileInfo *info; gboolean retval; file = g_file_new_for_uri (uri); info = g_file_query_info (file, ATTRIBUTES, G_FILE_QUERY_INFO_NONE, NULL, NULL); if (info == NULL) return FALSE; retval = add_single_file (bg_source, file, info, uri); return retval; } gboolean bg_pictures_source_remove (BgPicturesSource *bg_source, CcAppearanceItem *item) { GtkTreeModel *model; GtkTreeIter iter; gboolean cont; const char *uri; gboolean retval; retval = FALSE; model = GTK_TREE_MODEL (bg_source_get_liststore (BG_SOURCE (bg_source))); uri = cc_appearance_item_get_uri (item); cont = gtk_tree_model_get_iter_first (model, &iter); while (cont) { CcAppearanceItem *tmp_item; const char *tmp_uri; gtk_tree_model_get (model, &iter, 1, &tmp_item, -1); tmp_uri = cc_appearance_item_get_uri (tmp_item); if (g_str_equal (tmp_uri, uri)) { GFile *file; char *uuid; file = g_file_new_for_uri (uri); uuid = g_file_get_basename (file); g_hash_table_insert (bg_source->priv->known_items, uuid, NULL); gtk_list_store_remove (GTK_LIST_STORE (model), &iter); retval = TRUE; g_file_trash (file, NULL, NULL); g_object_unref (file); break; } g_object_unref (tmp_item); cont = gtk_tree_model_iter_next (model, &iter); } return retval; } static void file_info_async_ready (GObject *source, GAsyncResult *res, gpointer user_data) { BgPicturesSource *bg_source; GList *files, *l; GError *err = NULL; GFile *parent; files = g_file_enumerator_next_files_finish (G_FILE_ENUMERATOR (source), res, &err); if (err) { if (!g_error_matches (err, G_IO_ERROR, G_IO_ERROR_CANCELLED)) g_warning ("Could not get pictures file information: %s", err->message); g_error_free (err); g_list_foreach (files, (GFunc) g_object_unref, NULL); g_list_free (files); return; } bg_source = BG_PICTURES_SOURCE (user_data); parent = g_file_enumerator_get_container (G_FILE_ENUMERATOR (source)); /* iterate over the available files */ for (l = files; l; l = g_list_next (l)) { GFileInfo *info = l->data; GFile *file; file = g_file_get_child (parent, g_file_info_get_name (info)); add_single_file (bg_source, file, info, NULL); } g_list_foreach (files, (GFunc) g_object_unref, NULL); g_list_free (files); } static void dir_enum_async_ready (GObject *source, GAsyncResult *res, gpointer user_data) { BgPicturesSourcePrivate *priv; GFileEnumerator *enumerator; GError *err = NULL; enumerator = g_file_enumerate_children_finish (G_FILE (source), res, &err); if (err) { if (!g_error_matches (err, G_IO_ERROR, G_IO_ERROR_NOT_FOUND) && !g_error_matches (err, G_IO_ERROR, G_IO_ERROR_CANCELLED)) g_warning ("Could not fill pictures source: %s", err->message); g_error_free (err); return; } priv = BG_PICTURES_SOURCE (user_data)->priv; /* get the files */ g_file_enumerator_next_files_async (enumerator, G_MAXINT, G_PRIORITY_LOW, priv->cancellable, file_info_async_ready, user_data); } char * bg_pictures_source_get_cache_path (void) { return g_build_filename (g_get_user_cache_dir (), "gnome-control-center", "backgrounds", NULL); } static char * bg_pictures_source_get_unique_filename (const char *uri) { GChecksum *csum; char *ret; csum = g_checksum_new (G_CHECKSUM_SHA256); g_checksum_update (csum, (guchar *) uri, -1); ret = g_strdup (g_checksum_get_string (csum)); g_checksum_free (csum); return ret; } char * bg_pictures_source_get_unique_path (const char *uri) { GFile *parent, *file; char *cache_path; char *filename; char *ret; cache_path = bg_pictures_source_get_cache_path (); parent = g_file_new_for_path (cache_path); g_free (cache_path); filename = bg_pictures_source_get_unique_filename (uri); file = g_file_get_child (parent, filename); g_free (filename); ret = g_file_get_path (file); g_object_unref (file); return ret; } gboolean bg_pictures_source_is_known (BgPicturesSource *bg_source, const char *uri) { gboolean retval; char *uuid; uuid = bg_pictures_source_get_unique_filename (uri); retval = (GPOINTER_TO_INT (g_hash_table_lookup (bg_source->priv->known_items, uuid))); g_free (uuid); return retval; } static void bg_pictures_source_init (BgPicturesSource *self) { const gchar *pictures_path; BgPicturesSourcePrivate *priv; GFile *dir; char *cache_path; priv = self->priv = PICTURES_SOURCE_PRIVATE (self); priv->cancellable = g_cancellable_new (); priv->known_items = g_hash_table_new_full (g_str_hash, g_str_equal, (GDestroyNotify) g_free, NULL); pictures_path = g_get_user_special_dir (G_USER_DIRECTORY_PICTURES); dir = g_file_new_for_path (pictures_path); g_file_enumerate_children_async (dir, ATTRIBUTES, G_FILE_QUERY_INFO_NONE, G_PRIORITY_LOW, priv->cancellable, dir_enum_async_ready, self); g_object_unref (dir); cache_path = bg_pictures_source_get_cache_path (); dir = g_file_new_for_path (cache_path); g_file_enumerate_children_async (dir, ATTRIBUTES, G_FILE_QUERY_INFO_NONE, G_PRIORITY_LOW, priv->cancellable, dir_enum_async_ready, self); g_object_unref (dir); priv->thumb_factory = gnome_desktop_thumbnail_factory_new (GNOME_DESKTOP_THUMBNAIL_SIZE_NORMAL); } BgPicturesSource * bg_pictures_source_new (void) { return g_object_new (BG_TYPE_PICTURES_SOURCE, NULL); } const char * const * bg_pictures_get_support_content_types (void) { return content_types; }