gimp/app/tools/gimpbezierselecttool.c

4005 lines
99 KiB
C

/* The GIMP -- an 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 2 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, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "config.h"
#include <string.h>
#include <gtk/gtk.h>
#include <gdk/gdkkeysyms.h>
#include "libgimpwidgets/gimpwidgets.h"
#include "libgimpmath/gimpmath.h"
#include "apptypes.h"
#include "core/gimpchannel.h"
#include "core/gimpmarshal.h"
#include "core/gimpcontext.h"
#include "core/gimpimage.h"
#include "core/gimpimage-mask.h"
#include "cursorutil.h"
#include "drawable.h"
#include "errors.h"
#include "gdisplay.h"
#include "undo.h"
#include "pdb/procedural_db.h"
#include "gui/paths-dialog.h"
#include "floating_sel.h"
#include "pixel_region.h"
#include "gimpeditselectiontool.h"
#include "gimpbezierselecttool.h"
#include "gimptoolinfo.h"
#include "selection_options.h"
#include "tool_options.h"
#include "tool_manager.h"
#include "libgimp/gimpintl.h"
#include "pixmaps2.h"
#define STATUSBAR_SIZE 128
/* Old local stuff that (at worst) shouldn't break anything */
#define BEZIER_DRAW_CURVE 1
#define BEZIER_DRAW_CURRENT 2
#define BEZIER_DRAW_HANDLES 4
#define BEZIER_DRAW_ALL (BEZIER_DRAW_CURVE | BEZIER_DRAW_HANDLES)
#define BEZIER_WIDTH 8
#define BEZIER_HALFWIDTH 4
#define SUPERSAMPLE 3
#define SUPERSAMPLE2 9
/* the bezier select structures */
typedef gdouble BezierMatrix[4][4];
typedef struct
{
CountCurves curve_count; /* Must be the first element */
gdouble *stroke_points;
gint num_stroke_points; /* num of valid points */
gint len_stroke_points; /* allocated length */
gpointer next_curve; /* Next curve in list -- we draw all curves
* separately.
*/
} BezierRenderPnts;
typedef struct
{
CountCurves curve_count; /* Must be the first element */
gboolean firstpnt;
gdouble curdist;
gdouble dist;
gdouble *gradient;
gint *x;
gint *y;
gdouble lastx;
gdouble lasty;
gboolean found;
} BezierDistance;
typedef struct
{
CountCurves curve_count; /* Must be the first element */
gint x;
gint y;
gint halfwidth;
gint found;
} BezierCheckPnts;
/* the bezier selection tool options */
static SelectionOptions *bezier_options = NULL;
/* local variables */
static BezierMatrix basis =
{
{ -1, 3, -3, 1 },
{ 3, -6, 3, 0 },
{ -3, 3, 0, 0 },
{ 1, 0, 0, 0 },
};
/*
static BezierMatrix basis =
{
{ -1/6.0, 3/6.0, -3/6.0, 1/6.0 },
{ 3/6.0, -6/6.0, 3/6.0, 0 },
{ -3/6.0, 0, 3/6.0, 0 },
{ 1/6.0, 4/6.0, 1, 0 },
};
*/
/* Global Static Variable to maintain informations about the "context" */
static GimpBezierSelectTool *curSel = NULL;
static GimpTool *curTool = NULL;
static GDisplay *curGdisp = NULL;
static GimpDrawTool *curCore = NULL;
static gint ModeEdit = EXTEND_NEW;
enum
{
BEZIER_SELECT,
LAST_SIGNAL
};
static GimpSelectionToolClass *parent_class = NULL;
static guint bezier_select_signals[LAST_SIGNAL] = { 0 };
static void gimp_bezier_select_tool_class_init (GimpBezierSelectToolClass *klass);
static void gimp_bezier_select_tool_init (GimpBezierSelectTool *bezier_select);
static void gimp_bezier_select_tool_destroy (GtkObject *object);
static void gimp_bezier_select_tool_button_press (GimpTool *tool,
GdkEventButton *bevent,
GDisplay *gdisp);
static void gimp_bezier_select_tool_button_release (GimpTool *tool,
GdkEventButton *bevent,
GDisplay *gdisp);
static void gimp_bezier_select_tool_motion (GimpTool *tool,
GdkEventMotion *mevent,
GDisplay *gdisp);
static void gimp_bezier_select_tool_control (GimpTool *tool,
ToolAction action,
GDisplay *gdisp);
static void gimp_bezier_select_tool_cursor_update (GimpTool *tool,
GdkEventMotion *mevent,
GDisplay *gdisp);
static void bezier_select_draw (GimpDrawTool *draw_tool);
static void bezier_offset_point (GimpBezierSelectPoint *pt,
gint x,
gint y);
static gint bezier_check_point (GimpBezierSelectPoint *pt,
gint x,
gint y,
gint halfwidth);
static void bezier_draw_handles (GimpBezierSelectTool *bezier_sel,
gboolean do_all);
static void bezier_draw_current (GimpBezierSelectTool *bezier_sel);
static void bezier_draw_point (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt,
gboolean fill);
static void bezier_draw_line (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt1,
GimpBezierSelectPoint *pt2);
static void bezier_draw_segment (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *points,
gint subdivisions,
gint space,
GimpBezierSelectPointsFunc points_func,
gpointer data);
static void bezier_draw_segment_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data);
static void bezier_compose (BezierMatrix a,
BezierMatrix b,
BezierMatrix ab);
static void bezier_convert (GimpBezierSelectTool *bezier_sel,
GDisplay *gdisp,
gint subdivisions,
gboolean antialias);
static void bezier_convert_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data);
static void bezier_convert_line (GSList **scanlines,
gint x1,
gint y1,
gint x2,
gint y2);
static GSList * bezier_insert_in_list (GSList *list,
gint x);
static gboolean test_add_point_on_segment (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt,
gint subdivisions,
gint space,
gint xpos,
gint ypos,
gint halfwidth);
static void bezier_to_sel_internal (GimpBezierSelectTool *bezier_sel,
GimpTool *tool,
GDisplay *gdisp,
gint op,
gboolean replace);
static void bezier_stack_points_aux (GdkPoint *points,
gint start,
gint end,
gdouble error,
BezierRenderPnts *rpnts);
static void bezier_stack_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data);
static gboolean stroke_interpolatable (gint offx,
gint offy,
gint l_offx,
gint l_offy,
gdouble error);
static gint count_points_on_curve (GimpBezierSelectPoint *points);
static gboolean bezier_edit_point_on_curve (gint x,
gint y,
gint halfwidth,
GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
GimpTool *tool,
GdkEventButton *bevent);
static gboolean bezier_add_point_on_segment (gint x,
gint y,
gint halfwidth,
GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
GimpTool *tool);
static GimpBezierSelectPoint * find_start_open_curve (GimpBezierSelectTool *bsel);
static void bezier_start_new_segment (GimpBezierSelectTool *bezier_sel,
gint x,
gint y);
/* Public functions */
void
gimp_bezier_select_tool_register (void)
{
tool_manager_register_tool (GIMP_TYPE_BEZIER_SELECT_TOOL,
FALSE,
"gimp:bezier_select_tool",
_("Bezier Select"),
_("Select regions using Bezier curves"),
_("/Tools/Selection Tools/Bezier Select"), "B",
NULL, "tools/bezier_select.html",
(const gchar **) bezier_bits);
}
GtkType
gimp_bezier_select_tool_get_type (void)
{
static GtkType bezier_select_type = 0;
if (! bezier_select_type)
{
GtkTypeInfo bezier_select_info =
{
"GimpBezierSelectTool",
sizeof (GimpBezierSelectTool),
sizeof (GimpBezierSelectToolClass),
(GtkClassInitFunc) gimp_bezier_select_tool_class_init,
(GtkObjectInitFunc) gimp_bezier_select_tool_init,
/* reserved_1 */ NULL,
/* reserved_2 */ NULL
};
bezier_select_type = gtk_type_unique (GIMP_TYPE_SELECTION_TOOL,
&bezier_select_info);
}
return bezier_select_type;
}
/* Private functions */
static void
gimp_bezier_select_tool_class_init (GimpBezierSelectToolClass *klass)
{
GtkObjectClass *object_class;
GimpToolClass *tool_class;
GimpDrawToolClass *draw_tool_class;
object_class = (GtkObjectClass *) klass;
tool_class = (GimpToolClass *) klass;
draw_tool_class = (GimpDrawToolClass *) klass;
parent_class = gtk_type_class (GIMP_TYPE_SELECTION_TOOL);
bezier_select_signals[BEZIER_SELECT] =
gtk_signal_new ("bezier_select",
GTK_RUN_FIRST,
object_class->type,
GTK_SIGNAL_OFFSET (GimpBezierSelectToolClass,
bezier_select),
gimp_marshal_NONE__INT_INT_INT_INT,
GTK_TYPE_NONE, 4,
GTK_TYPE_INT,
GTK_TYPE_INT,
GTK_TYPE_INT,
GTK_TYPE_INT);
gtk_object_class_add_signals (object_class, bezier_select_signals, LAST_SIGNAL);
object_class->destroy = gimp_bezier_select_tool_destroy;
tool_class->button_press = gimp_bezier_select_tool_button_press;
tool_class->button_release = gimp_bezier_select_tool_button_release;
tool_class->motion = gimp_bezier_select_tool_motion;
tool_class->control = gimp_bezier_select_tool_control;
tool_class->cursor_update = gimp_bezier_select_tool_cursor_update;
draw_tool_class->draw = bezier_select_draw;
/* klass->bezier_select = gimp_bezier_select_tool_real_bezier_select; */
}
static void
gimp_bezier_select_tool_init (GimpBezierSelectTool *bezier_select)
{
GimpTool *tool;
GimpDrawTool *draw_tool;
GimpSelectionTool *select_tool;
tool = GIMP_TOOL (bezier_select);
draw_tool = GIMP_DRAW_TOOL (bezier_select);
select_tool = GIMP_SELECTION_TOOL (bezier_select);
if (! bezier_options)
{
bezier_options = selection_options_new (GIMP_TYPE_BEZIER_SELECT_TOOL,
selection_options_reset);
tool_manager_register_tool_options (GIMP_TYPE_BEZIER_SELECT_TOOL,
(ToolOptions *) bezier_options);
}
bezier_select->num_points = 0;
bezier_select->mask = NULL;
tool->tool_cursor = GIMP_BEZIER_SELECT_TOOL_CURSOR;
tool->preserve = FALSE; /* Don't preserve on drawable change */
curCore = draw_tool;
curSel = bezier_select;
curTool = tool;
bezier_select_reset (bezier_select);
paths_new_bezier_select_tool ();
}
static void
gimp_bezier_select_tool_destroy (GtkObject *object)
{
if (GTK_OBJECT_CLASS (parent_class)->destroy)
GTK_OBJECT_CLASS (parent_class)->destroy (object);
}
static void
gimp_bezier_select_tool_button_press (GimpTool *tool,
GdkEventButton *bevent,
GDisplay *gdisp)
{
GimpBezierSelectTool *bezier_sel;
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *curve_start;
gboolean grab_pointer;
gint op;
gint x, y;
gint halfwidth, dummy;
tool->drawable = gimp_image_active_drawable (gdisp->gimage);
bezier_sel = GIMP_BEZIER_SELECT_TOOL(tool);
grab_pointer = FALSE;
/* If the tool was being used in another image...reset it */
if (tool->state == ACTIVE && gdisp != tool->gdisp)
{
gimp_draw_tool_stop ((GimpDrawTool *)bezier_sel);
bezier_select_reset (bezier_sel);
}
gdisplay_untransform_coords (gdisp, bevent->x, bevent->y, &x, &y, TRUE, 0);
/* get halfwidth in image coord */
gdisplay_untransform_coords (gdisp, bevent->x + BEZIER_HALFWIDTH, 0,
&halfwidth, &dummy, TRUE, 0);
halfwidth -= x;
curTool = active_tool;
curSel = bezier_sel;
curGdisp = (GDisplay *) gdisp;
curCore = (GimpDrawTool *)bezier_sel;
switch (bezier_sel->state)
{
case BEZIER_START:
if (ModeEdit != EXTEND_NEW)
break;
grab_pointer = TRUE;
tool->state = ACTIVE;
tool->gdisp = gdisp;
bezier_sel->state = BEZIER_ADD;
/*bezier_sel->draw_mode = BEZIER_DRAW_CURVE; | BEZIER_DRAW_HANDLES;*/
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT;
bezier_add_point (bezier_sel, BEZIER_ANCHOR, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
gimp_draw_tool_start ((GimpDrawTool *)bezier_sel, gdisp->canvas->window);
break;
case BEZIER_ADD:
grab_pointer = TRUE;
if (ModeEdit == EXTEND_EDIT)
{
/* erase the handles */
if(bezier_sel->closed)
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
else
bezier_sel->draw_mode = BEZIER_DRAW_CURVE;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
if (grab_pointer)
{
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
}
else
{
paths_dialog_set_default_op ();
/* recursive call */
gimp_bezier_select_tool_button_press (tool, bevent, gdisp);
}
return;
}
if (ModeEdit == EXTEND_REMOVE)
{
/* if(bezier_sel->num_points < 6) */
/* return; */
/* erase the handles */
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
/*kkk*/
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
/*kkk*/
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
if (bezier_sel->num_points == 0)
{
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
paths_dialog_set_default_op ();
}
if (!grab_pointer)
{
paths_dialog_set_default_op ();
/* recursive call */
gimp_bezier_select_tool_button_press (tool, bevent, gdisp);
}
return;
}
if (ModeEdit == EXTEND_ADD)
{
if (bezier_sel->num_points < 5)
return;
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
grab_pointer = bezier_add_point_on_segment (x, y, halfwidth,
gdisp, bezier_sel, tool);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
if (grab_pointer)
{
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
}
else
{
paths_dialog_set_default_op ();
/* recursive call */
gimp_bezier_select_tool_button_press (tool, bevent, gdisp);
}
return;
}
if (bezier_sel->cur_anchor)
{
if (bezier_check_point (bezier_sel->cur_anchor, x, y, halfwidth))
{
break;
}
if (bezier_sel->cur_anchor->next &&
bezier_check_point (bezier_sel->cur_anchor->next, x, y, halfwidth))
{
bezier_sel->cur_control = bezier_sel->cur_anchor->next;
break;
}
if (bezier_sel->cur_anchor->prev &&
bezier_check_point (bezier_sel->cur_anchor->prev, x, y, halfwidth))
{
bezier_sel->cur_control = bezier_sel->cur_anchor->prev;
break;
}
}
curve_start = find_start_open_curve(bezier_sel);
if (curve_start && bezier_check_point (curve_start, x, y, halfwidth))
{
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_sel->last_point->next = curve_start;
curve_start->prev = bezier_sel->last_point;
bezier_sel->cur_anchor = curve_start;
bezier_sel->cur_control = curve_start->next;
bezier_sel->closed = 1;
bezier_sel->state = BEZIER_EDIT;
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
}
else
{
if(bezier_sel->cur_anchor)
bezier_sel->cur_anchor->pointflags = 1;
bezier_sel->draw_mode = BEZIER_DRAW_HANDLES;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_ANCHOR, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT | BEZIER_DRAW_HANDLES;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
}
break;
case BEZIER_EDIT:
if (!bezier_sel->closed)
gimp_fatal_error ("gimp_bezier_select_tool_button_press(): Tried to edit "
"on open bezier curve in edit selection");
/* erase the handles */
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (ModeEdit == EXTEND_ADD)
{
if (bezier_sel->closed)
{
grab_pointer = bezier_add_point_on_segment (x, y, halfwidth,
gdisp, bezier_sel,
tool);
}
}
else
{
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
if (grab_pointer && bezier_sel->num_points == 0)
{
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
paths_dialog_set_default_op ();
}
}
if (!grab_pointer && gimp_channel_value (bezier_sel->mask, x, y))
{
gboolean replace = FALSE;
if ((bevent->state & GDK_SHIFT_MASK) &&
!(bevent->state & GDK_CONTROL_MASK))
op = CHANNEL_OP_ADD;
else if ((bevent->state & GDK_CONTROL_MASK) &&
!(bevent->state & GDK_SHIFT_MASK))
op = CHANNEL_OP_SUB;
else if ((bevent->state & GDK_CONTROL_MASK) &&
(bevent->state & GDK_SHIFT_MASK))
op = CHANNEL_OP_INTERSECT;
else
{
op = CHANNEL_OP_ADD;
replace = TRUE;
}
bezier_to_sel_internal (bezier_sel, tool, gdisp, op, replace);
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
}
else
{
/* draw the handles */
if (!grab_pointer)
{
paths_dialog_set_default_op ();
bezier_start_new_segment (bezier_sel, x, y);
}
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
}
break;
}
if (grab_pointer)
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
/* Don't bother doing this if we don't have any points */
if (bezier_sel->num_points > 0)
paths_first_button_press(bezier_sel, gdisp);
}
static void
gimp_bezier_select_tool_button_release (GimpTool *tool,
GdkEventButton *bevent,
GDisplay *gdisp)
{
GimpBezierSelectTool *bezier_sel;
gdisp = tool->gdisp;
bezier_sel = GIMP_BEZIER_SELECT_TOOL(tool);
bezier_sel->state &= ~(BEZIER_DRAG);
gdk_pointer_ungrab (bevent->time);
gdk_flush ();
if (bezier_sel->closed)
bezier_convert (bezier_sel, tool->gdisp, SUBDIVIDE, FALSE);
/* Here ?*/
paths_newpoint_current (bezier_sel,gdisp);
}
static void
gimp_bezier_select_tool_motion (GimpTool *tool,
GdkEventMotion *mevent,
GDisplay *gdisp)
{
static gint lastx, lasty;
GimpBezierSelectTool *bezier_sel;
GimpBezierSelectPoint *anchor;
GimpBezierSelectPoint *opposite_control;
gint offsetx;
gint offsety;
gint x, y;
if (tool->state != ACTIVE)
return;
bezier_sel = GIMP_BEZIER_SELECT_TOOL(tool);
if (!bezier_sel->cur_anchor || !bezier_sel->cur_control)
return;
if (mevent->state & GDK_MOD1_MASK)
{
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
}
else
{
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT | BEZIER_DRAW_HANDLES;
}
gimp_draw_tool_pause ((GimpDrawTool *)bezier_sel);
gdisplay_untransform_coords (gdisp, mevent->x, mevent->y, &x, &y, TRUE, 0);
/* If this is the first point then change the state and "remember" the point.
*/
if (!(bezier_sel->state & BEZIER_DRAG))
{
bezier_sel->state |= BEZIER_DRAG;
lastx = x;
lasty = y;
}
/* The Alt key is down... Move all the points of the bezier curve */
if (mevent->state & GDK_MOD1_MASK)
{
GimpBezierSelectPoint *tmp = bezier_sel->points;
gint num_points = bezier_sel->num_points;
offsetx = x - lastx;
offsety = y - lasty;
if (mevent->state & GDK_SHIFT_MASK)
{
/* Only move this curve */
GimpBezierSelectPoint *start_pt = bezier_sel->cur_anchor;
/* g_print ("moving only one curve\n"); */
tmp = start_pt;
do
{
bezier_offset_point (tmp, offsetx, offsety);
tmp = tmp->next;
}
while (tmp != start_pt && tmp);
/* Check if need to go backwards because curve is open */
if (!tmp && bezier_sel->cur_anchor->prev)
{
GimpBezierSelectPoint *start_pt = bezier_sel->cur_anchor->prev;
tmp = start_pt;
do
{
bezier_offset_point (tmp, offsetx, offsety);
tmp = tmp->prev;
}
while (tmp != start_pt && tmp);
}
}
else
{
while (tmp && num_points)
{
bezier_offset_point (tmp, offsetx, offsety);
if(tmp->next_curve)
tmp = tmp->next_curve;
else
tmp = tmp->next;
num_points--;
}
}
}
else if (mevent->state & GDK_CONTROL_MASK)
{
/* the control key is down ... move the current anchor point */
/* we must also move the neighboring control points appropriately */
offsetx = x - lastx;
offsety = y - lasty;
bezier_offset_point (bezier_sel->cur_anchor, offsetx, offsety);
bezier_offset_point (bezier_sel->cur_anchor->next, offsetx, offsety);
bezier_offset_point (bezier_sel->cur_anchor->prev, offsetx, offsety);
}
else
{
/* the control key is not down ... we move the current control point */
offsetx = x - bezier_sel->cur_control->x;
offsety = y - bezier_sel->cur_control->y;
bezier_offset_point (bezier_sel->cur_control, offsetx, offsety);
/* if the shift key is not down then we align the opposite control */
/* point...ie the opposite control point acts like a mirror of the */
/* current control point */
if (!(mevent->state & GDK_SHIFT_MASK))
{
anchor = NULL;
opposite_control = NULL;
if (bezier_sel->cur_control->next)
{
if (bezier_sel->cur_control->next->type == BEZIER_ANCHOR)
{
anchor = bezier_sel->cur_control->next;
opposite_control = anchor->next;
}
}
if (bezier_sel->cur_control->prev)
{
if (bezier_sel->cur_control->prev->type == BEZIER_ANCHOR)
{
anchor = bezier_sel->cur_control->prev;
opposite_control = anchor->prev;
}
}
if (!anchor)
gimp_fatal_error ("bezier_select_motion(): Encountered orphaned "
"bezier control point");
if (opposite_control)
{
offsetx = bezier_sel->cur_control->x - anchor->x;
offsety = bezier_sel->cur_control->y - anchor->y;
opposite_control->x = anchor->x - offsetx;
opposite_control->y = anchor->y - offsety;
}
}
}
/* As we're moving all the control points of the curve,
we have to redraw all !!!
*/
if ( mevent->state & GDK_MOD1_MASK)
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
else
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT | BEZIER_DRAW_HANDLES;
gimp_draw_tool_resume ((GimpDrawTool *)bezier_sel);
lastx = x;
lasty = y;
}
gint
bezier_select_load (GDisplay *gdisp,
GimpBezierSelectPoint *pts,
gint num_pts,
gint closed)
{
GimpTool *tool;
GimpBezierSelectTool *bezier_sel;
/* select the bezier tool */
gimp_context_set_tool (gimp_context_get_user (),
tool_manager_get_info_by_type
( GIMP_TYPE_BEZIER_SELECT_TOOL));
tool = active_tool;
tool->state = ACTIVE;
tool->gdisp = gdisp;
bezier_sel = (GimpBezierSelectTool *) tool;
bezier_sel->points = pts;
bezier_sel->last_point = pts->prev;
bezier_sel->num_points = num_pts;
bezier_sel->closed = closed;
bezier_sel->state = BEZIER_EDIT;
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
bezier_convert (bezier_sel, tool->gdisp, SUBDIVIDE, FALSE);
gimp_draw_tool_start ((GimpDrawTool *) bezier_sel, gdisp->canvas->window);
return 1;
}
static GimpBezierSelectPoint *
valid_curve_segment (GimpBezierSelectPoint *points)
{
/* Valid curve segment is made up of four points */
if (points &&
points->next &&
points->next->next &&
points->next->next->next)
{
return points;
}
return NULL;
}
static GimpBezierSelectPoint *
next_anchor (GimpBezierSelectPoint *points,
GimpBezierSelectPoint **next_curve)
{
gint loop;
*next_curve = NULL;
if (!points)
return NULL;
for (loop = 0; loop < 3; loop++)
{
points = points->next;
if (!points)
return NULL;
if (points->next_curve)
{
*next_curve = points->next_curve;
}
}
return valid_curve_segment (points);
}
void
bezier_draw_curve (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPointsFunc func,
gint coord,
gpointer data)
{
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *next_curve;
CountCurves *cnt = (CountCurves *) data;
gint point_counts = 0;
/* printSel(bezier_sel); */
if (cnt)
cnt->count = 0;
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (bezier_sel->num_points >= 4)
{
do
{
point_counts = count_points_on_curve (points);
if (point_counts >= 4)
{
do
{
bezier_draw_segment (bezier_sel, points,
SUBDIVIDE, coord,
func,
data);
points = next_anchor (points,&next_curve);
/* g_print ("next_anchor = %p\n",points); */
}
while (points != start_pt && points);
if (cnt)
cnt->count++;
start_pt = next_curve;
points = next_curve;
}
else
{
break; /* must be last curve since only this one is allowed < 4
* points.
*/
}
}
while (next_curve);
}
}
void
bezier_select_reset (GimpBezierSelectTool *bezier_sel)
{
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *temp_pt;
GimpBezierSelectPoint *next_curve;
if (bezier_sel->num_points > 0)
{
points = bezier_sel->points;
start_pt = bezier_sel->points;
do
{
do
{
temp_pt = points;
next_curve = points->next_curve;
points = points->next;
if (points != start_pt && next_curve)
{
g_warning ("Curve points out of sync");
}
g_free (temp_pt);
}
while (points != start_pt && points);
points = next_curve;
start_pt = next_curve;
}
while (next_curve);
}
if (bezier_sel->mask)
gtk_object_unref (GTK_OBJECT (bezier_sel->mask));
bezier_sel->state = BEZIER_START; /* we are starting the curve */
bezier_sel->draw_mode = BEZIER_DRAW_ALL; /* draw everything by default */
bezier_sel->closed = 0; /* the curve is initally open */
bezier_sel->points = NULL; /* initially there are no points */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
bezier_sel->last_point = NULL;
bezier_sel->num_points = 0; /* intially there are no points */
bezier_sel->mask = NULL; /* empty mask */
bezier_sel->scanlines = NULL;
}
void
bezier_select_free (GimpBezierSelectTool *bezier_sel)
{
bezier_select_reset (bezier_sel);
g_free (bezier_sel);
}
/* Find the curve that points to this curve. This makes to_check point
* the start of a curve.
*/
static GimpBezierSelectPoint *
check_for_next_curve (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *to_check)
{
GimpBezierSelectPoint *points = bezier_sel->points;
gint num_points = bezier_sel->num_points;
while (points && num_points)
{
if (points->next_curve == to_check)
return points;
if (points->next_curve)
points = points->next_curve;
else
points = points->next;
num_points--;
}
return NULL;
}
static gint
count_points_on_curve (GimpBezierSelectPoint *points)
{
GimpBezierSelectPoint *start = points;
gint count = 0;
while (points && points->next != start)
{
points = points->next;
count++;
}
return count;
}
/* Find the start of the last open curve, if curve already closed
* this is an error..
*/
static GimpBezierSelectPoint *
find_start_open_curve (GimpBezierSelectTool *bsel)
{
GimpBezierSelectPoint *start_pnt = NULL;
GimpBezierSelectPoint *this_pnt = bsel->last_point;
/* Could be one of the first points */
if (! bsel->last_point)
return NULL;
if (bsel->closed)
{
g_message (_("Bezier path already closed."));
return NULL;
}
/* Step backwards until the prev point is null.
* in this case this is the start of the open curve.
* The start_pnt stuff is to stop us going around forever.
* If we even have a closed curve then we are in seroius
* trouble.
*/
while (this_pnt->prev && start_pnt != this_pnt)
{
if (!start_pnt)
start_pnt = this_pnt;
this_pnt = this_pnt->prev;
}
/* Must be an anchor to be the start */
if (start_pnt == this_pnt || this_pnt->type != BEZIER_ANCHOR)
{
g_message (_("Corrupt curve"));
return NULL;
}
/* g_print ("Returned start pnt of curve %p is %p\n",bsel->last_point,this_pnt); */
return this_pnt;
}
/* Delete a whole curve. Watch out for special cases.
* start_pnt must always be the start point if the curve to delete.
*/
static void
delete_whole_curve (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *start_pnt)
{
GimpBezierSelectPoint *tmppnt;
GimpBezierSelectPoint *next_curve = NULL; /* Next curve this one
* points at (if any)
*/
GimpBezierSelectPoint *prev_curve; /* Does a curve point to this one? */
gint cnt_pnts = 0; /* Count how many pnts deleted */
gint reset_last = FALSE;
/* shift and del means delete whole curve */
/* Three cases, this is first curve, middle curve
* or end curve.
*/
/* Does this curve have another chained on the end?
* or is this curve pointed to another one?
*/
/* g_print ("delete_whole_curve::\n"); */
tmppnt = start_pnt;
do
{
if (tmppnt->next_curve)
{
next_curve = tmppnt->next_curve;
break;
}
tmppnt = tmppnt->next;
}
while (tmppnt != start_pnt && tmppnt);
prev_curve = check_for_next_curve (bezier_sel, start_pnt);
/* First curve ?*/
if (bezier_sel->points == start_pnt)
{
bezier_sel->points = next_curve;
}
else
{
/* better have a previous curve else how did we get here? */
prev_curve->next_curve = next_curve;
}
/* start_pnt points to the curve we should free .. ignoring the next_curve */
tmppnt = start_pnt;
do
{
GimpBezierSelectPoint *fpnt;
fpnt = tmppnt;
tmppnt = tmppnt->next;
if (fpnt == bezier_sel->last_point)
reset_last = TRUE;
g_free (fpnt);
cnt_pnts++;
}
while (tmppnt != start_pnt && tmppnt);
bezier_sel->num_points -= cnt_pnts;
/* if deleted curve was unclosed then must have been the last curve
* and thus this curve becomes closed.
*/
if (!tmppnt && bezier_sel->num_points > 0)
{
bezier_sel->closed = 1;
bezier_sel->state = BEZIER_EDIT;
}
if (bezier_sel->num_points <= 0)
{
bezier_select_reset(bezier_sel);
}
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
/* The last point could have been on this curve as well... */
if (reset_last)
{
GimpBezierSelectPoint *points = bezier_sel->points;
GimpBezierSelectPoint *l_pnt = NULL;
gint num_points = bezier_sel->num_points;
while (points && num_points)
{
l_pnt = points;
if(points->next_curve)
points = points->next_curve;
else
points = points->next;
num_points--;
}
bezier_sel->last_point = l_pnt;
}
}
static gboolean
bezier_edit_point_on_curve (gint x,
gint y,
gint halfwidth,
GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
GimpTool *tool,
GdkEventButton *bevent)
{
gboolean grab_pointer = FALSE;
GimpBezierSelectPoint *next_curve;
GimpBezierSelectPoint *points = bezier_sel->points;
GimpBezierSelectPoint *start_pt = bezier_sel->points;
GimpBezierSelectPoint *last_curve = NULL;
gint point_counts = 0;
/* find if the button press occurred on a point */
do
{
point_counts = count_points_on_curve (points);
do
{
if (bezier_check_point (points, x, y, halfwidth))
{
GimpBezierSelectPoint *finded=points;
GimpBezierSelectPoint *start_op;
GimpBezierSelectPoint *end_op;
if (ModeEdit== EXTEND_REMOVE)
{
if ((bevent->state & GDK_SHIFT_MASK) || (point_counts <= 7))
{
/* Case 1: GDK_SHIFT_MASK - The user explicitly wishes
the present curve to go away.
Case 2: The current implementation cannot cope with
less than 7 points ie: 2 anchors points and
4 controls: the minimal closed curve.
Since the user wishes less than this implementation
minimum, we take this for an implicit wish that
the entire curve go away. G'bye dear curve.
*/
delete_whole_curve (bezier_sel,start_pt);
return TRUE;
}
else if (!finded->prev || !finded->prev->prev)
{
/* This is the first point on the curve */
/* Clear current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
/* Where are we? reset to first point... */
/*if(last_curve == NULL)*/
if (start_pt == bezier_sel->points)
{
finded = bezier_sel->points;
bezier_sel->points = finded->next->next->next;
bezier_sel->points->prev = NULL;
}
else
{
finded = last_curve->next_curve;
last_curve->next_curve = finded->next->next->next;
last_curve->next_curve->prev = NULL;
}
bezier_sel->num_points -= 3;
g_free (finded->next->next);
g_free (finded->next);
g_free (finded);
}
else if (!bezier_sel->closed &&
(finded == bezier_sel->last_point ||
finded == bezier_sel->last_point->prev ||
finded == bezier_sel->last_point->prev->prev))
{
/* This is the last point on the curve */
/* Clear current anchor and control */
/* Where are we? reset to last point... */
finded = bezier_sel->last_point->prev;
bezier_sel->last_point = finded->prev->prev;
bezier_sel->last_point->next = NULL;
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
bezier_sel->num_points -= 3;
g_free (finded->prev);
g_free (finded->next);
g_free (finded);
}
else
{
if (finded->type == BEZIER_CONTROL)
{
if (finded->next->type == BEZIER_CONTROL)
finded = finded->prev;
else
finded = finded->next;
}
start_op = finded->prev->prev;
end_op = finded->next->next;
/* we can use next_curve here since we are going to
* drop out the bottom anyways.
*/
next_curve = check_for_next_curve (bezier_sel,finded);
if (next_curve)
{
/* Deleteing first point of next curve*/
next_curve->next_curve = finded->prev->prev->prev;
}
else /* Can't be both first and a next curve!*/
{
if (bezier_sel->points == finded)
{
/* g_print ("Deleting first point %p\n",finded); */
bezier_sel->points = finded->prev->prev->prev;
}
}
/* Make sure the chain of curves is preserved */
if (finded->prev->next_curve)
{
/* g_print ("Moving curve on next_curve %p\n",finded->prev->next_curve); */
/* last point on closed multi-path */
finded->prev->prev->prev->prev->next_curve = finded->prev->next_curve;
}
if (bezier_sel->last_point == finded->prev)
{
/* g_print ("Deleting last point %p\n",finded->prev); */
bezier_sel->last_point = bezier_sel->last_point->prev->prev->prev;
}
start_op->next = end_op;
end_op->prev = start_op;
/* if ( (bezier_sel->last_point == finded) || */
/* (bezier_sel->last_point == finded->next) || */
/* (bezier_sel->last_point == finded->prev)) */
/* { */
/* bezier_sel->last_point = start_op->prev->prev; */
/* bezier_sel->points = start_op->prev; */
/* } */
bezier_sel->num_points -= 3;
g_free (finded->prev);
g_free (finded->next);
g_free (finded);
/* Clear current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
}
}
else
{
/* set the current anchor and control points */
switch (points->type)
{
case BEZIER_ANCHOR:
bezier_sel->cur_anchor = points;
bezier_sel->cur_control = bezier_sel->cur_anchor->next;
break;
case BEZIER_CONTROL:
bezier_sel->cur_control = points;
if (bezier_sel->cur_control->next &&
bezier_sel->cur_control->next->type == BEZIER_ANCHOR)
bezier_sel->cur_anchor = bezier_sel->cur_control->next;
else
bezier_sel->cur_anchor = bezier_sel->cur_control->prev;
break;
}
}
return TRUE;
}
next_curve = points->next_curve;
if (next_curve)
last_curve = points;
points = points->next;
}
while (points != start_pt && points);
start_pt = next_curve;
points = next_curve;
}
while (next_curve);
return grab_pointer;
}
static gboolean
bezier_add_point_on_segment (gint x,
gint y,
gint halfwidth,
GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
GimpTool *tool)
{
GimpBezierSelectPoint *points = bezier_sel->points;
GimpBezierSelectPoint *start_pt = bezier_sel->points;
GimpBezierSelectPoint *next_curve;
do
{
do
{
if (test_add_point_on_segment (bezier_sel,
points,
SUBDIVIDE,
IMAGE_COORDS,
x, y,
halfwidth))
{
return TRUE;
}
points = points->next;
if (!points)
return FALSE;
points = points->next;
if (!points)
return FALSE;
next_curve = points->next_curve;
points = points->next;
}
while (points != start_pt && points);
start_pt = next_curve;
points = next_curve;
}
while (next_curve);
return FALSE;
}
static void
bezier_start_new_segment (GimpBezierSelectTool *bezier_sel,
gint x,
gint y)
{
/* Must be closed to do this! */
if (!bezier_sel->closed)
return;
bezier_sel->closed = 0; /* End is no longer closed !*/
bezier_sel->state = BEZIER_ADD;
bezier_add_point (bezier_sel, BEZIER_MOVE, (gdouble) x, (gdouble) y);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble) x, (gdouble) y);
}
static void
bezier_select_button_press (GimpTool *tool,
GdkEventButton *bevent,
GDisplay *gdisp)
{
GimpBezierSelectTool *bezier_sel;
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *curve_start;
gboolean grab_pointer;
gint op;
gint x, y;
gint halfwidth, dummy;
tool->drawable = gimp_image_active_drawable (gdisp->gimage);
bezier_sel = (GimpBezierSelectTool *) tool;
grab_pointer = FALSE;
/* If the tool was being used in another image...reset it */
if (tool->state == ACTIVE && gdisp != tool->gdisp)
{
gimp_draw_tool_stop ( (GimpDrawTool *)bezier_sel);
bezier_select_reset (bezier_sel);
}
gdisplay_untransform_coords (gdisp, bevent->x, bevent->y, &x, &y, TRUE, 0);
/* get halfwidth in image coord */
gdisplay_untransform_coords (gdisp, bevent->x + BEZIER_HALFWIDTH, 0,
&halfwidth, &dummy, TRUE, 0);
halfwidth -= x;
curTool = active_tool;
curSel = (GimpBezierSelectTool *) curTool;
curGdisp = (GDisplay *) gdisp;
active_tool->gdisp = gdisp;
curCore = (GimpDrawTool *) bezier_sel;
switch (bezier_sel->state)
{
case BEZIER_START:
if (ModeEdit != EXTEND_NEW)
break;
grab_pointer = TRUE;
tool->state = ACTIVE;
tool->gdisp = gdisp;
/* if (bevent->state & GDK_MOD1_MASK) */
/* { */
/* init_edit_selection (tool, gdisp, bevent, EDIT_MASK_TRANSLATE); */
/* break; */
/* } */
/* else if (!(bevent->state & GDK_SHIFT_MASK) && !(bevent->state & GDK_CONTROL_MASK)) */
/* if (! (layer_is_floating_sel (gimp_image_get_active_layer (gdisp->gimage))) && */
/* gdisplay_mask_value (gdisp, bevent->x, bevent->y) > HALF_WAY) */
/* { */
/* init_edit_selection (tool, gdisp, bevent, EDIT_MASK_TO_LAYER_TRANSLATE); */
/* break; */
/* } */
bezier_sel->state = BEZIER_ADD;
/*bezier_sel->draw_mode = BEZIER_DRAW_CURVE; | BEZIER_DRAW_HANDLES;*/
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT;
bezier_add_point (bezier_sel, BEZIER_ANCHOR, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
gimp_draw_tool_start ( (GimpDrawTool *) bezier_sel, gdisp->canvas->window);
break;
case BEZIER_ADD:
grab_pointer = TRUE;
if (ModeEdit == EXTEND_EDIT)
{
/* erase the handles */
if(bezier_sel->closed)
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
else
bezier_sel->draw_mode = BEZIER_DRAW_CURVE;
gimp_draw_tool_pause ((GimpDrawTool *) bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
if (grab_pointer)
{
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
}
else
{
paths_dialog_set_default_op ();
/* recursive call */
bezier_select_button_press (tool, bevent, gdisp);
}
return;
}
if (ModeEdit == EXTEND_REMOVE)
{
/* if(bezier_sel->num_points < 6) */
/* return; */
/* erase the handles */
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
/*kkk*/
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
/*kkk*/
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
if (bezier_sel->num_points == 0)
{
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
paths_dialog_set_default_op ();
}
if (!grab_pointer)
{
paths_dialog_set_default_op ();
/* recursive call */
bezier_select_button_press (tool, bevent, gdisp);
}
return;
}
if (ModeEdit == EXTEND_ADD)
{
if (bezier_sel->num_points < 5)
return;
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
grab_pointer = bezier_add_point_on_segment (x, y, halfwidth,
gdisp, bezier_sel, tool);
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
if (grab_pointer)
{
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
}
else
{
paths_dialog_set_default_op ();
/* recursive call */
bezier_select_button_press (tool, bevent, gdisp);
}
return;
}
if (bezier_sel->cur_anchor)
{
if (bezier_check_point (bezier_sel->cur_anchor, x, y, halfwidth))
{
break;
}
if (bezier_sel->cur_anchor->next &&
bezier_check_point (bezier_sel->cur_anchor->next, x, y, halfwidth))
{
bezier_sel->cur_control = bezier_sel->cur_anchor->next;
break;
}
if (bezier_sel->cur_anchor->prev &&
bezier_check_point (bezier_sel->cur_anchor->prev, x, y, halfwidth))
{
bezier_sel->cur_control = bezier_sel->cur_anchor->prev;
break;
}
}
curve_start = find_start_open_curve(bezier_sel);
if (curve_start && bezier_check_point (curve_start, x, y, halfwidth))
{
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_sel->last_point->next = curve_start;
curve_start->prev = bezier_sel->last_point;
bezier_sel->cur_anchor = curve_start;
bezier_sel->cur_control = curve_start->next;
bezier_sel->closed = 1;
bezier_sel->state = BEZIER_EDIT;
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
}
else
{
if(bezier_sel->cur_anchor)
bezier_sel->cur_anchor->pointflags = 1;
bezier_sel->draw_mode = BEZIER_DRAW_HANDLES;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_ANCHOR, (gdouble)x, (gdouble)y);
bezier_add_point (bezier_sel, BEZIER_CONTROL, (gdouble)x, (gdouble)y);
bezier_sel->draw_mode = BEZIER_DRAW_CURRENT | BEZIER_DRAW_HANDLES;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
}
break;
case BEZIER_EDIT:
if (!bezier_sel->closed)
gimp_fatal_error ("bezier_select_button_press(): Tried to edit "
"on open bezier curve in edit selection");
/* erase the handles */
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
/* unset the current anchor and control */
bezier_sel->cur_anchor = NULL;
bezier_sel->cur_control = NULL;
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (ModeEdit == EXTEND_ADD)
{
if (bezier_sel->closed)
{
grab_pointer = bezier_add_point_on_segment (x, y, halfwidth,
gdisp, bezier_sel,
tool);
}
}
else
{
grab_pointer = bezier_edit_point_on_curve (x, y, halfwidth,
gdisp, bezier_sel,
tool, bevent);
if (grab_pointer && bezier_sel->num_points == 0)
{
gimp_draw_tool_pause ( (GimpDrawTool *) bezier_sel);
paths_dialog_set_default_op ();
}
}
if (!grab_pointer && gimp_channel_value (bezier_sel->mask, x, y))
{
gboolean replace = FALSE;
if ((bevent->state & GDK_SHIFT_MASK) &&
!(bevent->state & GDK_CONTROL_MASK))
op = CHANNEL_OP_ADD;
else if ((bevent->state & GDK_CONTROL_MASK) &&
!(bevent->state & GDK_SHIFT_MASK))
op = CHANNEL_OP_SUB;
else if ((bevent->state & GDK_CONTROL_MASK) &&
(bevent->state & GDK_SHIFT_MASK))
op = CHANNEL_OP_INTERSECT;
else
{
op = CHANNEL_OP_ADD;
replace = TRUE;
}
bezier_to_sel_internal (bezier_sel, tool, gdisp, op, replace);
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
}
else
{
/* draw the handles */
if (!grab_pointer)
{
paths_dialog_set_default_op ();
bezier_start_new_segment (bezier_sel, x, y);
}
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume ( (GimpDrawTool *) bezier_sel);
}
break;
}
if (grab_pointer)
gdk_pointer_grab (gdisp->canvas->window, FALSE,
GDK_POINTER_MOTION_HINT_MASK |
GDK_BUTTON1_MOTION_MASK |
GDK_BUTTON_RELEASE_MASK,
NULL, NULL, bevent->time);
/* Don't bother doing this if we don't have any points */
if (bezier_sel->num_points > 0)
paths_first_button_press(bezier_sel, gdisp);
}
/* returns 0 if not on control point, else BEZIER_ANCHOR or BEZIER_CONTROL */
static gint
bezier_on_control_point (GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
gint x,
gint y,
gint halfwidth)
{
GimpBezierSelectPoint *points;
gint num_points;
/* transform the points from image space to screen space */
points = bezier_sel->points;
num_points = bezier_sel->num_points;
while (points && num_points)
{
if (bezier_check_point (points, x, y, halfwidth))
return points->type;
if(points->next_curve)
points = points->next_curve;
else
points = points->next;
num_points--;
}
return FALSE;
}
static void
bezier_check_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data)
{
BezierCheckPnts *chkpnts = data;
gint halfwidth = chkpnts->halfwidth;
gint loop;
gint l, r, t, b;
/* Quick exit if already found */
if (chkpnts->found)
return;
for (loop = 0 ; loop < n_points; loop++)
{
l = points[loop].x - halfwidth;
r = points[loop].x + halfwidth;
t = points[loop].y - halfwidth;
b = points[loop].y + halfwidth;
/* g_print ("x,y = [%d,%d] halfwidth %d l,r,t,d [%d,%d,%d,%d]\n", */
/* points[loop].x, */
/* points[loop].y, */
/* halfwidth, */
/* l,r,t,b); */
if ((chkpnts->x >= l) &&
(chkpnts->x <= r) &&
(chkpnts->y >= t) &&
(chkpnts->y <= b))
{
chkpnts->found = TRUE;
}
}
}
static gboolean
points_in_box (GimpBezierSelectPoint *points,
gint x,
gint y)
{
/* below code adapted from Wm. Randolph Franklin <wrf@ecse.rpi.edu>
*/
gint i, j;
gint c = 0;
gdouble yp[4];
gdouble xp[4];
for (i = 0; i < 4; i++)
{
xp[i] = points->x;
yp[i] = points->y;
points = points->next;
}
/* Check if straight line ..below don't work if it is! */
if((xp[0] == xp[1] && yp[0] == yp[1]) ||
(xp[2] == xp[3] && yp[0] == yp[1]))
return TRUE;
for (i = 0, j = 3; i < 4; j = i++)
{
if ((((yp[i]<=y) && (y<yp[j])) ||
((yp[j]<=y) && (y<yp[i]))) &&
(x < (xp[j] - xp[i]) * (y - yp[i]) / (yp[j] - yp[i]) + xp[i]))
c = !c;
}
return c;
}
static gint
bezier_point_on_curve (GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel,
gint x,
gint y,
gint halfwidth)
{
BezierCheckPnts chkpnts;
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *next_curve;
CountCurves *cnt = (CountCurves *) &chkpnts;
gint point_counts = 0;
chkpnts.x = x;
chkpnts.y = y;
chkpnts.halfwidth = halfwidth;
chkpnts.found = FALSE;
if (cnt)
cnt->count = 0;
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (bezier_sel->num_points >= 4)
{
do
{
point_counts = count_points_on_curve(points);
if (point_counts >= 4)
{
do
{
if (points_in_box (points, x, y))
{
bezier_draw_segment (bezier_sel, points,
SUBDIVIDE, IMAGE_COORDS,
bezier_check_points,
&chkpnts);
}
points = next_anchor (points, &next_curve);
/* g_print ("next_anchor = %p\n",points); */
}
while (points != start_pt && points);
if (cnt)
cnt->count++;
start_pt = next_curve;
points = next_curve;
}
else
break; /* must be last curve since only this one is allowed < 4
* points.
*/
}
while (next_curve);
}
return chkpnts.found;
}
static void
gimp_bezier_select_tool_cursor_update (GimpTool *tool,
GdkEventMotion *mevent,
GDisplay *gdisp)
{
GimpBezierSelectTool *bezier_sel;
GimpDrawTool *draw_tool;
gboolean on_curve;
gboolean on_control_pnt;
gboolean in_selection_area;
gint halfwidth, dummy;
gint x, y;
bezier_sel = (GimpBezierSelectTool *) tool;
draw_tool = GIMP_DRAW_TOOL(bezier_sel);
if (gdisp != tool->gdisp || draw_tool->draw_state == INVISIBLE)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
return;
}
gdisplay_untransform_coords (gdisp, mevent->x, mevent->y, &x, &y, TRUE, 0);
/* get halfwidth in image coord */
gdisplay_untransform_coords (gdisp, mevent->x + BEZIER_HALFWIDTH, 0,
&halfwidth, &dummy, TRUE, 0);
halfwidth -= x;
on_control_pnt = bezier_on_control_point (gdisp, bezier_sel, x, y, halfwidth);
on_curve = bezier_point_on_curve (gdisp, bezier_sel, x, y, halfwidth);
if (bezier_sel->mask && bezier_sel->closed &&
gimp_channel_value (bezier_sel->mask, x, y) &&
!on_control_pnt &&
(!on_curve || ModeEdit != EXTEND_ADD))
{
in_selection_area = TRUE;
if ((mevent->state & GDK_SHIFT_MASK) &&
!(mevent->state & GDK_CONTROL_MASK))
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_RECT_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_PLUS);
}
else if ((mevent->state & GDK_CONTROL_MASK) &&
!(mevent->state & GDK_SHIFT_MASK))
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_RECT_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_MINUS);
}
else if ((mevent->state & GDK_CONTROL_MASK) &&
(mevent->state & GDK_SHIFT_MASK))
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_RECT_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_INTERSECT);
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_RECT_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
}
return;
}
if (mevent->state & GDK_MOD1_MASK)
{
/* Moving curve */
if (mevent->state & GDK_SHIFT_MASK)
{
/* moving on 1 curve */
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_MOVE);
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_MOVE);
}
}
else
{
switch (ModeEdit)
{
case EXTEND_NEW:
if (on_control_pnt && bezier_sel->closed)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_CONTROL);
/* g_print ("add to curve cursor\n"); */
}
else if (on_curve)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
/* g_print ("edit control point cursor\n"); */
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
}
break;
case EXTEND_ADD:
if (on_curve)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_PLUS);
/* g_print ("add to curve cursor\n"); */
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
/* g_print ("default no action cursor\n"); */
}
break;
case EXTEND_EDIT:
if (on_control_pnt)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_CONTROL);
/* g_print ("edit control point cursor\n"); */
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
/* g_print ("default no action cursor\n"); */
}
break;
case EXTEND_REMOVE:
if (on_control_pnt && mevent->state & GDK_SHIFT_MASK)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_MINUS);
/* g_print ("delete whole curve cursor\n"); */
}
else if (on_control_pnt)
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_MINUS);
/* g_print ("remove point cursor\n"); */
}
else
{
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
/* g_print ("default no action cursor\n"); */
}
break;
default:
g_print ("In default\n");
gdisplay_install_tool_cursor (gdisp,
GIMP_MOUSE_CURSOR,
GIMP_BEZIER_SELECT_TOOL_CURSOR,
GIMP_CURSOR_MODIFIER_NONE);
break;
}
}
}
static void
gimp_bezier_select_tool_control (GimpTool *tool,
ToolAction action,
GDisplay *gdisp)
{
GimpBezierSelectTool * bezier_sel;
bezier_sel = (GimpBezierSelectTool *) tool;
switch (action)
{
case PAUSE :
gimp_draw_tool_pause ((GimpDrawTool *) bezier_sel);
break;
case RESUME :
gimp_draw_tool_resume ((GimpDrawTool *) bezier_sel);
break;
case HALT :
gimp_draw_tool_stop ((GimpDrawTool *) bezier_sel);
bezier_select_reset (bezier_sel);
break;
default:
break;
}
}
void
bezier_draw (GDisplay *gdisp,
GimpBezierSelectTool *bezier_sel)
{
GimpBezierSelectPoint *points;
gint num_points;
gint draw_curve;
gint draw_handles;
gint draw_current;
if (!bezier_sel->draw_mode)
return;
draw_curve = bezier_sel->draw_mode & BEZIER_DRAW_CURVE;
draw_current = bezier_sel->draw_mode & BEZIER_DRAW_CURRENT;
draw_handles = bezier_sel->draw_mode & BEZIER_DRAW_HANDLES;
/* reset to the default drawing state of drawing the curve and handles */
bezier_sel->draw_mode = BEZIER_DRAW_ALL;
/* transform the points from image space to screen space */
points = bezier_sel->points;
num_points = bezier_sel->num_points;
while (points && num_points)
{
gdisplay_transform_coords (gdisp, points->x, points->y,
&points->sx, &points->sy, 0);
if (points->next_curve)
points = points->next_curve;
else
points = points->next;
num_points--;
}
if (draw_curve)
{
bezier_draw_curve (bezier_sel, bezier_draw_segment_points,
SCREEN_COORDS, NULL);
bezier_draw_handles (bezier_sel, TRUE);
}
else if (draw_current)
{
bezier_draw_current (bezier_sel);
bezier_draw_handles (bezier_sel, FALSE);
}
else if (draw_handles)
{
bezier_draw_handles (bezier_sel, FALSE);
}
}
static void
bezier_select_draw (GimpDrawTool *draw_tool)
{
GDisplay *gdisp;
GimpTool *tool;
GimpBezierSelectTool *bezier_sel;
bezier_sel = (GimpBezierSelectTool *)draw_tool;
tool = GIMP_TOOL(bezier_sel);
gdisp = tool->gdisp;
bezier_draw (gdisp, bezier_sel);
}
void
bezier_add_point (GimpBezierSelectTool *bezier_sel,
gint type,
gdouble x,
gdouble y)
{
GimpBezierSelectPoint *newpt;
newpt = g_new0 (GimpBezierSelectPoint,1);
newpt->type = type;
newpt->x = x;
newpt->y = y;
newpt->next = NULL;
newpt->prev = NULL;
newpt->next_curve = NULL;
if (type == BEZIER_MOVE &&
bezier_sel->last_point)
{
/* g_print ("Adding move point\n"); */
newpt->type = BEZIER_ANCHOR;
bezier_sel->last_point->next_curve = newpt;
bezier_sel->last_point = newpt;
bezier_sel->cur_anchor = newpt;
}
else
{
if(type == BEZIER_MOVE)
{
newpt->type = BEZIER_ANCHOR;
/* g_print ("Adding MOVE point to null curve\n"); */
}
if (bezier_sel->last_point)
{
bezier_sel->last_point->next = newpt;
newpt->prev = bezier_sel->last_point;
bezier_sel->last_point = newpt;
}
else
{
bezier_sel->points = newpt;
bezier_sel->last_point = newpt;
}
switch (type)
{
case BEZIER_ANCHOR:
bezier_sel->cur_anchor = newpt;
break;
case BEZIER_CONTROL:
bezier_sel->cur_control = newpt;
break;
}
}
bezier_sel->num_points += 1;
}
static void
bezier_offset_point (GimpBezierSelectPoint *pt,
gint x,
gint y)
{
if (pt)
{
pt->x += x;
pt->y += y;
}
}
static int
bezier_check_point (GimpBezierSelectPoint *pt,
gint x,
gint y,
gint halfwidth)
{
gint l, r, t, b;
if (pt)
{
l = pt->x - halfwidth;
r = pt->x + halfwidth;
t = pt->y - halfwidth;
b = pt->y + halfwidth;
return ((x >= l) && (x <= r) && (y >= t) && (y <= b));
}
return 0;
}
static void
bezier_draw_handles (GimpBezierSelectTool *bezier_sel,
gboolean do_all)
{
GimpBezierSelectPoint *points;
gint num_points;
points = bezier_sel->points;
num_points = bezier_sel->num_points;
if (num_points <= 0)
return;
while (num_points && num_points > 0)
{
if (points == bezier_sel->cur_anchor)
{
/* g_print ("bezier_draw_handles:: found cur_anchor %p\n",points); */
bezier_draw_point (bezier_sel, points, FALSE);
bezier_draw_point (bezier_sel, points->next, FALSE);
bezier_draw_point (bezier_sel, points->prev, FALSE);
bezier_draw_line (bezier_sel, points, points->next);
bezier_draw_line (bezier_sel, points, points->prev);
}
else
{
/* g_print ("bezier_draw_handles:: not found cur_anchor %p\n",points); */
if (do_all || points->pointflags == 1)
{
bezier_draw_point (bezier_sel, points, TRUE);
points->pointflags = 0;
}
}
if (points)
{
if (points->next_curve)
points = points->next_curve;
else
points = points->next;
}
if (points)
{
if (points->next_curve)
points = points->next_curve;
else
points = points->next;
}
if (points)
{
if (points->next_curve)
points = points->next_curve;
else
points = points->next;
}
num_points -= 3;
}
}
static void
bezier_draw_current (GimpBezierSelectTool *bezier_sel)
{
GimpBezierSelectPoint *points;
points = bezier_sel->cur_anchor;
if (points) points = points->prev;
if (points) points = points->prev;
if (points) points = points->prev;
if (points)
bezier_draw_segment (bezier_sel, points,
SUBDIVIDE, SCREEN_COORDS,
bezier_draw_segment_points,
NULL);
if (points != bezier_sel->cur_anchor)
{
points = bezier_sel->cur_anchor;
if (points) points = points->next;
if (points) points = points->next;
if (points) points = points->next;
if (points)
bezier_draw_segment (bezier_sel, bezier_sel->cur_anchor,
SUBDIVIDE, SCREEN_COORDS,
bezier_draw_segment_points,
NULL);
}
}
static void
bezier_draw_point (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt,
gboolean fill)
{
GimpDrawTool *draw_tool;
draw_tool = GIMP_DRAW_TOOL(bezier_sel);
if (pt)
{
switch (pt->type)
{
case BEZIER_ANCHOR:
if (fill)
{
gdk_draw_arc (draw_tool->win, draw_tool->gc,
TRUE,
pt->sx - BEZIER_HALFWIDTH, pt->sy - BEZIER_HALFWIDTH,
BEZIER_WIDTH, BEZIER_WIDTH, 0, 23040);
}
else
{
gdk_draw_arc (draw_tool->win, draw_tool->gc,
FALSE,
pt->sx - BEZIER_HALFWIDTH, pt->sy - BEZIER_HALFWIDTH,
BEZIER_WIDTH, BEZIER_WIDTH, 0, 23040);
}
break;
case BEZIER_CONTROL:
if (fill)
{
gdk_draw_rectangle (draw_tool->win, draw_tool->gc,
TRUE,
pt->sx - BEZIER_HALFWIDTH, pt->sy - BEZIER_HALFWIDTH,
BEZIER_WIDTH, BEZIER_WIDTH);
}
else
{
gdk_draw_rectangle (draw_tool->win, draw_tool->gc,
FALSE,
pt->sx - BEZIER_HALFWIDTH, pt->sy - BEZIER_HALFWIDTH,
BEZIER_WIDTH, BEZIER_WIDTH);
}
break;
}
}
}
static void
bezier_draw_line (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt1,
GimpBezierSelectPoint *pt2)
{
GimpDrawTool *draw_tool;
draw_tool = GIMP_DRAW_TOOL(bezier_sel);
if (pt1 && pt2)
{
gdk_draw_line (draw_tool->win,
draw_tool->gc,
pt1->sx, pt1->sy, pt2->sx, pt2->sy);
}
}
static void
bezier_draw_segment (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *points,
gint subdivisions,
gint space,
GimpBezierSelectPointsFunc points_func,
gpointer data)
{
static GdkPoint gdk_points[256];
static gint n_points = 256;
BezierMatrix geometry;
BezierMatrix tmp1, tmp2;
BezierMatrix deltas;
gdouble x, dx, dx2, dx3;
gdouble y, dy, dy2, dy3;
gdouble d, d2, d3;
gint lastx, lasty;
gint newx, newy;
gint index;
gint i;
/* construct the geometry matrix from the segment */
/* assumes that a valid segment containing 4 points is passed in */
for (i = 0; i < 4; i++)
{
if (!points)
gimp_fatal_error ("bezier_draw_segment(): Bad bezier segment");
switch (space)
{
case IMAGE_COORDS:
geometry[i][0] = points->x;
geometry[i][1] = points->y;
break;
case AA_IMAGE_COORDS:
geometry[i][0] = (points->x * SUPERSAMPLE);
geometry[i][1] = (points->y * SUPERSAMPLE);
break;
case SCREEN_COORDS:
geometry[i][0] = points->sx;
geometry[i][1] = points->sy;
break;
default:
gimp_fatal_error ("bezier_draw_segment(): Unknown coordinate space: %d", space);
break;
}
geometry[i][2] = 0;
geometry[i][3] = 0;
points = points->next;
}
/* subdivide the curve n times */
/* n can be adjusted to give a finer or coarser curve */
d = 1.0 / subdivisions;
d2 = d * d;
d3 = d * d * d;
/* construct a temporary matrix for determining the forward diffencing deltas */
tmp2[0][0] = 0; tmp2[0][1] = 0; tmp2[0][2] = 0; tmp2[0][3] = 1;
tmp2[1][0] = d3; tmp2[1][1] = d2; tmp2[1][2] = d; tmp2[1][3] = 0;
tmp2[2][0] = 6*d3; tmp2[2][1] = 2*d2; tmp2[2][2] = 0; tmp2[2][3] = 0;
tmp2[3][0] = 6*d3; tmp2[3][1] = 0; tmp2[3][2] = 0; tmp2[3][3] = 0;
/* compose the basis and geometry matrices */
bezier_compose (basis, geometry, tmp1);
/* compose the above results to get the deltas matrix */
bezier_compose (tmp2, tmp1, deltas);
/* extract the x deltas */
x = deltas[0][0];
dx = deltas[1][0];
dx2 = deltas[2][0];
dx3 = deltas[3][0];
/* extract the y deltas */
y = deltas[0][1];
dy = deltas[1][1];
dy2 = deltas[2][1];
dy3 = deltas[3][1];
lastx = x;
lasty = y;
gdk_points[0].x = (lastx);
gdk_points[0].y = (lasty);
index = 1;
/* loop over the curve */
for (i = 0; i < subdivisions; i++)
{
/* increment the x values */
x += dx;
dx += dx2;
dx2 += dx3;
/* increment the y values */
y += dy;
dy += dy2;
dy2 += dy3;
newx = ROUND (x);
newy = ROUND (y);
/* if this point is different than the last one...then draw it */
if ((lastx != newx) || (lasty != newy))
{
/* add the point to the point buffer */
gdk_points[index].x = (newx);
gdk_points[index].y = (newy);
index++;
/* if the point buffer is full put it to the screen and zero it out */
if (index >= n_points)
{
(* points_func) (bezier_sel, gdk_points, index, data);
index = 0;
}
}
lastx = newx;
lasty = newy;
}
/* if there are points in the buffer, then put them on the screen */
if (index)
(* points_func) (bezier_sel, gdk_points, index, data);
}
static void
bezier_draw_segment_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data)
{
GimpDrawTool *draw_tool;
draw_tool = GIMP_DRAW_TOOL (bezier_sel);
gdk_draw_points (draw_tool->win,
draw_tool->gc, points, n_points);
}
static void
bezier_compose (BezierMatrix a,
BezierMatrix b,
BezierMatrix ab)
{
int i, j;
for (i = 0; i < 4; i++)
{
for (j = 0; j < 4; j++)
{
ab[i][j] = (a[i][0] * b[0][j] +
a[i][1] * b[1][j] +
a[i][2] * b[2][j] +
a[i][3] * b[3][j]);
}
}
}
static int start_convert;
static int width, height;
static int lastx;
static int lasty;
static void
bezier_convert (GimpBezierSelectTool *bezier_sel,
GDisplay *gdisp,
gint subdivisions,
gboolean antialias)
{
PixelRegion maskPR;
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *next_curve;
GSList *list;
guchar *buf, *b;
gint draw_type;
gint *vals, val;
gint start, end;
gint x, x2, w;
gint i, j;
if (!bezier_sel->closed)
gimp_fatal_error ("bezier_convert(): tried to convert an open bezier curve");
/* destroy previous mask */
if (bezier_sel->mask)
{
gtk_object_unref (GTK_OBJECT (bezier_sel->mask));
bezier_sel->mask = NULL;
}
/* get the new mask's maximum extents */
if (antialias)
{
buf = (unsigned char *) g_malloc (width);
width = gdisp->gimage->width * SUPERSAMPLE;
height = gdisp->gimage->height * SUPERSAMPLE;
draw_type = AA_IMAGE_COORDS;
/* allocate value array */
vals = (int *) g_malloc (sizeof (int) * width);
}
else
{
buf = NULL;
width = gdisp->gimage->width;
height = gdisp->gimage->height;
draw_type = IMAGE_COORDS;
vals = NULL;
}
/* create a new mask */
bezier_sel->mask = gimp_channel_new_mask (gdisp->gimage,
gdisp->gimage->width,
gdisp->gimage->height);
gtk_object_ref (GTK_OBJECT (bezier_sel->mask));
gtk_object_sink (GTK_OBJECT (bezier_sel->mask));
/* allocate room for the scanlines */
bezier_sel->scanlines = g_malloc (sizeof (GSList *) * height);
/* zero out the scanlines */
for (i = 0; i < height; i++)
bezier_sel->scanlines[i] = NULL;
/* scan convert the curve */
points = bezier_sel->points;
start_pt = bezier_sel->points;
do {
start_convert = 1;
do {
bezier_draw_segment (bezier_sel, points,
subdivisions, draw_type,
bezier_convert_points,
NULL);
/* advance to the next segment */
points = next_anchor(points,&next_curve);
} while (points != start_pt && points);
start_pt = next_curve;
points = next_curve;
} while (next_curve);
pixel_region_init (&maskPR,
gimp_drawable_data (GIMP_DRAWABLE (bezier_sel->mask)),
0, 0,
gimp_drawable_width (GIMP_DRAWABLE (bezier_sel->mask)),
gimp_drawable_height (GIMP_DRAWABLE (bezier_sel->mask)),
TRUE);
for (i = 0; i < height; i++)
{
list = bezier_sel->scanlines[i];
/* zero the vals array */
if (antialias && !(i % SUPERSAMPLE))
memset (vals, 0, width * sizeof (int));
while (list)
{
x = (long) list->data;
list = list->next;
/*
if (!list)
g_message ("cannot properly scanline convert bezier curve: %d", i);
else
*/
{
/* bounds checking */
x = CLAMP (x, 0, width);
x2 = CLAMP ((long) list->data, 0, width);
w = x2 - x;
if (!antialias)
gimp_channel_add_segment (bezier_sel->mask, x, i, w, 255);
else
for (j = 0; j < w; j++)
vals[j + x] += 255;
list = g_slist_next (list);
}
}
if (antialias && !((i+1) % SUPERSAMPLE))
{
b = buf;
start = 0;
end = width;
for (j = start; j < end; j += SUPERSAMPLE)
{
val = 0;
for (x = 0; x < SUPERSAMPLE; x++)
val += vals[j + x];
*b++ = (unsigned char) (val / SUPERSAMPLE2);
}
pixel_region_set_row (&maskPR, 0, (i / SUPERSAMPLE),
gimp_drawable_width (GIMP_DRAWABLE (bezier_sel->mask)), buf);
}
g_slist_free (bezier_sel->scanlines[i]);
}
if (antialias)
{
g_free (vals);
g_free (buf);
}
g_free (bezier_sel->scanlines);
bezier_sel->scanlines = NULL;
gimp_channel_invalidate_bounds (bezier_sel->mask);
}
static void
bezier_convert_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data)
{
gint i;
if (start_convert)
start_convert = 0;
else
bezier_convert_line (bezier_sel->scanlines,
lastx, lasty, points[0].x, points[0].y);
for (i = 0; i < (n_points - 1); i++)
{
bezier_convert_line (bezier_sel->scanlines,
points[i].x, points[i].y,
points[i+1].x, points[i+1].y);
}
lastx = points[n_points - 1].x;
lasty = points[n_points - 1].y;
}
static void
bezier_convert_line (GSList **scanlines,
gint x1,
gint y1,
gint x2,
gint y2)
{
gint dx, dy;
gint error, inc;
gint tmp;
gdouble slope;
if (y1 == y2)
return;
if (y1 > y2)
{
tmp = y2; y2 = y1; y1 = tmp;
tmp = x2; x2 = x1; x1 = tmp;
}
if (y1 < 0)
{
if (y2 < 0)
return;
if (x2 == x1)
{
y1 = 0;
}
else
{
slope = (double) (y2 - y1) / (double) (x2 - x1);
x1 = x2 + (int)(0.5 + (double)(0 - y2) / slope);
y1 = 0;
}
}
if (y2 >= height)
{
if (y1 >= height)
return;
if (x2 == x1)
{
y2 = height;
}
else
{
slope = (double) (y2 - y1) / (double) (x2 - x1);
x2 = x1 + (int)(0.5 + (double)(height - y1) / slope);
y2 = height;
}
}
if (y1 == y2)
return;
dx = x2 - x1;
dy = y2 - y1;
scanlines = &scanlines[y1];
if (((dx < 0) ? -dx : dx) > ((dy < 0) ? -dy : dy))
{
if (dx < 0)
{
inc = -1;
dx = -dx;
}
else
{
inc = 1;
}
error = -dx /2;
while (x1 != x2)
{
error += dy;
if (error > 0)
{
error -= dx;
*scanlines = bezier_insert_in_list (*scanlines, x1);
scanlines++;
}
x1 += inc;
}
}
else
{
error = -dy /2;
if (dx < 0)
{
dx = -dx;
inc = -1;
}
else
{
inc = 1;
}
while (y1++ < y2)
{
*scanlines = bezier_insert_in_list (*scanlines, x1);
scanlines++;
error += dx;
if (error > 0)
{
error -= dy;
x1 += inc;
}
}
}
}
static GSList *
bezier_insert_in_list (GSList *list,
gint x)
{
GSList *orig = list;
GSList *rest;
if (!list)
return g_slist_prepend (list, (void *) ((long) x));
while (list)
{
rest = g_slist_next (list);
if (x < (long) list->data)
{
rest = g_slist_prepend (rest, list->data);
list->next = rest;
list->data = (void *) ((long) x);
return orig;
}
else if (!rest)
{
g_slist_append (list, (void *) ((long) x));
return orig;
}
list = g_slist_next (list);
}
return orig;
}
gboolean
bezier_tool_selected (void)
{
return (active_tool &&
GIMP_IS_BEZIER_SELECT_TOOL (active_tool) &&
active_tool->state == ACTIVE);
}
void
bezier_paste_bezierselect_to_current (GDisplay *gdisp,
GimpBezierSelectTool *bsel)
{
GimpBezierSelectPoint *pts;
gint i;
GimpTool *tool;
GimpBezierSelectPoint *bpnt = NULL;
gint need_move = 0;
/* g_print ("bezier_paste_bezierselect_to_current::\n"); */
/* printSel(bsel); */
/* If the tool was being used before clear it */
if (active_tool &&
GIMP_IS_BEZIER_SELECT_TOOL (active_tool) &&
active_tool->state == ACTIVE)
{
GimpBezierSelectTool *bezier_sel = (GimpBezierSelectTool *) active_tool;
if(bezier_sel)
{
gimp_draw_tool_stop ((GimpDrawTool *) curSel);
bezier_select_reset (bezier_sel);
}
}
gimp_context_set_tool (gimp_context_get_user (),
tool_manager_get_info_by_type
( GIMP_TYPE_BEZIER_SELECT_TOOL));
active_tool->paused_count = 0;
active_tool->gdisp = gdisp;
active_tool->drawable = gimp_image_active_drawable (gdisp->gimage);
tool = active_tool;
bezier_select_reset (curSel);
gimp_draw_tool_start ((GimpDrawTool *) curSel, gdisp->canvas->window);
tool->state = ACTIVE;
pts = (GimpBezierSelectPoint *) bsel->points;
for (i = 0; i < bsel->num_points; i++)
{
if (need_move)
{
bezier_add_point (curSel, BEZIER_MOVE, pts->x, pts->y);
need_move = 0;
}
else
{
bezier_add_point (curSel, pts->type, pts->x, pts->y);
}
if (bpnt == NULL)
bpnt = curSel->last_point;
if (pts->next_curve)
{
/* g_print ("bezier_paste_bezierselect_to_current:: Close last curve off \n"); */
curSel->last_point->next = bpnt;
bpnt->prev = curSel->last_point;
curSel->cur_anchor = NULL;
curSel->cur_control = NULL;
bpnt = NULL;
need_move = 1;
pts = pts->next_curve;
}
else
{
pts = pts->next;
}
}
if (bsel->closed)
{
curSel->last_point->next = bpnt;
bpnt->prev = curSel->last_point;
curSel->cur_anchor = curSel->points;
curSel->cur_control = curSel->points->next;
curSel->closed = 1;
if (curTool->gdisp)
bezier_convert (curSel, curTool->gdisp, SUBDIVIDE, FALSE);
}
/* g_print ("After pasting...\n"); */
/* printSel(curSel); */
if (bsel->num_points == 0)
{
curSel->state = BEZIER_START;
curSel->draw_mode = 0;
gimp_draw_tool_stop( (GimpDrawTool *) curSel);
}
else
{
curSel->state = bsel->state;
curSel->draw_mode = BEZIER_DRAW_ALL;
gimp_draw_tool_resume( (GimpDrawTool *) curSel);
}
}
static void
bezier_to_sel_internal (GimpBezierSelectTool *bezier_sel,
GimpTool *tool,
GDisplay *gdisp,
gint op,
gint replace)
{
/* If we're antialiased, then recompute the mask...
*/
if (bezier_options->antialias)
bezier_convert (bezier_sel, tool->gdisp, SUBDIVIDE, TRUE);
if (replace)
gimage_mask_clear (gdisp->gimage);
else
gimage_mask_undo (gdisp->gimage);
if (bezier_options->feather)
gimp_channel_feather (bezier_sel->mask,
gimp_image_get_mask (gdisp->gimage),
bezier_options->feather_radius,
bezier_options->feather_radius,
op, 0, 0);
else
gimp_channel_combine_mask (gimp_image_get_mask (gdisp->gimage),
bezier_sel->mask, op, 0, 0);
/* show selection on all views */
gdisplays_flush ();
}
static gboolean
test_add_point_on_segment (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *pt,
gint subdivisions,
gint space,
gint xpos,
gint ypos,
gint halfwidth)
{
GimpBezierSelectPoint *points;
BezierMatrix geometry;
BezierMatrix tmp1, tmp2;
BezierMatrix deltas;
gdouble x, dx, dx2, dx3;
gdouble y, dy, dy2, dy3;
gdouble d, d2, d3;
gint lastx, lasty;
gint newx, newy;
gint i;
gdouble ratio;
/* construct the geometry matrix from the segment assumes that a
* valid segment containing 4 points is passed in ALT ignore invalid
* segments since we might be working on an open curve
*/
points = pt;
ratio = -1.0;
for (i = 0; i < 4; i++)
{
if (!points)
return FALSE;
switch (space)
{
case IMAGE_COORDS:
geometry[i][0] = points->x;
geometry[i][1] = points->y;
break;
case AA_IMAGE_COORDS:
geometry[i][0] = RINT(points->x * SUPERSAMPLE);
geometry[i][1] = RINT(points->y * SUPERSAMPLE);
break;
case SCREEN_COORDS:
geometry[i][0] = points->sx;
geometry[i][1] = points->sy;
break;
default:
gimp_fatal_error ("test_add_point_on_segment(): Unknown coordinate space: %d", space);
break;
}
geometry[i][2] = 0;
geometry[i][3] = 0;
points = points->next;
}
/* subdivide the curve n times n can be adjusted to give a finer or
* coarser curve
*/
d = 1.0 / subdivisions;
d2 = d * d;
d3 = d * d * d;
/* construct a temporary matrix for determining the forward
* diffencing deltas
*/
tmp2[0][0] = 0; tmp2[0][1] = 0; tmp2[0][2] = 0; tmp2[0][3] = 1;
tmp2[1][0] = d3; tmp2[1][1] = d2; tmp2[1][2] = d; tmp2[1][3] = 0;
tmp2[2][0] = 6*d3; tmp2[2][1] = 2*d2; tmp2[2][2] = 0; tmp2[2][3] = 0;
tmp2[3][0] = 6*d3; tmp2[3][1] = 0; tmp2[3][2] = 0; tmp2[3][3] = 0;
/* compose the basis and geometry matrices */
bezier_compose (basis, geometry, tmp1);
/* compose the above results to get the deltas matrix */
bezier_compose (tmp2, tmp1, deltas);
/* extract the x deltas */
x = deltas[0][0];
dx = deltas[1][0];
dx2 = deltas[2][0];
dx3 = deltas[3][0];
/* extract the y deltas */
y = deltas[0][1];
dy = deltas[1][1];
dy2 = deltas[2][1];
dy3 = deltas[3][1];
lastx = x;
lasty = y;
/* loop over the curve */
for (i = 0; i < subdivisions; i++)
{
/* increment the x values */
x += dx;
dx += dx2;
dx2 += dx3;
/* increment the y values */
y += dy;
dy += dy2;
dy2 += dy3;
newx = ROUND (x);
newy = ROUND (y);
/* if this point is different than the last one...then test it */
if ((lastx != newx) || (lasty != newy))
{
int l, r, b, t;
l = newx - halfwidth;
r = newx + halfwidth;
t = newy - halfwidth;
b = newy + halfwidth;
if ((xpos >= l) && (xpos <= r) && (ypos >= t) && (ypos <= b))
{
/* so we found one point in the square hit */
ratio = (double)i/(double)subdivisions;
/* We found the exact point on the curve, so take it ...*/
if ((xpos==newx) && (ypos==newy)) break;
/* To Implement: keep each time the nearest point of the
* curve from where we've clicked in the case where we
* haven't click exactely on the curve.
*/
}
}
lastx = newx;
lasty = newy;
}
/* we found a point on the curve */
if (ratio >= 0.0)
{
GimpBezierSelectPoint *pts, *pt1, *pt2, *pt3;
GimpBezierSelectPoint *P00, *P01, *P02, *P03;
GimpBezierSelectPoint P10, P11, P12;
GimpBezierSelectPoint P20, P21;
GimpBezierSelectPoint P30;
pts = pt;
P00 = pts;
pts = pts->next;
P01 = pts;
pts = pts->next;
P02 = pts;
pts = pts->next;
P03 = pts;
/* De Casteljau algorithme
[Advanced Animation & Randering Technics / Alan & Mark WATT]
[ADDISON WESLEY ref 54412]
Iteratif way of drawing a Bezier curve by geometrical approch
P0x represent the four controls points ( anchor / control /control /anchor )
P30 represent the new anchor point to add on the curve
P2x represent the new control points of P30
P1x represent the new values of the control points P01 and P02
so if we moves ratio from 0 to 1 we draw the all curve between P00 and P03
*/
P10.x = (int)((1-ratio)*P00->x + ratio * P01->x);
P10.y = (int)((1-ratio)*P00->y + ratio * P01->y);
P11.x = (1-ratio)*P01->x + ratio * P02->x;
P11.y = (1-ratio)*P01->y + ratio * P02->y;
P12.x = (1-ratio)*P02->x + ratio * P03->x;
P12.y = (1-ratio)*P02->y + ratio * P03->y;
P20.x = (1-ratio)*P10.x + ratio * P11.x;
P20.y = (1-ratio)*P10.y + ratio * P11.y;
P21.x = (1-ratio)*P11.x + ratio * P12.x;
P21.y = (1-ratio)*P11.y + ratio * P12.y;
P30.x = (1-ratio)*P20.x + ratio * P21.x;
P30.y = (1-ratio)*P20.y + ratio * P21.y;
P01->x = P10.x;
P01->y = P10.y;
P02->x = P12.x;
P02->y = P12.y;
/* All the computes are done, let's insert the new point on the curve */
pt1 = g_new0(GimpBezierSelectPoint,1);
pt2 = g_new0(GimpBezierSelectPoint,1);
pt3 = g_new0(GimpBezierSelectPoint,1);
pt1->type = BEZIER_CONTROL;
pt2->type = BEZIER_ANCHOR;
pt3->type = BEZIER_CONTROL;
pt1->x = P20.x; pt1->y = P20.y;
pt2->x = P30.x; pt2->y = P30.y;
pt3->x = P21.x; pt3->y = P21.y;
pt3->next_curve = P01->next_curve;
P01->next = pt1;
pt1->prev = P01;
pt1->next = pt2;
pt2->prev = pt1;
pt2->next = pt3;
pt3->prev = pt2;
pt3->next = P02;
P02->prev = pt3;
bezier_sel->num_points += 3;
bezier_sel->cur_anchor = pt2;
bezier_sel->cur_control = pt1;
return TRUE;
}
return FALSE;
}
void
bezier_select_mode (gint mode)
{
ModeEdit = mode;
}
/* The curve has to be closed to do a selection. */
void
bezier_to_selection (GimpBezierSelectTool *bezier_sel,
GDisplay *gdisp)
{
/* Call the internal function */
if (!bezier_sel->closed)
{
g_message (_("Curve not closed!"));
return;
}
/* force the passed selection to be the current selection...
* This loads it into curSel for this image
*/
bezier_paste_bezierselect_to_current (gdisp, bezier_sel);
bezier_to_sel_internal (curSel, (GimpTool *) curTool, gdisp, CHANNEL_OP_ADD, TRUE);
}
/* check whether vectors (offx, offy), (l_offx, l_offy) have the same angle. */
static gboolean
stroke_interpolatable (gint offx,
gint offy,
gint l_offx,
gint l_offy,
gdouble error)
{
if ((offx == l_offx) & (offy == l_offy))
{
return TRUE;
}
else if ((offx == 0) | (l_offx == 0))
{
if (offx == l_offx)
return ((0 <= offy) & (0 <= offy)) | ((offy < 0) & (l_offy < 0));
else
return FALSE;
}
else if ((offy == 0) | (l_offy == 0))
{
if (offy == l_offy)
return ((0 < offx) & (0 < l_offx)) | ((offx < 0) & (l_offx < 0));
else
return FALSE;
}
/* At this point, we can assert: offx, offy, l_offx, l_offy != 0 */
else if (((0 < offx) & (0 < l_offx)) | ((offx < 0) & (l_offx < 0)))
{
gdouble grad1, grad2;
if (ABS (offy) < ABS (offx))
{
grad1 = (gdouble) offy / (gdouble) offx;
grad2 = (gdouble) l_offy / (gdouble) l_offx;
}
else
{
grad1 = (gdouble) offx / (gdouble) offy;
grad2 = (gdouble) l_offx / (gdouble) l_offy;
}
/* printf ("error: %f / %f\n", ABS (grad1 - grad2), error); */
return ABS (grad1 - grad2) <= error;
}
return FALSE;
}
static void
bezier_stack_points_aux (GdkPoint *points,
gint start,
gint end,
gdouble error,
BezierRenderPnts *rpnts)
{
const gint expand_size = 32;
gint med;
gint offx, offy, l_offx, l_offy;
if (rpnts->stroke_points == NULL)
return;
/* BASE CASE: stack the end point */
if (end - start <= 1)
{
if ((rpnts->stroke_points[rpnts->num_stroke_points * 2 - 2] == points[end].x)
&& (rpnts->stroke_points[rpnts->num_stroke_points * 2 - 1] == points[end].y))
return;
rpnts->num_stroke_points++;
if (rpnts->len_stroke_points <= rpnts->num_stroke_points)
{
rpnts->len_stroke_points += expand_size;
rpnts->stroke_points = g_renew (double, rpnts->stroke_points, 2 * rpnts->len_stroke_points);
if (rpnts->stroke_points == NULL)
{
rpnts->len_stroke_points = rpnts->num_stroke_points = 0;
return;
}
}
rpnts->stroke_points[rpnts->num_stroke_points * 2 - 2] = points[end].x;
rpnts->stroke_points[rpnts->num_stroke_points * 2 - 1] = points[end].y;
return;
}
if (end - start <= 32)
{
gint i;
for (i = start+ 1; i <= end; i++)
bezier_stack_points_aux (points, i, i, 0,rpnts);
return;
}
/* Otherwise, check whether to divide the segment recursively */
offx = points[end].x - points[start].x;
offy = points[end].y - points[start].y;
med = (end + start) / 2;
l_offx = points[med].x - points[start].x;
l_offy = points[med].y - points[start].y;
if (! stroke_interpolatable (offx, offy, l_offx, l_offy, error))
{
bezier_stack_points_aux (points, start, med, error, rpnts);
bezier_stack_points_aux (points, med, end, error, rpnts);
return;
}
l_offx = points[end].x - points[med].x;
l_offy = points[end].y - points[med].y;
if (! stroke_interpolatable (offx, offy, l_offx, l_offy, error))
{
bezier_stack_points_aux (points, start, med, error, rpnts);
bezier_stack_points_aux (points, med, end, error, rpnts);
return;
}
/* Now, the curve can be represented by a points pair: (start, end).
So, add the last point to stroke_points. */
bezier_stack_points_aux (points, end, end, 0, rpnts);
}
static void
bezier_stack_points (GimpBezierSelectTool *bezier_sel,
GdkPoint *points,
gint n_points,
gpointer data)
{
BezierRenderPnts *rpnts = data;
gint i;
gint expand_size = 32;
gint minx, maxx, miny, maxy;
gdouble error;
if (n_points < 2) /* Does this happen? */
return;
if (rpnts->stroke_points == NULL) /* initialize it here */
{
rpnts->num_stroke_points = 0;
rpnts->len_stroke_points = expand_size;
rpnts->stroke_points = g_new (gdouble, 2 * rpnts->len_stroke_points);
}
maxx = minx = points[0].x;
maxy = miny = points[0].y;
for (i = 1; i < n_points; i++)
{
if (points[i].x < minx)
minx = points[i].x;
else if (maxx < points[i].x)
maxx = points[i].x;
if (points[i].y < miny)
miny = points[i].x;
else if (maxy < points[i].y)
maxy = points[i].y;
}
/* allow one pixel fluctuation */
error = 2.0 / MAX (maxx - minx, maxy - miny);
error = 0.0; /* ALT */
/* add the start point */
bezier_stack_points_aux (points, 0, 0, 0, rpnts);
/* divide segments recursively */
bezier_stack_points_aux (points, 0, n_points - 1, error, rpnts);
/* printf ("n_points: %d\n", n_points); */
}
static gint
bezier_gen_points (GimpBezierSelectTool *bezier_sel,
gint open_path,
BezierRenderPnts *rpnts)
{
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *next_curve;
BezierRenderPnts *next_rpnts = rpnts;
gint point_counts = 0;
/* stack points */
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (bezier_sel->num_points >= 4)
{
do
{
point_counts = count_points_on_curve (points);
if (point_counts < 4)
return TRUE;
do
{
bezier_draw_segment (bezier_sel, points,
SUBDIVIDE, AA_IMAGE_COORDS,
bezier_stack_points,
(gpointer) next_rpnts);
points = next_anchor (points, &next_curve);
}
while (points != start_pt && points);
start_pt = next_curve;
points = next_curve;
if (next_curve)
{
next_rpnts->next_curve = g_new0 (BezierRenderPnts, 1);
next_rpnts = next_rpnts->next_curve;
}
}
while (next_curve);
}
return TRUE;
}
void
bezier_stroke (GimpBezierSelectTool *bezier_sel,
GDisplay *gdisp,
gint subdivisions,
gint open_path)
{
Argument *return_vals;
gint nreturn_vals;
BezierRenderPnts *next_rpnts;
BezierRenderPnts *rpnts;
rpnts = g_new0 (BezierRenderPnts, 1);
/* Start an undo group */
undo_push_group_start (gdisp->gimage, PAINT_CORE_UNDO);
bezier_gen_points (bezier_sel,open_path,rpnts);
do
{
if (rpnts->stroke_points)
{
GimpDrawable *drawable;
gint offset_x, offset_y;
gdouble *ptr;
drawable = gimp_image_active_drawable (gdisp->gimage);
gimp_drawable_offsets (drawable, &offset_x, &offset_y);
ptr = rpnts->stroke_points;
while (ptr < rpnts->stroke_points + (rpnts->num_stroke_points * 2))
{
*ptr /= SUPERSAMPLE;
*ptr++ -= offset_x;
*ptr /= SUPERSAMPLE;
*ptr++ -= offset_y;
}
/* Stroke with the correct tool */
return_vals =
procedural_db_run_proc (tool_manager_active_get_PDB_string (),
&nreturn_vals,
PDB_DRAWABLE, gimp_drawable_get_ID (drawable),
PDB_INT32, (gint32) rpnts->num_stroke_points * 2,
PDB_FLOATARRAY, rpnts->stroke_points,
PDB_END);
if (return_vals && return_vals[0].value.pdb_int != PDB_SUCCESS)
g_message (_("Paintbrush operation failed."));
procedural_db_destroy_args (return_vals, nreturn_vals);
g_free (rpnts->stroke_points);
}
next_rpnts = rpnts->next_curve;
rpnts->stroke_points = NULL;
rpnts->len_stroke_points = rpnts->num_stroke_points = 0;
g_free(rpnts);
rpnts = next_rpnts;
}
while (rpnts);
/* End an undo group */
undo_push_group_end (gdisp->gimage);
gdisplays_flush ();
}
static void
bezier_draw_segment_for_distance (GimpBezierSelectTool *bezier_sel,
GimpBezierSelectPoint *points,
gint subdivisions,
BezierDistance *bdist)
{
BezierMatrix geometry;
BezierMatrix tmp1, tmp2;
BezierMatrix deltas;
gdouble x, dx, dx2, dx3;
gdouble y, dy, dy2, dy3;
gdouble d, d2, d3;
gint index;
gint i;
/* construct the geometry matrix from the segment */
/* assumes that a valid segment containing 4 points is passed in */
if (bdist->found)
return;
for (i = 0; i < 4; i++)
{
if (!points)
gimp_fatal_error ("bezier_draw_segment_for_distance(): Bad bezier segment");
geometry[i][0] = points->x;
geometry[i][1] = points->y;
geometry[i][2] = 0;
geometry[i][3] = 0;
points = points->next;
}
/* subdivide the curve n times */
/* n can be adjusted to give a finer or coarser curve */
d = 1.0 / subdivisions;
d2 = d * d;
d3 = d * d * d;
/* construct a temporary matrix for determining the forward diffencing deltas */
tmp2[0][0] = 0; tmp2[0][1] = 0; tmp2[0][2] = 0; tmp2[0][3] = 1;
tmp2[1][0] = d3; tmp2[1][1] = d2; tmp2[1][2] = d; tmp2[1][3] = 0;
tmp2[2][0] = 6*d3; tmp2[2][1] = 2*d2; tmp2[2][2] = 0; tmp2[2][3] = 0;
tmp2[3][0] = 6*d3; tmp2[3][1] = 0; tmp2[3][2] = 0; tmp2[3][3] = 0;
/* compose the basis and geometry matrices */
bezier_compose (basis, geometry, tmp1);
/* compose the above results to get the deltas matrix */
bezier_compose (tmp2, tmp1, deltas);
/* extract the x deltas */
x = deltas[0][0];
dx = deltas[1][0];
dx2 = deltas[2][0];
dx3 = deltas[3][0];
/* extract the y deltas */
y = deltas[0][1];
dy = deltas[1][1];
dy2 = deltas[2][1];
dy3 = deltas[3][1];
index = 1;
/* loop over the curve */
for (i = 0; i < subdivisions; i++)
{
/* increment the x values */
x += dx;
dx += dx2;
dx2 += dx3;
/* increment the y values */
y += dy;
dy += dy2;
dy2 += dy3;
/* g_print ("x = %g, y = %g\n",x,y); */
/* if this point is different than the last one...then draw it */
/* Note :
* It assumes the data is the place we want the
* floating version of the coords to be stuffed.
* These are needed when we calculate the gradient of the
* curve.
*/
if (!bdist->firstpnt)
{
gdouble rx = x;
gdouble ry = y;
gdouble dx = bdist->lastx - rx;
gdouble dy = bdist->lasty - ry;
bdist->curdist += sqrt ((dx * dx) + (dy * dy));
if(bdist->curdist >= bdist->dist)
{
*(bdist->x) = ROUND ((rx + dx / 2));
*(bdist->y) = ROUND ((ry + dy / 2));
if (dx == 0.0)
*(bdist->gradient) = G_MAXDOUBLE;
else
*(bdist->gradient) = dy / dx;
/* g_print ("found x = %d, y = %d\n",*(bdist->x),*(bdist->y)); */
bdist->found = TRUE;
break;
}
bdist->lastx = rx;
bdist->lasty = ry;
}
else
{
bdist->firstpnt = FALSE;
bdist->lastx = x;
bdist->lasty = y;
}
}
}
static void
bezier_draw_curve_for_distance (GimpBezierSelectTool *bezier_sel,
BezierDistance *distance)
{
GimpBezierSelectPoint *points;
GimpBezierSelectPoint *start_pt;
GimpBezierSelectPoint *next_curve;
points = bezier_sel->points;
start_pt = bezier_sel->points;
if (bezier_sel->num_points >= 4)
{
do
{
do
{
bezier_draw_segment_for_distance (bezier_sel, points,
SUBDIVIDE,
distance);
points = next_anchor (points, &next_curve);
}
while (points != start_pt && points);
start_pt = next_curve;
points = next_curve;
}
while (next_curve);
}
}
gint
bezier_distance_along (GimpBezierSelectTool *bezier_sel,
gint open_path,
gdouble dist,
gint *x,
gint *y,
gdouble *gradient)
{
/* Render the curve as points then walk along it... */
BezierDistance *bdist;
gint ret;
bdist = g_new0 (BezierDistance, 1);
bdist->firstpnt = TRUE;
bdist->curdist = 0.0;
bdist->lastx = 0.0;
bdist->lasty = 0.0;
bdist->dist = dist;
bdist->x = x;
bdist->y = y;
bdist->gradient = gradient;
bdist->found = FALSE;
bezier_draw_curve_for_distance (bezier_sel, bdist);
ret = bdist->found;
g_free (bdist);
return ret;
}