geometrictransform: adds some properties to base class
authorThiago Santos <thiago.sousa.santos@collabora.co.uk>
Thu, 3 Jun 2010 01:01:31 +0000 (22:01 -0300)
committerThiago Santos <thiago.sousa.santos@collabora.co.uk>
Fri, 4 Jun 2010 18:31:22 +0000 (15:31 -0300)
Adds a property to select what to do with pixels that are mapped out of
edges: ignore, clamp or wrap.

gst/geometrictransform/gstcircle.c
gst/geometrictransform/gstgeometrictransform.c
gst/geometrictransform/gstgeometrictransform.h
gst/geometrictransform/gstkaleidoscope.c
gst/geometrictransform/gstpinch.c
gst/geometrictransform/gsttwirl.c

index 31c5d47..16528a7 100644 (file)
@@ -167,8 +167,6 @@ circle_map (GstGeometricTransform * gt, gint x, gint y, gdouble * in_x,
   *in_y =
       gt->height * (1 - (distance - cgt->radius) / (circle->height + 0.0001));
 
-  *in_x = CLAMP (*in_x, 0, gt->width - 1);
-  *in_y = CLAMP (*in_y, 0, gt->height - 1);
   GST_DEBUG_OBJECT (circle, "Inversely mapped %d %d into %lf %lf",
       x, y, *in_x, *in_y);
 
index 13d3d04..0795e34 100644 (file)
@@ -22,6 +22,7 @@
 #endif
 
 #include "gstgeometrictransform.h"
+#include "geometricmath.h"
 #include <string.h>
 
 GST_DEBUG_CATEGORY_STATIC (geometric_transform_debug);
@@ -43,6 +44,43 @@ static GstStaticPadTemplate gst_geometric_transform_sink_template =
 
 static GstVideoFilterClass *parent_class = NULL;
 
+enum
+{
+  PROP_0,
+  PROP_OFF_EDGE_PIXELS
+};
+
+enum
+{
+  GST_GT_OFF_EDGES_PIXELS_IGNORE = 0,
+  GST_GT_OFF_EDGES_PIXELS_CLAMP,
+  GST_GT_OFF_EDGES_PIXELS_WRAP
+};
+
+#define GST_GT_OFF_EDGES_PIXELS_METHOD_TYPE ( \
+    gst_geometric_transform_off_edges_pixels_method_get_type())
+static GType
+gst_geometric_transform_off_edges_pixels_method_get_type (void)
+{
+  static GType method_type = 0;
+
+  static const GEnumValue method_types[] = {
+    {GST_GT_OFF_EDGES_PIXELS_IGNORE, "Ignore", "ignore"},
+    {GST_GT_OFF_EDGES_PIXELS_CLAMP, "Clamp", "clamp"},
+    {GST_GT_OFF_EDGES_PIXELS_WRAP, "Wrap", "wrap"},
+    {0, NULL, NULL}
+  };
+
+  if (!method_type) {
+    method_type =
+        g_enum_register_static ("GstGeometricTransformOffEdgesPixelsMethod",
+        method_types);
+  }
+  return method_type;
+}
+
+#define DEFAULT_OFF_EDGE_PIXELS GST_GT_OFF_EDGES_PIXELS_IGNORE
+
 static gboolean
 gst_geometric_transform_generate_map (GstGeometricTransform * gt)
 {
@@ -124,16 +162,43 @@ static void
 gst_geometric_transform_do_map (GstGeometricTransform * gt, GstBuffer * inbuf,
     GstBuffer * outbuf, gint x, gint y, gdouble in_x, gdouble in_y)
 {
-  gint trunc_x = (gint) in_x;
-  gint trunc_y = (gint) in_y;
   gint in_offset;
   gint out_offset;
 
   out_offset = y * gt->row_stride + x * gt->pixel_stride;
-  in_offset = trunc_y * gt->row_stride + trunc_x * gt->pixel_stride;
 
-  memcpy (GST_BUFFER_DATA (outbuf) + out_offset,
-      GST_BUFFER_DATA (inbuf) + in_offset, gt->pixel_stride);
+  /* operate on out of edge pixels */
+  switch (gt->off_edge_pixels) {
+    case GST_GT_OFF_EDGES_PIXELS_CLAMP:
+      in_x = CLAMP (in_x, 0, gt->width - 1);
+      in_y = CLAMP (in_y, 0, gt->height - 1);
+      break;
+
+    case GST_GT_OFF_EDGES_PIXELS_WRAP:
+      in_x = mod_float (in_x, gt->width);
+      in_y = mod_float (in_y, gt->height);
+      if (in_x < 0)
+        in_x += gt->width;
+      if (in_y < 0)
+        in_y += gt->height;
+      break;
+
+    default:
+      break;
+  }
+
+  {
+    gint trunc_x = (gint) in_x;
+    gint trunc_y = (gint) in_y;
+    /* only set the values if the values are valid */
+    if (trunc_x >= 0 && trunc_x < gt->width && trunc_y >= 0 &&
+        trunc_y < gt->height) {
+      in_offset = trunc_y * gt->row_stride + trunc_x * gt->pixel_stride;
+
+      memcpy (GST_BUFFER_DATA (outbuf) + out_offset,
+          GST_BUFFER_DATA (inbuf) + in_offset, gt->pixel_stride);
+    }
+  }
 }
 
 static GstFlowReturn
@@ -160,6 +225,43 @@ gst_geometric_transform_transform (GstBaseTransform * trans, GstBuffer * buf,
   return ret;
 }
 
+static void
+gst_geometric_transform_set_property (GObject * object, guint prop_id,
+    const GValue * value, GParamSpec * pspec)
+{
+  GstGeometricTransform *gt;
+
+  gt = GST_GEOMETRIC_TRANSFORM (object);
+
+  switch (prop_id) {
+    case PROP_OFF_EDGE_PIXELS:
+      gt->off_edge_pixels = g_value_get_enum (value);
+      break;
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+      break;
+  }
+}
+
+static void
+gst_geometric_transform_get_property (GObject * object, guint prop_id,
+    GValue * value, GParamSpec * pspec)
+{
+  GstGeometricTransform *gt;
+
+  gt = GST_GEOMETRIC_TRANSFORM (object);
+
+  switch (prop_id) {
+    case PROP_OFF_EDGE_PIXELS:
+      g_value_set_enum (value, gt->off_edge_pixels);
+      break;
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+      break;
+  }
+}
+
+
 static gboolean
 gst_geometric_transform_stop (GstBaseTransform * trans)
 {
@@ -184,16 +286,29 @@ gst_geometric_transform_base_init (gpointer g_class)
 static void
 gst_geometric_transform_class_init (gpointer klass, gpointer class_data)
 {
+  GObjectClass *obj_class;
   GstBaseTransformClass *trans_class;
 
+  obj_class = (GObjectClass *) klass;
   trans_class = (GstBaseTransformClass *) klass;
 
   parent_class = g_type_class_peek_parent (klass);
 
+  obj_class->set_property =
+      GST_DEBUG_FUNCPTR (gst_geometric_transform_set_property);
+  obj_class->get_property =
+      GST_DEBUG_FUNCPTR (gst_geometric_transform_get_property);
+
   trans_class->stop = GST_DEBUG_FUNCPTR (gst_geometric_transform_stop);
   trans_class->set_caps = GST_DEBUG_FUNCPTR (gst_geometric_transform_set_caps);
   trans_class->transform =
       GST_DEBUG_FUNCPTR (gst_geometric_transform_transform);
+
+  g_object_class_install_property (obj_class, PROP_OFF_EDGE_PIXELS,
+      g_param_spec_enum ("off-edge-pixels", "Off edge pixels",
+          "What to do with off edge pixels",
+          GST_GT_OFF_EDGES_PIXELS_METHOD_TYPE, DEFAULT_OFF_EDGE_PIXELS,
+          G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS | G_PARAM_CONSTRUCT));
 }
 
 static void
index 1e8ddc8..066799f 100644 (file)
@@ -80,6 +80,9 @@ struct _GstGeometricTransform {
   gint pixel_stride;
   gint row_stride;
 
+  /* properties */
+  gint off_edge_pixels;
+
   gdouble *map;
 };
 
index f4f1bee..805cdeb 100644 (file)
@@ -172,8 +172,6 @@ kaleidoscope_map (GstGeometricTransform * gt, gint x, gint y, gdouble * in_x,
   *in_x = cgt->precalc_x_center + distance * cos (theta);
   *in_y = cgt->precalc_y_center + distance * sin (theta);
 
-  *in_x = CLAMP (*in_x, 0, gt->width - 1);
-  *in_y = CLAMP (*in_y, 0, gt->height - 1);
   GST_DEBUG_OBJECT (kaleidoscope, "Inversely mapped %d %d into %lf %lf",
       x, y, *in_x, *in_y);
 
index bb0dd40..fd48e5a 100644 (file)
@@ -161,9 +161,6 @@ pinch_map (GstGeometricTransform * gt, gint x, gint y, gdouble * in_x,
 
     *in_x = cgt->precalc_x_center + dx;
     *in_y = cgt->precalc_y_center + dy;
-
-    *in_x = CLAMP (*in_x, 0, gt->width - 1);
-    *in_y = CLAMP (*in_y, 0, gt->height - 1);
   }
 
   GST_DEBUG_OBJECT (pinch, "Inversely mapped %d %d into %lf %lf",
index 0d88b97..408bbbc 100644 (file)
@@ -150,9 +150,6 @@ twirl_map (GstGeometricTransform * gt, gint x, gint y, gdouble * in_x,
 
     *in_x = cgt->precalc_x_center + d * cos (a);
     *in_y = cgt->precalc_y_center + d * sin (a);
-
-    *in_x = CLAMP (*in_x, 0, gt->width - 1);
-    *in_y = CLAMP (*in_y, 0, gt->height - 1);
   }
 
   GST_DEBUG_OBJECT (twirl, "Inversely mapped %d %d into %lf %lf",