[gtk/wip/otte/dnd: 6/10] contentprovider: Add a union content provider



commit b927755d0358d89011beb291378d99c441c15d36
Author: Benjamin Otte <otte redhat com>
Date:   Mon Feb 17 02:21:13 2020 +0100

    contentprovider: Add a union content provider

 gdk/gdkcontentproviderimpl.c | 278 ++++++++++++++++++++++++++++++++++++++++++-
 gdk/gdkcontentproviderimpl.h |   3 +
 2 files changed, 279 insertions(+), 2 deletions(-)
---
diff --git a/gdk/gdkcontentproviderimpl.c b/gdk/gdkcontentproviderimpl.c
index 05857c75a6..a7ff1b6b5c 100644
--- a/gdk/gdkcontentproviderimpl.c
+++ b/gdk/gdkcontentproviderimpl.c
@@ -1,6 +1,6 @@
 /* GDK - The GIMP Drawing Kit
  *
- * Copyright (C) 2017 Benjamin Otte <otte gnome org>
+ * Copyright (C) 2017,2020 Benjamin Otte <otte gnome org>
  *
  * This library is free software; you can redistribute it and/or
  * modify it under the terms of the GNU Lesser General Public
@@ -18,7 +18,7 @@
 
 #include "config.h"
 
-#include "gdkcontentprovider.h"
+#include "gdkcontentproviderprivate.h"
 
 #include <gobject/gvaluecollector.h>
 
@@ -164,6 +164,280 @@ gdk_content_provider_new_typed (GType type,
   return GDK_CONTENT_PROVIDER (content);
 }
 
+#define GDK_TYPE_CONTENT_PROVIDER_UNION            (gdk_content_provider_union_get_type ())
+#define GDK_CONTENT_PROVIDER_UNION(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), 
GDK_TYPE_CONTENT_PROVIDER_UNION, GdkContentProviderUnion))
+#define GDK_IS_CONTENT_PROVIDER_UNION(obj)         (G_TYPE_CHECK_INSTANCE_TYPE ((obj), 
GDK_TYPE_CONTENT_PROVIDER_UNION))
+#define GDK_CONTENT_PROVIDER_UNION_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), 
GDK_TYPE_CONTENT_PROVIDER_UNION, GdkContentProviderUnionClass))
+#define GDK_IS_CONTENT_PROVIDER_UNION_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), 
GDK_TYPE_CONTENT_PROVIDER_UNION))
+#define GDK_CONTENT_PROVIDER_UNION_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS ((obj), 
GDK_TYPE_CONTENT_PROVIDER_UNION, GdkContentProviderUnionClass))
+
+typedef struct _GdkContentProviderUnion GdkContentProviderUnion;
+typedef struct _GdkContentProviderUnionClass GdkContentProviderUnionClass;
+
+struct _GdkContentProviderUnion
+{
+  GdkContentProvider parent;
+
+  GdkContentProvider **providers;
+  gsize n_providers;
+};
+
+struct _GdkContentProviderUnionClass
+{
+  GdkContentProviderClass parent_class;
+};
+
+GType gdk_content_provider_union_get_type (void) G_GNUC_CONST;
+
+G_DEFINE_TYPE (GdkContentProviderUnion, gdk_content_provider_union, GDK_TYPE_CONTENT_PROVIDER)
+
+static void
+gdk_content_provider_union_attach_clipboard (GdkContentProvider *provider,
+                                             GdkClipboard       *clipboard)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  gsize i;
+
+  for (i = 0; i < self->n_providers; i++)
+    gdk_content_provider_attach_clipboard (self->providers[i], clipboard);
+}
+
+static void
+gdk_content_provider_union_detach_clipboard (GdkContentProvider *provider,
+                                             GdkClipboard       *clipboard)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  gsize i;
+
+  for (i = 0; i < self->n_providers; i++)
+    gdk_content_provider_detach_clipboard (self->providers[i], clipboard);
+}
+
+static GdkContentFormats *
+gdk_content_provider_union_ref_formats (GdkContentProvider *provider)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  GdkContentFormatsBuilder *builder;
+  gsize i;
+
+  builder = gdk_content_formats_builder_new ();
+
+  for (i = 0; i < self->n_providers; i++)
+    {
+      GdkContentFormats *formats = gdk_content_provider_ref_formats (self->providers[i]);
+      gdk_content_formats_builder_add_formats (builder, formats);
+      gdk_content_formats_unref (formats);
+    }
+
+  return gdk_content_formats_builder_free_to_formats (builder);
+}
+
+static GdkContentFormats *
+gdk_content_provider_union_ref_storable_formats (GdkContentProvider *provider)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  GdkContentFormatsBuilder *builder;
+  gsize i;
+
+  builder = gdk_content_formats_builder_new ();
+
+  for (i = 0; i < self->n_providers; i++)
+    {
+      GdkContentFormats *formats = gdk_content_provider_ref_storable_formats (self->providers[i]);
+      gdk_content_formats_builder_add_formats (builder, formats);
+      gdk_content_formats_unref (formats);
+    }
+
+  return gdk_content_formats_builder_free_to_formats (builder);
+}
+
+static void
+gdk_content_provider_union_write_mime_type_done (GObject      *source_object,
+                                                 GAsyncResult *res,
+                                                 gpointer      data)
+{
+  GTask *task = data;
+  GError *error = NULL;
+
+  if (!gdk_content_provider_write_mime_type_finish (GDK_CONTENT_PROVIDER (source_object), res, &error))
+    {
+      g_task_return_error (task, error);
+    }
+  else
+    {
+      g_task_return_boolean (task, TRUE);
+    }
+
+  g_object_unref (task);
+}
+
+static void
+gdk_content_provider_union_write_mime_type_async (GdkContentProvider     *provider,
+                                                  const char             *mime_type,
+                                                  GOutputStream          *stream,
+                                                  int                     io_priority,
+                                                  GCancellable           *cancellable,
+                                                  GAsyncReadyCallback     callback,
+                                                  gpointer                user_data)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  GTask *task;
+  gsize i;
+
+  task = g_task_new (self, cancellable, callback, user_data);
+  g_task_set_priority (task, io_priority);
+  g_task_set_source_tag (task, gdk_content_provider_union_write_mime_type_async);
+
+  for (i = 0; i < self->n_providers; i++)
+    {
+      GdkContentFormats *formats = gdk_content_provider_ref_formats (self->providers[i]);
+
+      if (gdk_content_formats_contain_mime_type (formats, mime_type))
+        {
+          gdk_content_provider_write_mime_type_async (self->providers[i],
+                                                      mime_type,
+                                                      stream,
+                                                      io_priority,
+                                                      cancellable,
+                                                      gdk_content_provider_union_write_mime_type_done,
+                                                      task);
+          gdk_content_formats_unref (formats);
+        }
+      gdk_content_formats_unref (formats);
+    }
+
+  g_task_return_new_error (task, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED,
+                           _("Cannot provide contents as ā€œ%sā€"), mime_type);
+  g_object_unref (task);
+}
+
+static gboolean
+gdk_content_provider_union_write_mime_type_finish (GdkContentProvider  *provider,
+                                                   GAsyncResult        *result,
+                                                   GError             **error)
+{
+  g_return_val_if_fail (g_task_is_valid (result, provider), FALSE);
+  g_return_val_if_fail (g_task_get_source_tag (G_TASK (result)) == 
gdk_content_provider_union_write_mime_type_async, FALSE);
+
+  return g_task_propagate_boolean (G_TASK (result), error);
+}
+
+static gboolean
+gdk_content_provider_union_get_value (GdkContentProvider  *provider,
+                                      GValue              *value,
+                                      GError             **error)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (provider);
+  gsize i;
+
+  for (i = 0; i < self->n_providers; i++)
+    {
+      GError *provider_error = NULL;
+
+      if (gdk_content_provider_get_value (self->providers[i], value, &provider_error))
+        return TRUE;
+
+      if (!g_error_matches (provider_error, G_IO_ERROR, G_IO_ERROR_NOT_SUPPORTED))
+        {
+          g_propagate_error (error, provider_error);
+          return FALSE;
+        }
+
+      g_clear_error (&provider_error);
+    }
+
+  return FALSE;
+}
+
+static void
+gdk_content_provider_union_finalize (GObject *object)
+{
+  GdkContentProviderUnion *self = GDK_CONTENT_PROVIDER_UNION (object);
+  gsize i;
+
+  for (i = 0; i < self->n_providers; i++)
+    {
+      g_signal_handlers_disconnect_by_func (self->providers[i], gdk_content_provider_content_changed, self);
+      g_object_unref (self->providers[i]);
+    }
+
+  g_free (self->providers);
+
+  G_OBJECT_CLASS (gdk_content_provider_union_parent_class)->finalize (object);
+}
+
+static void
+gdk_content_provider_union_class_init (GdkContentProviderUnionClass *class)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (class);
+  GdkContentProviderClass *provider_class = GDK_CONTENT_PROVIDER_CLASS (class);
+
+  object_class->finalize = gdk_content_provider_union_finalize;
+
+  provider_class->attach_clipboard = gdk_content_provider_union_attach_clipboard;
+  provider_class->detach_clipboard = gdk_content_provider_union_detach_clipboard;
+  provider_class->ref_formats = gdk_content_provider_union_ref_formats;
+  provider_class->ref_storable_formats = gdk_content_provider_union_ref_storable_formats;
+  provider_class->write_mime_type_async = gdk_content_provider_union_write_mime_type_async;
+  provider_class->write_mime_type_finish = gdk_content_provider_union_write_mime_type_finish;
+  provider_class->get_value = gdk_content_provider_union_get_value;
+}
+
+static void
+gdk_content_provider_union_init (GdkContentProviderUnion *self)
+{
+}
+
+/**
+ * gdk_content_provider_new_union:
+ * @providers: (nullable) (array length=n_providers) (transfer elements):
+ *     The #GdkContentProviders to present the union of
+ * @n_providers: the number of providers
+ *
+ * Creates a content provider that represents all the given @providers.
+ *
+ * Whenever data needs to be written, the union provider will try the given
+ * @providers in the given order and the first one supporting a format will
+ * be chosen to provide it.
+ *
+ * This allows an easy way to support providing data in different formats.
+ * For example, an image may be provided by its file and by the iamge
+ * contents with a call such as
+ * |[<!-- language="C" -->
+ * gdk_content_provider_new_union ((GdkContentProvider *[2]) {
+ *                                   gdk_content_provider_new_typed (G_TYPE_FILE, file),
+ *                                   gdk_content_provider_new_typed (G_TYPE_TEXTURE, texture)
+ *                                 }, 2);
+ * ]|
+ *
+ *
+ * Returns: a new #GdkContentProvider
+ **/
+GdkContentProvider *
+gdk_content_provider_new_union (GdkContentProvider **providers,
+                                gsize                n_providers)
+{
+  GdkContentProviderUnion *result;
+  gsize i;
+
+  g_return_val_if_fail (providers != NULL || n_providers == 0, NULL);
+
+  result = g_object_new (GDK_TYPE_CONTENT_PROVIDER_UNION, NULL);
+
+  result->n_providers = n_providers;
+  result->providers = g_memdup (providers, sizeof (GdkContentProvider *) * n_providers);
+
+  for (i = 0; i < n_providers; i++)
+    {
+      g_signal_connect_swapped (result->providers[i],
+                                "content-changed",
+                                G_CALLBACK (gdk_content_provider_content_changed),
+                                result);
+    }
+
+  return GDK_CONTENT_PROVIDER (result);
+}
+
 #define GDK_TYPE_CONTENT_PROVIDER_BYTES            (gdk_content_provider_bytes_get_type ())
 #define GDK_CONTENT_PROVIDER_BYTES(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), 
GDK_TYPE_CONTENT_PROVIDER_BYTES, GdkContentProviderBytes))
 #define GDK_IS_CONTENT_PROVIDER_BYTES(obj)         (G_TYPE_CHECK_INSTANCE_TYPE ((obj), 
GDK_TYPE_CONTENT_PROVIDER_BYTES))
diff --git a/gdk/gdkcontentproviderimpl.h b/gdk/gdkcontentproviderimpl.h
index 907ed5015c..983465717c 100644
--- a/gdk/gdkcontentproviderimpl.h
+++ b/gdk/gdkcontentproviderimpl.h
@@ -35,6 +35,9 @@ GDK_AVAILABLE_IN_ALL
 GdkContentProvider *    gdk_content_provider_new_typed                  (GType                   type,
                                                                          ...);
 GDK_AVAILABLE_IN_ALL
+GdkContentProvider *    gdk_content_provider_new_union                  (GdkContentProvider    **providers,
+                                                                         gsize                   
n_providers);
+GDK_AVAILABLE_IN_ALL
 GdkContentProvider *    gdk_content_provider_new_for_bytes              (const char             *mime_type,
                                                                          GBytes                 *bytes);
 


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]