Implement a new design for the wallpaper selection. https://bugzilla.gnome.org/show_bug.cgi?id=676539
585 lines
16 KiB
C
585 lines
16 KiB
C
/* 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
*
|
|
* Author: Thomas Wood <thomas.wood@intel.com>
|
|
*
|
|
*/
|
|
|
|
#include "bg-pictures-source.h"
|
|
|
|
#include "cc-background-item.h"
|
|
|
|
#include <string.h>
|
|
#include <gio/gio.h>
|
|
#include <libgnome-desktop/gnome-desktop-thumbnail.h>
|
|
#include <gdesktop-enums.h>
|
|
|
|
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)
|
|
{
|
|
CcBackgroundItem *item_a;
|
|
CcBackgroundItem *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_background_item_get_name (item_a);
|
|
name_b = cc_background_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;
|
|
CcBackgroundItem *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_background_item_get_uri (item));
|
|
g_object_unref (pixbuf);
|
|
g_object_unref (item);
|
|
return;
|
|
}
|
|
|
|
cc_background_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_background_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_background_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;
|
|
CcBackgroundItem *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;
|
|
CcBackgroundItem *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 CcBackgroundItem */
|
|
uri = g_file_get_uri (file);
|
|
item = cc_background_item_new (uri);
|
|
g_free (uri);
|
|
g_object_set (G_OBJECT (item),
|
|
"flags", CC_BACKGROUND_ITEM_HAS_URI | CC_BACKGROUND_ITEM_HAS_SHADING,
|
|
"shading", G_DESKTOP_BACKGROUND_SHADING_SOLID,
|
|
"placement", G_DESKTOP_BACKGROUND_STYLE_ZOOM,
|
|
NULL);
|
|
if (source_uri != NULL)
|
|
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,
|
|
CcBackgroundItem *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_background_item_get_uri (item);
|
|
|
|
cont = gtk_tree_model_get_iter_first (model, &iter);
|
|
while (cont)
|
|
{
|
|
CcBackgroundItem *tmp_item;
|
|
const char *tmp_uri;
|
|
|
|
gtk_tree_model_get (model, &iter, 1, &tmp_item, -1);
|
|
tmp_uri = cc_background_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 = BG_PICTURES_SOURCE (user_data);
|
|
GList *files, *l;
|
|
GError *err = NULL;
|
|
GFile *parent;
|
|
files = g_file_enumerator_next_files_finish (G_FILE_ENUMERATOR (source),
|
|
res,
|
|
&err);
|
|
|
|
if (err)
|
|
{
|
|
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;
|
|
}
|
|
|
|
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 = BG_PICTURES_SOURCE (user_data)->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) == FALSE)
|
|
g_warning ("Could not fill pictures source: %s", err->message);
|
|
g_error_free (err);
|
|
return;
|
|
}
|
|
|
|
/* 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_LARGE);
|
|
}
|
|
|
|
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;
|
|
}
|