mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-19 05:45:58 +00:00
53d335b4ed
Unlike ges_container_add, this lets you set the index and will check that track selection did not fail. This is useful for time effects whose addition would create an unsupported timeline configuration. Also can use the clip add error in ges_timeline_add_clip to let the user know when adding a clip to a layer that its in-point is set larger than the max-duration of its core children. Part-of: <https://gitlab.freedesktop.org/gstreamer/gst-editing-services/-/merge_requests/177>
913 lines
30 KiB
C
913 lines
30 KiB
C
/* GStreamer Editing Services
|
|
* Copyright (C) 2010 Thibault Saunier <tsaunier@gnome.org>
|
|
*
|
|
* 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
|
|
* Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
|
* Boston, MA 02110-1301, USA.
|
|
*/
|
|
|
|
#include "test-utils.h"
|
|
#include <ges/ges.h>
|
|
#include <gst/check/gstcheck.h>
|
|
|
|
void
|
|
deep_prop_changed_cb (GESTrackElement * track_element, GstElement * element,
|
|
GParamSpec * spec);
|
|
|
|
GST_START_TEST (test_effect_basic)
|
|
{
|
|
GESEffect *effect;
|
|
|
|
ges_init ();
|
|
|
|
effect = ges_effect_new ("agingtv");
|
|
fail_unless (effect != NULL);
|
|
gst_object_unref (effect);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_add_effect_to_clip)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESTrack *track_audio, *track_video;
|
|
GESEffect *effect;
|
|
GESTestClip *source;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_audio = GES_TRACK (ges_audio_track_new ());
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_audio);
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
source = ges_test_clip_new ();
|
|
|
|
g_object_set (source, "duration", 10 * GST_SECOND, NULL);
|
|
|
|
ges_layer_add_clip (layer, (GESClip *) source);
|
|
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect = ges_effect_new ("agingtv");
|
|
|
|
fail_unless (GES_IS_EFFECT (effect));
|
|
fail_unless (ges_container_add (GES_CONTAINER (source),
|
|
GES_TIMELINE_ELEMENT (effect)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect)) !=
|
|
NULL);
|
|
|
|
assert_equals_int (GES_TRACK_ELEMENT (effect)->active, TRUE);
|
|
|
|
ges_layer_remove_clip (layer, (GESClip *) source);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_get_effects_from_tl)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESTrack *track_video;
|
|
GESTrackElement *video_source;
|
|
GESEffect *effect, *effect1, *effect2;
|
|
GESTestClip *source;
|
|
GList *effects, *tmp = NULL;
|
|
gint effect_prio = -1;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
source = ges_test_clip_new ();
|
|
|
|
g_object_set (source, "duration", 10 * GST_SECOND, NULL);
|
|
|
|
GST_DEBUG ("Adding source to layer");
|
|
ges_layer_add_clip (layer, (GESClip *) source);
|
|
assert_equals_int (g_list_length (GES_CONTAINER_CHILDREN (source)), 1);
|
|
video_source = GES_CONTAINER_CHILDREN (source)->data;
|
|
fail_unless (GES_IS_VIDEO_TEST_SOURCE (video_source));
|
|
assert_equals_int (_PRIORITY (video_source),
|
|
MIN_NLE_PRIO + TRANSITIONS_HEIGHT);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect = ges_effect_new ("agingtv");
|
|
effect1 = ges_effect_new ("agingtv");
|
|
effect2 = ges_effect_new ("agingtv");
|
|
|
|
fail_unless (GES_IS_EFFECT (effect));
|
|
fail_unless (GES_IS_EFFECT (effect1));
|
|
fail_unless (GES_IS_EFFECT (effect2));
|
|
|
|
GST_DEBUG ("Adding effect (0)");
|
|
fail_unless (ges_container_add (GES_CONTAINER (source),
|
|
GES_TIMELINE_ELEMENT (effect)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect)) ==
|
|
track_video);
|
|
assert_equals_int (_PRIORITY (effect), MIN_NLE_PRIO + TRANSITIONS_HEIGHT + 0);
|
|
assert_equals_int (_PRIORITY (video_source),
|
|
MIN_NLE_PRIO + TRANSITIONS_HEIGHT + 1);
|
|
|
|
GST_DEBUG ("Adding effect 1");
|
|
fail_unless (ges_container_add (GES_CONTAINER (source),
|
|
GES_TIMELINE_ELEMENT (effect1)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect1)) ==
|
|
track_video);
|
|
assert_equals_int (_PRIORITY (effect), MIN_NLE_PRIO + TRANSITIONS_HEIGHT);
|
|
assert_equals_int (_PRIORITY (effect1),
|
|
MIN_NLE_PRIO + TRANSITIONS_HEIGHT + 1);
|
|
assert_equals_int (_PRIORITY (video_source),
|
|
MIN_NLE_PRIO + TRANSITIONS_HEIGHT + 2);
|
|
|
|
GST_DEBUG ("Adding effect 2");
|
|
fail_unless (ges_container_add (GES_CONTAINER (source),
|
|
GES_TIMELINE_ELEMENT (effect2)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect2)) ==
|
|
track_video);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (source), 4);
|
|
|
|
effects = ges_clip_get_top_effects (GES_CLIP (source));
|
|
fail_unless (g_list_length (effects) == 3);
|
|
for (tmp = effects; tmp; tmp = tmp->next) {
|
|
gint priority = ges_clip_get_top_effect_position (GES_CLIP (source),
|
|
GES_BASE_EFFECT (tmp->data));
|
|
fail_unless (priority > effect_prio);
|
|
fail_unless (GES_IS_EFFECT (tmp->data));
|
|
effect_prio = priority;
|
|
|
|
gst_object_unref (tmp->data);
|
|
}
|
|
g_list_free (effects);
|
|
|
|
ges_layer_remove_clip (layer, (GESClip *) source);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_effect_clip)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESTrack *track_audio, *track_video;
|
|
GESEffectClip *effect_clip;
|
|
GESEffect *effect, *effect1, *core_effect, *core_effect1;
|
|
GList *children, *top_effects, *tmp;
|
|
gint clip_height;
|
|
gint core_effect_prio;
|
|
gint effect_index, effect1_index;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_audio = GES_TRACK (ges_audio_track_new ());
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_audio);
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
/* these are the core video and audio effects for the clip */
|
|
effect_clip = ges_effect_clip_new ("videobalance", "audioecho");
|
|
|
|
g_object_set (effect_clip, "duration", 25 * GST_SECOND, NULL);
|
|
|
|
ges_layer_add_clip (layer, (GESClip *) effect_clip);
|
|
|
|
/* core elements should now be created */
|
|
fail_unless (children = GES_CONTAINER_CHILDREN (effect_clip));
|
|
core_effect = GES_EFFECT (children->data);
|
|
fail_unless (children = children->next);
|
|
core_effect1 = GES_EFFECT (children->data);
|
|
fail_unless (children->next == NULL);
|
|
|
|
/* both effects are placed at the same priority since they are core
|
|
* children of the clip, destined for different tracks */
|
|
core_effect_prio = _PRIORITY (core_effect);
|
|
assert_equals_int (core_effect_prio, _PRIORITY (core_effect1));
|
|
g_object_get (effect_clip, "height", &clip_height, NULL);
|
|
assert_equals_int (clip_height, 1);
|
|
|
|
/* add additional non-core effects */
|
|
effect = ges_effect_new ("agingtv");
|
|
fail_unless (ges_container_add (GES_CONTAINER (effect_clip),
|
|
GES_TIMELINE_ELEMENT (effect)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect)) ==
|
|
track_video);
|
|
|
|
/* placed at a higher priority than the effects */
|
|
core_effect_prio = _PRIORITY (core_effect);
|
|
assert_equals_int (core_effect_prio, _PRIORITY (core_effect1));
|
|
fail_unless (_PRIORITY (effect) < core_effect_prio);
|
|
g_object_get (effect_clip, "height", &clip_height, NULL);
|
|
assert_equals_int (clip_height, 2);
|
|
|
|
effect_index =
|
|
ges_clip_get_top_effect_index (GES_CLIP (effect_clip),
|
|
GES_BASE_EFFECT (effect));
|
|
assert_equals_int (effect_index, 0);
|
|
|
|
/* 'effect1' is placed in between the core children and 'effect' */
|
|
effect1 = ges_effect_new ("audiopanorama");
|
|
fail_unless (ges_container_add (GES_CONTAINER (effect_clip),
|
|
GES_TIMELINE_ELEMENT (effect1)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect1)) ==
|
|
track_audio);
|
|
|
|
/* 'effect' is still the highest priority effect, and the core
|
|
* elements are at the lowest priority */
|
|
core_effect_prio = _PRIORITY (core_effect);
|
|
assert_equals_int (core_effect_prio, _PRIORITY (core_effect1));
|
|
fail_unless (_PRIORITY (effect1) < core_effect_prio);
|
|
fail_unless (_PRIORITY (effect1) > _PRIORITY (effect));
|
|
g_object_get (effect_clip, "height", &clip_height, NULL);
|
|
assert_equals_int (clip_height, 3);
|
|
|
|
effect_index =
|
|
ges_clip_get_top_effect_index (GES_CLIP (effect_clip),
|
|
GES_BASE_EFFECT (effect));
|
|
effect1_index =
|
|
ges_clip_get_top_effect_index (GES_CLIP (effect_clip),
|
|
GES_BASE_EFFECT (effect1));
|
|
assert_equals_int (effect_index, 0);
|
|
assert_equals_int (effect1_index, 1);
|
|
|
|
/* all effects are children of the effect_clip, ordered by priority */
|
|
fail_unless (children = GES_CONTAINER_CHILDREN (effect_clip));
|
|
fail_unless (children->data == effect);
|
|
fail_unless (children = children->next);
|
|
fail_unless (children->data == effect1);
|
|
fail_unless (children = children->next);
|
|
fail_unless (children->data == core_effect);
|
|
fail_unless (children = children->next);
|
|
fail_unless (children->data == core_effect1);
|
|
fail_unless (children->next == NULL);
|
|
|
|
/* but only the additional effects are part of the top effects */
|
|
top_effects = ges_clip_get_top_effects (GES_CLIP (effect_clip));
|
|
fail_unless (tmp = top_effects);
|
|
fail_unless (tmp->data == effect);
|
|
fail_unless (tmp = tmp->next);
|
|
fail_unless (tmp->data == effect1);
|
|
fail_unless (tmp->next == NULL);
|
|
|
|
g_list_free_full (top_effects, gst_object_unref);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_priorities_clip)
|
|
{
|
|
GList *top_effects, *tmp;
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESClip *effect_clip;
|
|
GESTrack *track_audio, *track_video, *track;
|
|
GESBaseEffect *effects[6], *audio_effect = NULL, *video_effect = NULL;
|
|
gint prev_index, i, num_effects = G_N_ELEMENTS (effects);
|
|
guint32 base_prio = MIN_NLE_PRIO + TRANSITIONS_HEIGHT;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_audio = GES_TRACK (ges_audio_track_new ());
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_audio);
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect_clip = GES_CLIP (ges_effect_clip_new ("videobalance", "audioecho"));
|
|
|
|
g_object_set (effect_clip, "duration", 25 * GST_SECOND, NULL);
|
|
|
|
ges_layer_add_clip ((layer), (GESClip *) effect_clip);
|
|
for (tmp = GES_CONTAINER_CHILDREN (effect_clip); tmp; tmp = tmp->next) {
|
|
if (ges_track_element_get_track_type (GES_TRACK_ELEMENT (tmp->data)) ==
|
|
GES_TRACK_TYPE_AUDIO)
|
|
audio_effect = tmp->data;
|
|
else if (ges_track_element_get_track_type (GES_TRACK_ELEMENT (tmp->data)) ==
|
|
GES_TRACK_TYPE_VIDEO)
|
|
video_effect = tmp->data;
|
|
else
|
|
g_assert (0);
|
|
}
|
|
fail_unless (GES_IS_EFFECT (audio_effect));
|
|
fail_unless (GES_IS_EFFECT (video_effect));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (audio_effect)) ==
|
|
track_audio);
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (video_effect)) ==
|
|
track_video);
|
|
|
|
/* both the core effects have the same priority */
|
|
assert_equals_int (_PRIORITY (audio_effect), base_prio);
|
|
assert_equals_int (_PRIORITY (video_effect), base_prio);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), 1);
|
|
|
|
/* can not change their priority using the top effect methods since
|
|
* they are not top effects */
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, audio_effect, 1)
|
|
== FALSE);
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, video_effect, 0)
|
|
== FALSE);
|
|
|
|
/* adding non-core effects */
|
|
GST_DEBUG ("Adding effects to the effect clip ");
|
|
for (i = 0; i < num_effects; i++) {
|
|
if (i % 2)
|
|
effects[i] = GES_BASE_EFFECT (ges_effect_new ("agingtv"));
|
|
else
|
|
effects[i] = GES_BASE_EFFECT (ges_effect_new ("audiopanorama"));
|
|
fail_unless (ges_container_add (GES_CONTAINER (effect_clip),
|
|
GES_TIMELINE_ELEMENT (effects[i])));
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), 2 + i);
|
|
track = ges_track_element_get_track (GES_TRACK_ELEMENT (effects[i]));
|
|
if (i % 2)
|
|
fail_unless (track == track_video);
|
|
else
|
|
fail_unless (track == track_audio);
|
|
}
|
|
|
|
/* change top effect index */
|
|
for (i = 0; i < num_effects; i++) {
|
|
assert_equals_int (ges_clip_get_top_effect_index (effect_clip, effects[i]),
|
|
i);
|
|
assert_equals_int (_PRIORITY (effects[i]), i + base_prio);
|
|
}
|
|
|
|
assert_equals_int (_PRIORITY (video_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (audio_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (effect_clip), 1);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), num_effects + 1);
|
|
|
|
/* moving 4th effect to index 1 should only change the priority of
|
|
* effects 1, 2, 3, and 4 because these lie between the new index (1)
|
|
* and the old index (4). */
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, effects[4], 1));
|
|
|
|
assert_equals_int (_PRIORITY (effects[0]), 0 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[1]), 2 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[2]), 3 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[3]), 4 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[4]), 1 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[5]), 5 + base_prio);
|
|
|
|
/* everything else stays the same */
|
|
assert_equals_int (_PRIORITY (video_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (audio_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (effect_clip), 1);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), num_effects + 1);
|
|
|
|
/* move back */
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, effects[4], 4));
|
|
|
|
for (i = 0; i < num_effects; i++) {
|
|
assert_equals_int (ges_clip_get_top_effect_index (effect_clip, effects[i]),
|
|
i);
|
|
assert_equals_int (_PRIORITY (effects[i]), i + base_prio);
|
|
}
|
|
|
|
assert_equals_int (_PRIORITY (video_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (audio_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (effect_clip), 1);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), num_effects + 1);
|
|
|
|
/* moving 2nd effect to index 4 should only change the priority of
|
|
* effects 2, 3 and 4 because these lie between the new index (4) and
|
|
* the old index (2). */
|
|
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, effects[2], 4));
|
|
|
|
assert_equals_int (_PRIORITY (effects[0]), 0 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[1]), 1 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[2]), 4 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[3]), 2 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[4]), 3 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[5]), 5 + base_prio);
|
|
|
|
/* everything else stays the same */
|
|
assert_equals_int (_PRIORITY (video_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (audio_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (effect_clip), 1);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), num_effects + 1);
|
|
|
|
/* move 4th effect to index 0 should only change the priority of
|
|
* effects 0, 1, 3 and 4 because these lie between the new index (0) and
|
|
* the old index (3) */
|
|
|
|
fail_unless (ges_clip_set_top_effect_index (effect_clip, effects[4], 0));
|
|
|
|
assert_equals_int (_PRIORITY (effects[0]), 1 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[1]), 2 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[2]), 4 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[3]), 3 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[4]), 0 + base_prio);
|
|
assert_equals_int (_PRIORITY (effects[5]), 5 + base_prio);
|
|
|
|
/* everything else stays the same */
|
|
assert_equals_int (_PRIORITY (video_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (audio_effect), num_effects + base_prio);
|
|
assert_equals_int (_PRIORITY (effect_clip), 1);
|
|
assert_equals_int (GES_CONTAINER_HEIGHT (effect_clip), num_effects + 1);
|
|
|
|
/* make sure top effects are ordered by index */
|
|
top_effects = ges_clip_get_top_effects (effect_clip);
|
|
prev_index = -1;
|
|
for (tmp = top_effects; tmp; tmp = tmp->next) {
|
|
gint index = ges_clip_get_top_effect_index (effect_clip,
|
|
GES_BASE_EFFECT (tmp->data));
|
|
fail_unless (index >= 0);
|
|
fail_unless (index > prev_index);
|
|
fail_unless (GES_IS_EFFECT (tmp->data));
|
|
prev_index = index;
|
|
}
|
|
g_list_free_full (top_effects, gst_object_unref);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_effect_set_properties)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESTrack *track_video;
|
|
GESEffectClip *effect_clip;
|
|
GESTimelineElement *effect;
|
|
guint scratch_line, n_props, i;
|
|
gboolean color_aging;
|
|
GParamSpec **pspecs, *spec;
|
|
GValue val = { 0 };
|
|
GValue nval = { 0 };
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect_clip = ges_effect_clip_new ("agingtv", NULL);
|
|
|
|
g_object_set (effect_clip, "duration", 25 * GST_SECOND, NULL);
|
|
|
|
ges_layer_add_clip (layer, (GESClip *) effect_clip);
|
|
|
|
effect = GES_TIMELINE_ELEMENT (ges_effect_new ("agingtv"));
|
|
fail_unless (ges_container_add (GES_CONTAINER (effect_clip),
|
|
GES_TIMELINE_ELEMENT (effect)));
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect)) ==
|
|
track_video);
|
|
|
|
ges_timeline_element_set_child_properties (effect,
|
|
"GstAgingTV::scratch-lines", 17, "color-aging", FALSE, NULL);
|
|
ges_timeline_element_get_child_properties (effect,
|
|
"GstAgingTV::scratch-lines", &scratch_line,
|
|
"color-aging", &color_aging, NULL);
|
|
fail_unless (scratch_line == 17);
|
|
fail_unless (color_aging == FALSE);
|
|
|
|
pspecs = ges_timeline_element_list_children_properties (effect, &n_props);
|
|
fail_unless (n_props == 7);
|
|
|
|
spec = pspecs[0];
|
|
i = 1;
|
|
while (g_strcmp0 (spec->name, "scratch-lines")) {
|
|
spec = pspecs[i++];
|
|
}
|
|
|
|
g_value_init (&val, G_TYPE_UINT);
|
|
g_value_init (&nval, G_TYPE_UINT);
|
|
g_value_set_uint (&val, 10);
|
|
|
|
ges_timeline_element_set_child_property_by_pspec (effect, spec, &val);
|
|
ges_timeline_element_get_child_property_by_pspec (effect, spec, &nval);
|
|
fail_unless (g_value_get_uint (&nval) == 10);
|
|
|
|
for (i = 0; i < n_props; i++) {
|
|
g_param_spec_unref (pspecs[i]);
|
|
}
|
|
g_free (pspecs);
|
|
|
|
ges_layer_remove_clip (layer, (GESClip *) effect_clip);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static void
|
|
effect_added_cb (GESClip * clip, GESBaseEffect * trop, gboolean * effect_added)
|
|
{
|
|
GST_DEBUG ("Effect added");
|
|
fail_unless (GES_IS_CLIP (clip));
|
|
fail_unless (GES_IS_EFFECT (trop));
|
|
*effect_added = TRUE;
|
|
}
|
|
|
|
void
|
|
deep_prop_changed_cb (GESTrackElement * track_element, GstElement * element,
|
|
GParamSpec * spec)
|
|
{
|
|
GST_DEBUG ("%s property changed", g_param_spec_get_name (spec));
|
|
fail_unless (GES_IS_TRACK_ELEMENT (track_element));
|
|
fail_unless (GST_IS_ELEMENT (element));
|
|
}
|
|
|
|
GST_START_TEST (test_clip_signals)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESLayer *layer;
|
|
GESTrack *track_video;
|
|
GESEffectClip *effect_clip;
|
|
GESTimelineElement *effect;
|
|
GValue val = { 0, };
|
|
gboolean effect_added = FALSE;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_layer_new ();
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect_clip = ges_effect_clip_new ("agingtv", NULL);
|
|
g_signal_connect (effect_clip, "child-added", (GCallback) effect_added_cb,
|
|
&effect_added);
|
|
|
|
g_object_set (effect_clip, "duration", 25 * GST_SECOND, NULL);
|
|
|
|
ges_layer_add_clip (layer, (GESClip *) effect_clip);
|
|
|
|
effect = GES_TIMELINE_ELEMENT (ges_effect_new ("agingtv"));
|
|
fail_unless (ges_container_add (GES_CONTAINER (effect_clip), effect));
|
|
fail_unless (effect_added);
|
|
g_signal_handlers_disconnect_by_func (effect_clip, effect_added_cb,
|
|
&effect_added);
|
|
fail_unless (ges_track_element_get_track (GES_TRACK_ELEMENT (effect)) ==
|
|
track_video);
|
|
g_signal_connect (effect, "deep-notify", (GCallback) deep_prop_changed_cb,
|
|
effect);
|
|
|
|
ges_timeline_element_set_child_properties (effect,
|
|
"GstAgingTV::scratch-lines", 17, NULL);
|
|
|
|
g_value_init (&val, G_TYPE_UINT);
|
|
ges_timeline_element_get_child_property (effect,
|
|
"GstAgingTV::scratch-lines", &val);
|
|
fail_unless (G_VALUE_HOLDS_UINT (&val));
|
|
g_value_unset (&val);
|
|
|
|
ges_layer_remove_clip (layer, (GESClip *) effect_clip);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_split_clip_effect_priorities)
|
|
{
|
|
GESLayer *layer;
|
|
GESTimeline *timeline;
|
|
GESTrack *track_video;
|
|
GESClip *clip, *nclip;
|
|
GESEffect *effect;
|
|
GESTrackElement *source, *nsource, *neffect;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new ();
|
|
layer = ges_timeline_append_layer (timeline);
|
|
track_video = GES_TRACK (ges_video_track_new ());
|
|
|
|
g_object_set (timeline, "auto-transition", TRUE, NULL);
|
|
ges_timeline_add_track (timeline, track_video);
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
GST_DEBUG ("Create effect");
|
|
effect = ges_effect_new ("agingtv");
|
|
clip = GES_CLIP (ges_test_clip_new ());
|
|
g_object_set (clip, "duration", GST_SECOND * 2, NULL);
|
|
|
|
fail_unless (ges_container_add (GES_CONTAINER (clip),
|
|
GES_TIMELINE_ELEMENT (effect)));
|
|
ges_layer_add_clip (layer, clip);
|
|
|
|
source = ges_clip_find_track_element (clip, NULL, GES_TYPE_VIDEO_SOURCE);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (effect), 3);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (source), 4);
|
|
|
|
nclip = ges_clip_split (clip, GST_SECOND);
|
|
fail_unless (nclip);
|
|
neffect = ges_clip_find_track_element (nclip, NULL, GES_TYPE_EFFECT);
|
|
nsource = ges_clip_find_track_element (nclip, NULL, GES_TYPE_VIDEO_SOURCE);
|
|
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (effect), 3);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (source), 4);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (neffect), 5);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (nsource), 6);
|
|
|
|
/* Create a transition ... */
|
|
ges_timeline_element_set_start (GES_TIMELINE_ELEMENT (clip), GST_SECOND / 2);
|
|
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (effect), 3);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (source), 4);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (neffect), 5);
|
|
assert_equals_uint64 (GES_TIMELINE_ELEMENT_PRIORITY (nsource), 6);
|
|
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
#define _NO_ERROR -1
|
|
#define _set_rate(videorate, rate, error_code) \
|
|
{ \
|
|
g_value_set_double (&val, rate); \
|
|
if (error_code != _NO_ERROR) { \
|
|
fail_if (ges_timeline_element_set_child_property_full ( \
|
|
GES_TIMELINE_ELEMENT (videorate), "rate", &val, &error)); \
|
|
assert_GESError (error, error_code); \
|
|
} else { \
|
|
fail_unless (ges_timeline_element_set_child_property_full ( \
|
|
GES_TIMELINE_ELEMENT (videorate), "rate", &val, &error)); \
|
|
fail_if (error); \
|
|
} \
|
|
}
|
|
|
|
#define _add_effect(clip, effect, _index, error_code) \
|
|
{ \
|
|
gint index = _index; \
|
|
if (error_code != _NO_ERROR) { \
|
|
fail_if (ges_clip_add_top_effect (clip, effect, index, &error)); \
|
|
assert_GESError (error, error_code); \
|
|
} else { \
|
|
GList *effects; \
|
|
gboolean res = ges_clip_add_top_effect (clip, effect, index, &error); \
|
|
fail_unless (res, "Adding effect " #effect " failed: %s", \
|
|
error ? error->message : "No error produced"); \
|
|
fail_if (error); \
|
|
effects = ges_clip_get_top_effects (clip); \
|
|
fail_unless (g_list_find (effects, effect)); \
|
|
if (index < 0 || index >= g_list_length (effects)) \
|
|
index = g_list_length (effects) - 1; \
|
|
assert_equals_int (ges_clip_get_top_effect_index (clip, effect), index); \
|
|
fail_unless (g_list_nth_data (effects, index) == effect); \
|
|
g_list_free_full (effects, gst_object_unref); \
|
|
} \
|
|
}
|
|
|
|
#define _remove_effect(clip, effect, error_code) \
|
|
{ \
|
|
if (error_code != _NO_ERROR) { \
|
|
fail_if (ges_clip_remove_top_effect (clip, effect, &error)); \
|
|
assert_GESError (error, error_code); \
|
|
} else { \
|
|
GList *effects; \
|
|
gboolean res = ges_clip_remove_top_effect (clip, effect, &error); \
|
|
fail_unless (res, "Removing effect " #effect " failed: %s", \
|
|
error ? error->message : "No error produced"); \
|
|
fail_if (error); \
|
|
effects = ges_clip_get_top_effects (clip); \
|
|
fail_if (g_list_find (effects, effect)); \
|
|
g_list_free_full (effects, gst_object_unref); \
|
|
} \
|
|
}
|
|
|
|
#define _move_effect(clip, effect, index, error_code) \
|
|
{ \
|
|
if (error_code != _NO_ERROR) { \
|
|
fail_if ( \
|
|
ges_clip_set_top_effect_index_full (clip, effect, index, &error)); \
|
|
assert_GESError (error, error_code); \
|
|
} else { \
|
|
GList *effects; \
|
|
gboolean res = \
|
|
ges_clip_set_top_effect_index_full (clip, effect, index, &error); \
|
|
fail_unless (res, "Moving effect " #effect " failed: %s", \
|
|
error ? error->message : "No error produced"); \
|
|
fail_if (error); \
|
|
effects = ges_clip_get_top_effects (clip); \
|
|
fail_unless (g_list_find (effects, effect)); \
|
|
assert_equals_int (ges_clip_get_top_effect_index (clip, effect), index); \
|
|
fail_unless (g_list_nth_data (effects, index) == effect); \
|
|
g_list_free_full (effects, gst_object_unref); \
|
|
} \
|
|
}
|
|
|
|
GST_START_TEST (test_move_time_effect)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESTrack *track;
|
|
GESLayer *layer;
|
|
GESAsset *asset;
|
|
GESClip *clip;
|
|
GESBaseEffect *rate0, *rate1, *overlay;
|
|
GError *error = NULL;
|
|
GValue val = G_VALUE_INIT;
|
|
|
|
ges_init ();
|
|
|
|
g_value_init (&val, G_TYPE_DOUBLE);
|
|
|
|
|
|
timeline = ges_timeline_new ();
|
|
track = GES_TRACK (ges_video_track_new ());
|
|
fail_unless (ges_timeline_add_track (timeline, track));
|
|
|
|
layer = ges_timeline_append_layer (timeline);
|
|
|
|
/* add a dummy clip for overlap */
|
|
asset = ges_asset_request (GES_TYPE_TEST_CLIP, "max-duration=16", &error);
|
|
fail_unless (asset);
|
|
fail_if (error);
|
|
|
|
fail_unless (ges_layer_add_asset_full (layer, asset, 0, 0, 16,
|
|
GES_TRACK_TYPE_UNKNOWN, &error));
|
|
fail_if (error);
|
|
|
|
clip = GES_CLIP (ges_asset_extract (asset, &error));
|
|
fail_unless (clip);
|
|
fail_if (error);
|
|
assert_set_start (clip, 8);
|
|
assert_set_duration (clip, 16);
|
|
|
|
rate0 = GES_BASE_EFFECT (ges_effect_new ("videorate"));
|
|
rate1 = GES_BASE_EFFECT (ges_effect_new ("videorate"));
|
|
overlay = GES_BASE_EFFECT (ges_effect_new ("textoverlay"));
|
|
|
|
ges_track_element_set_has_internal_source (GES_TRACK_ELEMENT (overlay), TRUE);
|
|
/* only has 8ns of content */
|
|
assert_set_inpoint (overlay, 13);
|
|
assert_set_max_duration (overlay, 21);
|
|
|
|
_set_rate (rate0, 2.0, _NO_ERROR);
|
|
_set_rate (rate1, 0.5, _NO_ERROR);
|
|
|
|
/* keep alive */
|
|
gst_object_ref (clip);
|
|
gst_object_ref (rate0);
|
|
gst_object_ref (rate1);
|
|
gst_object_ref (overlay);
|
|
|
|
/* cannot add to layer with rate effect because it would cause a full
|
|
* overlap */
|
|
_add_effect (clip, rate0, 0, _NO_ERROR);
|
|
fail_if (ges_layer_add_clip_full (layer, clip, &error));
|
|
assert_GESError (error, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
_remove_effect (clip, rate0, _NO_ERROR);
|
|
|
|
/* same with overlay */
|
|
_add_effect (clip, overlay, 0, _NO_ERROR);
|
|
fail_if (ges_layer_add_clip_full (layer, clip, &error));
|
|
assert_GESError (error, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
_remove_effect (clip, overlay, _NO_ERROR);
|
|
|
|
CHECK_OBJECT_PROPS (clip, 8, 0, 16);
|
|
|
|
fail_unless (ges_layer_add_clip_full (layer, clip, &error));
|
|
fail_if (error);
|
|
|
|
CHECK_OBJECT_PROPS_MAX (clip, 8, 0, 16, 16);
|
|
|
|
/* can't add rate0 or overlay in the same way */
|
|
_add_effect (clip, rate0, 0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
_add_effect (clip, overlay, 0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
|
|
/* rate1 extends the duration-limit instead */
|
|
_add_effect (clip, rate1, 0, _NO_ERROR);
|
|
|
|
/* can't add overlay next to the timeline */
|
|
_add_effect (clip, overlay, 0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
/* but next to source is ok */
|
|
_add_effect (clip, overlay, 1, _NO_ERROR);
|
|
|
|
/* can't add rate0 after overlay */
|
|
_add_effect (clip, rate0, 1, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
/* but before is ok */
|
|
_add_effect (clip, rate0, -1, _NO_ERROR);
|
|
|
|
/* can't move rate0 to end */
|
|
_move_effect (clip, rate0, 0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
/* can't move overlay to start or end */
|
|
_move_effect (clip, overlay, 0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
_move_effect (clip, overlay, 2, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
|
|
/* can now move: swap places with rate1 */
|
|
_set_rate (rate0, 0.5, _NO_ERROR);
|
|
_move_effect (clip, rate0, 0, _NO_ERROR);
|
|
_move_effect (clip, rate1, 2, _NO_ERROR);
|
|
_set_rate (rate1, 2.0, _NO_ERROR);
|
|
|
|
/* cannot speed up either rate too much */
|
|
_set_rate (rate0, 1.0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
_set_rate (rate1, 4.0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
|
|
/* cannot remove rate0 which is slowing down */
|
|
_remove_effect (clip, rate0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
|
|
/* removing the speed-up is fine */
|
|
_remove_effect (clip, rate1, _NO_ERROR);
|
|
|
|
/* removing the overlay is fine */
|
|
_remove_effect (clip, overlay, _NO_ERROR);
|
|
|
|
CHECK_OBJECT_PROPS_MAX (clip, 8, 0, 16, 16);
|
|
assert_set_max_duration (clip, 8);
|
|
CHECK_OBJECT_PROPS_MAX (clip, 8, 0, 16, 8);
|
|
/* still can't remove the slow down since it is the only thing stopping
|
|
* a full overlap */
|
|
_remove_effect (clip, rate0, GES_ERROR_INVALID_OVERLAP_IN_TRACK);
|
|
|
|
gst_object_unref (clip);
|
|
/* shouldn't have any problems when removing from the layer */
|
|
fail_unless (ges_layer_remove_clip (layer, clip));
|
|
|
|
g_value_reset (&val);
|
|
gst_object_unref (rate0);
|
|
gst_object_unref (rate1);
|
|
gst_object_unref (overlay);
|
|
gst_object_unref (asset);
|
|
gst_object_unref (timeline);
|
|
|
|
ges_deinit ();
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
static Suite *
|
|
ges_suite (void)
|
|
{
|
|
Suite *s = suite_create ("ges");
|
|
TCase *tc_chain = tcase_create ("effect");
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
|
|
tcase_add_test (tc_chain, test_effect_basic);
|
|
tcase_add_test (tc_chain, test_add_effect_to_clip);
|
|
tcase_add_test (tc_chain, test_get_effects_from_tl);
|
|
tcase_add_test (tc_chain, test_effect_clip);
|
|
tcase_add_test (tc_chain, test_priorities_clip);
|
|
tcase_add_test (tc_chain, test_effect_set_properties);
|
|
tcase_add_test (tc_chain, test_clip_signals);
|
|
tcase_add_test (tc_chain, test_split_clip_effect_priorities);
|
|
tcase_add_test (tc_chain, test_move_time_effect);
|
|
|
|
return s;
|
|
}
|
|
|
|
GST_CHECK_MAIN (ges);
|