mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-11 01:45:33 +00:00
geometrictransform: adds some properties to base class
Adds a property to select what to do with pixels that are mapped out of edges: ignore, clamp or wrap.
This commit is contained in:
parent
179e234361
commit
a00614064b
6 changed files with 123 additions and 15 deletions
|
@ -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);
|
||||
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -80,6 +80,9 @@ struct _GstGeometricTransform {
|
|||
gint pixel_stride;
|
||||
gint row_stride;
|
||||
|
||||
/* properties */
|
||||
gint off_edge_pixels;
|
||||
|
||||
gdouble *map;
|
||||
};
|
||||
|
||||
|
|
|
@ -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);
|
||||
|
||||
|
|
|
@ -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",
|
||||
|
|
|
@ -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",
|
||||
|
|
Loading…
Reference in a new issue