2501 lines
		
	
	
		
			79 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			2501 lines
		
	
	
		
			79 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* gst-editing-services
 | |
|  * Copyright (C) <2013> Thibault Saunier <thibault.saunier@collabora.com>
 | |
|  *               <2013> Collabora Ltd.
 | |
|  *
 | |
|  * gst-editing-services is free software: you can redistribute it and/or modify it
 | |
|  * under the terms of the GNU General Public License as published by the
 | |
|  * Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * gst-editing-services 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 General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * SECTION:gestimelineelement
 | |
|  * @title: GESTimelineElement
 | |
|  * @short_description: Base Class for all elements with some temporal extent
 | |
|  * within a #GESTimeline.
 | |
|  *
 | |
|  * A #GESTimelineElement will have some temporal extent in its
 | |
|  * corresponding #GESTimelineElement:timeline, controlled by its
 | |
|  * #GESTimelineElement:start and #GESTimelineElement:duration. This
 | |
|  * determines when its content will be displayed, or its effect applied,
 | |
|  * in the timeline. Several objects may overlap within a given
 | |
|  * #GESTimeline, in which case their #GESTimelineElement:priority is used
 | |
|  * to determine their ordering in the timeline. Priority is mostly handled
 | |
|  * internally by #GESLayer-s and #GESClip-s.
 | |
|  *
 | |
|  * A timeline element can have a #GESTimelineElement:parent,
 | |
|  * such as a #GESClip, which is responsible for controlling its timing.
 | |
|  *
 | |
|  * ## Editing
 | |
|  *
 | |
|  * Elements can be moved around in their #GESTimelineElement:timeline by
 | |
|  * setting their #GESTimelineElement:start and
 | |
|  * #GESTimelineElement:duration using ges_timeline_element_set_start()
 | |
|  * and ges_timeline_element_set_duration(). Additionally, which parts of
 | |
|  * the underlying content are played in the timeline can be adjusted by
 | |
|  * setting the #GESTimelineElement:in-point using
 | |
|  * ges_timeline_element_set_inpoint(). The library also provides
 | |
|  * ges_timeline_element_edit(), with various #GESEditMode-s, which can
 | |
|  * adjust these properties in a convenient way, as well as introduce
 | |
|  * similar changes in neighbouring or later elements in the timeline.
 | |
|  *
 | |
|  * However, a timeline may refuse a change in these properties if they
 | |
|  * would place the timeline in an unsupported configuration. See
 | |
|  * #GESTimeline for its overlap rules.
 | |
|  *
 | |
|  * Additionally, an edit may be refused if it would place one of the
 | |
|  * timing properties out of bounds (such as a negative time value for
 | |
|  * #GESTimelineElement:start, or having insufficient internal
 | |
|  * content to last for the desired #GESTimelineElement:duration).
 | |
|  *
 | |
|  * ## Time Coordinates
 | |
|  *
 | |
|  * There are three main sets of time coordinates to consider when using
 | |
|  * timeline elements:
 | |
|  *
 | |
|  * + Timeline coordinates: these are the time coordinates used in the
 | |
|  *   output of the timeline in its #GESTrack-s. Each track share the same
 | |
|  *   coordinates, so there is only one set of coordinates for the
 | |
|  *   timeline. These extend indefinitely from 0. The times used for
 | |
|  *   editing (including setting #GESTimelineElement:start and
 | |
|  *   #GESTimelineElement:duration) use these coordinates, since these
 | |
|  *   define when an element is present and for how long the element lasts
 | |
|  *   for in the timeline.
 | |
|  * + Internal source coordinates: these are the time coordinates used
 | |
|  *   internally at the element's output. This is only really defined for
 | |
|  *   #GESTrackElement-s, where it refers to time coordinates used at the
 | |
|  *   final source pad of the wrapped #GstElement-s. However, these
 | |
|  *   coordinates may also be used in a #GESClip in reference to its
 | |
|  *   children. In particular, these are the coordinates used for
 | |
|  *   #GESTimelineElement:in-point and #GESTimelineElement:max-duration.
 | |
|  * + Internal sink coordinates: these are the time coordinates used
 | |
|  *   internally at the element's input. A #GESSource has no input, so
 | |
|  *   these would be undefined. Otherwise, for most #GESTrackElement-s
 | |
|  *   these will be the same set of coordinates as the internal source
 | |
|  *   coordinates because the element does not change the timing
 | |
|  *   internally. Only #GESBaseEffect can support elements where these
 | |
|  *   are different. See #GESBaseEffect for more information.
 | |
|  *
 | |
|  * You can determine the timeline time for a given internal source time
 | |
|  * in a #GESTrack in a #GESClip using
 | |
|  * ges_clip_get_timeline_time_from_internal_time(), and vice versa using
 | |
|  * ges_clip_get_internal_time_from_timeline_time(), for the purposes of
 | |
|  * editing and setting timings properties.
 | |
|  *
 | |
|  * ## Children Properties
 | |
|  *
 | |
|  * If a timeline element owns another #GstObject and wishes to expose
 | |
|  * some of its properties, it can do so by registering the property as one
 | |
|  * of the timeline element's children properties using
 | |
|  * ges_timeline_element_add_child_property(). The registered property of
 | |
|  * the child can then be read and set using the
 | |
|  * ges_timeline_element_get_child_property() and
 | |
|  * ges_timeline_element_set_child_property() methods, respectively. Some
 | |
|  * sub-classed objects will be created with pre-registered children
 | |
|  * properties; for example, to expose part of an underlying #GstElement
 | |
|  * that is used internally. The registered properties can be listed with
 | |
|  * ges_timeline_element_list_children_properties().
 | |
|  */
 | |
| #ifdef HAVE_CONFIG_H
 | |
| #include "config.h"
 | |
| #endif
 | |
| 
 | |
| #include "ges-utils.h"
 | |
| #include "ges-timeline-element.h"
 | |
| #include "ges-extractable.h"
 | |
| #include "ges-meta-container.h"
 | |
| #include "ges-internal.h"
 | |
| #include "ges-effect.h"
 | |
| 
 | |
| #include <string.h>
 | |
| #include <gobject/gvaluecollector.h>
 | |
| 
 | |
| /* maps type name quark => count */
 | |
| static GData *object_name_counts = NULL;
 | |
| 
 | |
| static void
 | |
| extractable_set_asset (GESExtractable * extractable, GESAsset * asset)
 | |
| {
 | |
|   GES_TIMELINE_ELEMENT (extractable)->asset = asset;
 | |
| }
 | |
| 
 | |
| static void
 | |
| ges_extractable_interface_init (GESExtractableInterface * iface)
 | |
| {
 | |
|   iface->set_asset = extractable_set_asset;
 | |
| }
 | |
| 
 | |
| enum
 | |
| {
 | |
|   PROP_0,
 | |
|   PROP_PARENT,
 | |
|   PROP_TIMELINE,
 | |
|   PROP_START,
 | |
|   PROP_INPOINT,
 | |
|   PROP_DURATION,
 | |
|   PROP_MAX_DURATION,
 | |
|   PROP_PRIORITY,
 | |
|   PROP_NAME,
 | |
|   PROP_SERIALIZE,
 | |
|   PROP_LAST
 | |
| };
 | |
| 
 | |
| enum
 | |
| {
 | |
|   DEEP_NOTIFY,
 | |
|   CHILD_PROPERTY_ADDED,
 | |
|   CHILD_PROPERTY_REMOVED,
 | |
|   LAST_SIGNAL
 | |
| };
 | |
| 
 | |
| static guint ges_timeline_element_signals[LAST_SIGNAL] = { 0 };
 | |
| 
 | |
| static GParamSpec *properties[PROP_LAST] = { NULL, };
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   GObject *child;
 | |
|   GESTimelineElement *owner;
 | |
|   gulong handler_id;
 | |
|   GESTimelineElement *self;
 | |
| } ChildPropHandler;
 | |
| 
 | |
| struct _GESTimelineElementPrivate
 | |
| {
 | |
|   gboolean serialize;
 | |
| 
 | |
|   /* We keep a link between properties name and elements internally
 | |
|    * The hashtable should look like
 | |
|    * {GParamaSpec ---> child}*/
 | |
|   GHashTable *children_props;
 | |
| 
 | |
|   GESTimelineElement *copied_from;
 | |
| 
 | |
|   GESTimelineElementFlags flags;
 | |
| };
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   GObject *child;
 | |
|   GParamSpec *arg;
 | |
|   GESTimelineElement *self;
 | |
| } EmitDeepNotifyInIdleData;
 | |
| 
 | |
| G_DEFINE_ABSTRACT_TYPE_WITH_CODE (GESTimelineElement, ges_timeline_element,
 | |
|     G_TYPE_INITIALLY_UNOWNED, G_ADD_PRIVATE (GESTimelineElement)
 | |
|     G_IMPLEMENT_INTERFACE (GES_TYPE_EXTRACTABLE, ges_extractable_interface_init)
 | |
|     G_IMPLEMENT_INTERFACE (GES_TYPE_META_CONTAINER, NULL));
 | |
| 
 | |
| /*********************************************
 | |
|  *      Virtual methods implementation       *
 | |
|  *********************************************/
 | |
| static void
 | |
| _set_child_property (GESTimelineElement * self G_GNUC_UNUSED, GObject * child,
 | |
|     GParamSpec * pspec, GValue * value)
 | |
