- Make the various virtual methods of GimpPaintCore use a list of drawables as argument instead of a single drawable. - gimp_brush_core_eval_transform_dynamics() can work with an image as argument rather than a drawable as it doesn't actually depends on specific drawable data. - New function gimp_paint_tool_enable_multi_paint() to be used in init() method of paint tools to announce that this tool can work with multiple layers selected. - Use gimp_paint_tool_enable_multi_paint() in the GimpSourceTool base class only for now. This is a first step for multi-layer drawing, but we don't want it to be possible in just any random cases, which is why I add a special function to advertize this capability. We will use it for special-casing the clone (as well as heal and perspective tools most likely) tool to work on several layers at once. At this step, it is still very bugged and not really working properly. In particular, since we don't process the drawable offset early anymore (because it makes no sense when we pass a list of drawables with different offsets), I suspect that all the offset-related code will be very broken.
404 lines
12 KiB
C
404 lines
12 KiB
C
/* GIMP - The GNU Image Manipulation Program
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <gegl.h>
|
|
|
|
#include "libgimpmath/gimpmath.h"
|
|
|
|
#include "paint-types.h"
|
|
|
|
#include "core/gimpboundary.h"
|
|
#include "core/gimpdrawable.h"
|
|
#include "core/gimperror.h"
|
|
#include "core/gimpcoords.h"
|
|
|
|
#include "vectors/gimpstroke.h"
|
|
#include "vectors/gimpvectors.h"
|
|
|
|
#include "gimppaintcore.h"
|
|
#include "gimppaintcore-stroke.h"
|
|
#include "gimppaintoptions.h"
|
|
|
|
#include "gimp-intl.h"
|
|
|
|
|
|
static void gimp_paint_core_stroke_emulate_dynamics (GimpCoords *coords,
|
|
gint length);
|
|
|
|
|
|
static const GimpCoords default_coords = GIMP_COORDS_DEFAULT_VALUES;
|
|
|
|
|
|
gboolean
|
|
gimp_paint_core_stroke (GimpPaintCore *core,
|
|
GimpDrawable *drawable,
|
|
GimpPaintOptions *paint_options,
|
|
GimpCoords *strokes,
|
|
gint n_strokes,
|
|
gboolean push_undo,
|
|
GError **error)
|
|
{
|
|
GList *drawables;
|
|
gboolean success = FALSE;
|
|
|
|
g_return_val_if_fail (GIMP_IS_PAINT_CORE (core), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_DRAWABLE (drawable), FALSE);
|
|
g_return_val_if_fail (gimp_item_is_attached (GIMP_ITEM (drawable)), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_PAINT_OPTIONS (paint_options), FALSE);
|
|
g_return_val_if_fail (strokes != NULL, FALSE);
|
|
g_return_val_if_fail (n_strokes > 0, FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
drawables = g_list_prepend (NULL, drawable);
|
|
|
|
if (gimp_paint_core_start (core, drawables, paint_options, &strokes[0],
|
|
error))
|
|
{
|
|
gint i;
|
|
|
|
core->last_coords = strokes[0];
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_INIT, 0);
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_MOTION, 0);
|
|
|
|
for (i = 1; i < n_strokes; i++)
|
|
{
|
|
gimp_paint_core_interpolate (core, drawables, paint_options,
|
|
&strokes[i], 0);
|
|
}
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_FINISH, 0);
|
|
|
|
gimp_paint_core_finish (core, drawables, push_undo);
|
|
|
|
gimp_paint_core_cleanup (core);
|
|
|
|
success = TRUE;
|
|
}
|
|
|
|
g_list_free (drawables);
|
|
|
|
return success;
|
|
}
|
|
|
|
gboolean
|
|
gimp_paint_core_stroke_boundary (GimpPaintCore *core,
|
|
GimpDrawable *drawable,
|
|
GimpPaintOptions *paint_options,
|
|
gboolean emulate_dynamics,
|
|
const GimpBoundSeg *bound_segs,
|
|
gint n_bound_segs,
|
|
gint offset_x,
|
|
gint offset_y,
|
|
gboolean push_undo,
|
|
GError **error)
|
|
{
|
|
GList *drawables;
|
|
GimpBoundSeg *stroke_segs;
|
|
gint n_stroke_segs;
|
|
gint off_x;
|
|
gint off_y;
|
|
GimpCoords *coords;
|
|
gboolean initialized = FALSE;
|
|
gint n_coords;
|
|
gint seg;
|
|
gint s;
|
|
|
|
g_return_val_if_fail (GIMP_IS_PAINT_CORE (core), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_DRAWABLE (drawable), FALSE);
|
|
g_return_val_if_fail (gimp_item_is_attached (GIMP_ITEM (drawable)), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_PAINT_OPTIONS (paint_options), FALSE);
|
|
g_return_val_if_fail (bound_segs != NULL && n_bound_segs > 0, FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
stroke_segs = gimp_boundary_sort (bound_segs, n_bound_segs,
|
|
&n_stroke_segs);
|
|
|
|
if (n_stroke_segs == 0)
|
|
return TRUE;
|
|
|
|
gimp_item_get_offset (GIMP_ITEM (drawable), &off_x, &off_y);
|
|
|
|
off_x -= offset_x;
|
|
off_y -= offset_y;
|
|
|
|
coords = g_new0 (GimpCoords, n_bound_segs + 4);
|
|
|
|
seg = 0;
|
|
n_coords = 0;
|
|
|
|
/* we offset all coordinates by 0.5 to align the brush with the path */
|
|
|
|
coords[n_coords] = default_coords;
|
|
coords[n_coords].x = (gdouble) (stroke_segs[0].x1 - off_x + 0.5);
|
|
coords[n_coords].y = (gdouble) (stroke_segs[0].y1 - off_y + 0.5);
|
|
|
|
n_coords++;
|
|
|
|
drawables = g_list_prepend (NULL, drawable);
|
|
|
|
for (s = 0; s < n_stroke_segs; s++)
|
|
{
|
|
while (stroke_segs[seg].x1 != -1 ||
|
|
stroke_segs[seg].x2 != -1 ||
|
|
stroke_segs[seg].y1 != -1 ||
|
|
stroke_segs[seg].y2 != -1)
|
|
{
|
|
coords[n_coords] = default_coords;
|
|
coords[n_coords].x = (gdouble) (stroke_segs[seg].x1 - off_x + 0.5);
|
|
coords[n_coords].y = (gdouble) (stroke_segs[seg].y1 - off_y + 0.5);
|
|
|
|
n_coords++;
|
|
seg++;
|
|
}
|
|
|
|
/* Close the stroke points up */
|
|
coords[n_coords] = coords[0];
|
|
|
|
n_coords++;
|
|
|
|
if (emulate_dynamics)
|
|
gimp_paint_core_stroke_emulate_dynamics (coords, n_coords);
|
|
|
|
if (initialized ||
|
|
gimp_paint_core_start (core, drawables, paint_options, &coords[0],
|
|
error))
|
|
{
|
|
gint i;
|
|
|
|
initialized = TRUE;
|
|
|
|
core->cur_coords = coords[0];
|
|
core->last_coords = coords[0];
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_INIT, 0);
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_MOTION, 0);
|
|
|
|
for (i = 1; i < n_coords; i++)
|
|
{
|
|
gimp_paint_core_interpolate (core, drawables, paint_options,
|
|
&coords[i], 0);
|
|
}
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_FINISH, 0);
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
|
|
n_coords = 0;
|
|
seg++;
|
|
|
|
coords[n_coords] = default_coords;
|
|
coords[n_coords].x = (gdouble) (stroke_segs[seg].x1 - off_x + 0.5);
|
|
coords[n_coords].y = (gdouble) (stroke_segs[seg].y1 - off_y + 0.5);
|
|
|
|
n_coords++;
|
|
}
|
|
|
|
if (initialized)
|
|
{
|
|
gimp_paint_core_finish (core, drawables, push_undo);
|
|
|
|
gimp_paint_core_cleanup (core);
|
|
}
|
|
|
|
g_list_free (drawables);
|
|
g_free (coords);
|
|
g_free (stroke_segs);
|
|
|
|
return initialized;
|
|
}
|
|
|
|
gboolean
|
|
gimp_paint_core_stroke_vectors (GimpPaintCore *core,
|
|
GimpDrawable *drawable,
|
|
GimpPaintOptions *paint_options,
|
|
gboolean emulate_dynamics,
|
|
GimpVectors *vectors,
|
|
gboolean push_undo,
|
|
GError **error)
|
|
{
|
|
GList *drawables;
|
|
GList *stroke;
|
|
gboolean initialized = FALSE;
|
|
gboolean due_to_lack_of_points = FALSE;
|
|
gint off_x, off_y;
|
|
gint vectors_off_x, vectors_off_y;
|
|
|
|
g_return_val_if_fail (GIMP_IS_PAINT_CORE (core), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_DRAWABLE (drawable), FALSE);
|
|
g_return_val_if_fail (gimp_item_is_attached (GIMP_ITEM (drawable)), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_PAINT_OPTIONS (paint_options), FALSE);
|
|
g_return_val_if_fail (GIMP_IS_VECTORS (vectors), FALSE);
|
|
g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
|
|
|
|
gimp_item_get_offset (GIMP_ITEM (vectors), &vectors_off_x, &vectors_off_y);
|
|
gimp_item_get_offset (GIMP_ITEM (drawable), &off_x, &off_y);
|
|
|
|
off_x -= vectors_off_x;
|
|
off_y -= vectors_off_y;
|
|
|
|
drawables = g_list_prepend (NULL, drawable);
|
|
|
|
for (stroke = vectors->strokes->head;
|
|
stroke;
|
|
stroke = stroke->next)
|
|
{
|
|
GArray *coords;
|
|
gboolean closed;
|
|
|
|
coords = gimp_stroke_interpolate (GIMP_STROKE (stroke->data),
|
|
1.0, &closed);
|
|
|
|
if (coords && coords->len)
|
|
{
|
|
gint i;
|
|
|
|
for (i = 0; i < coords->len; i++)
|
|
{
|
|
g_array_index (coords, GimpCoords, i).x -= off_x;
|
|
g_array_index (coords, GimpCoords, i).y -= off_y;
|
|
}
|
|
|
|
if (emulate_dynamics)
|
|
gimp_paint_core_stroke_emulate_dynamics ((GimpCoords *) coords->data,
|
|
coords->len);
|
|
|
|
if (initialized ||
|
|
gimp_paint_core_start (core, drawables, paint_options,
|
|
&g_array_index (coords, GimpCoords, 0),
|
|
error))
|
|
{
|
|
initialized = TRUE;
|
|
|
|
core->cur_coords = g_array_index (coords, GimpCoords, 0);
|
|
core->last_coords = g_array_index (coords, GimpCoords, 0);
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_INIT, 0);
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_MOTION, 0);
|
|
|
|
for (i = 1; i < coords->len; i++)
|
|
{
|
|
gimp_paint_core_interpolate (core, drawables, paint_options,
|
|
&g_array_index (coords, GimpCoords, i),
|
|
0);
|
|
}
|
|
|
|
gimp_paint_core_paint (core, drawables, paint_options,
|
|
GIMP_PAINT_STATE_FINISH, 0);
|
|
}
|
|
else
|
|
{
|
|
if (coords)
|
|
g_array_free (coords, TRUE);
|
|
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
due_to_lack_of_points = TRUE;
|
|
}
|
|
|
|
if (coords)
|
|
g_array_free (coords, TRUE);
|
|
}
|
|
|
|
if (initialized)
|
|
{
|
|
gimp_paint_core_finish (core, drawables, push_undo);
|
|
|
|
gimp_paint_core_cleanup (core);
|
|
}
|
|
|
|
if (! initialized && due_to_lack_of_points && *error == NULL)
|
|
{
|
|
g_set_error_literal (error, GIMP_ERROR, GIMP_FAILED,
|
|
_("Not enough points to stroke"));
|
|
}
|
|
|
|
g_list_free (drawables);
|
|
|
|
return initialized;
|
|
}
|
|
|
|
static void
|
|
gimp_paint_core_stroke_emulate_dynamics (GimpCoords *coords,
|
|
gint length)
|
|
{
|
|
const gint ramp_length = length / 3;
|
|
|
|
/* Calculate and create pressure ramp parameters */
|
|
if (ramp_length > 0)
|
|
{
|
|
gdouble slope = 1.0 / (gdouble) (ramp_length);
|
|
gint i;
|
|
|
|
/* Calculate pressure start ramp */
|
|
for (i = 0; i < ramp_length; i++)
|
|
{
|
|
coords[i].pressure = i * slope;
|
|
}
|
|
|
|
/* Calculate pressure end ramp */
|
|
for (i = length - ramp_length; i < length; i++)
|
|
{
|
|
coords[i].pressure = 1.0 - (i - (length - ramp_length)) * slope;
|
|
}
|
|
}
|
|
|
|
/* Calculate and create velocity ramp parameters */
|
|
if (length > 0)
|
|
{
|
|
gdouble slope = 1.0 / length;
|
|
gint i;
|
|
|
|
/* Calculate velocity end ramp */
|
|
for (i = 0; i < length; i++)
|
|
{
|
|
coords[i].velocity = i * slope;
|
|
}
|
|
}
|
|
|
|
if (length > 1)
|
|
{
|
|
gint i;
|
|
/* Fill in direction */
|
|
for (i = 1; i < length; i++)
|
|
{
|
|
coords[i].direction = gimp_coords_direction (&coords[i-1], &coords[i]);
|
|
}
|
|
|
|
coords[0].direction = coords[1].direction;
|
|
}
|
|
}
|