[gegl/paint-system: 5/7] Extract vector rendering operations from path.c
- From: Damien de Lemeny <ddelemeny src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gegl/paint-system: 5/7] Extract vector rendering operations from path.c
- Date: Tue, 20 Jul 2010 18:09:16 +0000 (UTC)
commit 89403971d75507477e70b2c72af6112f90848d87
Author: Damien de Lemeny <d delemeny gmail com>
Date: Thu Jun 10 15:26:17 2010 +0200
Extract vector rendering operations from path.c
* add vector-fill and vector-stroke operations (use cairo)
* update Makefile
operations/external/Makefile.am | 8 +-
operations/external/vector-fill.c | 292 +++++++++++++++++++++++++++++++++++
operations/external/vector-stroke.c | 286 ++++++++++++++++++++++++++++++++++
3 files changed, 585 insertions(+), 1 deletions(-)
---
diff --git a/operations/external/Makefile.am b/operations/external/Makefile.am
index b33746e..81d8956 100644
--- a/operations/external/Makefile.am
+++ b/operations/external/Makefile.am
@@ -12,10 +12,16 @@ text_la_CFLAGS = $(AM_CFLAGS) $(PANGOCAIRO_CFLAGS)
endif
if HAVE_CAIRO
-ops += path.la
+ops += path.la vector-fill.la vector-stroke.la
path_la_SOURCES = path.c
path_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
path_la_CFLAGS = $(AM_CFLAGS) $(CAIRO_CFLAGS)
+vector_fill_la_SOURCES = vector-fill.c
+vector_fill_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
+vector_fill_la_CFLAGS = $(AM_CFLAGS) $(CAIRO_CFLAGS)
+vector_stroke_la_SOURCES = vector-stroke.c
+vector_stroke_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
+vector_stroke_la_CFLAGS = $(AM_CFLAGS) $(CAIRO_CFLAGS)
endif
if HAVE_PNG
diff --git a/operations/external/vector-fill.c b/operations/external/vector-fill.c
new file mode 100644
index 0000000..725e046
--- /dev/null
+++ b/operations/external/vector-fill.c
@@ -0,0 +1,292 @@
+/* This file is an image processing operation for GEGL
+ *
+ * GEGL is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * GEGL 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with GEGL; if not, see <http://www.gnu.org/licenses/>.
+ *
+ * Copyright 2006 �yvind Kolås <pippin gimp org>
+ */
+
+
+#include "config.h"
+#include <glib/gi18n-lib.h>
+
+
+#ifdef GEGL_CHANT_PROPERTIES
+
+
+gegl_chant_color (color, _("Color"), "rgba(0.0,0.0,0.0,0.6)",
+ _("Color of paint to use for filling."))
+
+gegl_chant_double (opacity, _("Opacity"), -2.0, 2.0, 1.0,
+ _("The fill opacity to use."))
+
+gegl_chant_string (fill_rule,_("Fill rule."), "nonzero",
+ _("how to determine what to fill (nonzero|evenodd"))
+
+gegl_chant_string (transform,_("Transform"), "",
+ _("svg style description of transform."))
+
+gegl_chant_path (d, _("Vector"),
+ _("A GeglVector representing the path of the stroke"))
+
+#else
+
+#define GEGL_CHANT_TYPE_FILTER
+#define GEGL_CHANT_C_FILE "vector-fill.c"
+
+#include "gegl-plugin.h"
+
+/* the path api isn't public yet */
+#include "property-types/gegl-path.h"
+static void path_changed (GeglPath *path,
+ const GeglRectangle *roi,
+ gpointer userdata);
+
+#include "gegl-chant.h"
+#include <cairo.h>
+
+static void path_changed (GeglPath *path,
+ const GeglRectangle *roi,
+ gpointer userdata)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (userdata);
+ GeglRectangle rect;
+ gdouble x0, x1, y0, y1;
+
+ gegl_path_get_bounds(o->d, &x0, &x1, &y0, &y1);
+ rect.x = x0;
+ rect.y = y0;
+ rect.width = x1 - x0;
+ rect.height = y1 - y0;
+
+ gegl_operation_invalidate (userdata, roi, TRUE);
+};
+
+static void
+prepare (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gegl_operation_set_format (operation, "output", babl_format ("RaGaBaA float"));
+ if (o->transform && o->transform[0] != '\0')
+ {
+ GeglMatrix3 matrix;
+ gegl_matrix3_parse_string (matrix, o->transform);
+ gegl_path_set_matrix (o->d, matrix);
+ }
+}
+
+static GeglRectangle
+get_bounding_box (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ GeglRectangle defined = { 0, 0, 512, 512 };
+ GeglRectangle *in_rect;
+ gdouble x0, x1, y0, y1;
+
+ in_rect = gegl_operation_source_get_bounding_box (operation, "input");
+
+ gegl_path_get_bounds (o->d, &x0, &x1, &y0, &y1);
+ defined.x = x0;
+ defined.y = y0;
+ defined.width = x1 - x0;
+ defined.height = y1 - y0;
+
+ if (in_rect)
+ {
+ gegl_rectangle_bounding_box (&defined, &defined, in_rect);
+ }
+
+ return defined;
+}
+
+
+
+#if 0
+static gboolean gegl_path_is_closed (GeglPath *path)
+{
+ const GeglPathItem *knot;
+
+ if (!path)
+ return FALSE;
+ knot = gegl_path_get_node (path, -1);
+ if (!knot)
+ return FALSE;
+ if (knot->type == 'z')
+ {
+ return TRUE;
+ }
+ return FALSE;
+}
+#endif
+
+
+#if 0
+static GeglRectangle
+get_cached_region (GeglOperation *operation)
+{
+ return get_bounding_box (operation);
+}
+#endif
+static void gegl_path_cairo_play (GeglPath *path,
+ cairo_t *cr);
+
+static gboolean
+process (GeglOperation *operation,
+ GeglBuffer *input,
+ GeglBuffer *output,
+ const GeglRectangle *result)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gboolean need_fill = FALSE;
+ gdouble r,g,b,a;
+
+ if (input)
+ {
+ gegl_buffer_copy (input, result, output, result);
+ }
+ else
+ {
+ gegl_buffer_clear (output, result);
+ }
+
+
+ if (o->opacity > 0.0001 && o->color)
+ {
+ gegl_color_get_rgba (o->color, &r,&g,&b,&a);
+ a *= o->opacity;
+ if (a>0.001)
+ need_fill=TRUE;
+ }
+
+ if (need_fill)
+ {
+ GStaticMutex mutex = G_STATIC_MUTEX_INIT;
+ cairo_t *cr;
+ cairo_surface_t *surface;
+ guchar *data;
+
+
+ g_static_mutex_lock (&mutex);
+ data = (void*)gegl_buffer_linear_open (output, result, NULL, babl_format ("B'aG'aR'aA u8"));
+ surface = cairo_image_surface_create_for_data (data,
+ CAIRO_FORMAT_ARGB32,
+ result->width,
+ result->height,
+ result->width * 4);
+
+ cr = cairo_create (surface);
+ cairo_translate (cr, -result->x, -result->y);
+ if (g_str_equal (o->fill_rule, "evenodd"))
+ cairo_set_fill_rule (cr, CAIRO_FILL_RULE_EVEN_ODD);
+
+ gegl_path_cairo_play (o->d, cr);
+ cairo_set_source_rgba (cr, r,g,b,a);
+ cairo_fill (cr);
+ cairo_destroy (cr);
+
+ gegl_buffer_linear_close (output, data);
+ g_static_mutex_unlock (&mutex);
+ }
+ return TRUE;
+}
+
+static void foreach_cairo (const GeglPathItem *knot,
+ gpointer cr)
+{
+ switch (knot->type)
+ {
+ case 'M':
+ cairo_move_to (cr, knot->point[0].x, knot->point[0].y);
+ break;
+ case 'L':
+ cairo_line_to (cr, knot->point[0].x, knot->point[0].y);
+ break;
+ case 'C':
+ cairo_curve_to (cr, knot->point[0].x, knot->point[0].y,
+ knot->point[1].x, knot->point[1].y,
+ knot->point[2].x, knot->point[2].y);
+ break;
+ case 'z':
+ cairo_close_path (cr);
+ break;
+ default:
+ g_print ("%s uh?:%c\n", G_STRLOC, knot->type);
+ }
+}
+
+static void gegl_path_cairo_play (GeglPath *path,
+ cairo_t *cr)
+{
+ gegl_path_foreach_flat (path, foreach_cairo, cr);
+}
+
+static GeglNode *detect (GeglOperation *operation,
+ gint x,
+ gint y)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ cairo_t *cr;
+ cairo_surface_t *surface;
+ gchar *data = " ";
+ gboolean result = FALSE;
+
+ surface = cairo_image_surface_create_for_data ((guchar*)data,
+ CAIRO_FORMAT_ARGB32,
+ 1,1,4);
+ cr = cairo_create (surface);
+ gegl_path_cairo_play (o->d, cr);
+
+ if (!result)
+ {
+ if (o->d)
+ {
+ gdouble r,g,b,a;
+ gegl_color_get_rgba (o->color, &r,&g,&b,&a);
+ if (a * o->opacity>0.8)
+ result = cairo_in_fill (cr, x, y);
+ }
+ }
+
+ cairo_destroy (cr);
+
+ if (result)
+ return operation->node;
+
+ return NULL;
+}
+
+static void
+gegl_chant_class_init (GeglChantClass *klass)
+{
+ GeglOperationClass *operation_class;
+ GeglOperationFilterClass *filter_class;
+
+ operation_class = GEGL_OPERATION_CLASS (klass);
+ filter_class = GEGL_OPERATION_FILTER_CLASS (klass);
+
+ filter_class->process = process;
+ operation_class->get_bounding_box = get_bounding_box;
+ operation_class->prepare = prepare;
+ operation_class->detect = detect;
+ /*operation_class->no_cache = TRUE;*/
+
+ operation_class->name = "gegl:vector-fill";
+ operation_class->categories = "render";
+ operation_class->description = _("Renders a filled region");
+#if 0
+ operation_class->get_cached_region = (void*)get_cached_region;
+#endif
+}
+
+
+#endif
diff --git a/operations/external/vector-stroke.c b/operations/external/vector-stroke.c
new file mode 100644
index 0000000..7250dc3
--- /dev/null
+++ b/operations/external/vector-stroke.c
@@ -0,0 +1,286 @@
+/* This file is an image processing operation for GEGL
+ *
+ * GEGL is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 3 of the License, or (at your option) any later version.
+ *
+ * GEGL 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
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with GEGL; if not, see <http://www.gnu.org/licenses/>.
+ *
+ * Copyright 2006 �yvind Kolås <pippin gimp org>
+ */
+
+
+#include "config.h"
+#include <glib/gi18n-lib.h>
+
+
+#ifdef GEGL_CHANT_PROPERTIES
+
+
+gegl_chant_color (color, _("Color"), "rgba(0.0,0.0,0.0,0.0)",
+ _("Color of paint to use for stroking."))
+
+gegl_chant_double (width, _("Width"), 0.0, 200.0, 2.0,
+ _("The width of the brush used to stroke the path."))
+
+gegl_chant_double (opacity, _("Opacity"), -2.0, 2.0, 1.0,
+ _("Opacity of stroke, note, does not behave like SVG since at the moment stroking is done using an airbrush tool."))
+
+gegl_chant_string (transform,_("Transform"), "",
+ _("svg style description of transform."))
+
+gegl_chant_path (d, _("Vector"),
+ _("A GeglVector representing the path of the stroke"))
+
+#else
+
+#define GEGL_CHANT_TYPE_FILTER
+#define GEGL_CHANT_C_FILE "vector-stroke.c"
+
+#include "gegl-plugin.h"
+
+/* the path api isn't public yet */
+#include "property-types/gegl-path.h"
+static void path_changed (GeglPath *path,
+ const GeglRectangle *roi,
+ gpointer userdata);
+
+#include "gegl-chant.h"
+#include <cairo.h>
+
+static void path_changed (GeglPath *path,
+ const GeglRectangle *roi,
+ gpointer userdata)
+{
+ GeglRectangle rect = *roi;
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (userdata);
+ /* invalidate the incoming rectangle */
+
+ rect.x -= o->width/2;
+ rect.y -= o->width/2;
+ rect.width += o->width;
+ rect.height += o->width;
+
+ gegl_operation_invalidate (userdata, &rect, FALSE);
+};
+
+static void
+prepare (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gegl_operation_set_format (operation, "output", babl_format ("RaGaBaA float"));
+ if (o->transform && o->transform[0] != '\0')
+ {
+ GeglMatrix3 matrix;
+ gegl_matrix3_parse_string (matrix, o->transform);
+ gegl_path_set_matrix (o->d, matrix);
+ }
+}
+
+static GeglRectangle
+get_bounding_box (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ GeglRectangle defined = { 0, 0, 512, 512 };
+ GeglRectangle *in_rect;
+ gdouble x0, x1, y0, y1;
+
+ in_rect = gegl_operation_source_get_bounding_box (operation, "input");
+
+ gegl_path_get_bounds (o->d, &x0, &x1, &y0, &y1);
+ defined.x = x0 - o->width/2;
+ defined.y = y0 - o->width/2;
+ defined.width = x1 - x0 + o->width;
+ defined.height = y1 - y0 + o->width;
+
+ if (in_rect)
+ {
+ gegl_rectangle_bounding_box (&defined, &defined, in_rect);
+ }
+
+ return defined;
+}
+
+
+
+#if 0
+static gboolean gegl_path_is_closed (GeglPath *path)
+{
+ const GeglPathItem *knot;
+
+ if (!path)
+ return FALSE;
+ knot = gegl_path_get_node (path, -1);
+ if (!knot)
+ return FALSE;
+ if (knot->type == 'z')
+ {
+ return TRUE;
+ }
+ return FALSE;
+}
+#endif
+
+
+#if 0
+static GeglRectangle
+get_cached_region (GeglOperation *operation)
+{
+ return get_bounding_box (operation);
+}
+#endif
+static void gegl_path_cairo_play (GeglPath *path,
+ cairo_t *cr);
+
+static gboolean
+process (GeglOperation *operation,
+ GeglBuffer *input,
+ GeglBuffer *output,
+ const GeglRectangle *result)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+
+ if (input)
+ {
+ gegl_buffer_copy (input, result, output, result);
+ }
+ else
+ {
+ gegl_buffer_clear (output, result);
+ }
+
+ gboolean need_stroke = FALSE;
+ gdouble r,g,b,a;
+
+ if (o->width > 0.1 && o->opacity > 0.0001)
+ {
+ gegl_color_get_rgba (o->color, &r,&g,&b,&a);
+ a *= o->opacity;
+ if (a>0.001)
+ need_stroke=TRUE;
+ }
+
+ if (need_stroke)
+ {
+ GStaticMutex mutex = G_STATIC_MUTEX_INIT;
+ cairo_t *cr;
+ cairo_surface_t *surface;
+ guchar *data;
+
+ g_static_mutex_lock (&mutex);
+ data = (void*)gegl_buffer_linear_open (output, result, NULL, babl_format ("B'aG'aR'aA u8"));
+ surface = cairo_image_surface_create_for_data (data,
+ CAIRO_FORMAT_ARGB32,
+ result->width,
+ result->height,
+ result->width * 4);
+
+ cr = cairo_create (surface);
+
+ cairo_translate (cr, -result->x, -result->y);
+
+ cairo_set_line_width (cr, o->width);
+ cairo_set_line_cap (cr, CAIRO_LINE_CAP_ROUND);
+ cairo_set_line_join (cr, CAIRO_LINE_JOIN_ROUND);
+
+ gegl_path_cairo_play (o->d, cr);
+ cairo_set_source_rgba (cr, r,g,b,a);
+ cairo_stroke (cr);
+ cairo_destroy (cr);
+
+ gegl_buffer_linear_close (output, data);
+ g_static_mutex_unlock (&mutex);
+ }
+ return TRUE;
+}
+static void foreach_cairo (const GeglPathItem *knot,
+ gpointer cr)
+{
+ switch (knot->type)
+ {
+ case 'M':
+ cairo_move_to (cr, knot->point[0].x, knot->point[0].y);
+ break;
+ case 'L':
+ cairo_line_to (cr, knot->point[0].x, knot->point[0].y);
+ break;
+ case 'C':
+ cairo_curve_to (cr, knot->point[0].x, knot->point[0].y,
+ knot->point[1].x, knot->point[1].y,
+ knot->point[2].x, knot->point[2].y);
+ break;
+ case 'z':
+ cairo_close_path (cr);
+ break;
+ default:
+ g_print ("%s uh?:%c\n", G_STRLOC, knot->type);
+ }
+}
+
+static void gegl_path_cairo_play (GeglPath *path,
+ cairo_t *cr)
+{
+ gegl_path_foreach_flat (path, foreach_cairo, cr);
+}
+
+static GeglNode *detect (GeglOperation *operation,
+ gint x,
+ gint y)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ cairo_t *cr;
+ cairo_surface_t *surface;
+ gchar *data = " ";
+ gboolean result = FALSE;
+
+ surface = cairo_image_surface_create_for_data ((guchar*)data,
+ CAIRO_FORMAT_ARGB32,
+ 1,1,4);
+ cr = cairo_create (surface);
+ gegl_path_cairo_play (o->d, cr);
+ cairo_set_line_width (cr, o->width);
+
+
+ if (o->width > 0.1 && o->opacity > 0.0001)
+ result = cairo_in_stroke (cr, x, y);
+
+ cairo_destroy (cr);
+
+ if (result)
+ return operation->node;
+
+ return NULL;
+}
+
+static void
+gegl_chant_class_init (GeglChantClass *klass)
+{
+ GeglOperationClass *operation_class;
+ GeglOperationFilterClass *filter_class;
+
+ operation_class = GEGL_OPERATION_CLASS (klass);
+ filter_class = GEGL_OPERATION_FILTER_CLASS (klass);
+
+ filter_class->process = process;
+ operation_class->get_bounding_box = get_bounding_box;
+ operation_class->prepare = prepare;
+ operation_class->detect = detect;
+ /*operation_class->no_cache = TRUE;*/
+
+ operation_class->name = "gegl:vector-stroke";
+ operation_class->categories = "render";
+ operation_class->description = _("Renders a vector stroke");
+#if 0
+ operation_class->get_cached_region = (void*)get_cached_region;
+#endif
+}
+
+
+#endif
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]