| {
 | |
|   if (G_VALUE_TYPE (value) != pspec->value_type
 | |
|       && G_VALUE_TYPE (value) == G_TYPE_STRING)
 | |
|     gst_util_set_object_arg (child, pspec->name, g_value_get_string (value));
 | |
|   else
 | |
|     g_object_set_property (child, pspec->name, value);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| _set_child_property_full (GESTimelineElement * self, GObject * child,
 | |
|     GParamSpec * pspec, const GValue * value, GError ** error)
 | |
| {
 | |
|   GES_TIMELINE_ELEMENT_GET_CLASS (self)->set_child_property (self, child,
 | |
|       pspec, (GValue *) value);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| _lookup_child (GESTimelineElement * self, const gchar * prop_name,
 | |
|     GObject ** child, GParamSpec ** pspec)
 | |
| {
 | |
|   GHashTableIter iter;
 | |
|   gpointer key, value;
 | |
|   gchar **names, *name, *classename;
 | |
|   gboolean res;
 | |
| 
 | |
|   classename = NULL;
 | |
|   res = FALSE;
 | |
| 
 | |
|   names = g_strsplit (prop_name, "::", 2);
 | |
|   if (names[1] != NULL) {
 | |
|     classename = names[0];
 | |
|     name = names[1];
 | |
|   } else
 | |
|     name = names[0];
 | |
| 
 | |
|   g_hash_table_iter_init (&iter, self->priv->children_props);
 | |
|   while (g_hash_table_iter_next (&iter, &key, &value)) {
 | |
|     if (g_strcmp0 (G_PARAM_SPEC (key)->name, name) == 0) {
 | |
|       ChildPropHandler *handler = (ChildPropHandler *) value;
 | |
|       if (classename == NULL ||
 | |
|           g_strcmp0 (G_OBJECT_TYPE_NAME (G_OBJECT (handler->child)),
 | |
|               classename) == 0 ||
 | |
|           g_strcmp0 (g_type_name (G_PARAM_SPEC (key)->owner_type),
 | |
|               classename) == 0) {
 | |
|         GST_DEBUG_OBJECT (self, "The %s property from %s has been found", name,
 | |
|             classename);
 | |
|         if (child)
 | |
|           *child = gst_object_ref (handler->child);
 | |
| 
 | |
|         if (pspec)
 | |
|           *pspec = g_param_spec_ref (key);
 | |
|         res = TRUE;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   g_strfreev (names);
 | |
| 
 | |
|   return res;
 | |
| }
 | |
| 
 | |
| GParamSpec **
 | |
| ges_timeline_element_get_children_properties (GESTimelineElement * self,
 | |
|     guint * n_properties)
 | |
| {
 | |
|   GParamSpec **pspec, *spec;
 | |
|   GHashTableIter iter;
 | |
|   gpointer key, value;
 | |
| 
 | |
|   guint i = 0;
 | |
| 
 | |
|   *n_properties = g_hash_table_size (self->priv->children_props);
 | |
|   pspec = g_new (GParamSpec *, *n_properties);
 | |
| 
 | |
|   g_hash_table_iter_init (&iter, self->priv->children_props);
 | |
|   while (g_hash_table_iter_next (&iter, &key, &value)) {
 | |
|     spec = G_PARAM_SPEC (key);
 | |
|     pspec[i] = g_param_spec_ref (spec);
 | |
|     i++;
 | |
|   }
 | |
| 
 | |
|   return pspec;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _get_property (GObject * object, guint property_id,
 | |
|     GValue * value, GParamSpec * pspec)
 | |
| {
 | |
|   GESTimelineElement *self = GES_TIMELINE_ELEMENT (object);
 | |
| 
 | |
|   switch (property_id) {
 | |
|     case PROP_PARENT:
 | |
|       g_value_take_object (value, self->parent);
 | |
|       break;
 | |
|     case PROP_TIMELINE:
 | |
|       g_value_take_object (value, self->timeline);
 | |
|       break;
 | |
|     case PROP_START:
 | |
|       g_value_set_uint64 (value, self->start);
 | |
|       break;
 | |
|     case PROP_INPOINT:
 | |
|       g_value_set_uint64 (value, self->inpoint);
 | |
|       break;
 | |
|     case PROP_DURATION:
 | |
|       g_value_set_uint64 (value, self->duration);
 | |
|       break;
 | |
|     case PROP_MAX_DURATION:
 | |
|       g_value_set_uint64 (value, self->maxduration);
 | |
|       break;
 | |
|     case PROP_PRIORITY:
 | |
|       g_value_set_uint (value, self->priority);
 | |
|       break;
 | |
|     case PROP_NAME:
 | |
|       g_value_take_string (value, ges_timeline_element_get_name (self));
 | |
|       break;
 | |
|     case PROP_SERIALIZE:
 | |
|       g_value_set_boolean (value, self->priv->serialize);
 | |
|       break;
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (self, property_id, pspec);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void
 | |
| _set_property (GObject * object, guint property_id,
 | |
|     const GValue * value, GParamSpec * pspec)
 | |
| {
 | |
|   GESTimelineElement *self = GES_TIMELINE_ELEMENT (object);
 | |
| 
 | |
|   switch (property_id) {
 | |
|     case PROP_PARENT:
 | |
|       ges_timeline_element_set_parent (self, g_value_get_object (value));
 | |
|       break;
 | |
|     case PROP_TIMELINE:
 | |
|       ges_timeline_element_set_timeline (self, g_value_get_object (value));
 | |
|       break;
 | |
|     case PROP_START:
 | |
|       ges_timeline_element_set_start (self, g_value_get_uint64 (value));
 | |
|       break;
 | |
|     case PROP_INPOINT:
 | |
|       ges_timeline_element_set_inpoint (self, g_value_get_uint64 (value));
 | |
|       break;
 | |
|     case PROP_DURATION:
 | |
|       ges_timeline_element_set_duration (self, g_value_get_uint64 (value));
 | |
|       break;
 | |
|     case PROP_PRIORITY:
 | |
|       ges_timeline_element_set_priority (self, g_value_get_uint (value));
 | |
|       break;
 | |
|     case PROP_MAX_DURATION:
 | |
|       ges_timeline_element_set_max_duration (self, g_value_get_uint64 (value));
 | |
|       break;
 | |
|     case PROP_NAME:
 | |
|       ges_timeline_element_set_name (self, g_value_get_string (value));
 | |
|       break;
 | |
|     case PROP_SERIALIZE:
 | |
|       self->priv->serialize = g_value_get_boolean (value);
 | |
|       break;
 | |
|     default:
 | |
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (self, property_id, pspec);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void
 | |
| ges_timeline_element_dispose (GObject * object)
 | |
| {
 | |
|   GESTimelineElement *self = GES_TIMELINE_ELEMENT (object);
 | |
| 
 | |
|   if (self->priv->children_props) {
 | |
|     g_hash_table_unref (self->priv->children_props);
 | |
|     self->priv->children_props = NULL;
 | |
|   }
 | |
| 
 | |
|   g_clear_object (&self->priv->copied_from);
 | |
| 
 | |
|   G_OBJECT_CLASS (ges_timeline_element_parent_class)->dispose (object);
 | |
| }
 | |
| 
 | |
| static void
 | |
| ges_timeline_element_finalize (GObject * self)
 | |
| {
 | |
|   GESTimelineElement *tle = GES_TIMELINE_ELEMENT (self);
 | |
| 
 | |
|   g_free (tle->name);
 | |
| 
 | |
|   G_OBJECT_CLASS (ges_timeline_element_parent_class)->finalize (self);
 | |
| }
 | |
| 
 | |
| static void
 | |
| _child_prop_handler_free (ChildPropHandler * handler)
 | |
| {
 | |
|   g_object_freeze_notify (handler->child);
 | |
|   if (handler->handler_id)
 | |
|     g_signal_handler_disconnect (handler->child, handler->handler_id);
 | |
|   g_object_thaw_notify (handler->child);
 | |
| 
 | |
|   if (handler->child != (GObject *) handler->self &&
 | |
|       handler->child != (GObject *) handler->owner)
 | |
|     gst_object_unref (handler->child);
 | |
|   g_slice_free (ChildPropHandler, handler);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| _get_natural_framerate (GESTimelineElement * self, gint * framerate_n,
 | |
|     gint * framerate_d)
 | |
| {
 | |
|   GST_INFO_OBJECT (self, "No natural framerate");
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| ges_timeline_element_init (GESTimelineElement * self)
 | |
| {
 | |
|   self->priv = ges_timeline_element_get_instance_private (self);
 | |
| 
 | |
|   self->priv->serialize = TRUE;
 | |
| 
 | |
|   self->priv->children_props =
 | |
|       g_hash_table_new_full ((GHashFunc) ges_pspec_hash, ges_pspec_equal,
 | |
|       (GDestroyNotify) g_param_spec_unref,
 | |
|       (GDestroyNotify) _child_prop_handler_free);
 | |
| }
 | |
| 
 | |
| static void
 | |
| ges_timeline_element_class_init (GESTimelineElementClass * klass)
 | |
| {
 | |
|   GObjectClass *object_class = G_OBJECT_CLASS (klass);
 | |
| 
 | |
|   object_class->get_property = _get_property;
 | |
|   object_class->set_property = _set_property;
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:parent:
 | |
|    *
 | |
|    * The parent container of the element.
 | |
|    */
 | |
|   properties[PROP_PARENT] =
 | |
|       g_param_spec_object ("parent", "Parent",
 | |
|       "The parent container of the object", GES_TYPE_TIMELINE_ELEMENT,
 | |
|       G_PARAM_READWRITE);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:timeline:
 | |
|    *
 | |
|    * The timeline that the element lies within.
 | |
|    */
 | |
|   properties[PROP_TIMELINE] =
 | |
|       g_param_spec_object ("timeline", "Timeline",
 | |
|       "The timeline the object is in", GES_TYPE_TIMELINE,
 | |
|       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:start:
 | |
|    *
 | |
|    * The starting position of the element in the timeline (in nanoseconds
 | |
|    * and in the time coordinates of the timeline). For example, for a
 | |
|    * source element, this would determine the time at which it should
 | |
|    * start outputting its internal content. For an operation element, this
 | |
|    * would determine the time at which it should start applying its effect
 | |
|    * to any source content.
 | |
|    */
 | |
|   properties[PROP_START] = g_param_spec_uint64 ("start", "Start",
 | |
|       "The position in the timeline", 0, G_MAXUINT64, 0,
 | |
|       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:in-point:
 | |
|    *
 | |
|    * The initial offset to use internally when outputting content (in
 | |
|    * nanoseconds, but in the time coordinates of the internal content).
 | |
|    *
 | |
|    * For example, for a #GESVideoUriSource that references some media
 | |
|    * file, the "internal content" is the media file data, and the
 | |
|    * in-point would correspond to some timestamp in the media file.
 | |
|    * When playing the timeline, and when the element is first reached at
 | |
|    * timeline-time #GESTimelineElement:start, it will begin outputting the
 | |
|    * data from the timestamp in-point **onwards**, until it reaches the
 | |
|    * end of its #GESTimelineElement:duration in the timeline.
 | |
|    *
 | |
|    * For elements that have no internal content, this should be kept
 | |
|    * as 0.
 | |
|    */
 | |
|   properties[PROP_INPOINT] =
 | |
|       g_param_spec_uint64 ("in-point", "In-point", "The in-point", 0,
 | |
|       G_MAXUINT64, 0, G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:duration:
 | |
|    *
 | |
|    * The duration that the element is in effect for in the timeline (a
 | |
|    * time difference in nanoseconds using the time coordinates of the
 | |
|    * timeline). For example, for a source element, this would determine
 | |
|    * for how long it should output its internal content for. For an
 | |
|    * operation element, this would determine for how long its effect
 | |
|    * should be applied to any source content.
 | |
|    */
 | |
|   properties[PROP_DURATION] =
 | |
|       g_param_spec_uint64 ("duration", "Duration", "The play duration", 0,
 | |
|       G_MAXUINT64, GST_CLOCK_TIME_NONE,
 | |
|       G_PARAM_READWRITE | G_PARAM_EXPLICIT_NOTIFY);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:max-duration:
 | |
|    *
 | |
|    * The full duration of internal content that is available (a time
 | |
|    * difference in nanoseconds using the time coordinates of the internal
 | |
|    * content).
 | |
|    *
 | |
|    * This will act as a cap on the #GESTimelineElement:in-point of the
 | |
|    * element (which is in the same time coordinates), and will sometimes
 | |
|    * be used to limit the #GESTimelineElement:duration of the element in
 | |
|    * the timeline.
 | |
|    *
 | |
|    * For example, for a #GESVideoUriSource that references some media
 | |
|    * file, this would be the length of the media file.
 | |
|    *
 | |
|    * For elements that have no internal content, or whose content is
 | |
|    * indefinite, this should be kept as #GST_CLOCK_TIME_NONE.
 | |
|    */
 | |
|   properties[PROP_MAX_DURATION] =
 | |
|       g_param_spec_uint64 ("max-duration", "Maximum duration",
 | |
|       "The maximum duration of the object", 0, G_MAXUINT64, GST_CLOCK_TIME_NONE,
 | |
|       G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_EXPLICIT_NOTIFY);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:priority:
 | |
|    *
 | |
|    * The priority of the element.
 | |
|    *
 | |
|    * Deprecated: 1.10: Priority management is now done by GES itself.
 | |
|    */
 | |
|   properties[PROP_PRIORITY] = g_param_spec_uint ("priority", "Priority",
 | |
|       "The priority of the object", 0, G_MAXUINT, 0, G_PARAM_READWRITE);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:name:
 | |
|    *
 | |
|    * The name of the element. This should be unique within its timeline.
 | |
|    */
 | |
|   properties[PROP_NAME] =
 | |
|       g_param_spec_string ("name", "Name", "The name of the timeline object",
 | |
|       NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT | G_PARAM_STATIC_STRINGS);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement:serialize:
 | |
|    *
 | |
|    * Whether the element should be serialized.
 | |
|    */
 | |
|   properties[PROP_SERIALIZE] = g_param_spec_boolean ("serialize", "Serialize",
 | |
|       "Whether the element should be serialized", TRUE,
 | |
|       G_PARAM_READWRITE | GES_PARAM_NO_SERIALIZATION);
 | |
| 
 | |
|   g_object_class_install_properties (object_class, PROP_LAST, properties);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement::deep-notify:
 | |
|    * @timeline_element: A #GESTtimelineElement
 | |
|    * @prop_object: The child whose property has been set
 | |
|    * @prop: The specification for the property that been set
 | |
|    *
 | |
|    * Emitted when a child of the element has one of its registered
 | |
|    * properties set. See ges_timeline_element_add_child_property().
 | |
|    * Note that unlike #GObject::notify, a child property name can not be
 | |
|    * used as a signal detail.
 | |
|    */
 | |
|   ges_timeline_element_signals[DEEP_NOTIFY] =
 | |
|       g_signal_new ("deep-notify", G_TYPE_FROM_CLASS (klass),
 | |
|       G_SIGNAL_RUN_FIRST | G_SIGNAL_NO_RECURSE | G_SIGNAL_DETAILED |
 | |
|       G_SIGNAL_NO_HOOKS, 0, NULL, NULL, NULL,
 | |
|       G_TYPE_NONE, 2, G_TYPE_OBJECT, G_TYPE_PARAM);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement::child-property-added:
 | |
|    * @timeline_element: A #GESTtimelineElement
 | |
|    * @prop_object: The child whose property has been registered
 | |
|    * @prop: The specification for the property that has been registered
 | |
|    *
 | |
|    * Emitted when the element has a new child property registered. See
 | |
|    * ges_timeline_element_add_child_property().
 | |
|    *
 | |
|    * Note that some GES elements will be automatically created with
 | |
|    * pre-registered children properties. You can use
 | |
|    * ges_timeline_element_list_children_properties() to list these.
 | |
|    *
 | |
|    * Since: 1.18
 | |
|    */
 | |
|   ges_timeline_element_signals[CHILD_PROPERTY_ADDED] =
 | |
|       g_signal_new ("child-property-added", G_TYPE_FROM_CLASS (klass),
 | |
|       G_SIGNAL_RUN_FIRST, 0, NULL, NULL, NULL, G_TYPE_NONE, 2,
 | |
|       G_TYPE_OBJECT, G_TYPE_PARAM);
 | |
| 
 | |
|   /**
 | |
|    * GESTimelineElement::child-property-removed:
 | |
|    * @timeline_element: A #GESTimelineElement
 | |
|    * @prop_object: The child whose property has been unregistered
 | |
|    * @prop: The specification for the property that has been unregistered
 | |
|    *
 | |
|    * Emitted when the element has a child property unregistered. See
 | |
|    * ges_timeline_element_remove_child_property().
 | |
|    *
 | |
|    * Since: 1.18
 | |
|    */
 | |
|   ges_timeline_element_signals[CHILD_PROPERTY_REMOVED] =
 | |
|       g_signal_new ("child-property-removed", G_TYPE_FROM_CLASS (klass),
 | |
|       G_SIGNAL_RUN_FIRST, 0, NULL, NULL, NULL, G_TYPE_NONE, 2,
 | |
|       G_TYPE_OBJECT, G_TYPE_PARAM);
 | |
| 
 | |
| 
 | |
|   object_class->dispose = ges_timeline_element_dispose;
 | |
|   object_class->finalize = ges_timeline_element_finalize;
 | |
| 
 | |
|   klass->set_parent = NULL;
 | |
|   klass->set_start = NULL;
 | |
|   klass->set_inpoint = NULL;
 | |
|   klass->set_duration = NULL;
 | |
|   klass->set_max_duration = NULL;
 | |
|   klass->set_priority = NULL;
 | |
| 
 | |
|   klass->ripple = NULL;
 | |
|   klass->ripple_end = NULL;
 | |
|   klass->roll_start = NULL;
 | |
|   klass->roll_end = NULL;
 | |
|   klass->trim = NULL;
 | |
| 
 | |
|   klass->list_children_properties =
 | |
|       ges_timeline_element_get_children_properties;
 | |
|   klass->lookup_child = _lookup_child;
 | |
|   klass->set_child_property = _set_child_property;
 | |
|   klass->set_child_property_full = _set_child_property_full;
 | |
|   klass->get_natural_framerate = _get_natural_framerate;
 | |
| }
 | |
| 
 | |
| static void
 | |
| _set_name (GESTimelineElement * self, const gchar * wanted_name)
 | |
| {
 | |
|   const gchar *type_name;
 | |
|   gchar *lowcase_type;
 | |
|   gint count;
 | |
|   GQuark q;
 | |
|   guint i, l;
 | |
|   gchar *name = NULL;
 | |
| 
 | |
|   if (!object_name_counts) {
 | |
|     g_datalist_init (&object_name_counts);
 | |
|   }
 | |
| 
 | |
|   q = g_type_qname (G_OBJECT_TYPE (self));
 | |
|   count = GPOINTER_TO_INT (g_datalist_id_get_data (&object_name_counts, q));
 | |
| 
 | |
|   /* GstFooSink -> foosink<N> */
 | |
|   type_name = g_quark_to_string (q);
 | |
|   if (strncmp (type_name, "GES", 3) == 0)
 | |
|     type_name += 3;
 | |
| 
 | |
|   lowcase_type = g_strdup (type_name);
 | |
|   l = strlen (lowcase_type);
 | |
|   for (i = 0; i < l; i++)
 | |
|     lowcase_type[i] = g_ascii_tolower (lowcase_type[i]);
 | |
| 
 | |
|   if (wanted_name == NULL) {
 | |
|     /* give the 20th "uriclip" element and the first "uriclip2" (if needed in the future)
 | |
|      * different names */
 | |
|     l = strlen (type_name);
 | |
|     if (l > 0 && g_ascii_isdigit (type_name[l - 1])) {
 | |
|       name = g_strdup_printf ("%s-%d", lowcase_type, count++);
 | |
|     } else {
 | |
|       name = g_strdup_printf ("%s%d", lowcase_type, count++);
 | |
|     }
 | |
|   } else {
 | |
|     /* If the wanted name uses the same 'namespace' as default, make
 | |
|      * sure it does not badly interfere with our counting system */
 | |
| 
 | |
|     /* FIXME: should we really be allowing a user to set the name
 | |
|      * "uriclip1" for, say, a GESTransition? The below code *does not*
 | |
|      * capture this case (because the prefix does not match "transition").
 | |
|      * If the user subsequently calls _set_name with name == NULL, on a
 | |
|      * GESClip *for the first time*, then the GES library will
 | |
|      * automatically choose the *same* name "uriclip1", but this is not
 | |
|      * unique! */
 | |
|     if (g_str_has_prefix (wanted_name, lowcase_type)) {
 | |
|       guint64 tmpcount =
 | |
|           g_ascii_strtoull (&wanted_name[strlen (lowcase_type)], NULL, 10);
 | |
| 
 | |
|       if (tmpcount > count) {
 | |
|         count = tmpcount + 1;
 | |
|         GST_DEBUG_OBJECT (self, "Using same naming %s but updated count to %i",
 | |
|             wanted_name, count);
 | |
|       } else if (tmpcount < count) {
 | |
|         /* FIXME: this can unexpectedly change names given by the user
 | |
|          * E.g. if "transition2" already exists, and a user then wants to
 | |
|          * set a GESTransition to have the name "transition-custom" or
 | |
|          * "transition 1 too many" then tmpcount would in fact be 0 or 1,
 | |
|          * and the name would then be changed to "transition3"! */
 | |
|         name = g_strdup_printf ("%s%d", lowcase_type, count);
 | |
|         count++;
 | |
|         GST_DEBUG_OBJECT (self, "Name %s already allocated, giving: %s instead"
 | |
|             " New count is %i", wanted_name, name, count);
 | |
|       } else {
 | |
|         count++;
 | |
|         GST_DEBUG_OBJECT (self, "Perfect name, just bumping object count");
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (name == NULL)
 | |
|       name = g_strdup (wanted_name);
 | |
|   }
 | |
| 
 | |
|   g_free (lowcase_type);
 | |
|   g_datalist_id_set_data (&object_name_counts, q, GINT_TO_POINTER (count));
 | |
| 
 | |
|   g_free (self->name);
 | |
|   self->name = name;
 | |
| }
 | |
| 
 | |
| /*********************************************
 | |
|  *       Internal and private helpers        *
 | |
|  *********************************************/
 | |
| 
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_peak_toplevel (GESTimelineElement * self)
 | |
| {
 | |
|   GESTimelineElement *toplevel = self;
 | |
| 
 | |
|   while (toplevel->parent)
 | |
|     toplevel = toplevel->parent;
 | |
| 
 | |
|   return toplevel;
 | |
| }
 | |
| 
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_get_copied_from (GESTimelineElement * self)
 | |
| {
 | |
|   GESTimelineElement *copied_from = self->priv->copied_from;
 | |
|   self->priv->copied_from = NULL;
 | |
|   return copied_from;
 | |
| }
 | |
| 
 | |
| GESTimelineElementFlags
 | |
| ges_timeline_element_flags (GESTimelineElement * self)
 | |
| {
 | |
|   return self->priv->flags;
 | |
| }
 | |
| 
 | |
| void
 | |
| ges_timeline_element_set_flags (GESTimelineElement * self,
 | |
|     GESTimelineElementFlags flags)
 | |
| {
 | |
|   self->priv->flags = flags;
 | |
| 
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| emit_deep_notify_in_idle (EmitDeepNotifyInIdleData * data)
 | |
| {
 | |
|   g_signal_emit (data->self, ges_timeline_element_signals[DEEP_NOTIFY], 0,
 | |
|       data->child, data->arg);
 | |
| 
 | |
|   gst_object_unref (data->child);
 | |
|   g_param_spec_unref (data->arg);
 | |
|   gst_object_unref (data->self);
 | |
|   g_slice_free (EmitDeepNotifyInIdleData, data);
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| static void
 | |
| child_prop_changed_cb (GObject * child, GParamSpec * arg,
 | |
|     GESTimelineElement * self)
 | |
| {
 | |
|   EmitDeepNotifyInIdleData *data;
 | |
| 
 | |
|   /* Emit "deep-notify" right away if in main thread */
 | |
|   if (g_main_context_acquire (g_main_context_default ())) {
 | |
|     g_main_context_release (g_main_context_default ());
 | |
|     g_signal_emit (self, ges_timeline_element_signals[DEEP_NOTIFY], 0,
 | |
|         child, arg);
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   data = g_slice_new (EmitDeepNotifyInIdleData);
 | |
| 
 | |
|   data->child = gst_object_ref (child);
 | |
|   data->arg = g_param_spec_ref (arg);
 | |
|   data->self = gst_object_ref (self);
 | |
| 
 | |
|   ges_idle_add ((GSourceFunc) emit_deep_notify_in_idle, data, NULL);
 | |
| }
 | |
| 
 | |
| static gboolean
 | |
| set_child_property_by_pspec (GESTimelineElement * self,
 | |
|     GParamSpec * pspec, const GValue * value, GError ** error)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
|   GESTimelineElement *setter = self;
 | |
|   ChildPropHandler *handler =
 | |
|       g_hash_table_lookup (self->priv->children_props, pspec);
 | |
| 
 | |
|   if (!handler) {
 | |
|     GST_ERROR_OBJECT (self, "The %s property doesn't exist", pspec->name);
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   if (handler->owner) {
 | |
|     klass = GES_TIMELINE_ELEMENT_GET_CLASS (handler->owner);
 | |
|     setter = handler->owner;
 | |
|   } else {
 | |
|     klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
|   }
 | |
| 
 | |
|   if (klass->set_child_property_full)
 | |
|     return klass->set_child_property_full (setter, handler->child, pspec,
 | |
|         value, error);
 | |
| 
 | |
|   g_assert (klass->set_child_property);
 | |
|   klass->set_child_property (setter, handler->child, pspec, (GValue *) value);
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| gboolean
 | |
| ges_timeline_element_add_child_property_full (GESTimelineElement * self,
 | |
|     GESTimelineElement * owner, GParamSpec * pspec, GObject * child)
 | |
| {
 | |
|   gchar *signame;
 | |
|   ChildPropHandler *handler;
 | |
| 
 | |
|   /* FIXME: allow the same pspec, provided the child is different. This
 | |
|    * is important for containers that may have duplicate children
 | |
|    * If this is changed, _remove_childs_child_property in ges-container.c
 | |
|    * should be changed to reflect this.
 | |
|    * We could hack around this by copying the pspec into a new instance
 | |
|    * of GParamSpec, but there is no such GLib method, and it would break
 | |
|    * the usage of get_..._from_pspec and set_..._from_pspec */
 | |
|   if (g_hash_table_contains (self->priv->children_props, pspec)) {
 | |
|     GST_INFO_OBJECT (self, "Child property already exists: %s", pspec->name);
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "Adding child property: %" GST_PTR_FORMAT "::%s",
 | |
|       child, pspec->name);
 | |
| 
 | |
|   signame = g_strconcat ("notify::", pspec->name, NULL);
 | |
|   handler = (ChildPropHandler *) g_slice_new0 (ChildPropHandler);
 | |
|   handler->self = self;
 | |
|   if (child == G_OBJECT (self) || child == G_OBJECT (owner))
 | |
|     handler->child = child;
 | |
|   else
 | |
|     handler->child = gst_object_ref (child);
 | |
|   handler->owner = owner;
 | |
|   handler->handler_id =
 | |
|       g_signal_connect (child, signame, G_CALLBACK (child_prop_changed_cb),
 | |
|       self);
 | |
|   g_hash_table_insert (self->priv->children_props, g_param_spec_ref (pspec),
 | |
|       handler);
 | |
| 
 | |
|   g_signal_emit (self, ges_timeline_element_signals[CHILD_PROPERTY_ADDED], 0,
 | |
|       child, pspec);
 | |
| 
 | |
|   g_free (signame);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| GObject *
 | |
| ges_timeline_element_get_child_from_child_property (GESTimelineElement * self,
 | |
|     GParamSpec * pspec)
 | |
| {
 | |
|   ChildPropHandler *handler =
 | |
|       g_hash_table_lookup (self->priv->children_props, pspec);
 | |
|   if (handler)
 | |
|     return handler->child;
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*********************************************
 | |
|  *            API implementation             *
 | |
|  *********************************************/
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_parent:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @parent (nullable): New parent of @self
 | |
|  *
 | |
|  * Sets the #GESTimelineElement:parent for the element.
 | |
|  *
 | |
|  * This is used internally and you should normally not call this. A
 | |
|  * #GESContainer will set the #GESTimelineElement:parent of its children
 | |
|  * in ges_container_add() and ges_container_remove().
 | |
|  *
 | |
|  * Note, if @parent is not %NULL, @self must not already have a parent
 | |
|  * set. Therefore, if you wish to switch parents, you will need to call
 | |
|  * this function twice: first to set the parent to %NULL, and then to the
 | |
|  * new parent.
 | |
|  *
 | |
|  * If @parent is not %NULL, you must ensure it already has a
 | |
|  * (non-floating) reference to @self before calling this.
 | |
|  *
 | |
|  * Returns: %TRUE if @parent could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_parent (GESTimelineElement * self,
 | |
|     GESTimelineElement * parent)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (parent == NULL
 | |
|       || GES_IS_TIMELINE_ELEMENT (parent), FALSE);
 | |
| 
 | |
|   if (self == parent) {
 | |
|     GST_INFO_OBJECT (self, "Trying to add %p in itself, not a good idea!",
 | |
|         self);
 | |
|     /* FIXME: why are we sinking and then unreffing self when we do not
 | |
|      * own it? */
 | |
|     gst_object_ref_sink (self);
 | |
|     gst_object_unref (self);
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "set parent to %" GST_PTR_FORMAT, parent);
 | |
| 
 | |
|   if (self->parent != NULL && parent != NULL)
 | |
|     goto had_parent;
 | |
| 
 | |
|   if (GES_TIMELINE_ELEMENT_GET_CLASS (self)->set_parent) {
 | |
|     if (!GES_TIMELINE_ELEMENT_GET_CLASS (self)->set_parent (self, parent))
 | |
|       return FALSE;
 | |
|   }
 | |
| 
 | |
|   self->parent = parent;
 | |
| 
 | |
|   g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_PARENT]);
 | |
|   return TRUE;
 | |
| 
 | |
|   /* ERROR handling */
 | |
| had_parent:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "set parent failed, object already had a parent");
 | |
|     /* FIXME: why are we sinking and then unreffing self when we do not
 | |
|      * own it? */
 | |
|     gst_object_ref_sink (self);
 | |
|     gst_object_unref (self);
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_parent:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:parent for the element.
 | |
|  *
 | |
|  * Returns: (transfer full) (nullable): The parent of @self, or %NULL if
 | |
|  * @self has no parent.
 | |
|  */
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_get_parent (GESTimelineElement * self)
 | |
| {
 | |
|   GESTimelineElement *result = NULL;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   result = self->parent;
 | |
|   if (G_LIKELY (result))
 | |
|     gst_object_ref (result);
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_timeline:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @timeline (nullable): The #GESTimeline @self should be in
 | |
|  *
 | |
|  * Sets the #GESTimelineElement:timeline of the element.
 | |
|  *
 | |
|  * This is used internally and you should normally not call this. A
 | |
|  * #GESClip will have its #GESTimelineElement:timeline set through its
 | |
|  * #GESLayer. A #GESTrack will similarly take care of setting the
 | |
|  * #GESTimelineElement:timeline of its #GESTrackElement-s. A #GESGroup
 | |
|  * will adopt the same #GESTimelineElement:timeline as its children.
 | |
|  *
 | |
|  * If @timeline is %NULL, this will stop its current
 | |
|  * #GESTimelineElement:timeline from tracking it, otherwise @timeline will
 | |
|  * start tracking @self. Note, in the latter case, @self must not already
 | |
|  * have a timeline set. Therefore, if you wish to switch timelines, you
 | |
|  * will need to call this function twice: first to set the timeline to
 | |
|  * %NULL, and then to the new timeline.
 | |
|  *
 | |
|  * Returns: %TRUE if @timeline could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_timeline (GESTimelineElement * self,
 | |
|     GESTimeline * timeline)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (timeline == NULL || GES_IS_TIMELINE (timeline), FALSE);
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "set timeline to %" GST_PTR_FORMAT, timeline);
 | |
| 
 | |
|   if (self->timeline == timeline)
 | |
|     return TRUE;
 | |
| 
 | |
|   if (timeline != NULL && G_UNLIKELY (self->timeline != NULL))
 | |
|     goto had_timeline;
 | |
| 
 | |
|   if (timeline == NULL) {
 | |
|     if (self->timeline) {
 | |
|       if (!timeline_remove_element (self->timeline, self)) {
 | |
|         GST_INFO_OBJECT (self, "Could not remove from"
 | |
|             " currently set timeline %" GST_PTR_FORMAT, self->timeline);
 | |
|         return FALSE;
 | |
|       }
 | |
|     }
 | |
|   } else {
 | |
|     if (!timeline_add_element (timeline, self)) {
 | |
|       GST_INFO_OBJECT (self, "Could not add to timeline %" GST_PTR_FORMAT,
 | |
|           self);
 | |
|       return FALSE;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   self->timeline = timeline;
 | |
| 
 | |
|   g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_TIMELINE]);
 | |
|   return TRUE;
 | |
| 
 | |
|   /* ERROR handling */
 | |
| had_timeline:
 | |
|   {
 | |
|     GST_DEBUG_OBJECT (self, "set timeline failed, object already had a "
 | |
|         "timeline");
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_timeline:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:timeline for the element.
 | |
|  *
 | |
|  * Returns: (transfer full) (nullable): The timeline of @self, or %NULL
 | |
|  * if @self has no timeline.
 | |
|  */
 | |
| GESTimeline *
 | |
| ges_timeline_element_get_timeline (GESTimelineElement * self)
 | |
| {
 | |
|   GESTimeline *result = NULL;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   result = self->timeline;
 | |
|   if (G_LIKELY (result))
 | |
|     gst_object_ref (result);
 | |
| 
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_start:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @start: The desired start position of the element in its timeline
 | |
|  *
 | |
|  * Sets #GESTimelineElement:start for the element. If the element has a
 | |
|  * parent, this will also move its siblings with the same shift.
 | |
|  *
 | |
|  * Whilst the element is part of a #GESTimeline, this is the same as
 | |
|  * editing the element with ges_timeline_element_edit() under
 | |
|  * #GES_EDIT_MODE_NORMAL with #GES_EDGE_NONE. In particular, the
 | |
|  * #GESTimelineElement:start of the element may be snapped to a different
 | |
|  * timeline time from the one given. In addition, setting may fail if it
 | |
|  * would place the timeline in an unsupported configuration.
 | |
|  *
 | |
|  * Returns: %TRUE if @start could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_start (GESTimelineElement * self, GstClockTime start)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
|   GESTimelineElement *toplevel_container, *parent;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (start), FALSE);
 | |
| 
 | |
|   if (self->start == start)
 | |
|     return TRUE;
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "current start: %" GST_TIME_FORMAT
 | |
|       " new start: %" GST_TIME_FORMAT,
 | |
|       GST_TIME_ARGS (GES_TIMELINE_ELEMENT_START (self)), GST_TIME_ARGS (start));
 | |
| 
 | |
|   if (self->timeline && !GES_TIMELINE_ELEMENT_BEING_EDITED (self))
 | |
|     return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_NORMAL,
 | |
|         GES_EDGE_NONE, start);
 | |
| 
 | |
|   toplevel_container = ges_timeline_element_peak_toplevel (self);
 | |
|   parent = self->parent;
 | |
| 
 | |
|   /* FIXME This should not belong to GESTimelineElement */
 | |
|   /* only check if no timeline, otherwise the timeline-tree will handle this
 | |
|    * check */
 | |
|   if (!self->timeline && toplevel_container &&
 | |
|       ((gint64) (_START (toplevel_container) + start - _START (self))) < 0 &&
 | |
|       parent
 | |
|       && GES_CONTAINER (parent)->children_control_mode == GES_CHILDREN_UPDATE) {
 | |
|     GST_INFO_OBJECT (self,
 | |
|         "Can not move the object as it would imply its "
 | |
|         "container to have a negative start value");
 | |
| 
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
|   if (klass->set_start) {
 | |
|     gint res = klass->set_start (self, start);
 | |
|     if (res == FALSE)
 | |
|       return FALSE;
 | |
|     if (res == TRUE) {
 | |
|       self->start = start;
 | |
|       g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_START]);
 | |
|     }
 | |
| 
 | |
|     GST_DEBUG_OBJECT (self, "New start: %" GST_TIME_FORMAT,
 | |
|         GST_TIME_ARGS (GES_TIMELINE_ELEMENT_START (self)));
 | |
| 
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   GST_WARNING_OBJECT (self, "No set_start virtual method implementation"
 | |
|       " on class %s. Can not set start %" GST_TIME_FORMAT,
 | |
|       G_OBJECT_CLASS_NAME (klass), GST_TIME_ARGS (start));
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_inpoint:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @inpoint: The in-point, in internal time coordinates
 | |
|  *
 | |
|  * Sets #GESTimelineElement:in-point for the element. If the new in-point
 | |
|  * is above the current #GESTimelineElement:max-duration of the element,
 | |
|  * this method will fail.
 | |
|  *
 | |
|  * Returns: %TRUE if @inpoint could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_inpoint (GESTimelineElement * self,
 | |
|     GstClockTime inpoint)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "current inpoint: %" GST_TIME_FORMAT
 | |
|       " new inpoint: %" GST_TIME_FORMAT, GST_TIME_ARGS (self->inpoint),
 | |
|       GST_TIME_ARGS (inpoint));
 | |
| 
 | |
|   if (G_UNLIKELY (inpoint == self->inpoint))
 | |
|     return TRUE;
 | |
| 
 | |
|   if (GES_CLOCK_TIME_IS_LESS (self->maxduration, inpoint)) {
 | |
|     GST_WARNING_OBJECT (self, "Can not set an in-point of %" GST_TIME_FORMAT
 | |
|         " because it exceeds the element's max-duration: %" GST_TIME_FORMAT,
 | |
|         GST_TIME_ARGS (inpoint), GST_TIME_ARGS (self->maxduration));
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->set_inpoint) {
 | |
|     /* FIXME: Could we instead use g_object_freeze_notify() to prevent
 | |
|      * duplicate notify signals? Rather than relying on the return value
 | |
|      * being -1 for setting that succeeds but does not want a notify
 | |
|      * signal because it will call this method on itself a second time. */
 | |
|     if (!klass->set_inpoint (self, inpoint))
 | |
|       return FALSE;
 | |
| 
 | |
|     self->inpoint = inpoint;
 | |
|     g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_INPOINT]);
 | |
| 
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "No set_inpoint virtual method implementation"
 | |
|       " on class %s. Can not set inpoint %" GST_TIME_FORMAT,
 | |
|       G_OBJECT_CLASS_NAME (klass), GST_TIME_ARGS (inpoint));
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_max_duration:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @maxduration: The maximum duration, in internal time coordinates
 | |
|  *
 | |
|  * Sets #GESTimelineElement:max-duration for the element. If the new
 | |
|  * maximum duration is below the current #GESTimelineElement:in-point of
 | |
|  * the element, this method will fail.
 | |
|  *
 | |
|  * Returns: %TRUE if @maxduration could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_max_duration (GESTimelineElement * self,
 | |
|     GstClockTime maxduration)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "current max-duration: %" GST_TIME_FORMAT
 | |
|       " new max-duration: %" GST_TIME_FORMAT,
 | |
|       GST_TIME_ARGS (self->maxduration), GST_TIME_ARGS (maxduration));
 | |
| 
 | |
|   if (G_UNLIKELY (maxduration == self->maxduration))
 | |
|     return TRUE;
 | |
| 
 | |
|   if (GES_CLOCK_TIME_IS_LESS (maxduration, self->inpoint)) {
 | |
|     GST_WARNING_OBJECT (self, "Can not set a max-duration of %"
 | |
|         GST_TIME_FORMAT " because it lies below the element's in-point: %"
 | |
|         GST_TIME_FORMAT, GST_TIME_ARGS (maxduration),
 | |
|         GST_TIME_ARGS (self->inpoint));
 | |
|     return FALSE;
 | |
|   }
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->set_max_duration) {
 | |
|     if (!klass->set_max_duration (self, maxduration))
 | |
|       return FALSE;
 | |
|     self->maxduration = maxduration;
 | |
|     g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_MAX_DURATION]);
 | |
| 
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "No set_max_duration virtual method implementation"
 | |
|       " on class %s. Can not set max-duration  %" GST_TIME_FORMAT,
 | |
|       G_OBJECT_CLASS_NAME (klass), GST_TIME_ARGS (maxduration));
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_duration:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @duration: The desired duration in its timeline
 | |
|  *
 | |
|  * Sets #GESTimelineElement:duration for the element.
 | |
|  *
 | |
|  * Whilst the element is part of a #GESTimeline, this is the same as
 | |
|  * editing the element with ges_timeline_element_edit() under
 | |
|  * #GES_EDIT_MODE_TRIM with #GES_EDGE_END. In particular, the
 | |
|  * #GESTimelineElement:duration of the element may be snapped to a
 | |
|  * different timeline time difference from the one given. In addition,
 | |
|  * setting may fail if it would place the timeline in an unsupported
 | |
|  * configuration, or the element does not have enough internal content to
 | |
|  * last the desired duration.
 | |
|  *
 | |
|  * Returns: %TRUE if @duration could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_duration (GESTimelineElement * self,
 | |
|     GstClockTime duration)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   if (duration == self->duration)
 | |
|     return TRUE;
 | |
| 
 | |
|   if (self->timeline && !GES_TIMELINE_ELEMENT_BEING_EDITED (self))
 | |
|     return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_TRIM,
 | |
|         GES_EDGE_END, self->start + duration);
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "current duration: %" GST_TIME_FORMAT
 | |
|       " new duration: %" GST_TIME_FORMAT,
 | |
|       GST_TIME_ARGS (GES_TIMELINE_ELEMENT_DURATION (self)),
 | |
|       GST_TIME_ARGS (duration));
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
|   if (klass->set_duration) {
 | |
|     gint res = klass->set_duration (self, duration);
 | |
|     if (res == FALSE)
 | |
|       return FALSE;
 | |
|     if (res == TRUE) {
 | |
|       self->duration = duration;
 | |
|       g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_DURATION]);
 | |
|     }
 | |
| 
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   GST_WARNING_OBJECT (self, "No set_duration virtual method implementation"
 | |
|       " on class %s. Can not set duration %" GST_TIME_FORMAT,
 | |
|       G_OBJECT_CLASS_NAME (klass), GST_TIME_ARGS (duration));
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_start:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:start for the element.
 | |
|  *
 | |
|  * Returns: The start of @self (in nanoseconds).
 | |
|  */
 | |
| GstClockTime
 | |
| ges_timeline_element_get_start (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), GST_CLOCK_TIME_NONE);
 | |
| 
 | |
|   return self->start;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_inpoint:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:in-point for the element.
 | |
|  *
 | |
|  * Returns: The in-point of @self (in nanoseconds).
 | |
|  */
 | |
| GstClockTime
 | |
| ges_timeline_element_get_inpoint (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), GST_CLOCK_TIME_NONE);
 | |
| 
 | |
|   return self->inpoint;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_duration:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:duration for the element.
 | |
|  *
 | |
|  * Returns: The duration of @self (in nanoseconds).
 | |
|  */
 | |
| GstClockTime
 | |
| ges_timeline_element_get_duration (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), GST_CLOCK_TIME_NONE);
 | |
| 
 | |
|   return self->duration;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_max_duration:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:max-duration for the element.
 | |
|  *
 | |
|  * Returns: The max-duration of @self (in nanoseconds).
 | |
|  */
 | |
| GstClockTime
 | |
| ges_timeline_element_get_max_duration (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), GST_CLOCK_TIME_NONE);
 | |
| 
 | |
|   return self->maxduration;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_priority:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:priority for the element.
 | |
|  *
 | |
|  * Returns: The priority of @self.
 | |
|  */
 | |
| guint32
 | |
| ges_timeline_element_get_priority (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), 0);
 | |
| 
 | |
|   return self->priority;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_priority:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @priority: The priority
 | |
|  *
 | |
|  * Sets the priority of the element within the containing layer.
 | |
|  *
 | |
|  * Deprecated:1.10: All priority management is done by GES itself now.
 | |
|  * To set #GESEffect priorities #ges_clip_set_top_effect_index should
 | |
|  * be used.
 | |
|  *
 | |
|  * Returns: %TRUE if @priority could be set for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_priority (GESTimelineElement * self, guint32 priority)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "current priority: %d new priority: %d",
 | |
|       self->priority, priority);
 | |
| 
 | |
|   if (klass->set_priority) {
 | |
|     gboolean res = klass->set_priority (self, priority);
 | |
|     if (res) {
 | |
|       self->priority = priority;
 | |
|       g_object_notify_by_pspec (G_OBJECT (self), properties[PROP_PRIORITY]);
 | |
|     }
 | |
| 
 | |
|     return res;
 | |
|   }
 | |
| 
 | |
|   GST_WARNING_OBJECT (self, "No set_priority virtual method implementation"
 | |
|       " on class %s. Can not set priority %d", G_OBJECT_CLASS_NAME (klass),
 | |
|       priority);
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_ripple:
 | |
|  * @self: The #GESTimelineElement to ripple
 | |
|  * @start: The new start time of @self in ripple mode
 | |
|  *
 | |
|  * Edits the start time of an element within its timeline in ripple mode.
 | |
|  * See ges_timeline_element_edit() with #GES_EDIT_MODE_RIPPLE and
 | |
|  * #GES_EDGE_NONE.
 | |
|  *
 | |
|  * Returns: %TRUE if the ripple edit of @self completed, %FALSE on
 | |
|  * failure.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_ripple (GESTimelineElement * self, GstClockTime start)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (start), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->ripple)
 | |
|     return klass->ripple (self, start);
 | |
| 
 | |
|   return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_RIPPLE,
 | |
|       GES_EDGE_NONE, start);
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_ripple_end:
 | |
|  * @self: The #GESTimelineElement to ripple
 | |
|  * @end: The new end time of @self in ripple mode
 | |
|  *
 | |
|  * Edits the end time of an element within its timeline in ripple mode.
 | |
|  * See ges_timeline_element_edit() with #GES_EDIT_MODE_RIPPLE and
 | |
|  * #GES_EDGE_END.
 | |
|  *
 | |
|  * Returns: %TRUE if the ripple edit of @self completed, %FALSE on
 | |
|  * failure.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_ripple_end (GESTimelineElement * self, GstClockTime end)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (end), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->ripple_end)
 | |
