summaryrefslogtreecommitdiffstats
path: root/app/tools/gimp-tools.c
diff options
context:
space:
mode:
authorDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-27 16:23:22 +0000
committerDaniel Baumann <daniel.baumann@progress-linux.org>2024-04-27 16:23:22 +0000
commite42129241681dde7adae7d20697e7b421682fbb4 (patch)
treeaf1fe815a5e639e68e59fabd8395ec69458b3e5e /app/tools/gimp-tools.c
parentInitial commit. (diff)
downloadgimp-upstream/2.10.22.tar.xz
gimp-upstream/2.10.22.zip
Adding upstream version 2.10.22.upstream/2.10.22upstream
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to '')
-rw-r--r--app/tools/gimp-tools.c831
1 files changed, 831 insertions, 0 deletions
diff --git a/app/tools/gimp-tools.c b/app/tools/gimp-tools.c
new file mode 100644
index 0000000..7c35be9
--- /dev/null
+++ b/app/tools/gimp-tools.c
@@ -0,0 +1,831 @@
+/* GIMP - The GNU Image Manipulation Program
+ * Copyright (C) 1995-2001 Spencer Kimball, Peter Mattis and others
+ *
+ * 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 <gegl.h>
+#include <gtk/gtk.h>
+
+#include "libgimpbase/gimpbase.h"
+#include "libgimpconfig/gimpconfig.h"
+#include "libgimpwidgets/gimpwidgets.h"
+
+#include "tools-types.h"
+
+#include "widgets/gimpwidgets-utils.h"
+
+#include "core/gimp.h"
+#include "core/gimp-contexts.h"
+#include "core/gimp-internal-data.h"
+#include "core/gimpcontext.h"
+#include "core/gimplist.h"
+#include "core/gimptoolgroup.h"
+#include "core/gimptoolinfo.h"
+#include "core/gimptooloptions.h"
+
+#include "gimp-tool-options-manager.h"
+#include "gimp-tools.h"
+#include "gimptooloptions-gui.h"
+#include "tool_manager.h"
+
+#include "gimpairbrushtool.h"
+#include "gimpaligntool.h"
+#include "gimpbrightnesscontrasttool.h"
+#include "gimpbucketfilltool.h"
+#include "gimpbycolorselecttool.h"
+#include "gimpcagetool.h"
+#include "gimpclonetool.h"
+#include "gimpcolorpickertool.h"
+#include "gimpconvolvetool.h"
+#include "gimpcroptool.h"
+#include "gimpcurvestool.h"
+#include "gimpdodgeburntool.h"
+#include "gimpellipseselecttool.h"
+#include "gimperasertool.h"
+#include "gimpfliptool.h"
+#include "gimpfreeselecttool.h"
+#include "gimpforegroundselecttool.h"
+#include "gimpfuzzyselecttool.h"
+#include "gimpgegltool.h"
+#include "gimpgradienttool.h"
+#include "gimphandletransformtool.h"
+#include "gimphealtool.h"
+#include "gimpinktool.h"
+#include "gimpiscissorstool.h"
+#include "gimplevelstool.h"
+#include "gimpoperationtool.h"
+#include "gimpmagnifytool.h"
+#include "gimpmeasuretool.h"
+#include "gimpmovetool.h"
+#include "gimpmybrushtool.h"
+#include "gimpnpointdeformationtool.h"
+#include "gimpoffsettool.h"
+#include "gimppaintbrushtool.h"
+#include "gimppenciltool.h"
+#include "gimpperspectiveclonetool.h"
+#include "gimpperspectivetool.h"
+#include "gimpthresholdtool.h"
+#include "gimprectangleselecttool.h"
+#include "gimprotatetool.h"
+#include "gimpseamlessclonetool.h"
+#include "gimpscaletool.h"
+#include "gimpsheartool.h"
+#include "gimpsmudgetool.h"
+#include "gimptexttool.h"
+#include "gimptransform3dtool.h"
+#include "gimpunifiedtransformtool.h"
+#include "gimpvectortool.h"
+#include "gimpwarptool.h"
+
+#include "gimp-intl.h"
+
+
+#define TOOL_RC_FILE_VERSION 1
+
+
+/* local function prototypes */
+
+static void gimp_tools_register (GType tool_type,
+ GType tool_options_type,
+ GimpToolOptionsGUIFunc options_gui_func,
+ GimpContextPropMask context_props,
+ const gchar *identifier,
+ const gchar *label,
+ const gchar *tooltip,
+ const gchar *menu_label,
+ const gchar *menu_accel,
+ const gchar *help_domain,
+ const gchar *help_data,
+ const gchar *icon_name,
+ gpointer data);
+
+static void gimp_tools_copy_structure (Gimp *gimp,
+ GimpContainer *src_container,
+ GimpContainer *dest_container,
+ GHashTable *tools);
+
+/* private variables */
+
+static GBinding *toolbox_groups_binding = NULL;
+static gboolean tool_options_deleted = FALSE;
+
+
+/* public functions */
+
+void
+gimp_tools_init (Gimp *gimp)
+{
+ GimpToolRegisterFunc register_funcs[] =
+ {
+ /* selection tools */
+
+ gimp_rectangle_select_tool_register,
+ gimp_ellipse_select_tool_register,
+ gimp_free_select_tool_register,
+ gimp_fuzzy_select_tool_register,
+ gimp_by_color_select_tool_register,
+ gimp_iscissors_tool_register,
+ gimp_foreground_select_tool_register,
+
+ /* path tool */
+
+ gimp_vector_tool_register,
+
+ /* non-modifying tools */
+
+ gimp_color_picker_tool_register,
+ gimp_magnify_tool_register,
+ gimp_measure_tool_register,
+
+ /* transform tools */
+
+ gimp_move_tool_register,
+ gimp_align_tool_register,
+ gimp_crop_tool_register,
+ gimp_unified_transform_tool_register,
+ gimp_rotate_tool_register,
+ gimp_scale_tool_register,
+ gimp_shear_tool_register,
+ gimp_handle_transform_tool_register,
+ gimp_perspective_tool_register,
+ gimp_transform_3d_tool_register,
+ gimp_flip_tool_register,
+ gimp_cage_tool_register,
+ gimp_warp_tool_register,
+ gimp_n_point_deformation_tool_register,
+
+ /* paint tools */
+
+ gimp_seamless_clone_tool_register,
+ gimp_text_tool_register,
+ gimp_bucket_fill_tool_register,
+ gimp_gradient_tool_register,
+ gimp_pencil_tool_register,
+ gimp_paintbrush_tool_register,
+ gimp_eraser_tool_register,
+ gimp_airbrush_tool_register,
+ gimp_ink_tool_register,
+ gimp_mybrush_tool_register,
+ gimp_clone_tool_register,
+ gimp_heal_tool_register,
+ gimp_perspective_clone_tool_register,
+ gimp_convolve_tool_register,
+ gimp_smudge_tool_register,
+ gimp_dodge_burn_tool_register,
+
+ /* filter tools */
+
+ gimp_brightness_contrast_tool_register,
+ gimp_threshold_tool_register,
+ gimp_levels_tool_register,
+ gimp_curves_tool_register,
+ gimp_offset_tool_register,
+ gimp_gegl_tool_register,
+ gimp_operation_tool_register
+ };
+
+ gint i;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+ gimp_tool_options_create_folder ();
+
+ gimp_container_freeze (gimp->tool_info_list);
+
+ for (i = 0; i < G_N_ELEMENTS (register_funcs); i++)
+ {
+ register_funcs[i] (gimp_tools_register, gimp);
+ }
+
+ gimp_container_thaw (gimp->tool_info_list);
+
+ gimp_tool_options_manager_init (gimp);
+
+ tool_manager_init (gimp);
+
+ toolbox_groups_binding = g_object_bind_property (
+ gimp->config, "toolbox-groups",
+ gimp->tool_item_ui_list, "flat",
+ G_BINDING_INVERT_BOOLEAN |
+ G_BINDING_SYNC_CREATE);
+}
+
+void
+gimp_tools_exit (Gimp *gimp)
+{
+ GList *list;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+ g_clear_object (&toolbox_groups_binding);
+
+ tool_manager_exit (gimp);
+
+ gimp_tool_options_manager_exit (gimp);
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = list->data;
+
+ gimp_tools_set_tool_options_gui (tool_info->tool_options, NULL);
+ }
+}
+
+void
+gimp_tools_restore (Gimp *gimp)
+{
+ GimpObject *object;
+ GList *list;
+ GError *error = NULL;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+ /* restore tool order */
+ gimp_tools_reset (gimp, gimp->tool_item_list, TRUE);
+
+ /* make the generic operation tool invisible by default */
+ object = gimp_container_get_child_by_name (gimp->tool_info_list,
+ "gimp-operation-tool");
+ if (object)
+ g_object_set (object, "visible", FALSE, NULL);
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = GIMP_TOOL_INFO (list->data);
+
+ /* get default values from prefs (see bug #120832) */
+ gimp_config_reset (GIMP_CONFIG (tool_info->tool_options));
+ }
+
+ if (! gimp_contexts_load (gimp, &error))
+ {
+ gimp_message_literal (gimp, NULL, GIMP_MESSAGE_WARNING, error->message);
+ g_clear_error (&error);
+ }
+
+ if (! gimp_internal_data_load (gimp, &error))
+ {
+ gimp_message_literal (gimp, NULL, GIMP_MESSAGE_WARNING, error->message);
+ g_clear_error (&error);
+ }
+
+ /* make sure there is always a tool active, so broken config files
+ * can't leave us with no initial tool
+ */
+ if (! gimp_context_get_tool (gimp_get_user_context (gimp)))
+ {
+ gimp_context_set_tool (gimp_get_user_context (gimp),
+ gimp_get_tool_info_iter (gimp)->data);
+ }
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = GIMP_TOOL_INFO (list->data);
+ GimpToolOptionsGUIFunc options_gui_func;
+
+ /* copy all context properties except those the tool actually
+ * uses, because the subsequent deserialize() on the tool
+ * options will only set the properties that were set to
+ * non-default values at the time of saving, and we want to
+ * keep these default values as if they have been saved.
+ * (see bug #541586).
+ */
+ gimp_context_copy_properties (gimp_get_user_context (gimp),
+ GIMP_CONTEXT (tool_info->tool_options),
+ GIMP_CONTEXT_PROP_MASK_ALL &~
+ (tool_info->context_props |
+ GIMP_CONTEXT_PROP_MASK_TOOL |
+ GIMP_CONTEXT_PROP_MASK_PAINT_INFO));
+
+ gimp_tool_options_deserialize (tool_info->tool_options, NULL);
+
+ options_gui_func = g_object_get_data (G_OBJECT (tool_info),
+ "gimp-tool-options-gui-func");
+
+ if (! options_gui_func)
+ options_gui_func = gimp_tool_options_empty_gui;
+
+ gimp_tools_set_tool_options_gui_func (tool_info->tool_options,
+ options_gui_func);
+ }
+}
+
+void
+gimp_tools_save (Gimp *gimp,
+ gboolean save_tool_options,
+ gboolean always_save)
+{
+ GimpConfigWriter *writer;
+ GFile *file;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+
+ if (save_tool_options && (! tool_options_deleted || always_save))
+ {
+ GList *list;
+ GError *error = NULL;
+
+ if (! gimp_contexts_save (gimp, &error))
+ {
+ gimp_message_literal (gimp, NULL, GIMP_MESSAGE_WARNING,
+ error->message);
+ g_clear_error (&error);
+ }
+
+ if (! gimp_internal_data_save (gimp, &error))
+ {
+ gimp_message_literal (gimp, NULL, GIMP_MESSAGE_WARNING,
+ error->message);
+ g_clear_error (&error);
+ }
+
+ gimp_tool_options_create_folder ();
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = GIMP_TOOL_INFO (list->data);
+
+ gimp_tool_options_serialize (tool_info->tool_options, NULL);
+ }
+ }
+
+ file = gimp_directory_file ("toolrc", NULL);
+
+ if (gimp->be_verbose)
+ g_print ("Writing '%s'\n", gimp_file_get_utf8_name (file));
+
+ writer = gimp_config_writer_new_gfile (file, TRUE, "GIMP toolrc", NULL);
+
+ if (writer)
+ {
+ gimp_tools_serialize (gimp, gimp->tool_item_list, writer);
+
+ gimp_config_writer_finish (writer, "end of toolrc", NULL);
+ }
+
+ g_object_unref (file);
+}
+
+gboolean
+gimp_tools_clear (Gimp *gimp,
+ GError **error)
+{
+ GList *list;
+ gboolean success = TRUE;
+
+ g_return_val_if_fail (GIMP_IS_GIMP (gimp), FALSE);
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list && success;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = GIMP_TOOL_INFO (list->data);
+
+ success = gimp_tool_options_delete (tool_info->tool_options, NULL);
+ }
+
+ if (success)
+ success = gimp_contexts_clear (gimp, error);
+
+ if (success)
+ success = gimp_internal_data_clear (gimp, error);
+
+ if (success)
+ tool_options_deleted = TRUE;
+
+ return success;
+}
+
+gboolean
+gimp_tools_serialize (Gimp *gimp,
+ GimpContainer *container,
+ GimpConfigWriter *writer)
+{
+ g_return_val_if_fail (GIMP_IS_GIMP (gimp), FALSE);
+ g_return_val_if_fail (GIMP_IS_CONTAINER (container), FALSE);
+
+ gimp_config_writer_open (writer, "file-version");
+ gimp_config_writer_printf (writer, "%d", TOOL_RC_FILE_VERSION);
+ gimp_config_writer_close (writer);
+
+ gimp_config_writer_linefeed (writer);
+
+ return gimp_config_serialize (GIMP_CONFIG (container), writer, NULL);
+}
+
+gboolean
+gimp_tools_deserialize (Gimp *gimp,
+ GimpContainer *container,
+ GScanner *scanner)
+{
+ enum
+ {
+ FILE_VERSION = 1
+ };
+
+ GimpContainer *src_container;
+ GTokenType token;
+ guint scope_id;
+ guint old_scope_id;
+ gint file_version = 0;
+ gboolean result = FALSE;
+
+ scope_id = g_type_qname (GIMP_TYPE_TOOL_GROUP);
+ old_scope_id = g_scanner_set_scope (scanner, scope_id);
+
+ g_scanner_scope_add_symbol (scanner, scope_id,
+ "file-version",
+ GINT_TO_POINTER (FILE_VERSION));
+
+ token = G_TOKEN_LEFT_PAREN;
+
+ while (g_scanner_peek_next_token (scanner) == token &&
+ (token != G_TOKEN_LEFT_PAREN ||
+ ! file_version))
+ {
+ token = g_scanner_get_next_token (scanner);
+
+ switch (token)
+ {
+ case G_TOKEN_LEFT_PAREN:
+ token = G_TOKEN_SYMBOL;
+ break;
+
+ case G_TOKEN_SYMBOL:
+ switch (GPOINTER_TO_INT (scanner->value.v_symbol))
+ {
+ case FILE_VERSION:
+ token = G_TOKEN_INT;
+ if (gimp_scanner_parse_int (scanner, &file_version))
+ token = G_TOKEN_RIGHT_PAREN;
+ break;
+ }
+ break;
+
+ case G_TOKEN_RIGHT_PAREN:
+ token = G_TOKEN_LEFT_PAREN;
+ break;
+
+ default:
+ break;
+ }
+ }
+
+ g_scanner_set_scope (scanner, old_scope_id);
+
+ if (token != G_TOKEN_LEFT_PAREN)
+ {
+ g_scanner_get_next_token (scanner);
+ g_scanner_unexp_token (scanner, token, NULL, NULL, NULL,
+ _("fatal parse error"), TRUE);
+
+ return FALSE;
+ }
+ else if (file_version != TOOL_RC_FILE_VERSION)
+ {
+ g_scanner_error (scanner, "wrong toolrc file format version");
+
+ return FALSE;
+ }
+
+ gimp_container_freeze (container);
+
+ /* make sure the various GimpToolItem types are registered */
+ g_type_class_unref (g_type_class_ref (GIMP_TYPE_TOOL_GROUP));
+ g_type_class_unref (g_type_class_ref (GIMP_TYPE_TOOL_INFO));
+
+ gimp_container_clear (container);
+
+ src_container = g_object_new (GIMP_TYPE_LIST,
+ "children-type", GIMP_TYPE_TOOL_ITEM,
+ "append", TRUE,
+ NULL);
+
+ if (gimp_config_deserialize (GIMP_CONFIG (src_container),
+ scanner, 0, NULL))
+ {
+ GHashTable *tools;
+ GList *list;
+
+ result = TRUE;
+
+ tools = g_hash_table_new (g_direct_hash, g_direct_equal);
+
+ gimp_tools_copy_structure (gimp, src_container, container, tools);
+
+ for (list = gimp_get_tool_info_iter (gimp);
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolInfo *tool_info = list->data;
+
+ if (! tool_info->hidden && ! g_hash_table_contains (tools, tool_info))
+ {
+ if (tool_info->experimental)
+ {
+ /* if an experimental tool is not in the file, just add it to
+ * the tool-item list.
+ */
+ gimp_container_add (container, GIMP_OBJECT (tool_info));
+ }
+ else
+ {
+ /* otherwise, it means we added a new stable tool. this must
+ * be the user toolrc file; rejct it, so that we fall back to
+ * the default toolrc file, which should contain the missing
+ * tool.
+ */
+ g_scanner_error (scanner, "missing tools in toolrc file");
+
+ result = FALSE;
+
+ break;
+ }
+ }
+ }
+
+ g_hash_table_unref (tools);
+ }
+
+ g_object_unref (src_container);
+
+ gimp_container_thaw (container);
+
+ return result;
+}
+
+void
+gimp_tools_reset (Gimp *gimp,
+ GimpContainer *container,
+ gboolean user_toolrc)
+{
+ GList *files = NULL;
+ GList *list;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+ g_return_if_fail (GIMP_IS_CONTAINER (container));
+
+ if (user_toolrc)
+ files = g_list_prepend (files, gimp_directory_file ("toolrc", NULL));
+ files = g_list_prepend (files, gimp_sysconf_directory_file ("toolrc", NULL));
+
+ files = g_list_reverse (files);
+
+ gimp_container_freeze (container);
+
+ gimp_container_clear (container);
+
+ for (list = files; list; list = g_list_next (list))
+ {
+ GScanner *scanner;
+ GFile *file = list->data;
+ GError *error = NULL;
+
+ if (gimp->be_verbose)
+ g_print ("Parsing '%s'\n", gimp_file_get_utf8_name (file));
+
+ scanner = gimp_scanner_new_gfile (file, &error);
+
+ if (scanner && gimp_tools_deserialize (gimp, container, scanner))
+ {
+ gimp_scanner_destroy (scanner);
+
+ break;
+ }
+ else
+ {
+ if (error->code != G_IO_ERROR_NOT_FOUND)
+ {
+ gimp_message_literal (gimp, NULL,
+ GIMP_MESSAGE_WARNING, error->message);
+ }
+
+ g_clear_error (&error);
+
+ gimp_container_clear (container);
+ }
+
+ g_clear_pointer (&scanner, gimp_scanner_destroy);
+ }
+
+ g_list_free_full (files, g_object_unref);
+
+ if (gimp_container_is_empty (container))
+ {
+ if (gimp->be_verbose)
+ g_print ("Using default tool order\n");
+
+ gimp_tools_copy_structure (gimp, gimp->tool_info_list, container, NULL);
+ }
+
+ gimp_container_thaw (container);
+}
+
+
+/* private functions */
+
+static void
+gimp_tools_register (GType tool_type,
+ GType tool_options_type,
+ GimpToolOptionsGUIFunc options_gui_func,
+ GimpContextPropMask context_props,
+ const gchar *identifier,
+ const gchar *label,
+ const gchar *tooltip,
+ const gchar *menu_label,
+ const gchar *menu_accel,
+ const gchar *help_domain,
+ const gchar *help_data,
+ const gchar *icon_name,
+ gpointer data)
+{
+ Gimp *gimp = (Gimp *) data;
+ GimpToolInfo *tool_info;
+ const gchar *paint_core_name;
+ gboolean visible;
+
+ g_return_if_fail (GIMP_IS_GIMP (gimp));
+ g_return_if_fail (g_type_is_a (tool_type, GIMP_TYPE_TOOL));
+ g_return_if_fail (tool_options_type == G_TYPE_NONE ||
+ g_type_is_a (tool_options_type, GIMP_TYPE_TOOL_OPTIONS));
+
+ if (tool_options_type == G_TYPE_NONE)
+ tool_options_type = GIMP_TYPE_TOOL_OPTIONS;
+
+ if (tool_type == GIMP_TYPE_PENCIL_TOOL)
+ {
+ paint_core_name = "gimp-pencil";
+ }
+ else if (tool_type == GIMP_TYPE_PAINTBRUSH_TOOL)
+ {
+ paint_core_name = "gimp-paintbrush";
+ }
+ else if (tool_type == GIMP_TYPE_ERASER_TOOL)
+ {
+ paint_core_name = "gimp-eraser";
+ }
+ else if (tool_type == GIMP_TYPE_AIRBRUSH_TOOL)
+ {
+ paint_core_name = "gimp-airbrush";
+ }
+ else if (tool_type == GIMP_TYPE_CLONE_TOOL)
+ {
+ paint_core_name = "gimp-clone";
+ }
+ else if (tool_type == GIMP_TYPE_HEAL_TOOL)
+ {
+ paint_core_name = "gimp-heal";
+ }
+ else if (tool_type == GIMP_TYPE_PERSPECTIVE_CLONE_TOOL)
+ {
+ paint_core_name = "gimp-perspective-clone";
+ }
+ else if (tool_type == GIMP_TYPE_CONVOLVE_TOOL)
+ {
+ paint_core_name = "gimp-convolve";
+ }
+ else if (tool_type == GIMP_TYPE_SMUDGE_TOOL)
+ {
+ paint_core_name = "gimp-smudge";
+ }
+ else if (tool_type == GIMP_TYPE_DODGE_BURN_TOOL)
+ {
+ paint_core_name = "gimp-dodge-burn";
+ }
+ else if (tool_type == GIMP_TYPE_INK_TOOL)
+ {
+ paint_core_name = "gimp-ink";
+ }
+ else if (tool_type == GIMP_TYPE_MYBRUSH_TOOL)
+ {
+ paint_core_name = "gimp-mybrush";
+ }
+ else
+ {
+ paint_core_name = "gimp-paintbrush";
+ }
+
+ tool_info = gimp_tool_info_new (gimp,
+ tool_type,
+ tool_options_type,
+ context_props,
+ identifier,
+ label,
+ tooltip,
+ menu_label,
+ menu_accel,
+ help_domain,
+ help_data,
+ paint_core_name,
+ icon_name);
+
+ visible = (! g_type_is_a (tool_type, GIMP_TYPE_FILTER_TOOL));
+
+ gimp_tool_item_set_visible (GIMP_TOOL_ITEM (tool_info), visible);
+
+ /* hack to hide the operation tool entirely */
+ if (tool_type == GIMP_TYPE_OPERATION_TOOL)
+ tool_info->hidden = TRUE;
+
+ /* hack to not require experimental tools to be present in toolrc */
+ if (tool_type == GIMP_TYPE_N_POINT_DEFORMATION_TOOL ||
+ tool_type == GIMP_TYPE_SEAMLESS_CLONE_TOOL)
+ {
+ tool_info->experimental = TRUE;
+ }
+
+ g_object_set_data (G_OBJECT (tool_info), "gimp-tool-options-gui-func",
+ options_gui_func);
+
+ gimp_container_add (gimp->tool_info_list, GIMP_OBJECT (tool_info));
+ g_object_unref (tool_info);
+
+ if (tool_type == GIMP_TYPE_PAINTBRUSH_TOOL)
+ gimp_tool_info_set_standard (gimp, tool_info);
+}
+
+static void
+gimp_tools_copy_structure (Gimp *gimp,
+ GimpContainer *src_container,
+ GimpContainer *dest_container,
+ GHashTable *tools)
+{
+ GList *list;
+
+ for (list = GIMP_LIST (src_container)->queue->head;
+ list;
+ list = g_list_next (list))
+ {
+ GimpToolItem *src_tool_item = list->data;
+ GimpToolItem *dest_tool_item = NULL;
+
+ if (GIMP_IS_TOOL_GROUP (src_tool_item))
+ {
+ dest_tool_item = GIMP_TOOL_ITEM (gimp_tool_group_new ());
+
+ gimp_tools_copy_structure (
+ gimp,
+ gimp_viewable_get_children (GIMP_VIEWABLE (src_tool_item)),
+ gimp_viewable_get_children (GIMP_VIEWABLE (dest_tool_item)),
+ tools);
+
+ gimp_tool_group_set_active_tool (
+ GIMP_TOOL_GROUP (dest_tool_item),
+ gimp_tool_group_get_active_tool (GIMP_TOOL_GROUP (src_tool_item)));
+ }
+ else
+ {
+ dest_tool_item = GIMP_TOOL_ITEM (
+ gimp_get_tool_info (gimp, gimp_object_get_name (src_tool_item)));
+
+ if (dest_tool_item)
+ {
+ if (! GIMP_TOOL_INFO (dest_tool_item)->hidden)
+ {
+ g_object_ref (dest_tool_item);
+
+ if (tools)
+ g_hash_table_add (tools, dest_tool_item);
+ }
+ else
+ {
+ dest_tool_item = NULL;
+ }
+ }
+ }
+
+ if (dest_tool_item)
+ {
+ gimp_tool_item_set_visible (
+ dest_tool_item,
+ gimp_tool_item_get_visible (src_tool_item));
+
+ gimp_container_add (dest_container,
+ GIMP_OBJECT (dest_tool_item));
+
+ g_object_unref (dest_tool_item);
+ }
+ }
+}