[gimp] app: replace gegl:watershed-transform with custom algorithm.
- From: Jehan <jehanp src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gimp] app: replace gegl:watershed-transform with custom algorithm.
- Date: Thu, 22 Nov 2018 13:25:10 +0000 (UTC)
commit 3467acf096fab265d55e2882b91b870da5db975b
Author: Jehan <jehan girinstud io>
Date: Wed Nov 21 15:52:51 2018 +0100
app: replace gegl:watershed-transform with custom algorithm.
We don't really need to flow every line art pixel and this new
implementation is simpler (because we don't actually need over-featured
watershedding), and a lot lot faster, making the line art bucket fill
now very reactive.
For this, I am keeping the computed distance map, as well as local
thickness map around to be used when flooding the line art pixels
(basically I try to flood half the stroke thickness).
Note that there are still some issues with this new implementation as it
doesn't properly flood yet created (i.e. invisible) splines and
segments, and in particular the ones between 2 colored sections. I am
going to fix this next.
app/core/gimpchannel-select.c | 2 +-
app/core/gimpdrawable-bucket-fill.c | 9 +-
app/core/gimpdrawable-bucket-fill.h | 4 +
app/core/gimplineart.c | 68 ++++---
app/core/gimplineart.h | 32 +--
app/core/gimppickable-contiguous-region.c | 327 +++++++++++++++++++++++++-----
app/core/gimppickable-contiguous-region.h | 12 +-
app/pdb/drawable-edit-cmds.c | 2 +-
app/tools/gimpbucketfilltool.c | 31 ++-
app/tools/gimpfuzzyselecttool.c | 2 +-
pdb/groups/drawable_edit.pdb | 2 +-
11 files changed, 383 insertions(+), 108 deletions(-)
---
diff --git a/app/core/gimpchannel-select.c b/app/core/gimpchannel-select.c
index 892dbb203e..6abe01f960 100644
--- a/app/core/gimpchannel-select.c
+++ b/app/core/gimpchannel-select.c
@@ -516,7 +516,7 @@ gimp_channel_select_fuzzy (GimpChannel *channel,
pickable = GIMP_PICKABLE (drawable);
add_on = gimp_pickable_contiguous_region_by_seed (pickable,
- NULL,
+ NULL, NULL, NULL,
antialias,
threshold,
select_transparent,
diff --git a/app/core/gimpdrawable-bucket-fill.c b/app/core/gimpdrawable-bucket-fill.c
index 4aebb63535..7025018ff8 100644
--- a/app/core/gimpdrawable-bucket-fill.c
+++ b/app/core/gimpdrawable-bucket-fill.c
@@ -50,6 +50,8 @@
void
gimp_drawable_bucket_fill (GimpDrawable *drawable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
GimpFillOptions *options,
gboolean fill_transparent,
GimpSelectCriterion fill_criterion,
@@ -72,7 +74,8 @@ gimp_drawable_bucket_fill (GimpDrawable *drawable,
image = gimp_item_get_image (GIMP_ITEM (drawable));
gimp_set_busy (image->gimp);
- buffer = gimp_drawable_get_bucket_fill_buffer (drawable, line_art, options,
+ buffer = gimp_drawable_get_bucket_fill_buffer (drawable, line_art,
+ distmap, thickmap, options,
fill_transparent, fill_criterion,
threshold, sample_merged,
diagonal_neighbors,
@@ -134,6 +137,8 @@ gimp_drawable_bucket_fill (GimpDrawable *drawable,
GeglBuffer *
gimp_drawable_get_bucket_fill_buffer (GimpDrawable *drawable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
GimpFillOptions *options,
gboolean fill_transparent,
GimpSelectCriterion fill_criterion,
@@ -196,7 +201,7 @@ gimp_drawable_get_bucket_fill_buffer (GimpDrawable *drawable,
* contiguous region.
*/
new_mask = gimp_pickable_contiguous_region_by_seed (pickable,
- line_art,
+ line_art, distmap, thickmap,
antialias,
threshold,
fill_transparent,
diff --git a/app/core/gimpdrawable-bucket-fill.h b/app/core/gimpdrawable-bucket-fill.h
index 9cb566f699..7b6cae9888 100644
--- a/app/core/gimpdrawable-bucket-fill.h
+++ b/app/core/gimpdrawable-bucket-fill.h
@@ -21,6 +21,8 @@
void gimp_drawable_bucket_fill (GimpDrawable *drawable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
GimpFillOptions *options,
gboolean fill_transparent,
GimpSelectCriterion fill_criterion,
@@ -32,6 +34,8 @@ void gimp_drawable_bucket_fill (GimpDrawable *drawabl
gdouble y);
GeglBuffer * gimp_drawable_get_bucket_fill_buffer (GimpDrawable *drawable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
GimpFillOptions *options,
gboolean fill_transparent,
GimpSelectCriterion fill_criterion,
diff --git a/app/core/gimplineart.c b/app/core/gimplineart.c
index 640cb9f7b1..c8373eb87c 100644
--- a/app/core/gimplineart.c
+++ b/app/core/gimplineart.c
@@ -105,7 +105,8 @@ static GArray * gimp_lineart_line_segment_until_hit (const GeglBuffer
Pixel start,
GimpVector2 direction,
int size);
-static gfloat * gimp_lineart_estimate_strokes_radii (GeglBuffer *mask);
+static gfloat * gimp_lineart_estimate_strokes_radii (GeglBuffer *mask,
+ gfloat **lineart_distmap);
/* Some callback-type functions. */
@@ -163,7 +164,7 @@ static void gimp_edgelset_next8 (const GeglBuffer *buffer,
/**
* gimp_lineart_close:
- * @line_art: the input #GeglBuffer.
+ * @buffer: the input #GeglBuffer.
* @select_transparent: whether we binarize the alpha channel or the
* luminosity.
* @stroke_threshold: [0-1] threshold value for detecting stroke pixels
@@ -187,8 +188,10 @@ static void gimp_edgelset_next8 (const GeglBuffer *buffer,
* @segments_max_length: the maximum length for creating segments
* between end points. Unlike splines, segments
* are straight lines.
+ * @lineart_distmap: a distance map of the line art pixels.
+ * @lineart_radii: a map of estimated radii of line art border pixels.
*
- * Creates a binarized version of the strokes of @line_art, detected either
+ * Creates a binarized version of the strokes of @buffer, detected either
* with luminosity (light means background) or alpha values depending on
* @select_transparent. This binary version of the strokes will have closed
* regions allowing adequate selection of "nearly closed regions".
@@ -201,24 +204,28 @@ static void gimp_edgelset_next8 (const GeglBuffer *buffer,
* Fourey, David Tschumperlé, David Revoy.
*
* Returns: a new #GeglBuffer of format "Y u8" representing the
- * binarized @line_art. A value of
+ * binarized @line_art. If @lineart_radii and @lineart_distmap
+ * are not #NULL, newly allocated float buffer are returned,
+ * which can be used for overflowing created masks later.
*/
GeglBuffer *
-gimp_lineart_close (GeglBuffer *line_art,
- gboolean select_transparent,
- gfloat stroke_threshold,
- gint minimal_lineart_area,
- gint normal_estimate_mask_size,
- gfloat end_point_rate,
- gint spline_max_length,
- gfloat spline_max_angle,
- gint end_point_connectivity,
- gfloat spline_roundness,
- gboolean allow_self_intersections,
- gint created_regions_significant_area,
- gint created_regions_minimum_area,
- gboolean small_segments_from_spline_sources,
- gint segments_max_length)
+gimp_lineart_close (GeglBuffer *buffer,
+ gboolean select_transparent,
+ gfloat stroke_threshold,
+ gint minimal_lineart_area,
+ gint normal_estimate_mask_size,
+ gfloat end_point_rate,
+ gint spline_max_length,
+ gfloat spline_max_angle,
+ gint end_point_connectivity,
+ gfloat spline_roundness,
+ gboolean allow_self_intersections,
+ gint created_regions_significant_area,
+ gint created_regions_minimum_area,
+ gboolean small_segments_from_spline_sources,
+ gint segments_max_length,
+ gfloat **lineart_distmap,
+ gfloat **lineart_radii)
{
const Babl *gray_format;
gfloat *normals;
@@ -236,8 +243,8 @@ gimp_lineart_close (GeglBuffer *line_art,
guchar max_value = 0;
gfloat threshold;
gfloat clamped_threshold;
- gint width = gegl_buffer_get_width (line_art);
- gint height = gegl_buffer_get_height (line_art);
+ gint width = gegl_buffer_get_width (buffer);
+ gint height = gegl_buffer_get_height (buffer);
gint i;
normals = g_new0 (gfloat, width * height * 2);
@@ -252,9 +259,9 @@ gimp_lineart_close (GeglBuffer *line_art,
gray_format = babl_format ("Y' u8");
/* Transform the line art from any format to gray. */
- strokes = gegl_buffer_new (gegl_buffer_get_extent (line_art),
+ strokes = gegl_buffer_new (gegl_buffer_get_extent (buffer),
gray_format);
- gegl_buffer_copy (line_art, NULL, GEGL_ABYSS_NONE, strokes, NULL);
+ gegl_buffer_copy (buffer, NULL, GEGL_ABYSS_NONE, strokes, NULL);
gegl_buffer_set_format (strokes, babl_format ("Y' u8"));
if (! select_transparent)
@@ -306,7 +313,7 @@ gimp_lineart_close (GeglBuffer *line_art,
smoothed_curvatures,
normal_estimate_mask_size);
- radii = gimp_lineart_estimate_strokes_radii (strokes);
+ radii = gimp_lineart_estimate_strokes_radii (strokes, lineart_distmap);
threshold = 1.0f - end_point_rate;
clamped_threshold = MAX (0.25f, threshold);
for (i = 0; i < width; i++)
@@ -321,7 +328,10 @@ gimp_lineart_close (GeglBuffer *line_art,
curvatures[i + j * width] = 0.0;
}
}
- g_free (radii);
+ if (lineart_radii)
+ *lineart_radii = radii;
+ else
+ g_free (radii);
keypoints = gimp_lineart_curvature_extremums (curvatures, smoothed_curvatures,
width, height);
@@ -1266,7 +1276,8 @@ gimp_lineart_line_segment_until_hit (const GeglBuffer *mask,
}
static gfloat *
-gimp_lineart_estimate_strokes_radii (GeglBuffer *mask)
+gimp_lineart_estimate_strokes_radii (GeglBuffer *mask,
+ gfloat **lineart_distmap)
{
GeglBufferIterator *gi;
gfloat *dist;
@@ -1411,7 +1422,10 @@ gimp_lineart_estimate_strokes_radii (GeglBuffer *mask)
}
}
- g_free (dist);
+ if (lineart_distmap)
+ *lineart_distmap = dist;
+ else
+ g_free (dist);
g_object_unref (distmap);
return thickness;
diff --git a/app/core/gimplineart.h b/app/core/gimplineart.h
index 54ee9e546b..8fa43f78da 100644
--- a/app/core/gimplineart.h
+++ b/app/core/gimplineart.h
@@ -22,21 +22,23 @@
#define __GIMP_LINEART__
-GeglBuffer * gimp_lineart_close (GeglBuffer *line_art,
- gboolean select_transparent,
- gfloat stroke_threshold,
- gint minimal_lineart_area,
- gint normal_estimate_mask_size,
- gfloat end_point_rate,
- gint spline_max_length,
- gfloat spline_max_angle,
- gint end_point_connectivity,
- gfloat spline_roundness,
- gboolean allow_self_intersections,
- gint created_regions_significant_area,
- gint created_regions_minimum_area,
- gboolean small_segments_from_spline_sources,
- gint segments_max_length);
+GeglBuffer * gimp_lineart_close (GeglBuffer *buffer,
+ gboolean select_transparent,
+ gfloat stroke_threshold,
+ gint minimal_lineart_area,
+ gint normal_estimate_mask_size,
+ gfloat end_point_rate,
+ gint spline_max_length,
+ gfloat spline_max_angle,
+ gint end_point_connectivity,
+ gfloat spline_roundness,
+ gboolean allow_self_intersections,
+ gint created_regions_significant_area,
+ gint created_regions_minimum_area,
+ gboolean small_segments_from_spline_sources,
+ gint segments_max_length,
+ gfloat **lineart_distmap,
+ gfloat **lineart_radii);
#endif /* __GIMP_LINEART__ */
diff --git a/app/core/gimppickable-contiguous-region.c b/app/core/gimppickable-contiguous-region.c
index 9149643c8f..396758e03a 100644
--- a/app/core/gimppickable-contiguous-region.c
+++ b/app/core/gimppickable-contiguous-region.c
@@ -46,6 +46,13 @@ typedef struct
gfloat stroke_threshold;
} LineArtData;
+typedef struct
+{
+ gint x;
+ gint y;
+ gfloat dist;
+} BorderPixel;
+
/* local function prototypes */
@@ -106,10 +113,20 @@ static void find_contiguous_region (GeglBuffer *src_buffer,
gint y,
const gfloat *col);
-static LineArtData * line_art_data_new (GeglBuffer *buffer,
- gboolean select_transparent,
- gfloat stroke_threshold);
-static void line_art_data_free (LineArtData *data);
+static LineArtData * line_art_data_new (GeglBuffer *buffer,
+ gboolean select_transparent,
+ gfloat stroke_threshold);
+static void line_art_data_free (LineArtData *data);
+static GimpPickableLineArtAsyncResult *
+ line_art_result_new (GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap);
+static void line_art_result_free (GimpPickableLineArtAsyncResult
+ *data);
+static void line_art_queue_pixel (GQueue *queue,
+ gint x,
+ gint y,
+ gfloat dist);
/* public functions */
@@ -119,6 +136,8 @@ gimp_pickable_contiguous_region_prepare_line_art_async_func (GimpAsync *async,
LineArtData *data)
{
GeglBuffer *lineart;
+ gfloat *distmap;
+ gfloat *thickmap;
gboolean has_alpha;
gboolean select_transparent = FALSE;
@@ -201,23 +220,29 @@ gimp_pickable_contiguous_region_prepare_line_art_async_func (GimpAsync *async,
/*small_segments_from_spline_sources,*/
TRUE,
/*segments_max_length*/
- 20);
+ 20,
+ &distmap, &thickmap);
GIMP_TIMER_END("close line-art");
- gimp_async_finish_full (async, lineart, g_object_unref);
+ gimp_async_finish_full (async,
+ line_art_result_new (lineart, distmap, thickmap),
+ (GDestroyNotify) line_art_result_free);
line_art_data_free (data);
}
GeglBuffer *
-gimp_pickable_contiguous_region_prepare_line_art (GimpPickable *pickable,
- gboolean select_transparent,
- gfloat stroke_threshold)
+gimp_pickable_contiguous_region_prepare_line_art (GimpPickable *pickable,
+ gboolean select_transparent,
+ gfloat stroke_threshold,
+ gfloat **distmap,
+ gfloat **thickmap)
{
- GimpAsync *async;
- LineArtData *data;
- GeglBuffer *lineart;
+ GimpAsync *async;
+ LineArtData *data;
+ GimpPickableLineArtAsyncResult *result;
+ GeglBuffer *lineart;
g_return_val_if_fail (GIMP_IS_PICKABLE (pickable), NULL);
@@ -230,7 +255,13 @@ gimp_pickable_contiguous_region_prepare_line_art (GimpPickable *pickable,
gimp_pickable_contiguous_region_prepare_line_art_async_func (async, data);
- lineart = g_object_ref (gimp_async_get_result (async));
+ result = gimp_async_get_result (async);
+
+ lineart = g_object_ref (result->line_art);
+ *distmap = result->distmap;
+ *thickmap = result->thickmap;
+ result->distmap = NULL;
+ result->thickmap = NULL;
g_object_unref (async);
@@ -272,6 +303,8 @@ gimp_pickable_contiguous_region_prepare_line_art_async (GimpPickable *pickable,
GeglBuffer *
gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
gboolean antialias,
gfloat threshold,
gboolean select_transparent,
@@ -288,7 +321,6 @@ gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
gint n_components;
gboolean has_alpha;
gfloat start_col[MAX_CHANNELS];
- gfloat flag = 2.0;
gboolean smart_line_art = FALSE;
gboolean free_line_art = FALSE;
@@ -296,6 +328,9 @@ gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
if (select_criterion == GIMP_SELECT_CRITERION_LINE_ART)
{
+ g_return_val_if_fail ((line_art && distmap && thickmap) ||
+ (! line_art && ! distmap && ! thickmap),
+ NULL);
if (line_art == NULL)
{
/* It is much better experience to pre-compute the line art,
@@ -303,7 +338,8 @@ gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
* selecting/filling through a PDB call).
*/
line_art = gimp_pickable_contiguous_region_prepare_line_art (pickable, select_transparent,
- stroke_threshold);
+ stroke_threshold,
+ &distmap, &thickmap);
free_line_art = TRUE;
}
@@ -382,59 +418,198 @@ gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
/* The last step of the line art algorithm is to make sure that
* selections does not leave "holes" between its borders and the
* line arts, while not stepping over as well.
- * To achieve this, I label differently the selection and the rest
- * and leave the stroke pixels unlabelled, then I let these
- * unknown pixels be labelled by flooding through watershed.
+ * I used to run the "gegl:watershed-transform" operation to flood
+ * the stroke pixels, but for such simple need, this simple code
+ * is so much faster while producing better results.
*/
- GeglBufferIterator *gi;
- GeglNode *graph;
- GeglNode *input;
- GeglNode *op;
+ gfloat *mask;
+ GQueue *queue = g_queue_new ();
+ gint width = gegl_buffer_get_width (line_art);
+ gint height = gegl_buffer_get_height (line_art);
+ gint nx, ny;
GIMP_TIMER_START();
- /* Flag the unselected line art pixels. */
- gi = gegl_buffer_iterator_new (src_buffer, NULL, 0, NULL,
- GEGL_ACCESS_READ, GEGL_ABYSS_NONE, 2);
- gegl_buffer_iterator_add (gi, mask_buffer, NULL, 0,
- babl_format ("Y float"),
- GEGL_ACCESS_READWRITE, GEGL_ABYSS_NONE);
-
- while (gegl_buffer_iterator_next (gi))
+ mask = g_new (gfloat, width * height);
+ gegl_buffer_get (mask_buffer, NULL, 1.0, NULL,
+ mask, GEGL_AUTO_ROWSTRIDE, GEGL_ABYSS_NONE);
+
+ for (y = 0; y < height; y++)
+ for (x = 0; x < width; x++)
+ {
+ gfloat thickness = thickmap[x + y * width];
+
+ if (thickness > 0.0)
+ {
+ if (x > 0)
+ {
+ nx = x - 1;
+ if (y > 0)
+ {
+ ny = y - 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ ny = y;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ if (y < height - 1)
+ {
+ ny = y + 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ }
+ if (x < width - 1)
+ {
+ nx = x + 1;
+ if (y > 0)
+ {
+ ny = y - 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ ny = y;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ if (y < height - 1)
+ {
+ ny = y + 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ }
+ nx = x;
+ if (y > 0)
+ {
+ ny = y - 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ if (y < height - 1)
+ {
+ ny = y + 1;
+ if (mask[nx + ny * width] != 0.0)
+ {
+ line_art_queue_pixel (queue, x, y, thickness);
+ continue;
+ }
+ }
+ }
+ }
+
+ while (! g_queue_is_empty (queue))
{
- guchar *lineart = (guchar*) gi->items[0].data;
- gfloat *mask = (gfloat*) gi->items[1].data;
- gint k;
+ BorderPixel *c = g_queue_pop_head (queue);
- for (k = 0; k < gi->length; k++)
+ if (mask[c->x + c->y * width] != 1.0)
{
- if (*lineart && ! *mask)
- *mask = flag;
- lineart++;
- mask++;
+ mask[c->x + c->y * width] = 1.0;
+ if (c->x > 0)
+ {
+ nx = c->x - 1;
+ if (c->y > 0)
+ {
+ ny = c->y - 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
+ ny = c->y;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ if (c->y < height - 1)
+ {
+ ny = c->y - 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
+ }
+ if (c->x < width - 1)
+ {
+ nx = c->x + 1;
+ if (c->y > 0)
+ {
+ ny = c->y - 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
+ ny = c->y;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ if (c->y < height - 1)
+ {
+ ny = c->y - 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
+ }
+ nx = c->x;
+ if (c->y > 0)
+ {
+ ny = c->y - 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
+ if (c->y < height - 1)
+ {
+ ny = c->y + 1;
+ if (mask[nx + ny * width] == 0.0 &&
+ distmap[nx + ny * width] > distmap[c->x + c->y * width] &&
+ distmap[nx + ny * width] < c->dist)
+ line_art_queue_pixel (queue, nx, ny, c->dist);
+ }
}
+ g_free (c);
}
-
- /* Watershed the line art. */
- graph = gegl_node_new ();
- input = gegl_node_new_child (graph,
- "operation", "gegl:buffer-source",
- "buffer", mask_buffer,
- NULL);
- op = gegl_node_new_child (graph,
- "operation", "gegl:watershed-transform",
- "flag-component", 0,
- "flag", &flag,
- NULL);
- gegl_node_connect_to (input, "output",
- op, "input");
- gegl_node_blit_buffer (op, mask_buffer, NULL, 0, GEGL_ABYSS_NONE);
- g_object_unref (graph);
+ g_queue_free (queue);
+ gegl_buffer_set (mask_buffer, gegl_buffer_get_extent (mask_buffer),
+ 0, NULL, mask, GEGL_AUTO_ROWSTRIDE);
+ g_free (mask);
GIMP_TIMER_END("watershed line art");
+
+ if (free_line_art)
+ {
+ g_object_unref (src_buffer);
+ g_free (distmap);
+ g_free (thickmap);
+ }
}
- if (free_line_art)
- g_object_unref (src_buffer);
return mask_buffer;
}
@@ -1001,3 +1176,43 @@ line_art_data_free (LineArtData *data)
g_slice_free (LineArtData, data);
}
+
+static GimpPickableLineArtAsyncResult *
+line_art_result_new (GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap)
+{
+ GimpPickableLineArtAsyncResult *data;
+
+ data = g_slice_new (GimpPickableLineArtAsyncResult);
+ data->line_art = line_art;
+ data->distmap = distmap;
+ data->thickmap = thickmap;
+
+ return data;
+}
+
+static void
+line_art_result_free (GimpPickableLineArtAsyncResult *data)
+{
+ g_object_unref (data->line_art);
+ g_clear_pointer (&data->distmap, g_free);
+ g_clear_pointer (&data->thickmap, g_free);
+
+ g_slice_free (GimpPickableLineArtAsyncResult, data);
+}
+
+static void
+line_art_queue_pixel (GQueue *queue,
+ gint x,
+ gint y,
+ gfloat dist)
+{
+ BorderPixel *p = g_new (BorderPixel, 1);
+
+ p->x = x;
+ p->y = y;
+ p->dist = dist;
+
+ g_queue_push_head (queue, p);
+}
diff --git a/app/core/gimppickable-contiguous-region.h b/app/core/gimppickable-contiguous-region.h
index 95543a59e5..8c2f61e82d 100644
--- a/app/core/gimppickable-contiguous-region.h
+++ b/app/core/gimppickable-contiguous-region.h
@@ -18,10 +18,18 @@
#ifndef __GIMP_PICKABLE_CONTIGUOUS_REGION_H__
#define __GIMP_PICKABLE_CONTIGUOUS_REGION_H__
+typedef struct
+{
+ GeglBuffer *line_art;
+ gfloat *distmap;
+ gfloat *thickmap;
+} GimpPickableLineArtAsyncResult;
GeglBuffer * gimp_pickable_contiguous_region_prepare_line_art (GimpPickable *pickable,
gboolean select_transparent,
- gfloat stroke_threshold);
+ gfloat stroke_threshold,
+ gfloat **distmap,
+ gfloat **thickmap);
GimpAsync * gimp_pickable_contiguous_region_prepare_line_art_async (GimpPickable *pickable,
gboolean select_transparent,
gfloat stroke_threshold,
@@ -29,6 +37,8 @@ GimpAsync * gimp_pickable_contiguous_region_prepare_line_art_async (GimpPickabl
GeglBuffer * gimp_pickable_contiguous_region_by_seed (GimpPickable *pickable,
GeglBuffer *line_art,
+ gfloat *distmap,
+ gfloat *thickmap,
gboolean antialias,
gfloat threshold,
gboolean select_transparent,
diff --git a/app/pdb/drawable-edit-cmds.c b/app/pdb/drawable-edit-cmds.c
index 3c5e307bd7..5a88a1deb4 100644
--- a/app/pdb/drawable-edit-cmds.c
+++ b/app/pdb/drawable-edit-cmds.c
@@ -165,7 +165,7 @@ drawable_edit_bucket_fill_invoker (GimpProcedure *procedure,
if (gimp_fill_options_set_by_fill_type (options, context,
fill_type, error))
{
- gimp_drawable_bucket_fill (drawable, NULL, options,
+ gimp_drawable_bucket_fill (drawable, NULL, NULL, NULL, options,
GIMP_PDB_CONTEXT (context)->sample_transparent,
GIMP_PDB_CONTEXT (context)->sample_criterion,
GIMP_PDB_CONTEXT (context)->sample_threshold,
diff --git a/app/tools/gimpbucketfilltool.c b/app/tools/gimpbucketfilltool.c
index 413f871cbc..b46da842b5 100644
--- a/app/tools/gimpbucketfilltool.c
+++ b/app/tools/gimpbucketfilltool.c
@@ -61,6 +61,8 @@ struct _GimpBucketFillToolPrivate
{
GimpAsync *async;
GeglBuffer *line_art;
+ gfloat *distmap;
+ gfloat *thickmap;
GWeakRef cached_image;
GWeakRef cached_drawable;
@@ -243,6 +245,8 @@ gimp_bucket_fill_tool_finalize (GObject *object)
}
g_clear_object (&tool->priv->line_art);
+ g_clear_pointer (&tool->priv->distmap, g_free);
+ g_clear_pointer (&tool->priv->thickmap, g_free);
if (image)
{
@@ -363,6 +367,8 @@ gimp_bucket_fill_tool_preview (GimpBucketFillTool *tool,
{
GeglBuffer *fill = NULL;
GeglBuffer *line_art = NULL;
+ gfloat *distmap = NULL;
+ gfloat *thickmap = NULL;
gdouble x = coords->x;
gdouble y = coords->y;
@@ -377,10 +383,14 @@ gimp_bucket_fill_tool_preview (GimpBucketFillTool *tool,
}
if (options->fill_criterion == GIMP_SELECT_CRITERION_LINE_ART)
- line_art = g_object_ref (tool->priv->line_art);
+ {
+ line_art = g_object_ref (tool->priv->line_art);
+ distmap = tool->priv->distmap;
+ thickmap = tool->priv->thickmap;
+ }
fill = gimp_drawable_get_bucket_fill_buffer (drawable,
- line_art,
+ line_art, distmap, thickmap,
fill_options,
options->fill_transparent,
options->fill_criterion,
@@ -696,7 +706,17 @@ gimp_bucket_fill_compute_line_art_cb (GimpAsync *async,
return;
if (gimp_async_is_finished (async))
- tool->priv->line_art = g_object_ref (gimp_async_get_result (async));
+ {
+ GimpPickableLineArtAsyncResult *result;
+
+ result = gimp_async_get_result (async);
+
+ tool->priv->line_art = g_object_ref (result->line_art);
+ tool->priv->distmap = result->distmap;
+ tool->priv->thickmap = result->thickmap;
+ result->distmap = NULL;
+ result->thickmap = NULL;
+ }
g_clear_object (&tool->priv->async);
}
@@ -719,6 +739,9 @@ gimp_bucket_fill_compute_line_art (GimpBucketFillTool *tool)
}
g_clear_object (&tool->priv->line_art);
+ g_clear_pointer (&tool->priv->distmap, g_free);
+ g_clear_pointer (&tool->priv->thickmap, g_free);
+
if (options->fill_criterion == GIMP_SELECT_CRITERION_LINE_ART)
{
GimpPickable *pickable = NULL;
@@ -822,6 +845,8 @@ gimp_bucket_fill_tool_image_changed (GimpContext *context,
GimpImage *prev_drawable = g_weak_ref_get (&tool->priv->cached_drawable);
g_clear_object (&tool->priv->line_art);
+ g_clear_pointer (&tool->priv->distmap, g_free);
+ g_clear_pointer (&tool->priv->thickmap, g_free);
if (prev_image)
{
diff --git a/app/tools/gimpfuzzyselecttool.c b/app/tools/gimpfuzzyselecttool.c
index 2f43b406c5..e28dc71f47 100644
--- a/app/tools/gimpfuzzyselecttool.c
+++ b/app/tools/gimpfuzzyselecttool.c
@@ -122,7 +122,7 @@ gimp_fuzzy_select_tool_get_mask (GimpRegionSelectTool *region_select,
pickable = GIMP_PICKABLE (image);
}
- return gimp_pickable_contiguous_region_by_seed (pickable, NULL,
+ return gimp_pickable_contiguous_region_by_seed (pickable, NULL, NULL, NULL,
sel_options->antialias,
options->threshold / 255.0,
options->select_transparent,
diff --git a/pdb/groups/drawable_edit.pdb b/pdb/groups/drawable_edit.pdb
index 9687d9989f..193a735ec0 100644
--- a/pdb/groups/drawable_edit.pdb
+++ b/pdb/groups/drawable_edit.pdb
@@ -169,7 +169,7 @@ HELP
if (gimp_fill_options_set_by_fill_type (options, context,
fill_type, error))
{
- gimp_drawable_bucket_fill (drawable, NULL, options,
+ gimp_drawable_bucket_fill (drawable, NULL, NULL, NULL, options,
GIMP_PDB_CONTEXT (context)->sample_transparent,
GIMP_PDB_CONTEXT (context)->sample_criterion,
GIMP_PDB_CONTEXT (context)->sample_threshold,
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]