|     return klass->ripple_end (self, end);
 | |
| 
 | |
|   return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_RIPPLE,
 | |
|       GES_EDGE_END, end);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_roll_start:
 | |
|  * @self: The #GESTimelineElement to roll
 | |
|  * @start: The new start time of @self in roll mode
 | |
|  *
 | |
|  * Edits the start time of an element within its timeline in roll mode.
 | |
|  * See ges_timeline_element_edit() with #GES_EDIT_MODE_ROLL and
 | |
|  * #GES_EDGE_START.
 | |
|  *
 | |
|  * Returns: %TRUE if the roll edit of @self completed, %FALSE on failure.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_roll_start (GESTimelineElement * self, GstClockTime start)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (start), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->roll_start)
 | |
|     return klass->roll_start (self, start);
 | |
| 
 | |
|   return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_ROLL,
 | |
|       GES_EDGE_START, start);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_roll_end:
 | |
|  * @self: The #GESTimelineElement to roll
 | |
|  * @end: The new end time of @self in roll mode
 | |
|  *
 | |
|  * Edits the end time of an element within its timeline in roll mode.
 | |
|  * See ges_timeline_element_edit() with #GES_EDIT_MODE_ROLL and
 | |
|  * #GES_EDGE_END.
 | |
|  *
 | |
