diff options
Diffstat (limited to 'src/dmap/grl-dpap-db.c')
-rw-r--r-- | src/dmap/grl-dpap-db.c | 368 |
1 files changed, 368 insertions, 0 deletions
diff --git a/src/dmap/grl-dpap-db.c b/src/dmap/grl-dpap-db.c new file mode 100644 index 0000000..ea418d2 --- /dev/null +++ b/src/dmap/grl-dpap-db.c @@ -0,0 +1,368 @@ +/* + * Copyright (C) 2012 W. Michael Petullo. + * + * Contact: W. Michael Petullo <mike@flyn.org> + * + * This library is free software; you can redistribute it and/or + * modify it under the terms of the GNU Lesser General Public License + * as published by the Free Software Foundation; version 2.1 of + * the License, or (at your option) any later version. + * + * This library 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 + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this library; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +/* See grl-daap-db.c for a description of this database. */ + +#ifdef HAVE_CONFIG_H +#include "config.h" +#endif + +#include <glib/gi18n-lib.h> +#include <sys/stat.h> +#include <sys/types.h> +#include <glib.h> +#include <string.h> + +#include "grl-dpap-db.h" + +#define PHOTOS_ID "photos" +#define PHOTOS_NAME _("Photos") + +/* Media IDs start at max and go down. Container IDs start at 1 and go up. */ +static guint nextid = G_MAXINT; /* NOTE: this should be G_MAXUINT, but iPhoto can't handle it. */ + +struct GrlDPAPDbPrivate { + /* Contains each picture box (tracked with photos hash table) */ + GrlMediaBox *photos_box; + + GHashTable *root; + GHashTable *photos; +}; + +enum { + PROP_0, + PROP_RECORD_FACTORY, +}; + +static guint +box_hash (gconstpointer a) +{ + return g_str_hash (grl_media_get_id (GRL_MEDIA (a))); +} + +static gboolean +box_equal (gconstpointer a, gconstpointer b) +{ + return g_str_equal (grl_media_get_id (GRL_MEDIA (a)), grl_media_get_id (GRL_MEDIA (b))); +} + +GrlDPAPDb * +grl_dpap_db_new (void) +{ + GrlDPAPDb *db = g_object_new (TYPE_GRL_DPAP_DB, NULL); + + return db; +} + +static DMAPRecord * +grl_dpap_db_lookup_by_id (const DMAPDb *db, guint id) +{ + g_warning ("Not implemented"); + return NULL; +} + +static void +grl_dpap_db_foreach (const DMAPDb *db, + GHFunc func, + gpointer data) +{ + g_warning ("Not implemented"); +} + +static gint64 +grl_dpap_db_count (const DMAPDb *db) +{ + g_warning ("Not implemented"); + return 0; +} + +static void +set_insert (GHashTable *category, const char *category_name, char *set_name, GrlMedia *media) +{ + gchar *id = NULL; + GrlMedia *box; + GHashTable *set; + + id = g_strdup_printf ("%s-%s", category_name, set_name); + + box = g_object_new (GRL_TYPE_MEDIA_BOX, NULL); + grl_media_set_id (box, id); + grl_media_set_title (box, set_name); + + set = g_hash_table_lookup (category, box); + if (set == NULL) { + set = g_hash_table_new_full (box_hash, box_equal, g_object_unref, NULL); + g_hash_table_insert (category, g_object_ref (box), set); + } + + g_hash_table_insert (set, g_object_ref (media), NULL); + + g_free (id); + g_object_unref (box); +} + +static guint +grl_dpap_db_add (DMAPDb *_db, DMAPRecord *_record) +{ + g_assert (IS_GRL_DPAP_DB (_db)); + g_assert (IS_DPAP_RECORD (_record)); + + GrlDPAPDb *db = GRL_DPAP_DB (_db); + DPAPRecord *record = DPAP_RECORD (_record); + + gint height = 0, + width = 0, + largefilesize = 0, + creationdate = 0, + rating = 0; + GByteArray *thumbnail = NULL; + gchar *id_s = NULL, + *filename = NULL, + *aspectratio = NULL, + *format = NULL, + *comments = NULL, + *url = NULL; + GrlMedia *media; + + g_object_get (record, + "large-filesize", + &largefilesize, + "creation-date", + &creationdate, + "rating", + &rating, + "filename", + &filename, + "aspect-ratio", + &aspectratio, + "pixel-height", + &height, + "pixel-width", + &width, + "format", + &format, + "comments", + &comments, + "thumbnail", + &thumbnail, + "location", + &url, + NULL); + + id_s = g_strdup_printf ("%u", nextid); + + media = grl_media_image_new (); + + grl_media_set_id (media, id_s); + + if (filename) + grl_media_set_title (media, filename); + + if (url) { + /* Replace URL's dpap:// with http:// */ + memcpy (url, "http", 4); + grl_media_set_url (media, url); + } + + GrlMediaImage *media_image = GRL_MEDIA_IMAGE (media); + + grl_media_image_set_width (media_image, width); + grl_media_image_set_height (media_image, height); + + set_insert (db->priv->photos, PHOTOS_ID, "Unknown", media); + + g_free (id_s); + g_object_unref (media); + + return --nextid; +} + +static gboolean +same_media (GrlMedia *a, GrlMedia *b) +{ + return (strcmp (grl_media_get_id (a), grl_media_get_id (b)) == 0); +} + +void +grl_dpap_db_browse (GrlDPAPDb *db, + GrlMedia *container, + GrlSource *source, + guint op_id, + guint skip, + guint count, + GrlSourceResultCb func, + gpointer user_data) +{ + g_assert (IS_GRL_DPAP_DB (db)); + + int i; + guint remaining; + GHashTable *hash_table; + GHashTableIter iter; + gpointer key, val; + + const gchar *box_id = grl_media_get_id (container); + if (box_id == NULL) { + hash_table = db->priv->root; + } else if (same_media (container, GRL_MEDIA (db->priv->photos_box))) { + hash_table = db->priv->photos; + } else { + hash_table = g_hash_table_lookup (db->priv->photos, container); + } + + /* Should not be NULL; this means the container requested + does not exist in the database. */ + if (hash_table == NULL) { + GError *error = g_error_new (GRL_CORE_ERROR, + GRL_CORE_ERROR_BROWSE_FAILED, + "Invalid container identifier %s", + box_id); + func (source, op_id, NULL, 0, user_data, error); + goto done; + } + + remaining = g_hash_table_size (hash_table) - skip; + remaining = remaining < count ? remaining : count; + g_hash_table_iter_init (&iter, hash_table); + for (i = 0; g_hash_table_iter_next (&iter, &key, &val) && i < skip + count; i++) { + if (i < skip) + continue; + if (GRL_IS_MEDIA_BOX (key)) + grl_media_box_set_childcount (GRL_MEDIA_BOX (key), g_hash_table_size (val)); + func (source, op_id, GRL_MEDIA (g_object_ref (key)), --remaining, user_data, NULL); + } +done: + return; +} + +void +grl_dpap_db_search (GrlDPAPDb *db, + GrlSource *source, + guint op_id, + GHRFunc predicate, + gpointer pred_user_data, + GrlSourceResultCb func, + gpointer user_data) +{ + g_assert (IS_GRL_DPAP_DB (db)); + + gint i, j, k; + guint remaining = 0; + gpointer key1, val1, key2, val2; + GHashTable *hash_tables[] = { db->priv->photos }; + + /* Use hash table to avoid duplicates */ + GHashTable *results = NULL; + GHashTableIter iter1, iter2; + + results = g_hash_table_new (g_str_hash, g_str_equal); + + /* For photos ... */ + for (i = 0; i < G_N_ELEMENTS (hash_tables); i++) { + g_hash_table_iter_init (&iter1, hash_tables[i]); + /* For each album or artist in above... */ + for (j = 0; g_hash_table_iter_next (&iter1, &key1, &val1); j++) { + if (GRL_IS_MEDIA_BOX (key1)) { + g_hash_table_iter_init (&iter2, val1); + /* For each media item in above... */ + for (k = 0; g_hash_table_iter_next (&iter2, &key2, &val2); k++) { + const char *id = grl_media_get_id (GRL_MEDIA (key2)); + /* If the predicate returns true, add to results set. */ + if (predicate (key2, val2, pred_user_data) + && ! g_hash_table_contains (results, id)) { + remaining++; + g_hash_table_insert (results, (gpointer) id, key2); + } + } + } + } + } + + /* Process results set. */ + g_hash_table_iter_init (&iter1, results); + for (i = 0; g_hash_table_iter_next (&iter1, &key1, &val1); i++) { + func (source, op_id, GRL_MEDIA (g_object_ref (val1)), --remaining, user_data, NULL); + } +} + +static void +dmap_db_interface_init (gpointer iface, gpointer data) +{ + DMAPDbIface *dpap_db = iface; + + g_assert (G_TYPE_FROM_INTERFACE (dpap_db) == DMAP_TYPE_DB); + + dpap_db->add = grl_dpap_db_add; + dpap_db->lookup_by_id = grl_dpap_db_lookup_by_id; + dpap_db->foreach = grl_dpap_db_foreach; + dpap_db->count = grl_dpap_db_count; +} + +G_DEFINE_TYPE_WITH_CODE (GrlDPAPDb, grl_dpap_db, G_TYPE_OBJECT, + G_IMPLEMENT_INTERFACE (DMAP_TYPE_DB, dmap_db_interface_init)) + +static GObject* +grl_dpap_db_constructor (GType type, guint n_construct_params, GObjectConstructParam *construct_params) +{ + GObject *object; + + object = G_OBJECT_CLASS (grl_dpap_db_parent_class)->constructor (type, n_construct_params, construct_params); + + return object; +} + +static void +grl_dpap_db_init (GrlDPAPDb *db) +{ + db->priv = GRL_DPAP_DB_GET_PRIVATE (db); + + db->priv->photos_box = g_object_new (GRL_TYPE_MEDIA_BOX, NULL); + + grl_media_set_id (GRL_MEDIA (db->priv->photos_box), PHOTOS_ID); + grl_media_set_title (GRL_MEDIA (db->priv->photos_box), PHOTOS_NAME); + + db->priv->root = g_hash_table_new_full (box_hash, box_equal, g_object_unref, (GDestroyNotify) g_hash_table_destroy); + db->priv->photos = g_hash_table_new_full (box_hash, box_equal, g_object_unref, (GDestroyNotify) g_hash_table_destroy); + + g_hash_table_insert (db->priv->root, g_object_ref (db->priv->photos_box), db->priv->photos); +} + +static void +grl_dpap_db_finalize (GObject *object) +{ + GrlDPAPDb *db = GRL_DPAP_DB (object); + + GRL_DEBUG ("Finalizing GrlDPAPDb"); + + g_object_unref (db->priv->photos_box); + + g_hash_table_destroy (db->priv->photos); +} + +static void +grl_dpap_db_class_init (GrlDPAPDbClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + g_type_class_add_private (klass, sizeof (GrlDPAPDbPrivate)); + + object_class->finalize = grl_dpap_db_finalize; + object_class->constructor = grl_dpap_db_constructor; +} |