diff options
Diffstat (limited to '')
-rw-r--r-- | app/tools/gimpguidetool.c | 546 |
1 files changed, 546 insertions, 0 deletions
diff --git a/app/tools/gimpguidetool.c b/app/tools/gimpguidetool.c new file mode 100644 index 0000000..55d5732 --- /dev/null +++ b/app/tools/gimpguidetool.c @@ -0,0 +1,546 @@ +/* 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 <gegl.h> +#include <gtk/gtk.h> + +#include "libgimpmath/gimpmath.h" + +#include "tools-types.h" + +#include "core/gimp.h" +#include "core/gimpguide.h" +#include "core/gimpimage.h" +#include "core/gimpimage-guides.h" +#include "core/gimpimage-undo.h" + +#include "display/gimpdisplay.h" +#include "display/gimpdisplayshell.h" +#include "display/gimpdisplayshell-selection.h" +#include "display/gimpdisplayshell-transform.h" + +#include "gimpguidetool.h" +#include "gimptoolcontrol.h" +#include "tool_manager.h" + +#include "gimp-intl.h" + + +#define SWAP_ORIENT(orient) ((orient) == GIMP_ORIENTATION_HORIZONTAL ? \ + GIMP_ORIENTATION_VERTICAL : \ + GIMP_ORIENTATION_HORIZONTAL) + + +/* local function prototypes */ + +static void gimp_guide_tool_finalize (GObject *object); + +static void gimp_guide_tool_button_release (GimpTool *tool, + const GimpCoords *coords, + guint32 time, + GdkModifierType state, + GimpButtonReleaseType release_type, + GimpDisplay *display); +static void gimp_guide_tool_motion (GimpTool *tool, + const GimpCoords *coords, + guint32 time, + GdkModifierType state, + GimpDisplay *display); + +static void gimp_guide_tool_draw (GimpDrawTool *draw_tool); + +static void gimp_guide_tool_start (GimpTool *parent_tool, + GimpDisplay *display, + GList *guides, + GimpOrientationType orientation); + +static void gimp_guide_tool_push_status (GimpGuideTool *guide_tool, + GimpDisplay *display, + gboolean remove_guides); + + +G_DEFINE_TYPE (GimpGuideTool, gimp_guide_tool, GIMP_TYPE_DRAW_TOOL) + +#define parent_class gimp_guide_tool_parent_class + + +static void +gimp_guide_tool_class_init (GimpGuideToolClass *klass) +{ + GObjectClass *object_class = G_OBJECT_CLASS (klass); + GimpToolClass *tool_class = GIMP_TOOL_CLASS (klass); + GimpDrawToolClass *draw_tool_class = GIMP_DRAW_TOOL_CLASS (klass); + + object_class->finalize = gimp_guide_tool_finalize; + + tool_class->button_release = gimp_guide_tool_button_release; + tool_class->motion = gimp_guide_tool_motion; + + draw_tool_class->draw = gimp_guide_tool_draw; +} + +static void +gimp_guide_tool_init (GimpGuideTool *guide_tool) +{ + GimpTool *tool = GIMP_TOOL (guide_tool); + + gimp_tool_control_set_snap_to (tool->control, FALSE); + gimp_tool_control_set_handle_empty_image (tool->control, TRUE); + gimp_tool_control_set_tool_cursor (tool->control, + GIMP_TOOL_CURSOR_MOVE); + gimp_tool_control_set_scroll_lock (tool->control, TRUE); + gimp_tool_control_set_precision (tool->control, + GIMP_CURSOR_PRECISION_PIXEL_BORDER); + + guide_tool->guides = NULL; + guide_tool->n_guides = 0; +} + +static void +gimp_guide_tool_finalize (GObject *object) +{ + GimpGuideTool *guide_tool = GIMP_GUIDE_TOOL (object); + gint i; + + for (i = 0; i < guide_tool->n_guides; i++) + g_clear_object (&guide_tool->guides[i].guide); + + g_free (guide_tool->guides); + + G_OBJECT_CLASS (parent_class)->finalize (object); +} + +static void +gimp_guide_tool_button_release (GimpTool *tool, + const GimpCoords *coords, + guint32 time, + GdkModifierType state, + GimpButtonReleaseType release_type, + GimpDisplay *display) +{ + GimpGuideTool *guide_tool = GIMP_GUIDE_TOOL (tool); + GimpDisplayShell *shell = gimp_display_get_shell (display); + GimpImage *image = gimp_display_get_image (display); + gint i; + + gimp_tool_pop_status (tool, display); + + gimp_tool_control_halt (tool->control); + + gimp_draw_tool_stop (GIMP_DRAW_TOOL (tool)); + + if (release_type == GIMP_BUTTON_RELEASE_CANCEL) + { + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + + /* custom guides are moved live */ + if (guide->custom) + { + gimp_image_move_guide (image, guide->guide, guide->old_position, + TRUE); + } + } + } + else + { + gint n_non_custom_guides = 0; + gboolean remove_guides = FALSE; + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + gint max_position; + + if (guide->orientation == GIMP_ORIENTATION_HORIZONTAL) + max_position = gimp_image_get_height (image); + else + max_position = gimp_image_get_width (image); + + n_non_custom_guides += ! guide->custom; + + if (guide->position == GIMP_GUIDE_POSITION_UNDEFINED || + guide->position < 0 || + guide->position > max_position) + { + remove_guides = TRUE; + } + } + + if (n_non_custom_guides > 1) + { + gimp_image_undo_group_start (image, GIMP_UNDO_GROUP_GUIDE, + remove_guides ? + C_("undo-type", "Remove Guides") : + C_("undo-type", "Move Guides")); + } + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + + if (remove_guides) + { + /* removing a guide can cause other guides to be removed as well + * (in particular, in case of symmetry guides). these guides + * will be kept alive, since we hold a reference on them, but we + * need to make sure that they're still part of the image. + */ + if (g_list_find (gimp_image_get_guides (image), guide->guide)) + gimp_image_remove_guide (image, guide->guide, TRUE); + } + else + { + if (guide->guide) + { + /* custom guides are moved live */ + if (! guide->custom) + { + gimp_image_move_guide (image, guide->guide, + guide->position, TRUE); + } + } + else + { + switch (guide->orientation) + { + case GIMP_ORIENTATION_HORIZONTAL: + gimp_image_add_hguide (image, + guide->position, + TRUE); + break; + + case GIMP_ORIENTATION_VERTICAL: + gimp_image_add_vguide (image, + guide->position, + TRUE); + break; + + default: + gimp_assert_not_reached (); + } + } + } + } + + if (n_non_custom_guides > 1) + gimp_image_undo_group_end (image); + + gimp_image_flush (image); + } + + gimp_display_shell_selection_resume (shell); + + tool_manager_pop_tool (display->gimp); + g_object_unref (guide_tool); + + { + GimpTool *active_tool = tool_manager_get_active (display->gimp); + + if (GIMP_IS_DRAW_TOOL (active_tool)) + gimp_draw_tool_pause (GIMP_DRAW_TOOL (active_tool)); + + tool_manager_oper_update_active (display->gimp, coords, state, + TRUE, display); + tool_manager_cursor_update_active (display->gimp, coords, state, + display); + + if (GIMP_IS_DRAW_TOOL (active_tool)) + gimp_draw_tool_resume (GIMP_DRAW_TOOL (active_tool)); + } +} + +static void +gimp_guide_tool_motion (GimpTool *tool, + const GimpCoords *coords, + guint32 time, + GdkModifierType state, + GimpDisplay *display) + +{ + GimpGuideTool *guide_tool = GIMP_GUIDE_TOOL (tool); + GimpDisplayShell *shell = gimp_display_get_shell (display); + GimpImage *image = gimp_display_get_image (display); + gboolean remove_guides = FALSE; + gint tx, ty; + gint i; + + gimp_draw_tool_pause (GIMP_DRAW_TOOL (tool)); + + gimp_display_shell_transform_xy (shell, + coords->x, coords->y, + &tx, &ty); + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + gint max_position; + gint position; + + if (guide->orientation == GIMP_ORIENTATION_HORIZONTAL) + max_position = gimp_image_get_height (image); + else + max_position = gimp_image_get_width (image); + + if (guide->orientation == GIMP_ORIENTATION_HORIZONTAL) + guide->position = RINT (coords->y); + else + guide->position = RINT (coords->x); + + position = CLAMP (guide->position, 0, max_position); + + if (tx < 0 || tx >= shell->disp_width || + ty < 0 || ty >= shell->disp_height) + { + guide->position = GIMP_GUIDE_POSITION_UNDEFINED; + + remove_guides = TRUE; + } + else + { + if (guide->position < 0 || guide->position > max_position) + remove_guides = TRUE; + } + + /* custom guides are moved live */ + if (guide->custom) + gimp_image_move_guide (image, guide->guide, position, TRUE); + } + + gimp_draw_tool_resume (GIMP_DRAW_TOOL (tool)); + + gimp_tool_pop_status (tool, display); + + gimp_guide_tool_push_status (guide_tool, display, remove_guides); +} + +static void +gimp_guide_tool_draw (GimpDrawTool *draw_tool) +{ + GimpGuideTool *guide_tool = GIMP_GUIDE_TOOL (draw_tool); + gint i; + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + + if (guide->position != GIMP_GUIDE_POSITION_UNDEFINED) + { + /* custom guides are moved live */ + if (! guide->custom) + { + gimp_draw_tool_add_guide (draw_tool, + guide->orientation, + guide->position, + GIMP_GUIDE_STYLE_NONE); + } + } + } +} + +static void +gimp_guide_tool_start (GimpTool *parent_tool, + GimpDisplay *display, + GList *guides, + GimpOrientationType orientation) +{ + GimpGuideTool *guide_tool; + GimpTool *tool; + + guide_tool = g_object_new (GIMP_TYPE_GUIDE_TOOL, + "tool-info", parent_tool->tool_info, + NULL); + + tool = GIMP_TOOL (guide_tool); + + gimp_display_shell_selection_pause (gimp_display_get_shell (display)); + + if (guides) + { + gint i; + + guide_tool->n_guides = g_list_length (guides); + guide_tool->guides = g_new (GimpGuideToolGuide, guide_tool->n_guides); + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuide *guide = guides->data; + + guide_tool->guides[i].guide = g_object_ref (guide); + guide_tool->guides[i].old_position = gimp_guide_get_position (guide); + guide_tool->guides[i].position = gimp_guide_get_position (guide); + guide_tool->guides[i].orientation = gimp_guide_get_orientation (guide); + guide_tool->guides[i].custom = gimp_guide_is_custom (guide); + + guides = g_list_next (guides); + } + } + else + { + guide_tool->n_guides = 1; + guide_tool->guides = g_new (GimpGuideToolGuide, 1); + + guide_tool->guides[0].guide = NULL; + guide_tool->guides[0].old_position = 0; + guide_tool->guides[0].position = GIMP_GUIDE_POSITION_UNDEFINED; + guide_tool->guides[0].orientation = orientation; + guide_tool->guides[0].custom = FALSE; + } + + gimp_tool_set_cursor (tool, display, + GIMP_CURSOR_MOUSE, + GIMP_TOOL_CURSOR_HAND, + GIMP_CURSOR_MODIFIER_MOVE); + + tool_manager_push_tool (display->gimp, tool); + + tool->display = display; + gimp_tool_control_activate (tool->control); + + gimp_draw_tool_start (GIMP_DRAW_TOOL (guide_tool), display); + + gimp_guide_tool_push_status (guide_tool, display, FALSE); +} + +static void +gimp_guide_tool_push_status (GimpGuideTool *guide_tool, + GimpDisplay *display, + gboolean remove_guides) +{ + GimpTool *tool = GIMP_TOOL (guide_tool); + + if (remove_guides) + { + gimp_tool_push_status (tool, display, + guide_tool->n_guides > 1 ? _("Remove Guides") : + guide_tool->guides[0].guide ? _("Remove Guide") : + _("Cancel Guide")); + } + else + { + GimpGuideToolGuide *guides[2]; + gint n_guides = 0; + gint i; + + for (i = 0; i < guide_tool->n_guides; i++) + { + GimpGuideToolGuide *guide = &guide_tool->guides[i]; + + if (guide_tool->guides[i].guide) + { + if (n_guides == 0 || guide->orientation != guides[0]->orientation) + { + guides[n_guides++] = guide; + + if (n_guides == 2) + break; + } + } + } + + if (n_guides == 2 && + guides[0]->orientation == GIMP_ORIENTATION_HORIZONTAL) + { + GimpGuideToolGuide *temp; + + temp = guides[0]; + guides[0] = guides[1]; + guides[1] = temp; + } + + if (n_guides == 1) + { + gimp_tool_push_status_length (tool, display, + _("Move Guide: "), + SWAP_ORIENT (guides[0]->orientation), + guides[0]->position - + guides[0]->old_position, + NULL); + } + else if (n_guides == 2) + { + gimp_tool_push_status_coords (tool, display, + GIMP_CURSOR_PRECISION_PIXEL_BORDER, + _("Move Guides: "), + guides[0]->position - + guides[0]->old_position, + ", ", + guides[1]->position - + guides[1]->old_position, + NULL); + } + else + { + gimp_tool_push_status_length (tool, display, + _("Add Guide: "), + SWAP_ORIENT (guide_tool->guides[0].orientation), + guide_tool->guides[0].position, + NULL); + } + } +} + + +/* public functions */ + +void +gimp_guide_tool_start_new (GimpTool *parent_tool, + GimpDisplay *display, + GimpOrientationType orientation) +{ + g_return_if_fail (GIMP_IS_TOOL (parent_tool)); + g_return_if_fail (GIMP_IS_DISPLAY (display)); + g_return_if_fail (orientation != GIMP_ORIENTATION_UNKNOWN); + + gimp_guide_tool_start (parent_tool, display, + NULL, orientation); +} + +void +gimp_guide_tool_start_edit (GimpTool *parent_tool, + GimpDisplay *display, + GimpGuide *guide) +{ + GList *guides = NULL; + + g_return_if_fail (GIMP_IS_TOOL (parent_tool)); + g_return_if_fail (GIMP_IS_DISPLAY (display)); + g_return_if_fail (GIMP_IS_GUIDE (guide)); + + guides = g_list_append (guides, guide); + + gimp_guide_tool_start (parent_tool, display, + guides, GIMP_ORIENTATION_UNKNOWN); + + g_list_free (guides); +} + +void +gimp_guide_tool_start_edit_many (GimpTool *parent_tool, + GimpDisplay *display, + GList *guides) +{ + g_return_if_fail (GIMP_IS_TOOL (parent_tool)); + g_return_if_fail (GIMP_IS_DISPLAY (display)); + g_return_if_fail (guides != NULL); + + gimp_guide_tool_start (parent_tool, display, + guides, GIMP_ORIENTATION_UNKNOWN); +} |