|  * Returns: %TRUE if the roll edit of @self completed, %FALSE on failure.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_roll_end (GESTimelineElement * self, GstClockTime end)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (end), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->roll_end)
 | |
|     return klass->roll_end (self, end);
 | |
| 
 | |
|   return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_ROLL,
 | |
|       GES_EDGE_END, end);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_trim:
 | |
|  * @self: The #GESTimelineElement to trim
 | |
|  * @start: The new start time of @self in trim mode
 | |
|  *
 | |
|  * Edits the start time of an element within its timeline in trim mode.
 | |
|  * See ges_timeline_element_edit() with #GES_EDIT_MODE_TRIM and
 | |
|  * #GES_EDGE_START.
 | |
|  *
 | |
|  * Returns: %TRUE if the trim edit of @self completed, %FALSE on failure.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_trim (GESTimelineElement * self, GstClockTime start)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (start), FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (klass->trim)
 | |
|     return klass->trim (self, start);
 | |
| 
 | |
|   return ges_timeline_element_edit (self, NULL, -1, GES_EDIT_MODE_TRIM,
 | |
|       GES_EDGE_START, start);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_copy:
 | |
|  * @self: The #GESTimelineElement to copy
 | |
|  * @deep: Whether the copy is needed for pasting
 | |
|  *
 | |
