[gimp/gimp-2-10] app: add GimpTreeProxy
- From: Ell <ell src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp/gimp-2-10] app: add GimpTreeProxy
- Date: Sat, 1 Feb 2020 11:22:45 +0000 (UTC)
commit 6c86a4c611e8b8669e27b48f3432054048fc8e5f
Author: Ell <ell_se yahoo com>
Date: Sat Feb 1 13:01:17 2020 +0200
app: add GimpTreeProxy
Add a new GimpTreeProxy container class, which proxies a
GimpViewable tree. The proxy has a dynamically-settable boolean
"flat" property, which controls if the tree hierarchy is preserved,
or if it's viewed as a flat list.
(cherry picked from commit 2caa518b196af6113df178ab55e4a9561d8fa7a6)
app/core/Makefile.am | 2 +
app/core/core-types.h | 1 +
app/core/gimptreeproxy.c | 634 +++++++++++++++++++++++++++++++++++++++++++++++
app/core/gimptreeproxy.h | 66 +++++
4 files changed, 703 insertions(+)
---
diff --git a/app/core/Makefile.am b/app/core/Makefile.am
index bda67d0837..e489c4ac3f 100644
--- a/app/core/Makefile.am
+++ b/app/core/Makefile.am
@@ -473,6 +473,8 @@ libappcore_a_sources = \
gimptoolpreset-save.h \
gimptreehandler.c \
gimptreehandler.h \
+ gimptreeproxy.c \
+ gimptreeproxy.h \
gimptriviallycancelablewaitable.c \
gimptriviallycancelablewaitable.h \
gimpuncancelablewaitable.c \
diff --git a/app/core/core-types.h b/app/core/core-types.h
index 40d4e3a560..579d7e6173 100644
--- a/app/core/core-types.h
+++ b/app/core/core-types.h
@@ -101,6 +101,7 @@ typedef struct _GimpFilterStack GimpFilterStack;
typedef struct _GimpItemStack GimpItemStack;
typedef struct _GimpLayerStack GimpLayerStack;
typedef struct _GimpTaggedContainer GimpTaggedContainer;
+typedef struct _GimpTreeProxy GimpTreeProxy;
/* not really a container */
diff --git a/app/core/gimptreeproxy.c b/app/core/gimptreeproxy.c
new file mode 100644
index 0000000000..76fad91bf2
--- /dev/null
+++ b/app/core/gimptreeproxy.c
@@ -0,0 +1,634 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimptreeproxy.c
+ * Copyright (C) 2020 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 "libgimpbase/gimpbase.h"
+
+#include "core-types.h"
+
+#include "gimpviewable.h"
+#include "gimptreeproxy.h"
+
+
+enum
+{
+ PROP_0,
+ PROP_CONTAINER,
+ PROP_FLAT
+};
+
+
+struct _GimpTreeProxyPrivate
+{
+ GimpContainer *container;
+ gboolean flat;
+};
+
+
+/* local function prototypes */
+
+static void gimp_tree_proxy_dispose (GObject *object);
+static void gimp_tree_proxy_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec);
+static void gimp_tree_proxy_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec);
+
+static void gimp_tree_proxy_container_add (GimpContainer *container,
+ GimpObject *object,
+ GimpTreeProxy *tree_proxy);
+static void gimp_tree_proxy_container_remove (GimpContainer *container,
+ GimpObject *object,
+ GimpTreeProxy *tree_proxy);
+static void gimp_tree_proxy_container_reorder (GimpContainer *container,
+ GimpObject *object,
+ gint new_index,
+ GimpTreeProxy *tree_proxy);
+static void gimp_tree_proxy_container_freeze (GimpContainer *container,
+ GimpTreeProxy *tree_proxy);
+static void gimp_tree_proxy_container_thaw (GimpContainer *container,
+ GimpTreeProxy *tree_proxy);
+
+static gint gimp_tree_proxy_add_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container,
+ gint index);
+static void gimp_tree_proxy_remove_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container);
+
+static gint gimp_tree_proxy_add_object (GimpTreeProxy *tree_proxy,
+ GimpObject *object,
+ gint index);
+static void gimp_tree_proxy_remove_object (GimpTreeProxy *tree_proxy,
+ GimpObject *object);
+
+static gint gimp_tree_proxy_find_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container);
+static gint gimp_tree_proxy_find_object (GimpContainer *container,
+ GimpObject *object);
+
+
+G_DEFINE_TYPE_WITH_PRIVATE (GimpTreeProxy, gimp_tree_proxy, GIMP_TYPE_LIST)
+
+#define parent_class gimp_tree_proxy_parent_class
+
+
+/* private functions */
+
+static void
+gimp_tree_proxy_class_init (GimpTreeProxyClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->dispose = gimp_tree_proxy_dispose;
+ object_class->set_property = gimp_tree_proxy_set_property;
+ object_class->get_property = gimp_tree_proxy_get_property;
+
+ g_object_class_install_property (object_class, PROP_CONTAINER,
+ g_param_spec_object ("container", NULL, NULL,
+ GIMP_TYPE_CONTAINER,
+ GIMP_PARAM_READWRITE));
+
+ g_object_class_install_property (object_class, PROP_FLAT,
+ g_param_spec_boolean ("flat", NULL, NULL,
+ FALSE,
+ GIMP_PARAM_READWRITE));
+}
+
+static void
+gimp_tree_proxy_init (GimpTreeProxy *tree_proxy)
+{
+ tree_proxy->priv = gimp_tree_proxy_get_instance_private (tree_proxy);
+}
+
+static void
+gimp_tree_proxy_dispose (GObject *object)
+{
+ GimpTreeProxy *tree_proxy = GIMP_TREE_PROXY (object);
+
+ gimp_tree_proxy_set_container (tree_proxy, NULL);
+
+ G_OBJECT_CLASS (parent_class)->dispose (object);;
+}
+
+static void
+gimp_tree_proxy_set_property (GObject *object,
+ guint property_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GimpTreeProxy *tree_proxy = GIMP_TREE_PROXY (object);
+
+ switch (property_id)
+ {
+ case PROP_CONTAINER:
+ gimp_tree_proxy_set_container (tree_proxy, g_value_get_object (value));
+ break;
+
+ case PROP_FLAT:
+ gimp_tree_proxy_set_flat (tree_proxy, g_value_get_boolean (value));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ break;
+ }
+}
+
+static void
+gimp_tree_proxy_get_property (GObject *object,
+ guint property_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GimpTreeProxy *tree_proxy = GIMP_TREE_PROXY (object);
+
+ switch (property_id)
+ {
+ case PROP_CONTAINER:
+ g_value_set_object (value, tree_proxy->priv->container);
+ break;
+
+ case PROP_FLAT:
+ g_value_set_boolean (value, tree_proxy->priv->flat);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
+ break;
+ }
+}
+
+static void
+gimp_tree_proxy_container_add (GimpContainer *container,
+ GimpObject *object,
+ GimpTreeProxy *tree_proxy)
+{
+ gint index;
+
+ if (tree_proxy->priv->flat)
+ {
+ index = gimp_tree_proxy_find_container (tree_proxy, container) +
+ gimp_tree_proxy_find_object (container, object);
+ }
+ else
+ {
+ index = gimp_container_get_child_index (container, object);
+ }
+
+ gimp_tree_proxy_add_object (tree_proxy, object, index);
+}
+
+static void
+gimp_tree_proxy_container_remove (GimpContainer *container,
+ GimpObject *object,
+ GimpTreeProxy *tree_proxy)
+{
+ gimp_tree_proxy_remove_object (tree_proxy, object);
+}
+
+static void
+gimp_tree_proxy_container_reorder (GimpContainer *container,
+ GimpObject *object,
+ gint new_index,
+ GimpTreeProxy *tree_proxy)
+{
+ gint index;
+
+ if (tree_proxy->priv->flat)
+ {
+ index = gimp_tree_proxy_find_container (tree_proxy, container) +
+ gimp_tree_proxy_find_object (container, object);
+
+ if (gimp_viewable_get_children (GIMP_VIEWABLE (object)))
+ {
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+
+ gimp_tree_proxy_remove_object (tree_proxy, object);
+ gimp_tree_proxy_add_object (tree_proxy, object, index);
+
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+
+ return;
+ }
+ }
+ else
+ {
+ index = new_index;
+ }
+
+ gimp_container_reorder (GIMP_CONTAINER (tree_proxy), object, index);
+}
+
+static void
+gimp_tree_proxy_container_freeze (GimpContainer *container,
+ GimpTreeProxy *tree_proxy)
+{
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+}
+
+static void
+gimp_tree_proxy_container_thaw (GimpContainer *container,
+ GimpTreeProxy *tree_proxy)
+{
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+}
+
+typedef struct
+{
+ GimpTreeProxy *tree_proxy;
+ gint index;
+} AddContainerData;
+
+static void
+gimp_tree_proxy_add_container_func (GimpObject *object,
+ AddContainerData *data)
+{
+ data->index = gimp_tree_proxy_add_object (data->tree_proxy,
+ object, data->index);
+}
+
+static gint
+gimp_tree_proxy_add_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container,
+ gint index)
+{
+ AddContainerData data;
+
+ g_signal_connect (container, "add",
+ G_CALLBACK (gimp_tree_proxy_container_add),
+ tree_proxy);
+ g_signal_connect (container, "remove",
+ G_CALLBACK (gimp_tree_proxy_container_remove),
+ tree_proxy);
+ g_signal_connect (container, "reorder",
+ G_CALLBACK (gimp_tree_proxy_container_reorder),
+ tree_proxy);
+ g_signal_connect (container, "freeze",
+ G_CALLBACK (gimp_tree_proxy_container_freeze),
+ tree_proxy);
+ g_signal_connect (container, "thaw",
+ G_CALLBACK (gimp_tree_proxy_container_thaw),
+ tree_proxy);
+
+ data.tree_proxy = tree_proxy;
+ data.index = index;
+
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+
+ gimp_container_foreach (container,
+ (GFunc) gimp_tree_proxy_add_container_func,
+ &data);
+
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+
+ return data.index;
+}
+
+static void
+gimp_tree_proxy_remove_container_func (GimpObject *object,
+ GimpTreeProxy *tree_proxy)
+{
+ gimp_tree_proxy_remove_object (tree_proxy, object);
+}
+
+static void
+gimp_tree_proxy_remove_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container)
+{
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+
+ gimp_container_foreach (container,
+ (GFunc) gimp_tree_proxy_remove_container_func,
+ tree_proxy);
+
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+
+ g_signal_handlers_disconnect_by_func (
+ container,
+ gimp_tree_proxy_container_add,
+ tree_proxy);
+ g_signal_handlers_disconnect_by_func (
+ container,
+ gimp_tree_proxy_container_remove,
+ tree_proxy);
+ g_signal_handlers_disconnect_by_func (
+ container,
+ gimp_tree_proxy_container_reorder,
+ tree_proxy);
+ g_signal_handlers_disconnect_by_func (
+ container,
+ gimp_tree_proxy_container_freeze,
+ tree_proxy);
+ g_signal_handlers_disconnect_by_func (
+ container,
+ gimp_tree_proxy_container_thaw,
+ tree_proxy);
+}
+
+static gint
+gimp_tree_proxy_add_object (GimpTreeProxy *tree_proxy,
+ GimpObject *object,
+ gint index)
+{
+ if (index == gimp_container_get_n_children (GIMP_CONTAINER (tree_proxy)))
+ index = -1;
+
+ if (tree_proxy->priv->flat)
+ {
+ GimpContainer *children;
+
+ children = gimp_viewable_get_children (GIMP_VIEWABLE (object));
+
+ if (children)
+ return gimp_tree_proxy_add_container (tree_proxy, children, index);
+ }
+
+ if (index >= 0)
+ {
+ gimp_container_insert (GIMP_CONTAINER (tree_proxy), object, index);
+
+ return index + 1;
+ }
+ else
+ {
+ gimp_container_add (GIMP_CONTAINER (tree_proxy), object);
+
+ return index;
+ }
+}
+
+static void
+gimp_tree_proxy_remove_object (GimpTreeProxy *tree_proxy,
+ GimpObject *object)
+{
+ if (tree_proxy->priv->flat)
+ {
+ GimpContainer *children;
+
+ children = gimp_viewable_get_children (GIMP_VIEWABLE (object));
+
+ if (children)
+ return gimp_tree_proxy_remove_container (tree_proxy, children);
+ }
+
+ gimp_container_remove (GIMP_CONTAINER (tree_proxy), object);
+}
+
+typedef struct
+{
+ GimpContainer *container;
+ gint index;
+} FindContainerData;
+
+static gboolean
+gimp_tree_proxy_find_container_search_func (GimpObject *object,
+ FindContainerData *data)
+{
+ GimpContainer *children;
+
+ children = gimp_viewable_get_children (GIMP_VIEWABLE (object));
+
+ if (children)
+ {
+ if (children == data->container)
+ return TRUE;
+
+ return gimp_container_search (
+ children,
+ (GimpContainerSearchFunc) gimp_tree_proxy_find_container_search_func,
+ data) != NULL;
+ }
+
+ data->index++;
+
+ return FALSE;
+}
+
+static gint
+gimp_tree_proxy_find_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container)
+{
+ FindContainerData data;
+
+ if (container == tree_proxy->priv->container)
+ return 0;
+
+ data.container = container;
+ data.index = 0;
+
+ if (gimp_container_search (
+ tree_proxy->priv->container,
+ (GimpContainerSearchFunc) gimp_tree_proxy_find_container_search_func,
+ &data))
+ {
+ return data.index;
+ }
+
+ g_return_val_if_reached (0);
+}
+
+typedef struct
+{
+ GimpObject *object;
+ gint index;
+} FindObjectData;
+
+static gboolean
+gimp_tree_proxy_find_object_search_func (GimpObject *object,
+ FindObjectData *data)
+{
+ GimpContainer *children;
+
+ if (object == data->object)
+ return TRUE;
+
+ children = gimp_viewable_get_children (GIMP_VIEWABLE (object));
+
+ if (children)
+ {
+ return gimp_container_search (
+ children,
+ (GimpContainerSearchFunc) gimp_tree_proxy_find_object_search_func,
+ data) != NULL;
+ }
+
+ data->index++;
+
+ return FALSE;
+}
+
+static gint
+gimp_tree_proxy_find_object (GimpContainer *container,
+ GimpObject *object)
+{
+ FindObjectData data;
+
+ data.object = object;
+ data.index = 0;
+
+ if (gimp_container_search (
+ container,
+ (GimpContainerSearchFunc) gimp_tree_proxy_find_object_search_func,
+ &data))
+ {
+ return data.index;
+ }
+
+ g_return_val_if_reached (0);
+}
+
+
+/* public functions */
+
+GimpContainer *
+gimp_tree_proxy_new (GType children_type)
+{
+ GTypeClass *children_class;
+
+ children_class = g_type_class_ref (children_type);
+
+ g_return_val_if_fail (G_TYPE_CHECK_CLASS_TYPE (children_class,
+ GIMP_TYPE_VIEWABLE),
+ NULL);
+
+ g_type_class_unref (children_class);
+
+ return g_object_new (GIMP_TYPE_TREE_PROXY,
+ "children-type", children_type,
+ "policy", GIMP_CONTAINER_POLICY_WEAK,
+ "append", TRUE,
+ NULL);
+}
+
+GimpContainer *
+gimp_tree_proxy_new_for_container (GimpContainer *container)
+{
+ GimpTreeProxy *tree_proxy;
+
+ g_return_val_if_fail (GIMP_IS_CONTAINER (container), NULL);
+
+ tree_proxy = GIMP_TREE_PROXY (
+ gimp_tree_proxy_new (gimp_container_get_children_type (container)));
+
+ gimp_tree_proxy_set_container (tree_proxy, container);
+
+ return GIMP_CONTAINER (tree_proxy);
+}
+
+void
+gimp_tree_proxy_set_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container)
+{
+ g_return_if_fail (GIMP_IS_TREE_PROXY (tree_proxy));
+ g_return_if_fail (container == NULL || GIMP_IS_CONTAINER (container));
+
+ if (container)
+ {
+ GTypeClass *children_class;
+
+ children_class = g_type_class_ref (
+ gimp_container_get_children_type (container));
+
+ g_return_if_fail (
+ G_TYPE_CHECK_CLASS_TYPE (
+ children_class,
+ gimp_container_get_children_type (GIMP_CONTAINER (tree_proxy))));
+
+ g_type_class_unref (children_class);
+ }
+
+ if (container != tree_proxy->priv->container)
+ {
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+
+ if (tree_proxy->priv->container)
+ {
+ gimp_tree_proxy_remove_container (tree_proxy,
+ tree_proxy->priv->container);
+ }
+
+ g_set_object (&tree_proxy->priv->container, container);
+
+ if (tree_proxy->priv->container)
+ {
+ gimp_tree_proxy_add_container (tree_proxy,
+ tree_proxy->priv->container,
+ -1);
+ }
+
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+
+ g_object_notify (G_OBJECT (tree_proxy), "container");
+ }
+}
+
+GimpContainer *
+gimp_tree_proxy_get_container (GimpTreeProxy *tree_proxy)
+{
+ g_return_val_if_fail (GIMP_IS_TREE_PROXY (tree_proxy), NULL);
+
+ return tree_proxy->priv->container;
+}
+
+void
+gimp_tree_proxy_set_flat (GimpTreeProxy *tree_proxy,
+ gboolean flat)
+{
+ g_return_if_fail (GIMP_IS_TREE_PROXY (tree_proxy));
+
+ if (flat != tree_proxy->priv->flat)
+ {
+ gimp_container_freeze (GIMP_CONTAINER (tree_proxy));
+
+ if (tree_proxy->priv->container)
+ {
+ gimp_tree_proxy_remove_container (tree_proxy,
+ tree_proxy->priv->container);
+ }
+
+ tree_proxy->priv->flat = flat;
+
+ if (tree_proxy->priv->container)
+ {
+ gimp_tree_proxy_add_container (tree_proxy,
+ tree_proxy->priv->container,
+ -1);
+ }
+
+ gimp_container_thaw (GIMP_CONTAINER (tree_proxy));
+
+ g_object_notify (G_OBJECT (tree_proxy), "flat");
+ }
+}
+
+gboolean
+gimp_tree_proxy_get_flat (GimpTreeProxy *tree_proxy)
+{
+ g_return_val_if_fail (GIMP_IS_TREE_PROXY (tree_proxy), FALSE);
+
+ return tree_proxy->priv->flat;
+}
diff --git a/app/core/gimptreeproxy.h b/app/core/gimptreeproxy.h
new file mode 100644
index 0000000000..8ae39f8173
--- /dev/null
+++ b/app/core/gimptreeproxy.h
@@ -0,0 +1,66 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimptreeproxy.h
+ * Copyright (C) 2020 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/>.
+ */
+
+#ifndef __GIMP_TREE_PROXY_H__
+#define __GIMP_TREE_PROXY_H__
+
+
+#include "gimplist.h"
+
+
+#define GIMP_TYPE_TREE_PROXY (gimp_tree_proxy_get_type ())
+#define GIMP_TREE_PROXY(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GIMP_TYPE_TREE_PROXY,
GimpTreeProxy))
+#define GIMP_TREE_PROXY_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GIMP_TYPE_TREE_PROXY,
GimpTreeProxyClass))
+#define GIMP_IS_TREE_PROXY(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GIMP_TYPE_TREE_PROXY))
+#define GIMP_IS_TREE_PROXY_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GIMP_TYPE_TREE_PROXY))
+#define GIMP_TREE_PROXY_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GIMP_TYPE_TREE_PROXY,
GimpTreeProxyClass))
+
+
+typedef struct _GimpTreeProxyPrivate GimpTreeProxyPrivate;
+typedef struct _GimpTreeProxyClass GimpTreeProxyClass;
+
+struct _GimpTreeProxy
+{
+ GimpList parent_instance;
+
+ GimpTreeProxyPrivate *priv;
+};
+
+struct _GimpTreeProxyClass
+{
+ GimpListClass parent_class;
+};
+
+
+GType gimp_tree_proxy_get_type (void) G_GNUC_CONST;
+
+GimpContainer * gimp_tree_proxy_new (GType children_type);
+GimpContainer * gimp_tree_proxy_new_for_container (GimpContainer *container);
+
+void gimp_tree_proxy_set_container (GimpTreeProxy *tree_proxy,
+ GimpContainer *container);
+GimpContainer * gimp_tree_proxy_get_container (GimpTreeProxy *tree_proxy);
+
+void gimp_tree_proxy_set_flat (GimpTreeProxy *tree_proxy,
+ gboolean flat);
+gboolean gimp_tree_proxy_get_flat (GimpTreeProxy *tree_proxy);
+
+
+#endif /* __GIMP_TREE_PROXY_H__ */
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]