mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-01 13:08:49 +00:00
364 lines
13 KiB
C
364 lines
13 KiB
C
/* GStreamer Editing Services
|
|
* Copyright (C) 2010 Edward Hervey <bilboed@bilboed.com>
|
|
*
|
|
* 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., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <ges/ges.h>
|
|
#include <gst/check/gstcheck.h>
|
|
|
|
#define LAYER_HEIGHT 1000
|
|
|
|
static gboolean
|
|
my_fill_track_func (GESTimelineObject * object,
|
|
GESTrackObject * trobject, GstElement * gnlobj, gpointer user_data)
|
|
{
|
|
GstElement *src;
|
|
|
|
GST_DEBUG ("timelineobj:%p, trackobjec:%p, gnlobj:%p",
|
|
object, trobject, gnlobj);
|
|
|
|
/* Let's just put a fakesource in for the time being */
|
|
src = gst_element_factory_make ("fakesrc", NULL);
|
|
|
|
/* If this fails... that means that there already was something
|
|
* in it */
|
|
fail_unless (gst_bin_add (GST_BIN (gnlobj), src));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#define gnl_object_check(gnlobj, start, duration, mstart, mduration, priority, active) { \
|
|
guint64 pstart, pdur, pmstart, pmdur; \
|
|
guint32 pprio; \
|
|
gboolean pact; \
|
|
g_object_get (gnlobj, "start", &pstart, "duration", &pdur, \
|
|
"media-start", &pmstart, "media-duration", &pmdur, \
|
|
"priority", &pprio, "active", &pact, \
|
|
NULL); \
|
|
assert_equals_uint64 (pstart, start); \
|
|
assert_equals_uint64 (pdur, duration); \
|
|
assert_equals_uint64 (pmstart, mstart); \
|
|
assert_equals_uint64 (pmdur, mduration); \
|
|
assert_equals_int (pprio, priority); \
|
|
assert_equals_int (pact, active); \
|
|
}
|
|
|
|
|
|
GST_START_TEST (test_layer_properties)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESTimelineLayer *layer;
|
|
GESTrack *track;
|
|
GESTrackObject *trackobject;
|
|
GESTimelineObject *object;
|
|
|
|
ges_init ();
|
|
|
|
/* Timeline and 1 Layer */
|
|
timeline = ges_timeline_new ();
|
|
layer = (GESTimelineLayer *) ges_timeline_layer_new ();
|
|
|
|
/* The default priority is 0 */
|
|
fail_unless_equals_int (ges_timeline_layer_get_priority (layer), 0);
|
|
|
|
/* Layers are initially floating, once we add them to the timeline,
|
|
* the timeline will take that reference. */
|
|
fail_unless (g_object_is_floating (layer));
|
|
fail_unless (ges_timeline_add_layer (timeline, layer));
|
|
fail_if (g_object_is_floating (layer));
|
|
|
|
track = ges_track_new (GES_TRACK_TYPE_CUSTOM, GST_CAPS_ANY);
|
|
fail_unless (track != NULL);
|
|
fail_unless (ges_timeline_add_track (timeline, track));
|
|
|
|
object =
|
|
(GESTimelineObject *) ges_custom_timeline_source_new (my_fill_track_func,
|
|
NULL);
|
|
fail_unless (object != NULL);
|
|
|
|
/* Set some properties */
|
|
g_object_set (object, "start", (guint64) 42, "duration", (guint64) 51,
|
|
"in-point", (guint64) 12, NULL);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_START (object), 42);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_DURATION (object), 51);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_INPOINT (object), 12);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_PRIORITY (object), 0);
|
|
|
|
/* Add the object to the timeline */
|
|
fail_unless (g_object_is_floating (object));
|
|
fail_unless (ges_timeline_layer_add_object (layer,
|
|
GES_TIMELINE_OBJECT (object)));
|
|
fail_if (g_object_is_floating (object));
|
|
trackobject = ges_timeline_object_find_track_object (object, track,
|
|
G_TYPE_NONE);
|
|
fail_unless (trackobject != NULL);
|
|
|
|
/* This is not a SimpleLayer, therefore the properties shouldn't have changed */
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_START (object), 42);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_DURATION (object), 51);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_INPOINT (object), 12);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_PRIORITY (object), 0);
|
|
gnl_object_check (ges_track_object_get_gnlobject (trackobject), 42, 51, 12,
|
|
51, 0, TRUE);
|
|
|
|
/* Change the priority of the layer */
|
|
g_object_set (layer, "priority", 1, NULL);
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer), 1);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_PRIORITY (object), 0);
|
|
gnl_object_check (ges_track_object_get_gnlobject (trackobject), 42, 51, 12,
|
|
51, LAYER_HEIGHT, TRUE);
|
|
|
|
/* Change it to an insanely high value */
|
|
g_object_set (layer, "priority", 31, NULL);
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer), 31);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_PRIORITY (object), 0);
|
|
gnl_object_check (ges_track_object_get_gnlobject (trackobject), 42, 51, 12,
|
|
51, LAYER_HEIGHT * 31, TRUE);
|
|
|
|
/* and back to 0 */
|
|
g_object_set (layer, "priority", 0, NULL);
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer), 0);
|
|
assert_equals_uint64 (GES_TIMELINE_OBJECT_PRIORITY (object), 0);
|
|
gnl_object_check (ges_track_object_get_gnlobject (trackobject), 42, 51, 12,
|
|
51, 0, TRUE);
|
|
|
|
g_object_unref (trackobject);
|
|
fail_unless (ges_timeline_layer_remove_object (layer, object));
|
|
fail_unless (ges_timeline_remove_track (timeline, track));
|
|
fail_unless (ges_timeline_remove_layer (timeline, layer));
|
|
g_object_unref (timeline);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_layer_priorities)
|
|
{
|
|
GESTrack *track;
|
|
GESTimeline *timeline;
|
|
GESTimelineLayer *layer1, *layer2, *layer3;
|
|
GESTrackObject *tckobj1, *tckobj2, *tckobj3;
|
|
GESTimelineObject *object1, *object2, *object3;
|
|
GstElement *gnlobj1, *gnlobj2, *gnlobj3;
|
|
guint prio1, prio2, prio3;
|
|
GList *objs, *tmp;
|
|
|
|
ges_init ();
|
|
|
|
/* Timeline and 3 Layer */
|
|
timeline = ges_timeline_new ();
|
|
layer1 = (GESTimelineLayer *) ges_timeline_layer_new ();
|
|
layer2 = (GESTimelineLayer *) ges_timeline_layer_new ();
|
|
layer3 = (GESTimelineLayer *) ges_timeline_layer_new ();
|
|
|
|
ges_timeline_layer_set_priority (layer2, 1);
|
|
ges_timeline_layer_set_priority (layer3, 2);
|
|
|
|
fail_unless (ges_timeline_add_layer (timeline, layer1));
|
|
fail_unless (ges_timeline_add_layer (timeline, layer2));
|
|
fail_unless (ges_timeline_add_layer (timeline, layer3));
|
|
fail_unless_equals_int (ges_timeline_layer_get_priority (layer1), 0);
|
|
fail_unless_equals_int (ges_timeline_layer_get_priority (layer2), 1);
|
|
fail_unless_equals_int (ges_timeline_layer_get_priority (layer3), 2);
|
|
|
|
track = ges_track_video_raw_new ();
|
|
fail_unless (track != NULL);
|
|
fail_unless (ges_timeline_add_track (timeline, track));
|
|
|
|
object1 =
|
|
GES_TIMELINE_OBJECT (ges_custom_timeline_source_new (my_fill_track_func,
|
|
NULL));
|
|
object2 =
|
|
GES_TIMELINE_OBJECT (ges_custom_timeline_source_new (my_fill_track_func,
|
|
NULL));
|
|
object3 =
|
|
GES_TIMELINE_OBJECT (ges_custom_timeline_source_new (my_fill_track_func,
|
|
NULL));
|
|
fail_unless (object1 != NULL);
|
|
fail_unless (object2 != NULL);
|
|
fail_unless (object3 != NULL);
|
|
|
|
/* Set priorities on the objects */
|
|
g_object_set (object1, "priority", 0, NULL);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object1), 0);
|
|
g_object_set (object2, "priority", 1, NULL);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object2), 1);
|
|
g_object_set (object3, "priority", LAYER_HEIGHT + 1, NULL);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object3), LAYER_HEIGHT + 1);
|
|
|
|
/* Add objects to the timeline */
|
|
fail_unless (ges_timeline_layer_add_object (layer1, object1));
|
|
tckobj1 = ges_timeline_object_find_track_object (object1, track, G_TYPE_NONE);
|
|
fail_unless (tckobj1 != NULL);
|
|
|
|
fail_unless (ges_timeline_layer_add_object (layer2, object2));
|
|
tckobj2 = ges_timeline_object_find_track_object (object2, track, G_TYPE_NONE);
|
|
fail_unless (tckobj2 != NULL);
|
|
|
|
fail_unless (ges_timeline_layer_add_object (layer3, object3));
|
|
tckobj3 = ges_timeline_object_find_track_object (object3, track, G_TYPE_NONE);
|
|
fail_unless (tckobj3 != NULL);
|
|
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object1), 0);
|
|
gnlobj1 = ges_track_object_get_gnlobject (tckobj1);
|
|
fail_unless (gnlobj1 != NULL);
|
|
g_object_get (gnlobj1, "priority", &prio1, NULL);
|
|
assert_equals_int (prio1, 0);
|
|
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object2), 1);
|
|
gnlobj2 = ges_track_object_get_gnlobject (tckobj2);
|
|
fail_unless (gnlobj2 != NULL);
|
|
g_object_get (gnlobj2, "priority", &prio2, NULL);
|
|
/* object2 is on the second layer and has a priority of 1 */
|
|
assert_equals_int (prio2, LAYER_HEIGHT + 1);
|
|
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object3), LAYER_HEIGHT - 1);
|
|
gnlobj3 = ges_track_object_get_gnlobject (tckobj3);
|
|
fail_unless (gnlobj3 != NULL);
|
|
/* object3 is on the third layer and has a priority of LAYER_HEIGHT + 1
|
|
* it priority must have the maximum priority of this layer*/
|
|
g_object_get (gnlobj3, "priority", &prio3, NULL);
|
|
assert_equals_int (prio3, LAYER_HEIGHT * 3 - 1);
|
|
|
|
/* Move layers around */
|
|
g_object_set (layer1, "priority", 2, NULL);
|
|
g_object_set (layer2, "priority", 0, NULL);
|
|
g_object_set (layer3, "priority", 1, NULL);
|
|
|
|
/* And check the new priorities */
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer1), 2);
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer2), 0);
|
|
assert_equals_int (ges_timeline_layer_get_priority (layer3), 1);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object1), 0);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object2), 1);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object3), LAYER_HEIGHT - 1);
|
|
g_object_get (gnlobj1, "priority", &prio1, NULL);
|
|
g_object_get (gnlobj2, "priority", &prio2, NULL);
|
|
g_object_get (gnlobj3, "priority", &prio3, NULL);
|
|
assert_equals_int (prio1, 2 * LAYER_HEIGHT);
|
|
assert_equals_int (prio2, 1);
|
|
assert_equals_int (prio3, LAYER_HEIGHT * 2 - 1);
|
|
|
|
/* And move objects around */
|
|
fail_unless (ges_timeline_object_move_to_layer (object2, layer1));
|
|
fail_unless (ges_timeline_object_move_to_layer (object3, layer1));
|
|
|
|
objs = ges_timeline_layer_get_objects (layer1);
|
|
assert_equals_int (g_list_length (objs), 3);
|
|
fail_unless (ges_timeline_layer_get_objects (layer2) == NULL);
|
|
fail_unless (ges_timeline_layer_get_objects (layer3) == NULL);
|
|
|
|
for (tmp = objs; tmp; tmp = g_list_next (tmp)) {
|
|
g_object_unref (tmp->data);
|
|
}
|
|
g_list_free (objs);
|
|
|
|
/* Check their priorities (layer1 priority is now 2) */
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object1), 0);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object2), 1);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object3), LAYER_HEIGHT - 1);
|
|
g_object_get (gnlobj1, "priority", &prio1, NULL);
|
|
g_object_get (gnlobj2, "priority", &prio2, NULL);
|
|
g_object_get (gnlobj3, "priority", &prio3, NULL);
|
|
assert_equals_int (prio1, 2 * LAYER_HEIGHT);
|
|
assert_equals_int (prio2, 2 * LAYER_HEIGHT + 1);
|
|
assert_equals_int (prio3, LAYER_HEIGHT * 3 - 1);
|
|
|
|
/* And change TrackObject-s priorities and check that changes are well
|
|
* refected on it containing TimelineObject */
|
|
ges_track_object_set_priority (tckobj3, LAYER_HEIGHT * 2);
|
|
g_object_get (gnlobj3, "priority", &prio3, NULL);
|
|
assert_equals_int (prio3, 2 * LAYER_HEIGHT);
|
|
assert_equals_int (GES_TIMELINE_OBJECT_PRIORITY (object3), 0);
|
|
|
|
g_object_unref (tckobj1);
|
|
g_object_unref (tckobj2);
|
|
g_object_unref (tckobj3);
|
|
g_object_unref (timeline);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
GST_START_TEST (test_layer_automatic_transition)
|
|
{
|
|
GESTimeline *timeline;
|
|
GESTimelineLayer *layer;
|
|
GESTimelineTestSource *src, *srcbis;
|
|
GList *objects = NULL, *tmp = NULL;
|
|
gboolean res = FALSE;
|
|
|
|
ges_init ();
|
|
|
|
timeline = ges_timeline_new_audio_video ();
|
|
layer = ges_timeline_layer_new ();
|
|
ges_timeline_add_layer (timeline, layer);
|
|
|
|
g_object_set (layer, "auto-transition", TRUE, NULL);
|
|
src = ges_timeline_test_source_new ();
|
|
srcbis = ges_timeline_test_source_new ();
|
|
|
|
g_object_set (srcbis, "start", (gint64) 5000, "duration", (gint64) 10000LL,
|
|
NULL);
|
|
g_object_set (src, "start", (gint64) 0, "duration", (gint64) 10000LL, NULL);
|
|
|
|
ges_timeline_layer_add_object (layer, GES_TIMELINE_OBJECT (src));
|
|
ges_timeline_layer_add_object (layer, GES_TIMELINE_OBJECT (srcbis));
|
|
|
|
objects = ges_timeline_layer_get_objects (layer);
|
|
|
|
for (tmp = objects; tmp; tmp = tmp->next) {
|
|
if (GES_IS_TIMELINE_STANDARD_TRANSITION (tmp->data)) {
|
|
res = TRUE;
|
|
}
|
|
}
|
|
|
|
fail_unless (res == TRUE);
|
|
}
|
|
|
|
GST_END_TEST;
|
|
|
|
|
|
static Suite *
|
|
ges_suite (void)
|
|
{
|
|
Suite *s = suite_create ("ges-timeline-layer");
|
|
TCase *tc_chain = tcase_create ("timeline-layer");
|
|
|
|
suite_add_tcase (s, tc_chain);
|
|
|
|
tcase_add_test (tc_chain, test_layer_properties);
|
|
tcase_add_test (tc_chain, test_layer_priorities);
|
|
tcase_add_test (tc_chain, test_layer_automatic_transition);
|
|
|
|
return s;
|
|
}
|
|
|
|
int
|
|
main (int argc, char **argv)
|
|
{
|
|
int nf;
|
|
|
|
Suite *s = ges_suite ();
|
|
SRunner *sr = srunner_create (s);
|
|
|
|
gst_check_init (&argc, &argv);
|
|
|
|
srunner_run_all (sr, CK_NORMAL);
|
|
nf = srunner_ntests_failed (sr);
|
|
srunner_free (sr);
|
|
|
|
return nf;
|
|
}
|