/* 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-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 . */ #include "config.h" #include #include #include "libpikamath/pikamath.h" #include "libpikawidgets/pikawidgets.h" #include "tools-types.h" #include "config/pikaguiconfig.h" #include "core/pika.h" #include "core/pikadrawable-transform.h" #include "core/pikaerror.h" #include "core/pikaimage.h" #include "core/pikaimage-item-list.h" #include "core/pikaimage-transform.h" #include "core/pikaimage-undo.h" #include "core/pikalayer.h" #include "core/pikalayermask.h" #include "core/pikaprogress.h" #include "core/pika-transform-resize.h" #include "vectors/pikavectors.h" #include "display/pikadisplay.h" #include "display/pikadisplayshell.h" #include "widgets/pikamessagedialog.h" #include "widgets/pikamessagebox.h" #include "widgets/pikawidgets-utils.h" #include "pikatoolcontrol.h" #include "pikatools-utils.h" #include "pikatransformoptions.h" #include "pikatransformtool.h" #include "pika-intl.h" /* the minimal ratio between the transformed item size and the image size, * above which confirmation is required. */ #define MIN_CONFIRMATION_RATIO 10 /* local function prototypes */ static void pika_transform_tool_control (PikaTool *tool, PikaToolAction action, PikaDisplay *display); static gchar * pika_transform_tool_real_get_undo_desc (PikaTransformTool *tr_tool); static PikaTransformDirection pika_transform_tool_real_get_direction (PikaTransformTool *tr_tool); static GeglBuffer * pika_transform_tool_real_transform (PikaTransformTool *tr_tool, GList *objects, GeglBuffer *orig_buffer, gint orig_offset_x, gint orig_offset_y, PikaColorProfile **buffer_profile, gint *new_offset_x, gint *new_offset_y); static void pika_transform_tool_halt (PikaTransformTool *tr_tool); static gboolean pika_transform_tool_confirm (PikaTransformTool *tr_tool, PikaDisplay *display); G_DEFINE_TYPE (PikaTransformTool, pika_transform_tool, PIKA_TYPE_DRAW_TOOL) #define parent_class pika_transform_tool_parent_class /* private functions */ static void pika_transform_tool_class_init (PikaTransformToolClass *klass) { PikaToolClass *tool_class = PIKA_TOOL_CLASS (klass); tool_class->control = pika_transform_tool_control; klass->recalc_matrix = NULL; klass->get_undo_desc = pika_transform_tool_real_get_undo_desc; klass->get_direction = pika_transform_tool_real_get_direction; klass->transform = pika_transform_tool_real_transform; klass->undo_desc = _("Transform"); klass->progress_text = _("Transforming"); } static void pika_transform_tool_init (PikaTransformTool *tr_tool) { pika_matrix3_identity (&tr_tool->transform); tr_tool->transform_valid = TRUE; tr_tool->restore_type = FALSE; } static void pika_transform_tool_control (PikaTool *tool, PikaToolAction action, PikaDisplay *display) { PikaTransformTool *tr_tool = PIKA_TRANSFORM_TOOL (tool); switch (action) { case PIKA_TOOL_ACTION_PAUSE: case PIKA_TOOL_ACTION_RESUME: break; case PIKA_TOOL_ACTION_HALT: pika_transform_tool_halt (tr_tool); break; case PIKA_TOOL_ACTION_COMMIT: break; } PIKA_TOOL_CLASS (parent_class)->control (tool, action, display); } static gchar * pika_transform_tool_real_get_undo_desc (PikaTransformTool *tr_tool) { return g_strdup (PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->undo_desc); } static PikaTransformDirection pika_transform_tool_real_get_direction (PikaTransformTool *tr_tool) { PikaTransformOptions *options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); return options->direction; } static GeglBuffer * pika_transform_tool_real_transform (PikaTransformTool *tr_tool, GList *objects, GeglBuffer *orig_buffer, gint orig_offset_x, gint orig_offset_y, PikaColorProfile **buffer_profile, gint *new_offset_x, gint *new_offset_y) { PikaTransformToolClass *klass = PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool); PikaTool *tool = PIKA_TOOL (tr_tool); PikaTransformOptions *options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tool); PikaContext *context = PIKA_CONTEXT (options); GeglBuffer *ret = NULL; PikaTransformResize clip = options->clip; PikaTransformDirection direction; PikaProgress *progress; direction = PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->get_direction (tr_tool); progress = pika_progress_start (PIKA_PROGRESS (tool), FALSE, "%s", klass->progress_text); while (g_main_context_pending (NULL)) g_main_context_iteration (NULL, FALSE); if (orig_buffer) { /* this happens when transforming a selection cut out of * normal drawables. */ ret = pika_drawable_transform_buffer_affine (objects->data, context, orig_buffer, orig_offset_x, orig_offset_y, &tr_tool->transform, direction, options->interpolation, clip, buffer_profile, new_offset_x, new_offset_y, progress); } else if (g_list_length (objects) == 1 && PIKA_IS_IMAGE (objects->data)) { /* this happens for images */ pika_image_transform (objects->data, context, &tr_tool->transform, direction, options->interpolation, clip, progress); } else { GList *items; /* this happens for entire drawables, paths and layer groups */ g_return_val_if_fail (g_list_length (objects) > 0, NULL); items = pika_image_item_list_filter (g_list_copy (objects)); pika_image_item_list_transform (pika_item_get_image (objects->data), items, context, &tr_tool->transform, direction, options->interpolation, clip, progress); g_list_free (items); } if (progress) pika_progress_end (progress); return ret; } static void pika_transform_tool_halt (PikaTransformTool *tr_tool) { PikaTransformOptions *options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); tr_tool->x1 = 0; tr_tool->y1 = 0; tr_tool->x2 = 0; tr_tool->y2 = 0; if (tr_tool->restore_type) { g_object_set (options, "type", tr_tool->saved_type, NULL); tr_tool->restore_type = FALSE; } } static gboolean pika_transform_tool_confirm (PikaTransformTool *tr_tool, PikaDisplay *display) { PikaTransformOptions *options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); PikaDisplayShell *shell = pika_display_get_shell (display); PikaImage *image = pika_display_get_image (display); GList *selected_objects; gdouble max_ratio = 0.0; PikaObject *max_ratio_object = NULL; selected_objects = pika_transform_tool_get_selected_objects (tr_tool, display); if (PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->recalc_matrix) { PikaMatrix3 transform; PikaTransformDirection direction; GeglRectangle selection_bounds; gboolean selection_empty = TRUE; GList *objects = NULL; GList *iter; transform = tr_tool->transform; direction = PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->get_direction ( tr_tool); if (direction == PIKA_TRANSFORM_BACKWARD) pika_matrix3_invert (&transform); if (options->type == PIKA_TRANSFORM_TYPE_LAYER) { for (iter = selected_objects; iter; iter = iter->next) if (! pika_viewable_get_children (PIKA_VIEWABLE (iter->data))) { if (pika_item_bounds (PIKA_ITEM (pika_image_get_mask (image)), &selection_bounds.x, &selection_bounds.y, &selection_bounds.width, &selection_bounds.height)) { selection_empty = FALSE; break; } } } if (selection_empty && selected_objects && PIKA_IS_ITEM (selected_objects->data)) { objects = pika_image_item_list_filter (g_list_copy (selected_objects)); g_list_free (selected_objects); } else { objects = selected_objects; } if (options->type == PIKA_TRANSFORM_TYPE_IMAGE) { objects = g_list_concat ( objects, pika_image_item_list_get_list (image, PIKA_ITEM_TYPE_ALL, PIKA_ITEM_SET_ALL)); } for (iter = objects; iter; iter = g_list_next (iter)) { PikaObject *object = iter->data; PikaTransformResize clip = options->clip; GeglRectangle orig_bounds; GeglRectangle new_bounds; gdouble ratio = 0.0; if (PIKA_IS_DRAWABLE (object)) { if (selection_empty) { PikaItem *item = PIKA_ITEM (object); pika_item_get_offset (item, &orig_bounds.x, &orig_bounds.y); orig_bounds.width = pika_item_get_width (item); orig_bounds.height = pika_item_get_height (item); clip = pika_item_get_clip (item, clip); } else { orig_bounds = selection_bounds; } } else if (PIKA_IS_ITEM (object)) { PikaItem *item = PIKA_ITEM (object); pika_item_bounds (item, &orig_bounds.x, &orig_bounds.y, &orig_bounds.width, &orig_bounds.height); clip = pika_item_get_clip (item, clip); } else { PikaImage *image; g_return_val_if_fail (PIKA_IS_IMAGE (object), FALSE); image = PIKA_IMAGE (object); orig_bounds.x = 0; orig_bounds.y = 0; orig_bounds.width = pika_image_get_width (image); orig_bounds.height = pika_image_get_height (image); } pika_transform_resize_boundary (&transform, clip, orig_bounds.x, orig_bounds.y, orig_bounds.x + orig_bounds.width, orig_bounds.y + orig_bounds.height, &new_bounds.x, &new_bounds.y, &new_bounds.width, &new_bounds.height); new_bounds.width -= new_bounds.x; new_bounds.height -= new_bounds.y; if (new_bounds.width > orig_bounds.width) { ratio = MAX (ratio, (gdouble) new_bounds.width / (gdouble) pika_image_get_width (image)); } if (new_bounds.height > orig_bounds.height) { ratio = MAX (ratio, (gdouble) new_bounds.height / (gdouble) pika_image_get_height (image)); } if (ratio > max_ratio) { max_ratio = ratio; max_ratio_object = object; } } g_list_free (objects); } if (max_ratio > MIN_CONFIRMATION_RATIO) { GtkWidget *dialog; gint response; dialog = pika_message_dialog_new (_("Confirm Transformation"), PIKA_ICON_DIALOG_WARNING, GTK_WIDGET (shell), GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT, pika_standard_help_func, NULL, _("_Cancel"), GTK_RESPONSE_CANCEL, _("_Transform"), GTK_RESPONSE_OK, NULL); pika_dialog_set_alternative_button_order (GTK_DIALOG (dialog), GTK_RESPONSE_OK, GTK_RESPONSE_CANCEL, -1); if (PIKA_IS_ITEM (max_ratio_object)) { pika_message_box_set_primary_text (PIKA_MESSAGE_DIALOG (dialog)->box, _("Transformation creates " "a very large item.")); pika_message_box_set_text ( PIKA_MESSAGE_DIALOG (dialog)->box, _("Applying the transformation will result " "in an item that is over %g times larger " "than the image."), floor (max_ratio)); } else { pika_message_box_set_primary_text (PIKA_MESSAGE_DIALOG (dialog)->box, _("Transformation creates " "a very large image.")); pika_message_box_set_text ( PIKA_MESSAGE_DIALOG (dialog)->box, _("Applying the transformation will enlarge " "the image by a factor of %g."), floor (max_ratio)); } response = gtk_dialog_run (GTK_DIALOG (dialog)); gtk_widget_destroy (dialog); if (response != GTK_RESPONSE_OK) return FALSE; } return TRUE; } /* public functions */ gboolean pika_transform_tool_bounds (PikaTransformTool *tr_tool, PikaDisplay *display) { PikaTransformOptions *options; PikaDisplayShell *shell; PikaImage *image; gboolean non_empty = TRUE; g_return_val_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool), FALSE); options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); image = pika_display_get_image (display); shell = pika_display_get_shell (display); g_return_val_if_fail (PIKA_IS_IMAGE (image), FALSE); switch (options->type) { case PIKA_TRANSFORM_TYPE_LAYER: { GList *drawables; gint offset_x; gint offset_y; gint x, y; gint width, height; drawables = pika_image_get_selected_drawables (image); pika_item_get_offset (PIKA_ITEM (drawables->data), &offset_x, &offset_y); non_empty = pika_item_mask_intersect (PIKA_ITEM (drawables->data), &x, &y, &width, &height); tr_tool->x1 = x + offset_x; tr_tool->y1 = y + offset_y; tr_tool->x2 = x + width + offset_x; tr_tool->y2 = y + height + offset_y; g_list_free (drawables); } break; case PIKA_TRANSFORM_TYPE_SELECTION: { pika_item_bounds (PIKA_ITEM (pika_image_get_mask (image)), &tr_tool->x1, &tr_tool->y1, &tr_tool->x2, &tr_tool->y2); tr_tool->x2 += tr_tool->x1; tr_tool->y2 += tr_tool->y1; } break; case PIKA_TRANSFORM_TYPE_PATH: { PikaChannel *selection = pika_image_get_mask (image); /* if selection is not empty, use its bounds to perform the * transformation of the path */ if (! pika_channel_is_empty (selection)) { pika_item_bounds (PIKA_ITEM (selection), &tr_tool->x1, &tr_tool->y1, &tr_tool->x2, &tr_tool->y2); tr_tool->x2 += tr_tool->x1; tr_tool->y2 += tr_tool->y1; } else { GList *iter; /* without selection, test the emptiness of the path bounds : * if empty, use the canvas bounds * else use the path bounds */ tr_tool->x1 = G_MAXINT; tr_tool->y1 = G_MAXINT; tr_tool->x2 = G_MININT; tr_tool->y2 = G_MININT; for (iter = pika_image_get_selected_vectors (image); iter; iter = iter->next) { PikaItem *item = iter->data; gint x; gint y; gint width; gint height; if (pika_item_bounds (item, &x, &y, &width, &height)) { tr_tool->x1 = MIN (tr_tool->x1, x); tr_tool->y1 = MIN (tr_tool->y1, y); tr_tool->x2 = MAX (tr_tool->x2, x + width); tr_tool->y2 = MAX (tr_tool->y2, y + height); } } if (tr_tool->x2 <= tr_tool->x1 || tr_tool->y2 <= tr_tool->y1) { tr_tool->x1 = 0; tr_tool->y1 = 0; tr_tool->x2 = pika_image_get_width (image); tr_tool->y2 = pika_image_get_height (image); } } } break; case PIKA_TRANSFORM_TYPE_IMAGE: if (! shell->show_all) { tr_tool->x1 = 0; tr_tool->y1 = 0; tr_tool->x2 = pika_image_get_width (image); tr_tool->y2 = pika_image_get_height (image); } else { GeglRectangle bounding_box; bounding_box = pika_display_shell_get_bounding_box (shell); tr_tool->x1 = bounding_box.x; tr_tool->y1 = bounding_box.y; tr_tool->x2 = bounding_box.x + bounding_box.width; tr_tool->y2 = bounding_box.y + bounding_box.height; } break; } return non_empty; } void pika_transform_tool_recalc_matrix (PikaTransformTool *tr_tool, PikaDisplay *display) { g_return_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool)); g_return_if_fail (PIKA_IS_DISPLAY (display)); if (tr_tool->x1 == tr_tool->x2 && tr_tool->y1 == tr_tool->y2) pika_transform_tool_bounds (tr_tool, display); if (PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->recalc_matrix) PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->recalc_matrix (tr_tool); } GList * pika_transform_tool_get_selected_objects (PikaTransformTool *tr_tool, PikaDisplay *display) { PikaTransformOptions *options; PikaImage *image; GList *objects = NULL; g_return_val_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool), NULL); g_return_val_if_fail (PIKA_IS_DISPLAY (display), NULL); options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); image = pika_display_get_image (display); g_return_val_if_fail (PIKA_IS_IMAGE (image), NULL); if (tr_tool->objects) return g_list_copy (tr_tool->objects); switch (options->type) { case PIKA_TRANSFORM_TYPE_LAYER: objects = pika_image_get_selected_drawables (image); break; case PIKA_TRANSFORM_TYPE_SELECTION: if (! pika_channel_is_empty (pika_image_get_mask (image))) objects = g_list_prepend (NULL, pika_image_get_mask (image)); break; case PIKA_TRANSFORM_TYPE_PATH: objects = g_list_copy (pika_image_get_selected_vectors (image)); break; case PIKA_TRANSFORM_TYPE_IMAGE: objects = g_list_prepend (NULL, image); break; } return objects; } GList * pika_transform_tool_check_selected_objects (PikaTransformTool *tr_tool, PikaDisplay *display, GError **error) { PikaTransformOptions *options; GList *objects; GList *iter; const gchar *null_message = NULL; const gchar *locked_message = NULL; PikaItem *locked_item = NULL; PikaGuiConfig *config = PIKA_GUI_CONFIG (display->pika->config); g_return_val_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool), NULL); g_return_val_if_fail (PIKA_IS_DISPLAY (display), NULL); g_return_val_if_fail (error == NULL || *error == NULL, NULL); options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); objects = pika_transform_tool_get_selected_objects (tr_tool, display); switch (options->type) { case PIKA_TRANSFORM_TYPE_LAYER: null_message = _("There is no layer to transform."); for (iter = objects; iter; iter = iter->next) { PikaItem *item = iter->data; if (pika_item_is_content_locked (item, &locked_item)) locked_message = _("A selected layer's pixels are locked."); else if (pika_item_is_position_locked (item, &locked_item)) locked_message = _("A selected layer's position and size are locked."); if (! pika_item_is_visible (item) && ! config->edit_non_visible && ! g_list_find (tr_tool->objects, item)) /* see bug #759194 */ { g_set_error_literal (error, PIKA_ERROR, PIKA_FAILED, _("A selected layer is not visible.")); return NULL; } if (! pika_transform_tool_bounds (tr_tool, display)) { g_set_error_literal (error, PIKA_ERROR, PIKA_FAILED, _("The selection does not intersect with a selected layer.")); return NULL; } } break; case PIKA_TRANSFORM_TYPE_SELECTION: null_message = _("There is no selection to transform."); for (iter = objects; iter; iter = iter->next) { PikaItem *item = iter->data; /* cannot happen, so don't translate these messages */ if (pika_item_is_content_locked (item, &locked_item)) locked_message = "The selection's pixels are locked."; else if (pika_item_is_position_locked (item, &locked_item)) locked_message = "The selection's position and size are locked."; } break; case PIKA_TRANSFORM_TYPE_PATH: null_message = _("There is no path to transform."); for (iter = objects; iter; iter = iter->next) { PikaItem *item = iter->data; if (pika_item_is_content_locked (item, &locked_item)) locked_message = _("The selected path's strokes are locked."); else if (pika_item_is_position_locked (item, &locked_item)) locked_message = _("The selected path's position is locked."); else if (! pika_vectors_get_n_strokes (PIKA_VECTORS (item))) locked_message = _("The selected path has no strokes."); } break; case PIKA_TRANSFORM_TYPE_IMAGE: /* cannot happen, so don't translate this message */ null_message = "There is no image to transform."; break; } if (! objects) { g_set_error_literal (error, PIKA_ERROR, PIKA_FAILED, null_message); if (error) { pika_tools_show_tool_options (display->pika); pika_widget_blink (options->type_box); } return NULL; } if (locked_message) { g_set_error_literal (error, PIKA_ERROR, PIKA_FAILED, locked_message); if (error) { if (locked_item == NULL) locked_item = PIKA_ITEM (objects->data); pika_tools_blink_lock_box (display->pika, locked_item); } return NULL; } return objects; } gboolean pika_transform_tool_transform (PikaTransformTool *tr_tool, PikaDisplay *display) { PikaTool *tool; PikaTransformOptions *options; PikaImage *image; GList *selected_objects; GeglBuffer *orig_buffer = NULL; gint orig_offset_x = 0; gint orig_offset_y = 0; GeglBuffer *new_buffer; gint new_offset_x; gint new_offset_y; PikaColorProfile *buffer_profile; gchar *undo_desc = NULL; gboolean new_layer = FALSE; GError *error = NULL; g_return_val_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool), FALSE); g_return_val_if_fail (PIKA_IS_DISPLAY (display), FALSE); tool = PIKA_TOOL (tr_tool); options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tool); image = pika_display_get_image (display); g_return_val_if_fail (PIKA_IS_IMAGE (image), FALSE); selected_objects = pika_transform_tool_check_selected_objects (tr_tool, display, &error); if (! selected_objects) { pika_tool_message_literal (tool, display, error->message); g_clear_error (&error); return FALSE; } pika_transform_tool_recalc_matrix (tr_tool, display); if (! tr_tool->transform_valid) { pika_tool_message_literal (tool, display, _("The current transform is invalid")); return FALSE; } if (PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->recalc_matrix && pika_matrix3_is_identity (&tr_tool->transform)) { /* No need to commit an identity transformation! */ return TRUE; } if (! pika_transform_tool_confirm (tr_tool, display)) return FALSE; pika_set_busy (display->pika); /* We're going to dirty this image, but we want to keep the tool around */ pika_tool_control_push_preserve (tool->control, TRUE); undo_desc = PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->get_undo_desc (tr_tool); pika_image_undo_group_start (image, PIKA_UNDO_GROUP_TRANSFORM, undo_desc); g_free (undo_desc); switch (options->type) { case PIKA_TRANSFORM_TYPE_LAYER: if (! pika_channel_is_empty (pika_image_get_mask (image))) { orig_buffer = pika_drawable_transform_cut ( selected_objects, PIKA_CONTEXT (options), &orig_offset_x, &orig_offset_y, &new_layer); } break; case PIKA_TRANSFORM_TYPE_SELECTION: case PIKA_TRANSFORM_TYPE_PATH: case PIKA_TRANSFORM_TYPE_IMAGE: break; } /* Send the request for the transformation to the tool... */ new_buffer = PIKA_TRANSFORM_TOOL_GET_CLASS (tr_tool)->transform ( tr_tool, selected_objects, orig_buffer, orig_offset_x, orig_offset_y, &buffer_profile, &new_offset_x, &new_offset_y); if (orig_buffer) g_object_unref (orig_buffer); switch (options->type) { case PIKA_TRANSFORM_TYPE_LAYER: if (new_buffer) { /* paste the new transformed image to the image...also implement * undo... */ pika_drawable_transform_paste (PIKA_DRAWABLE (selected_objects->data), new_buffer, buffer_profile, new_offset_x, new_offset_y, new_layer); g_object_unref (new_buffer); } break; case PIKA_TRANSFORM_TYPE_SELECTION: case PIKA_TRANSFORM_TYPE_PATH: case PIKA_TRANSFORM_TYPE_IMAGE: /* Nothing to be done */ break; } pika_image_undo_group_end (image); /* We're done dirtying the image, and would like to be restarted if * the image gets dirty while the tool exists */ pika_tool_control_pop_preserve (tool->control); pika_unset_busy (display->pika); pika_image_flush (image); g_list_free (selected_objects); return TRUE; } void pika_transform_tool_set_type (PikaTransformTool *tr_tool, PikaTransformType type) { PikaTransformOptions *options; g_return_if_fail (PIKA_IS_TRANSFORM_TOOL (tr_tool)); options = PIKA_TRANSFORM_TOOL_GET_OPTIONS (tr_tool); if (! tr_tool->restore_type) tr_tool->saved_type = options->type; tr_tool->restore_type = FALSE; g_object_set (options, "type", type, NULL); tr_tool->restore_type = TRUE; }