|  * Create a copy of @self. All the properties of @self are copied into
 | |
|  * a new element, with the exception of #GESTimelineElement:parent,
 | |
|  * #GESTimelineElement:timeline and #GESTimelineElement:name. Other data,
 | |
|  * such the list of a #GESContainer's children, is **not** copied.
 | |
|  *
 | |
|  * If @deep is %TRUE, then the new element is prepared so that it can be
 | |
|  * used in ges_timeline_element_paste() or ges_timeline_paste_element().
 | |
|  * In the case of copying a #GESContainer, this ensures that the children
 | |
|  * of @self will also be pasted. The new element should not be used for
 | |
|  * anything else and can only be used **once** in a pasting operation. In
 | |
|  * particular, the new element itself is not an actual 'deep' copy of
 | |
|  * @self, but should be thought of as an intermediate object used for a
 | |
|  * single paste operation.
 | |
|  *
 | |
|  * Returns: (transfer floating): The newly create element,
 | |
|  * copied from @self.
 | |
|  */
 | |
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS;       /* Start ignoring GParameter deprecation */
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_copy (GESTimelineElement * self, gboolean deep)
 | |
| {
 | |
|   GESAsset *asset;
 | |
|   GParamSpec **specs;
 | |
|   GESTimelineElementClass *klass;
 | |
|   guint n, n_specs;
 | |
| 
 | |
|   GESTimelineElement *ret = NULL;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   specs = g_object_class_list_properties (G_OBJECT_GET_CLASS (self), &n_specs);
 | |
| 
 | |
|   asset = ges_extractable_get_asset (GES_EXTRACTABLE (self));
 | |
|   g_assert (asset);
 | |
|   ret = GES_TIMELINE_ELEMENT (ges_asset_extract (asset, NULL));
 | |
|   for (n = 0; n < n_specs; ++n) {
 | |
|     /* We do not want the timeline or the name to be copied */
 | |
|     if (g_strcmp0 (specs[n]->name, "parent") &&
 | |
|         g_strcmp0 (specs[n]->name, "timeline") &&
 | |
|         g_strcmp0 (specs[n]->name, "name") &&
 | |
|         (specs[n]->flags & G_PARAM_READWRITE) == G_PARAM_READWRITE &&
 | |
|         (specs[n]->flags & G_PARAM_CONSTRUCT_ONLY) == 0) {
 | |
|       GValue v = G_VALUE_INIT;
 | |
|       g_value_init (&v, specs[n]->value_type);
 | |
|       g_object_get_property (G_OBJECT (self), specs[n]->name, &v);
 | |
| 
 | |
|       g_object_set_property (G_OBJECT (ret), specs[n]->name, &v);
 | |
|       g_value_reset (&v);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   g_free (specs);
 | |
|   if (deep) {
 | |
|     if (klass->deep_copy)
 | |
|       klass->deep_copy (self, ret);
 | |
|     else
 | |
|       GST_WARNING_OBJECT (self, "No deep_copy virtual method implementation"
 | |
|           " on class %s. Can not finish the copy", G_OBJECT_CLASS_NAME (klass));
 | |
|   }
 | |
| 
 | |
|   if (deep) {
 | |
|     ret->priv->copied_from = gst_object_ref (self);
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| G_GNUC_END_IGNORE_DEPRECATIONS; /* End ignoring GParameter deprecation */
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_toplevel_parent:
 | |
|  * @self: The #GESTimelineElement to get the toplevel parent from
 | |
|  *
 | |
|  * Gets the toplevel #GESTimelineElement:parent of the element.
 | |
|  *
 | |
|  * Returns: (transfer full): The toplevel parent of @self.
 | |
|  */
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_get_toplevel_parent (GESTimelineElement * self)
 | |
| {
 | |
|   GESTimelineElement *toplevel;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   toplevel = ges_timeline_element_peak_toplevel (self);
 | |
| 
 | |
|   return gst_object_ref (toplevel);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_name:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the #GESTimelineElement:name for the element.
 | |
|  *
 | |
|  * Returns: (transfer full): The name of @self.
 | |
|  */
 | |
| gchar *
 | |
| ges_timeline_element_get_name (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   return g_strdup (self->name);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_name:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @name: (allow-none): The name @self should take
 | |
|  *
 | |
|  * Sets the #GESTimelineElement:name for the element. If %NULL is given
 | |
|  * for @name, then the library will instead generate a new name based on
 | |
|  * the type name of the element, such as the name "uriclip3" for a
 | |
|  * #GESUriClip, and will set that name instead.
 | |
|  *
 | |
|  * If @self already has a #GESTimelineElement:timeline, you should not
 | |
|  * call this function with @name set to %NULL.
 | |
|  *
 | |
|  * You should ensure that, within each #GESTimeline, every element has a
 | |
|  * unique name. If you call this function with @name as %NULL, then
 | |
|  * the library should ensure that the set generated name is unique from
 | |
|  * previously **generated** names. However, if you choose a @name that
 | |
|  * interferes with the naming conventions of the library, the library will
 | |
|  * attempt to ensure that the generated names will not conflict with the
 | |
|  * chosen name, which may lead to a different name being set instead, but
 | |
|  * the uniqueness between generated and user-chosen names is not
 | |
|  * guaranteed.
 | |
|  *
 | |
|  * Returns: %TRUE if @name or a generated name for @self could be set.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_name (GESTimelineElement * self, const gchar * name)
 | |
| {
 | |
|   gboolean result = TRUE, readd_to_timeline = FALSE;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   if (name != NULL && !g_strcmp0 (name, self->name)) {
 | |
|     GST_DEBUG_OBJECT (self, "Same name!");
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   /* parented objects cannot be renamed */
 | |
|   if (self->timeline != NULL && name) {
 | |
|     GESTimelineElement *tmp = ges_timeline_get_element (self->timeline, name);
 | |
| 
 | |
|     /* FIXME: if tmp == self then this means that we setting the name of
 | |
|      * self to its existing name. There is no need to throw an error */
 | |
|     if (tmp) {
 | |
|       gst_object_unref (tmp);
 | |
|       goto had_timeline;
 | |
|     }
 | |
| 
 | |
|     timeline_remove_element (self->timeline, self);
 | |
|     readd_to_timeline = TRUE;
 | |
|   }
 | |
|   /* FIXME: if self already has a timeline and name is NULL, then it also
 | |
|    * needs to be re-added to the timeline (or, at least its entry in
 | |
|    * timeline->priv->all_elements needs its key to be updated) using the
 | |
|    * new generated name */
 | |
| 
 | |
|   _set_name (self, name);
 | |
| 
 | |
|   /* FIXME: the set name may not always be unique in a given timeline, see
 | |
|    * _set_name(). This can cause timeline_add_element to fail! */
 | |
|   if (readd_to_timeline)
 | |
|     timeline_add_element (self->timeline, self);
 | |
| 
 | |
|   return result;
 | |
| 
 | |
|   /* error */
 | |
| had_timeline:
 | |
|   {
 | |
|     /* FIXME: message is misleading. We are here if some other object in
 | |
|      * the timeline was added under @name (see above) */
 | |
|     GST_WARNING ("Object %s already in a timeline can't be renamed to %s",
 | |
|         self->name, name);
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_add_child_property:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @pspec: The specification for the property to add
 | |
|  * @child: The #GstObject who the property belongs to
 | |
|  *
 | |
|  * Register a property of a child of the element to allow it to be
 | |
|  * written with ges_timeline_element_set_child_property() and read with
 | |
|  * ges_timeline_element_get_child_property(). A change in the property
 | |
|  * will also appear in the #GESTimelineElement::deep-notify signal.
 | |
|  *
 | |
|  * @pspec should be unique from other children properties that have been
 | |
|  * registered on @self.
 | |
|  *
 | |
|  * Returns: %TRUE if the property was successfully registered.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_add_child_property (GESTimelineElement * self,
 | |
|     GParamSpec * pspec, GObject * child)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
 | |
|   g_return_val_if_fail (G_IS_OBJECT (child), FALSE);
 | |
| 
 | |
|   return ges_timeline_element_add_child_property_full (self, NULL, pspec,
 | |
|       child);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_child_property_by_pspec:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @pspec: The specification of a registered child property to get
 | |
|  * @value: (out): The return location for the value
 | |
|  *
 | |
|  * Gets the property of a child of the element. Specifically, the property
 | |
|  * corresponding to the @pspec used in
 | |
|  * ges_timeline_element_add_child_property() is copied into @value.
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_get_child_property_by_pspec (GESTimelineElement * self,
 | |
|     GParamSpec * pspec, GValue * value)
 | |
| {
 | |
|   ChildPropHandler *handler;
 | |
| 
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
|   g_return_if_fail (G_IS_PARAM_SPEC (pspec));
 | |
| 
 | |
|   handler = g_hash_table_lookup (self->priv->children_props, pspec);
 | |
|   if (!handler)
 | |
|     goto not_found;
 | |
| 
 | |
|   g_object_get_property (G_OBJECT (handler->child), pspec->name, value);
 | |
| 
 | |
|   return;
 | |
| 
 | |
| not_found:
 | |
|   {
 | |
|     GST_ERROR_OBJECT (self, "The %s property doesn't exist", pspec->name);
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_child_property_by_pspec:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @pspec: The specification of a registered child property to set
 | |
|  * @value: The value to set the property to
 | |
|  *
 | |
|  * Sets the property of a child of the element. Specifically, the property
 | |
|  * corresponding to the @pspec used in
 | |
|  * ges_timeline_element_add_child_property() is set to @value.
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_set_child_property_by_pspec (GESTimelineElement * self,
 | |
|     GParamSpec * pspec, const GValue * value)
 | |
| {
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
|   g_return_if_fail (G_IS_PARAM_SPEC (pspec));
 | |
| 
 | |
|   set_child_property_by_pspec (self, pspec, value, NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_child_property_full:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @property_name: The name of the child property to set
 | |
|  * @value: The value to set the property to
 | |
|  * @error: (nullable): Return location for an error
 | |
|  *
 | |
|  * Sets the property of a child of the element.
 | |
|  *
 | |
|  * @property_name can either be in the format "prop-name" or
 | |
|  * "TypeName::prop-name", where "prop-name" is the name of the property
 | |
|  * to set (as used in g_object_set()), and "TypeName" is the type name of
 | |
|  * the child (as returned by G_OBJECT_TYPE_NAME()). The latter format is
 | |
|  * useful when two children of different types share the same property
 | |
|  * name.
 | |
|  *
 | |
|  * The first child found with the given "prop-name" property that was
 | |
|  * registered with ges_timeline_element_add_child_property() (and of the
 | |
|  * type "TypeName", if it was given) will have the corresponding
 | |
|  * property set to @value. Other children that may have also matched the
 | |
|  * property name (and type name) are left unchanged!
 | |
|  *
 | |
|  * Returns: %TRUE if the property was found and set.
 | |
|  * Since: 1.18
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_child_property_full (GESTimelineElement * self,
 | |
|     const gchar * property_name, const GValue * value, GError ** error)
 | |
| {
 | |
|   GParamSpec *pspec;
 | |
|   GObject *child;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (!error || !*error, FALSE);
 | |
| 
 | |
|   if (!ges_timeline_element_lookup_child (self, property_name, &child, &pspec))
 | |
|     goto not_found;
 | |
| 
 | |
|   return set_child_property_by_pspec (self, pspec, value, error);
 | |
| 
 | |
| not_found:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "The %s property doesn't exist", property_name);
 | |
| 
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_child_property:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @property_name: The name of the child property to set
 | |
|  * @value: The value to set the property to
 | |
|  *
 | |
|  * See ges_timeline_element_set_child_property_full(), which also gives an
 | |
|  * error.
 | |
|  *
 | |
|  * Note that ges_timeline_element_set_child_properties() may be more
 | |
|  * convenient for C programming.
 | |
|  *
 | |
|  * Returns: %TRUE if the property was found and set.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_set_child_property (GESTimelineElement * self,
 | |
|     const gchar * property_name, const GValue * value)
 | |
| {
 | |
|   return ges_timeline_element_set_child_property_full (self, property_name,
 | |
|       value, NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_child_property:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @property_name: The name of the child property to get
 | |
|  * @value: (out): The return location for the value
 | |
|  *
 | |
|  * Gets the property of a child of the element.
 | |
|  *
 | |
|  * @property_name can either be in the format "prop-name" or
 | |
|  * "TypeName::prop-name", where "prop-name" is the name of the property
 | |
|  * to get (as used in g_object_get()), and "TypeName" is the type name of
 | |
|  * the child (as returned by G_OBJECT_TYPE_NAME()). The latter format is
 | |
|  * useful when two children of different types share the same property
 | |
|  * name.
 | |
|  *
 | |
|  * The first child found with the given "prop-name" property that was
 | |
|  * registered with ges_timeline_element_add_child_property() (and of the
 | |
|  * type "TypeName", if it was given) will have the corresponding
 | |
|  * property copied into @value.
 | |
|  *
 | |
|  * Note that ges_timeline_element_get_child_properties() may be more
 | |
|  * convenient for C programming.
 | |
|  *
 | |
|  * Returns: %TRUE if the property was found and copied to @value.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_get_child_property (GESTimelineElement * self,
 | |
|     const gchar * property_name, GValue * value)
 | |
| {
 | |
|   GParamSpec *pspec;
 | |
|   GObject *child;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
| 
 | |
|   if (!ges_timeline_element_lookup_child (self, property_name, &child, &pspec))
 | |
|     goto not_found;
 | |
| 
 | |
|   /* FIXME: since GLib 2.60, g_object_get_property() will automatically
 | |
|    * initialize the type */
 | |
|   if (G_VALUE_TYPE (value) == G_TYPE_INVALID)
 | |
|     g_value_init (value, pspec->value_type);
 | |
| 
 | |
|   g_object_get_property (child, pspec->name, value);
 | |
| 
 | |
|   gst_object_unref (child);
 | |
|   g_param_spec_unref (pspec);
 | |
| 
 | |
|   return TRUE;
 | |
| 
 | |
| not_found:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "The %s property doesn't exist", property_name);
 | |
| 
 | |
|     return FALSE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_lookup_child:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @prop_name: The name of a child property
 | |
|  * @child: (out) (optional) (transfer full): The return location for the
 | |
|  * found child
 | |
|  * @pspec: (out) (optional) (transfer full): The return location for the
 | |
|  * specification of the child property
 | |
|  *
 | |
|  * Looks up a child property of the element.
 | |
|  *
 | |
|  * @prop_name can either be in the format "prop-name" or
 | |
|  * "TypeName::prop-name", where "prop-name" is the name of the property
 | |
|  * to look up (as used in g_object_get()), and "TypeName" is the type name
 | |
|  * of the child (as returned by G_OBJECT_TYPE_NAME()). The latter format is
 | |
|  * useful when two children of different types share the same property
 | |
|  * name.
 | |
|  *
 | |
|  * The first child found with the given "prop-name" property that was
 | |
|  * registered with ges_timeline_element_add_child_property() (and of the
 | |
|  * type "TypeName", if it was given) will be passed to @child, and the
 | |
|  * registered specification of this property will be passed to @pspec.
 | |
|  *
 | |
|  * Returns: %TRUE if a child corresponding to the property was found, in
 | |
|  * which case @child and @pspec are set.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_lookup_child (GESTimelineElement * self,
 | |
|     const gchar * prop_name, GObject ** child, GParamSpec ** pspec)
 | |
| {
 | |
|   GESTimelineElementClass *class;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   class = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
|   g_return_val_if_fail (class->lookup_child, FALSE);
 | |
| 
 | |
|   return class->lookup_child (self, prop_name, child, pspec);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_child_property_valist:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @first_property_name: The name of the first child property to set
 | |
|  * @var_args: The value for the first property, followed optionally by more
 | |
|  * name/value pairs, followed by %NULL
 | |
|  *
 | |
|  * Sets several of the children properties of the element. See
 | |
|  * ges_timeline_element_set_child_property().
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_set_child_property_valist (GESTimelineElement * self,
 | |
|     const gchar * first_property_name, va_list var_args)
 | |
| {
 | |
|   const gchar *name;
 | |
|   GParamSpec *pspec;
 | |
| 
 | |
|   gchar *error = NULL;
 | |
|   GValue value = { 0, };
 | |
| 
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
| 
 | |
|   name = first_property_name;
 | |
| 
 | |
|   /* Note: This part is in big part copied from the gst_child_object_set_valist
 | |
|    * method. */
 | |
| 
 | |
|   /* iterate over pairs */
 | |
|   while (name) {
 | |
|     if (!ges_timeline_element_lookup_child (self, name, NULL, &pspec))
 | |
|       goto not_found;
 | |
| 
 | |
|     G_VALUE_COLLECT_INIT (&value, pspec->value_type, var_args,
 | |
|         G_VALUE_NOCOPY_CONTENTS, &error);
 | |
| 
 | |
|     if (error)
 | |
|       goto cant_copy;
 | |
| 
 | |
|     set_child_property_by_pspec (self, pspec, &value, NULL);
 | |
| 
 | |
|     g_param_spec_unref (pspec);
 | |
|     g_value_unset (&value);
 | |
| 
 | |
|     name = va_arg (var_args, gchar *);
 | |
|   }
 | |
|   return;
 | |
| 
 | |
| not_found:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "No property %s in OBJECT\n", name);
 | |
|     return;
 | |
|   }
 | |
| cant_copy:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "error copying value %s in %p: %s", pspec->name,
 | |
|         self, error);
 | |
| 
 | |
|     g_param_spec_unref (pspec);
 | |
|     g_value_unset (&value);
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_set_child_properties:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @first_property_name: The name of the first child property to set
 | |
|  * @...: The value for the first property, followed optionally by more
 | |
|  * name/value pairs, followed by %NULL
 | |
|  *
 | |
|  * Sets several of the children properties of the element. See
 | |
|  * ges_timeline_element_set_child_property().
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_set_child_properties (GESTimelineElement * self,
 | |
|     const gchar * first_property_name, ...)
 | |
| {
 | |
|   va_list var_args;
 | |
| 
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
| 
 | |
|   va_start (var_args, first_property_name);
 | |
|   ges_timeline_element_set_child_property_valist (self, first_property_name,
 | |
|       var_args);
 | |
|   va_end (var_args);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_child_property_valist:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @first_property_name: The name of the first child property to get
 | |
|  * @var_args: The return location for the first property, followed
 | |
|  * optionally by more name/return location pairs, followed by %NULL
 | |
|  *
 | |
|  * Gets several of the children properties of the element. See
 | |
|  * ges_timeline_element_get_child_property().
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_get_child_property_valist (GESTimelineElement * self,
 | |
|     const gchar * first_property_name, va_list var_args)
 | |
| {
 | |
|   const gchar *name;
 | |
|   gchar *error = NULL;
 | |
|   GValue value = { 0, };
 | |
|   GParamSpec *pspec;
 | |
|   GObject *child;
 | |
| 
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
| 
 | |
|   name = first_property_name;
 | |
| 
 | |
|   /* This part is in big part copied from the gst_child_object_get_valist method */
 | |
|   while (name) {
 | |
|     if (!ges_timeline_element_lookup_child (self, name, &child, &pspec))
 | |
|       goto not_found;
 | |
| 
 | |
|     g_value_init (&value, pspec->value_type);
 | |
|     g_object_get_property (child, pspec->name, &value);
 | |
|     gst_object_unref (child);
 | |
|     g_param_spec_unref (pspec);
 | |
| 
 | |
|     G_VALUE_LCOPY (&value, var_args, 0, &error);
 | |
|     if (error)
 | |
|       goto cant_copy;
 | |
|     g_value_unset (&value);
 | |
|     name = va_arg (var_args, gchar *);
 | |
|   }
 | |
|   return;
 | |
| 
 | |
| not_found:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "no child property %s", name);
 | |
|     return;
 | |
|   }
 | |
| cant_copy:
 | |
|   {
 | |
|     GST_WARNING_OBJECT (self, "error copying value %s in %s", pspec->name,
 | |
|         error);
 | |
| 
 | |
|     g_value_unset (&value);
 | |
|     return;
 | |
|   }
 | |
| }
 | |
| 
 | |
| static gint
 | |
| compare_gparamspec (GParamSpec ** a, GParamSpec ** b, gpointer udata)
 | |
| {
 | |
|   return g_strcmp0 ((*a)->name, (*b)->name);
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_list_children_properties:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @n_properties: (out): The return location for the length of the
 | |
|  * returned array
 | |
|  *
 | |
|  * Get a list of children properties of the element, which is a list of
 | |
|  * all the specifications passed to
 | |
|  * ges_timeline_element_add_child_property().
 | |
|  *
 | |
|  * Returns: (transfer full) (array length=n_properties): An array of
 | |
|  * #GParamSpec corresponding to the child properties of @self, or %NULL if
 | |
|  * something went wrong.
 | |
|  */
 | |
| GParamSpec **
 | |
| ges_timeline_element_list_children_properties (GESTimelineElement * self,
 | |
|     guint * n_properties)
 | |
| {
 | |
|   GParamSpec **ret;
 | |
|   GESTimelineElementClass *class;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), NULL);
 | |
| 
 | |
|   class = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   if (!class->list_children_properties) {
 | |
|     GST_INFO_OBJECT (self, "No %s->list_children_properties implementation",
 | |
|         G_OBJECT_TYPE_NAME (self));
 | |
| 
 | |
|     *n_properties = 0;
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   ret = class->list_children_properties (self, n_properties);
 | |
|   g_qsort_with_data (ret, *n_properties, sizeof (GParamSpec *),
 | |
|       (GCompareDataFunc) compare_gparamspec, NULL);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_child_properties:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @first_property_name: The name of the first child property to get
 | |
|  * @...: The return location for the first property, followed
 | |
|  * optionally by more name/return location pairs, followed by %NULL
 | |
|  *
 | |
|  * Gets several of the children properties of the element. See
 | |
|  * ges_timeline_element_get_child_property().
 | |
|  */
 | |
| void
 | |
| ges_timeline_element_get_child_properties (GESTimelineElement * self,
 | |
|     const gchar * first_property_name, ...)
 | |
| {
 | |
|   va_list var_args;
 | |
| 
 | |
|   g_return_if_fail (GES_IS_TIMELINE_ELEMENT (self));
 | |
| 
 | |
|   va_start (var_args, first_property_name);
 | |
|   ges_timeline_element_get_child_property_valist (self, first_property_name,
 | |
|       var_args);
 | |
|   va_end (var_args);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_remove_child_property:
 | |
|  * @self: A #GESTimelineElement
 | |
|  * @pspec: The specification for the property to remove
 | |
|  *
 | |
|  * Remove a child property from the element. @pspec should be a
 | |
|  * specification that was passed to
 | |
|  * ges_timeline_element_add_child_property(). The corresponding property
 | |
|  * will no longer be registered as a child property for the element.
 | |
|  *
 | |
|  * Returns: %TRUE if the property was successfully un-registered for @self.
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_remove_child_property (GESTimelineElement * self,
 | |
|     GParamSpec * pspec)
 | |
| {
 | |
|   gpointer key, value;
 | |
|   GParamSpec *found_pspec;
 | |
|   ChildPropHandler *handler;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (G_IS_PARAM_SPEC (pspec), FALSE);
 | |
| 
 | |
|   if (!g_hash_table_lookup_extended (self->priv->children_props, pspec,
 | |
|           &key, &value)) {
 | |
|     GST_WARNING_OBJECT (self, "No child property with pspec %p (%s) found",
 | |
|         pspec, pspec->name);
 | |
|     return FALSE;
 | |
|   }
 | |
|   g_hash_table_steal (self->priv->children_props, pspec);
 | |
|   found_pspec = G_PARAM_SPEC (key);
 | |
|   handler = (ChildPropHandler *) value;
 | |
| 
 | |
|   g_signal_emit (self, ges_timeline_element_signals[CHILD_PROPERTY_REMOVED], 0,
 | |
|       handler->child, found_pspec);
 | |
| 
 | |
|   g_param_spec_unref (found_pspec);
 | |
|   _child_prop_handler_free (handler);
 | |
| 
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_track_types:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the track types that the element can interact with, i.e. the type
 | |
|  * of #GESTrack it can exist in, or will create #GESTrackElement-s for.
 | |
|  *
 | |
|  * Returns: The track types that @self supports.
 | |
|  *
 | |
|  * Since: 1.6.0
 | |
|  */
 | |
| GESTrackType
 | |
| ges_timeline_element_get_track_types (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), 0);
 | |
|   g_return_val_if_fail (GES_TIMELINE_ELEMENT_GET_CLASS (self)->get_track_types,
 | |
|       0);
 | |
| 
 | |
|   return GES_TIMELINE_ELEMENT_GET_CLASS (self)->get_track_types (self);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_paste:
 | |
|  * @self: The #GESTimelineElement to paste
 | |
|  * @paste_position: The position in the timeline @element should be pasted
 | |
|  * to, i.e. the #GESTimelineElement:start value for the pasted element.
 | |
|  *
 | |
|  * Paste an element inside the same timeline and layer as @self. @self
 | |
|  * **must** be the return of ges_timeline_element_copy() with `deep=TRUE`,
 | |
|  * and it should not be changed before pasting.
 | |
|  * @self is not placed in the timeline, instead a new element is created,
 | |
|  * alike to the originally copied element. Note that the originally
 | |
|  * copied element must stay within the same timeline and layer, at both
 | |
|  * the point of copying and pasting.
 | |
|  *
 | |
|  * Pasting may fail if it would place the timeline in an unsupported
 | |
|  * configuration.
 | |
|  *
 | |
|  * After calling this function @element should not be used. In particular,
 | |
|  * @element can **not** be pasted again. Instead, you can copy the
 | |
|  * returned element and paste that copy (although, this is only possible
 | |
|  * if the paste was successful).
 | |
|  *
 | |
|  * See also ges_timeline_paste_element().
 | |
|  *
 | |
|  * Returns: (transfer full) (nullable): The newly created element, or
 | |
|  * %NULL if pasting fails.
 | |
|  *
 | |
|  * Since: 1.6.0
 | |
|  */
 | |
| GESTimelineElement *
 | |
| ges_timeline_element_paste (GESTimelineElement * self,
 | |
|     GstClockTime paste_position)
 | |
| {
 | |
|   GESTimelineElement *res;
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (paste_position), FALSE);
 | |
| 
 | |
|   if (!self->priv->copied_from) {
 | |
|     GST_ERROR_OBJECT (self, "Is not being 'deeply' copied!");
 | |
| 
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   if (!GES_TIMELINE_ELEMENT_GET_CLASS (self)->paste) {
 | |
|     GST_ERROR_OBJECT (self, "No paste vmethod implemented");
 | |
| 
 | |
|     return NULL;
 | |
|   }
 | |
| 
 | |
|   res = GES_TIMELINE_ELEMENT_GET_CLASS (self)->paste (self,
 | |
|       self->priv->copied_from, paste_position);
 | |
| 
 | |
|   g_clear_object (&self->priv->copied_from);
 | |
| 
 | |
|   return res ? g_object_ref_sink (res) : res;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_layer_priority:
 | |
|  * @self: A #GESTimelineElement
 | |
|  *
 | |
|  * Gets the priority of the layer the element is in. A #GESGroup may span
 | |
|  * several layers, so this would return the highest priority (numerically,
 | |
|  * the smallest) amongst them.
 | |
|  *
 | |
|  * Returns: The priority of the layer @self is in, or
 | |
|  * #GES_TIMELINE_ELEMENT_NO_LAYER_PRIORITY if @self does not exist in a
 | |
|  * layer.
 | |
|  *
 | |
|  * Since: 1.16
 | |
|  */
 | |
| guint32
 | |
| ges_timeline_element_get_layer_priority (GESTimelineElement * self)
 | |
| {
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self),
 | |
|       GES_TIMELINE_ELEMENT_NO_LAYER_PRIORITY);
 | |
| 
 | |
|   if (!GES_TIMELINE_ELEMENT_GET_CLASS (self)->get_layer_priority)
 | |
|     return self->priority;
 | |
| 
 | |
|   return GES_TIMELINE_ELEMENT_GET_CLASS (self)->get_layer_priority (self);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_edit_full:
 | |
|  * @self: The #GESTimelineElement to edit
 | |
|  * @new_layer_priority: The priority/index of the layer @self should be
 | |
|  * moved to. -1 means no move
 | |
|  * @mode: The edit mode
 | |
|  * @edge: The edge of @self where the edit should occur
 | |
|  * @position: The edit position: a new location for the edge of @self
 | |
|  * (in nanoseconds) in the timeline coordinates
 | |
|  * @error: (nullable): Return location for an error
 | |
|  *
 | |
|  * Edits the element within its timeline by adjusting its
 | |
|  * #GESTimelineElement:start, #GESTimelineElement:duration or
 | |
|  * #GESTimelineElement:in-point, and potentially doing the same for
 | |
|  * other elements in the timeline. See #GESEditMode for details about each
 | |
|  * edit mode. An edit may fail if it would place one of these properties
 | |
|  * out of bounds, or if it would place the timeline in an unsupported
 | |
|  * configuration.
 | |
|  *
 | |
|  * Note that if you act on a #GESTrackElement, this will edit its parent
 | |
|  * #GESClip instead. Moreover, for any #GESTimelineElement, if you select
 | |
|  * #GES_EDGE_NONE for #GES_EDIT_MODE_NORMAL or #GES_EDIT_MODE_RIPPLE, this
 | |
|  * will edit the toplevel instead, but still in such a way as to make the
 | |
|  * #GESTimelineElement:start of @self reach the edit @position.
 | |
|  *
 | |
|  * Note that if the element's timeline has a
 | |
|  * #GESTimeline:snapping-distance set, then the edit position may be
 | |
|  * snapped to the edge of some element under the edited element.
 | |
|  *
 | |
|  * @new_layer_priority can be used to switch @self, and other elements
 | |
|  * moved by the edit, to a new layer. New layers may be be created if the
 | |
|  * the corresponding layer priority/index does not yet exist for the
 | |
|  * timeline.
 | |
|  *
 | |
|  * Returns: %TRUE if the edit of @self completed, %FALSE on failure.
 | |
|  * Since: 1.18
 | |
|  */
 | |
| 
 | |
| gboolean
 | |
| ges_timeline_element_edit_full (GESTimelineElement * self,
 | |
|     gint64 new_layer_priority, GESEditMode mode, GESEdge edge, guint64 position,
 | |
|     GError ** error)
 | |
| {
 | |
|   GESTimeline *timeline;
 | |
|   guint32 layer_prio;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (position), FALSE);
 | |
|   g_return_val_if_fail (!error || !*error, FALSE);
 | |
| 
 | |
|   timeline = GES_TIMELINE_ELEMENT_TIMELINE (self);
 | |
|   g_return_val_if_fail (timeline, FALSE);
 | |
| 
 | |
|   layer_prio = GES_TIMELINE_ELEMENT_LAYER_PRIORITY (self);
 | |
| 
 | |
|   if (new_layer_priority < 0)
 | |
|     new_layer_priority = layer_prio;
 | |
| 
 | |
|   GST_DEBUG_OBJECT (self, "Editing %s at edge %s to position %"
 | |
|       GST_TIME_FORMAT " under %s mode, and to layer %" G_GINT64_FORMAT,
 | |
|       self->name, ges_edge_name (edge), GST_TIME_ARGS (position),
 | |
|       ges_edit_mode_name (mode), new_layer_priority);
 | |
| 
 | |
|   return ges_timeline_edit (timeline, self, new_layer_priority, mode,
 | |
|       edge, position, error);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_edit:
 | |
|  * @self: The #GESTimelineElement to edit
 | |
|  * @layers: (element-type GESLayer) (nullable): A whitelist of layers
 | |
|  * where the edit can be performed, %NULL allows all layers in the
 | |
|  * timeline.
 | |
|  * @new_layer_priority: The priority/index of the layer @self should be
 | |
|  * moved to. -1 means no move
 | |
|  * @mode: The edit mode
 | |
|  * @edge: The edge of @self where the edit should occur
 | |
|  * @position: The edit position: a new location for the edge of @self
 | |
|  * (in nanoseconds) in the timeline coordinates
 | |
|  *
 | |
|  * See ges_timeline_element_edit_full(), which also gives an error.
 | |
|  *
 | |
|  * Note that the @layers argument is currently ignored, so you should
 | |
|  * just pass %NULL.
 | |
|  *
 | |
|  * Returns: %TRUE if the edit of @self completed, %FALSE on failure.
 | |
|  *
 | |
|  * Since: 1.18
 | |
|  */
 | |
| 
 | |
| /* FIXME: handle the layers argument. Currently we always treat it as if
 | |
|  * it is NULL in the ges-timeline code */
 | |
| gboolean
 | |
| ges_timeline_element_edit (GESTimelineElement * self, GList * layers,
 | |
|     gint64 new_layer_priority, GESEditMode mode, GESEdge edge, guint64 position)
 | |
| {
 | |
|   return ges_timeline_element_edit_full (self, new_layer_priority, mode, edge,
 | |
|       position, NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * ges_timeline_element_get_natural_framerate:
 | |
|  * @self: The #GESTimelineElement to get "natural" framerate from
 | |
|  * @framerate_n: (out): The framerate numerator
 | |
|  * @framerate_d: (out): The framerate denominator
 | |
|  *
 | |
|  * Get the "natural" framerate of @self. This is to say, for example
 | |
|  * for a #GESVideoUriSource the framerate of the source.
 | |
|  *
 | |
|  * Note that a #GESAudioSource may also have a natural framerate if it derives
 | |
|  * from the same #GESSourceClip asset as a #GESVideoSource, and its value will
 | |
|  * be that of the video source. For example, if the uri of a #GESUriClip points
 | |
|  * to a file that contains both a video and audio stream, then the corresponding
 | |
|  * #GESAudioUriSource will share the natural framerate of the corresponding
 | |
|  * #GESVideoUriSource.
 | |
|  *
 | |
|  * Returns: Whether @self has a natural framerate or not, @framerate_n
 | |
|  * and @framerate_d will be set to, respectively, 0 and -1 if it is
 | |
|  * not the case.
 | |
|  *
 | |
|  * Since: 1.18
 | |
|  */
 | |
| gboolean
 | |
| ges_timeline_element_get_natural_framerate (GESTimelineElement * self,
 | |
|     gint * framerate_n, gint * framerate_d)
 | |
| {
 | |
|   GESTimelineElementClass *klass;
 | |
| 
 | |
|   g_return_val_if_fail (GES_IS_TIMELINE_ELEMENT (self), FALSE);
 | |
|   g_return_val_if_fail (framerate_n && framerate_d, FALSE);
 | |
| 
 | |
|   klass = GES_TIMELINE_ELEMENT_GET_CLASS (self);
 | |
| 
 | |
|   *framerate_n = 0;
 | |
|   *framerate_d = -1;
 | |
|   return klass->get_natural_framerate (self, framerate_n, framerate_d);
 | |
| }
 |