mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-25 00:28:21 +00:00
clip: secure adding clip to layer
Add more checks when adding a clip to a layer, or moving a clip to a new layer. Also, mark the "layer" property as explicit-notify.
This commit is contained in:
parent
46e6a074bb
commit
e57c345f72
2 changed files with 75 additions and 29 deletions
|
@ -106,6 +106,7 @@ struct _GESClipPrivate
|
||||||
|
|
||||||
GList *copied_track_elements;
|
GList *copied_track_elements;
|
||||||
GESLayer *copied_layer;
|
GESLayer *copied_layer;
|
||||||
|
GESTimeline *copied_timeline;
|
||||||
gboolean prevent_priority_offset_update;
|
gboolean prevent_priority_offset_update;
|
||||||
gboolean prevent_resort;
|
gboolean prevent_resort;
|
||||||
|
|
||||||
|
@ -1225,6 +1226,7 @@ _deep_copy (GESTimelineElement * element, GESTimelineElement * copy)
|
||||||
}
|
}
|
||||||
|
|
||||||
ccopy->priv->copied_layer = g_object_ref (self->priv->layer);
|
ccopy->priv->copied_layer = g_object_ref (self->priv->layer);
|
||||||
|
ccopy->priv->copied_timeline = self->priv->layer->timeline;
|
||||||
}
|
}
|
||||||
|
|
||||||
static GESTimelineElement *
|
static GESTimelineElement *
|
||||||
|
@ -1233,6 +1235,7 @@ _paste (GESTimelineElement * element, GESTimelineElement * ref,
|
||||||
{
|
{
|
||||||
GList *tmp;
|
GList *tmp;
|
||||||
GESClip *self = GES_CLIP (element);
|
GESClip *self = GES_CLIP (element);
|
||||||
|
GESLayer *layer = self->priv->copied_layer;
|
||||||
GESClip *nclip = GES_CLIP (ges_timeline_element_copy (element, FALSE));
|
GESClip *nclip = GES_CLIP (ges_timeline_element_copy (element, FALSE));
|
||||||
|
|
||||||
ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (nclip), paste_position);
|
ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (nclip), paste_position);
|
||||||
|
@ -1241,8 +1244,18 @@ _paste (GESTimelineElement * element, GESTimelineElement * ref,
|
||||||
for (tmp = self->priv->copied_track_elements; tmp; tmp = tmp->next)
|
for (tmp = self->priv->copied_track_elements; tmp; tmp = tmp->next)
|
||||||
ges_clip_copy_track_element_into (nclip, tmp->data, GST_CLOCK_TIME_NONE);
|
ges_clip_copy_track_element_into (nclip, tmp->data, GST_CLOCK_TIME_NONE);
|
||||||
|
|
||||||
if (self->priv->copied_layer) {
|
if (layer) {
|
||||||
if (!ges_layer_add_clip (self->priv->copied_layer, nclip)) {
|
if (layer->timeline != self->priv->copied_timeline) {
|
||||||
|
GST_WARNING_OBJECT (self, "Cannot be pasted into the layer %"
|
||||||
|
GST_PTR_FORMAT " because its timeline has changed", layer);
|
||||||
|
gst_object_ref_sink (nclip);
|
||||||
|
gst_object_unref (nclip);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* adding the clip to the layer will add it to the tracks, but not
|
||||||
|
* necessarily the same ones depending on select-tracks-for-object */
|
||||||
|
if (!ges_layer_add_clip (layer, nclip)) {
|
||||||
GST_INFO ("%" GES_FORMAT " could not be pasted to %" GST_TIME_FORMAT,
|
GST_INFO ("%" GES_FORMAT " could not be pasted to %" GST_TIME_FORMAT,
|
||||||
GES_ARGS (element), GST_TIME_ARGS (paste_position));
|
GES_ARGS (element), GST_TIME_ARGS (paste_position));
|
||||||
|
|
||||||
|
@ -1366,7 +1379,7 @@ ges_clip_class_init (GESClipClass * klass)
|
||||||
*/
|
*/
|
||||||
properties[PROP_LAYER] = g_param_spec_object ("layer", "Layer",
|
properties[PROP_LAYER] = g_param_spec_object ("layer", "Layer",
|
||||||
"The GESLayer where this clip is being used.",
|
"The GESLayer where this clip is being used.",
|
||||||
GES_TYPE_LAYER, G_PARAM_READABLE);
|
GES_TYPE_LAYER, G_PARAM_READABLE | G_PARAM_EXPLICIT_NOTIFY);
|
||||||
g_object_class_install_property (object_class, PROP_LAYER,
|
g_object_class_install_property (object_class, PROP_LAYER,
|
||||||
properties[PROP_LAYER]);
|
properties[PROP_LAYER]);
|
||||||
|
|
||||||
|
@ -1585,13 +1598,42 @@ ges_clip_is_moving_from_layer (GESClip * clip)
|
||||||
gboolean
|
gboolean
|
||||||
ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
|
ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
|
||||||
{
|
{
|
||||||
gboolean ret;
|
gboolean ret = FALSE;
|
||||||
GESLayer *current_layer;
|
GESLayer *current_layer;
|
||||||
|
GESTimeline *current_timeline = GES_TIMELINE_ELEMENT_TIMELINE (clip);
|
||||||
|
|
||||||
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
||||||
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
|
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
|
||||||
|
|
||||||
|
current_layer = clip->priv->layer;
|
||||||
|
|
||||||
|
if (current_layer == layer) {
|
||||||
|
GST_INFO_OBJECT (clip, "Already in the layer %" GST_PTR_FORMAT, layer);
|
||||||
|
return TRUE;
|
||||||
|
}
|
||||||
|
|
||||||
|
gst_object_ref (clip);
|
||||||
|
|
||||||
|
if (current_layer == NULL) {
|
||||||
|
GST_DEBUG ("Not moving %p, only adding it to %p", clip, layer);
|
||||||
|
|
||||||
|
ret = ges_layer_add_clip (layer, clip);
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING);
|
ELEMENT_SET_FLAG (clip, GES_CLIP_IS_MOVING);
|
||||||
|
|
||||||
|
if (current_timeline != layer->timeline) {
|
||||||
|
/* make sure we can perform the can_move_element_check in the timeline
|
||||||
|
* of the layer */
|
||||||
|
GST_WARNING_OBJECT (layer, "Cannot move clip %" GES_FORMAT " into "
|
||||||
|
"the layer because its timeline %" GST_PTR_FORMAT " does not "
|
||||||
|
"match the timeline of the layer %" GST_PTR_FORMAT,
|
||||||
|
GES_ARGS (clip), current_timeline, layer->timeline);
|
||||||
|
ret = FALSE;
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
if (layer->timeline
|
if (layer->timeline
|
||||||
&& !timeline_tree_can_move_element (timeline_get_tree (layer->timeline),
|
&& !timeline_tree_can_move_element (timeline_get_tree (layer->timeline),
|
||||||
GES_TIMELINE_ELEMENT (clip),
|
GES_TIMELINE_ELEMENT (clip),
|
||||||
|
@ -1600,36 +1642,30 @@ ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
|
||||||
GES_TIMELINE_ELEMENT_DURATION (clip), NULL)) {
|
GES_TIMELINE_ELEMENT_DURATION (clip), NULL)) {
|
||||||
GST_INFO_OBJECT (layer, "Clip %" GES_FORMAT " can't move to layer %d",
|
GST_INFO_OBJECT (layer, "Clip %" GES_FORMAT " can't move to layer %d",
|
||||||
GES_ARGS (clip), ges_layer_get_priority (layer));
|
GES_ARGS (clip), ges_layer_get_priority (layer));
|
||||||
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
goto done;
|
||||||
return FALSE;
|
|
||||||
}
|
|
||||||
|
|
||||||
current_layer = clip->priv->layer;
|
|
||||||
|
|
||||||
if (current_layer == NULL) {
|
|
||||||
GST_DEBUG ("Not moving %p, only adding it to %p", clip, layer);
|
|
||||||
|
|
||||||
return ges_layer_add_clip (layer, clip);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (clip, "moving to layer %p, priority: %d", layer,
|
GST_DEBUG_OBJECT (clip, "moving to layer %p, priority: %d", layer,
|
||||||
ges_layer_get_priority (layer));
|
ges_layer_get_priority (layer));
|
||||||
|
|
||||||
gst_object_ref (clip);
|
|
||||||
ret = ges_layer_remove_clip (current_layer, clip);
|
ret = ges_layer_remove_clip (current_layer, clip);
|
||||||
|
|
||||||
if (!ret) {
|
if (!ret) {
|
||||||
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
goto done;
|
||||||
gst_object_unref (clip);
|
|
||||||
return FALSE;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = ges_layer_add_clip (layer, clip);
|
ret = ges_layer_add_clip (layer, clip);
|
||||||
|
|
||||||
|
if (ret) {
|
||||||
|
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
|
||||||
|
} else {
|
||||||
|
/* try and move back into the original layer */
|
||||||
|
ges_layer_add_clip (current_layer, clip);
|
||||||
|
}
|
||||||
|
|
||||||
|
done:
|
||||||
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
ELEMENT_UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
|
||||||
|
|
||||||
gst_object_unref (clip);
|
gst_object_unref (clip);
|
||||||
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
|
|
||||||
|
|
||||||
|
|
||||||
return ret && (clip->priv->layer == layer);
|
return ret && (clip->priv->layer == layer);
|
||||||
}
|
}
|
||||||
|
|
|
@ -690,29 +690,41 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
|
||||||
GESAsset *asset;
|
GESAsset *asset;
|
||||||
GESLayerPrivate *priv;
|
GESLayerPrivate *priv;
|
||||||
GESLayer *current_layer;
|
GESLayer *current_layer;
|
||||||
|
GESTimeline *timeline = GES_TIMELINE_ELEMENT_TIMELINE (clip);
|
||||||
|
|
||||||
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
|
g_return_val_if_fail (GES_IS_LAYER (layer), FALSE);
|
||||||
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (layer, "adding clip:%p", clip);
|
GST_DEBUG_OBJECT (layer, "adding clip:%p", clip);
|
||||||
|
gst_object_ref_sink (clip);
|
||||||
|
|
||||||
priv = layer->priv;
|
priv = layer->priv;
|
||||||
current_layer = ges_clip_get_layer (clip);
|
current_layer = ges_clip_get_layer (clip);
|
||||||
if (G_UNLIKELY (current_layer)) {
|
if (G_UNLIKELY (current_layer)) {
|
||||||
GST_WARNING ("Clip %p already belongs to another layer", clip);
|
GST_WARNING_OBJECT (layer, "Clip %" GES_FORMAT " already belongs to "
|
||||||
gst_object_ref_sink (clip);
|
"another layer", GES_ARGS (clip));
|
||||||
gst_object_unref (clip);
|
gst_object_unref (clip);
|
||||||
gst_object_unref (current_layer);
|
gst_object_unref (current_layer);
|
||||||
|
|
||||||
return FALSE;
|
return FALSE;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (timeline && timeline != layer->timeline) {
|
||||||
|
/* if a clip is not in any layer, its timeline should not be set */
|
||||||
|
GST_ERROR_OBJECT (layer, "Clip %" GES_FORMAT " timeline %"
|
||||||
|
GST_PTR_FORMAT " does not match that of the layer %"
|
||||||
|
GST_PTR_FORMAT, GES_ARGS (clip), timeline, layer->timeline);
|
||||||
|
gst_object_unref (clip);
|
||||||
|
return FALSE;
|
||||||
|
}
|
||||||
|
|
||||||
|
timeline = layer->timeline;
|
||||||
|
|
||||||
asset = ges_extractable_get_asset (GES_EXTRACTABLE (clip));
|
asset = ges_extractable_get_asset (GES_EXTRACTABLE (clip));
|
||||||
if (asset == NULL) {
|
if (asset == NULL) {
|
||||||
gchar *id;
|
gchar *id;
|
||||||
NewAssetUData *mudata = g_slice_new (NewAssetUData);
|
NewAssetUData *mudata = g_slice_new (NewAssetUData);
|
||||||
|
|
||||||
mudata->clip = gst_object_ref_sink (clip);
|
mudata->clip = clip;
|
||||||
mudata->layer = layer;
|
mudata->layer = layer;
|
||||||
|
|
||||||
GST_DEBUG_OBJECT (layer, "%" GST_PTR_FORMAT " as no reference to any "
|
GST_DEBUG_OBJECT (layer, "%" GST_PTR_FORMAT " as no reference to any "
|
||||||
|
@ -741,8 +753,6 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
|
||||||
|
|
||||||
g_slice_free (NewAssetUData, mudata);
|
g_slice_free (NewAssetUData, mudata);
|
||||||
gst_clear_object (&asset);
|
gst_clear_object (&asset);
|
||||||
} else {
|
|
||||||
gst_object_ref_sink (clip);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Take a reference to the clip and store it stored by start/priority */
|
/* Take a reference to the clip and store it stored by start/priority */
|
||||||
|
@ -775,9 +785,9 @@ ges_layer_add_clip (GESLayer * layer, GESClip * clip)
|
||||||
* invoked by ges_timeline_add_clip */
|
* invoked by ges_timeline_add_clip */
|
||||||
g_signal_emit (layer, ges_layer_signals[OBJECT_ADDED], 0, clip);
|
g_signal_emit (layer, ges_layer_signals[OBJECT_ADDED], 0, clip);
|
||||||
|
|
||||||
if (layer->timeline && !ges_timeline_add_clip (layer->timeline, clip)) {
|
if (timeline && !ges_timeline_add_clip (timeline, clip)) {
|
||||||
GST_WARNING_OBJECT (layer, "Could not add the clip %" GES_FORMAT
|
GST_WARNING_OBJECT (layer, "Could not add the clip %" GES_FORMAT
|
||||||
" to the timeline %" GST_PTR_FORMAT, GES_ARGS (clip), layer->timeline);
|
" to the timeline %" GST_PTR_FORMAT, GES_ARGS (clip), timeline);
|
||||||
/* FIXME: change emit signal to FALSE once we are able to delay the
|
/* FIXME: change emit signal to FALSE once we are able to delay the
|
||||||
* "clip-added" signal until after ges_timeline_add_clip */
|
* "clip-added" signal until after ges_timeline_add_clip */
|
||||||
ges_layer_remove_clip_internal (layer, clip, TRUE);
|
ges_layer_remove_clip_internal (layer, clip, TRUE);
|
||||||
|
|
Loading…
Reference in a new issue