[gegl/paint-system: 7/7] Add brush-dab operations
- From: Damien de Lemeny <ddelemeny src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gegl/paint-system: 7/7] Add brush-dab operations
- Date: Tue, 20 Jul 2010 18:09:26 +0000 (UTC)
commit 0b56a00d3db1b7f6f78752b6f359536531a0c619
Author: Damien de Lemeny <d delemeny gmail com>
Date: Tue Jul 20 14:19:13 2010 +0200
Add brush-dab operations
* new operations/workshop/brush-dab.c
* new operations/workshop/parametric-dab.c
operations/workshop/brush-dab.c | 150 +++++++++++++++++
operations/workshop/parametric-dab.c | 292 ++++++++++++++++++++++++++++++++++
2 files changed, 442 insertions(+), 0 deletions(-)
---
diff --git a/operations/workshop/brush-dab.c b/operations/workshop/brush-dab.c
new file mode 100644
index 0000000..14b36fd
--- /dev/null
+++ b/operations/workshop/brush-dab.c
@@ -0,0 +1,150 @@
+
+/* 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, 2010 �yvind Kolås <pippin gimp org>
+ */
+
+#include "config.h"
+#include <glib/gi18n-lib.h>
+
+
+#ifdef GEGL_CHANT_PROPERTIES
+
+/* Brush shape parameters */
+gegl_chant_double (scale, _("Scale Factor"), 0.0, 10.0, 1.0,
+ _("Brush Scale Factor"))
+gegl_chant_double (aspect, _("Aspect Ratio"), 0.1, 10.0, 1.0,
+ _("Brush Aspect, -10 for pancake 10.0 for spike."))
+gegl_chant_double (angle, _("Angle"), 0.0, 360.0, 0.0,
+ _("Brush Angle."))
+gegl_chant_double (hardness, _("Hardness"), 0.0, 1.0, 0.6,
+ _("Brush Hardness, 0.0 for soft 1.0 for hard."))
+gegl_chant_double (force, _("Force"), 0.0, 1.0, 0.6,
+ _("Brush Force."))
+
+gegl_chant_color (color, _("Color"), "rgb(0.0,0.0,0.0)",
+ _("Color of paint to use for stroking."))
+gegl_chant_double (opacity, _("Opacity"), -2.0, 2.0, 1.0,
+ _("Opacity."))
+/* dab position parameters */
+gegl_chant_double (x, _("X"), -G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
+ _("Horizontal offset."))
+gegl_chant_double (y, _("Y"), -G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
+ _("Vertical offset."))
+
+gegl_chant_double (radius, _("Radius"), -G_MAXDOUBLE, G_MAXDOUBLE, 10.0,
+ _("Blur radius."))
+
+#else
+
+#define GEGL_CHANT_C_FILE "brush-dab.c"
+
+#include "gegl-plugin.h"
+
+struct _GeglChant
+{
+ GeglOperationMeta parent_instance;
+ gpointer properties;
+
+ GeglNode *input;
+ GeglNode *output;
+
+ GeglNode *scale;
+ GeglNode *rotate;
+ GeglNode *translate;
+
+ GeglNode *hardness;
+ GeglNode *force;
+
+ GeglNode *opacity;
+ GeglNode *color;
+};
+
+typedef struct
+{
+ GeglOperationMetaClass parent_class;
+} GeglChantClass;
+
+#include "gegl-chant.h"
+GEGL_DEFINE_DYNAMIC_OPERATION(GEGL_TYPE_OPERATION_META)
+
+/* in attach we hook into graph adding the needed nodes */
+static void attach (GeglOperation *operation)
+{
+ GeglChant *self = GEGL_CHANT (operation);
+ GeglNode *gegl = operation->node;
+
+ self->input = gegl_node_get_input_proxy (gegl, "input");
+ self->output = gegl_node_get_output_proxy (gegl, "output");
+
+ self->scale = gegl_node_new_child (gegl, "operation", "gegl:scale", NULL);
+ self->rotate = gegl_node_new_child (gegl, "operation", "gegl:rotate", NULL);
+ self->translate = gegl_node_new_child (gegl, "operation", "gegl:translate", NULL);
+
+ self->opacity = gegl_node_new_child (gegl, "operation", "gegl:opacity", NULL);
+
+ self->color = gegl_node_new_child (gegl, "operation", "gegl:color", NULL);
+
+ gegl_node_link_many (self->input,
+ self->translate, NULL);
+ gegl_node_link_many (self->color, self->opacity, self->output, NULL);
+
+ gegl_node_connect_from (self->opacity, "aux", self->translate, "output");
+ gegl_operation_meta_redirect (operation, "angle", self->rotate, "degrees");
+#if 0
+ gegl_operation_meta_redirect (operation, "hardness", self->hardness, "");
+ gegl_operation_meta_redirect (operation, "force", self->force, "");
+#endif
+ gegl_operation_meta_redirect (operation, "x", self->translate, "x");
+ gegl_operation_meta_redirect (operation, "y", self->translate, "y");
+ gegl_operation_meta_redirect (operation, "opacity", self->opacity, "value");
+ gegl_operation_meta_redirect (operation, "color", self->color, "value");
+}
+
+static void prepare (GeglOperation *operation)
+{
+ GeglChant *self = GEGL_CHANT (operation);
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gdouble ar = o->aspect;
+ gdouble scale = o->scale;
+
+ gdouble scalex = (ar>0.0)?scale*ar:scale;
+ gdouble scaley = (ar<0.0)?scale*ar:scale;
+
+ gegl_node_set (self->scale, "x", scalex, NULL);
+ gegl_node_set (self->scale, "y", scaley, NULL);
+}
+
+
+static void
+gegl_chant_class_init (GeglChantClass *klass)
+{
+ GObjectClass *object_class;
+ GeglOperationClass *operation_class;
+
+ object_class = G_OBJECT_CLASS (klass);
+ operation_class = GEGL_OPERATION_CLASS (klass);
+
+ operation_class->attach = attach;
+ operation_class->prepare = prepare;
+
+ operation_class->name = "gegl:brush-dab";
+ operation_class->categories = "meta:render";
+ operation_class->description =
+ _("Transform a brush dab.");
+}
+
+#endif
diff --git a/operations/workshop/parametric-dab.c b/operations/workshop/parametric-dab.c
new file mode 100644
index 0000000..6c678d4
--- /dev/null
+++ b/operations/workshop/parametric-dab.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
+
+
+/* Brush shape parameters */
+gegl_chant_double (scale, _("Scale Factor"), 0.0, 10.0, 2.0,
+ _("Brush Scale Factor"))
+gegl_chant_curve (scale_curve, _("Scale Curve"),
+ _("A Gegl Curve pondering the scale factor along the path"))
+gegl_chant_double (hardness, _("Hardness"), 0.0, 1.0, 0.6,
+ _("Brush Hardness, 0.0 for soft 1.0 for hard."))
+gegl_chant_double (angle, _("Angle"), 0.0, 360.0, 0.0,
+ _("Brush Angle."))
+gegl_chant_double (aspect, _("Aspect Ratio"), 0.1, 10.0, 1.0,
+ _("Brush Aspect, 0.1 for pancake 10.0 for spike."))
+gegl_chant_double (force, _("Force"), 0.0, 1.0, 0.6,
+ _("Brush Force."))
+
+gegl_chant_color (color, _("Color"), "rgba(0.0,0.0,0.0,0.0)",
+ _("Color of paint to use for stroking."))
+gegl_chant_double (opacity, _("Opacity"), -2.0, 2.0, 1.0,
+ _("Stroke Opacity."))
+
+/* dab position parameters */
+gegl_chant_double (x, _("X"), -G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
+ _("Horizontal offset."))
+gegl_chant_double (y, _("Y"), -G_MAXDOUBLE, G_MAXDOUBLE, 0.0,
+ _("Vertical offset."))
+gegl_chant_double (pos, _("Position"), 0.0, G_MAXDOUBLE, 0.0,
+ _("Position."))
+
+#else
+
+#define GEGL_CHANT_TYPE_FILTER
+#define GEGL_CHANT_C_FILE "parametric-dab.c"
+#define BASE_RADIUS 50.0
+
+#include "gegl-plugin.h"
+#include "gegl-buffer-private.h"
+
+#include "gegl-chant.h"
+#include <stdio.h>
+
+
+typedef struct StampStatic {
+ gboolean valid;
+ Babl *format;
+ gfloat *buf;
+ gdouble radius;
+}StampStatic;
+
+static void stamp (GeglBuffer *buffer,
+ const GeglRectangle *clip_rect,
+ gdouble x,
+ gdouble y,
+ gdouble radius,
+ gdouble hardness,
+ GeglColor *color,
+ gdouble opacity)
+{
+ gfloat col[4];
+ static StampStatic s = {FALSE,}; /* XXX:
+ we will ultimately leak the last valid
+ cached brush. */
+
+ GeglRectangle temp;
+ GeglRectangle roi;
+
+ roi.x = floor(x-radius);
+ roi.y = floor(y-radius);
+ roi.width = ceil (x+radius) - floor (x-radius);
+ roi.height = ceil (y+radius) - floor (y-radius);
+
+ gegl_color_get_rgba4f (color, col);
+
+ /* bail out if we wouldn't leave a mark on the buffer */
+ if (!gegl_rectangle_intersect (&temp, &roi, clip_rect))
+ {
+ return;
+ }
+
+ if (s.format == NULL)
+ s.format = babl_format ("RaGaBaA float");
+
+ if (s.buf == NULL ||
+ s.radius != radius)
+ {
+ if (s.buf != NULL)
+ g_free (s.buf);
+ /* allocate a little bit more, just in case due to rounding errors and
+ * such */
+ s.buf = g_malloc (4*4* (roi.width + 2 ) * (roi.height + 2));
+ s.radius = radius;
+ s.valid = TRUE;
+ }
+ g_assert (s.buf);
+
+ gegl_buffer_get_unlocked (buffer, 1.0, &roi, s.format, s.buf, 0);
+
+ {
+ gint u, v;
+ gint i=0;
+
+ gfloat radius_squared = radius * radius;
+ gfloat inner_radius_squared = (radius * hardness)*(radius * hardness);
+ gfloat soft_range = radius_squared - inner_radius_squared;
+
+ for (v= roi.y; v < roi.y + roi.height ; v++)
+ {
+ gfloat vy2 = (v-y)*(v-y);
+ for (u= roi.x; u < roi.x + roi.width; u++)
+ {
+ gfloat o = (u-x) * (u-x) + vy2;
+
+ if (o < inner_radius_squared)
+ o = col[3];
+ else if (o < radius_squared)
+ {
+ o = (1.0 - (o-inner_radius_squared) / (soft_range)) * col[3];
+ }
+ else
+ {
+ o=0.0;
+ }
+ if (o!=0.0)
+ {
+ gint c;
+ o = o*opacity;
+ for (c=0;c<4;c++)
+ s.buf[i*4+c] = (s.buf[i*4+c] * (1.0-o) + col[c] * o);
+ }
+ i++;
+ }
+ }
+ }
+ gegl_buffer_set_unlocked (buffer, &roi, s.format, s.buf, 0);
+}
+
+static void
+prepare (GeglOperation *operation)
+{
+ gegl_operation_set_format (operation, "output", babl_format ("RaGaBaA float"));
+ /*
+ gegl_operation_set_format (operation, "input", babl_format ("RaGaBaA float"));
+ gegl_operation_set_format (operation, "aux", babl_format ("Y float"));
+ */
+}
+
+static GeglRectangle
+get_bounding_box (GeglOperation *operation)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gfloat r = o->scale * BASE_RADIUS;
+ GeglRectangle result = {0,0,0,0};
+ GeglRectangle *in_rect = gegl_operation_source_get_bounding_box (operation, "input");
+
+ //if (in_rect)
+ {
+ result.x = o->x - r;
+ result.y = o->y - r;
+ result.width = r*2;
+ result.height = r*2;
+ }
+
+ return result;
+}
+
+#if 0
+static GeglRectangle
+get_cached_region (GeglOperation *operation)
+{
+ return get_bounding_box (operation);
+}
+#endif
+
+static gboolean
+process (GeglOperation *operation,
+ GeglBuffer *input,
+ GeglBuffer *output,
+ const GeglRectangle *result)
+{
+ GeglChantO *o = GEGL_CHANT_PROPERTIES (operation);
+ gdouble scale = BASE_RADIUS * o->scale;
+
+ if (input)
+ {
+ gegl_buffer_copy (input, result, output, result);
+ }
+ else
+ {
+ gegl_buffer_clear (output, result);
+ }
+ if (o->scale_curve)
+ {
+ scale = scale * gegl_curve_calc_value(o->scale_curve, o->pos);
+ fprintf(stderr,"scale curve found, scale=%f\n",scale);
+ }
+
+
+ /* Paint dab */
+ if (gegl_buffer_is_shared (output))
+ while (!gegl_buffer_try_lock (output));
+
+ stamp (output, result, o->x, o->y,
+ scale, o->hardness, o->color, o->opacity);
+
+ if (gegl_buffer_is_shared (output))
+ gegl_buffer_unlock (output);
+
+ return TRUE;
+}
+
+static GeglNode *detect (GeglOperation *operation,
+ gint x,
+ gint y)
+{
+#if 0
+ 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->radius);
+
+
+ if (o->radius > 0.1 && o->stroke_opacity > 0.0001)
+ result = cairo_in_stroke (cr, x, y);
+
+
+ cairo_destroy (cr);
+
+ if (result)
+ return operation->node;
+
+#endif
+ 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:parametric-dab";
+ operation_class->categories = "render";
+ operation_class->description = _("Renders a brush dab");
+#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]