tags:

views:

2849

answers:

5

I want to use Qt to create a simple GUI application that can play a local video file. I could use Phonon which does all the work behind the scenes, but I need to have a little more control. I have already succeeded in implementing an GStreamer pipeline using the decodebin and autovideosink elements. Now I want to use a Qt widget to channel the output to.

Has anyone ever succeeded in doing this? (I suppose so since there are Qt-based video players that build upon GStreamer.) Can someone point me in the right direction on how to do it?

Note: This question is similar to my previous posted question on how to connect Qt with an incoming RTP stream. This seemed to be quite challenging. This question will be easier to answer I think.

Edit:
Patrice's suggestion to use libVLC is very helpful already. Here's a somewhat cleaner version of the code found on VLC's website: Sample for Qt + libVLC.

However, my original question remains: How do I connect GStreamer to a Qt widget?

+2  A: 

VLC version is a QT-based video player (since version 0.99). It allows too to stream or read a stream: You can find all information you need here: http://wiki.videolan.org/Developers_Corner. You only have create an instance of the player and associate it to a widget. Then you have full control on the player.

I have already tested it (on Linux and Windows) playing local music and video files and it works fine.

Give it a try and see by yourself.

Hope that helps.

Edit: It seems if you want to use VLC, you need to write or find (I do not know if one exists) a GStreamer codec as explain on the videolan wiki. I think I would do that.

Patrice Bernassola
I'm a big fan of VLC player, but I didn't know you can connect it to a Qt widget. Are you talking about libVLC, or communicating with VLC using DBus?
StackedCrooked
I'm talking about libVLC. You have an example here: http://wiki.videolan.org/LibVLC_SampleCode_Qt
Patrice Bernassola
@Patrice: thanks, see my edit.
StackedCrooked
+5  A: 

Since Phonon is based on gstreamer, the place to look for details is the Phonon source tree (available here: http://gitorious.org/phonon/import/trees/master). For a video player you are most likely going to need a video display widget, such as the gstreamer/videowidget.h (cpp) that in turn used the X11 renderer (gstreamer/x11renderer.h, cpp). The sink used is the xvimagesink, falling back onto the ximagesink if the first cannot be created.

The basic trick is to overlay the VideoWidget with the video output. The X11 handle needed to do this is retrieved using the QWidget::winId method, which is platform specific (as are the sinks, so no biggie).

Also, if overlay is unavailable, a QWidgetVideoSink is used, which converts the video frames into individual frames for the WidgetRenderer class. This class, in turn, makes the current frame available as a QImage object, ready for any type of processing.

So to answer your question - use either overlays (as X11Renderer) or extract individual QImages from the video stream (as QWidgetVideoSink).

e8johan
+2  A: 

To connect Gstreamer with your QWidget, you need to get the window handle using QWidget::winId() and you pass it to gst_x_overlay_set_xwindow_id();

Rough sample code:

    sink = gst_element_factory_make("xvimagesink", "sink");
    gst_element_set_state(sink, GST_STATE_READY);

    QApplication::syncX();
    gst_x_overlay_set_xwindow_id(GST_X_OVERLAY(sink), widget->winId());

Also, you will want your widget to be backed by a native window which is achieved by setting the Qt::AA_NativeWindows attribute at the application level or the Qt::WA_NativeWindow attribute at the widget level.

Gregory Pakosz
This actually does not work for me. It works if I call, QApplication::syncX() AFTER the call to winId. In recent Qt version, calling winId automatically transform your widget to a native one. See http://stackoverflow.com/questions/3573483/targeting-qt-child-widget-with-gstreamer for the ordering thing
shodanex
A: 

Also, if overlay is unavailable, a QWidgetVideoSink is used, which converts the video frames into individual frames for the WidgetRenderer class. This class, in turn, makes the current frame available as a QImage object, ready for any type of processing.

Dear e8johan:

I would like to get the individual images for processing as you suggest. But how? I have successfully written a program to play the video stream, but how can I gain access to the individual frames? I wanted to do motion detection by differencing successive frames.

Many thanks for your advice.

wpoet.

wpoet
A: 

With respect to the answer marked as the best answer, could you please suggest as to how exactly can we force a Gstreamer video played on its window to be played on the Qt widget? Am eagerly waiting for your response ThankQ in advance

raggio
StackedCrooked