gegl r2799 - in trunk: . operations/external
- From: ok svn gnome org
- To: svn-commits-list gnome org
- Subject: gegl r2799 - in trunk: . operations/external
- Date: Mon, 24 Nov 2008 00:54:50 +0000 (UTC)
Author: ok
Date: Mon Nov 24 00:54:50 2008
New Revision: 2799
URL: http://svn.gnome.org/viewvc/gegl?rev=2799&view=rev
Log:
* operations/external/path.c: added new op that takes over the
roles of gegl:fill and gegl:stroke, it uses mostly the same names
for properties as the svg fill op. The op can be used both as a filter
and as an image source.
* operations/external/Makefile.am: added new op.
* operations/external/stroke.c: removed.
* operations/external/fill.c: removed.
Added:
trunk/operations/external/path.c
Removed:
trunk/operations/external/fill.c
trunk/operations/external/stroke.c
Modified:
trunk/ChangeLog
trunk/operations/external/Makefile.am
Modified: trunk/operations/external/Makefile.am
==============================================================================
--- trunk/operations/external/Makefile.am (original)
+++ trunk/operations/external/Makefile.am Mon Nov 24 00:54:50 2008
@@ -12,13 +12,10 @@
endif
if HAVE_CAIRO
-fill_la_SOURCES = fill.c
-fill_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
-fill_la_CFLAGS = $(CAIRO_CFLAGS)
-stroke_la_SOURCES = stroke.c
-stroke_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
-stroke_la_CFLAGS = $(CAIRO_CFLAGS)
-ops += fill.la stroke.la
+path_la_SOURCES = path.c
+path_la_LIBADD = $(op_libs) $(CAIRO_LIBS)
+path_la_CFLAGS = $(CAIRO_CFLAGS)
+ops += path.la
endif
if HAVE_PNG
Added: trunk/operations/external/path.c
==============================================================================
--- (empty file)
+++ trunk/operations/external/path.c Mon Nov 24 00:54:50 2008
@@ -0,0 +1,274 @@
+/* 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 (fill, _("Fill Color"), "black",
+ _("Color of paint to use for filling, use 0 opacity to disable filling."))
+gegl_chant_color (stroke, _("Stroke Color"), "rgba(0.0,0.0,0.0,0.0)",
+ _("Color of paint to use for stroking."))
+
+gegl_chant_double (stroke_width,_("Stroke width"), 0.0, 200.0, 2.0,
+ _("The width of the brush used to stroke the path."))
+
+gegl_chant_double (stroke_opacity, _("Stroke 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_double (stroke_hardness, _("Hardness"), 0.0, 1.0, 0.6,
+ _("hardness of brush, 0.0 for soft brush 1.0 for hard brush."))
+
+gegl_chant_string (fill_rule,_("Fill rule."), "nonzero",
+ _("how to determine what to fill (nonzero|evenodd"))
+
+gegl_chant_double (fill_opacity, _("Fill opacity"), -2.0, 2.0, 1.0,
+ _("The fill opacity to use."))
+
+gegl_chant_path (d, _("Vector"),
+ _("A GeglVector representing the path of the stroke"))
+gegl_chant_pointer (pad, "", "")
+
+#else
+
+#define GEGL_CHANT_TYPE_FILTER
+#define GEGL_CHANT_C_FILE "path.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/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->stroke_width/2;
+ rect.y -= o->stroke_width/2;
+ rect.width += o->stroke_width;
+ rect.height += o->stroke_width;
+
+ gegl_operation_invalidate (userdata, &rect, FALSE);
+};
+
+static void
+prepare (GeglOperation *operation)
+{
+ gegl_operation_set_format (operation, "output", babl_format ("RaGaBaA float"));
+}
+
+static GeglRectangle
+get_bounding_box (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ GeglRectangle defined = { 0, 0, 512, 512 };
+ gdouble x0, x1, y0, y1;
+
+ gegl_path_get_bounds (o->d, &x0, &x1, &y0, &y1);
+ defined.x = x0 - o->stroke_width/2;
+ defined.y = y0 - o->stroke_width/2;
+ defined.width = x1 - x0 + o->stroke_width;
+ defined.height = y1 - y0 + o->stroke_width;
+
+ return defined;
+}
+
+#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);
+ }
+
+ if (o->fill_opacity > 0.0001 && o->fill)
+ {
+ gfloat r,g,b,a;
+ gegl_color_get_rgba (o->fill, &r,&g,&b,&a);
+ a *= o->fill_opacity;
+ if (a>0.001)
+ {
+ cairo_t *cr;
+ cairo_surface_t *surface;
+ guchar *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);
+ gegl_buffer_linear_close (output, data);
+ }
+ }
+
+ g_object_set_data (G_OBJECT (operation), "path-radius", GINT_TO_POINTER((gint)(o->stroke_width+1)/2));
+
+ if (o->stroke_width > 0.1 && o->stroke_opacity > 0.0001)
+ {
+ gegl_path_stroke (output, result,
+ o->d,
+ o->stroke,
+ o->stroke_width,
+ o->stroke_hardness,
+ o->stroke_opacity);
+ }
+
+ 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->stroke_width);
+
+
+ if (o->stroke_width > 0.1 && o->stroke_opacity > 0.0001)
+ result = cairo_in_stroke (cr, x, y);
+
+ if (!result)
+ {
+ if (o->d)
+ {
+ gfloat r,g,b,a;
+ gegl_color_get_rgba (o->fill, &r,&g,&b,&a);
+ if (a>0.001)
+ 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;
+ GeglOperationSourceClass *source_class;
+
+ operation_class = GEGL_OPERATION_CLASS (klass);
+ source_class = GEGL_OPERATION_SOURCE_CLASS (klass);
+
+ source_class->process = process;
+ operation_class->get_bounding_box = get_bounding_box;
+ operation_class->prepare = prepare;
+ operation_class->detect = detect;
+
+ operation_class->name = "gegl:path";
+ operation_class->categories = "render";
+ operation_class->description = _("Renders a brush 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]