clip: Make sure to never snap when splitting clips

It makes no sense to snap in that context.

https://gitlab.gnome.org/GNOME/pitivi/issues/2193
This commit is contained in:
Thibault Saunier 2018-05-13 21:12:35 -04:00
parent ab956c20e8
commit 33d6490308
2 changed files with 50 additions and 10 deletions

View file

@ -49,6 +49,17 @@ static void _compute_height (GESContainer * container);
G_DEFINE_ABSTRACT_TYPE (GESClip, ges_clip, GES_TYPE_CONTAINER); G_DEFINE_ABSTRACT_TYPE (GESClip, ges_clip, GES_TYPE_CONTAINER);
typedef enum
{
GES_CLIP_IS_SPLITTING = (1 << 0),
GES_CLIP_IS_MOVING = (1 << 1),
} GESClipFlags;
#define FLAGS(obj) (GES_CLIP(obj)->priv->flags)
#define SET_FLAG(obj,flag) (FLAGS(obj) |= (flag))
#define UNSET_FLAG(obj,flag) (FLAGS(obj) &= ~(flag))
#define FLAG_IS_SET(obj,flag) (FLAGS(obj) == (flag))
struct _GESClipPrivate struct _GESClipPrivate
{ {
/*< public > */ /*< public > */
@ -59,7 +70,7 @@ struct _GESClipPrivate
/* Set to TRUE when the clip is doing updates of track element /* Set to TRUE when the clip is doing updates of track element
* properties so we don't end up in infinite property update loops * properties so we don't end up in infinite property update loops
*/ */
gboolean is_moving; GESClipFlags flags;
guint nb_effects; guint nb_effects;
@ -153,7 +164,8 @@ _set_start (GESTimelineElement * element, GstClockTime start)
/* Make the snapping happen if in a timeline */ /* Make the snapping happen if in a timeline */
timeline = GES_TIMELINE_ELEMENT_TIMELINE (child); timeline = GES_TIMELINE_ELEMENT_TIMELINE (child);
if (timeline && !container->initiated_move) { if (timeline && !container->initiated_move) {
if (ges_timeline_move_object_simple (timeline, child, NULL, GES_EDGE_NONE, start)) if (ges_timeline_move_object_simple (timeline, child, NULL,
GES_EDGE_NONE, start))
continue; continue;
} }
@ -199,9 +211,12 @@ _set_duration (GESTimelineElement * element, GstClockTime duration)
if (child != container->initiated_move) { if (child != container->initiated_move) {
/* Make the snapping happen if in a timeline */ /* Make the snapping happen if in a timeline */
timeline = GES_TIMELINE_ELEMENT_TIMELINE (child); timeline = GES_TIMELINE_ELEMENT_TIMELINE (child);
if (timeline == NULL || ges_timeline_trim_object_simple (timeline, child, if (timeline == NULL || FLAG_IS_SET (element, GES_CLIP_IS_SPLITTING) ||
NULL, GES_EDGE_END, _START (child) + duration, TRUE) == FALSE) (ges_timeline_trim_object_simple (timeline, child,
NULL, GES_EDGE_END, _START (child) + duration,
TRUE) == FALSE)) {
_set_duration0 (GES_TIMELINE_ELEMENT (child), duration); _set_duration0 (GES_TIMELINE_ELEMENT (child), duration);
}
} }
} }
container->children_control_mode = GES_CHILDREN_UPDATE; container->children_control_mode = GES_CHILDREN_UPDATE;
@ -841,7 +856,7 @@ ges_clip_init (GESClip * self)
/* FIXME, check why it was done this way _DURATION (self) = GST_SECOND; */ /* FIXME, check why it was done this way _DURATION (self) = GST_SECOND; */
self->priv->layer = NULL; self->priv->layer = NULL;
self->priv->nb_effects = 0; self->priv->nb_effects = 0;
self->priv->is_moving = FALSE; self->priv->flags = 0;
} }
/** /**
@ -984,7 +999,7 @@ ges_clip_set_layer (GESClip * clip, GESLayer * layer)
* it is actually the result of a move between layer (as we know * 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 * that it will be added to another layer right after, and this
* is what imports here.) */ * is what imports here.) */
if (!clip->priv->is_moving) if (!FLAG_IS_SET (clip, GES_CLIP_IS_MOVING))
g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]); g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
} }
@ -1012,7 +1027,10 @@ ges_clip_set_moving_from_layer (GESClip * clip, gboolean is_moving)
{ {
g_return_if_fail (GES_IS_CLIP (clip)); g_return_if_fail (GES_IS_CLIP (clip));
clip->priv->is_moving = is_moving; if (is_moving)
SET_FLAG (clip, GES_CLIP_IS_MOVING);
else
UNSET_FLAG (clip, GES_CLIP_IS_MOVING);
} }
/** /**
@ -1031,7 +1049,7 @@ ges_clip_is_moving_from_layer (GESClip * clip)
{ {
g_return_val_if_fail (GES_IS_CLIP (clip), FALSE); g_return_val_if_fail (GES_IS_CLIP (clip), FALSE);
return clip->priv->is_moving; return FLAG_IS_SET (clip, GES_CLIP_IS_MOVING);
} }
/** /**
@ -1064,7 +1082,7 @@ ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
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));
clip->priv->is_moving = TRUE; SET_FLAG (clip, GES_CLIP_IS_MOVING);
gst_object_ref (clip); gst_object_ref (clip);
ret = ges_layer_remove_clip (current_layer, clip); ret = ges_layer_remove_clip (current_layer, clip);
@ -1074,7 +1092,7 @@ ges_clip_move_to_layer (GESClip * clip, GESLayer * layer)
} }
ret = ges_layer_add_clip (layer, clip); ret = ges_layer_add_clip (layer, clip);
clip->priv->is_moving = FALSE; 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]); g_object_notify_by_pspec (G_OBJECT (clip), properties[PROP_LAYER]);
@ -1383,7 +1401,9 @@ ges_clip_split (GESClip * clip, guint64 position)
position - start + inpoint); position - start + inpoint);
} }
SET_FLAG (clip, GES_CLIP_IS_SPLITTING);
_set_duration0 (GES_TIMELINE_ELEMENT (clip), old_duration); _set_duration0 (GES_TIMELINE_ELEMENT (clip), old_duration);
UNSET_FLAG (clip, GES_CLIP_IS_SPLITTING);
if (GES_TIMELINE_ELEMENT_TIMELINE (clip)) { if (GES_TIMELINE_ELEMENT_TIMELINE (clip)) {
for (tmp = GES_CONTAINER_CHILDREN (new_object); tmp; tmp = tmp->next) { for (tmp = GES_CONTAINER_CHILDREN (new_object); tmp; tmp = tmp->next) {

View file

@ -193,6 +193,26 @@ class TestSnapping(GESSimpleTimelineTest):
clip2.props.start - 1) clip2.props.start - 1)
self.assertEqual(clip2.props.start, split_position) self.assertEqual(clip2.props.start, split_position)
def test_no_snapping_on_split(self):
self.timeline.props.auto_transition = True
self.timeline.set_snapping_distance(1)
not_called = []
def snapping_started_cb(timeline, element1, element2, dist, self):
Gst.error("Here %s %s" % (Gst.TIME_ARGS(element1.props.start + element1.props.duration),
Gst.TIME_ARGS(element2.props.start)))
not_called.append("No snapping should happen")
self.timeline.connect('snapping-started', snapping_started_cb, self)
clip1 = self.add_clip(0, 0, 100)
# Split clip1.
split_position = 50
clip2 = clip1.split(split_position)
self.assertEqual(not_called, [])
self.assertEqual(len(self.layer.get_clips()), 2)
self.assertEqual(clip1.props.duration, split_position)
self.assertEqual(clip2.props.start, split_position)
class TestTransitions(GESSimpleTimelineTest): class TestTransitions(GESSimpleTimelineTest):