diff options
Diffstat (limited to 'app/core/gimpasyncset.c')
-rw-r--r-- | app/core/gimpasyncset.c | 348 |
1 files changed, 348 insertions, 0 deletions
diff --git a/app/core/gimpasyncset.c b/app/core/gimpasyncset.c new file mode 100644 index 0000000..f1b611e --- /dev/null +++ b/app/core/gimpasyncset.c @@ -0,0 +1,348 @@ +/* GIMP - The GNU Image Manipulation Program + * Copyright (C) 1995 Spencer Kimball and Peter Mattis + * + * gimpasyncset.c + * Copyright (C) 2018 Ell + * + * 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 3 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 <https://www.gnu.org/licenses/>. + */ + +#include "config.h" + +#include <gdk-pixbuf/gdk-pixbuf.h> +#include <gegl.h> + +#include "core-types.h" + +#include "gimpasync.h" +#include "gimpasyncset.h" +#include "gimpcancelable.h" +#include "gimpwaitable.h" + + +enum +{ + PROP_0, + PROP_EMPTY +}; + + +struct _GimpAsyncSetPrivate +{ + GHashTable *asyncs; +}; + + +/* local function prototypes */ + +static void gimp_async_set_waitable_iface_init (GimpWaitableInterface *iface); + +static void gimp_async_set_cancelable_iface_init (GimpCancelableInterface *iface); + +static void gimp_async_set_dispose (GObject *object); +static void gimp_async_set_finalize (GObject *object); +static void gimp_async_set_set_property (GObject *object, + guint property_id, + const GValue *value, + GParamSpec *pspec); +static void gimp_async_set_get_property (GObject *object, + guint property_id, + GValue *value, + GParamSpec *pspec); + +static void gimp_async_set_wait (GimpWaitable *waitable); +static gboolean gimp_async_set_try_wait (GimpWaitable *waitable); +static gboolean gimp_async_set_wait_until (GimpWaitable *waitable, + gint64 end_time); + +static void gimp_async_set_cancel (GimpCancelable *cancelable); + +static void gimp_async_set_async_callback (GimpAsync *async, + GimpAsyncSet *async_set); + + +G_DEFINE_TYPE_WITH_CODE (GimpAsyncSet, gimp_async_set, G_TYPE_OBJECT, + G_ADD_PRIVATE (GimpAsyncSet) + G_IMPLEMENT_INTERFACE (GIMP_TYPE_WAITABLE, + gimp_async_set_waitable_iface_init) + G_IMPLEMENT_INTERFACE (GIMP_TYPE_CANCELABLE, + gimp_async_set_cancelable_iface_init)) + +#define parent_class gimp_async_set_parent_class + + +/* private functions */ + + +static void +gimp_async_set_class_init (GimpAsyncSetClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + + object_class->dispose = gimp_async_set_dispose; + object_class->finalize = gimp_async_set_finalize; + object_class->set_property = gimp_async_set_set_property; + object_class->get_property = gimp_async_set_get_property; + + g_object_class_install_property (object_class, PROP_EMPTY, + g_param_spec_boolean ("empty", + NULL, NULL, + FALSE, + GIMP_PARAM_READABLE)); +} + +static void +gimp_async_set_waitable_iface_init (GimpWaitableInterface *iface) +{ + iface->wait = gimp_async_set_wait; + iface->try_wait = gimp_async_set_try_wait; + iface->wait_until = gimp_async_set_wait_until; +} + +static void +gimp_async_set_cancelable_iface_init (GimpCancelableInterface *iface) +{ + iface->cancel = gimp_async_set_cancel; +} + +static void +gimp_async_set_init (GimpAsyncSet *async_set) +{ + async_set->priv = gimp_async_set_get_instance_private (async_set); + + async_set->priv->asyncs = g_hash_table_new (NULL, NULL); +} + +static void +gimp_async_set_dispose (GObject *object) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (object); + + gimp_async_set_clear (async_set); + + G_OBJECT_CLASS (parent_class)->dispose (object); +} + +static void +gimp_async_set_finalize (GObject *object) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (object); + + g_hash_table_unref (async_set->priv->asyncs); + + G_OBJECT_CLASS (parent_class)->finalize (object); +} + +static void +gimp_async_set_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); + break; + } +} + +static void +gimp_async_set_get_property (GObject *object, + guint property_id, + GValue *value, + GParamSpec *pspec) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (object); + + switch (property_id) + { + case PROP_EMPTY: + g_value_set_boolean (value, gimp_async_set_is_empty (async_set)); + break; + + default: + G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); + break; + } +} + +static void +gimp_async_set_wait (GimpWaitable *waitable) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (waitable); + + while (! gimp_async_set_is_empty (async_set)) + { + GimpAsync *async; + GHashTableIter iter; + + g_hash_table_iter_init (&iter, async_set->priv->asyncs); + + g_hash_table_iter_next (&iter, (gpointer *) &async, NULL); + + gimp_waitable_wait (GIMP_WAITABLE (async)); + } +} + +static gboolean +gimp_async_set_try_wait (GimpWaitable *waitable) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (waitable); + + while (! gimp_async_set_is_empty (async_set)) + { + GimpAsync *async; + GHashTableIter iter; + + g_hash_table_iter_init (&iter, async_set->priv->asyncs); + + g_hash_table_iter_next (&iter, (gpointer *) &async, NULL); + + if (! gimp_waitable_try_wait (GIMP_WAITABLE (async))) + return FALSE; + } + + return TRUE; +} + +static gboolean +gimp_async_set_wait_until (GimpWaitable *waitable, + gint64 end_time) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (waitable); + + while (! gimp_async_set_is_empty (async_set)) + { + GimpAsync *async; + GHashTableIter iter; + + g_hash_table_iter_init (&iter, async_set->priv->asyncs); + + g_hash_table_iter_next (&iter, (gpointer *) &async, NULL); + + if (! gimp_waitable_wait_until (GIMP_WAITABLE (async), end_time)) + return FALSE; + } + + return TRUE; +} + +static void +gimp_async_set_cancel (GimpCancelable *cancelable) +{ + GimpAsyncSet *async_set = GIMP_ASYNC_SET (cancelable); + GList *list; + + list = g_hash_table_get_keys (async_set->priv->asyncs); + + g_list_foreach (list, (GFunc) g_object_ref, NULL); + + g_list_foreach (list, (GFunc) gimp_cancelable_cancel, NULL); + + g_list_free_full (list, g_object_unref); +} + +static void +gimp_async_set_async_callback (GimpAsync *async, + GimpAsyncSet *async_set) +{ + g_hash_table_remove (async_set->priv->asyncs, async); + + if (gimp_async_set_is_empty (async_set)) + g_object_notify (G_OBJECT (async_set), "empty"); +} + + +/* public functions */ + + +GimpAsyncSet * +gimp_async_set_new (void) +{ + return g_object_new (GIMP_TYPE_ASYNC_SET, + NULL); +} + +void +gimp_async_set_add (GimpAsyncSet *async_set, + GimpAsync *async) +{ + g_return_if_fail (GIMP_IS_ASYNC_SET (async_set)); + g_return_if_fail (GIMP_IS_ASYNC (async)); + + if (g_hash_table_add (async_set->priv->asyncs, async)) + { + if (g_hash_table_size (async_set->priv->asyncs) == 1) + g_object_notify (G_OBJECT (async_set), "empty"); + + gimp_async_add_callback ( + async, + (GimpAsyncCallback) gimp_async_set_async_callback, + async_set); + } +} + +void +gimp_async_set_remove (GimpAsyncSet *async_set, + GimpAsync *async) +{ + g_return_if_fail (GIMP_IS_ASYNC_SET (async_set)); + g_return_if_fail (GIMP_IS_ASYNC (async)); + + if (g_hash_table_remove (async_set->priv->asyncs, async)) + { + gimp_async_remove_callback ( + async, + (GimpAsyncCallback) gimp_async_set_async_callback, + async_set); + + if (g_hash_table_size (async_set->priv->asyncs) == 0) + g_object_notify (G_OBJECT (async_set), "empty"); + } +} + +void +gimp_async_set_clear (GimpAsyncSet *async_set) +{ + GimpAsync *async; + GHashTableIter iter; + + g_return_if_fail (GIMP_IS_ASYNC_SET (async_set)); + + if (gimp_async_set_is_empty (async_set)) + return; + + g_hash_table_iter_init (&iter, async_set->priv->asyncs); + + while (g_hash_table_iter_next (&iter, (gpointer *) &async, NULL)) + { + gimp_async_remove_callback ( + async, + (GimpAsyncCallback) gimp_async_set_async_callback, + async_set); + } + + g_hash_table_remove_all (async_set->priv->asyncs); + + g_object_notify (G_OBJECT (async_set), "empty"); +} + +gboolean +gimp_async_set_is_empty (GimpAsyncSet *async_set) +{ + g_return_val_if_fail (GIMP_IS_ASYNC_SET (async_set), FALSE); + + return g_hash_table_size (async_set->priv->asyncs) == 0; +} |