blob: 2f9bf8954ade6b98fb72aca967affd75921389ad [file] [log] [blame]
/*
* Copyright (C) 2014-2015 Collabora Ltd.
* @author George Kiagiadakis <george.kiagiadakis@collabora.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., 51 Franklin St, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include <gst/gst.h>
#include <gtk/gtk.h>
#include <gdk/gdk.h>
#ifdef GDK_WINDOWING_WAYLAND
#include <gdk/gdkwayland.h>
#else
#error "Wayland is not supported in GTK+"
#endif
#include <gst/video/videooverlay.h>
#include <gst/wayland/wayland.h>
static gboolean live = FALSE;
static GOptionEntry entries[] = {
{"live", 'l', 0, G_OPTION_ARG_NONE, &live, "Use a live source", NULL},
{NULL}
};
typedef struct
{
GtkWidget *app_widget;
GtkWidget *video_widget;
GstElement *pipeline;
GstVideoOverlay *overlay;
gchar **argv;
gint current_uri; /* index for argv */
} DemoApp;
static void
on_about_to_finish (GstElement * playbin, DemoApp * d)
{
if (d->argv[++d->current_uri] == NULL)
d->current_uri = 1;
g_print ("Now playing %s\n", d->argv[d->current_uri]);
g_object_set (playbin, "uri", d->argv[d->current_uri], NULL);
}
static void
error_cb (GstBus * bus, GstMessage * msg, gpointer user_data)
{
DemoApp *d = user_data;
gchar *debug = NULL;
GError *err = NULL;
gst_message_parse_error (msg, &err, &debug);
g_print ("Error: %s\n", err->message);
g_error_free (err);
if (debug) {
g_print ("Debug details: %s\n", debug);
g_free (debug);
}
gst_element_set_state (d->pipeline, GST_STATE_NULL);
}
static GstBusSyncReply
bus_sync_handler (GstBus * bus, GstMessage * message, gpointer user_data)
{
DemoApp *d = user_data;
if (gst_is_wayland_display_handle_need_context_message (message)) {
GstContext *context;
GdkDisplay *display;
struct wl_display *display_handle;
display = gtk_widget_get_display (d->video_widget);
display_handle = gdk_wayland_display_get_wl_display (display);
context = gst_wayland_display_handle_context_new (display_handle);
gst_element_set_context (GST_ELEMENT (GST_MESSAGE_SRC (message)), context);
goto drop;
} else if (gst_is_video_overlay_prepare_window_handle_message (message)) {
GtkAllocation allocation;
GdkWindow *window;
struct wl_surface *window_handle;
/* GST_MESSAGE_SRC (message) will be the overlay object that we have to
* use. This may be waylandsink, but it may also be playbin. In the latter
* case, we must make sure to use playbin instead of waylandsink, because
* playbin resets the window handle and render_rectangle after restarting
* playback and the actual window size is lost */
d->overlay = GST_VIDEO_OVERLAY (GST_MESSAGE_SRC (message));
gtk_widget_get_allocation (d->video_widget, &allocation);
window = gtk_widget_get_window (d->video_widget);
window_handle = gdk_wayland_window_get_wl_surface (window);
g_print ("setting window handle and size (%d x %d)\n",
allocation.width, allocation.height);
gst_video_overlay_set_window_handle (d->overlay, (guintptr) window_handle);
gst_video_overlay_set_render_rectangle (d->overlay, allocation.x,
allocation.y, allocation.width, allocation.height);
goto drop;
}
return GST_BUS_PASS;
drop:
gst_message_unref (message);
return GST_BUS_DROP;
}
/* We use the "draw" callback to change the size of the sink
* because the "configure-event" is only sent to top-level widgets. */
static gboolean
video_widget_draw_cb (GtkWidget * widget, cairo_t * cr, gpointer user_data)
{
DemoApp *d = user_data;
GtkAllocation allocation;
gtk_widget_get_allocation (widget, &allocation);
g_print ("draw_cb x %d, y %d, w %d, h %d\n",
allocation.x, allocation.y, allocation.width, allocation.height);
if (d->overlay) {
gst_video_overlay_set_render_rectangle (d->overlay, allocation.x,
allocation.y, allocation.width, allocation.height);
}
/* There is no need to call gst_video_overlay_expose().
* The wayland compositor can always re-draw the window
* based on its last contents if necessary */
return FALSE;
}
static void
playing_clicked_cb (GtkButton * button, DemoApp * d)
{
gst_element_set_state (d->pipeline, GST_STATE_PLAYING);
}
static void
paused_clicked_cb (GtkButton * button, DemoApp * d)
{
gst_element_set_state (d->pipeline, GST_STATE_PAUSED);
}
static void
ready_clicked_cb (GtkButton * button, DemoApp * d)
{
gst_element_set_state (d->pipeline, GST_STATE_READY);
}
static void
null_clicked_cb (GtkButton * button, DemoApp * d)
{
gst_element_set_state (d->pipeline, GST_STATE_NULL);
}
static void
build_window (DemoApp * d)
{
GtkBuilder *builder;
GtkWidget *button;
GError *error = NULL;
builder = gtk_builder_new ();
if (!gtk_builder_add_from_file (builder, "window.ui", &error)) {
g_error ("Failed to load window.ui: %s", error->message);
g_error_free (error);
goto exit;
}
d->app_widget = GTK_WIDGET (gtk_builder_get_object (builder, "window"));
g_object_ref (d->app_widget);
g_signal_connect (d->app_widget, "destroy", G_CALLBACK (gtk_main_quit), NULL);
d->video_widget = GTK_WIDGET (gtk_builder_get_object (builder, "videoarea"));
g_signal_connect (d->video_widget, "draw",
G_CALLBACK (video_widget_draw_cb), d);
button = GTK_WIDGET (gtk_builder_get_object (builder, "button_playing"));
g_signal_connect (button, "clicked", G_CALLBACK (playing_clicked_cb), d);
button = GTK_WIDGET (gtk_builder_get_object (builder, "button_paused"));
g_signal_connect (button, "clicked", G_CALLBACK (paused_clicked_cb), d);
button = GTK_WIDGET (gtk_builder_get_object (builder, "button_ready"));
g_signal_connect (button, "clicked", G_CALLBACK (ready_clicked_cb), d);
button = GTK_WIDGET (gtk_builder_get_object (builder, "button_null"));
g_signal_connect (button, "clicked", G_CALLBACK (null_clicked_cb), d);
gtk_widget_show_all (d->app_widget);
exit:
g_object_unref (builder);
}
int
main (int argc, char **argv)
{
DemoApp *d;
GOptionContext *context;
GstBus *bus;
GError *error = NULL;
gtk_init (&argc, &argv);
gst_init (&argc, &argv);
context = g_option_context_new ("- waylandsink gtk demo");
g_option_context_add_main_entries (context, entries, NULL);
if (!g_option_context_parse (context, &argc, &argv, &error)) {
g_printerr ("option parsing failed: %s\n", error->message);
return 1;
}
d = g_slice_new0 (DemoApp);
build_window (d);
if (argc > 1) {
d->argv = argv;
d->current_uri = 1;
d->pipeline = gst_parse_launch ("playbin video-sink=waylandsink", NULL);
g_object_set (d->pipeline, "uri", argv[d->current_uri], NULL);
/* enable looping */
g_signal_connect (d->pipeline, "about-to-finish",
G_CALLBACK (on_about_to_finish), d);
} else {
if (live) {
d->pipeline = gst_parse_launch ("videotestsrc pattern=18 "
"background-color=0x000062FF is-live=true ! waylandsink", NULL);
} else {
d->pipeline = gst_parse_launch ("videotestsrc pattern=18 "
"background-color=0x000062FF ! waylandsink", NULL);
}
}
bus = gst_pipeline_get_bus (GST_PIPELINE (d->pipeline));
gst_bus_add_signal_watch (bus);
g_signal_connect (bus, "message::error", G_CALLBACK (error_cb), d);
gst_bus_set_sync_handler (bus, bus_sync_handler, d, NULL);
gst_object_unref (bus);
gst_element_set_state (d->pipeline, GST_STATE_PLAYING);
gtk_main ();
gst_element_set_state (d->pipeline, GST_STATE_NULL);
gst_object_unref (d->pipeline);
g_object_unref (d->app_widget);
g_slice_free (DemoApp, d);
return 0;
}