923 lines
35 KiB
C
923 lines
35 KiB
C
/* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gegl.h>
|
|
#include <gtk/gtk.h>
|
|
|
|
#include "libpikamath/pikamath.h"
|
|
#include "libpikawidgets/pikawidgets.h"
|
|
|
|
#include "tools-types.h"
|
|
|
|
#include "core/pikachannel-select.h"
|
|
#include "core/pikachannel.h"
|
|
#include "core/pikaimage.h"
|
|
#include "core/pikalayer-floating-selection.h"
|
|
#include "core/pikapickable.h"
|
|
|
|
#include "widgets/pikahelp-ids.h"
|
|
#include "widgets/pikawidgets-utils.h"
|
|
|
|
#include "display/pikadisplay.h"
|
|
#include "display/pikadisplayshell.h"
|
|
#include "display/pikatoolrectangle.h"
|
|
|
|
#include "pikaeditselectiontool.h"
|
|
#include "pikarectangleoptions.h"
|
|
#include "pikarectangleselecttool.h"
|
|
#include "pikarectangleselectoptions.h"
|
|
#include "pikatoolcontrol.h"
|
|
|
|
#include "pika-intl.h"
|
|
|
|
|
|
struct _PikaRectangleSelectToolPrivate
|
|
{
|
|
PikaChannelOps operation; /* remember for use when modifying */
|
|
gboolean use_saved_op; /* use operation or get from options */
|
|
|
|
gdouble press_x;
|
|
gdouble press_y;
|
|
|
|
PikaToolWidget *widget;
|
|
PikaToolWidget *grab_widget;
|
|
GList *bindings;
|
|
};
|
|
|
|
|
|
static void pika_rectangle_select_tool_control (PikaTool *tool,
|
|
PikaToolAction action,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_button_press (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaButtonPressType press_type,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_button_release (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaButtonReleaseType release_type,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_motion (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaDisplay *display);
|
|
static gboolean pika_rectangle_select_tool_key_press (PikaTool *tool,
|
|
GdkEventKey *kevent,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_oper_update (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
GdkModifierType state,
|
|
gboolean proximity,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_cursor_update (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
GdkModifierType state,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_options_notify (PikaTool *tool,
|
|
PikaToolOptions *options,
|
|
const GParamSpec *pspec);
|
|
|
|
static gboolean pika_rectangle_select_tool_select (PikaRectangleSelectTool *rect_tool,
|
|
gint x,
|
|
gint y,
|
|
gint w,
|
|
gint h);
|
|
static void pika_rectangle_select_tool_real_select (PikaRectangleSelectTool *rect_tool,
|
|
PikaChannelOps operation,
|
|
gint x,
|
|
gint y,
|
|
gint w,
|
|
gint h);
|
|
|
|
static void pika_rectangle_select_tool_rectangle_response
|
|
(PikaToolWidget *widget,
|
|
gint response_id,
|
|
PikaRectangleSelectTool *rect_tool);
|
|
static void pika_rectangle_select_tool_rectangle_change_complete
|
|
(PikaToolWidget *widget,
|
|
PikaRectangleSelectTool *rect_tool);
|
|
|
|
static void pika_rectangle_select_tool_start (PikaRectangleSelectTool *rect_tool,
|
|
PikaDisplay *display);
|
|
static void pika_rectangle_select_tool_commit (PikaRectangleSelectTool *rect_tool);
|
|
static void pika_rectangle_select_tool_halt (PikaRectangleSelectTool *rect_tool);
|
|
|
|
static PikaChannelOps
|
|
pika_rectangle_select_tool_get_operation (PikaRectangleSelectTool *rect_tool);
|
|
static void pika_rectangle_select_tool_update_option_defaults
|
|
(PikaRectangleSelectTool *rect_tool,
|
|
gboolean ignore_pending);
|
|
static void pika_rectangle_select_tool_update (PikaRectangleSelectTool *rect_tool);
|
|
static void pika_rectangle_select_tool_auto_shrink (PikaRectangleSelectTool *rect_tool);
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (PikaRectangleSelectTool, pika_rectangle_select_tool,
|
|
PIKA_TYPE_SELECTION_TOOL)
|
|
|
|
#define parent_class pika_rectangle_select_tool_parent_class
|
|
|
|
|
|
void
|
|
pika_rectangle_select_tool_register (PikaToolRegisterCallback callback,
|
|
gpointer data)
|
|
{
|
|
(* callback) (PIKA_TYPE_RECTANGLE_SELECT_TOOL,
|
|
PIKA_TYPE_RECTANGLE_SELECT_OPTIONS,
|
|
pika_rectangle_select_options_gui,
|
|
0,
|
|
"pika-rect-select-tool",
|
|
_("Rectangle Select"),
|
|
_("Rectangle Select Tool: Select a rectangular region"),
|
|
N_("_Rectangle Select"), "R",
|
|
NULL, PIKA_HELP_TOOL_RECT_SELECT,
|
|
PIKA_ICON_TOOL_RECT_SELECT,
|
|
data);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_class_init (PikaRectangleSelectToolClass *klass)
|
|
{
|
|
PikaToolClass *tool_class = PIKA_TOOL_CLASS (klass);
|
|
|
|
tool_class->control = pika_rectangle_select_tool_control;
|
|
tool_class->button_press = pika_rectangle_select_tool_button_press;
|
|
tool_class->button_release = pika_rectangle_select_tool_button_release;
|
|
tool_class->motion = pika_rectangle_select_tool_motion;
|
|
tool_class->key_press = pika_rectangle_select_tool_key_press;
|
|
tool_class->oper_update = pika_rectangle_select_tool_oper_update;
|
|
tool_class->cursor_update = pika_rectangle_select_tool_cursor_update;
|
|
tool_class->options_notify = pika_rectangle_select_tool_options_notify;
|
|
|
|
klass->select = pika_rectangle_select_tool_real_select;
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_init (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
|
|
rect_tool->private =
|
|
pika_rectangle_select_tool_get_instance_private (rect_tool);
|
|
|
|
pika_tool_control_set_wants_click (tool->control, TRUE);
|
|
pika_tool_control_set_active_modifiers (tool->control,
|
|
PIKA_TOOL_ACTIVE_MODIFIERS_SEPARATE);
|
|
pika_tool_control_set_precision (tool->control,
|
|
PIKA_CURSOR_PRECISION_PIXEL_BORDER);
|
|
pika_tool_control_set_tool_cursor (tool->control,
|
|
PIKA_TOOL_CURSOR_RECT_SELECT);
|
|
pika_tool_control_set_preserve (tool->control, FALSE);
|
|
pika_tool_control_set_dirty_mask (tool->control,
|
|
PIKA_DIRTY_IMAGE_SIZE |
|
|
PIKA_DIRTY_SELECTION);
|
|
pika_tool_control_set_dirty_action (tool->control,
|
|
PIKA_TOOL_ACTION_COMMIT);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_control (PikaTool *tool,
|
|
PikaToolAction action,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
|
|
switch (action)
|
|
{
|
|
case PIKA_TOOL_ACTION_PAUSE:
|
|
case PIKA_TOOL_ACTION_RESUME:
|
|
break;
|
|
|
|
case PIKA_TOOL_ACTION_HALT:
|
|
pika_rectangle_select_tool_halt (rect_tool);
|
|
break;
|
|
|
|
case PIKA_TOOL_ACTION_COMMIT:
|
|
pika_rectangle_select_tool_commit (rect_tool);
|
|
break;
|
|
}
|
|
|
|
PIKA_TOOL_CLASS (parent_class)->control (tool, action, display);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_button_press (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaButtonPressType press_type,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *private = rect_tool->private;
|
|
PikaRectangleFunction function;
|
|
|
|
if (tool->display && display != tool->display)
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_COMMIT, tool->display);
|
|
|
|
if (pika_selection_tool_start_edit (PIKA_SELECTION_TOOL (tool),
|
|
display, coords))
|
|
{
|
|
/* In some cases we want to finish the rectangle select tool
|
|
* and hand over responsibility to the selection tool
|
|
*/
|
|
|
|
gboolean zero_rect = FALSE;
|
|
|
|
if (private->widget)
|
|
{
|
|
gdouble x1, y1, x2, y2;
|
|
|
|
pika_tool_rectangle_get_public_rect (PIKA_TOOL_RECTANGLE (private->widget),
|
|
&x1, &y1, &x2, &y2);
|
|
if (x1 == x2 && y1 == y2)
|
|
zero_rect = TRUE;
|
|
}
|
|
|
|
/* Don't commit a zero-size rectangle, it would look like a
|
|
* click to commit() and that could anchor the floating
|
|
* selection or do other evil things. Instead, simply cancel a
|
|
* zero-size rectangle. See bug #796073.
|
|
*/
|
|
if (zero_rect)
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_HALT, display);
|
|
else
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_COMMIT, display);
|
|
|
|
pika_rectangle_select_tool_update_option_defaults (rect_tool, TRUE);
|
|
return;
|
|
}
|
|
|
|
if (! tool->display)
|
|
{
|
|
pika_rectangle_select_tool_start (rect_tool, display);
|
|
|
|
pika_tool_widget_hover (private->widget, coords, state, TRUE);
|
|
|
|
/* HACK: force CREATING on a newly created rectangle; otherwise,
|
|
* the above binding of properties would cause the rectangle to
|
|
* start with the size from tool options.
|
|
*/
|
|
pika_tool_rectangle_set_function (PIKA_TOOL_RECTANGLE (private->widget),
|
|
PIKA_TOOL_RECTANGLE_CREATING);
|
|
}
|
|
|
|
/* if the shift or ctrl keys are down, we don't want to adjust, we
|
|
* want to create a new rectangle, regardless of pointer loc
|
|
*/
|
|
if (state & (pika_get_extend_selection_mask () |
|
|
pika_get_modify_selection_mask ()))
|
|
{
|
|
pika_tool_rectangle_set_function (PIKA_TOOL_RECTANGLE (private->widget),
|
|
PIKA_TOOL_RECTANGLE_CREATING);
|
|
}
|
|
|
|
if (pika_tool_widget_button_press (private->widget, coords, time, state,
|
|
press_type))
|
|
{
|
|
private->grab_widget = private->widget;
|
|
}
|
|
|
|
private->press_x = coords->x;
|
|
private->press_y = coords->y;
|
|
|
|
/* if we have an existing rectangle in the current display, then
|
|
* we have already "executed", and need to undo at this point,
|
|
* unless the user has done something in the meantime
|
|
*/
|
|
function =
|
|
pika_tool_rectangle_get_function (PIKA_TOOL_RECTANGLE (private->widget));
|
|
|
|
if (function == PIKA_TOOL_RECTANGLE_CREATING)
|
|
private->use_saved_op = FALSE;
|
|
|
|
pika_selection_tool_start_change (
|
|
PIKA_SELECTION_TOOL (tool),
|
|
function == PIKA_TOOL_RECTANGLE_CREATING,
|
|
pika_rectangle_select_tool_get_operation (rect_tool));
|
|
|
|
pika_tool_control_activate (tool->control);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_button_release (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaButtonReleaseType release_type,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
pika_tool_control_halt (tool->control);
|
|
|
|
pika_selection_tool_end_change (PIKA_SELECTION_TOOL (tool),
|
|
/* if the user has not moved the mouse,
|
|
* cancel the change
|
|
*/
|
|
release_type == PIKA_BUTTON_RELEASE_CLICK ||
|
|
release_type == PIKA_BUTTON_RELEASE_CANCEL);
|
|
|
|
pika_tool_pop_status (tool, display);
|
|
|
|
if (priv->grab_widget)
|
|
{
|
|
pika_tool_widget_button_release (priv->grab_widget,
|
|
coords, time, state, release_type);
|
|
priv->grab_widget = NULL;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_motion (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
guint32 time,
|
|
GdkModifierType state,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
if (priv->grab_widget)
|
|
{
|
|
pika_tool_widget_motion (priv->grab_widget, coords, time, state);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
pika_rectangle_select_tool_key_press (PikaTool *tool,
|
|
GdkEventKey *kevent,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
if (priv->widget && display == tool->display)
|
|
{
|
|
if (pika_tool_widget_key_press (priv->widget, kevent))
|
|
return TRUE;
|
|
}
|
|
|
|
return pika_edit_selection_tool_key_press (tool, kevent, display);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_oper_update (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
GdkModifierType state,
|
|
gboolean proximity,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
if (priv->widget && display == tool->display)
|
|
{
|
|
pika_tool_widget_hover (priv->widget, coords, state, proximity);
|
|
}
|
|
|
|
PIKA_TOOL_CLASS (parent_class)->oper_update (tool, coords, state, proximity,
|
|
display);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_cursor_update (PikaTool *tool,
|
|
const PikaCoords *coords,
|
|
GdkModifierType state,
|
|
PikaDisplay *display)
|
|
{
|
|
PikaRectangleSelectTool *rect_tool = PIKA_RECTANGLE_SELECT_TOOL (tool);
|
|
PikaRectangleSelectToolPrivate *private = rect_tool->private;
|
|
PikaCursorType cursor = PIKA_CURSOR_CROSSHAIR_SMALL;
|
|
PikaCursorModifier modifier = PIKA_CURSOR_MODIFIER_NONE;
|
|
|
|
if (private->widget && display == tool->display)
|
|
{
|
|
pika_tool_widget_get_cursor (private->widget, coords, state,
|
|
&cursor, NULL, &modifier);
|
|
}
|
|
|
|
pika_tool_control_set_cursor (tool->control, cursor);
|
|
pika_tool_control_set_cursor_modifier (tool->control, modifier);
|
|
|
|
/* override the previous if shift or ctrl are down */
|
|
if (state & (pika_get_extend_selection_mask () |
|
|
pika_get_modify_selection_mask ()))
|
|
{
|
|
pika_tool_control_set_cursor (tool->control,
|
|
PIKA_CURSOR_CROSSHAIR_SMALL);
|
|
}
|
|
|
|
PIKA_TOOL_CLASS (parent_class)->cursor_update (tool, coords, state, display);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_options_notify (PikaTool *tool,
|
|
PikaToolOptions *options,
|
|
const GParamSpec *pspec)
|
|
{
|
|
if (! strcmp (pspec->name, "antialias") ||
|
|
! strcmp (pspec->name, "feather") ||
|
|
! strcmp (pspec->name, "feather-radius") ||
|
|
! strcmp (pspec->name, "round-corners") ||
|
|
! strcmp (pspec->name, "corner-radius"))
|
|
{
|
|
pika_rectangle_select_tool_update (PIKA_RECTANGLE_SELECT_TOOL (tool));
|
|
}
|
|
|
|
PIKA_TOOL_CLASS (parent_class)->options_notify (tool, options, pspec);
|
|
}
|
|
|
|
static gboolean
|
|
pika_rectangle_select_tool_select (PikaRectangleSelectTool *rect_tool,
|
|
gint x,
|
|
gint y,
|
|
gint w,
|
|
gint h)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaImage *image = pika_display_get_image (tool->display);
|
|
gboolean rectangle_exists;
|
|
PikaChannelOps operation;
|
|
|
|
pika_tool_pop_status (tool, tool->display);
|
|
|
|
rectangle_exists = (x <= pika_image_get_width (image) &&
|
|
y <= pika_image_get_height (image) &&
|
|
x + w >= 0 &&
|
|
y + h >= 0 &&
|
|
w > 0 &&
|
|
h > 0);
|
|
|
|
operation = pika_rectangle_select_tool_get_operation (rect_tool);
|
|
|
|
/* if rectangle exists, turn it into a selection */
|
|
if (rectangle_exists)
|
|
{
|
|
pika_selection_tool_start_change (PIKA_SELECTION_TOOL (rect_tool),
|
|
FALSE,
|
|
operation);
|
|
|
|
PIKA_RECTANGLE_SELECT_TOOL_GET_CLASS (rect_tool)->select (rect_tool,
|
|
operation,
|
|
x, y, w, h);
|
|
|
|
pika_selection_tool_end_change (PIKA_SELECTION_TOOL (rect_tool),
|
|
FALSE);
|
|
}
|
|
|
|
return rectangle_exists;
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_real_select (PikaRectangleSelectTool *rect_tool,
|
|
PikaChannelOps operation,
|
|
gint x,
|
|
gint y,
|
|
gint w,
|
|
gint h)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaSelectionOptions *options = PIKA_SELECTION_TOOL_GET_OPTIONS (tool);
|
|
PikaRectangleSelectOptions *rect_options;
|
|
PikaChannel *channel;
|
|
|
|
rect_options = PIKA_RECTANGLE_SELECT_TOOL_GET_OPTIONS (tool);
|
|
|
|
channel = pika_image_get_mask (pika_display_get_image (tool->display));
|
|
|
|
if (rect_options->round_corners)
|
|
{
|
|
/* To prevent elliptification of the rectangle,
|
|
* we must cap the corner radius.
|
|
*/
|
|
gdouble max = MIN (w / 2.0, h / 2.0);
|
|
gdouble radius = MIN (rect_options->corner_radius, max);
|
|
|
|
pika_channel_select_round_rect (channel,
|
|
x, y, w, h,
|
|
radius, radius,
|
|
operation,
|
|
options->antialias,
|
|
options->feather,
|
|
options->feather_radius,
|
|
options->feather_radius,
|
|
TRUE);
|
|
}
|
|
else
|
|
{
|
|
pika_channel_select_rectangle (channel,
|
|
x, y, w, h,
|
|
operation,
|
|
options->feather,
|
|
options->feather_radius,
|
|
options->feather_radius,
|
|
TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_rectangle_response (PikaToolWidget *widget,
|
|
gint response_id,
|
|
PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
|
|
switch (response_id)
|
|
{
|
|
case PIKA_TOOL_WIDGET_RESPONSE_CONFIRM:
|
|
{
|
|
gdouble x1, y1, x2, y2;
|
|
|
|
pika_tool_rectangle_get_public_rect (PIKA_TOOL_RECTANGLE (widget),
|
|
&x1, &y1, &x2, &y2);
|
|
if (x1 == x2 && y1 == y2)
|
|
{
|
|
/* if there are no extents, we got here because of a
|
|
* click, call commit() directly because we might want to
|
|
* reconfigure the rectangle and continue, instead of
|
|
* HALTing it like calling COMMIT would do
|
|
*/
|
|
pika_rectangle_select_tool_commit (rect_tool);
|
|
|
|
pika_tool_rectangle_get_public_rect (PIKA_TOOL_RECTANGLE (widget),
|
|
&x1, &y1, &x2, &y2);
|
|
if (x1 == x2 && y1 == y2)
|
|
{
|
|
/* if there still is no rectangle after the
|
|
* tool_commit(), the click was outside the selection
|
|
* and we HALT to get rid of a zero-size tool widget.
|
|
*/
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_HALT, tool->display);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_COMMIT, tool->display);
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PIKA_TOOL_WIDGET_RESPONSE_CANCEL:
|
|
pika_tool_control (tool, PIKA_TOOL_ACTION_HALT, tool->display);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_rectangle_change_complete (PikaToolWidget *widget,
|
|
PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
pika_rectangle_select_tool_update (rect_tool);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_start (PikaRectangleSelectTool *rect_tool,
|
|
PikaDisplay *display)
|
|
{
|
|
static const gchar *properties[] =
|
|
{
|
|
"highlight",
|
|
"highlight-opacity",
|
|
"guide",
|
|
"round-corners",
|
|
"corner-radius",
|
|
"x",
|
|
"y",
|
|
"width",
|
|
"height",
|
|
"fixed-rule-active",
|
|
"fixed-rule",
|
|
"desired-fixed-width",
|
|
"desired-fixed-height",
|
|
"desired-fixed-size-width",
|
|
"desired-fixed-size-height",
|
|
"aspect-numerator",
|
|
"aspect-denominator",
|
|
"fixed-center"
|
|
};
|
|
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaRectangleSelectToolPrivate *private = rect_tool->private;
|
|
PikaDisplayShell *shell = pika_display_get_shell (display);
|
|
PikaRectangleSelectOptions *options;
|
|
PikaToolWidget *widget;
|
|
gboolean draw_ellipse;
|
|
gint i;
|
|
|
|
options = PIKA_RECTANGLE_SELECT_TOOL_GET_OPTIONS (rect_tool);
|
|
|
|
tool->display = display;
|
|
|
|
private->widget = widget = pika_tool_rectangle_new (shell);
|
|
|
|
draw_ellipse = PIKA_RECTANGLE_SELECT_TOOL_GET_CLASS (rect_tool)->draw_ellipse;
|
|
|
|
g_object_set (widget,
|
|
"draw-ellipse", draw_ellipse,
|
|
"status-title", draw_ellipse ? _("Ellipse: ") : _("Rectangle: "),
|
|
NULL);
|
|
|
|
pika_draw_tool_set_widget (PIKA_DRAW_TOOL (tool), widget);
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (properties); i++)
|
|
{
|
|
GBinding *binding =
|
|
g_object_bind_property (G_OBJECT (options), properties[i],
|
|
G_OBJECT (widget), properties[i],
|
|
G_BINDING_SYNC_CREATE |
|
|
G_BINDING_BIDIRECTIONAL);
|
|
|
|
private->bindings = g_list_prepend (private->bindings, binding);
|
|
}
|
|
|
|
pika_rectangle_options_connect (PIKA_RECTANGLE_OPTIONS (options),
|
|
pika_display_get_image (shell->display),
|
|
G_CALLBACK (pika_rectangle_select_tool_auto_shrink),
|
|
rect_tool);
|
|
|
|
g_signal_connect (widget, "response",
|
|
G_CALLBACK (pika_rectangle_select_tool_rectangle_response),
|
|
rect_tool);
|
|
g_signal_connect (widget, "change-complete",
|
|
G_CALLBACK (pika_rectangle_select_tool_rectangle_change_complete),
|
|
rect_tool);
|
|
|
|
pika_draw_tool_start (PIKA_DRAW_TOOL (tool), display);
|
|
}
|
|
|
|
/* This function is called if the user clicks and releases the left
|
|
* button without moving it. There are the things we might want
|
|
* to do here:
|
|
* 1) If there is an existing rectangle and we are inside it, we
|
|
* convert it into a selection.
|
|
* 2) If there is an existing rectangle and we are outside it, we
|
|
* clear it.
|
|
* 3) If there is no rectangle and there is a floating selection,
|
|
* we anchor it.
|
|
* 4) If there is no rectangle and we are inside the selection, we
|
|
* create a rectangle from the selection bounds.
|
|
* 5) If there is no rectangle and we are outside the selection,
|
|
* we clear the selection.
|
|
*/
|
|
static void
|
|
pika_rectangle_select_tool_commit (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
if (priv->widget)
|
|
{
|
|
gdouble x1, y1, x2, y2;
|
|
gint w, h;
|
|
|
|
pika_tool_rectangle_get_public_rect (PIKA_TOOL_RECTANGLE (priv->widget),
|
|
&x1, &y1, &x2, &y2);
|
|
w = x2 - x1;
|
|
h = y2 - y1;
|
|
|
|
if (w == 0 && h == 0)
|
|
{
|
|
PikaImage *image = pika_display_get_image (tool->display);
|
|
PikaChannel *selection = pika_image_get_mask (image);
|
|
gint press_x;
|
|
gint press_y;
|
|
|
|
if (pika_image_get_floating_selection (image))
|
|
{
|
|
floating_sel_anchor (pika_image_get_floating_selection (image));
|
|
pika_image_flush (image);
|
|
|
|
return;
|
|
}
|
|
|
|
press_x = ROUND (priv->press_x);
|
|
press_y = ROUND (priv->press_y);
|
|
|
|
/* if the click was inside the marching ants */
|
|
if (pika_pickable_get_opacity_at (PIKA_PICKABLE (selection),
|
|
press_x, press_y) > 0.5)
|
|
{
|
|
gint x, y, w, h;
|
|
|
|
if (pika_item_bounds (PIKA_ITEM (selection), &x, &y, &w, &h))
|
|
{
|
|
g_object_set (priv->widget,
|
|
"x1", (gdouble) x,
|
|
"y1", (gdouble) y,
|
|
"x2", (gdouble) (x + w),
|
|
"y2", (gdouble) (y + h),
|
|
NULL);
|
|
}
|
|
|
|
pika_rectangle_select_tool_update (rect_tool);
|
|
}
|
|
else
|
|
{
|
|
PikaChannelOps operation;
|
|
|
|
/* prevent this change from halting the tool */
|
|
pika_tool_control_push_preserve (tool->control, TRUE);
|
|
|
|
/* We can conceptually think of a click outside of the
|
|
* selection as adding a 0px selection. Behave intuitively
|
|
* for the current selection mode
|
|
*/
|
|
operation = pika_rectangle_select_tool_get_operation (rect_tool);
|
|
|
|
switch (operation)
|
|
{
|
|
case PIKA_CHANNEL_OP_REPLACE:
|
|
case PIKA_CHANNEL_OP_INTERSECT:
|
|
pika_channel_clear (selection, NULL, TRUE);
|
|
pika_image_flush (image);
|
|
break;
|
|
|
|
case PIKA_CHANNEL_OP_ADD:
|
|
case PIKA_CHANNEL_OP_SUBTRACT:
|
|
default:
|
|
/* Do nothing */
|
|
break;
|
|
}
|
|
|
|
pika_tool_control_pop_preserve (tool->control);
|
|
}
|
|
}
|
|
|
|
pika_rectangle_select_tool_update_option_defaults (rect_tool, FALSE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_halt (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
PikaRectangleSelectOptions *options;
|
|
|
|
options = PIKA_RECTANGLE_SELECT_TOOL_GET_OPTIONS (rect_tool);
|
|
|
|
if (tool->display)
|
|
{
|
|
PikaDisplayShell *shell = pika_display_get_shell (tool->display);
|
|
|
|
pika_display_shell_set_highlight (shell, NULL, 0.0);
|
|
|
|
pika_rectangle_options_disconnect (PIKA_RECTANGLE_OPTIONS (options),
|
|
G_CALLBACK (pika_rectangle_select_tool_auto_shrink),
|
|
rect_tool);
|
|
}
|
|
|
|
if (pika_draw_tool_is_active (PIKA_DRAW_TOOL (tool)))
|
|
pika_draw_tool_stop (PIKA_DRAW_TOOL (tool));
|
|
|
|
/* disconnect bindings manually so they are really gone *now*, we
|
|
* might be in the middle of a signal emission that keeps the
|
|
* widget and its bindings alive.
|
|
*/
|
|
g_list_free_full (priv->bindings, (GDestroyNotify) g_object_unref);
|
|
priv->bindings = NULL;
|
|
|
|
pika_draw_tool_set_widget (PIKA_DRAW_TOOL (tool), NULL);
|
|
g_clear_object (&priv->widget);
|
|
|
|
tool->display = NULL;
|
|
|
|
pika_rectangle_select_tool_update_option_defaults (rect_tool, TRUE);
|
|
}
|
|
|
|
static PikaChannelOps
|
|
pika_rectangle_select_tool_get_operation (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
PikaSelectionOptions *options;
|
|
|
|
options = PIKA_SELECTION_TOOL_GET_OPTIONS (rect_tool);
|
|
|
|
if (priv->use_saved_op)
|
|
return priv->operation;
|
|
else
|
|
return options->operation;
|
|
}
|
|
|
|
/**
|
|
* pika_rectangle_select_tool_update_option_defaults:
|
|
* @crop_tool:
|
|
* @ignore_pending: %TRUE to ignore any pending crop rectangle.
|
|
*
|
|
* Sets the default Fixed: Aspect ratio and Fixed: Size option
|
|
* properties.
|
|
*/
|
|
static void
|
|
pika_rectangle_select_tool_update_option_defaults (PikaRectangleSelectTool *rect_tool,
|
|
gboolean ignore_pending)
|
|
{
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaRectangleOptions *rect_options;
|
|
|
|
rect_options = PIKA_RECTANGLE_OPTIONS (pika_tool_get_options (tool));
|
|
|
|
if (priv->widget && ! ignore_pending)
|
|
{
|
|
/* There is a pending rectangle and we should not ignore it, so
|
|
* set default Fixed: Size to the same as the current pending
|
|
* rectangle width/height.
|
|
*/
|
|
|
|
pika_tool_rectangle_pending_size_set (PIKA_TOOL_RECTANGLE (priv->widget),
|
|
G_OBJECT (rect_options),
|
|
"default-aspect-numerator",
|
|
"default-aspect-denominator");
|
|
|
|
g_object_set (G_OBJECT (rect_options),
|
|
"use-string-current", TRUE,
|
|
NULL);
|
|
}
|
|
else
|
|
{
|
|
g_object_set (G_OBJECT (rect_options),
|
|
"default-aspect-numerator", 1.0,
|
|
"default-aspect-denominator", 1.0,
|
|
NULL);
|
|
|
|
g_object_set (G_OBJECT (rect_options),
|
|
"use-string-current", FALSE,
|
|
NULL);
|
|
}
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_update (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaTool *tool = PIKA_TOOL (rect_tool);
|
|
PikaRectangleSelectToolPrivate *priv = rect_tool->private;
|
|
|
|
/* prevent change in selection from halting the tool */
|
|
pika_tool_control_push_preserve (tool->control, TRUE);
|
|
|
|
if (tool->display && ! pika_tool_control_is_active (tool->control))
|
|
{
|
|
gdouble x1, y1, x2, y2;
|
|
|
|
pika_tool_rectangle_get_public_rect (PIKA_TOOL_RECTANGLE (priv->widget),
|
|
&x1, &y1, &x2, &y2);
|
|
|
|
pika_rectangle_select_tool_select (rect_tool,
|
|
x1, y1, x2 - x1, y2 - y1);
|
|
|
|
if (! priv->use_saved_op)
|
|
{
|
|
PikaSelectionOptions *options;
|
|
|
|
options = PIKA_SELECTION_TOOL_GET_OPTIONS (tool);
|
|
|
|
/* remember the operation now in case we modify the rectangle */
|
|
priv->operation = options->operation;
|
|
priv->use_saved_op = TRUE;
|
|
}
|
|
}
|
|
|
|
pika_tool_control_pop_preserve (tool->control);
|
|
|
|
pika_rectangle_select_tool_update_option_defaults (rect_tool, FALSE);
|
|
}
|
|
|
|
static void
|
|
pika_rectangle_select_tool_auto_shrink (PikaRectangleSelectTool *rect_tool)
|
|
{
|
|
PikaRectangleSelectToolPrivate *private = rect_tool->private;
|
|
gboolean shrink_merged;
|
|
|
|
g_object_get (pika_tool_get_options (PIKA_TOOL (rect_tool)),
|
|
"shrink-merged", &shrink_merged,
|
|
NULL);
|
|
|
|
pika_tool_rectangle_auto_shrink (PIKA_TOOL_RECTANGLE (private->widget),
|
|
shrink_merged);
|
|
}
|