/* PIKA - Photo and Image Kooker Application * a rebranding of The GNU Image Manipulation Program (created with heckimp) * A derived work which may be trivial. However, any changes may be (C)2023 by Aldercone Studio * * Original copyright, applying to most contents (license remains unchanged): * Copyright (C) 1995 Spencer Kimball and Peter Mattis * * 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 . */ #include "config.h" #include #include #include #include "libpikabase/pikabase.h" #include "libpikacolor/pikacolor.h" #include "libpikaconfig/pikaconfig.h" #include "libpikawidgets/pikawidgets.h" #include "tools-types.h" #include "gegl/pika-gegl-utils.h" #include "core/pikachannel.h" #include "core/pikacontainer.h" #include "core/pikacontext.h" #include "core/pikadrawable.h" #include "core/pikaerror.h" #include "core/pikaimage.h" #include "core/pikalist.h" #include "core/pikapickable.h" #include "core/pikasettings.h" #include "widgets/pikabuffersourcebox.h" #include "widgets/pikahelp-ids.h" #include "widgets/pikapickablebutton.h" #include "propgui/pikapropgui.h" #include "display/pikadisplay.h" #include "display/pikatoolgui.h" #include "pikafilteroptions.h" #include "pikaoperationtool.h" #include "pika-intl.h" typedef struct _AuxInput AuxInput; struct _AuxInput { PikaOperationTool *tool; gchar *pad; GeglNode *node; GtkWidget *box; }; /* local function prototypes */ static void pika_operation_tool_finalize (GObject *object); static gboolean pika_operation_tool_initialize (PikaTool *tool, PikaDisplay *display, GError **error); static void pika_operation_tool_control (PikaTool *tool, PikaToolAction action, PikaDisplay *display); static gchar * pika_operation_tool_get_operation (PikaFilterTool *filter_tool, gchar **description); static void pika_operation_tool_dialog (PikaFilterTool *filter_tool); static void pika_operation_tool_reset (PikaFilterTool *filter_tool); static void pika_operation_tool_set_config (PikaFilterTool *filter_tool, PikaConfig *config); static void pika_operation_tool_region_changed (PikaFilterTool *filter_tool); static void pika_operation_tool_color_picked (PikaFilterTool *filter_tool, gpointer identifier, gdouble x, gdouble y, const Babl *sample_format, const PikaRGB *color); static void pika_operation_tool_halt (PikaOperationTool *op_tool); static void pika_operation_tool_commit (PikaOperationTool *op_tool); static void pika_operation_tool_sync_op (PikaOperationTool *op_tool, gboolean sync_colors); static void pika_operation_tool_create_gui (PikaOperationTool *tool); static void pika_operation_tool_add_gui (PikaOperationTool *tool); static AuxInput * pika_operation_tool_aux_input_new (PikaOperationTool *tool, GeglNode *operation, const gchar *input_pad, const gchar *label); static void pika_operation_tool_aux_input_detach(AuxInput *input); static void pika_operation_tool_aux_input_clear (AuxInput *input); static void pika_operation_tool_aux_input_free (AuxInput *input); static void pika_operation_tool_unlink_chains (PikaOperationTool *op_tool); static void pika_operation_tool_relink_chains (PikaOperationTool *op_tool); G_DEFINE_TYPE (PikaOperationTool, pika_operation_tool, PIKA_TYPE_FILTER_TOOL) #define parent_class pika_operation_tool_parent_class void pika_operation_tool_register (PikaToolRegisterCallback callback, gpointer data) { (* callback) (PIKA_TYPE_OPERATION_TOOL, PIKA_TYPE_FILTER_OPTIONS, pika_color_options_gui, PIKA_CONTEXT_PROP_MASK_FOREGROUND | PIKA_CONTEXT_PROP_MASK_BACKGROUND, "pika-operation-tool", _("GEGL Operation"), _("Operation Tool: Use an arbitrary GEGL operation"), NULL, NULL, NULL, PIKA_HELP_TOOL_GEGL, PIKA_ICON_GEGL, data); } static void pika_operation_tool_class_init (PikaOperationToolClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); PikaToolClass *tool_class = PIKA_TOOL_CLASS (klass); PikaFilterToolClass *filter_tool_class = PIKA_FILTER_TOOL_CLASS (klass); object_class->finalize = pika_operation_tool_finalize; tool_class->initialize = pika_operation_tool_initialize; tool_class->control = pika_operation_tool_control; filter_tool_class->get_operation = pika_operation_tool_get_operation; filter_tool_class->dialog = pika_operation_tool_dialog; filter_tool_class->reset = pika_operation_tool_reset; filter_tool_class->set_config = pika_operation_tool_set_config; filter_tool_class->region_changed = pika_operation_tool_region_changed; filter_tool_class->color_picked = pika_operation_tool_color_picked; } static void pika_operation_tool_init (PikaOperationTool *op_tool) { } static void pika_operation_tool_finalize (GObject *object) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (object); g_clear_pointer (&op_tool->operation, g_free); g_clear_pointer (&op_tool->description, g_free); g_list_free_full (op_tool->aux_inputs, (GDestroyNotify) pika_operation_tool_aux_input_free); op_tool->aux_inputs = NULL; G_OBJECT_CLASS (parent_class)->finalize (object); } static gboolean pika_operation_tool_initialize (PikaTool *tool, PikaDisplay *display, GError **error) { if (PIKA_TOOL_CLASS (parent_class)->initialize (tool, display, error)) { PikaFilterTool *filter_tool = PIKA_FILTER_TOOL (tool); PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (tool); if (filter_tool->config) { GtkWidget *options_gui; pika_operation_tool_sync_op (op_tool, TRUE); options_gui = g_weak_ref_get (&op_tool->options_gui_ref); if (! options_gui) { pika_operation_tool_create_gui (op_tool); pika_operation_tool_add_gui (op_tool); } else { g_object_unref (options_gui); } } return TRUE; } return FALSE; } static void pika_operation_tool_control (PikaTool *tool, PikaToolAction action, PikaDisplay *display) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (tool); switch (action) { case PIKA_TOOL_ACTION_PAUSE: case PIKA_TOOL_ACTION_RESUME: break; case PIKA_TOOL_ACTION_HALT: pika_operation_tool_halt (op_tool); break; case PIKA_TOOL_ACTION_COMMIT: pika_operation_tool_commit (op_tool); break; } PIKA_TOOL_CLASS (parent_class)->control (tool, action, display); } static gchar * pika_operation_tool_get_operation (PikaFilterTool *filter_tool, gchar **description) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (filter_tool); *description = g_strdup (op_tool->description); return g_strdup (op_tool->operation); } static void pika_operation_tool_dialog (PikaFilterTool *filter_tool) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (filter_tool); GtkWidget *main_vbox; GtkWidget *options_sw; GtkWidget *options_gui; GtkWidget *options_box; main_vbox = pika_filter_tool_dialog_get_vbox (filter_tool); /* The options scrolled window */ options_sw = gtk_scrolled_window_new (NULL, NULL); g_weak_ref_set (&op_tool->options_sw_ref, options_sw); gtk_scrolled_window_set_overlay_scrolling (GTK_SCROLLED_WINDOW (options_sw), FALSE); gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (options_sw), GTK_SHADOW_NONE); gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (options_sw), GTK_POLICY_NEVER, GTK_POLICY_NEVER); gtk_box_pack_start (GTK_BOX (main_vbox), options_sw, TRUE, TRUE, 0); gtk_widget_show (options_sw); /* The options vbox */ options_box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 2); g_weak_ref_set (&op_tool->options_box_ref, options_box); gtk_container_add (GTK_CONTAINER (options_sw), options_box); gtk_widget_show (options_box); options_gui = g_weak_ref_get (&op_tool->options_gui_ref); if (options_gui) { pika_operation_tool_add_gui (op_tool); g_object_unref (options_gui); } } static void pika_operation_tool_reset (PikaFilterTool *filter_tool) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (filter_tool); pika_operation_tool_unlink_chains (op_tool); PIKA_FILTER_TOOL_CLASS (parent_class)->reset (filter_tool); if (filter_tool->config && PIKA_TOOL (op_tool)->drawables) pika_operation_tool_sync_op (op_tool, TRUE); pika_operation_tool_relink_chains (op_tool); } static void pika_operation_tool_set_config (PikaFilterTool *filter_tool, PikaConfig *config) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (filter_tool); pika_operation_tool_unlink_chains (op_tool); PIKA_FILTER_TOOL_CLASS (parent_class)->set_config (filter_tool, config); if (filter_tool->config && PIKA_TOOL (op_tool)->drawables) pika_operation_tool_sync_op (op_tool, FALSE); pika_operation_tool_relink_chains (op_tool); } static void pika_operation_tool_region_changed (PikaFilterTool *filter_tool) { PikaOperationTool *op_tool = PIKA_OPERATION_TOOL (filter_tool); /* when the region changes, do we want the operation's on-canvas * controller to move to a new position, or the operation to * change its properties to match the on-canvas controller? * * decided to leave the on-canvas controller where it is and * pretend it has changed, so the operation is updated * accordingly... */ if (filter_tool->widget) g_signal_emit_by_name (filter_tool->widget, "changed"); pika_operation_tool_sync_op (op_tool, FALSE); } static void pika_operation_tool_color_picked (PikaFilterTool *filter_tool, gpointer identifier, gdouble x, gdouble y, const Babl *sample_format, const PikaRGB *color) { gchar **pspecs = g_strsplit (identifier, ":", 2); if (pspecs[1]) { GObjectClass *object_class = G_OBJECT_GET_CLASS (filter_tool->config); GParamSpec *pspec_x; GParamSpec *pspec_y; gint off_x, off_y; GeglRectangle area; pika_filter_tool_get_drawable_area (filter_tool, &off_x, &off_y, &area); x -= off_x + area.x; y -= off_y + area.y; pspec_x = g_object_class_find_property (object_class, pspecs[0]); pspec_y = g_object_class_find_property (object_class, pspecs[1]); if (pspec_x && pspec_y && G_PARAM_SPEC_TYPE (pspec_x) == G_PARAM_SPEC_TYPE (pspec_y)) { GValue value_x = G_VALUE_INIT; GValue value_y = G_VALUE_INIT; g_value_init (&value_x, G_PARAM_SPEC_VALUE_TYPE (pspec_x)); g_value_init (&value_y, G_PARAM_SPEC_VALUE_TYPE (pspec_y)); #define HAS_KEY(p,k,v) pika_gegl_param_spec_has_key (p, k, v) if (HAS_KEY (pspec_x, "unit", "relative-coordinate") && HAS_KEY (pspec_y, "unit", "relative-coordinate")) { x /= (gdouble) area.width; y /= (gdouble) area.height; } if (G_IS_PARAM_SPEC_INT (pspec_x)) { g_value_set_int (&value_x, x); g_value_set_int (&value_y, y); g_param_value_validate (pspec_x, &value_x); g_param_value_validate (pspec_y, &value_y); g_object_set (filter_tool->config, pspecs[0], g_value_get_int (&value_x), pspecs[1], g_value_get_int (&value_y), NULL); } else if (G_IS_PARAM_SPEC_DOUBLE (pspec_x)) { g_value_set_double (&value_x, x); g_value_set_double (&value_y, y); g_param_value_validate (pspec_x, &value_x); g_param_value_validate (pspec_y, &value_y); g_object_set (filter_tool->config, pspecs[0], g_value_get_double (&value_x), pspecs[1], g_value_get_double (&value_y), NULL); } else { g_warning ("%s: unhandled param spec of type %s", G_STRFUNC, G_PARAM_SPEC_TYPE_NAME (pspec_x)); } g_value_unset (&value_x); g_value_unset (&value_y); } } else { g_object_set (filter_tool->config, pspecs[0], color, NULL); } g_strfreev (pspecs); } static void pika_operation_tool_halt (PikaOperationTool *op_tool) { /* don't reset op_tool->operation and op_tool->description so the * tool can be properly restarted by clicking on an image */ g_list_free_full (op_tool->aux_inputs, (GDestroyNotify) pika_operation_tool_aux_input_free); op_tool->aux_inputs = NULL; } static void pika_operation_tool_commit (PikaOperationTool *op_tool) { /* remove the aux input boxes from the dialog, so they don't get * destroyed when the parent class runs its commit() */ g_list_foreach (op_tool->aux_inputs, (GFunc) pika_operation_tool_aux_input_detach, NULL); } static void pika_operation_tool_sync_op (PikaOperationTool *op_tool, gboolean sync_colors) { PikaFilterTool *filter_tool = PIKA_FILTER_TOOL (op_tool); PikaToolOptions *options = PIKA_TOOL_GET_OPTIONS (op_tool); GParamSpec **pspecs; guint n_pspecs; gint off_x, off_y; GeglRectangle area; gint i; pika_filter_tool_get_drawable_area (filter_tool, &off_x, &off_y, &area); pspecs = g_object_class_list_properties (G_OBJECT_GET_CLASS (filter_tool->config), &n_pspecs); for (i = 0; i < n_pspecs; i++) { GParamSpec *pspec = pspecs[i]; #define HAS_KEY(p,k,v) pika_gegl_param_spec_has_key (p, k, v) if (HAS_KEY (pspec, "role", "output-extent")) { if (HAS_KEY (pspec, "unit", "pixel-coordinate") && HAS_KEY (pspec, "axis", "x")) { g_object_set (filter_tool->config, pspec->name, 0, NULL); } else if (HAS_KEY (pspec, "unit", "pixel-coordinate") && HAS_KEY (pspec, "axis", "y")) { g_object_set (filter_tool->config, pspec->name, 0, NULL); } else if (HAS_KEY (pspec, "unit", "pixel-distance") && HAS_KEY (pspec, "axis", "x")) { g_object_set (filter_tool->config, pspec->name, area.width, NULL); } else if (HAS_KEY (pspec, "unit", "pixel-distance") && HAS_KEY (pspec, "axis", "y")) { g_object_set (filter_tool->config, pspec->name, area.height, NULL); } } else if (sync_colors) { if (HAS_KEY (pspec, "role", "color-primary")) { PikaRGB color; pika_context_get_foreground (PIKA_CONTEXT (options), &color); g_object_set (filter_tool->config, pspec->name, &color, NULL); } else if (sync_colors && HAS_KEY (pspec, "role", "color-secondary")) { PikaRGB color; pika_context_get_background (PIKA_CONTEXT (options), &color); g_object_set (filter_tool->config, pspec->name, &color, NULL); } } } g_free (pspecs); } static void pika_operation_tool_create_gui (PikaOperationTool *op_tool) { PikaFilterTool *filter_tool = PIKA_FILTER_TOOL (op_tool); GtkWidget *options_gui; gint off_x, off_y; GeglRectangle area; gchar **input_pads; pika_filter_tool_get_drawable_area (filter_tool, &off_x, &off_y, &area); options_gui = pika_prop_gui_new (G_OBJECT (filter_tool->config), G_TYPE_FROM_INSTANCE (filter_tool->config), 0, &area, PIKA_CONTEXT (PIKA_TOOL_GET_OPTIONS (op_tool)), (PikaCreatePickerFunc) pika_filter_tool_add_color_picker, (PikaCreateControllerFunc) pika_filter_tool_add_controller, filter_tool); g_weak_ref_set (&op_tool->options_gui_ref, options_gui); input_pads = gegl_node_list_input_pads (filter_tool->operation); if (input_pads) { gint i; for (i = 0; input_pads[i]; i++) { AuxInput *input; GRegex *regex; gchar *label; if (! strcmp (input_pads[i], "input")) continue; regex = g_regex_new ("^aux(\\d*)$", 0, 0, NULL); g_return_if_fail (regex != NULL); /* Translators: don't translate "Aux" */ label = g_regex_replace (regex, input_pads[i], -1, 0, _("Aux\\1 Input"), 0, NULL); input = pika_operation_tool_aux_input_new (op_tool, filter_tool->operation, input_pads[i], label); op_tool->aux_inputs = g_list_prepend (op_tool->aux_inputs, input); g_free (label); g_regex_unref (regex); } g_strfreev (input_pads); } } static void pika_operation_tool_add_gui (PikaOperationTool *op_tool) { GtkSizeGroup *size_group = NULL; GtkWidget *options_gui; GtkWidget *options_box; GtkWidget *options_sw; GtkWidget *shell; GdkRectangle workarea; GtkRequisition minimum; GList *list; gboolean scrolling; options_gui = g_weak_ref_get (&op_tool->options_gui_ref); options_box = g_weak_ref_get (&op_tool->options_box_ref); options_sw = g_weak_ref_get (&op_tool->options_sw_ref); g_return_if_fail (options_gui && options_box && options_sw); for (list = op_tool->aux_inputs; list; list = g_list_next (list)) { AuxInput *input = list->data; GtkWidget *toggle; if (! size_group) size_group = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL); toggle = pika_buffer_source_box_get_toggle (PIKA_BUFFER_SOURCE_BOX (input->box)); gtk_size_group_add_widget (size_group, toggle); gtk_box_pack_start (GTK_BOX (options_box), input->box, FALSE, FALSE, 0); gtk_widget_show (input->box); } if (size_group) g_object_unref (size_group); gtk_box_pack_start (GTK_BOX (options_box), options_gui, TRUE, TRUE, 0); gtk_widget_show (options_gui); shell = GTK_WIDGET (pika_display_get_shell (PIKA_TOOL (op_tool)->display)); gdk_monitor_get_workarea (pika_widget_get_monitor (shell), &workarea); gtk_widget_get_preferred_size (options_box, &minimum, NULL); scrolling = minimum.height > workarea.height / 2; gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (options_sw), GTK_POLICY_NEVER, scrolling ? GTK_POLICY_AUTOMATIC : GTK_POLICY_NEVER); if (scrolling) gtk_widget_set_size_request (options_sw, -1, workarea.height / 2); else gtk_widget_set_size_request (options_sw, -1, -1); g_object_unref (options_gui); g_object_unref (options_box); g_object_unref (options_sw); } /* aux input utility functions */ static void pika_operation_tool_aux_input_notify (PikaBufferSourceBox *box, const GParamSpec *pspec, AuxInput *input) { PikaFilterTool *filter_tool = PIKA_FILTER_TOOL (input->tool); /* emit "notify" so PikaFilterTool will update its preview * * FIXME: this is a bad hack that should go away once PikaImageMap * and PikaFilterTool are refactored to be more filter-ish. */ if (filter_tool->config) g_signal_emit_by_name (filter_tool->config, "notify", NULL); } static AuxInput * pika_operation_tool_aux_input_new (PikaOperationTool *op_tool, GeglNode *operation, const gchar *input_pad, const gchar *label) { AuxInput *input = g_slice_new (AuxInput); PikaContext *context; input->tool = op_tool; input->pad = g_strdup (input_pad); input->node = gegl_node_new_child (NULL, "operation", "gegl:buffer-source", NULL); gegl_node_connect (input->node, "output", operation, input_pad); context = PIKA_CONTEXT (PIKA_TOOL_GET_OPTIONS (op_tool)); input->box = pika_buffer_source_box_new (context, input->node, label); /* make AuxInput owner of the box */ g_object_ref_sink (input->box); g_signal_connect (input->box, "notify::pickable", G_CALLBACK (pika_operation_tool_aux_input_notify), input); g_signal_connect (input->box, "notify::enabled", G_CALLBACK (pika_operation_tool_aux_input_notify), input); return input; } static void pika_operation_tool_aux_input_detach (AuxInput *input) { GtkWidget *parent = gtk_widget_get_parent (input->box); if (parent) gtk_container_remove (GTK_CONTAINER (parent), input->box); } static void pika_operation_tool_aux_input_clear (AuxInput *input) { pika_operation_tool_aux_input_detach (input); g_object_set (input->box, "pickable", NULL, NULL); } static void pika_operation_tool_aux_input_free (AuxInput *input) { pika_operation_tool_aux_input_clear (input); g_free (input->pad); g_object_unref (input->node); g_object_unref (input->box); g_slice_free (AuxInput, input); } static void pika_operation_tool_unlink_chains (PikaOperationTool *op_tool) { GObject *options_gui = g_weak_ref_get (&op_tool->options_gui_ref); GList *chains; g_return_if_fail (options_gui != NULL); chains = g_object_get_data (options_gui, "chains"); while (chains) { PikaChainButton *chain = chains->data; gboolean active; active = pika_chain_button_get_active (chain); g_object_set_data (G_OBJECT (chain), "was-active", GINT_TO_POINTER (active)); if (active) { pika_chain_button_set_active (chain, FALSE); } chains = chains->next; } g_object_unref (options_gui); } static void pika_operation_tool_relink_chains (PikaOperationTool *op_tool) { PikaFilterTool *filter_tool = PIKA_FILTER_TOOL (op_tool); GObject *options_gui = g_weak_ref_get (&op_tool->options_gui_ref); GList *chains; g_return_if_fail (options_gui != NULL); chains = g_object_get_data (options_gui, "chains"); while (chains) { PikaChainButton *chain = chains->data; if (g_object_get_data (G_OBJECT (chain), "was-active")) { const gchar *name_x = g_object_get_data (chains->data, "x-property"); const gchar *name_y = g_object_get_data (chains->data, "y-property"); const gchar *names[2] = { name_x, name_y }; GValue values[2] = { G_VALUE_INIT, G_VALUE_INIT }; GValue double_x = G_VALUE_INIT; GValue double_y = G_VALUE_INIT; g_object_getv (filter_tool->config, 2, names, values); g_value_init (&double_x, G_TYPE_DOUBLE); g_value_init (&double_y, G_TYPE_DOUBLE); if (g_value_transform (&values[0], &double_x) && g_value_transform (&values[1], &double_y) && g_value_get_double (&double_x) == g_value_get_double (&double_y)) { pika_chain_button_set_active (chain, TRUE); } g_value_unset (&double_x); g_value_unset (&double_y); g_value_unset (&values[0]); g_value_unset (&values[1]); g_object_set_data (G_OBJECT (chain), "was-active", NULL); } chains = chains->next; } g_object_unref (options_gui); } /* public functions */ void pika_operation_tool_set_operation (PikaOperationTool *op_tool, const gchar *operation, const gchar *title, const gchar *description, const gchar *undo_desc, const gchar *icon_name, const gchar *help_id) { PikaTool *tool; PikaFilterTool *filter_tool; GtkWidget *options_gui; g_return_if_fail (PIKA_IS_OPERATION_TOOL (op_tool)); tool = PIKA_TOOL (op_tool); filter_tool = PIKA_FILTER_TOOL (op_tool); g_free (op_tool->operation); g_free (op_tool->description); op_tool->operation = g_strdup (operation); op_tool->description = g_strdup (description); pika_tool_set_label (tool, title); pika_tool_set_undo_desc (tool, undo_desc); pika_tool_set_icon_name (tool, icon_name); pika_tool_set_help_id (tool, help_id); g_list_free_full (op_tool->aux_inputs, (GDestroyNotify) pika_operation_tool_aux_input_free); op_tool->aux_inputs = NULL; pika_filter_tool_set_widget (filter_tool, NULL); options_gui = g_weak_ref_get (&op_tool->options_gui_ref); if (options_gui) { pika_filter_tool_disable_color_picking (filter_tool); g_object_unref (options_gui); gtk_widget_destroy (options_gui); } if (! operation) return; pika_filter_tool_get_operation (filter_tool); if (tool->drawables) pika_operation_tool_sync_op (op_tool, TRUE); if (filter_tool->config && tool->display) { GtkWidget *options_box; pika_operation_tool_create_gui (op_tool); options_box = g_weak_ref_get (&op_tool->options_box_ref); if (options_box) { pika_operation_tool_add_gui (op_tool); g_object_unref (options_box); } } }