mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-03-30 12:49:40 +00:00
playback/player: player: Refactor video rendering API
There's a GstPlayerVideoRenderer interface now, which defines how video rendering happens in GstPlayer. Included is an implementation for the GstVideoOverlay interface, and inside the GTK example application one for gtksink/gtkglsink.
This commit is contained in:
parent
6ed6b58929
commit
a8f8d1b032
5 changed files with 249 additions and 28 deletions
|
@ -352,7 +352,7 @@ play_new (gchar ** uris, gdouble initial_volume)
|
||||||
play->cur_idx = -1;
|
play->cur_idx = -1;
|
||||||
|
|
||||||
play->player =
|
play->player =
|
||||||
gst_player_new_full (gst_player_g_main_context_signal_dispatcher_new
|
gst_player_new_full (NULL, gst_player_g_main_context_signal_dispatcher_new
|
||||||
(NULL));
|
(NULL));
|
||||||
|
|
||||||
g_signal_connect (play->player, "position-updated",
|
g_signal_connect (play->player, "position-updated",
|
||||||
|
|
|
@ -26,11 +26,11 @@ gtk-play-resources.h: resources/gresources.xml \
|
||||||
|
|
||||||
BUILT_SOURCES: gtk-play-resources.c gtk-play-resources.h
|
BUILT_SOURCES: gtk-play-resources.c gtk-play-resources.h
|
||||||
|
|
||||||
gtk_play_SOURCES = gtk-play.c gtk-play-resources.c
|
gtk_play_SOURCES = gtk-play.c gtk-play-resources.c gtk-video-renderer.c
|
||||||
|
|
||||||
LDADD = $(top_builddir)/lib/gst/player/.libs/libgstplayer-@GST_PLAYER_API_VERSION@.la \
|
LDADD = $(top_builddir)/lib/gst/player/.libs/libgstplayer-@GST_PLAYER_API_VERSION@.la \
|
||||||
$(GSTREAMER_LIBS) $(GTK_LIBS) $(GTK_X11_LIBS) $(GLIB_LIBS) $(LIBM) $(GMODULE_LIBS)
|
$(GSTREAMER_LIBS) $(GTK_LIBS) $(GTK_X11_LIBS) $(GLIB_LIBS) $(LIBM) $(GMODULE_LIBS)
|
||||||
|
|
||||||
AM_CFLAGS = -I$(top_srcdir)/lib -I$(top_builddir)/lib $(GSTREAMER_CFLAGS) $(GTK_CFLAGS) $(GTK_X11_CFLAGS) $(GLIB_CFLAGS) $(GMODULE_CFLAGS)
|
AM_CFLAGS = -I$(top_srcdir)/lib -I$(top_builddir)/lib $(GSTREAMER_CFLAGS) $(GTK_CFLAGS) $(GTK_X11_CFLAGS) $(GLIB_CFLAGS) $(GMODULE_CFLAGS)
|
||||||
|
|
||||||
noinst_HEADERS = gtk-play-resources.h
|
noinst_HEADERS = gtk-play-resources.h gtk-video-renderer.h
|
||||||
|
|
|
@ -39,6 +39,7 @@
|
||||||
#include <gtk/gtk.h>
|
#include <gtk/gtk.h>
|
||||||
|
|
||||||
#include <gst/player/player.h>
|
#include <gst/player/player.h>
|
||||||
|
#include "gtk-video-renderer.h"
|
||||||
|
|
||||||
#define APP_NAME "gtk-play"
|
#define APP_NAME "gtk-play"
|
||||||
|
|
||||||
|
@ -58,6 +59,7 @@ typedef struct
|
||||||
GtkApplicationWindow parent;
|
GtkApplicationWindow parent;
|
||||||
|
|
||||||
GstPlayer *player;
|
GstPlayer *player;
|
||||||
|
GstPlayerVideoRenderer *renderer;
|
||||||
gchar *uri;
|
gchar *uri;
|
||||||
|
|
||||||
GList *uris;
|
GList *uris;
|
||||||
|
@ -182,7 +184,8 @@ video_area_realize_cb (GtkWidget * widget, GtkPlay * play)
|
||||||
#elif defined (GDK_WINDOWING_X11)
|
#elif defined (GDK_WINDOWING_X11)
|
||||||
window_handle = GDK_WINDOW_XID (window);
|
window_handle = GDK_WINDOW_XID (window);
|
||||||
#endif
|
#endif
|
||||||
g_object_set (play->player, "window-handle", (gpointer) window_handle, NULL);
|
g_object_set (play->renderer, "window-handle", (gpointer) window_handle,
|
||||||
|
NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -1413,24 +1416,14 @@ create_ui (GtkPlay * play)
|
||||||
gtk_application_add_window (GTK_APPLICATION (g_application_get_default ()),
|
gtk_application_add_window (GTK_APPLICATION (g_application_get_default ()),
|
||||||
GTK_WINDOW (play));
|
GTK_WINDOW (play));
|
||||||
|
|
||||||
if ((gtk_sink = gst_element_factory_make ("gtkglsink", NULL))) {
|
play->renderer = gst_player_gtk_video_renderer_new ();
|
||||||
GstElement *video_sink;
|
if (play->renderer) {
|
||||||
|
play->video_area =
|
||||||
g_object_get (gtk_sink, "widget", &play->video_area, NULL);
|
gst_player_gtk_video_renderer_get_widget (GST_PLAYER_GTK_VIDEO_RENDERER
|
||||||
|
(play->renderer));
|
||||||
video_sink = gst_element_factory_make ("glsinkbin", NULL);
|
|
||||||
g_object_set (video_sink, "sink", gtk_sink, NULL);
|
|
||||||
|
|
||||||
playbin = gst_player_get_pipeline (play->player);
|
|
||||||
g_object_set (playbin, "video-sink", video_sink, NULL);
|
|
||||||
gst_object_unref (playbin);
|
|
||||||
} else if ((gtk_sink = gst_element_factory_make ("gtksink", NULL))) {
|
|
||||||
g_object_get (gtk_sink, "widget", &play->video_area, NULL);
|
|
||||||
|
|
||||||
playbin = gst_player_get_pipeline (play->player);
|
|
||||||
g_object_set (playbin, "video-sink", gtk_sink, NULL);
|
|
||||||
gst_object_unref (playbin);
|
|
||||||
} else {
|
} else {
|
||||||
|
play->renderer = gst_player_video_overlay_video_renderer_new (NULL);
|
||||||
|
|
||||||
play->video_area = gtk_drawing_area_new ();
|
play->video_area = gtk_drawing_area_new ();
|
||||||
g_signal_connect (play->video_area, "realize",
|
g_signal_connect (play->video_area, "realize",
|
||||||
G_CALLBACK (video_area_realize_cb), play);
|
G_CALLBACK (video_area_realize_cb), play);
|
||||||
|
@ -1506,10 +1499,6 @@ create_ui (GtkPlay * play)
|
||||||
if (play->fullscreen)
|
if (play->fullscreen)
|
||||||
gtk_toggle_button_set_active
|
gtk_toggle_button_set_active
|
||||||
(GTK_TOGGLE_BUTTON (play->fullscreen_button), TRUE);
|
(GTK_TOGGLE_BUTTON (play->fullscreen_button), TRUE);
|
||||||
|
|
||||||
/* enable visualization (by default laybin uses goom) */
|
|
||||||
/* if visualization is enabled then use the first element */
|
|
||||||
gst_player_set_visualization_enabled (play->player, TRUE);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
|
@ -1730,9 +1719,6 @@ gtk_play_constructor (GType type, guint n_construct_params,
|
||||||
(GtkPlay *) G_OBJECT_CLASS (gtk_play_parent_class)->constructor (type,
|
(GtkPlay *) G_OBJECT_CLASS (gtk_play_parent_class)->constructor (type,
|
||||||
n_construct_params, construct_params);
|
n_construct_params, construct_params);
|
||||||
|
|
||||||
self->player =
|
|
||||||
gst_player_new_full (gst_player_g_main_context_signal_dispatcher_new
|
|
||||||
(NULL));
|
|
||||||
self->playing = TRUE;
|
self->playing = TRUE;
|
||||||
|
|
||||||
if (self->inhibit_cookie)
|
if (self->inhibit_cookie)
|
||||||
|
@ -1744,6 +1730,10 @@ gtk_play_constructor (GType type, guint n_construct_params,
|
||||||
|
|
||||||
create_ui (self);
|
create_ui (self);
|
||||||
|
|
||||||
|
self->player =
|
||||||
|
gst_player_new_full (self->renderer,
|
||||||
|
gst_player_g_main_context_signal_dispatcher_new (NULL));
|
||||||
|
|
||||||
g_signal_connect (self->player, "position-updated",
|
g_signal_connect (self->player, "position-updated",
|
||||||
G_CALLBACK (position_updated_cb), self);
|
G_CALLBACK (position_updated_cb), self);
|
||||||
g_signal_connect (self->player, "duration-changed",
|
g_signal_connect (self->player, "duration-changed",
|
||||||
|
@ -1754,6 +1744,10 @@ gtk_play_constructor (GType type, guint n_construct_params,
|
||||||
g_signal_connect (self->player, "volume-changed",
|
g_signal_connect (self->player, "volume-changed",
|
||||||
G_CALLBACK (player_volume_changed_cb), self);
|
G_CALLBACK (player_volume_changed_cb), self);
|
||||||
|
|
||||||
|
/* enable visualization (by default playbin uses goom) */
|
||||||
|
/* if visualization is enabled then use the first element */
|
||||||
|
gst_player_set_visualization_enabled (self->player, TRUE);
|
||||||
|
|
||||||
g_signal_connect (G_OBJECT (self), "show", G_CALLBACK (show_cb), NULL);
|
g_signal_connect (G_OBJECT (self), "show", G_CALLBACK (show_cb), NULL);
|
||||||
|
|
||||||
return G_OBJECT (self);
|
return G_OBJECT (self);
|
||||||
|
|
178
playback/player/gtk/gtk-video-renderer.c
Normal file
178
playback/player/gtk/gtk-video-renderer.c
Normal file
|
@ -0,0 +1,178 @@
|
||||||
|
/* GStreamer
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 Sebastian Dröge <sebastian@centricular.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 "gtk-video-renderer.h"
|
||||||
|
|
||||||
|
struct _GstPlayerGtkVideoRenderer
|
||||||
|
{
|
||||||
|
GObject parent;
|
||||||
|
|
||||||
|
GstElement *sink;
|
||||||
|
GtkWidget *widget;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct _GstPlayerGtkVideoRendererClass
|
||||||
|
{
|
||||||
|
GObjectClass parent_class;
|
||||||
|
};
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_interface_init
|
||||||
|
(GstPlayerVideoRendererInterface * iface);
|
||||||
|
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
GTK_VIDEO_RENDERER_PROP_0,
|
||||||
|
GTK_VIDEO_RENDERER_PROP_WIDGET,
|
||||||
|
GTK_VIDEO_RENDERER_PROP_LAST
|
||||||
|
};
|
||||||
|
|
||||||
|
G_DEFINE_TYPE_WITH_CODE (GstPlayerGtkVideoRenderer,
|
||||||
|
gst_player_gtk_video_renderer, G_TYPE_OBJECT,
|
||||||
|
G_IMPLEMENT_INTERFACE (GST_TYPE_PLAYER_VIDEO_RENDERER,
|
||||||
|
gst_player_gtk_video_renderer_interface_init));
|
||||||
|
|
||||||
|
static GParamSpec
|
||||||
|
* gtk_video_renderer_param_specs[GTK_VIDEO_RENDERER_PROP_LAST] = { NULL, };
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_get_property (GObject * object,
|
||||||
|
guint prop_id, GValue * value, GParamSpec * pspec)
|
||||||
|
{
|
||||||
|
GstPlayerGtkVideoRenderer *self = GST_PLAYER_GTK_VIDEO_RENDERER (object);
|
||||||
|
|
||||||
|
switch (prop_id) {
|
||||||
|
case GTK_VIDEO_RENDERER_PROP_WIDGET:
|
||||||
|
g_value_set_object (value, self->widget);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_finalize (GObject * object)
|
||||||
|
{
|
||||||
|
GstPlayerGtkVideoRenderer *self = GST_PLAYER_GTK_VIDEO_RENDERER (object);
|
||||||
|
|
||||||
|
if (self->sink)
|
||||||
|
gst_object_unref (self->sink);
|
||||||
|
if (self->widget)
|
||||||
|
g_object_unref (self->widget);
|
||||||
|
|
||||||
|
G_OBJECT_CLASS
|
||||||
|
(gst_player_gtk_video_renderer_parent_class)->finalize (object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_class_init
|
||||||
|
(GstPlayerGtkVideoRendererClass * klass)
|
||||||
|
{
|
||||||
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
||||||
|
|
||||||
|
gobject_class->get_property = gst_player_gtk_video_renderer_get_property;
|
||||||
|
gobject_class->finalize = gst_player_gtk_video_renderer_finalize;
|
||||||
|
|
||||||
|
gtk_video_renderer_param_specs
|
||||||
|
[GTK_VIDEO_RENDERER_PROP_WIDGET] =
|
||||||
|
g_param_spec_object ("widget", "Widget",
|
||||||
|
"Widget to render the video into", GTK_TYPE_WIDGET,
|
||||||
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS);
|
||||||
|
|
||||||
|
g_object_class_install_properties (gobject_class,
|
||||||
|
GTK_VIDEO_RENDERER_PROP_LAST, gtk_video_renderer_param_specs);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_init (GstPlayerGtkVideoRenderer * self)
|
||||||
|
{
|
||||||
|
GstElement *gtk_sink = gst_element_factory_make ("gtkglsink", NULL);
|
||||||
|
|
||||||
|
if (gtk_sink) {
|
||||||
|
GstElement *sink = sink = gst_element_factory_make ("glsinkbin", NULL);
|
||||||
|
g_object_set (sink, "sink", gtk_sink, NULL);
|
||||||
|
|
||||||
|
self->sink = sink;
|
||||||
|
} else {
|
||||||
|
gtk_sink = gst_element_factory_make ("gtksink", NULL);
|
||||||
|
|
||||||
|
self->sink = gtk_sink;
|
||||||
|
}
|
||||||
|
|
||||||
|
g_assert (self->sink != NULL);
|
||||||
|
|
||||||
|
g_object_get (gtk_sink, "widget", &self->widget, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static GstElement *gst_player_gtk_video_renderer_create_video_sink
|
||||||
|
(GstPlayerVideoRenderer * iface, GstPlayer * player)
|
||||||
|
{
|
||||||
|
GstElement *gtk;
|
||||||
|
GstPlayerGtkVideoRenderer *self = GST_PLAYER_GTK_VIDEO_RENDERER (iface);
|
||||||
|
|
||||||
|
return gst_object_ref (self->sink);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
gst_player_gtk_video_renderer_interface_init
|
||||||
|
(GstPlayerVideoRendererInterface * iface)
|
||||||
|
{
|
||||||
|
iface->create_video_sink = gst_player_gtk_video_renderer_create_video_sink;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_player_gtk_video_renderer_new:
|
||||||
|
*
|
||||||
|
* Returns: (transfer full):
|
||||||
|
*/
|
||||||
|
GstPlayerVideoRenderer *
|
||||||
|
gst_player_gtk_video_renderer_new (void)
|
||||||
|
{
|
||||||
|
GstElementFactory *factory;
|
||||||
|
|
||||||
|
factory = gst_element_factory_find ("gtkglsink");
|
||||||
|
if (!factory)
|
||||||
|
factory = gst_element_factory_find ("gtksink");
|
||||||
|
if (!factory)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
gst_object_unref (factory);
|
||||||
|
|
||||||
|
return g_object_new (GST_TYPE_PLAYER_GTK_VIDEO_RENDERER, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gst_player_gtk_video_renderer_get_widget:
|
||||||
|
* @self: #GstPlayerVideoRenderer instance
|
||||||
|
*
|
||||||
|
* Returns: (transfer full): The GtkWidget
|
||||||
|
*/
|
||||||
|
GtkWidget *gst_player_gtk_video_renderer_get_widget
|
||||||
|
(GstPlayerGtkVideoRenderer * self)
|
||||||
|
{
|
||||||
|
GtkWidget *widget;
|
||||||
|
|
||||||
|
g_return_if_fail (GST_IS_PLAYER_GTK_VIDEO_RENDERER (self));
|
||||||
|
|
||||||
|
g_object_get (self, "widget", &widget, NULL);
|
||||||
|
|
||||||
|
return widget;
|
||||||
|
}
|
49
playback/player/gtk/gtk-video-renderer.h
Normal file
49
playback/player/gtk/gtk-video-renderer.h
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
/* GStreamer
|
||||||
|
*
|
||||||
|
* Copyright (C) 2015 Sebastian Dröge <sebastian@centricular.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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __GTK_VIDEO_RENDERER_H__
|
||||||
|
#define __GTK_VIDEO_RENDERER_H__
|
||||||
|
|
||||||
|
#include <gst/player/player.h>
|
||||||
|
#include <gtk/gtk.h>
|
||||||
|
|
||||||
|
G_BEGIN_DECLS
|
||||||
|
|
||||||
|
typedef struct _GstPlayerGtkVideoRenderer
|
||||||
|
GstPlayerGtkVideoRenderer;
|
||||||
|
typedef struct _GstPlayerGtkVideoRendererClass
|
||||||
|
GstPlayerGtkVideoRendererClass;
|
||||||
|
|
||||||
|
#define GST_TYPE_PLAYER_GTK_VIDEO_RENDERER (gst_player_gtk_video_renderer_get_type ())
|
||||||
|
#define GST_IS_PLAYER_GTK_VIDEO_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), GST_TYPE_PLAYER_GTK_VIDEO_RENDERER))
|
||||||
|
#define GST_IS_PLAYER_GTK_VIDEO_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), GST_TYPE_PLAYER_GTK_VIDEO_RENDERER))
|
||||||
|
#define GST_PLAYER_GTK_VIDEO_RENDERER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), GST_TYPE_PLAYER_GTK_VIDEO_RENDERER, GstPlayerGtkVideoRendererClass))
|
||||||
|
#define GST_PLAYER_GTK_VIDEO_RENDERER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), GST_TYPE_PLAYER_GTK_VIDEO_RENDERER, GstPlayerGtkVideoRenderer))
|
||||||
|
#define GST_PLAYER_GTK_VIDEO_RENDERER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), GST_TYPE_PLAYER_GTK_VIDEO_RENDERER, GstPlayerGtkVideoRendererClass))
|
||||||
|
#define GST_PLAYER_GTK_VIDEO_RENDERER_CAST(obj) ((GstPlayerGtkVideoRenderer*)(obj))
|
||||||
|
|
||||||
|
GType gst_player_gtk_video_renderer_get_type (void);
|
||||||
|
|
||||||
|
GstPlayerVideoRenderer * gst_player_gtk_video_renderer_new (void);
|
||||||
|
GtkWidget * gst_player_gtk_video_renderer_get_widget (GstPlayerGtkVideoRenderer * self);
|
||||||
|
|
||||||
|
G_END_DECLS
|
||||||
|
|
||||||
|
#endif /* __GTK_VIDEO_RENDERER_H__ */
|
Loading…
Reference in a new issue