[gimp/gimp-2-10] app: add gimp-tool-options-manager.[ch]



commit d03b31780bdf29af9cfef3b03d15f616a8c191ff
Author: Michael Natterer <mitch gimp org>
Date:   Thu Jun 28 21:41:32 2018 +0200

    app: add gimp-tool-options-manager.[ch]
    
    The way we currently manage tool options, and particularly copy things
    around for "global_brush", "global_pattern" etc. sucks, is spread
    across files, happens only on tool change, is thus buggy and leads to
    all sorts of small inconsistencies.
    
    This new manager will replace all of that stuff, and it does it in one
    place, and will keep the user context, the global paint options, and
    all tool options connected permanently, and only connect/disconnect
    things when settings change, so everything is always in a consistent
    state.
    
    Pushed for review only, nothing is used yet.
    
    (cherry picked from commit ef952f29262587a1a643dbbce407e47913b7f789)

 app/tools/Makefile.am                 |   2 +
 app/tools/gimp-tool-options-manager.c | 534 ++++++++++++++++++++++++++++++++++
 app/tools/gimp-tool-options-manager.h |  29 ++
 3 files changed, 565 insertions(+)
---
diff --git a/app/tools/Makefile.am b/app/tools/Makefile.am
index 23ce3b288b..42342440d6 100644
--- a/app/tools/Makefile.am
+++ b/app/tools/Makefile.am
@@ -15,6 +15,8 @@ noinst_LIBRARIES = libapptools.a
 libapptools_a_sources = \
        tools-enums.h                   \
        tools-types.h                   \
+       gimp-tool-options-manager.c     \
+       gimp-tool-options-manager.h     \
        gimp-tools.c                    \
        gimp-tools.h                    \
        tool_manager.c                  \
