Gimp/app/tools/gimpoffsettool.c
Jehan d493f0537f Issue #8900 and #9923: reimplementing GimpUnit as a proper class.
This fixes all our GObject Introspection issues with GimpUnit which was
both an enum and an int-derived type of user-defined units *completing*
the enum values. GIR clearly didn't like this!

Now GimpUnit is a proper class and units are unique objects, allowing to
compare them with an identity test (i.e. `unit == gimp_unit_pixel ()`
tells us if unit is the pixel unit or not), which makes it easy to use,
just like with int, yet adding also methods, making for nicer
introspected API.

As an aside, this also fixes #10738, by having all the built-in units
retrievable even if libgimpbase had not been properly initialized with
gimp_base_init().
I haven't checked in details how GIR works to introspect, but it looks
like it loads the library to inspect and runs functions, hence
triggering some CRITICALS because virtual methods (supposed to be
initialized with gimp_base_init() run by libgimp) are not set. This new
code won't trigger any critical because the vtable method are now not
necessary, at least for all built-in units.

Note that GimpUnit is still in libgimpbase. It could have been moved to
libgimp in order to avoid any virtual method table (since we need to
keep core and libgimp side's units in sync, PDB is required), but too
many libgimpwidgets widgets were already using GimpUnit. And technically
most of GimpUnit logic doesn't require PDB (only the creation/sync
part). This is one of the reasons why user-created GimpUnit list is
handled and stored differently from other types of objects.

Globally this simplifies the code a lot too and we don't need separate
implementations of various utils for core and libgimp, which means less
prone to errors.
2024-08-02 10:46:38 +02:00

797 lines
28 KiB
C

