2011-02-10 11:17:50 +00:00
|
|
|
/* GStreamer Editing Services
|
|
|
|
* Copyright (C) 2010 Thibault Saunier <thibault.saunier@collabora.co.uk>
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
2012-11-04 00:25:20 +00:00
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
2011-02-10 11:17:50 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2014-04-07 19:02:48 +00:00
|
|
|
* SECTION:geseffect
|
2011-02-10 11:17:50 +00:00
|
|
|
* @short_description: adds an effect build from a parse-launch style
|
2014-02-26 03:17:36 +00:00
|
|
|
* bin description to a stream in a GESSourceClip or a GESLayer
|
2011-02-10 11:17:50 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ges-internal.h"
|
2012-11-20 02:42:47 +00:00
|
|
|
#include "ges-extractable.h"
|
2013-01-26 15:31:33 +00:00
|
|
|
#include "ges-track-element.h"
|
2013-01-26 15:35:19 +00:00
|
|
|
#include "ges-base-effect.h"
|
2013-03-14 15:14:31 +00:00
|
|
|
#include "ges-effect-asset.h"
|
2013-01-26 15:40:51 +00:00
|
|
|
#include "ges-effect.h"
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2012-11-20 02:42:47 +00:00
|
|
|
static void ges_extractable_interface_init (GESExtractableInterface * iface);
|
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
G_DEFINE_TYPE_WITH_CODE (GESEffect,
|
|
|
|
ges_effect, GES_TYPE_BASE_EFFECT,
|
2012-11-20 02:42:47 +00:00
|
|
|
G_IMPLEMENT_INTERFACE (GES_TYPE_EXTRACTABLE,
|
|
|
|
ges_extractable_interface_init));
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
static void ges_effect_dispose (GObject * object);
|
|
|
|
static void ges_effect_finalize (GObject * object);
|
|
|
|
static GstElement *ges_effect_create_element (GESTrackElement * self);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
struct _GESEffectPrivate
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
|
|
|
gchar *bin_description;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_BIN_DESCRIPTION,
|
|
|
|
};
|
|
|
|
|
2012-11-20 02:42:47 +00:00
|
|
|
static gchar *
|
|
|
|
extractable_check_id (GType type, const gchar * id, GError ** error)
|
|
|
|
{
|
|
|
|
GstElement *effect = gst_parse_bin_from_description (id, TRUE, error);
|
|
|
|
|
|
|
|
if (effect == NULL)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
gst_object_unref (effect);
|
|
|
|
return g_strdup (id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static GParameter *
|
|
|
|
extractable_get_parameters_from_id (const gchar * id, guint * n_params)
|
|
|
|
{
|
|
|
|
GParameter *params = g_new0 (GParameter, 2);
|
|
|
|
|
2012-12-29 20:52:42 +00:00
|
|
|
params[0].name = "bin-description";
|
2012-11-20 02:42:47 +00:00
|
|
|
g_value_init (¶ms[0].value, G_TYPE_STRING);
|
|
|
|
g_value_set_string (¶ms[0].value, id);
|
|
|
|
|
|
|
|
*n_params = 1;
|
|
|
|
|
|
|
|
return params;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gchar *
|
|
|
|
extractable_get_id (GESExtractable * self)
|
|
|
|
{
|
2013-01-26 15:40:51 +00:00
|
|
|
return g_strdup (GES_EFFECT (self)->priv->bin_description);
|
2012-11-20 02:42:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
ges_extractable_interface_init (GESExtractableInterface * iface)
|
|
|
|
{
|
2013-03-14 15:14:31 +00:00
|
|
|
iface->asset_type = GES_TYPE_EFFECT_ASSET;
|
2012-11-20 02:42:47 +00:00
|
|
|
iface->check_id = (GESExtractableCheckId) extractable_check_id;
|
|
|
|
iface->get_parameters_from_id = extractable_get_parameters_from_id;
|
|
|
|
iface->get_id = extractable_get_id;
|
|
|
|
}
|
|
|
|
|
2011-02-10 11:17:50 +00:00
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_get_property (GObject * object,
|
2011-02-10 11:17:50 +00:00
|
|
|
guint property_id, GValue * value, GParamSpec * pspec)
|
|
|
|
{
|
2013-01-26 15:40:51 +00:00
|
|
|
GESEffectPrivate *priv = GES_EFFECT (object)->priv;
|
2011-06-21 18:35:47 +00:00
|
|
|
|
2011-02-10 11:17:50 +00:00
|
|
|
switch (property_id) {
|
2011-06-21 18:35:47 +00:00
|
|
|
case PROP_BIN_DESCRIPTION:
|
|
|
|
g_value_set_string (value, priv->bin_description);
|
|
|
|
break;
|
2011-02-10 11:17:50 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_set_property (GObject * object,
|
2011-02-10 11:17:50 +00:00
|
|
|
guint property_id, const GValue * value, GParamSpec * pspec)
|
|
|
|
{
|
2013-01-26 15:40:51 +00:00
|
|
|
GESEffect *self = GES_EFFECT (object);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
switch (property_id) {
|
|
|
|
case PROP_BIN_DESCRIPTION:
|
|
|
|
self->priv->bin_description = g_value_dup_string (value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_class_init (GESEffectClass * klass)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
|
|
|
GObjectClass *object_class;
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElementClass *obj_bg_class;
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
object_class = G_OBJECT_CLASS (klass);
|
2013-01-26 15:31:33 +00:00
|
|
|
obj_bg_class = GES_TRACK_ELEMENT_CLASS (klass);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
g_type_class_add_private (klass, sizeof (GESEffectPrivate));
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
object_class->get_property = ges_effect_get_property;
|
|
|
|
object_class->set_property = ges_effect_set_property;
|
|
|
|
object_class->dispose = ges_effect_dispose;
|
|
|
|
object_class->finalize = ges_effect_finalize;
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
obj_bg_class->create_element = ges_effect_create_element;
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
/**
|
2013-01-26 15:40:51 +00:00
|
|
|
* GESEffect:bin-description:
|
2011-02-10 11:17:50 +00:00
|
|
|
*
|
|
|
|
* The description of the effect bin with a gst-launch-style
|
|
|
|
* pipeline description.
|
2011-05-06 17:38:26 +00:00
|
|
|
*
|
|
|
|
* Example: "videobalance saturation=1.5 hue=+0.5"
|
2011-02-10 11:17:50 +00:00
|
|
|
*/
|
|
|
|
g_object_class_install_property (object_class, PROP_BIN_DESCRIPTION,
|
|
|
|
g_param_spec_string ("bin-description",
|
|
|
|
"bin description",
|
|
|
|
"Bin description of the effect",
|
|
|
|
NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_init (GESEffect * self)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
|
|
|
self->priv =
|
2013-01-26 15:40:51 +00:00
|
|
|
G_TYPE_INSTANCE_GET_PRIVATE (self, GES_TYPE_EFFECT, GESEffectPrivate);
|
2011-02-10 11:17:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_dispose (GObject * object)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
2013-01-26 15:40:51 +00:00
|
|
|
G_OBJECT_CLASS (ges_effect_parent_class)->dispose (object);
|
2011-02-10 11:17:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_finalize (GObject * object)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
2013-01-26 15:40:51 +00:00
|
|
|
GESEffect *self = GES_EFFECT (object);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
if (self->priv->bin_description)
|
|
|
|
g_free (self->priv->bin_description);
|
|
|
|
|
2013-01-26 15:40:51 +00:00
|
|
|
G_OBJECT_CLASS (ges_effect_parent_class)->finalize (object);
|
2011-02-10 11:17:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GstElement *
|
2013-01-26 15:40:51 +00:00
|
|
|
ges_effect_create_element (GESTrackElement * object)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
|
|
|
GstElement *effect;
|
2011-03-16 21:06:08 +00:00
|
|
|
gchar *bin_desc;
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
GError *error = NULL;
|
2013-01-26 15:40:51 +00:00
|
|
|
GESEffect *self = GES_EFFECT (object);
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrack *track = ges_track_element_get_track (object);
|
2013-08-27 03:31:14 +00:00
|
|
|
const gchar *wanted_categories[] = { "Effect", NULL };
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2011-03-16 21:06:08 +00:00
|
|
|
if (!track) {
|
|
|
|
GST_WARNING
|
2011-05-06 17:39:26 +00:00
|
|
|
("The object %p should be in a Track for the element to be created",
|
|
|
|
object);
|
2011-03-16 21:06:08 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (track->type == GES_TRACK_TYPE_VIDEO) {
|
2011-10-11 07:51:43 +00:00
|
|
|
bin_desc = g_strconcat ("videoconvert name=pre_video_convert ! ",
|
|
|
|
self->priv->bin_description, " ! videoconvert name=post_video_convert",
|
2011-05-07 11:41:11 +00:00
|
|
|
NULL);
|
2011-03-16 21:06:08 +00:00
|
|
|
} else if (track->type == GES_TRACK_TYPE_AUDIO) {
|
2011-05-06 17:39:26 +00:00
|
|
|
bin_desc =
|
|
|
|
g_strconcat ("audioconvert ! audioresample !",
|
|
|
|
self->priv->bin_description, NULL);
|
2011-03-16 21:06:08 +00:00
|
|
|
} else {
|
2011-05-06 17:39:26 +00:00
|
|
|
GST_DEBUG ("Track type not supported");
|
2011-03-16 21:06:08 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
effect = gst_parse_bin_from_description (bin_desc, TRUE, &error);
|
|
|
|
|
|
|
|
g_free (bin_desc);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
|
|
|
if (error != NULL) {
|
2011-05-07 11:41:11 +00:00
|
|
|
GST_ERROR ("An error occured while creating the GstElement: %s",
|
2011-05-06 17:39:26 +00:00
|
|
|
error->message);
|
2011-03-16 21:06:08 +00:00
|
|
|
g_error_free (error);
|
2011-02-10 11:17:50 +00:00
|
|
|
return NULL;
|
2011-03-16 21:06:08 +00:00
|
|
|
}
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2011-05-06 17:38:26 +00:00
|
|
|
GST_DEBUG ("Created effect %p", effect);
|
2011-02-10 11:17:50 +00:00
|
|
|
|
2013-08-27 03:31:14 +00:00
|
|
|
ges_track_element_add_children_props (object, effect, wanted_categories,
|
|
|
|
NULL, NULL);
|
|
|
|
|
2011-02-10 11:17:50 +00:00
|
|
|
return effect;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-01-26 15:40:51 +00:00
|
|
|
* ges_effect_new:
|
2011-05-06 17:38:26 +00:00
|
|
|
* @bin_description: The gst-launch like bin description of the effect
|
|
|
|
*
|
2013-01-26 15:40:51 +00:00
|
|
|
* Creates a new #GESEffect from the description of the bin.
|
2011-05-06 17:38:26 +00:00
|
|
|
*
|
2013-01-26 15:40:51 +00:00
|
|
|
* Returns: a newly created #GESEffect, or %NULL if something went
|
2011-05-06 17:38:26 +00:00
|
|
|
* wrong.
|
|
|
|
*/
|
2013-01-26 15:40:51 +00:00
|
|
|
GESEffect *
|
|
|
|
ges_effect_new (const gchar * bin_description)
|
2011-02-10 11:17:50 +00:00
|
|
|
{
|
2013-03-14 15:14:31 +00:00
|
|
|
GESEffect *effect;
|
|
|
|
GESAsset *asset = ges_asset_request (GES_TYPE_EFFECT,
|
2011-02-10 11:17:50 +00:00
|
|
|
bin_description, NULL);
|
2013-03-14 15:14:31 +00:00
|
|
|
|
|
|
|
g_return_val_if_fail (asset, NULL);
|
|
|
|
|
|
|
|
effect = GES_EFFECT (ges_asset_extract (asset, NULL));
|
|
|
|
|
|
|
|
gst_object_unref (asset);
|
|
|
|
|
|
|
|
return effect;
|
2011-02-10 11:17:50 +00:00
|
|
|
}
|