2012-07-03 15:42:00 +00:00
|
|
|
#include <string.h>
|
2013-10-31 09:01:26 +00:00
|
|
|
#include <stdio.h>
|
2012-07-03 15:42:00 +00:00
|
|
|
#include <gst/gst.h>
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2022-12-13 17:42:11 +00:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include <TargetConditionals.h>
|
|
|
|
#endif
|
|
|
|
|
2019-02-07 19:32:58 +00:00
|
|
|
typedef struct _CustomData
|
|
|
|
{
|
2012-07-03 15:42:00 +00:00
|
|
|
GstElement *pipeline;
|
2012-07-03 16:31:49 +00:00
|
|
|
GstElement *video_sink;
|
2012-07-03 15:42:00 +00:00
|
|
|
GMainLoop *loop;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2019-02-07 19:32:58 +00:00
|
|
|
gboolean playing; /* Playing or Paused */
|
|
|
|
gdouble rate; /* Current playback rate (can be negative) */
|
2012-07-03 15:42:00 +00:00
|
|
|
} CustomData;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
/* Send seek event to change rate */
|
2019-02-07 19:32:58 +00:00
|
|
|
static void
|
|
|
|
send_seek_event (CustomData * data)
|
|
|
|
{
|
2012-07-03 16:31:49 +00:00
|
|
|
gint64 position;
|
|
|
|
GstEvent *seek_event;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
/* Obtain the current position, needed for the seek event */
|
2013-10-31 09:01:26 +00:00
|
|
|
if (!gst_element_query_position (data->pipeline, GST_FORMAT_TIME, &position)) {
|
2012-07-03 16:31:49 +00:00
|
|
|
g_printerr ("Unable to retrieve current position.\n");
|
|
|
|
return;
|
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
/* Create the seek event */
|
|
|
|
if (data->rate > 0) {
|
2019-02-07 19:32:58 +00:00
|
|
|
seek_event =
|
|
|
|
gst_event_new_seek (data->rate, GST_FORMAT_TIME,
|
|
|
|
GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_ACCURATE, GST_SEEK_TYPE_SET,
|
2019-08-14 21:12:34 +00:00
|
|
|
position, GST_SEEK_TYPE_END, 0);
|
2012-07-03 16:31:49 +00:00
|
|
|
} else {
|
2019-02-07 19:32:58 +00:00
|
|
|
seek_event =
|
|
|
|
gst_event_new_seek (data->rate, GST_FORMAT_TIME,
|
|
|
|
GST_SEEK_FLAG_FLUSH | GST_SEEK_FLAG_ACCURATE, GST_SEEK_TYPE_SET, 0,
|
|
|
|
GST_SEEK_TYPE_SET, position);
|
2012-07-03 16:31:49 +00:00
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
if (data->video_sink == NULL) {
|
|
|
|
/* If we have not done so, obtain the sink through which we will send the seek events */
|
2012-07-05 11:06:18 +00:00
|
|
|
g_object_get (data->pipeline, "video-sink", &data->video_sink, NULL);
|
2012-07-03 16:31:49 +00:00
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
/* Send the event */
|
|
|
|
gst_element_send_event (data->video_sink, seek_event);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 16:31:49 +00:00
|
|
|
g_print ("Current rate: %g\n", data->rate);
|
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Process keyboard input */
|
2019-02-07 19:32:58 +00:00
|
|
|
static gboolean
|
|
|
|
handle_keyboard (GIOChannel * source, GIOCondition cond, CustomData * data)
|
|
|
|
{
|
2012-07-03 15:42:00 +00:00
|
|
|
gchar *str = NULL;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2019-02-07 19:32:58 +00:00
|
|
|
if (g_io_channel_read_line (source, &str, NULL, NULL,
|
|
|
|
NULL) != G_IO_STATUS_NORMAL) {
|
2012-07-03 15:42:00 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
switch (g_ascii_tolower (str[0])) {
|
2019-02-07 19:32:58 +00:00
|
|
|
case 'p':
|
|
|
|
data->playing = !data->playing;
|
|
|
|
gst_element_set_state (data->pipeline,
|
|
|
|
data->playing ? GST_STATE_PLAYING : GST_STATE_PAUSED);
|
|
|
|
g_print ("Setting state to %s\n", data->playing ? "PLAYING" : "PAUSE");
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
if (g_ascii_isupper (str[0])) {
|
|
|
|
data->rate *= 2.0;
|
|
|
|
} else {
|
|
|
|
data->rate /= 2.0;
|
|
|
|
}
|
|
|
|
send_seek_event (data);
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
data->rate *= -1.0;
|
|
|
|
send_seek_event (data);
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
if (data->video_sink == NULL) {
|
|
|
|
/* If we have not done so, obtain the sink through which we will send the step events */
|
|
|
|
g_object_get (data->pipeline, "video-sink", &data->video_sink, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
gst_element_send_event (data->video_sink,
|
|
|
|
gst_event_new_step (GST_FORMAT_BUFFERS, 1, ABS (data->rate), TRUE,
|
|
|
|
FALSE));
|
|
|
|
g_print ("Stepping one frame\n");
|
|
|
|
break;
|
|
|
|
case 'q':
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
2012-07-03 15:42:00 +00:00
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
g_free (str);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2019-02-07 19:32:58 +00:00
|
|
|
int
|
2022-12-13 17:42:11 +00:00
|
|
|
tutorial_main (int argc, char *argv[])
|
2019-02-07 19:32:58 +00:00
|
|
|
{
|
2012-07-03 15:42:00 +00:00
|
|
|
CustomData data;
|
|
|
|
GstStateChangeReturn ret;
|
|
|
|
GIOChannel *io_stdin;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Initialize GStreamer */
|
|
|
|
gst_init (&argc, &argv);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Initialize our data structure */
|
|
|
|
memset (&data, 0, sizeof (data));
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Print usage map */
|
2019-02-07 19:32:58 +00:00
|
|
|
g_print ("USAGE: Choose one of the following options, then press enter:\n"
|
|
|
|
" 'P' to toggle between PAUSE and PLAY\n"
|
|
|
|
" 'S' to increase playback speed, 's' to decrease playback speed\n"
|
|
|
|
" 'D' to toggle playback direction\n"
|
|
|
|
" 'N' to move to next frame (in the current direction, better in PAUSE)\n"
|
|
|
|
" 'Q' to quit\n");
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Build the pipeline */
|
2019-02-07 19:32:58 +00:00
|
|
|
data.pipeline =
|
|
|
|
gst_parse_launch
|
|
|
|
("playbin uri=https://www.freedesktop.org/software/gstreamer-sdk/data/media/sintel_trailer-480p.webm",
|
|
|
|
NULL);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Add a keyboard watch so we get notified of keystrokes */
|
2013-10-31 09:01:26 +00:00
|
|
|
#ifdef G_OS_WIN32
|
2012-07-03 15:42:00 +00:00
|
|
|
io_stdin = g_io_channel_win32_new_fd (fileno (stdin));
|
|
|
|
#else
|
|
|
|
io_stdin = g_io_channel_unix_new (fileno (stdin));
|
|
|
|
#endif
|
2019-02-07 19:32:58 +00:00
|
|
|
g_io_add_watch (io_stdin, G_IO_IN, (GIOFunc) handle_keyboard, &data);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Start playing */
|
|
|
|
ret = gst_element_set_state (data.pipeline, GST_STATE_PLAYING);
|
|
|
|
if (ret == GST_STATE_CHANGE_FAILURE) {
|
|
|
|
g_printerr ("Unable to set the pipeline to the playing state.\n");
|
|
|
|
gst_object_unref (data.pipeline);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
data.playing = TRUE;
|
|
|
|
data.rate = 1.0;
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Create a GLib Main Loop and set it to run */
|
|
|
|
data.loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
g_main_loop_run (data.loop);
|
2016-11-05 08:18:49 +00:00
|
|
|
|
2012-07-03 15:42:00 +00:00
|
|
|
/* Free resources */
|
|
|
|
g_main_loop_unref (data.loop);
|
|
|
|
g_io_channel_unref (io_stdin);
|
|
|
|
gst_element_set_state (data.pipeline, GST_STATE_NULL);
|
2012-07-03 16:31:49 +00:00
|
|
|
if (data.video_sink != NULL)
|
|
|
|
gst_object_unref (data.video_sink);
|
2012-07-03 15:42:00 +00:00
|
|
|
gst_object_unref (data.pipeline);
|
|
|
|
return 0;
|
|
|
|
}
|
2022-12-13 17:42:11 +00:00
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
{
|
|
|
|
#if defined(__APPLE__) && TARGET_OS_MAC && !TARGET_OS_IPHONE
|
|
|
|
return gst_macos_main (tutorial_main, argc, argv, NULL);
|
|
|
|
#else
|
|
|
|
return tutorial_main (argc, argv);
|
|
|
|
#endif
|
|
|
|
}
|