diff --git a/app/tools/gimp-tool-options-manager.c b/app/tools/gimp-tool-options-manager.c
new file mode 100644
index 0000000000..a675e5ae8d
--- /dev/null
+++ b/app/tools/gimp-tool-options-manager.c
@@ -0,0 +1,534 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimp-tool-options-manager.c
+ * Copyright (C) 2018 Michael Natterer <mitch gimp org>
+ *
+ * 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 <http://www.gnu.org/licenses/>.
+ */
+
+#include "config.h"
+
+#include <gegl.h>
+#include <gtk/gtk.h>
+
+#include "libgimpconfig/gimpconfig.h"
+
+#include "tools-types.h"
+
+#include "config/gimpcoreconfig.h"
+
+#include "core/gimp.h"
+#include "core/gimpcontext.h"
+#include "core/gimptoolinfo.h"
+
+#include "paint/gimppaintoptions.h"
+
+#include "gimp-tool-options-manager.h"
+
+
+typedef struct _GimpToolOptionsManager GimpToolOptionsManager;
+
+struct _GimpToolOptionsManager
+{
+  Gimp                *gimp;
+  GimpPaintOptions    *global_paint_options;
+  GimpContextPropMask  global_props;
+
+  GimpToolInfo        *active_tool;
+};
+
+
+static GQuark manager_quark = 0;
+
+
+/*  local function prototypes  */
+
+static GimpContextPropMask
+              tool_options_manager_get_global_props
+                                                 (GimpCoreConfig         *config);
+
+static void   tool_options_manager_global_notify (GimpCoreConfig         *config,
+                                                  const GParamSpec       *pspec,
+                                                  GimpToolOptionsManager *manager);
+static void   tool_options_manager_paint_options_notify
+                                                 (GimpPaintOptions       *src,
+                                                  const GParamSpec       *pspec,
+                                                  GimpPaintOptions       *dest);
+
+static void   tool_options_manager_copy_paint_props
+                                                 (GimpPaintOptions       *src,
+                                                  GimpPaintOptions       *dest,
+                                                  GimpContextPropMask     prop_mask);
+
+static void   tool_options_manager_tool_changed  (GimpContext            *user_context,
+                                                  GimpToolInfo           *tool_info,
+                                                  GimpToolOptionsManager *manager);
+
+
+/*  public functions  */
+
+void
+gimp_tool_options_manager_init (Gimp *gimp)
+{
+  GimpToolOptionsManager *manager;
+  GimpContext            *user_context;
+  GimpCoreConfig         *config;
+  GList                  *list;
+
+  g_return_if_fail (GIMP_IS_GIMP (gimp));
+  g_return_if_fail (manager_quark == 0);
+
+  manager_quark = g_quark_from_static_string ("gimp-tool-options-manager");
+
+  config = gimp->config;
+
+  manager = g_slice_new0 (GimpToolOptionsManager);
+
+  manager->gimp = gimp;
+
+  manager->global_paint_options =
+    g_object_new (GIMP_TYPE_PAINT_OPTIONS,
+                  "gimp", gimp,
+                  "name", "tool-options-manager-global-paint-options",
+                  NULL);
+
+  manager->global_props = tool_options_manager_get_global_props (config);
+
+  g_object_set_qdata (G_OBJECT (gimp), manager_quark, manager);
+
+  user_context = gimp_get_user_context (gimp);
+
+  for (list = gimp_get_tool_info_iter (gimp);
+       list;
+       list = g_list_next (list))
+    {
+      GimpToolInfo *tool_info = list->data;
+
+      /*  the global props that are actually used by the tool are
+       *  always shared with the user context by undefining them...
+       */
+      gimp_context_define_properties (GIMP_CONTEXT (tool_info->tool_options),
+                                      manager->global_props &
+                                      tool_info->context_props,
+                                      FALSE);
+
+      /*  ...and setting the user context as parent
+       */
+      gimp_context_set_parent (GIMP_CONTEXT (tool_info->tool_options),
+                               user_context);
+
+      /*  make sure paint tools also share their brush, dynamics,
+       *  gradient properties if the resp. context properties are
+       *  global
+       */
+      if (GIMP_IS_PAINT_OPTIONS (tool_info->tool_options))
+        {
+          g_signal_connect (tool_info->tool_options, "notify",
+                            G_CALLBACK (tool_options_manager_paint_options_notify),
+                            manager->global_paint_options);
+
+          g_signal_connect (manager->global_paint_options, "notify",
+                            G_CALLBACK (tool_options_manager_paint_options_notify),
+                            tool_info->tool_options);
+
+          tool_options_manager_copy_paint_props (manager->global_paint_options,
+                                                 GIMP_PAINT_OPTIONS (tool_info->tool_options),
+                                                 tool_info->context_props &
+                                                 manager->global_props);
+        }
+    }
+
+  g_signal_connect (gimp->config, "notify::global-brush",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+  g_signal_connect (gimp->config, "notify::global-dynamics",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+  g_signal_connect (gimp->config, "notify::global-pattern",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+  g_signal_connect (gimp->config, "notify::global-palette",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+  g_signal_connect (gimp->config, "notify::global-gradient",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+  g_signal_connect (gimp->config, "notify::global-font",
+                    G_CALLBACK (tool_options_manager_global_notify),
+                    manager);
+
+  g_signal_connect (user_context, "tool-changed",
+                    G_CALLBACK (tool_options_manager_tool_changed),
+                    manager);
+
+  tool_options_manager_tool_changed (user_context,
+                                     gimp_context_get_tool (user_context),
+                                     manager);
+}
+
+void
+gimp_tool_options_manager_exit (Gimp *gimp)
+{
+  GimpToolOptionsManager *manager;
+  GimpContext            *user_context;
+  GList                  *list;
+
+  g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+  manager = g_object_get_qdata (G_OBJECT (gimp), manager_quark);
+
+  g_return_if_fail (manager != NULL);
+
+  user_context = gimp_get_user_context (gimp);
+
+  g_signal_handlers_disconnect_by_func (user_context,
+                                        tool_options_manager_tool_changed,
+                                        manager);
+
+  g_signal_handlers_disconnect_by_func (gimp->config,
+                                        tool_options_manager_global_notify,
+                                        manager);
+
+  for (list = gimp_get_tool_info_iter (gimp);
+       list;
+       list = g_list_next (list))
+    {
+      GimpToolInfo *tool_info = list->data;
+
+      gimp_context_set_parent (GIMP_CONTEXT (tool_info->tool_options), NULL);
+
+      if (GIMP_IS_PAINT_OPTIONS (tool_info->tool_options))
+        {
+          g_signal_handlers_disconnect_by_func (tool_info->tool_options,
+                                                tool_options_manager_paint_options_notify,
+                                                manager->global_paint_options);
+
+          g_signal_handlers_disconnect_by_func (manager->global_paint_options,
+                                                tool_options_manager_paint_options_notify,
+                                                tool_info->tool_options);
+        }
+    }
+
+  g_clear_object (&manager->global_paint_options);
+
+  g_slice_free (GimpToolOptionsManager, manager);
+
+  g_object_set_qdata (G_OBJECT (gimp), manager_quark, NULL);
+}
+
+
+/*  private functions  */
+
+static GimpContextPropMask
+tool_options_manager_get_global_props (GimpCoreConfig *config)
+{
+  GimpContextPropMask global_props = 0;
+
+  /*  FG and BG are always shared between all tools  */
+  global_props |= GIMP_CONTEXT_PROP_MASK_FOREGROUND;
+  global_props |= GIMP_CONTEXT_PROP_MASK_BACKGROUND;
+
+  if (config->global_brush)
+    global_props |= GIMP_CONTEXT_PROP_MASK_BRUSH;
+  if (config->global_dynamics)
+    global_props |= GIMP_CONTEXT_PROP_MASK_DYNAMICS;
+  if (config->global_pattern)
+    global_props |= GIMP_CONTEXT_PROP_MASK_PATTERN;
+  if (config->global_palette)
+    global_props |= GIMP_CONTEXT_PROP_MASK_PALETTE;
+  if (config->global_gradient)
+    global_props |= GIMP_CONTEXT_PROP_MASK_GRADIENT;
+  if (config->global_font)
+    global_props |= GIMP_CONTEXT_PROP_MASK_FONT;
+
+  return global_props;
+}
+
+static void
+tool_options_manager_global_notify (GimpCoreConfig         *config,
+                                    const GParamSpec       *pspec,
+                                    GimpToolOptionsManager *manager)
+{
+  GimpContextPropMask  global_props;
+  GimpContextPropMask  enabled_global_props;
+  GimpContextPropMask  disabled_global_props;
+  GList               *list;
+
+  global_props = tool_options_manager_get_global_props (config);
+
+  enabled_global_props  = global_props & ~manager->global_props;
+  disabled_global_props = manager->global_props & ~global_props;
+
+  /*  if the active tool uses any of the newly enabled global props,
+   *  copy them to the user_context/global_paint_options first, so
+   *  what the user currently uses becomes the new global values
+   */
+  if (manager->active_tool &&
+      manager->active_tool->context_props & enabled_global_props)
+    {
+      GimpToolInfo *active = manager->active_tool;
+
+      gimp_context_copy_properties (GIMP_CONTEXT (active->tool_options),
+                                    gimp_get_user_context (manager->gimp),
+                                    active->context_props &
+                                    enabled_global_props);
+
+      if (GIMP_IS_PAINT_OPTIONS (active->tool_options))
+        tool_options_manager_copy_paint_props (GIMP_PAINT_OPTIONS (active->tool_options),
+                                               manager->global_paint_options,
+                                               active->context_props &
+                                               enabled_global_props);
+    }
+
+  /*  then, copy the newly enabled global props to all tool options,
+   *  and disconnect the newly disabled ones from the user context
+   */
+  for (list = gimp_get_tool_info_iter (manager->gimp);
+       list;
+       list = g_list_next (list))
+    {
+      GimpToolInfo *tool_info = list->data;
+
+      /*  defining the newly disabled ones disconnects them from the
+       *  parent user context
+       */
+      gimp_context_define_properties (GIMP_CONTEXT (tool_info->tool_options),
+                                      tool_info->context_props &
+                                      disabled_global_props,
+                                      TRUE);
+
+      /*  undefining the newly enabled ones copies the value from the
+       *  parent user context
+       */
+      gimp_context_define_properties (GIMP_CONTEXT (tool_info->tool_options),
+                                      tool_info->context_props &
+                                      enabled_global_props,
+                                      FALSE);
+
+      if (GIMP_IS_PAINT_OPTIONS (tool_info->tool_options))
+        tool_options_manager_copy_paint_props (manager->global_paint_options,
+                                               GIMP_PAINT_OPTIONS (tool_info->tool_options),
+                                               tool_info->context_props &
+                                               enabled_global_props);
+    }
+
+  manager->global_props = global_props;
+}
+
+static const gchar *brush_props[] =
+{
+  "brush-size",
+  "brush-angle",
+  "brush-aspect-ratio",
+  "brush-spacing",
+  "brush-hardness",
+  "brush-force",
+  "brush-link-size",
+  "brush-link-angle",
+  "brush-link-aspect-ratio",
+  "brush-link-spacing",
+  "brush-link-hardness",
+  "brush-lock-to-view"
+};
+
+static const gchar *dynamics_props[] =
+{
+  "dynamics-expanded",
+  "fade-reverse",
+  "fade-length",
+  "fade-unit",
+  "fade-repeat"
+};
+
+static const gchar *gradient_props[] =
+{
+  "gradient-reverse",
+  "gradient-blend-color-space"
+};
+
+static const gint max_n_props = (G_N_ELEMENTS (brush_props) +
+                                 G_N_ELEMENTS (dynamics_props) +
+                                 G_N_ELEMENTS (gradient_props));
+
+static void
+tool_options_manager_paint_options_notify (GimpPaintOptions *src,
+                                           const GParamSpec *pspec,
+                                           GimpPaintOptions *dest)
+{
+  Gimp                   *gimp   = GIMP_CONTEXT (src)->gimp;
+  GimpCoreConfig         *config = gimp->config;
+  GimpToolOptionsManager *manager;
+  GimpToolInfo           *tool_info;
+  GValue                  value = G_VALUE_INIT;
+  gint                    i;
+
+  manager = g_object_get_qdata (G_OBJECT (gimp), manager_quark);
+
+  /*  one of the options is the global one, the other is the tool's,
+   *  get the tool_info from the tool's options
+   */
+  if (manager->global_paint_options == src)
+    tool_info = gimp_context_get_tool (GIMP_CONTEXT (dest));
+  else
+    tool_info = gimp_context_get_tool (GIMP_CONTEXT (src));
+
+  if (config->global_brush &&
+      tool_info->context_props & GIMP_CONTEXT_PROP_MASK_BRUSH)
+    {
+      for (i = 0; i < G_N_ELEMENTS (brush_props); i++)
+        if (! strcmp (pspec->name, brush_props[i]))
+          goto copy_value;
+    }
+
+  if (config->global_dynamics &&
+      tool_info->context_props & GIMP_CONTEXT_PROP_MASK_DYNAMICS)
+    {
+      for (i = 0; i < G_N_ELEMENTS (dynamics_props); i++)
+        if (! strcmp (pspec->name, dynamics_props[i]))
+          goto copy_value;
+    }
+
+  if (config->global_gradient &&
+      tool_info->context_props & GIMP_CONTEXT_PROP_MASK_GRADIENT)
+    {
+      for (i = 0; i < G_N_ELEMENTS (gradient_props); i++)
+        if (! strcmp (pspec->name, gradient_props[i]))
+          goto copy_value;
+    }
+
+  return;
+
+ copy_value:
+
+  g_value_init (&value, pspec->value_type);
+
+  g_object_get_property (G_OBJECT (src), pspec->name, &value);
+
+  g_signal_handlers_block_by_func (dest,
+                                   tool_options_manager_paint_options_notify,
+                                   src);
+
+  g_object_set_property (G_OBJECT (dest), pspec->name, &value);
+
+  g_signal_handlers_unblock_by_func (dest,
+                                     tool_options_manager_paint_options_notify,
+                                     src);
+
+  g_value_unset (&value);
+}
+
+static void
+tool_options_manager_copy_paint_props (GimpPaintOptions    *src,
+                                       GimpPaintOptions    *dest,
+                                       GimpContextPropMask  prop_mask)
+{
+  const gchar *names[max_n_props];
+  GValue       values[max_n_props];
+  gint         n_props = 0;
+  gint         i;
+
+  if (prop_mask & GIMP_CONTEXT_PROP_MASK_BRUSH)
+    {
+      for (i = 0; i < G_N_ELEMENTS (brush_props); i++)
+        names[n_props++] = brush_props[i];
+    }
+
+  if (prop_mask & GIMP_CONTEXT_PROP_MASK_DYNAMICS)
+    {
+      for (i = 0; i < G_N_ELEMENTS (dynamics_props); i++)
+        names[n_props++] = dynamics_props[i];
+    }
+
+  if (prop_mask & GIMP_CONTEXT_PROP_MASK_GRADIENT)
+    {
+      for (i = 0; i < G_N_ELEMENTS (gradient_props); i++)
+        names[n_props++] = gradient_props[i];
+    }
+
+  if (n_props > 0)
+    {
+      g_object_getv (G_OBJECT (src), n_props, names, values);
+
+      g_signal_handlers_block_by_func (dest,
+                                       tool_options_manager_paint_options_notify,
+                                       src);
+
+      g_object_setv (G_OBJECT (dest), n_props, names, values);
+
+      g_signal_handlers_unblock_by_func (dest,
+                                         tool_options_manager_paint_options_notify,
+                                         src);
+
+      while (n_props--)
+        g_value_unset (&values[n_props]);
+    }
+}
+
+static void
+tool_options_manager_tool_changed (GimpContext            *user_context,
+                                   GimpToolInfo           *tool_info,
+                                   GimpToolOptionsManager *manager)
+{
+  if (tool_info == manager->active_tool)
+    return;
+
+  /*  FIXME: gimp_busy HACK
+   *  the tool manager will stop the emission, so simply return
+   */
+  if (user_context->gimp->busy)
+    return;
+
+  /*  note that in this function we only deal with non-global
+   *  properties, so we never have to copy from or to the global paint
+   *  options
+   */
+
+  if (manager->active_tool)
+    {
+      GimpToolInfo *active = manager->active_tool;
+
+      /*  disconnect the old active tool from all context properties
+       *  it uses, but are not currently global
+       */
+      gimp_context_define_properties (GIMP_CONTEXT (active->tool_options),
+                                      active->context_props &
+                                      ~manager->global_props,
+                                      TRUE);
+    }
+
+  manager->active_tool = tool_info;
+
+  if (manager->active_tool)
+    {
+      GimpToolInfo *active = manager->active_tool;
+
+      /*  copy the new tool's context properties that are not
+       *  currently global to the user context, so they get used by
+       *  everything
+       */
+      gimp_context_copy_properties (GIMP_CONTEXT (active->tool_options),
+                                    gimp_get_user_context (manager->gimp),
+                                    active->context_props &
+                                    ~manager->global_props);
+
+      /*  then, undefine these properties so the tool syncs with the
+       *  user context automatically
+       */
+      gimp_context_define_properties (GIMP_CONTEXT (active->tool_options),
+                                      active->context_props &
+                                      ~manager->global_props,
+                                      FALSE);
+    }
+}
diff --git a/app/tools/gimp-tool-options-manager.h b/app/tools/gimp-tool-options-manager.h
new file mode 100644
index 0000000000..8024e638c0
--- /dev/null
+++ b/app/tools/gimp-tool-options-manager.h
@@ -0,0 +1,29 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995 Spencer Kimball and Peter Mattis
+ *
+ * gimp-tool-options-manager.h
+ * Copyright (C) 2018 Michael Natterer <mitch gimp org>
+ *
+ * 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 <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef __GIMP_TOOL_OPTIONS_MANAGER_H__
+#define __GIMP_TOOL_OPTIONS_MANAGER_H__
+
+
+void   gimp_tool_options_manager_init (Gimp *gimp);
+void   gimp_tool_options_manager_exit (Gimp *gimp);
+
+
+#endif  /*  __GIMP_TOOL_OPTIONS_MANAGER_H__  */


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