2009-08-04 15:13:11 +00:00
|
|
|
/* GStreamer Editing Services
|
2009-11-30 14:14:25 +00:00
|
|
|
* Copyright (C) 2009 Edward Hervey <edward.hervey@collabora.co.uk>
|
|
|
|
* 2009 Nokia Corporation
|
2012-12-20 23:23:54 +00:00
|
|
|
* 2012 Collabora Ltd.
|
|
|
|
* Author: Sebastian Dröge <sebastian.droege@collabora.co.uk>
|
2009-08-04 15:13:11 +00:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Library General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Library General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Library General Public
|
|
|
|
* License along with this library; if not, write to the
|
2012-11-04 00:25:20 +00:00
|
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
2009-08-04 15:13:11 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2014-04-07 19:02:48 +00:00
|
|
|
* SECTION:gesclip
|
2017-03-08 21:13:48 +00:00
|
|
|
* @title: GESClip
|
2020-01-09 12:11:35 +00:00
|
|
|
* @short_description: Base class for elements that occupy a single
|
|
|
|
* #GESLayer and maintain equal timings of their children
|
2009-08-04 15:13:11 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* #GESClip-s are the core objects of a #GESLayer. Each clip may exist in
|
|
|
|
* a single layer but may control several #GESTrackElement-s that span
|
|
|
|
* several #GESTrack-s. A clip will ensure that all its children share the
|
|
|
|
* same #GESTimelineElement:start and #GESTimelineElement:duration in
|
|
|
|
* their tracks, which will match the #GESTimelineElement:start and
|
|
|
|
* #GESTimelineElement:duration of the clip itself. Therefore, changing
|
|
|
|
* the timing of the clip will change the timing of the children, and a
|
|
|
|
* change in the timing of a child will change the timing of the clip and
|
|
|
|
* subsequently all its siblings. As such, a clip can be treated as a
|
|
|
|
* singular object in its layer.
|
2009-08-04 15:13:11 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* For most uses of a #GESTimeline, it is often sufficient to only
|
|
|
|
* interact with #GESClip-s directly, which will take care of creating and
|
|
|
|
* organising the elements of the timeline's tracks.
|
|
|
|
*
|
|
|
|
* Most subclassed clips will be associated with some *core*
|
|
|
|
* #GESTrackElement-s. When such a clip is added to a layer in a timeline,
|
|
|
|
* it will create these children and they will be added to the timeline's
|
|
|
|
* tracks. You can connect to the #GESContainer::child-added signal to be
|
|
|
|
* notified of their creation. If a clip will produce several core
|
|
|
|
* elements of the same #GESTrackElement:track-type but they are destined
|
|
|
|
* for separate tracks, you should connect to the timeline's
|
|
|
|
* #GESTimeline::select-tracks-for-object signal to coordinate which
|
|
|
|
* tracks each element should land in.
|
|
|
|
*
|
|
|
|
* The #GESTimelineElement:in-point of the clip will control the
|
|
|
|
* #GESTimelineElement:in-point of these core elements to be the same
|
2020-03-10 11:38:58 +00:00
|
|
|
* value if their #GESTrackElement:has-internal-source is set to %TRUE.
|
|
|
|
* The #GESTimelineElement:max-duration of the clip is the minimum
|
|
|
|
* #GESTimelineElement:max-duration of its children. If you set its value
|
|
|
|
* to anything other than its current value, this will also set the
|
|
|
|
* #GESTimelineElement:max-duration of all its core children to the same
|
|
|
|
* value if their #GESTrackElement:has-internal-source is set to %TRUE.
|
2020-01-09 12:11:35 +00:00
|
|
|
*
|
|
|
|
* ## Effects
|
|
|
|
*
|
2020-03-02 12:56:03 +00:00
|
|
|
* Some subclasses (#GESSourceClip and #GESBaseEffect) may also allow
|
|
|
|
* their objects to have additional non-core #GESBaseEffect-s elements as
|
|
|
|
* children. These are additional effects that are applied to the output
|
|
|
|
* data of the core elements. They can be added to the clip using
|
|
|
|
* ges_container_add(), which will take care of adding the effect to the
|
|
|
|
* timeline's tracks. The new effect will be placed between the clip's
|
|
|
|
* core track elements and its other effects. As such, the newly added
|
|
|
|
* effect will be applied to any source data **before** the other existing
|
|
|
|
* effects. You can change the ordering of effects using
|
2020-01-09 12:11:35 +00:00
|
|
|
* ges_clip_set_top_effect_index().
|
2009-08-04 15:13:11 +00:00
|
|
|
*/
|
2018-09-24 14:41:24 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
2009-08-04 15:13:11 +00:00
|
|
|
|
2013-01-20 15:42:29 +00:00
|
|
|
#include "ges-clip.h"
|
2009-09-10 16:40:51 +00:00
|
|
|
#include "ges.h"
|
|
|
|
#include "ges-internal.h"
|
|
|
|
|
2012-03-12 15:46:42 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2017-08-07 19:35:58 +00:00
|
|
|
static GList *ges_clip_create_track_elements_func (GESClip * clip,
|
|
|
|
GESTrackType type);
|
2013-01-15 13:52:17 +00:00
|
|
|
static gboolean _ripple (GESTimelineElement * element, GstClockTime start);
|
|
|
|
static gboolean _ripple_end (GESTimelineElement * element, GstClockTime end);
|
|
|
|
static gboolean _roll_start (GESTimelineElement * element, GstClockTime start);
|
|
|
|
static gboolean _roll_end (GESTimelineElement * element, GstClockTime end);
|
|
|
|
static gboolean _trim (GESTimelineElement * element, GstClockTime start);
|
2013-06-28 18:39:16 +00:00
|
|
|
static void _compute_height (GESContainer * container);
|
2012-09-10 00:20:46 +00:00
|
|
|
|
2013-01-20 15:42:29 +00:00
|
|
|
struct _GESClipPrivate
|
2010-11-28 12:24:07 +00:00
|
|
|
{
|
|
|
|
/*< public > */
|
2013-04-23 23:04:04 +00:00
|
|
|
GESLayer *layer;
|
2010-11-28 12:24:07 +00:00
|
|
|
|
|
|
|
/*< private > */
|
2011-01-31 10:10:35 +00:00
|
|
|
guint nb_effects;
|
2011-12-01 03:18:30 +00:00
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
GList *copied_track_elements;
|
|
|
|
GESLayer *copied_layer;
|
2020-03-02 13:25:21 +00:00
|
|
|
gboolean prevent_priority_offset_update;
|
|
|
|
gboolean prevent_resort;
|
2016-01-01 10:56:27 +00:00
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
gboolean updating_max_duration;
|
|
|
|
gboolean prevent_max_duration_update;
|
|
|
|
gboolean setting_inpoint;
|
|
|
|
|
2013-01-20 15:42:29 +00:00
|
|
|
/* The formats supported by this Clip */
|
2011-12-01 03:18:30 +00:00
|
|
|
GESTrackType supportedformats;
|
2010-11-28 12:24:07 +00:00
|
|
|
};
|
|
|
|
|
2013-03-02 21:35:34 +00:00
|
|
|
typedef struct _CheckTrack
|
|
|
|
{
|
|
|
|
GESTrack *track;
|
|
|
|
GESTrackElement *source;
|
|
|
|
} CheckTrack;
|
|
|
|
|
2009-08-07 18:33:18 +00:00
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
2010-11-28 12:24:07 +00:00
|
|
|
PROP_LAYER,
|
2011-12-01 03:18:30 +00:00
|
|
|
PROP_SUPPORTED_FORMATS,
|
2010-12-16 18:29:14 +00:00
|
|
|
PROP_LAST
|
2009-08-07 18:33:18 +00:00
|
|
|
};
|
|
|
|
|
2010-12-16 18:29:14 +00:00
|
|
|
static GParamSpec *properties[PROP_LAST];
|
|
|
|
|
2018-09-06 01:55:02 +00:00
|
|
|
G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (GESClip, ges_clip, GES_TYPE_CONTAINER);
|
|
|
|
|
2013-07-01 21:51:32 +00:00
|
|
|
/****************************************************
|
|
|
|
* Listen to our children *
|
|
|
|
****************************************************/
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2016-08-11 12:54:23 +00:00
|
|
|
/* @min_priority: The absolute minimum priority a child of @container should have
|
|
|
|
* @max_priority: The absolute maximum priority a child of @container should have
|
|
|
|
*/
|
2013-06-29 23:31:23 +00:00
|
|
|
static void
|
2020-03-02 13:25:21 +00:00
|
|
|
_get_priority_range_full (GESContainer * container, guint32 * min_priority,
|
|
|
|
guint32 * max_priority, guint32 priority_base)
|
2013-06-29 23:31:23 +00:00
|
|
|
{
|
|
|
|
GESLayer *layer = GES_CLIP (container)->priv->layer;
|
|
|
|
|
|
|
|
if (layer) {
|
2020-03-02 13:25:21 +00:00
|
|
|
*min_priority = priority_base + layer->min_nle_priority;
|
2014-08-15 13:48:14 +00:00
|
|
|
*max_priority = layer->max_nle_priority;
|
2013-06-29 23:31:23 +00:00
|
|
|
} else {
|
2020-03-02 13:25:21 +00:00
|
|
|
*min_priority = priority_base + MIN_NLE_PRIO;
|
2013-06-29 23:31:23 +00:00
|
|
|
*max_priority = G_MAXUINT32;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
static void
|
|
|
|
_get_priority_range (GESContainer * container, guint32 * min_priority,
|
|
|
|
guint32 * max_priority)
|
|
|
|
{
|
|
|
|
_get_priority_range_full (container, min_priority, max_priority,
|
|
|
|
_PRIORITY (container));
|
|
|
|
}
|
|
|
|
|
2013-07-01 21:51:32 +00:00
|
|
|
static void
|
|
|
|
_child_priority_changed_cb (GESTimelineElement * child,
|
|
|
|
GParamSpec * arg G_GNUC_UNUSED, GESContainer * container)
|
|
|
|
{
|
2020-03-02 13:25:21 +00:00
|
|
|
/* we do not change the rest of the clip in response to a change in
|
|
|
|
* the child priority */
|
2013-07-01 21:51:32 +00:00
|
|
|
guint32 min_prio, max_prio;
|
2020-03-02 13:25:21 +00:00
|
|
|
GESClipPrivate *priv = GES_CLIP (container)->priv;
|
2013-07-01 21:51:32 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
GST_DEBUG_OBJECT (container, "TimelineElement %" GES_FORMAT
|
|
|
|
" priority changed to %u", GES_ARGS (child), _PRIORITY (child));
|
2013-07-01 21:51:32 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
if (!priv->prevent_priority_offset_update) {
|
|
|
|
/* Update mapping */
|
|
|
|
_get_priority_range (container, &min_prio, &max_prio);
|
2013-07-01 21:51:32 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
_ges_container_set_priority_offset (container, child,
|
|
|
|
min_prio - _PRIORITY (child));
|
|
|
|
}
|
2013-07-01 21:51:32 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
if (!priv->prevent_resort) {
|
|
|
|
_ges_container_sort_children (container);
|
|
|
|
_compute_height (container);
|
|
|
|
}
|
2013-07-01 21:51:32 +00:00
|
|
|
}
|
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
static void
|
|
|
|
_child_inpoint_changed_cb (GESTimelineElement * child, GParamSpec * pspec,
|
|
|
|
GESContainer * container)
|
|
|
|
{
|
|
|
|
if (GES_CLIP (container)->priv->setting_inpoint)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* ignore non-core */
|
2020-03-18 15:56:06 +00:00
|
|
|
if (!ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2020-03-10 11:38:58 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* if the track element has no internal content, then this means its
|
|
|
|
* in-point has been set (back) to 0, we can ignore this update */
|
|
|
|
if (!ges_track_element_has_internal_source (GES_TRACK_ELEMENT (child)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* If the child->inpoint is the same as our own, set_inpoint will do
|
|
|
|
* nothing. For example, when we set them in add_child (the notifies for
|
|
|
|
* this are released after child_added is called because
|
|
|
|
* ges_container_add freezes them) */
|
|
|
|
_set_inpoint0 (GES_TIMELINE_ELEMENT (container), child->inpoint);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* called when a child is added, removed or their max-duration changes */
|
|
|
|
static void
|
|
|
|
_update_max_duration (GESContainer * container)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
GstClockTime min = GST_CLOCK_TIME_NONE;
|
|
|
|
GESClipPrivate *priv = GES_CLIP (container)->priv;
|
|
|
|
|
|
|
|
if (priv->prevent_max_duration_update)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (tmp = container->children; tmp; tmp = tmp->next) {
|
|
|
|
GESTimelineElement *child = tmp->data;
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child))
|
2020-03-10 11:38:58 +00:00
|
|
|
&& GST_CLOCK_TIME_IS_VALID (child->maxduration))
|
|
|
|
min = GST_CLOCK_TIME_IS_VALID (min) ? MIN (min, child->maxduration) :
|
|
|
|
child->maxduration;
|
|
|
|
}
|
|
|
|
priv->updating_max_duration = TRUE;
|
|
|
|
ges_timeline_element_set_max_duration (GES_TIMELINE_ELEMENT (container), min);
|
|
|
|
priv->updating_max_duration = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_child_max_duration_changed_cb (GESTimelineElement * child,
|
|
|
|
GParamSpec * pspec, GESContainer * container)
|
|
|
|
{
|
|
|
|
/* ignore non-core */
|
2020-03-18 15:56:06 +00:00
|
|
|
if (!ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2020-03-10 11:38:58 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
_update_max_duration (container);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_child_has_internal_source_changed_cb (GESTimelineElement * child,
|
|
|
|
GParamSpec * pspec, GESContainer * container)
|
|
|
|
{
|
|
|
|
/* ignore non-core */
|
2020-03-18 15:56:06 +00:00
|
|
|
if (!ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2020-03-10 11:38:58 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* if the track element is now registered to have no internal content,
|
|
|
|
* we don't have to do anything */
|
|
|
|
if (!ges_track_element_has_internal_source (GES_TRACK_ELEMENT (child)))
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* otherwise, we need to make its in-point match ours */
|
|
|
|
_set_inpoint0 (child, _INPOINT (container));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-07-01 21:51:32 +00:00
|
|
|
/*****************************************************
|
|
|
|
* *
|
|
|
|
* GESTimelineElement virtual methods implementation *
|
|
|
|
* *
|
|
|
|
*****************************************************/
|
|
|
|
|
2013-06-26 20:54:02 +00:00
|
|
|
static gboolean
|
|
|
|
_set_start (GESTimelineElement * element, GstClockTime start)
|
|
|
|
{
|
2020-03-02 12:23:07 +00:00
|
|
|
GList *tmp, *children;
|
2013-06-26 20:54:02 +00:00
|
|
|
GESContainer *container = GES_CONTAINER (element);
|
|
|
|
|
|
|
|
GST_DEBUG_OBJECT (element, "Setting children start, (initiated_move: %"
|
|
|
|
GST_PTR_FORMAT ")", container->initiated_move);
|
|
|
|
|
2020-03-02 12:23:07 +00:00
|
|
|
/* get copy of children, since GESContainer may resort the clip */
|
|
|
|
children = ges_container_get_children (container, FALSE);
|
2013-07-01 21:51:32 +00:00
|
|
|
container->children_control_mode = GES_CHILDREN_IGNORE_NOTIFIES;
|
2020-03-02 12:23:07 +00:00
|
|
|
for (tmp = children; tmp; tmp = g_list_next (tmp)) {
|
2013-06-26 20:54:02 +00:00
|
|
|
GESTimelineElement *child = (GESTimelineElement *) tmp->data;
|
|
|
|
|
2019-12-14 17:04:54 +00:00
|
|
|
if (child != container->initiated_move)
|
|
|
|
_set_start0 (GES_TIMELINE_ELEMENT (child), start);
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
2013-07-01 21:51:32 +00:00
|
|
|
container->children_control_mode = GES_CHILDREN_UPDATE;
|
2020-03-02 12:23:07 +00:00
|
|
|
g_list_free_full (children, gst_object_unref);
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2019-12-14 17:04:54 +00:00
|
|
|
return TRUE;
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
|
|
|
|
2020-03-10 15:27:20 +00:00
|
|
|
/* Whether @clip can have its in-point set to @inpoint because none of
|
|
|
|
* its children have a max-duration below it */
|
|
|
|
gboolean
|
|
|
|
ges_clip_can_set_inpoint_of_child (GESClip * clip, GESTimelineElement * child,
|
|
|
|
GstClockTime inpoint)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
|
|
|
|
/* don't bother checking if we are setting the value */
|
|
|
|
if (clip->priv->setting_inpoint)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* non-core children do not effect our in-point */
|
2020-03-18 15:56:06 +00:00
|
|
|
if (!ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2020-03-10 15:27:20 +00:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = tmp->next) {
|
|
|
|
GESTimelineElement *child = tmp->data;
|
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child))
|
2020-03-10 15:27:20 +00:00
|
|
|
&& ges_track_element_has_internal_source (GES_TRACK_ELEMENT (child))) {
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (child->maxduration)
|
|
|
|
&& child->maxduration < inpoint)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* returns TRUE if we did not break early */
|
2013-06-26 20:54:02 +00:00
|
|
|
static gboolean
|
2020-03-10 15:27:20 +00:00
|
|
|
_set_childrens_inpoint (GESTimelineElement * element, GstClockTime inpoint,
|
|
|
|
gboolean break_on_failure)
|
2013-06-26 20:54:02 +00:00
|
|
|
{
|
2020-03-10 11:38:58 +00:00
|
|
|
GList *tmp;
|
|
|
|
GESClipPrivate *priv = GES_CLIP (element)->priv;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
priv->setting_inpoint = TRUE;
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (element); tmp; tmp = tmp->next) {
|
|
|
|
GESTimelineElement *child = tmp->data;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child))
|
2020-03-10 15:27:20 +00:00
|
|
|
&& ges_track_element_has_internal_source (GES_TRACK_ELEMENT (child))) {
|
|
|
|
if (!_set_inpoint0 (child, inpoint)) {
|
|
|
|
GST_ERROR_OBJECT ("Could not set the in-point of child %"
|
|
|
|
GES_FORMAT " to %" GST_TIME_FORMAT, GES_ARGS (child),
|
|
|
|
GST_TIME_ARGS (inpoint));
|
|
|
|
if (break_on_failure)
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
2020-03-10 11:38:58 +00:00
|
|
|
priv->setting_inpoint = FALSE;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-03-10 15:27:20 +00:00
|
|
|
static gboolean
|
|
|
|
_set_inpoint (GESTimelineElement * element, GstClockTime inpoint)
|
|
|
|
{
|
|
|
|
if (!_set_childrens_inpoint (element, inpoint, TRUE)) {
|
|
|
|
_set_childrens_inpoint (element, element->inpoint, FALSE);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-06-26 20:54:02 +00:00
|
|
|
static gboolean
|
|
|
|
_set_duration (GESTimelineElement * element, GstClockTime duration)
|
|
|
|
{
|
2020-03-02 12:23:07 +00:00
|
|
|
GList *tmp, *children;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
|
|
|
GESContainer *container = GES_CONTAINER (element);
|
|
|
|
|
2020-03-02 12:23:07 +00:00
|
|
|
/* get copy of children, since GESContainer may resort the clip */
|
|
|
|
children = ges_container_get_children (container, FALSE);
|
2013-07-01 21:51:32 +00:00
|
|
|
container->children_control_mode = GES_CHILDREN_IGNORE_NOTIFIES;
|
2013-06-26 20:54:02 +00:00
|
|
|
for (tmp = container->children; tmp; tmp = g_list_next (tmp)) {
|
|
|
|
GESTimelineElement *child = (GESTimelineElement *) tmp->data;
|
|
|
|
|
2020-03-05 18:56:28 +00:00
|
|
|
if (child != container->initiated_move) {
|
|
|
|
ELEMENT_SET_FLAG (child, GES_TIMELINE_ELEMENT_SET_SIMPLE);
|
2019-03-01 22:32:19 +00:00
|
|
|
_set_duration0 (GES_TIMELINE_ELEMENT (child), duration);
|
2020-03-05 18:56:28 +00:00
|
|
|
ELEMENT_UNSET_FLAG (child, GES_TIMELINE_ELEMENT_SET_SIMPLE);
|
|
|
|
}
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
2013-07-01 21:51:32 +00:00
|
|
|
container->children_control_mode = GES_CHILDREN_UPDATE;
|
2020-03-02 12:23:07 +00:00
|
|
|
g_list_free_full (children, gst_object_unref);
|
2013-06-26 20:54:02 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_set_max_duration (GESTimelineElement * element, GstClockTime maxduration)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
2020-03-10 11:38:58 +00:00
|
|
|
GESClipPrivate *priv = GES_CLIP (element)->priv;
|
|
|
|
GstClockTime new_min = GST_CLOCK_TIME_NONE;
|
|
|
|
|
|
|
|
/* if we are setting based on a change in the minimum */
|
|
|
|
if (priv->updating_max_duration)
|
|
|
|
return TRUE;
|
|
|
|
|
2020-03-18 16:36:47 +00:00
|
|
|
if (!GES_CONTAINER_CHILDREN (element)) {
|
|
|
|
/* If any child added later on has a lower max duration, this max duration
|
|
|
|
* will be used instead anyway */
|
|
|
|
GST_INFO_OBJECT (element,
|
|
|
|
"Setting max duration %" GST_TIME_FORMAT " as %" GES_FORMAT
|
|
|
|
" doesn't have any child yet",
|
|
|
|
GST_TIME_ARGS (maxduration), GES_ARGS (element));
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
/* else, we set every core child to have the same max duration */
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
priv->prevent_max_duration_update = TRUE;
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (element); tmp; tmp = tmp->next) {
|
|
|
|
GESTimelineElement *child = tmp->data;
|
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child))
|
2020-03-10 11:38:58 +00:00
|
|
|
&& ges_track_element_has_internal_source (GES_TRACK_ELEMENT (child))) {
|
2020-03-10 15:27:20 +00:00
|
|
|
if (!ges_timeline_element_set_max_duration (child, maxduration)) {
|
|
|
|
GST_ERROR_OBJECT ("Could not set the max-duration of child %"
|
|
|
|
GES_FORMAT " to %" GST_TIME_FORMAT, GES_ARGS (child),
|
|
|
|
GST_TIME_ARGS (maxduration));
|
|
|
|
}
|
2020-03-10 11:38:58 +00:00
|
|
|
if (GST_CLOCK_TIME_IS_VALID (child->maxduration)) {
|
|
|
|
new_min = GST_CLOCK_TIME_IS_VALID (new_min) ?
|
|
|
|
MIN (new_min, child->maxduration) : child->maxduration;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
priv->prevent_max_duration_update = FALSE;
|
|
|
|
|
|
|
|
if (new_min != maxduration) {
|
|
|
|
if (GST_CLOCK_TIME_IS_VALID (new_min))
|
|
|
|
GST_WARNING_OBJECT (element, "Failed to set the max-duration of the "
|
|
|
|
"clip to %" GST_TIME_FORMAT " because it was not possible to "
|
|
|
|
"match this with the actual minimum of %" GST_TIME_FORMAT,
|
|
|
|
GST_TIME_ARGS (maxduration), GST_TIME_ARGS (new_min));
|
|
|
|
else
|
|
|
|
GST_WARNING_OBJECT (element, "Failed to set the max-duration of the "
|
|
|
|
"clip to %" GST_TIME_FORMAT " because it has no core children "
|
|
|
|
"whose max-duration could be set to anything other than "
|
|
|
|
"GST_CLOCK_TIME_NONE", GST_TIME_ARGS (maxduration));
|
|
|
|
priv->updating_max_duration = TRUE;
|
|
|
|
ges_timeline_element_set_max_duration (element, new_min);
|
|
|
|
priv->updating_max_duration = FALSE;
|
|
|
|
return FALSE;
|
|
|
|
}
|
2013-06-26 20:54:02 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_set_priority (GESTimelineElement * element, guint32 priority)
|
|
|
|
{
|
2020-03-02 13:25:21 +00:00
|
|
|
GESClipPrivate *priv = GES_CLIP (element)->priv;
|
2013-06-26 20:54:02 +00:00
|
|
|
GList *tmp;
|
|
|
|
guint32 min_prio, max_prio;
|
|
|
|
|
|
|
|
GESContainer *container = GES_CONTAINER (element);
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* send the new 'priority' to determine what the new 'min_prio' should
|
|
|
|
* be for the clip */
|
|
|
|
_get_priority_range_full (container, &min_prio, &max_prio, priority);
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* offsets will remain constant for the children */
|
|
|
|
priv->prevent_resort = TRUE;
|
|
|
|
priv->prevent_priority_offset_update = TRUE;
|
2013-06-26 20:54:02 +00:00
|
|
|
for (tmp = container->children; tmp; tmp = g_list_next (tmp)) {
|
2016-08-11 12:54:23 +00:00
|
|
|
guint32 track_element_prio;
|
2013-06-26 20:54:02 +00:00
|
|
|
GESTimelineElement *child = (GESTimelineElement *) tmp->data;
|
2015-07-01 09:35:42 +00:00
|
|
|
gint off = _ges_container_get_priority_offset (container, child);
|
|
|
|
|
2015-06-30 21:13:28 +00:00
|
|
|
if (off >= LAYER_HEIGHT) {
|
|
|
|
GST_ERROR ("%s child %s as a priority offset %d >= LAYER_HEIGHT %d"
|
2015-07-01 09:35:42 +00:00
|
|
|
" ==> clamping it to 0", GES_TIMELINE_ELEMENT_NAME (element),
|
2015-06-30 21:13:28 +00:00
|
|
|
GES_TIMELINE_ELEMENT_NAME (child), off, LAYER_HEIGHT);
|
2013-06-29 23:31:23 +00:00
|
|
|
off = 0;
|
2015-06-30 21:13:28 +00:00
|
|
|
}
|
2013-06-29 23:31:23 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* Want the priority offset 'off' of each child to stay the same with
|
|
|
|
* the new priority. The offset is calculated from
|
|
|
|
* offset = min_priority - child_priority
|
2016-08-11 12:54:23 +00:00
|
|
|
*/
|
2020-03-02 13:25:21 +00:00
|
|
|
track_element_prio = min_prio - off;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2016-08-11 12:54:23 +00:00
|
|
|
if (track_element_prio > max_prio) {
|
2013-06-26 20:54:02 +00:00
|
|
|
GST_WARNING ("%p priority of %i, is outside of the its containing "
|
|
|
|
"layer space. (%d/%d) setting it to the maximum it can be",
|
2020-03-02 13:25:21 +00:00
|
|
|
container, priority, min_prio, max_prio);
|
2013-06-26 20:54:02 +00:00
|
|
|
|
2016-08-11 12:54:23 +00:00
|
|
|
track_element_prio = max_prio;
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
2016-08-11 12:54:23 +00:00
|
|
|
_set_priority0 (child, track_element_prio);
|
2013-06-26 20:54:02 +00:00
|
|
|
}
|
2020-03-02 13:25:21 +00:00
|
|
|
/* no need to re-sort the container since we maintained the relative
|
|
|
|
* offsets. As such, the height remains the same as well. */
|
|
|
|
priv->prevent_resort = FALSE;
|
|
|
|
priv->prevent_priority_offset_update = FALSE;
|
2013-06-26 20:54:02 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-03-01 22:08:39 +00:00
|
|
|
static guint32
|
|
|
|
_get_layer_priority (GESTimelineElement * element)
|
|
|
|
{
|
|
|
|
GESClip *clip = GES_CLIP (element);
|
|
|
|
|
|
|
|
if (clip->priv->layer == NULL)
|
|
|
|
return GES_TIMELINE_ELEMENT_NO_LAYER_PRIORITY;
|
|
|
|
|
|
|
|
return ges_layer_get_priority (clip->priv->layer);
|
|
|
|
}
|
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
/****************************************************
|
|
|
|
* *
|
|
|
|
* GESContainer virtual methods implementation *
|
|
|
|
* *
|
|
|
|
****************************************************/
|
|
|
|
|
2013-06-28 18:39:16 +00:00
|
|
|
static void
|
2013-03-24 17:42:55 +00:00
|
|
|
_compute_height (GESContainer * container)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
guint32 min_prio = G_MAXUINT32, max_prio = 0;
|
|
|
|
|
2013-06-28 18:39:16 +00:00
|
|
|
if (container->children == NULL) {
|
|
|
|
/* FIXME Why not 0! */
|
|
|
|
_ges_container_set_height (container, 1);
|
|
|
|
return;
|
|
|
|
}
|
2013-03-24 17:42:55 +00:00
|
|
|
|
|
|
|
/* Go over all childs and check if height has changed */
|
|
|
|
for (tmp = container->children; tmp; tmp = tmp->next) {
|
|
|
|
guint tck_priority = _PRIORITY (tmp->data);
|
|
|
|
|
|
|
|
if (tck_priority < min_prio)
|
|
|
|
min_prio = tck_priority;
|
|
|
|
if (tck_priority > max_prio)
|
|
|
|
max_prio = tck_priority;
|
|
|
|
}
|
|
|
|
|
2013-06-28 18:39:16 +00:00
|
|
|
_ges_container_set_height (container, max_prio - min_prio + 1);
|
2013-03-24 17:42:55 +00:00
|
|
|
}
|
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
static gboolean
|
|
|
|
_add_child (GESContainer * container, GESTimelineElement * element)
|
|
|
|
{
|
2020-03-02 12:56:03 +00:00
|
|
|
GESClipClass *klass = GES_CLIP_GET_CLASS (GES_CLIP (container));
|
2013-03-01 01:27:50 +00:00
|
|
|
guint max_prio, min_prio;
|
|
|
|
GESClipPrivate *priv = GES_CLIP (container)->priv;
|
2020-03-18 15:56:06 +00:00
|
|
|
GList *child_core_owners;
|
2013-03-01 01:27:50 +00:00
|
|
|
|
2013-06-28 23:15:59 +00:00
|
|
|
g_return_val_if_fail (GES_IS_TRACK_ELEMENT (element), FALSE);
|
|
|
|
|
2020-03-10 11:53:09 +00:00
|
|
|
if (element->timeline
|
|
|
|
&& element->timeline != GES_TIMELINE_ELEMENT_TIMELINE (container)) {
|
|
|
|
GST_WARNING_OBJECT (container, "Can not add the child %" GES_FORMAT
|
|
|
|
" because its timeline is %" GST_PTR_FORMAT " rather than the "
|
|
|
|
"clip's timeline %" GST_PTR_FORMAT, GES_ARGS (element),
|
|
|
|
element->timeline, GES_TIMELINE_ELEMENT_TIMELINE (container));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
child_core_owners =
|
|
|
|
ges_track_element_get_owners (GES_TRACK_ELEMENT (element));
|
|
|
|
if (child_core_owners
|
|
|
|
&& !g_list_find (child_core_owners, GES_CLIP (container))) {
|
|
|
|
GST_WARNING_OBJECT (container,
|
|
|
|
"Can not add child %" GES_FORMAT " because we are not in it core owner"
|
|
|
|
" element list", GES_ARGS (element));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-03-02 12:23:07 +00:00
|
|
|
/* NOTE: notifies are currently frozen by ges_container_add */
|
2013-03-30 11:30:47 +00:00
|
|
|
_get_priority_range (container, &min_prio, &max_prio);
|
2020-03-18 15:56:06 +00:00
|
|
|
if (child_core_owners) {
|
2020-03-02 12:56:03 +00:00
|
|
|
/* NOTE: Core track elements that are base effects are added like any
|
|
|
|
* other core clip. In particular, they are *not* added to the list of
|
|
|
|
* added effects, so we don not increase nb_effects. */
|
|
|
|
|
|
|
|
/* Set the core element to have the same in-point, which we don't
|
|
|
|
* apply to effects */
|
2020-03-10 15:27:20 +00:00
|
|
|
if (ges_track_element_has_internal_source (GES_TRACK_ELEMENT (element))) {
|
|
|
|
/* adding can fail if the max-duration of the element is smaller
|
|
|
|
* than the current in-point of the clip */
|
|
|
|
if (!_set_inpoint0 (element, _INPOINT (container))) {
|
|
|
|
GST_ERROR_OBJECT (element, "Could not set the in-point of the "
|
|
|
|
"element %" GES_FORMAT " to %" GST_TIME_FORMAT ". Not adding "
|
|
|
|
"as a child", GES_ARGS (element),
|
|
|
|
GST_TIME_ARGS (_INPOINT (container)));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Always add at the same priority, on top of existing effects */
|
|
|
|
_set_priority0 (element, min_prio + priv->nb_effects);
|
2020-03-18 15:56:06 +00:00
|
|
|
} else if (GES_CLIP_CLASS_CAN_ADD_EFFECTS (klass) &&
|
|
|
|
GES_IS_BASE_EFFECT (element)) {
|
2020-03-02 12:56:03 +00:00
|
|
|
GList *tmp;
|
|
|
|
/* Add the effect at the lowest priority among effects (just after
|
|
|
|
* the core elements). Need to shift the core elements up by 1
|
|
|
|
* to make room. */
|
|
|
|
GST_DEBUG_OBJECT (container, "Adding %ith effect: %" GES_FORMAT
|
|
|
|
" Priority %i", priv->nb_effects + 1, GES_ARGS (element),
|
2016-08-11 12:54:23 +00:00
|
|
|
min_prio + priv->nb_effects);
|
2013-03-01 01:27:50 +00:00
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
/* changing priorities, and updating their offset */
|
2020-03-02 13:25:21 +00:00
|
|
|
priv->prevent_resort = TRUE;
|
2013-03-01 01:27:50 +00:00
|
|
|
tmp = g_list_nth (GES_CONTAINER_CHILDREN (container), priv->nb_effects);
|
2020-03-02 12:56:03 +00:00
|
|
|
for (; tmp; tmp = tmp->next)
|
2013-03-01 01:27:50 +00:00
|
|
|
ges_timeline_element_set_priority (GES_TIMELINE_ELEMENT (tmp->data),
|
|
|
|
GES_TIMELINE_ELEMENT_PRIORITY (tmp->data) + 1);
|
|
|
|
|
2016-08-11 12:54:23 +00:00
|
|
|
_set_priority0 (element, min_prio + priv->nb_effects);
|
2013-03-01 01:27:50 +00:00
|
|
|
priv->nb_effects++;
|
2020-03-02 13:25:21 +00:00
|
|
|
priv->prevent_resort = FALSE;
|
2020-03-02 12:56:03 +00:00
|
|
|
/* no need to call _ges_container_sort_children (container) since
|
|
|
|
* there is no change to the ordering yet (this happens after the
|
|
|
|
* child is actually added) */
|
|
|
|
/* The height has already changed (increased by 1) */
|
|
|
|
_compute_height (container);
|
2013-03-01 01:27:50 +00:00
|
|
|
} else {
|
2020-03-02 12:56:03 +00:00
|
|
|
if (GES_IS_BASE_EFFECT (element))
|
|
|
|
GST_WARNING_OBJECT (container, "Can not add the effect %" GES_FORMAT
|
|
|
|
" because it is not a core element created by the clip itself "
|
|
|
|
"and the %s class does not allow for the adding extra effects",
|
|
|
|
GES_ARGS (element), G_OBJECT_CLASS_NAME (klass));
|
|
|
|
else if (GES_CLIP_CLASS_CAN_ADD_EFFECTS (klass))
|
|
|
|
GST_WARNING_OBJECT (container, "Can not add the track element %"
|
|
|
|
GES_FORMAT " because it is neither a core element created by "
|
|
|
|
"the clip itself, nor a GESBaseEffect", GES_ARGS (element));
|
|
|
|
else
|
|
|
|
GST_WARNING_OBJECT (container, "Can not add the track element %"
|
|
|
|
GES_FORMAT " because it is not a core element created by the "
|
|
|
|
"clip itself", GES_ARGS (element));
|
|
|
|
return FALSE;
|
2013-03-01 01:27:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_set_start0 (element, GES_TIMELINE_ELEMENT_START (container));
|
|
|
|
_set_duration0 (element, GES_TIMELINE_ELEMENT_DURATION (container));
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_remove_child (GESContainer * container, GESTimelineElement * element)
|
|
|
|
{
|
2020-03-02 13:35:20 +00:00
|
|
|
GESClipPrivate *priv = GES_CLIP (container)->priv;
|
|
|
|
|
2020-03-02 12:23:07 +00:00
|
|
|
/* NOTE: notifies are currently frozen by ges_container_add */
|
2020-03-18 15:56:06 +00:00
|
|
|
if (!ges_track_element_get_owners (GES_TRACK_ELEMENT (element))
|
2020-03-02 12:56:03 +00:00
|
|
|
&& GES_IS_BASE_EFFECT (element)) {
|
2020-03-02 13:35:20 +00:00
|
|
|
GList *tmp;
|
|
|
|
GST_DEBUG_OBJECT (container, "Resyncing effects priority.");
|
2013-03-01 01:27:50 +00:00
|
|
|
|
2020-03-02 13:35:20 +00:00
|
|
|
/* changing priorities, so preventing a re-sort */
|
|
|
|
priv->prevent_resort = TRUE;
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (container); tmp; tmp = tmp->next) {
|
|
|
|
guint32 sibling_prio = GES_TIMELINE_ELEMENT_PRIORITY (tmp->data);
|
|
|
|
if (sibling_prio > element->priority)
|
|
|
|
ges_timeline_element_set_priority (GES_TIMELINE_ELEMENT (tmp->data),
|
|
|
|
sibling_prio - 1);
|
|
|
|
}
|
|
|
|
priv->nb_effects--;
|
|
|
|
priv->prevent_resort = FALSE;
|
|
|
|
/* no need to re-sort the children since the rest keep the same
|
|
|
|
* relative priorities */
|
|
|
|
/* height may have changed */
|
|
|
|
_compute_height (container);
|
|
|
|
}
|
2020-03-18 15:56:06 +00:00
|
|
|
/* Core owner is not reset so that the child can be readded here in @container
|
|
|
|
* but not in any other clip by the end user */
|
2013-03-01 01:27:50 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2013-06-28 18:39:16 +00:00
|
|
|
static void
|
|
|
|
_child_added (GESContainer * container, GESTimelineElement * element)
|
|
|
|
{
|
2013-07-01 21:51:32 +00:00
|
|
|
g_signal_connect (G_OBJECT (element), "notify::priority",
|
|
|
|
G_CALLBACK (_child_priority_changed_cb), container);
|
2020-03-10 11:38:58 +00:00
|
|
|
g_signal_connect (G_OBJECT (element), "notify::in-point",
|
|
|
|
G_CALLBACK (_child_inpoint_changed_cb), container);
|
|
|
|
g_signal_connect (G_OBJECT (element), "notify::max-duration",
|
|
|
|
G_CALLBACK (_child_max_duration_changed_cb), container);
|
|
|
|
g_signal_connect (G_OBJECT (element), "notify::has-internal-source",
|
|
|
|
G_CALLBACK (_child_has_internal_source_changed_cb), container);
|
2016-02-09 11:31:10 +00:00
|
|
|
|
|
|
|
_child_priority_changed_cb (element, NULL, container);
|
2020-03-10 11:38:58 +00:00
|
|
|
_update_max_duration (container);
|
2013-06-28 18:39:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
_child_removed (GESContainer * container, GESTimelineElement * element)
|
|
|
|
{
|
2013-08-28 17:56:29 +00:00
|
|
|
g_signal_handlers_disconnect_by_func (element, _child_priority_changed_cb,
|
|
|
|
container);
|
2020-03-10 11:38:58 +00:00
|
|
|
g_signal_handlers_disconnect_by_func (element, _child_inpoint_changed_cb,
|
|
|
|
container);
|
|
|
|
g_signal_handlers_disconnect_by_func (element,
|
|
|
|
_child_max_duration_changed_cb, container);
|
|
|
|
g_signal_handlers_disconnect_by_func (element,
|
|
|
|
_child_has_internal_source_changed_cb, container);
|
|
|
|
|
|
|
|
_update_max_duration (container);
|
2013-06-28 18:39:16 +00:00
|
|
|
}
|
|
|
|
|
2013-03-01 23:25:17 +00:00
|
|
|
static void
|
2014-04-26 06:55:31 +00:00
|
|
|
add_clip_to_list (gpointer key, gpointer clip, GList ** list)
|
2013-03-01 23:25:17 +00:00
|
|
|
{
|
2014-04-26 06:55:31 +00:00
|
|
|
*list = g_list_prepend (*list, gst_object_ref (clip));
|
2013-03-01 23:25:17 +00:00
|
|
|
}
|
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
static void
|
|
|
|
_transfer_child (GESClip * from_clip, GESClip * to_clip,
|
|
|
|
GESTrackElement * child)
|
|
|
|
{
|
|
|
|
/* We need to bump the refcount to avoid the object to be destroyed */
|
|
|
|
gst_object_ref (child);
|
|
|
|
ges_container_remove (GES_CONTAINER (from_clip),
|
|
|
|
GES_TIMELINE_ELEMENT (child));
|
2020-03-18 15:56:06 +00:00
|
|
|
|
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
|
|
|
ges_track_element_add_owner (GES_TRACK_ELEMENT (child), to_clip);
|
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
ges_container_add (GES_CONTAINER (to_clip), GES_TIMELINE_ELEMENT (child));
|
|
|
|
gst_object_unref (child);
|
|
|
|
}
|
|
|
|
|
2013-03-01 23:25:17 +00:00
|
|
|
static GList *
|
|
|
|
_ungroup (GESContainer * container, gboolean recursive)
|
|
|
|
{
|
|
|
|
GESClip *tmpclip;
|
|
|
|
GESTrackType track_type;
|
|
|
|
GESTrackElement *track_element;
|
|
|
|
|
|
|
|
gboolean first_obj = TRUE;
|
2013-05-05 10:13:24 +00:00
|
|
|
GList *tmp, *children, *ret = NULL;
|
2013-03-01 23:25:17 +00:00
|
|
|
GESClip *clip = GES_CLIP (container);
|
|
|
|
GESTimelineElement *element = GES_TIMELINE_ELEMENT (container);
|
2013-04-23 23:04:04 +00:00
|
|
|
GESLayer *layer = clip->priv->layer;
|
2013-03-01 23:25:17 +00:00
|
|
|
GHashTable *_tracktype_clip = g_hash_table_new (g_int_hash, g_int_equal);
|
|
|
|
|
|
|
|
/* If there is no TrackElement, just return @container in a list */
|
|
|
|
if (GES_CONTAINER_CHILDREN (container) == NULL) {
|
|
|
|
GST_DEBUG ("No TrackElement, simply returning");
|
|
|
|
return g_list_prepend (ret, container);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We need a copy of the current list of tracks */
|
2013-07-10 19:24:28 +00:00
|
|
|
children = ges_container_get_children (container, FALSE);
|
2013-05-05 10:13:24 +00:00
|
|
|
for (tmp = children; tmp; tmp = tmp->next) {
|
2013-03-01 23:25:17 +00:00
|
|
|
track_element = GES_TRACK_ELEMENT (tmp->data);
|
|
|
|
track_type = ges_track_element_get_track_type (track_element);
|
|
|
|
|
|
|
|
tmpclip = g_hash_table_lookup (_tracktype_clip, &track_type);
|
|
|
|
if (tmpclip == NULL) {
|
|
|
|
if (G_UNLIKELY (first_obj == TRUE)) {
|
|
|
|
tmpclip = clip;
|
|
|
|
first_obj = FALSE;
|
|
|
|
} else {
|
|
|
|
tmpclip = GES_CLIP (ges_timeline_element_copy (element, FALSE));
|
|
|
|
if (layer) {
|
|
|
|
/* Add new container to the same layer as @container */
|
|
|
|
ges_clip_set_moving_from_layer (tmpclip, TRUE);
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_layer_add_clip (layer, tmpclip);
|
2013-03-01 23:25:17 +00:00
|
|
|
ges_clip_set_moving_from_layer (tmpclip, FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_hash_table_insert (_tracktype_clip, &track_type, tmpclip);
|
|
|
|
ges_clip_set_supported_formats (tmpclip, track_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Move trackelement to the container it is supposed to land into */
|
2020-03-02 12:56:03 +00:00
|
|
|
if (tmpclip != clip)
|
|
|
|
_transfer_child (clip, tmpclip, track_element);
|
2013-03-01 23:25:17 +00:00
|
|
|
}
|
2013-05-05 10:13:24 +00:00
|
|
|
g_list_free_full (children, gst_object_unref);
|
2014-04-26 06:55:31 +00:00
|
|
|
g_hash_table_foreach (_tracktype_clip, (GHFunc) add_clip_to_list, &ret);
|
2013-03-01 23:25:17 +00:00
|
|
|
g_hash_table_unref (_tracktype_clip);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-03-02 21:35:34 +00:00
|
|
|
static GESContainer *
|
|
|
|
_group (GList * containers)
|
|
|
|
{
|
|
|
|
CheckTrack *tracks = NULL;
|
|
|
|
GESTimeline *timeline = NULL;
|
|
|
|
GESTrackType supported_formats;
|
2013-04-23 23:04:04 +00:00
|
|
|
GESLayer *layer = NULL;
|
2013-03-02 21:35:34 +00:00
|
|
|
GList *tmp, *tmpclip, *tmpelement;
|
|
|
|
GstClockTime start, inpoint, duration;
|
|
|
|
|
|
|
|
GESAsset *asset = NULL;
|
|
|
|
GESContainer *ret = NULL;
|
|
|
|
guint nb_tracks = 0, i = 0;
|
|
|
|
|
|
|
|
start = inpoint = duration = GST_CLOCK_TIME_NONE;
|
|
|
|
|
2013-07-10 21:33:51 +00:00
|
|
|
if (!containers)
|
|
|
|
return NULL;
|
|
|
|
|
2013-03-02 21:35:34 +00:00
|
|
|
/* First check if all the containers are clips, if they
|
|
|
|
* all have the same start/inpoint/duration and are in the same
|
|
|
|
* layer.
|
|
|
|
*
|
|
|
|
* We also need to make sure that all source have been created by the
|
|
|
|
* same asset, keep the information */
|
|
|
|
for (tmp = containers; tmp; tmp = tmp->next) {
|
|
|
|
GESClip *clip;
|
|
|
|
GESTimeline *tmptimeline;
|
|
|
|
GESContainer *tmpcontainer;
|
|
|
|
GESTimelineElement *element;
|
|
|
|
|
|
|
|
tmpcontainer = GES_CONTAINER (tmp->data);
|
|
|
|
element = GES_TIMELINE_ELEMENT (tmp->data);
|
|
|
|
if (GES_IS_CLIP (element) == FALSE) {
|
|
|
|
GST_DEBUG ("Can only work with clips");
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
clip = GES_CLIP (tmp->data);
|
2013-06-28 23:15:59 +00:00
|
|
|
tmptimeline = GES_TIMELINE_ELEMENT_TIMELINE (element);
|
2013-03-02 21:35:34 +00:00
|
|
|
if (!timeline) {
|
|
|
|
GList *tmptrack;
|
|
|
|
|
|
|
|
start = _START (tmpcontainer);
|
|
|
|
inpoint = _INPOINT (tmpcontainer);
|
|
|
|
duration = _DURATION (tmpcontainer);
|
|
|
|
timeline = tmptimeline;
|
|
|
|
layer = clip->priv->layer;
|
|
|
|
nb_tracks = g_list_length (GES_TIMELINE_GET_TRACKS (timeline));
|
|
|
|
tracks = g_new0 (CheckTrack, nb_tracks);
|
|
|
|
|
|
|
|
for (tmptrack = GES_TIMELINE_GET_TRACKS (timeline); tmptrack;
|
|
|
|
tmptrack = tmptrack->next) {
|
|
|
|
tracks[i].track = tmptrack->data;
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
} else {
|
2020-01-09 12:11:35 +00:00
|
|
|
/* FIXME: we should allow the inpoint to be different if not a
|
|
|
|
* core track element of the clip.
|
|
|
|
* E.g. a GESEffect on a GESUriClip */
|
2013-03-02 21:35:34 +00:00
|
|
|
if (start != _START (tmpcontainer) ||
|
|
|
|
inpoint != _INPOINT (tmpcontainer) ||
|
|
|
|
duration != _DURATION (tmpcontainer) || clip->priv->layer != layer) {
|
|
|
|
GST_INFO ("All children must have the same start, inpoint, duration "
|
2020-03-18 15:56:06 +00:00
|
|
|
"and be in the same layer");
|
2013-03-02 21:35:34 +00:00
|
|
|
|
|
|
|
goto done;
|
|
|
|
} else {
|
|
|
|
GList *tmp2;
|
|
|
|
|
|
|
|
for (tmp2 = GES_CONTAINER_CHILDREN (tmp->data); tmp2; tmp2 = tmp2->next) {
|
|
|
|
GESTrackElement *track_element = GES_TRACK_ELEMENT (tmp2->data);
|
|
|
|
|
|
|
|
if (GES_IS_SOURCE (track_element)) {
|
|
|
|
guint i;
|
|
|
|
|
|
|
|
for (i = 0; i < nb_tracks; i++) {
|
|
|
|
if (tracks[i].track ==
|
|
|
|
ges_track_element_get_track (track_element)) {
|
|
|
|
if (tracks[i].source) {
|
|
|
|
GST_INFO ("Can not link clips with various source for a "
|
|
|
|
"same track");
|
|
|
|
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
tracks[i].source = track_element;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Then check that all sources have been created by the same asset,
|
|
|
|
* otherwise we can not group */
|
|
|
|
for (i = 0; i < nb_tracks; i++) {
|
|
|
|
if (tracks[i].source == NULL) {
|
|
|
|
GST_FIXME ("Check what to do here as we might end up having a mess");
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME Check what to do if we have source that have no assets */
|
|
|
|
if (!asset) {
|
|
|
|
asset =
|
|
|
|
ges_extractable_get_asset (GES_EXTRACTABLE
|
|
|
|
(ges_timeline_element_get_parent (GES_TIMELINE_ELEMENT (tracks
|
|
|
|
[i].source))));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (asset !=
|
|
|
|
ges_extractable_get_asset (GES_EXTRACTABLE
|
|
|
|
(ges_timeline_element_get_parent (GES_TIMELINE_ELEMENT (tracks
|
|
|
|
[i].source))))) {
|
|
|
|
GST_INFO ("Can not link clips with source coming from different assets");
|
|
|
|
|
|
|
|
goto done;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* And now pass all TrackElements to the first clip,
|
|
|
|
* and remove others from the layer (updating the supported formats) */
|
|
|
|
ret = containers->data;
|
|
|
|
supported_formats = GES_CLIP (ret)->priv->supportedformats;
|
|
|
|
for (tmpclip = containers->next; tmpclip; tmpclip = tmpclip->next) {
|
|
|
|
GESClip *cclip = tmpclip->data;
|
2013-07-10 19:24:28 +00:00
|
|
|
GList *children = ges_container_get_children (GES_CONTAINER (cclip), FALSE);
|
2013-03-02 21:35:34 +00:00
|
|
|
|
2013-07-02 14:56:40 +00:00
|
|
|
for (tmpelement = children; tmpelement; tmpelement = tmpelement->next) {
|
2020-03-02 12:56:03 +00:00
|
|
|
GESTrackElement *celement = GES_TRACK_ELEMENT (tmpelement->data);
|
|
|
|
_transfer_child (cclip, GES_CLIP (ret), celement);
|
|
|
|
supported_formats |= ges_track_element_get_track_type (celement);
|
2013-03-02 21:35:34 +00:00
|
|
|
}
|
2013-07-02 14:56:40 +00:00
|
|
|
g_list_free_full (children, gst_object_unref);
|
2013-03-02 21:35:34 +00:00
|
|
|
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_layer_remove_clip (layer, tmpclip->data);
|
2013-03-02 21:35:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ges_clip_set_supported_formats (GES_CLIP (ret), supported_formats);
|
|
|
|
|
|
|
|
done:
|
|
|
|
if (tracks)
|
|
|
|
g_free (tracks);
|
|
|
|
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
static void
|
|
|
|
_deep_copy (GESTimelineElement * element, GESTimelineElement * copy)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
GESClip *self = GES_CLIP (element), *ccopy = GES_CLIP (copy);
|
2020-03-02 12:56:03 +00:00
|
|
|
GESTrackElement *el, *el_copy;
|
2016-01-01 10:56:27 +00:00
|
|
|
|
2019-05-25 22:51:08 +00:00
|
|
|
if (!self->priv->layer)
|
|
|
|
return;
|
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (element); tmp; tmp = tmp->next) {
|
2020-03-02 12:56:03 +00:00
|
|
|
el = GES_TRACK_ELEMENT (tmp->data);
|
2020-03-03 18:00:51 +00:00
|
|
|
/* copies the children properties */
|
2020-03-18 15:56:06 +00:00
|
|
|
el_copy = GES_TRACK_ELEMENT (ges_timeline_element_copy (tmp->data, TRUE));
|
|
|
|
|
|
|
|
if (ges_track_element_get_owners (el))
|
|
|
|
ges_track_element_add_owner (el_copy, ccopy);
|
|
|
|
|
2020-03-03 18:00:51 +00:00
|
|
|
ges_track_element_copy_bindings (el, el_copy, GST_CLOCK_TIME_NONE);
|
2016-01-01 10:56:27 +00:00
|
|
|
ccopy->priv->copied_track_elements =
|
2020-03-02 12:56:03 +00:00
|
|
|
g_list_append (ccopy->priv->copied_track_elements, el_copy);
|
2016-01-01 10:56:27 +00:00
|
|
|
}
|
|
|
|
|
2019-05-25 22:51:08 +00:00
|
|
|
ccopy->priv->copied_layer = g_object_ref (self->priv->layer);
|
2016-01-01 10:56:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static GESTimelineElement *
|
2015-06-29 16:04:32 +00:00
|
|
|
_paste (GESTimelineElement * element, GESTimelineElement * ref,
|
|
|
|
GstClockTime paste_position)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
GESClip *self = GES_CLIP (element);
|
2016-01-01 10:56:27 +00:00
|
|
|
GESClip *nclip = GES_CLIP (ges_timeline_element_copy (element, FALSE));
|
|
|
|
|
|
|
|
if (self->priv->copied_layer)
|
|
|
|
nclip->priv->copied_layer = g_object_ref (self->priv->copied_layer);
|
|
|
|
ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (nclip), paste_position);
|
2015-06-29 16:04:32 +00:00
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
for (tmp = self->priv->copied_track_elements; tmp; tmp = tmp->next) {
|
2015-06-29 16:04:32 +00:00
|
|
|
GESTrackElement *new_trackelement, *trackelement =
|
|
|
|
GES_TRACK_ELEMENT (tmp->data);
|
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
/* NOTE: we do not deep copy the track element, we instead call
|
|
|
|
* ges_track_element_copy_properties explicitly, which is the
|
|
|
|
* deep_copy for the GESTrackElementClass. */
|
2015-06-29 16:04:32 +00:00
|
|
|
new_trackelement =
|
2020-03-18 15:56:06 +00:00
|
|
|
GES_TRACK_ELEMENT (ges_timeline_element_copy (GES_TIMELINE_ELEMENT
|
|
|
|
(trackelement), FALSE));
|
2015-06-29 16:04:32 +00:00
|
|
|
if (new_trackelement == NULL) {
|
|
|
|
GST_WARNING_OBJECT (trackelement, "Could not create a copy");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (trackelement))
|
|
|
|
ges_track_element_add_owner (new_trackelement, nclip);
|
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
ges_container_add (GES_CONTAINER (nclip),
|
2015-06-29 16:04:32 +00:00
|
|
|
GES_TIMELINE_ELEMENT (new_trackelement));
|
|
|
|
|
|
|
|
ges_track_element_copy_properties (GES_TIMELINE_ELEMENT (trackelement),
|
|
|
|
GES_TIMELINE_ELEMENT (new_trackelement));
|
|
|
|
|
|
|
|
ges_track_element_copy_bindings (trackelement, new_trackelement,
|
|
|
|
GST_CLOCK_TIME_NONE);
|
|
|
|
}
|
|
|
|
|
2020-03-03 18:00:51 +00:00
|
|
|
/* FIXME: should we bypass the select-tracks-for-object signal when
|
|
|
|
* copying and pasting? */
|
|
|
|
if (self->priv->copied_layer) {
|
|
|
|
if (!ges_layer_add_clip (self->priv->copied_layer, nclip)) {
|
|
|
|
GST_INFO ("%" GES_FORMAT " could not be pasted to %" GST_TIME_FORMAT,
|
|
|
|
GES_ARGS (element), GST_TIME_ARGS (paste_position));
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
return GES_TIMELINE_ELEMENT (nclip);
|
2015-06-29 16:04:32 +00:00
|
|
|
}
|
2013-06-28 15:23:27 +00:00
|
|
|
|
2016-07-28 20:02:05 +00:00
|
|
|
static gboolean
|
|
|
|
_lookup_child (GESTimelineElement * self, const gchar * prop_name,
|
|
|
|
GObject ** child, GParamSpec ** pspec)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
|
|
|
|
if (GES_TIMELINE_ELEMENT_CLASS (ges_clip_parent_class)->lookup_child (self,
|
|
|
|
prop_name, child, pspec))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (self); tmp; tmp = tmp->next) {
|
|
|
|
if (ges_timeline_element_lookup_child (tmp->data, prop_name, child, pspec))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
2013-03-02 21:35:34 +00:00
|
|
|
|
2013-03-01 23:25:17 +00:00
|
|
|
/****************************************************
|
|
|
|
* *
|
|
|
|
* GObject virtual methods implementation *
|
|
|
|
* *
|
|
|
|
****************************************************/
|
2009-08-06 10:14:37 +00:00
|
|
|
static void
|
2013-01-20 15:42:29 +00:00
|
|
|
ges_clip_get_property (GObject * object, guint property_id,
|
2009-08-04 15:16:31 +00:00
|
|
|
GValue * value, GParamSpec * pspec)
|
2009-08-04 15:13:11 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
GESClip *clip = GES_CLIP (object);
|
2009-08-07 18:33:18 +00:00
|
|
|
|
2009-08-04 15:13:11 +00:00
|
|
|
switch (property_id) {
|
2010-11-28 12:24:07 +00:00
|
|
|
case PROP_LAYER:
|
2013-02-15 02:34:48 +00:00
|
|
|
g_value_set_object (value, clip->priv->layer);
|
2010-11-28 12:24:07 +00:00
|
|
|
break;
|
2011-12-01 03:18:30 +00:00
|
|
|
case PROP_SUPPORTED_FORMATS:
|
2013-02-15 02:34:48 +00:00
|
|
|
g_value_set_flags (value, clip->priv->supportedformats);
|
2011-12-01 03:18:30 +00:00
|
|
|
break;
|
2009-08-04 15:16:31 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
2009-08-04 15:13:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-20 15:42:29 +00:00
|
|
|
ges_clip_set_property (GObject * object, guint property_id,
|
2009-08-04 15:16:31 +00:00
|
|
|
const GValue * value, GParamSpec * pspec)
|
2009-08-04 15:13:11 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
GESClip *clip = GES_CLIP (object);
|
2009-08-07 18:33:18 +00:00
|
|
|
|
2009-08-04 15:13:11 +00:00
|
|
|
switch (property_id) {
|
2011-12-01 03:18:30 +00:00
|
|
|
case PROP_SUPPORTED_FORMATS:
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_set_supported_formats (clip, g_value_get_flags (value));
|
2011-12-01 03:18:30 +00:00
|
|
|
break;
|
2009-08-04 15:16:31 +00:00
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec);
|
2009-08-04 15:13:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-01 10:56:27 +00:00
|
|
|
static void
|
2019-03-08 20:45:27 +00:00
|
|
|
ges_clip_dispose (GObject * object)
|
2016-01-01 10:56:27 +00:00
|
|
|
{
|
|
|
|
GESClip *self = GES_CLIP (object);
|
|
|
|
|
|
|
|
g_list_free_full (self->priv->copied_track_elements, g_object_unref);
|
2019-03-08 20:45:27 +00:00
|
|
|
self->priv->copied_track_elements = NULL;
|
|
|
|
g_clear_object (&self->priv->copied_layer);
|
2016-01-01 10:56:27 +00:00
|
|
|
|
2019-03-08 20:45:27 +00:00
|
|
|
G_OBJECT_CLASS (ges_clip_parent_class)->dispose (object);
|
2016-01-01 10:56:27 +00:00
|
|
|
}
|
|
|
|
|
2019-03-08 20:45:27 +00:00
|
|
|
|
2009-08-04 15:13:11 +00:00
|
|
|
static void
|
2013-01-20 15:42:29 +00:00
|
|
|
ges_clip_class_init (GESClipClass * klass)
|
2009-08-04 15:13:11 +00:00
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2013-03-01 01:27:50 +00:00
|
|
|
GESContainerClass *container_class = GES_CONTAINER_CLASS (klass);
|
2013-01-15 13:52:17 +00:00
|
|
|
GESTimelineElementClass *element_class = GES_TIMELINE_ELEMENT_CLASS (klass);
|
2009-08-04 15:13:11 +00:00
|
|
|
|
2013-01-20 15:42:29 +00:00
|
|
|
object_class->get_property = ges_clip_get_property;
|
|
|
|
object_class->set_property = ges_clip_set_property;
|
2019-03-08 20:45:27 +00:00
|
|
|
object_class->dispose = ges_clip_dispose;
|
2013-01-26 15:31:33 +00:00
|
|
|
klass->create_track_elements = ges_clip_create_track_elements_func;
|
|
|
|
klass->create_track_element = NULL;
|
2010-07-09 09:51:21 +00:00
|
|
|
|
2011-12-01 03:18:30 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* GESClip:supported-formats:
|
2011-12-01 03:18:30 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* The #GESTrackType-s that the clip supports, which it can create
|
|
|
|
* #GESTrackElement-s for. Note that this can be a combination of
|
|
|
|
* #GESTrackType flags to indicate support for several
|
|
|
|
* #GESTrackElement:track-type elements.
|
2011-12-01 03:18:30 +00:00
|
|
|
*/
|
|
|
|
properties[PROP_SUPPORTED_FORMATS] = g_param_spec_flags ("supported-formats",
|
2020-01-09 12:11:35 +00:00
|
|
|
"Supported formats", "Formats supported by the clip",
|
2012-11-18 15:46:05 +00:00
|
|
|
GES_TYPE_TRACK_TYPE, GES_TRACK_TYPE_AUDIO | GES_TRACK_TYPE_VIDEO,
|
2011-12-01 03:18:30 +00:00
|
|
|
G_PARAM_READWRITE | G_PARAM_CONSTRUCT);
|
|
|
|
|
|
|
|
g_object_class_install_property (object_class, PROP_SUPPORTED_FORMATS,
|
|
|
|
properties[PROP_SUPPORTED_FORMATS]);
|
|
|
|
|
2011-02-09 10:21:02 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* GESClip:layer:
|
2010-11-28 12:24:07 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* The layer this clip lies in.
|
|
|
|
*
|
|
|
|
* If you want to connect to this property's #GObject::notify signal,
|
|
|
|
* you should connect to it with g_signal_connect_after() since the
|
|
|
|
* signal emission may be stopped internally.
|
2010-11-28 12:24:07 +00:00
|
|
|
*/
|
2011-02-09 10:21:02 +00:00
|
|
|
properties[PROP_LAYER] = g_param_spec_object ("layer", "Layer",
|
2013-04-23 23:04:04 +00:00
|
|
|
"The GESLayer where this clip is being used.",
|
|
|
|
GES_TYPE_LAYER, G_PARAM_READABLE);
|
2010-11-28 12:24:07 +00:00
|
|
|
g_object_class_install_property (object_class, PROP_LAYER,
|
2011-02-09 10:21:02 +00:00
|
|
|
properties[PROP_LAYER]);
|
2010-11-28 12:24:07 +00:00
|
|
|
|
2013-01-15 13:52:17 +00:00
|
|
|
element_class->ripple = _ripple;
|
|
|
|
element_class->ripple_end = _ripple_end;
|
|
|
|
element_class->roll_start = _roll_start;
|
|
|
|
element_class->roll_end = _roll_end;
|
|
|
|
element_class->trim = _trim;
|
2013-06-26 20:54:02 +00:00
|
|
|
element_class->set_start = _set_start;
|
|
|
|
element_class->set_duration = _set_duration;
|
|
|
|
element_class->set_inpoint = _set_inpoint;
|
|
|
|
element_class->set_priority = _set_priority;
|
|
|
|
element_class->set_max_duration = _set_max_duration;
|
2015-06-29 16:04:32 +00:00
|
|
|
element_class->paste = _paste;
|
2016-01-01 10:56:27 +00:00
|
|
|
element_class->deep_copy = _deep_copy;
|
2016-07-28 20:02:05 +00:00
|
|
|
element_class->lookup_child = _lookup_child;
|
2019-03-01 22:08:39 +00:00
|
|
|
element_class->get_layer_priority = _get_layer_priority;
|
2013-03-01 01:27:50 +00:00
|
|
|
|
|
|
|
container_class->add_child = _add_child;
|
|
|
|
container_class->remove_child = _remove_child;
|
2013-06-28 18:39:16 +00:00
|
|
|
container_class->child_removed = _child_removed;
|
|
|
|
container_class->child_added = _child_added;
|
2013-03-01 23:25:17 +00:00
|
|
|
container_class->ungroup = _ungroup;
|
2013-03-02 21:35:34 +00:00
|
|
|
container_class->group = _group;
|
2013-03-23 08:46:38 +00:00
|
|
|
container_class->grouping_priority = G_MAXUINT;
|
2009-08-04 15:13:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-20 15:42:29 +00:00
|
|
|
ges_clip_init (GESClip * self)
|
2009-08-04 15:13:11 +00:00
|
|
|
{
|
2020-03-10 11:38:58 +00:00
|
|
|
GESClipPrivate *priv;
|
|
|
|
priv = self->priv = ges_clip_get_instance_private (self);
|
|
|
|
priv->layer = NULL;
|
|
|
|
priv->nb_effects = 0;
|
|
|
|
priv->prevent_priority_offset_update = FALSE;
|
|
|
|
priv->prevent_resort = FALSE;
|
|
|
|
priv->updating_max_duration = FALSE;
|
|
|
|
priv->prevent_max_duration_update = FALSE;
|
|
|
|
priv->setting_inpoint = FALSE;
|
2012-11-20 03:29:23 +00:00
|
|
|
}
|
|
|
|
|
2009-08-06 09:23:01 +00:00
|
|
|
/**
|
2013-01-26 15:31:33 +00:00
|
|
|
* ges_clip_create_track_element:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @type: The track to create an element for
|
2009-08-06 09:23:01 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Creates the core #GESTrackElement of the clip, of the given track type.
|
2009-08-06 09:23:01 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Note, unlike ges_clip_create_track_elements(), this does not add the
|
|
|
|
* created track element to the clip or set their timings.
|
|
|
|
*
|
|
|
|
* Returns: (transfer floating) (nullable): The element created
|
|
|
|
* by @clip, or %NULL if @clip can not provide a track element for the
|
|
|
|
* given @type or an error occurred.
|
2009-08-06 09:23:01 +00:00
|
|
|
*/
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_create_track_element (GESClip * clip, GESTrackType type)
|
2009-08-06 09:23:01 +00:00
|
|
|
{
|
2013-01-20 15:42:29 +00:00
|
|
|
GESClipClass *class;
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *res;
|
2009-08-06 09:23:01 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Creating track element for %s",
|
2012-12-20 23:23:54 +00:00
|
|
|
ges_track_type_name (type));
|
2013-02-15 02:34:48 +00:00
|
|
|
if (!(type & clip->priv->supportedformats)) {
|
|
|
|
GST_DEBUG_OBJECT (clip, "We don't support this track type %i", type);
|
2012-11-18 15:46:05 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
class = GES_CLIP_GET_CLASS (clip);
|
2009-08-06 09:23:01 +00:00
|
|
|
|
2013-01-26 15:31:33 +00:00
|
|
|
if (G_UNLIKELY (class->create_track_element == NULL)) {
|
|
|
|
GST_ERROR ("No 'create_track_element' implementation available fo type %s",
|
2013-02-15 02:34:48 +00:00
|
|
|
G_OBJECT_TYPE_NAME (clip));
|
2009-08-06 15:38:43 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
res = class->create_track_element (clip, type);
|
2009-08-06 15:38:43 +00:00
|
|
|
return res;
|
2010-07-07 15:07:33 +00:00
|
|
|
|
2009-08-06 15:38:43 +00:00
|
|
|
}
|
|
|
|
|
2010-07-07 14:51:39 +00:00
|
|
|
/**
|
2013-01-26 15:31:33 +00:00
|
|
|
* ges_clip_create_track_elements:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @type: The track-type to create elements for
|
2010-07-07 14:51:39 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Creates the core #GESTrackElement-s of the clip, of the given track
|
|
|
|
* type, and adds them to the clip.
|
2010-07-07 14:51:39 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer container) (element-type GESTrackElement): A list of
|
|
|
|
* the #GESTrackElement-s created by @clip for the given @type, or %NULL
|
|
|
|
* if no track elements are created or an error occurred.
|
2010-07-07 14:51:39 +00:00
|
|
|
*/
|
|
|
|
|
2012-12-20 23:23:54 +00:00
|
|
|
GList *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_create_track_elements (GESClip * clip, GESTrackType type)
|
2010-07-07 14:51:39 +00:00
|
|
|
{
|
2016-06-18 20:16:00 +00:00
|
|
|
GList *result = NULL, *tmp, *children;
|
2013-01-20 15:42:29 +00:00
|
|
|
GESClipClass *klass;
|
2019-02-21 20:24:51 +00:00
|
|
|
gboolean readding_effects_only = TRUE;
|
2010-07-07 14:51:39 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
klass = GES_CLIP_GET_CLASS (clip);
|
2010-07-07 14:51:39 +00:00
|
|
|
|
2013-01-26 15:31:33 +00:00
|
|
|
if (!(klass->create_track_elements)) {
|
|
|
|
GST_WARNING ("no GESClip::create_track_elements implentation");
|
2012-12-20 23:23:54 +00:00
|
|
|
return NULL;
|
2010-07-07 14:51:39 +00:00
|
|
|
}
|
2012-04-20 23:02:19 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Creating TrackElements for type: %s",
|
2012-12-20 23:23:54 +00:00
|
|
|
ges_track_type_name (type));
|
2016-06-18 20:16:00 +00:00
|
|
|
children = ges_container_get_children (GES_CONTAINER (clip), TRUE);
|
|
|
|
for (tmp = children; tmp; tmp = tmp->next) {
|
|
|
|
GESTrackElement *child = GES_TRACK_ELEMENT (tmp->data);
|
|
|
|
|
2019-02-21 20:24:51 +00:00
|
|
|
if (!ges_track_element_get_track (child)
|
|
|
|
&& ges_track_element_get_track_type (child) & type) {
|
2016-06-18 20:16:00 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Removing for reusage: %" GST_PTR_FORMAT, child);
|
2019-02-21 20:24:51 +00:00
|
|
|
result = g_list_append (result, g_object_ref (child));
|
2016-06-18 20:16:00 +00:00
|
|
|
ges_container_remove (GES_CONTAINER (clip), tmp->data);
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2019-02-21 20:24:51 +00:00
|
|
|
readding_effects_only = FALSE;
|
2016-06-18 20:16:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
g_list_free_full (children, gst_object_unref);
|
|
|
|
|
2020-01-09 12:11:35 +00:00
|
|
|
/* FIXME: we need something smarter to determine whether we should
|
|
|
|
* create the track elements.
|
|
|
|
* Currently, if the clip contains at least one element with a matching
|
|
|
|
* track-type, not in a track and not a GESBaseEffect, we will not
|
|
|
|
* recreate the track elements! But this is not a reliable indicator.
|
|
|
|
*
|
|
|
|
* For example, consider a uri clip that creates two audio track
|
|
|
|
* elements: El_A and El_B. First, we add the clip to a timeline that
|
|
|
|
* only has a single track: Track_A, and we connect to the timeline's
|
|
|
|
* ::select-tracks-for-object signal to only allow El_A to end up in
|
|
|
|
* Track_A. As such, whilst both El_A and El_B are initially created,
|
|
|
|
* El_B will eventually be removed from the clip since it has no track
|
|
|
|
* (see clip_track_element_added_cb in ges-timeline.c). As such, we now
|
|
|
|
* have a clip that only contains El_A.
|
|
|
|
* Next, we remove Track_A from the timeline. El_A will remain a child
|
|
|
|
* of the clip, but now has its track unset.
|
|
|
|
* Next, we add Track_B to the timeline, and we connect to the
|
|
|
|
* timeline's ::select-tracks-for-object signal to only allow El_B to
|
|
|
|
* end up in Track_B.
|
|
|
|
*
|
|
|
|
* However, since the clip contains an audio track element, that is not
|
|
|
|
* an effect and has no track set: El_A. Therefore, the
|
|
|
|
* create_track_elements method below will not be called, so we will not
|
|
|
|
* have an El_B created for Track_B!
|
|
|
|
*
|
|
|
|
* Moreover, even if we did recreate the track elements, we would be
|
|
|
|
* creating El_A again! We could destroy and recreate El_A instead, or
|
|
|
|
* we would need a way to determine exactly which elements need to be
|
|
|
|
* recreated.
|
|
|
|
*/
|
2019-02-21 20:24:51 +00:00
|
|
|
if (readding_effects_only) {
|
2020-03-02 12:56:03 +00:00
|
|
|
GList *track_elements = klass->create_track_elements (clip, type);
|
2020-03-18 15:56:06 +00:00
|
|
|
for (tmp = track_elements; tmp; tmp = tmp->next)
|
|
|
|
ges_track_element_add_owner (tmp->data, clip);
|
2020-03-02 12:56:03 +00:00
|
|
|
result = g_list_concat (track_elements, result);
|
2016-06-18 20:16:00 +00:00
|
|
|
}
|
2010-07-07 14:51:39 +00:00
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
for (tmp = result; tmp; tmp = tmp->next) {
|
2020-03-02 12:56:03 +00:00
|
|
|
ges_container_add (GES_CONTAINER (clip), GES_TIMELINE_ELEMENT (tmp->data));
|
2013-03-01 01:27:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
2012-01-16 12:37:22 +00:00
|
|
|
}
|
|
|
|
|
2010-12-09 11:53:07 +00:00
|
|
|
/*
|
2013-01-26 15:31:33 +00:00
|
|
|
* default implementation of GESClipClass::create_track_elements
|
2010-12-09 11:53:07 +00:00
|
|
|
*/
|
2012-12-20 23:23:54 +00:00
|
|
|
GList *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_create_track_elements_func (GESClip * clip, GESTrackType type)
|
2010-07-07 14:51:39 +00:00
|
|
|
{
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *result;
|
2010-07-07 14:51:39 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Creating trackelement for track: %s",
|
2012-12-20 23:23:54 +00:00
|
|
|
ges_track_type_name (type));
|
2013-02-15 02:34:48 +00:00
|
|
|
result = ges_clip_create_track_element (clip, type);
|
2010-07-07 14:51:39 +00:00
|
|
|
if (!result) {
|
2013-02-08 19:39:18 +00:00
|
|
|
GST_DEBUG ("Did not create track element");
|
2012-12-20 23:23:54 +00:00
|
|
|
return NULL;
|
2010-07-07 14:51:39 +00:00
|
|
|
}
|
2012-01-30 16:47:42 +00:00
|
|
|
|
2012-12-20 23:23:54 +00:00
|
|
|
return g_list_append (NULL, result);
|
2010-07-07 14:51:39 +00:00
|
|
|
}
|
|
|
|
|
2009-08-06 15:38:43 +00:00
|
|
|
void
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_clip_set_layer (GESClip * clip, GESLayer * layer)
|
2009-08-06 15:38:43 +00:00
|
|
|
{
|
2013-07-02 23:47:48 +00:00
|
|
|
if (layer == clip->priv->layer)
|
|
|
|
return;
|
2009-08-06 15:38:43 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
clip->priv->layer = layer;
|
2013-03-23 07:45:00 +00:00
|
|
|
|
2013-07-02 23:47:48 +00:00
|
|
|
GST_DEBUG ("clip:%p, layer:%p", clip, layer);
|
|
|
|
|
2013-03-23 07:45:00 +00:00
|
|
|
/* We do not want to notify the setting of layer = NULL when
|
|
|
|
* it is actually the result of a move between layer (as we know
|
|
|
|
* that it will be added to another layer right after, and this
|
|
|
|
* is what imports here.) */
|
2019-02-09 21:59:08 +00:00
|
|
|
if (!ELEMENT_FLAG_IS_SET (clip, GES_CLIP_IS_MOVING))
|
2013-03-23 07:45:00 +00:00
|
|
|
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
|
2009-08-06 15:38:43 +00:00
|
|
|
}
|
|
|
|
|
2011-06-06 19:55:47 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_set_moving_from_layer:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
2013-02-15 02:34:48 +00:00
|
|
|
* @is_moving: %TRUE if you want to start moving @clip to another layer
|
2020-01-09 12:11:35 +00:00
|
|
|
* %FALSE when you finished moving it
|
2011-06-06 19:55:47 +00:00
|
|
|
*
|
2013-02-15 02:34:48 +00:00
|
|
|
* Sets the clip in a moving to layer state. You might rather use the
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_move_to_layer function to move #GESClip-s
|
2011-06-06 19:55:47 +00:00
|
|
|
* from a layer to another.
|
|
|
|
**/
|
|
|
|
void
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_set_moving_from_layer (GESClip * clip, gboolean is_moving)
|
2011-06-06 19:55:47 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_if_fail (GES_IS_CLIP (clip));
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2018-05-14 01:12:35 +00:00
|
|
|
if (is_moving)
|
2019-02-09 21:59:08 +00:00
|
|
|
ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING);
|
2018-05-14 01:12:35 +00:00
|
|
|
else
|
2019-02-09 21:59:08 +00:00
|
|
|
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
2011-06-06 19:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_is_moving_from_layer:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
2011-06-06 19:55:47 +00:00
|
|
|
*
|
2013-02-15 02:34:48 +00:00
|
|
|
* Tells you if the clip is currently moving from a layer to another.
|
2013-01-20 15:42:29 +00:00
|
|
|
* You might rather use the ges_clip_move_to_layer function to
|
|
|
|
* move #GESClip-s from a layer to another.
|
2011-06-06 19:55:47 +00:00
|
|
|
*
|
2013-02-15 02:34:48 +00:00
|
|
|
* Returns: %TRUE if @clip is currently moving from its current layer
|
2020-01-09 12:11:35 +00:00
|
|
|
* %FALSE otherwize.
|
2011-06-06 19:55:47 +00:00
|
|
|
**/
|
|
|
|
gboolean
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_is_moving_from_layer (GESClip * clip)
|
2011-06-06 19:55:47 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
2012-04-20 23:02:19 +00:00
|
|
|
|
2019-02-09 21:59:08 +00:00
|
|
|
return ELEMENT_FLAG_IS_SET (clip, GES_CLIP_IS_MOVING);
|
2011-06-06 19:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_move_to_layer:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @layer: The new layer
|
2011-06-06 19:55:47 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Moves a clip to a new layer. If the clip already exists in a layer, it
|
|
|
|
* is first removed from its current layer before being added to the new
|
|
|
|
* layer.
|
2011-06-06 19:55:47 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: %TRUE if @clip was successfully moved to @layer.
|
2011-06-06 19:55:47 +00:00
|
|
|
*/
|
|
|
|
gboolean
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
|
2011-06-06 19:55:47 +00:00
|
|
|
{
|
|
|
|
gboolean ret;
|
2013-04-23 23:04:04 +00:00
|
|
|
GESLayer *current_layer;
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
2013-04-23 23:04:04 +00:00
|
|
|
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2019-03-01 22:32:19 +00:00
|
|
|
ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING);
|
|
|
|
if (layer->timeline
|
|
|
|
&& !timeline_tree_can_move_element (timeline_get_tree (layer->timeline),
|
|
|
|
GES_TIMELINE_ELEMENT (clip),
|
|
|
|
ges_layer_get_priority (layer),
|
|
|
|
GES_TIMELINE_ELEMENT_START (clip),
|
|
|
|
GES_TIMELINE_ELEMENT_DURATION (clip), NULL)) {
|
|
|
|
GST_INFO_OBJECT (layer, "Clip %" GES_FORMAT " can't move to layer %d",
|
|
|
|
GES_ARGS (clip), ges_layer_get_priority (layer));
|
|
|
|
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
current_layer = clip->priv->layer;
|
2011-06-06 19:55:47 +00:00
|
|
|
|
|
|
|
if (current_layer == NULL) {
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG ("Not moving %p, only adding it to %p", clip, layer);
|
2011-06-06 19:55:47 +00:00
|
|
|
|
2013-04-23 23:04:04 +00:00
|
|
|
return ges_layer_add_clip (layer, clip);
|
2011-06-06 19:55:47 +00:00
|
|
|
}
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "moving to layer %p, priority: %d", layer,
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_layer_get_priority (layer));
|
2012-01-20 20:03:58 +00:00
|
|
|
|
2013-03-16 22:05:04 +00:00
|
|
|
gst_object_ref (clip);
|
2013-04-23 23:04:04 +00:00
|
|
|
ret = ges_layer_remove_clip (current_layer, clip);
|
2011-06-06 19:55:47 +00:00
|
|
|
|
|
|
|
if (!ret) {
|
2019-03-01 22:32:19 +00:00
|
|
|
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
2013-03-16 22:05:04 +00:00
|
|
|
gst_object_unref (clip);
|
2011-06-06 19:55:47 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2013-04-23 23:04:04 +00:00
|
|
|
ret = ges_layer_add_clip (layer, clip);
|
2019-02-09 21:59:08 +00:00
|
|
|
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
2011-06-06 19:55:47 +00:00
|
|
|
|
2013-03-16 22:05:04 +00:00
|
|
|
gst_object_unref (clip);
|
2013-07-02 23:47:48 +00:00
|
|
|
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
|
2011-06-06 19:55:47 +00:00
|
|
|
|
2013-07-02 23:47:48 +00:00
|
|
|
|
|
|
|
return ret && (clip->priv->layer == layer);
|
2011-06-06 19:55:47 +00:00
|
|
|
}
|
|
|
|
|
2010-03-12 16:17:30 +00:00
|
|
|
/**
|
2013-01-26 15:31:33 +00:00
|
|
|
* ges_clip_find_track_element:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @track: (allow-none): The track to search in, or %NULL to search in
|
|
|
|
* all tracks
|
|
|
|
* @type: The type of track element to search for, or `G_TYPE_NONE` to
|
|
|
|
* match any type
|
2010-03-12 16:17:30 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Finds an element controlled by the clip. If @track is given,
|
|
|
|
* then only the track elements in @track are searched for. If @type is
|
|
|
|
* given, then this function searches for a track element of the given
|
|
|
|
* @type.
|
2010-03-12 16:17:30 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Note, if multiple track elements in the clip match the given criteria,
|
|
|
|
* this will return the element amongst them with the highest
|
|
|
|
* #GESTimelineElement:priority (numerically, the smallest). See
|
|
|
|
* ges_clip_find_track_elements() if you wish to find all such elements.
|
2010-03-12 16:17:30 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer full) (nullable): The element controlled by
|
|
|
|
* @clip, in @track, and of the given @type, or %NULL if no such element
|
|
|
|
* could be found.
|
2010-03-12 16:17:30 +00:00
|
|
|
*/
|
|
|
|
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_find_track_element (GESClip * clip, GESTrack * track, GType type)
|
2010-03-12 16:17:30 +00:00
|
|
|
{
|
2010-12-16 15:27:26 +00:00
|
|
|
GList *tmp;
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *otmp;
|
2010-03-12 16:17:30 +00:00
|
|
|
|
2013-10-29 10:59:22 +00:00
|
|
|
GESTrackElement *ret = NULL;
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2013-10-29 10:59:22 +00:00
|
|
|
g_return_val_if_fail (!(track == NULL && type == G_TYPE_NONE), NULL);
|
2011-05-06 17:40:22 +00:00
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = g_list_next (tmp)) {
|
2013-01-26 15:31:33 +00:00
|
|
|
otmp = (GESTrackElement *) tmp->data;
|
2010-12-16 14:00:46 +00:00
|
|
|
|
2013-10-29 10:59:22 +00:00
|
|
|
if ((type != G_TYPE_NONE) && !G_TYPE_CHECK_INSTANCE_TYPE (tmp->data, type))
|
|
|
|
continue;
|
2010-07-13 16:42:46 +00:00
|
|
|
|
2013-10-29 10:59:22 +00:00
|
|
|
if ((track == NULL) || (ges_track_element_get_track (otmp) == track)) {
|
2013-01-26 15:31:33 +00:00
|
|
|
ret = GES_TRACK_ELEMENT (tmp->data);
|
2013-03-16 22:05:04 +00:00
|
|
|
gst_object_ref (ret);
|
2010-12-16 15:27:26 +00:00
|
|
|
break;
|
2010-12-16 14:00:46 +00:00
|
|
|
}
|
2010-03-12 16:17:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2010-07-09 11:49:23 +00:00
|
|
|
|
2010-11-28 12:24:07 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_get_layer:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
2010-11-28 12:24:07 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Gets the #GESClip:layer of the clip.
|
2010-11-28 12:24:07 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer full) (nullable): The layer @clip is in, or %NULL if
|
|
|
|
* @clip is not in any layer.
|
2010-11-28 12:24:07 +00:00
|
|
|
*/
|
2013-04-23 23:04:04 +00:00
|
|
|
GESLayer *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_get_layer (GESClip * clip)
|
2010-11-28 12:24:07 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2010-11-28 12:24:07 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
if (clip->priv->layer != NULL)
|
2013-03-16 22:05:04 +00:00
|
|
|
gst_object_ref (G_OBJECT (clip->priv->layer));
|
2010-11-28 12:24:07 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
return clip->priv->layer;
|
2010-11-28 12:24:07 +00:00
|
|
|
}
|
|
|
|
|
2011-01-31 10:32:14 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_get_top_effects:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Gets the #GESBaseEffect-s that have been added to the clip. The
|
|
|
|
* returned list is ordered by their internal index in the clip. See
|
|
|
|
* ges_clip_get_top_effect_index().
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer full) (element-type GESTrackElement): A list of all
|
|
|
|
* #GESBaseEffect-s that have been added to @clip.
|
2011-05-07 14:59:06 +00:00
|
|
|
*/
|
2011-01-31 10:32:14 +00:00
|
|
|
GList *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_get_top_effects (GESClip * clip)
|
2011-01-31 10:32:14 +00:00
|
|
|
{
|
|
|
|
GList *tmp, *ret;
|
2020-03-02 12:56:03 +00:00
|
|
|
GESTimelineElement *child;
|
2011-01-31 10:32:14 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Getting the %i top effects", clip->priv->nb_effects);
|
2011-01-31 10:32:14 +00:00
|
|
|
ret = NULL;
|
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = tmp->next) {
|
|
|
|
child = tmp->data;
|
|
|
|
if (GES_IS_BASE_EFFECT (child)
|
2020-03-18 15:56:06 +00:00
|
|
|
&& !ges_track_element_get_owners (GES_TRACK_ELEMENT (child)))
|
2020-03-02 12:56:03 +00:00
|
|
|
ret = g_list_append (ret, gst_object_ref (child));
|
2011-01-31 10:32:14 +00:00
|
|
|
}
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* list is already sorted by index because the list of children is
|
|
|
|
* sorted by priority */
|
|
|
|
return ret;
|
2011-01-31 10:32:14 +00:00
|
|
|
}
|
2010-12-16 15:27:26 +00:00
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
static gboolean
|
|
|
|
_is_added_effect (GESClip * clip, GESBaseEffect * effect)
|
|
|
|
{
|
|
|
|
if (GES_TIMELINE_ELEMENT_PARENT (effect) != GES_TIMELINE_ELEMENT (clip)) {
|
|
|
|
GST_WARNING_OBJECT (clip, "The effect %" GES_FORMAT
|
|
|
|
" doe not belong to this clip", GES_ARGS (effect));
|
|
|
|
return FALSE;
|
|
|
|
}
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (GES_TRACK_ELEMENT (effect))) {
|
2020-03-02 12:56:03 +00:00
|
|
|
GST_WARNING_OBJECT (clip, "The effect %" GES_FORMAT " is not a top "
|
|
|
|
"effect of this clip (it is a core element of the clip)",
|
|
|
|
GES_ARGS (effect));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-01-31 10:41:37 +00:00
|
|
|
/**
|
2014-06-05 02:05:06 +00:00
|
|
|
* ges_clip_get_top_effect_index:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @effect: The effect we want to get the index of
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Gets the internal index of an effect in the clip. The index of effects
|
|
|
|
* in a clip will run from 0 to n-1, where n is the total number of
|
|
|
|
* effects. If two effects share the same #GESTrackElement:track, the
|
|
|
|
* effect with the numerically lower index will be applied to the source
|
|
|
|
* data **after** the other effect, i.e. output data will always flow from
|
|
|
|
* a higher index effect to a lower index effect.
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: The index of @effect in @clip, or -1 if something went wrong.
|
2011-05-07 14:59:06 +00:00
|
|
|
*/
|
2011-01-31 10:41:37 +00:00
|
|
|
gint
|
2014-06-05 02:05:06 +00:00
|
|
|
ges_clip_get_top_effect_index (GESClip * clip, GESBaseEffect * effect)
|
2011-01-31 10:41:37 +00:00
|
|
|
{
|
2013-03-01 01:27:50 +00:00
|
|
|
guint max_prio, min_prio;
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), -1);
|
2013-03-01 01:27:50 +00:00
|
|
|
g_return_val_if_fail (GES_IS_BASE_EFFECT (effect), -1);
|
2020-03-02 12:56:03 +00:00
|
|
|
if (!_is_added_effect (clip, effect))
|
|
|
|
return -1;
|
2013-03-01 01:27:50 +00:00
|
|
|
|
2013-03-30 11:30:47 +00:00
|
|
|
_get_priority_range (GES_CONTAINER (clip), &min_prio, &max_prio);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2016-08-11 12:54:23 +00:00
|
|
|
return GES_TIMELINE_ELEMENT_PRIORITY (effect) - min_prio;
|
2011-01-31 10:41:37 +00:00
|
|
|
}
|
|
|
|
|
2014-06-05 02:05:06 +00:00
|
|
|
/* TODO 2.0 remove as it is Deprecated */
|
|
|
|
gint
|
|
|
|
ges_clip_get_top_effect_position (GESClip * clip, GESBaseEffect * effect)
|
|
|
|
{
|
|
|
|
return ges_clip_get_top_effect_index (clip, effect);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* TODO 2.0 remove as it is Deprecated */
|
|
|
|
gboolean
|
|
|
|
ges_clip_set_top_effect_priority (GESClip * clip,
|
|
|
|
GESBaseEffect * effect, guint newpriority)
|
|
|
|
{
|
|
|
|
return ges_clip_set_top_effect_index (clip, effect, newpriority);
|
|
|
|
}
|
|
|
|
|
2011-02-04 10:44:19 +00:00
|
|
|
/**
|
2014-06-05 02:05:06 +00:00
|
|
|
* ges_clip_set_top_effect_index:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @effect: An effect within @clip to move
|
|
|
|
* @newindex: The index for @effect in @clip
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Set the index of an effect within the clip. See
|
|
|
|
* ges_clip_get_top_effect_index(). The new index must be an existing
|
|
|
|
* index of the clip. The effect is moved to the new index, and the other
|
|
|
|
* effects may be shifted in index accordingly to otherwise maintain the
|
|
|
|
* ordering.
|
2011-05-07 14:59:06 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: %TRUE if @effect was successfully moved to @newindex.
|
2011-05-07 14:59:06 +00:00
|
|
|
*/
|
2011-02-04 10:44:19 +00:00
|
|
|
gboolean
|
2014-06-05 02:05:06 +00:00
|
|
|
ges_clip_set_top_effect_index (GESClip * clip, GESBaseEffect * effect,
|
|
|
|
guint newindex)
|
2011-02-04 10:44:19 +00:00
|
|
|
{
|
2011-02-16 16:51:21 +00:00
|
|
|
gint inc;
|
2011-12-07 23:50:13 +00:00
|
|
|
GList *tmp;
|
2014-06-05 02:20:15 +00:00
|
|
|
guint current_prio, min_prio, max_prio;
|
2013-02-10 00:49:16 +00:00
|
|
|
GESTrackElement *track_element;
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
2020-03-02 12:56:03 +00:00
|
|
|
g_return_val_if_fail (GES_IS_BASE_EFFECT (effect), FALSE);
|
2011-12-07 23:50:13 +00:00
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
if (!_is_added_effect (clip, effect))
|
2019-04-17 21:53:14 +00:00
|
|
|
return FALSE;
|
|
|
|
|
2020-03-02 12:56:03 +00:00
|
|
|
track_element = GES_TRACK_ELEMENT (effect);
|
2013-02-10 00:49:16 +00:00
|
|
|
current_prio = _PRIORITY (track_element);
|
2011-02-04 10:44:19 +00:00
|
|
|
|
2014-06-05 02:20:15 +00:00
|
|
|
_get_priority_range (GES_CONTAINER (clip), &min_prio, &max_prio);
|
|
|
|
|
|
|
|
newindex = newindex + min_prio;
|
2011-02-04 10:44:19 +00:00
|
|
|
/* We don't change the priority */
|
2019-04-17 21:51:13 +00:00
|
|
|
if (current_prio == newindex)
|
|
|
|
return TRUE;
|
|
|
|
|
2014-06-05 02:20:15 +00:00
|
|
|
if (newindex > (clip->priv->nb_effects - 1 + min_prio)) {
|
2011-02-16 16:51:21 +00:00
|
|
|
GST_DEBUG ("You are trying to make %p not a top effect", effect);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-06-05 02:20:15 +00:00
|
|
|
if (current_prio > clip->priv->nb_effects + min_prio) {
|
|
|
|
GST_ERROR ("%p is not a top effect", effect);
|
2011-02-16 16:51:21 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Setting top effect %" GST_PTR_FORMAT "priority: %i",
|
|
|
|
effect, newindex);
|
|
|
|
|
|
|
|
if (current_prio < newindex)
|
2011-02-04 10:44:19 +00:00
|
|
|
inc = -1;
|
|
|
|
else
|
|
|
|
inc = +1;
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* prevent a re-sort of the list whilst we are traversing it! */
|
|
|
|
clip->priv->prevent_resort = TRUE;
|
2013-03-01 01:27:50 +00:00
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = tmp->next) {
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *tmpo = GES_TRACK_ELEMENT (tmp->data);
|
2013-01-15 13:52:17 +00:00
|
|
|
guint tck_priority = _PRIORITY (tmpo);
|
2011-02-04 10:44:19 +00:00
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
if (tmpo == track_element)
|
|
|
|
continue;
|
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
/* only need to change the priority for those between the new and old
|
|
|
|
* index */
|
|
|
|
if ((inc == +1 && tck_priority >= newindex && tck_priority < current_prio)
|
|
|
|
|| (inc == -1 && tck_priority <= newindex
|
|
|
|
&& tck_priority > current_prio)) {
|
2013-01-15 13:52:17 +00:00
|
|
|
_set_priority0 (GES_TIMELINE_ELEMENT (tmpo), tck_priority + inc);
|
2011-02-04 10:44:19 +00:00
|
|
|
}
|
|
|
|
}
|
2014-06-05 02:05:06 +00:00
|
|
|
_set_priority0 (GES_TIMELINE_ELEMENT (track_element), newindex);
|
2011-02-04 10:44:19 +00:00
|
|
|
|
2020-03-02 13:25:21 +00:00
|
|
|
clip->priv->prevent_resort = FALSE;
|
|
|
|
_ges_container_sort_children (GES_CONTAINER (clip));
|
|
|
|
/* height should have stayed the same */
|
|
|
|
|
2011-02-04 10:44:19 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2011-08-24 09:48:14 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_split:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: The #GESClip to split
|
|
|
|
* @position: The timeline position at which to perform the split
|
2011-08-24 09:48:14 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Splits a clip at the given timeline position into two clips. The clip
|
|
|
|
* must already have a #GESClip:layer.
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* The original clip's #GESTimelineElement:duration is reduced such that
|
|
|
|
* its end point matches the split position. Then a new clip is created in
|
|
|
|
* the same layer, whose #GESTimelineElement:start matches the split
|
|
|
|
* position and #GESTimelineElement:duration will be set such that its end
|
|
|
|
* point matches the old end point of the original clip. Thus, the two
|
|
|
|
* clips together will occupy the same positions in the timeline as the
|
|
|
|
* original clip did.
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* The children of the new clip will be new copies of the original clip's
|
|
|
|
* children, so it will share the same sources and use the same
|
|
|
|
* operations.
|
2011-08-24 09:48:14 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* The new clip will also have its #GESTimelineElement:in-point set so
|
|
|
|
* that any internal data will appear in the timeline at the same time.
|
|
|
|
* Thus, when the timeline is played, the playback of data should
|
|
|
|
* appear the same. This may be complicated by any additional
|
|
|
|
* #GESEffect-s that have been placed on the original clip that depend on
|
|
|
|
* the playback time or change the data consumption rate of sources. This
|
|
|
|
* method will attempt to translate these effects such that the playback
|
|
|
|
* appears the same. In such complex situations, you may get a better
|
|
|
|
* result if you place the clip in a separate sub #GESProject, which only
|
|
|
|
* contains this clip (and its effects), and in the original layer
|
|
|
|
* create two neighbouring #GESUriClip-s that reference this sub-project,
|
|
|
|
* but at a different #GESTimelineElement:in-point.
|
|
|
|
*
|
|
|
|
* Returns: (transfer none) (nullable): The newly created clip resulting
|
|
|
|
* from the splitting @clip, or %NULL if @clip can't be split.
|
2011-08-24 09:48:14 +00:00
|
|
|
*/
|
2013-01-20 15:42:29 +00:00
|
|
|
GESClip *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_split (GESClip * clip, guint64 position)
|
2011-08-24 09:48:14 +00:00
|
|
|
{
|
2012-04-20 23:19:49 +00:00
|
|
|
GList *tmp;
|
2013-01-20 15:42:29 +00:00
|
|
|
GESClip *new_object;
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
GstClockTime start, inpoint, duration, old_duration, new_duration;
|
|
|
|
gdouble media_duration_factor;
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
2013-03-18 13:02:10 +00:00
|
|
|
g_return_val_if_fail (clip->priv->layer, NULL);
|
2012-04-20 23:22:56 +00:00
|
|
|
g_return_val_if_fail (GST_CLOCK_TIME_IS_VALID (position), NULL);
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
duration = _DURATION (clip);
|
|
|
|
start = _START (clip);
|
|
|
|
inpoint = _INPOINT (clip);
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2012-04-20 23:19:49 +00:00
|
|
|
if (position >= start + duration || position <= start) {
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_WARNING_OBJECT (clip, "Can not split %" GST_TIME_FORMAT
|
2012-04-20 23:19:49 +00:00
|
|
|
" out of boundaries", GST_TIME_ARGS (position));
|
|
|
|
return NULL;
|
|
|
|
}
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2020-03-04 20:16:18 +00:00
|
|
|
old_duration = position - start;
|
|
|
|
if (!timeline_tree_can_move_element (timeline_get_tree
|
|
|
|
(GES_TIMELINE_ELEMENT_TIMELINE (clip)), GES_TIMELINE_ELEMENT (clip),
|
|
|
|
GES_TIMELINE_ELEMENT_LAYER_PRIORITY (clip),
|
|
|
|
start, old_duration, NULL)) {
|
|
|
|
GST_WARNING_OBJECT (clip,
|
|
|
|
"Can not split %" GES_FORMAT " at %" GST_TIME_FORMAT
|
|
|
|
" as timeline would be in an illegal" " state.", GES_ARGS (clip),
|
|
|
|
GST_TIME_ARGS (position));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_duration = duration + start - position;
|
|
|
|
if (!timeline_tree_can_move_element (timeline_get_tree
|
|
|
|
(GES_TIMELINE_ELEMENT_TIMELINE (clip)), GES_TIMELINE_ELEMENT (clip),
|
|
|
|
GES_TIMELINE_ELEMENT_LAYER_PRIORITY (clip), position, new_duration,
|
|
|
|
NULL)) {
|
|
|
|
GST_WARNING_OBJECT (clip,
|
|
|
|
"Can not split %" GES_FORMAT " at %" GST_TIME_FORMAT
|
|
|
|
" as timeline would end up in an illegal" " state.", GES_ARGS (clip),
|
|
|
|
GST_TIME_ARGS (position));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
GST_DEBUG_OBJECT (clip, "Spliting at %" GST_TIME_FORMAT,
|
2012-04-20 23:19:49 +00:00
|
|
|
GST_TIME_ARGS (position));
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2013-01-20 15:42:29 +00:00
|
|
|
/* Create the new Clip */
|
2013-03-01 01:27:50 +00:00
|
|
|
new_object = GES_CLIP (ges_timeline_element_copy (GES_TIMELINE_ELEMENT (clip),
|
|
|
|
FALSE));
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
GST_DEBUG_OBJECT (new_object, "New 'splitted' clip");
|
2013-01-20 15:42:29 +00:00
|
|
|
/* Set new timing properties on the Clip */
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
media_duration_factor =
|
|
|
|
ges_timeline_element_get_media_duration_factor (GES_TIMELINE_ELEMENT
|
|
|
|
(clip));
|
2013-01-15 13:52:17 +00:00
|
|
|
_set_start0 (GES_TIMELINE_ELEMENT (new_object), position);
|
|
|
|
_set_inpoint0 (GES_TIMELINE_ELEMENT (new_object),
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
inpoint + old_duration * media_duration_factor);
|
|
|
|
_set_duration0 (GES_TIMELINE_ELEMENT (new_object), new_duration);
|
2012-04-20 23:19:49 +00:00
|
|
|
|
2019-03-08 15:28:31 +00:00
|
|
|
_DURATION (clip) = old_duration;
|
|
|
|
g_object_notify (G_OBJECT (clip), "duration");
|
|
|
|
|
2013-03-18 13:02:10 +00:00
|
|
|
/* We do not want the timeline to create again TrackElement-s */
|
|
|
|
ges_clip_set_moving_from_layer (new_object, TRUE);
|
2013-04-23 23:04:04 +00:00
|
|
|
ges_layer_add_clip (clip->priv->layer, new_object);
|
2013-03-18 13:02:10 +00:00
|
|
|
ges_clip_set_moving_from_layer (new_object, FALSE);
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2013-03-01 01:27:50 +00:00
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = tmp->next) {
|
2013-01-26 15:31:33 +00:00
|
|
|
GESTrackElement *new_trackelement, *trackelement =
|
|
|
|
GES_TRACK_ELEMENT (tmp->data);
|
2012-04-20 23:19:49 +00:00
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
new_trackelement =
|
|
|
|
GES_TRACK_ELEMENT (ges_timeline_element_copy (GES_TIMELINE_ELEMENT
|
|
|
|
(trackelement), FALSE));
|
2013-01-26 15:31:33 +00:00
|
|
|
if (new_trackelement == NULL) {
|
|
|
|
GST_WARNING_OBJECT (trackelement, "Could not create a copy");
|
2012-04-20 23:19:49 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-03-18 15:56:06 +00:00
|
|
|
if (ges_track_element_get_owners (trackelement))
|
|
|
|
ges_track_element_add_owner (new_trackelement, new_object);
|
|
|
|
|
2020-03-10 11:38:58 +00:00
|
|
|
/* FIXME: in-point for non-core track elements should be shifted by
|
|
|
|
* the split (adding them to the new clip will not set their in-point)
|
|
|
|
* Handle this once generic time effects are supported in splitting */
|
2013-03-01 01:27:50 +00:00
|
|
|
ges_container_add (GES_CONTAINER (new_object),
|
|
|
|
GES_TIMELINE_ELEMENT (new_trackelement));
|
2013-07-24 18:26:18 +00:00
|
|
|
ges_track_element_copy_properties (GES_TIMELINE_ELEMENT (trackelement),
|
|
|
|
GES_TIMELINE_ELEMENT (new_trackelement));
|
2013-08-13 16:05:55 +00:00
|
|
|
|
2020-01-09 12:11:35 +00:00
|
|
|
/* FIXME: is position - start + inpoint always the correct splitting
|
|
|
|
* point for control bindings? What coordinate system are control
|
|
|
|
* bindings given in? */
|
|
|
|
/* NOTE: control bindings that are not registered in GES are not
|
|
|
|
* handled */
|
2015-06-29 16:04:32 +00:00
|
|
|
ges_track_element_copy_bindings (trackelement, new_trackelement,
|
2013-10-29 23:27:36 +00:00
|
|
|
position - start + inpoint);
|
2012-04-20 23:19:49 +00:00
|
|
|
}
|
2011-08-24 09:48:14 +00:00
|
|
|
|
2020-01-09 12:11:35 +00:00
|
|
|
/* FIXME: The below leads to a *second* notify signal for duration */
|
2019-03-01 22:32:19 +00:00
|
|
|
ELEMENT_SET_FLAG (clip, GES_TIMELINE_ELEMENT_SET_SIMPLE);
|
2019-03-08 15:28:31 +00:00
|
|
|
_DURATION (clip) = duration;
|
Handle changing playback rate
Before this patch, NLE and GES did not support NleOperations (respectively
GESEffects) that changed the speed/tempo/rate at which the source plays. For
example, the 'pitch' element can make audio play faster or slower. In GES 1.5.90
and before, an NleOperation containing the pitch element to change the rate (or
tempo) would cause a pipeline state change to PAUSED after that stack; that has
been fixed in 1.5.91 (see #755012 [0]). But even then, in 1.5.91 and later,
NleComposition would send segment events to its NleSources assuming that one
source second is equal to one pipeline second. The resulting early EOS event
(in the case of a source rate higher than 1.0) would cause it to switch stacks
too early, causing confusion in the timeline and spectacularly messed up
output.
This patch fixes that by searching for rate-changing elements in
GESTrackElements such as GESEffects. If such rate-changing elements are found,
their final effect on the playing rate is stored in the corresponding NleObject
as the 'media duration factor', named like this because the 'media duration',
or source duration, of an NleObject can be computed by multiplying the duration
with the media duration factor of that object and its parents (this is called
the 'recursive media duration factor'). For example, a 4-second NleSource with
an NleOperation with a media duration factor of 2.0 will have an 8-second media
duration, which means that for playing 4 seconds in the pipeline, the seek
event sent to it must span 8 seconds of media. (So, the 'duration' of an
NleObject or GES object always refers to its duration in the timeline, not the
media duration.)
To summarize:
* Rate-changing elements are registered in the GESEffectClass (pitch::tempo and
pitch::rate are registered by default);
* GESTimelineElement is responsible for detecting rate-changing elements and
computing the media_duration_factor;
* GESTrackElement is responsible for storing the media_duration_factor in
NleObject;
* NleComposition is responsible for the recursive_media_duration_factor;
* The latter property finally fixes media time computations in NleObject.
NLE and GES tests are included.
[0] https://bugzilla.gnome.org/show_bug.cgi?id=755012
Differential Revision: https://phabricator.freedesktop.org/D276
2015-12-20 13:03:57 +00:00
|
|
|
_set_duration0 (GES_TIMELINE_ELEMENT (clip), old_duration);
|
2019-03-01 22:32:19 +00:00
|
|
|
ELEMENT_UNSET_FLAG (clip, GES_TIMELINE_ELEMENT_SET_SIMPLE);
|
2018-03-18 14:03:00 +00:00
|
|
|
|
2011-08-24 09:48:14 +00:00
|
|
|
return new_object;
|
|
|
|
}
|
|
|
|
|
2011-12-07 23:17:55 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_set_supported_formats:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @supportedformats: The #GESTrackType-s supported by @clip
|
2011-12-07 23:17:55 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Sets the #GESClip:supported-formats of the clip. This should normally
|
|
|
|
* only be called by subclasses, which should be responsible for updating
|
|
|
|
* its value, rather than the user.
|
2011-12-07 23:17:55 +00:00
|
|
|
*/
|
|
|
|
void
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_set_supported_formats (GESClip * clip, GESTrackType supportedformats)
|
2011-12-07 23:17:55 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_if_fail (GES_IS_CLIP (clip));
|
2011-12-07 23:17:55 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
clip->priv->supportedformats = supportedformats;
|
2011-12-07 23:17:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_get_supported_formats:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
2011-12-07 23:17:55 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Gets the #GESClip:supported-formats of the clip.
|
2011-12-07 23:17:55 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: The #GESTrackType-s supported by @clip.
|
2011-12-07 23:17:55 +00:00
|
|
|
*/
|
|
|
|
GESTrackType
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_get_supported_formats (GESClip * clip)
|
2011-12-07 23:17:55 +00:00
|
|
|
{
|
2013-02-15 02:34:48 +00:00
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), GES_TRACK_TYPE_UNKNOWN);
|
2011-12-07 23:17:55 +00:00
|
|
|
|
2013-02-15 02:34:48 +00:00
|
|
|
return clip->priv->supportedformats;
|
2011-12-07 23:17:55 +00:00
|
|
|
}
|
|
|
|
|
2012-04-24 00:52:45 +00:00
|
|
|
gboolean
|
2013-01-15 13:52:17 +00:00
|
|
|
_ripple (GESTimelineElement * element, GstClockTime start)
|
2012-04-24 00:52:45 +00:00
|
|
|
{
|
2019-03-01 22:32:19 +00:00
|
|
|
return ges_container_edit (GES_CONTAINER (element), NULL,
|
|
|
|
ges_timeline_element_get_layer_priority (element),
|
|
|
|
GES_EDIT_MODE_RIPPLE, GES_EDGE_NONE, start);
|
2012-04-24 00:52:45 +00:00
|
|
|
}
|
|
|
|
|
2013-01-15 13:52:17 +00:00
|
|
|
static gboolean
|
|
|
|
_ripple_end (GESTimelineElement * element, GstClockTime end)
|
2012-04-24 00:52:45 +00:00
|
|
|
{
|
2019-03-01 22:32:19 +00:00
|
|
|
return ges_container_edit (GES_CONTAINER (element), NULL,
|
|
|
|
ges_timeline_element_get_layer_priority (element),
|
|
|
|
GES_EDIT_MODE_RIPPLE, GES_EDGE_END, end);
|
2012-04-24 00:52:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
2013-01-15 13:52:17 +00:00
|
|
|
_roll_start (GESTimelineElement * element, GstClockTime start)
|
2012-04-24 00:52:45 +00:00
|
|
|
{
|
2019-03-01 22:32:19 +00:00
|
|
|
return ges_container_edit (GES_CONTAINER (element), NULL,
|
|
|
|
ges_timeline_element_get_layer_priority (element),
|
|
|
|
GES_EDIT_MODE_ROLL, GES_EDGE_START, start);
|
2012-04-24 00:52:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
2013-01-15 13:52:17 +00:00
|
|
|
_roll_end (GESTimelineElement * element, GstClockTime end)
|
2012-04-24 00:52:45 +00:00
|
|
|
{
|
2019-03-01 22:32:19 +00:00
|
|
|
return ges_container_edit (GES_CONTAINER (element), NULL,
|
|
|
|
ges_timeline_element_get_layer_priority (element),
|
|
|
|
GES_EDIT_MODE_ROLL, GES_EDGE_END, end);
|
2012-04-24 00:52:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
2013-01-15 13:52:17 +00:00
|
|
|
_trim (GESTimelineElement * element, GstClockTime start)
|
2012-04-24 00:52:45 +00:00
|
|
|
{
|
2019-03-01 22:32:19 +00:00
|
|
|
return ges_container_edit (GES_CONTAINER (element), NULL, -1,
|
|
|
|
GES_EDIT_MODE_TRIM, GES_EDGE_START, start);
|
2012-04-24 00:52:45 +00:00
|
|
|
}
|
|
|
|
|
2012-11-27 15:18:27 +00:00
|
|
|
/**
|
2013-01-20 15:42:29 +00:00
|
|
|
* ges_clip_add_asset:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @asset: An asset with #GES_TYPE_TRACK_ELEMENT as its
|
|
|
|
* #GESAsset:extractable-type
|
2012-11-27 15:18:27 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Extracts a #GESTrackElement from an asset and adds it to the clip.
|
|
|
|
* This can be used to add effects that derive from the asset to the
|
|
|
|
* clip, but this method is not intended to be used to create the core
|
|
|
|
* elements of the clip.
|
2012-11-27 15:18:27 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer none)(allow-none): The newly created element, or
|
|
|
|
* %NULL if an error occurred.
|
2012-11-27 15:18:27 +00:00
|
|
|
*/
|
2020-01-09 12:11:35 +00:00
|
|
|
/* FIXME: this is not used elsewhere in the GES library */
|
2013-09-28 16:09:49 +00:00
|
|
|
GESTrackElement *
|
2013-02-15 02:34:48 +00:00
|
|
|
ges_clip_add_asset (GESClip * clip, GESAsset * asset)
|
2012-11-27 15:18:27 +00:00
|
|
|
{
|
2013-09-28 16:09:49 +00:00
|
|
|
GESTrackElement *element;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
|
|
|
g_return_val_if_fail (GES_IS_ASSET (asset), NULL);
|
2012-11-27 15:18:27 +00:00
|
|
|
g_return_val_if_fail (g_type_is_a (ges_asset_get_extractable_type
|
2013-09-28 16:09:49 +00:00
|
|
|
(asset), GES_TYPE_TRACK_ELEMENT), NULL);
|
|
|
|
|
|
|
|
element = GES_TRACK_ELEMENT (ges_asset_extract (asset, NULL));
|
|
|
|
|
|
|
|
if (!ges_container_add (GES_CONTAINER (clip), GES_TIMELINE_ELEMENT (element)))
|
|
|
|
return NULL;
|
2012-11-27 15:18:27 +00:00
|
|
|
|
2013-09-28 16:09:49 +00:00
|
|
|
return element;
|
2014-05-19 10:21:52 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ges_clip_find_track_elements:
|
2020-01-09 12:11:35 +00:00
|
|
|
* @clip: A #GESClip
|
|
|
|
* @track: (allow-none): The track to search in, or %NULL to search in
|
|
|
|
* all tracks
|
|
|
|
* @track_type: The track-type of the track element to search for, or
|
|
|
|
* #GES_TRACK_TYPE_UNKNOWN to match any track type
|
2020-02-18 18:02:08 +00:00
|
|
|
* @type: The type of track element to search for, or %G_TYPE_NONE to
|
|
|
|
* match any type
|
2020-01-09 12:11:35 +00:00
|
|
|
*
|
|
|
|
* Finds the #GESTrackElement-s controlled by the clip that match the
|
2020-02-18 18:02:08 +00:00
|
|
|
* given criteria. If @track is given as %NULL and @track_type is given as
|
|
|
|
* #GES_TRACK_TYPE_UNKNOWN, then the search will match all elements in any
|
|
|
|
* track, including those with no track, and of any
|
|
|
|
* #GESTrackElement:track-type. Otherwise, if @track is not %NULL, but
|
|
|
|
* @track_type is #GES_TRACK_TYPE_UNKNOWN, then only the track elements in
|
|
|
|
* @track are searched for. Otherwise, if @track_type is not
|
|
|
|
* #GES_TRACK_TYPE_UNKNOWN, but @track is %NULL, then only the track
|
|
|
|
* elements whose #GESTrackElement:track-type matches @track_type are
|
|
|
|
* searched for. Otherwise, when both are given, the track elements that
|
|
|
|
* match **either** criteria are searched for. Therefore, if you wish to
|
|
|
|
* only find elements in a specific track, you should give the track as
|
|
|
|
* @track, but you should not give the track's #GESTrack:track-type as
|
|
|
|
* @track_type because this would also select elements from other tracks
|
|
|
|
* of the same type.
|
2014-05-19 10:21:52 +00:00
|
|
|
*
|
2020-02-18 18:02:08 +00:00
|
|
|
* You may also give @type to _further_ restrict the search to track
|
2020-01-09 12:11:35 +00:00
|
|
|
* elements of the given @type.
|
2014-05-19 10:21:52 +00:00
|
|
|
*
|
2020-01-09 12:11:35 +00:00
|
|
|
* Returns: (transfer full) (element-type GESTrackElement): A list of all
|
|
|
|
* the #GESTrackElement-s controlled by @clip, in @track or of the given
|
|
|
|
* @track_type, and of the given @type.
|
2014-05-19 10:21:52 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
GList *
|
|
|
|
ges_clip_find_track_elements (GESClip * clip, GESTrack * track,
|
|
|
|
GESTrackType track_type, GType type)
|
|
|
|
{
|
|
|
|
GList *tmp;
|
|
|
|
GESTrackElement *otmp;
|
|
|
|
|
|
|
|
GList *ret = NULL;
|
|
|
|
|
|
|
|
g_return_val_if_fail (GES_IS_CLIP (clip), NULL);
|
|
|
|
g_return_val_if_fail (!(track == NULL && type == G_TYPE_NONE &&
|
|
|
|
track_type == GES_TRACK_TYPE_UNKNOWN), NULL);
|
|
|
|
|
|
|
|
for (tmp = GES_CONTAINER_CHILDREN (clip); tmp; tmp = g_list_next (tmp)) {
|
|
|
|
otmp = (GESTrackElement *) tmp->data;
|
|
|
|
|
|
|
|
if ((type != G_TYPE_NONE) && !G_TYPE_CHECK_INSTANCE_TYPE (tmp->data, type))
|
|
|
|
continue;
|
|
|
|
|
2020-02-18 18:02:08 +00:00
|
|
|
/* TODO 2.0: an AND condition, using a condition like the above type
|
|
|
|
* check would have made more sense here. Especially when both
|
|
|
|
* track != NULL and track_type != GES_TRACK_TYPE_UNKNOWN are given */
|
|
|
|
if ((track == NULL && track_type == GES_TRACK_TYPE_UNKNOWN) ||
|
|
|
|
(track != NULL && ges_track_element_get_track (otmp) == track) ||
|
2014-05-19 10:21:52 +00:00
|
|
|
(track_type != GES_TRACK_TYPE_UNKNOWN
|
2020-02-18 18:02:08 +00:00
|
|
|
&& ges_track_element_get_track_type (otmp) == track_type))
|
|
|
|
ret = g_list_append (ret, gst_object_ref (otmp));
|
2014-05-19 10:21:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2010-07-09 11:49:23 +00:00
|
|
|
}
|