/* GIMP - The GNU Image Manipulation Program
* 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 <https://www.gnu.org/licenses/>.
*/
#include "config.h"
#include <string.h>
#include <gegl.h>
#include <gtk/gtk.h>
#include "libgimpbase/gimpbase.h"
#include "libgimpmath/gimpmath.h"
#include "libgimpwidgets/gimpwidgets.h"
#include "tools-types.h"
#include "core/gimpchannel.h"
#include "core/gimpdrawable.h"
#include "core/gimpdrawablefilter.h"
#include "core/gimpimage.h"
#include "core/gimplayer.h"
#include "core/gimplayermask.h"
#include "widgets/gimphelp-ids.h"
#include "display/gimpdisplay.h"
#include "display/gimptoolgui.h"
#include "gimpoffsettool.h"
#include "gimpfilteroptions.h"
#include "gimptoolcontrol.h"
#include "gimp-intl.h"
static gboolean gimp_offset_tool_initialize (GimpTool *tool,
GimpDisplay *display,
GError **error);
static void gimp_offset_tool_control (GimpTool *tool,
GimpToolAction action,
GimpDisplay *display);
static void gimp_offset_tool_button_press (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpButtonPressType press_type,
GimpDisplay *display);
static void gimp_offset_tool_button_release (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpButtonReleaseType release_type,
GimpDisplay *display);
static void gimp_offset_tool_motion (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpDisplay *display);
static void gimp_offset_tool_oper_update (GimpTool *tool,
const GimpCoords *coords,
GdkModifierType state,
gboolean proximity,
GimpDisplay *display);
static void gimp_offset_tool_cursor_update (GimpTool *tool,
const GimpCoords *coords,
GdkModifierType state,
GimpDisplay *display);
static gchar * gimp_offset_tool_get_operation (GimpFilterTool *filter_tool,
gchar **description);
static void gimp_offset_tool_dialog (GimpFilterTool *filter_tool);
static void gimp_offset_tool_config_notify (GimpFilterTool *filter_tool,
GimpConfig *config,
const GParamSpec *pspec);
static void gimp_offset_tool_region_changed (GimpFilterTool *filter_tool);
static void gimp_offset_tool_offset_changed (GimpSizeEntry *se,
GimpOffsetTool *offset_tool);
static void gimp_offset_tool_half_xy_clicked (GtkButton *button,
GimpOffsetTool *offset_tool);
static void gimp_offset_tool_half_x_clicked (GtkButton *button,
GimpOffsetTool *offset_tool);
static void gimp_offset_tool_half_y_clicked (GtkButton *button,
GimpOffsetTool *offset_tool);
static void gimp_offset_tool_edge_behavior_toggled (GtkToggleButton *toggle,
GimpOffsetTool *offset_tool);
static void gimp_offset_tool_background_changed (GimpContext *context,
GeglColor *color,
GimpOffsetTool *offset_tool);
static gint gimp_offset_tool_get_width (GimpOffsetTool *offset_tool);
static gint gimp_offset_tool_get_height (GimpOffsetTool *offset_tool);
static void gimp_offset_tool_update (GimpOffsetTool *offset_tool);
static void gimp_offset_tool_halt (GimpOffsetTool *offset_tool);
G_DEFINE_TYPE (GimpOffsetTool, gimp_offset_tool,
GIMP_TYPE_FILTER_TOOL)
#define parent_class gimp_offset_tool_parent_class
void
gimp_offset_tool_register (GimpToolRegisterCallback callback,
gpointer data)
{
(* callback) (GIMP_TYPE_OFFSET_TOOL,
GIMP_TYPE_FILTER_OPTIONS, NULL,
GIMP_CONTEXT_PROP_MASK_BACKGROUND,
"gimp-offset-tool",
_("Offset"),
_("Shift the pixels, optionally wrapping them at the borders"),
N_("_Offset..."), NULL,
NULL, GIMP_HELP_TOOL_OFFSET,
GIMP_ICON_TOOL_OFFSET,
data);
}
static void
gimp_offset_tool_class_init (GimpOffsetToolClass *klass)
{
GimpToolClass *tool_class = GIMP_TOOL_CLASS (klass);
GimpFilterToolClass *filter_tool_class = GIMP_FILTER_TOOL_CLASS (klass);
tool_class->initialize = gimp_offset_tool_initialize;
tool_class->control = gimp_offset_tool_control;
tool_class->button_press = gimp_offset_tool_button_press;
tool_class->button_release = gimp_offset_tool_button_release;
tool_class->motion = gimp_offset_tool_motion;
tool_class->oper_update = gimp_offset_tool_oper_update;
tool_class->cursor_update = gimp_offset_tool_cursor_update;
filter_tool_class->get_operation = gimp_offset_tool_get_operation;
filter_tool_class->dialog = gimp_offset_tool_dialog;
filter_tool_class->config_notify = gimp_offset_tool_config_notify;
filter_tool_class->region_changed = gimp_offset_tool_region_changed;
}
static void
gimp_offset_tool_init (GimpOffsetTool *offset_tool)
{
GimpTool *tool = GIMP_TOOL (offset_tool);
gimp_tool_control_set_scroll_lock (tool->control, TRUE);
gimp_tool_control_set_precision (tool->control,
GIMP_CURSOR_PRECISION_PIXEL_CENTER);
}
static gboolean
gimp_offset_tool_initialize (GimpTool *tool,
GimpDisplay *display,
GError **error)
{
GimpFilterTool *filter_tool = GIMP_FILTER_TOOL (tool);
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (tool);
GimpContext *context = GIMP_CONTEXT (GIMP_TOOL_GET_OPTIONS (tool));
GimpImage *image;
GimpDrawable *drawable;
gdouble xres;
gdouble yres;
if (! GIMP_TOOL_CLASS (parent_class)->initialize (tool, display, error))
return FALSE;
if (g_list_length (tool->drawables) != 1)
{
if (g_list_length (tool->drawables) > 1)
gimp_tool_message_literal (tool, display,
_("Cannot modify multiple drawables. Select only one."));
else
gimp_tool_message_literal (tool, display, _("No selected drawables."));
return FALSE;
}
drawable = tool->drawables->data;
image = gimp_item_get_image (GIMP_ITEM (drawable));
gimp_image_get_resolution (image, &xres, &yres);
g_signal_handlers_block_by_func (offset_tool->offset_se,
gimp_offset_tool_offset_changed,
offset_tool);
gimp_size_entry_set_resolution (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 0,
xres, FALSE);
gimp_size_entry_set_resolution (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 1,
yres, FALSE);
if (GIMP_IS_LAYER (drawable))
gimp_tool_gui_set_description (filter_tool->gui, _("Offset Layer"));
else if (GIMP_IS_LAYER_MASK (drawable))
gimp_tool_gui_set_description (filter_tool->gui, _("Offset Layer Mask"));
else if (GIMP_IS_CHANNEL (drawable))
gimp_tool_gui_set_description (filter_tool->gui, _("Offset Channel"));
else
g_warning ("%s: unexpected drawable type", G_STRFUNC);
gtk_widget_set_sensitive (offset_tool->transparent_radio,
gimp_drawable_has_alpha (drawable));
g_signal_handlers_unblock_by_func (offset_tool->offset_se,
gimp_offset_tool_offset_changed,
offset_tool);
gegl_node_set (
filter_tool->operation,
"context", context,
NULL);
g_signal_connect (context, "background-changed",
G_CALLBACK (gimp_offset_tool_background_changed),
offset_tool);
gimp_offset_tool_update (offset_tool);
return TRUE;
}
static void
gimp_offset_tool_control (GimpTool *tool,
GimpToolAction action,
GimpDisplay *display)
{
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (tool);
switch (action)
{
case GIMP_TOOL_ACTION_PAUSE:
case GIMP_TOOL_ACTION_RESUME:
break;
case GIMP_TOOL_ACTION_HALT:
gimp_offset_tool_halt (offset_tool);
break;
case GIMP_TOOL_ACTION_COMMIT:
break;
}
GIMP_TOOL_CLASS (parent_class)->control (tool, action, display);
}
static gchar *
gimp_offset_tool_get_operation (GimpFilterTool *filter_tool,
gchar **description)
{
return g_strdup ("gimp:offset");
}
static void
gimp_offset_tool_button_press (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpButtonPressType press_type,
GimpDisplay *display)
{
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (tool);
offset_tool->dragging = ! gimp_filter_tool_on_guide (GIMP_FILTER_TOOL (tool),
coords, display);
if (! offset_tool->dragging)
{
GIMP_TOOL_CLASS (parent_class)->button_press (tool, coords, time, state,
press_type, display);
}
else
{
offset_tool->x = coords->x;
offset_tool->y = coords->y;
g_object_get (GIMP_FILTER_TOOL (tool)->config,
"x", &offset_tool->offset_x,
"y", &offset_tool->offset_y,
NULL);
tool->display = display;
gimp_tool_control_activate (tool->control);
gimp_tool_pop_status (tool, display);
gimp_tool_push_status_coords (tool, display,
GIMP_CURSOR_PRECISION_PIXEL_CENTER,
_("Offset: "),
0,
", ",
0,
NULL);
}
}
static void
gimp_offset_tool_button_release (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpButtonReleaseType release_type,
GimpDisplay *display)
{
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (tool);
if (! offset_tool->dragging)
{
GIMP_TOOL_CLASS (parent_class)->button_release (tool, coords, time, state,
release_type, display);
}
else
{
gimp_tool_control_halt (tool->control);
offset_tool->dragging = FALSE;
if (release_type == GIMP_BUTTON_RELEASE_CANCEL)
{
g_object_set (GIMP_FILTER_TOOL (tool)->config,
"x", offset_tool->offset_x,
"y", offset_tool->offset_y,
NULL);
}
}
}
static void
gimp_offset_tool_motion (GimpTool *tool,
const GimpCoords *coords,
guint32 time,
GdkModifierType state,
GimpDisplay *display)
{
GimpFilterTool *filter_tool = GIMP_FILTER_TOOL (tool);
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (tool);
if (! offset_tool->dragging)
{
GIMP_TOOL_CLASS (parent_class)->motion (tool, coords, time, state,
display);
}
else
{
GimpOffsetType type;
gint offset_x;
gint offset_y;
gint x;
gint y;
gint width;
gint height;
g_object_get (filter_tool->config,
"type", &type,
NULL);
offset_x = RINT (coords->x - offset_tool->x);
offset_y = RINT (coords->y - offset_tool->y);
x = offset_tool->offset_x + offset_x;
y = offset_tool->offset_y + offset_y;
width = gimp_offset_tool_get_width (offset_tool);
height = gimp_offset_tool_get_height (offset_tool);
if (type == GIMP_OFFSET_WRAP_AROUND)
{
x %= MAX (width, 1);
y %= MAX (height, 1);
}
else
{
x = CLAMP (x, -width, +width);
y = CLAMP (y, -height, +height);
}
g_object_set (filter_tool->config,
"x", x,
"y", y,
NULL);
gimp_tool_pop_status (tool, display);
gimp_tool_push_status_coords (tool, display,
GIMP_CURSOR_PRECISION_PIXEL_CENTER,
_("Offset: "),
offset_x,
", ",
offset_y,
NULL);
}
}
static void
gimp_offset_tool_oper_update (GimpTool *tool,
const GimpCoords *coords,
GdkModifierType state,
gboolean proximity,
GimpDisplay *display)
{
if (! tool->drawables ||
gimp_filter_tool_on_guide (GIMP_FILTER_TOOL (tool),
coords, display))
{
GIMP_TOOL_CLASS (parent_class)->oper_update (tool, coords, state,
proximity, display);
}
else
{
gimp_tool_pop_status (tool, display);
gimp_tool_push_status (tool, display, "%s",
_("Click-Drag to offset drawable"));
}
}
static void
gimp_offset_tool_cursor_update (GimpTool *tool,
const GimpCoords *coords,
GdkModifierType state,
GimpDisplay *display)
{
if (! tool->drawables ||
gimp_filter_tool_on_guide (GIMP_FILTER_TOOL (tool),
coords, display))
{
GIMP_TOOL_CLASS (parent_class)->cursor_update (tool, coords, state,
display);
}
else
{
gimp_tool_set_cursor (tool, display,
GIMP_CURSOR_MOUSE,
GIMP_TOOL_CURSOR_MOVE,
GIMP_CURSOR_MODIFIER_NONE);
}
}
static void
gimp_offset_tool_dialog (GimpFilterTool *filter_tool)
{
GimpOffsetTool *offset_tool = GIMP_OFFSET_TOOL (filter_tool);
GtkWidget *main_vbox;
GtkWidget *vbox;
GtkWidget *hbox;
GtkWidget *button;
GtkWidget *spinbutton;
GtkWidget *frame;
GtkAdjustment *adjustment;
main_vbox = gimp_filter_tool_dialog_get_vbox (filter_tool);
/* The offset frame */
frame = gimp_frame_new (_("Offset"));
gtk_box_pack_start (GTK_BOX (main_vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 6);
gtk_container_add (GTK_CONTAINER (frame), vbox);
gtk_widget_show (vbox);
adjustment = (GtkAdjustment *)
gtk_adjustment_new (1, 1, 1, 1, 10, 0);
spinbutton = gimp_spin_button_new (adjustment, 1.0, 2);
gtk_spin_button_set_numeric (GTK_SPIN_BUTTON (spinbutton), TRUE);
gtk_entry_set_width_chars (GTK_ENTRY (spinbutton), 10);
offset_tool->offset_se = gimp_size_entry_new (1, gimp_unit_pixel (), "%a",
TRUE, TRUE, FALSE, 10,
GIMP_SIZE_ENTRY_UPDATE_SIZE);
gimp_size_entry_add_field (GIMP_SIZE_ENTRY (offset_tool->offset_se),
GTK_SPIN_BUTTON (spinbutton), NULL);
gtk_grid_attach (GTK_GRID (offset_tool->offset_se), spinbutton, 1, 0, 1, 1);
gtk_widget_show (spinbutton);
gimp_size_entry_attach_label (GIMP_SIZE_ENTRY (offset_tool->offset_se),
_("_X:"), 0, 0, 0.0);
gimp_size_entry_attach_label (GIMP_SIZE_ENTRY (offset_tool->offset_se),
_("_Y:"), 1, 0, 0.0);
gtk_box_pack_start (GTK_BOX (vbox), offset_tool->offset_se, FALSE, FALSE, 0);
gtk_widget_show (offset_tool->offset_se);
gimp_size_entry_set_unit (GIMP_SIZE_ENTRY (offset_tool->offset_se),
gimp_unit_pixel ());
g_signal_connect (offset_tool->offset_se, "refval-changed",
G_CALLBACK (gimp_offset_tool_offset_changed),
offset_tool);
g_signal_connect (offset_tool->offset_se, "value-changed",
G_CALLBACK (gimp_offset_tool_offset_changed),
offset_tool);
button = gtk_button_new_with_mnemonic (_("By width/_2, height/2"));
gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
gtk_widget_show (button);
g_signal_connect (button, "clicked",
G_CALLBACK (gimp_offset_tool_half_xy_clicked),
offset_tool);
hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
gtk_widget_show (hbox);
button = gtk_button_new_with_mnemonic (_("By _width/2"));
gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 0);
gtk_widget_show (button);
g_signal_connect (button, "clicked",
G_CALLBACK (gimp_offset_tool_half_x_clicked),
offset_tool);
button = gtk_button_new_with_mnemonic (_("By _height/2"));
gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 0);
gtk_widget_show (button);
g_signal_connect (button, "clicked",
G_CALLBACK (gimp_offset_tool_half_y_clicked),
offset_tool);
/* The edge behavior frame */
frame = gimp_int_radio_group_new (TRUE, _("Edge Behavior"),
G_CALLBACK (gimp_offset_tool_edge_behavior_toggled),
offset_tool, NULL,
GIMP_OFFSET_WRAP_AROUND,
_("W_rap around"),
GIMP_OFFSET_WRAP_AROUND, NULL,
_("Fill with _background color"),
GIMP_OFFSET_BACKGROUND, NULL,
_("Make _transparent"),
GIMP_OFFSET_TRANSPARENT,
&offset_tool->transparent_radio,
NULL);
gtk_box_pack_start (GTK_BOX (main_vbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
}
static void
gimp_offset_tool_config_notify (GimpFilterTool *filter_tool,
GimpConfig *config,
const GParamSpec *pspec)
{
gimp_offset_tool_update (GIMP_OFFSET_TOOL (filter_tool));
GIMP_FILTER_TOOL_CLASS (parent_class)->config_notify (filter_tool,
config, pspec);
}
static void
gimp_offset_tool_region_changed (GimpFilterTool *filter_tool)
{
gimp_offset_tool_update (GIMP_OFFSET_TOOL (filter_tool));
}
static void
gimp_offset_tool_offset_changed (GimpSizeEntry *se,
GimpOffsetTool *offset_tool)
{
g_object_set (GIMP_FILTER_TOOL (offset_tool)->config,
"x", (gint) gimp_size_entry_get_refval (se, 0),
"y", (gint) gimp_size_entry_get_refval (se, 1),
NULL);
}
static void
gimp_offset_tool_half_xy_clicked (GtkButton *button,
GimpOffsetTool *offset_tool)
{
g_object_set (GIMP_FILTER_TOOL (offset_tool)->config,
"x", gimp_offset_tool_get_width (offset_tool) / 2,
"y", gimp_offset_tool_get_height (offset_tool) / 2,
NULL);
}
static void
gimp_offset_tool_half_x_clicked (GtkButton *button,
GimpOffsetTool *offset_tool)
{
g_object_set (GIMP_FILTER_TOOL (offset_tool)->config,
"x", gimp_offset_tool_get_width (offset_tool) / 2,
NULL);
}
static void
gimp_offset_tool_half_y_clicked (GtkButton *button,
GimpOffsetTool *offset_tool)
{
g_object_set (GIMP_FILTER_TOOL (offset_tool)->config,
"y", gimp_offset_tool_get_height (offset_tool) / 2,
NULL);
}
static void
gimp_offset_tool_edge_behavior_toggled (GtkToggleButton *toggle,
GimpOffsetTool *offset_tool)
{
if (gtk_toggle_button_get_active (toggle))
{
GimpOffsetType type;
type = GPOINTER_TO_INT (g_object_get_data (G_OBJECT (toggle),
"gimp-item-data"));
g_object_set (GIMP_FILTER_TOOL (offset_tool)->config,
"type", type,
NULL);
}
}
static void
gimp_offset_tool_background_changed (GimpContext *context,
GeglColor *color,
GimpOffsetTool *offset_tool)
{
GimpFilterTool *filter_tool = GIMP_FILTER_TOOL (offset_tool);
GimpOffsetType type;
g_object_get (filter_tool->config,
"type", &type,
NULL);
if (type == GIMP_OFFSET_BACKGROUND)
{
gegl_node_set (filter_tool->operation,
"context", context,
NULL);
gimp_drawable_filter_apply (filter_tool->filter, NULL);
}
}
static gint
gimp_offset_tool_get_width (GimpOffsetTool *offset_tool)
{
GeglRectangle drawable_area;
gint drawable_offset_x;
gint drawable_offset_y;
if (gimp_filter_tool_get_drawable_area (GIMP_FILTER_TOOL (offset_tool),
&drawable_offset_x,
&drawable_offset_y,
&drawable_area) &&
! gegl_rectangle_is_empty (&drawable_area))
{
return drawable_area.width;
}
return 0;
}
static gint
gimp_offset_tool_get_height (GimpOffsetTool *offset_tool)
{
GeglRectangle drawable_area;
gint drawable_offset_x;
gint drawable_offset_y;
if (gimp_filter_tool_get_drawable_area (GIMP_FILTER_TOOL (offset_tool),
&drawable_offset_x,
&drawable_offset_y,
&drawable_area) &&
! gegl_rectangle_is_empty (&drawable_area))
{
return drawable_area.height;
}
return 0;
}
static void
gimp_offset_tool_update (GimpOffsetTool *offset_tool)
{
GimpTool *tool = GIMP_TOOL (offset_tool);
GimpFilterTool *filter_tool = GIMP_FILTER_TOOL (offset_tool);
GimpOffsetType orig_type;
gint orig_x;
gint orig_y;
GimpOffsetType type;
gint x;
gint y;
gint width;
gint height;
g_object_get (filter_tool->config,
"type", &orig_type,
"x", &orig_x,
"y", &orig_y,
NULL);
width = gimp_offset_tool_get_width (offset_tool);
height = gimp_offset_tool_get_height (offset_tool);
x = CLAMP (orig_x, -width, +width);
y = CLAMP (orig_y, -height, +height);
type = orig_type;
if (tool->drawables &&
! gimp_drawable_has_alpha (tool->drawables->data) &&
type == GIMP_OFFSET_TRANSPARENT)
{
type = GIMP_OFFSET_BACKGROUND;
}
if (x != orig_x ||
y != orig_y ||
type != orig_type)
{
g_object_set (filter_tool->config,
"type", type,
"x", x,
"y", y,
NULL);
}
if (offset_tool->offset_se)
{
gint width = gimp_offset_tool_get_width (offset_tool);
gint height = gimp_offset_tool_get_height (offset_tool);
g_signal_handlers_block_by_func (offset_tool->offset_se,
gimp_offset_tool_offset_changed,
offset_tool);
gimp_size_entry_set_refval_boundaries (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 0,
-width, +width);
gimp_size_entry_set_refval_boundaries (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 1,
-height, +height);
gimp_size_entry_set_size (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 0,
0, width);
gimp_size_entry_set_size (
GIMP_SIZE_ENTRY (offset_tool->offset_se), 1,
0, height);
gimp_size_entry_set_refval (GIMP_SIZE_ENTRY (offset_tool->offset_se), 0,
x);
gimp_size_entry_set_refval (GIMP_SIZE_ENTRY (offset_tool->offset_se), 1,
y);
g_signal_handlers_unblock_by_func (offset_tool->offset_se,
gimp_offset_tool_offset_changed,
offset_tool);
}
if (offset_tool->transparent_radio)
{
gimp_int_radio_group_set_active (
GTK_RADIO_BUTTON (offset_tool->transparent_radio),
type);
}
}
static void
gimp_offset_tool_halt (GimpOffsetTool *offset_tool)
{
GimpContext *context = GIMP_CONTEXT (GIMP_TOOL_GET_OPTIONS (offset_tool));
offset_tool->offset_se = NULL;
offset_tool->transparent_radio = NULL;
g_signal_handlers_disconnect_by_func (
context,
gimp_offset_tool_background_changed,
offset_tool);
}