1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        unix/mediactrl.cpp 
   3 // Purpose:     Built-in Media Backends for Unix 
   4 // Author:      Ryan Norton <wxprojects@comcast.net> 
   8 // Copyright:   (c) 2004-2005 Ryan Norton 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 //=========================================================================== 
  14 //=========================================================================== 
  16 //--------------------------------------------------------------------------- 
  17 // Pre-compiled header stuff 
  18 //--------------------------------------------------------------------------- 
  20 #if defined(__GNUG__) && !defined(NO_GCC_PRAGMA) 
  21 #pragma implementation "mediactrl.h" 
  24 // For compilers that support precompilation, includes "wx.h". 
  25 #include "wx/wxprec.h" 
  31 //--------------------------------------------------------------------------- 
  33 //--------------------------------------------------------------------------- 
  34 #include "wx/mediactrl.h" 
  36 //--------------------------------------------------------------------------- 
  38 //--------------------------------------------------------------------------- 
  41 //=========================================================================== 
  42 //  BACKEND DECLARATIONS 
  43 //=========================================================================== 
  45 //--------------------------------------------------------------------------- 
  47 //  wxGStreamerMediaBackend 
  50 //TODO:  This is really not the best way to play-stop - 
  51 //TODO:  it should just have one playbin and stick with it the whole 
  52 //TODO:  instance of wxGStreamerMediaBackend - but stopping appears 
  53 //TODO:  to invalidate the playbin object... 
  56 //--------------------------------------------------------------------------- 
  59 //--------------------------------------------------------------------------- 
  61 //--------------------------------------------------------------------------- 
  63 #include <gst/xoverlay/xoverlay.h> 
  65 #include <string.h> //strstr 
  70     //for <gdk/gdkx.h>/related for GDK_WINDOW_XWINDOW 
  71 #    include "wx/gtk/win_gtk.h" 
  72 #    include <gtk/gtksignal.h> 
  73 #    if wxUSE_DYNLIB_CLASS 
  74 #        include "wx/dynlib.h" 
  76 //#    include <gst/gconf/gconf.h> //gstreamer gnome interface - needs deps 
  80 class WXDLLIMPEXP_MEDIA wxGStreamerMediaBackend 
: public wxMediaBackend
 
  84     wxGStreamerMediaBackend(); 
  85     ~wxGStreamerMediaBackend(); 
  87     virtual bool CreateControl(wxControl
* ctrl
, wxWindow
* parent
, 
  92                                      const wxValidator
& validator
, 
  93                                      const wxString
& name
); 
  99     virtual bool Load(const wxString
& fileName
); 
 100     virtual bool Load(const wxURI
& location
); 
 102     virtual wxMediaState 
GetState(); 
 104     virtual bool SetPosition(wxLongLong where
); 
 105     virtual wxLongLong 
GetPosition(); 
 106     virtual wxLongLong 
GetDuration(); 
 108     virtual void Move(int x
, int y
, int w
, int h
); 
 109     wxSize 
GetVideoSize() const; 
 111     virtual double GetPlaybackRate(); 
 112     virtual bool SetPlaybackRate(double dRate
); 
 116     static void OnFinish(GstElement 
*play
,  gpointer data
); 
 117     static void OnError (GstElement 
*play
,  GstElement 
*src
, 
 118                          GError     
*err
,   gchar      
*debug
, 
 120     static void OnVideoCapsReady(GstPad
* pad
,  GParamSpec
* pspec
, gpointer data
); 
 122     static bool TransCapsToVideoSize(wxGStreamerMediaBackend
* be
, GstPad
* caps
); 
 123     void PostRecalcSize(); 
 126     static gint 
OnGTKRealize(GtkWidget
* theWidget
, wxGStreamerMediaBackend
* be
); 
 129     GstElement
* m_player
;       //GStreamer media element 
 134     wxLongLong m_nPausedPos
; 
 136     DECLARE_DYNAMIC_CLASS(wxGStreamerMediaBackend
); 
 140 //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 
 142 // wxGStreamerMediaBackend 
 144 //+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ 
 146 IMPLEMENT_DYNAMIC_CLASS(wxGStreamerMediaBackend
, wxMediaBackend
); 
 148 //--------------------------------------------------------------------------- 
 149 // wxGStreamerMediaBackend Constructor 
 151 // Sets m_player to NULL signifying we havn't loaded anything yet 
 152 //--------------------------------------------------------------------------- 
 153 wxGStreamerMediaBackend::wxGStreamerMediaBackend() : m_player(NULL
), m_videoSize(0,0) 
 157 //--------------------------------------------------------------------------- 
 158 // wxGStreamerMediaBackend Destructor 
 160 // Stops/cleans up memory   
 161 //--------------------------------------------------------------------------- 
 162 wxGStreamerMediaBackend::~wxGStreamerMediaBackend() 
 167 //--------------------------------------------------------------------------- 
 168 // wxGStreamerMediaBackend::OnGTKRealize 
 170 // If the window wasn't realized when Load was called, this is the 
 171 // callback for when it is. 
 173 // 1) Installs GTK idle handler if it doesn't exist 
 174 // 2) Yeilds to avoid an X11 bug (?) 
 175 // 3) Tells GStreamer to play the video in our control 
 176 //--------------------------------------------------------------------------- 
 182 #   define DEBUG_MAIN_THREAD if (wxThread::IsMain() && g_mainThreadLocked) printf("gui reentrance"); 
 184 #   define DEBUG_MAIN_THREAD 
 187 #define DEBUG_MAIN_THREAD 
 190 extern void wxapp_install_idle_handler(); 
 191 extern bool g_isIdle
; 
 192 extern bool g_mainThreadLocked
; 
 194 gint 
wxGStreamerMediaBackend::OnGTKRealize(GtkWidget
* theWidget
,  
 195                                            wxGStreamerMediaBackend
* be
) 
 200         wxapp_install_idle_handler(); 
 202     wxYield();    //FIXME: X Server gets an error if I don't do this or a messagebox beforehand?!?!?? 
 204     GdkWindow 
*window 
= GTK_PIZZA(theWidget
)->bin_window
; 
 207     GstElement
* videosink
; 
 208     g_object_get (G_OBJECT (be
->m_player
), "video-sink", &videosink
, NULL
); 
 210     GstElement
* overlay 
= gst_bin_get_by_interface (GST_BIN (videosink
), 
 212     gst_x_overlay_set_xwindow_id( GST_X_OVERLAY(overlay
), 
 213                                 GDK_WINDOW_XWINDOW( window 
) 
 222 //--------------------------------------------------------------------------- 
 223 // wxGStreamerMediaBackend::Cleanup 
 225 // Frees the gstreamer interfaces if there were any created 
 226 //--------------------------------------------------------------------------- 
 227 void wxGStreamerMediaBackend::Cleanup() 
 229     if(m_player 
&& GST_IS_OBJECT(m_player
)) 
 231         gst_element_set_state (m_player
, GST_STATE_NULL
); 
 232         gst_object_unref (GST_OBJECT (m_player
)); 
 236 //--------------------------------------------------------------------------- 
 237 // wxGStreamerMediaBackend::CreateControl 
 239 // Initializes GStreamer and creates the wx side of our media control 
 240 //--------------------------------------------------------------------------- 
 241 bool wxGStreamerMediaBackend::CreateControl(wxControl
* ctrl
, wxWindow
* parent
, 
 246                                 const wxValidator
& validator
, 
 247                                 const wxString
& name
) 
 250     gst_init(NULL
, NULL
); 
 254     return m_ctrl
->wxControl::Create(parent
, id
, pos
, size
, 
 255                             style
,  //remove borders??? 
 259 //--------------------------------------------------------------------------- 
 260 // wxGStreamerMediaBackend::TransCapsToVideoSize 
 262 // Gets the size of our video (in wxSize) from a GstPad 
 263 //--------------------------------------------------------------------------- 
 264 bool wxGStreamerMediaBackend::TransCapsToVideoSize(wxGStreamerMediaBackend
* be
, GstPad
* pad
) 
 266     const GstCaps
* caps 
= GST_PAD_CAPS (pad
); 
 270         const GstStructure 
*s
; 
 271         s 
= gst_caps_get_structure (caps
, 0); 
 274         gst_structure_get_int (s
, "width", &be
->m_videoSize
.x
); 
 275         gst_structure_get_int (s
, "height", &be
->m_videoSize
.y
); 
 278         par 
= gst_structure_get_value (s
, "pixel-aspect-ratio"); 
 282             int num 
= gst_value_get_fraction_numerator (par
), 
 283                 den 
= gst_value_get_fraction_denominator (par
); 
 285             //TODO: maybe better fraction normalization... 
 287                 be
->m_videoSize
.x 
= (int) ((float) num 
* be
->m_videoSize
.x 
/ den
); 
 289                 be
->m_videoSize
.y 
= (int) ((float) den 
* be
->m_videoSize
.y 
/ num
); 
 292         be
->PostRecalcSize();         
 299 //--------------------------------------------------------------------------- 
 300 // wxGStreamerMediaBackend::PostRecalcSize 
 302 // Forces parent to recalc its layout if it has sizers to update 
 303 // to the new video size 
 304 //--------------------------------------------------------------------------- 
 305 void wxGStreamerMediaBackend::PostRecalcSize() 
 307         m_ctrl
->InvalidateBestSize(); 
 308         m_ctrl
->GetParent()->Layout(); 
 309         m_ctrl
->GetParent()->Refresh(); 
 310         m_ctrl
->GetParent()->Update(); 
 313 //--------------------------------------------------------------------------- 
 314 // wxGStreamerMediaBackend::OnFinish 
 316 // Called by gstreamer when the media is done playing 
 318 // 1) Send a wxEVT_MEDIA_STOP to the control 
 319 // 2) If veteod, break out 
 320 // 3) really stop the media 
 321 // 4) Send a wxEVT_MEDIA_FINISHED to the control 
 322 //--------------------------------------------------------------------------- 
 323 void wxGStreamerMediaBackend::OnFinish(GstElement 
*play
, gpointer    data
) 
 325     wxGStreamerMediaBackend
* m_parent 
= (wxGStreamerMediaBackend
*) data
; 
 327     wxMediaEvent 
theEvent(wxEVT_MEDIA_STOP
, 
 328                         m_parent
->m_ctrl
->GetId()); 
 329     m_parent
->m_ctrl
->ProcessEvent(theEvent
); 
 331     if(theEvent
.IsAllowed()) 
 333         bool bOk 
= m_parent
->Stop(); 
 336         //send the event to our child 
 337         wxMediaEvent 
theEvent(wxEVT_MEDIA_FINISHED
, 
 338                             m_parent
->m_ctrl
->GetId()); 
 339         m_parent
->m_ctrl
->ProcessEvent(theEvent
); 
 343 //--------------------------------------------------------------------------- 
 344 // wxGStreamerMediaBackend::OnError 
 346 // Called by gstreamer when an error is encountered playing the media 
 348 // TODO: Make this better - maybe some more intelligent wxLog stuff 
 349 //--------------------------------------------------------------------------- 
 350 void wxGStreamerMediaBackend::OnError(GstElement 
*play
, 
 358             wxT("Error in wxMediaCtrl!\nError Message:%s\nDebug:%s\n"),  
 359             (const wxChar
*)wxConvUTF8
.cMB2WX(err
->message
), 
 360             (const wxChar
*)wxConvUTF8
.cMB2WX(debug
) 
 366 //--------------------------------------------------------------------------- 
 367 // wxGStreamerMediaBackend::Load (File version) 
 369 // Just calls the URI version 
 370 //--------------------------------------------------------------------------- 
 371 bool wxGStreamerMediaBackend::Load(const wxString
& fileName
) 
 375                             wxString( wxT("file://") ) + fileName 
 
 380 //--------------------------------------------------------------------------- 
 381 // wxGStreamerMediaBackend::OnVideoCapsReady 
 383 // Called by gstreamer when the video caps for the media is ready 
 384 //--------------------------------------------------------------------------- 
 385 void wxGStreamerMediaBackend::OnVideoCapsReady(GstPad
* pad
, GParamSpec
* pspec
, gpointer data
) 
 387     wxGStreamerMediaBackend::TransCapsToVideoSize((wxGStreamerMediaBackend
*) data
, pad
);     
 390 //--------------------------------------------------------------------------- 
 391 // wxGStreamerMediaBackend::Load (URI version) 
 393 // 1) Stops/Cleanups the previous instance if there is any 
 394 // 2) Creates the gstreamer playbin 
 395 // 3) If there is no playbin bail out 
 396 // 4) Set up the error and end-of-stream callbacks for our player 
 397 // 5) Make our video sink and make sure it supports the x overlay interface 
 398 // 6) Make sure the passed URI is valid and tell playbin to load it 
 399 // 7) Use the xoverlay extension to tell gstreamer to play in our window 
 400 // 8) Get the video size - pause required to set the stream in action 
 401 //--------------------------------------------------------------------------- 
 402 bool wxGStreamerMediaBackend::Load(const wxURI
& location
) 
 408     m_player    
= gst_element_factory_make ("playbin", "play"); 
 415     g_signal_connect (m_player
, "eos", G_CALLBACK (OnFinish
), this); 
 416     g_signal_connect (m_player
, "error", G_CALLBACK (OnError
), this); 
 419     GstElement
* overlay 
= NULL
; 
 420     GstElement
* videosink
; 
 422 #if defined(__WXGTK__) && wxUSE_DYNLIB_CLASS 
 424     //use gnome-specific gstreamer extensions 
 425     //if synthisis (?) file not found, it  
 426     //spits out a warning and uses ximagesink 
 427     wxDynamicLibrary gstgconf
; 
 428     if(gstgconf
.Load(gstgconf
.CanonicalizeName(wxT("gstgconf-0.8")))) 
 430         typedef GstElement
* (*LPgst_gconf_get_default_video_sink
) (void); 
 431         LPgst_gconf_get_default_video_sink pGst_gconf_get_default_video_sink 
=  
 432         (LPgst_gconf_get_default_video_sink
) 
 433             gstgconf
.GetSymbol(wxT("gst_gconf_get_default_video_sink")); 
 435         if (pGst_gconf_get_default_video_sink
)         
 437             videosink 
= (*pGst_gconf_get_default_video_sink
) (); 
 438             wxASSERT( GST_IS_BIN(videosink
) ); 
 439             overlay 
= gst_bin_get_by_interface (GST_BIN (videosink
), 
 446         if ( ! GST_IS_X_OVERLAY(overlay
) ) 
 449             videosink 
= gst_element_factory_make ("xvimagesink", "videosink"); 
 450             if ( !GST_IS_OBJECT(videosink
) ) 
 451                 videosink 
= gst_element_factory_make ("ximagesink", "videosink"); 
 455             wxASSERT( GST_IS_X_OVERLAY(overlay
) ); 
 456             if ( ! GST_IS_X_OVERLAY(overlay
) ) 
 458 #if defined(__WXGTK__) && wxUSE_DYNLIB_CLASS 
 462     g_object_set (G_OBJECT (m_player
), 
 463                     "video-sink", videosink
, 
 464 //                    "audio-sink", m_audiosink, 
 468     wxString locstring 
= location
.BuildUnescapedURI(); 
 469     wxASSERT(gst_uri_protocol_is_valid("file")); 
 470     wxASSERT(gst_uri_is_valid(locstring
.mb_str())); 
 472     g_object_set (G_OBJECT (m_player
), "uri", (const char*)locstring
.mb_str(), NULL
); 
 476     if(!GTK_WIDGET_REALIZED(m_ctrl
->m_wxwindow
)) 
 478         //Not realized yet - set to connect at realization time 
 479         gtk_signal_connect( GTK_OBJECT(m_ctrl
->m_wxwindow
),  
 481                             GTK_SIGNAL_FUNC(wxGStreamerMediaBackend::OnGTKRealize
), 
 486         wxYield(); //see realize callback... 
 487         GdkWindow 
*window 
= GTK_PIZZA(m_ctrl
->m_wxwindow
)->bin_window
; 
 492     gst_x_overlay_set_xwindow_id( GST_X_OVERLAY(overlay
), 
 494                         GDK_WINDOW_XWINDOW( window 
) 
 506     wxASSERT(gst_element_set_state (m_player
, 
 507                 GST_STATE_PAUSED
)    == GST_STATE_SUCCESS
);             
 509     const GList 
*list 
= NULL
; 
 510     g_object_get (G_OBJECT (m_player
), "stream-info", &list
, NULL
); 
 512     for ( ; list 
!= NULL
; list 
= list
->next
) 
 514         GObject 
*info 
= (GObject 
*) list
->data
; 
 520         g_object_get (info
, "type", &type
, NULL
); 
 521         pspec 
= g_object_class_find_property ( 
 522                         G_OBJECT_GET_CLASS (info
), "type"); 
 523         val 
= g_enum_get_value (G_PARAM_SPEC_ENUM (pspec
)->enum_class
, type
); 
 525         if (strstr (val
->value_name
, "VIDEO")) 
 527             //Newer gstreamer 0.8+ is SUPPOSED to have "object"... 
 528             //but a lot of old plugins still use "pad" :) 
 529             pspec 
= g_object_class_find_property ( 
 530                         G_OBJECT_GET_CLASS (info
), "object"); 
 533                 g_object_get (info
, "pad", &pad
, NULL
); 
 535                 g_object_get (info
, "object", &pad
, NULL
); 
 537             pad 
= (GstPad 
*) GST_PAD_REALIZE (pad
); 
 540             if(!wxGStreamerMediaBackend::TransCapsToVideoSize(this, pad
)); 
 542                 //wait for those caps to get ready 
 546                 G_CALLBACK(wxGStreamerMediaBackend::OnVideoCapsReady
), 
 552             m_videoSize 
= wxSize(0,0); 
 555     }//end searching through info list     
 561 //--------------------------------------------------------------------------- 
 562 // wxGStreamerMediaBackend::Play 
 564 // Sets the stream to a playing state 
 565 //--------------------------------------------------------------------------- 
 566 bool wxGStreamerMediaBackend::Play() 
 568     if (gst_element_set_state (m_player
, GST_STATE_PLAYING
) 
 569             != GST_STATE_SUCCESS
) 
 574 //--------------------------------------------------------------------------- 
 575 // wxGStreamerMediaBackend::Pause 
 577 // Marks where we paused and pauses the stream 
 578 //--------------------------------------------------------------------------- 
 579 bool wxGStreamerMediaBackend::Pause() 
 581     m_nPausedPos 
= GetPosition(); 
 582     if (gst_element_set_state (m_player
, GST_STATE_PAUSED
) 
 583             != GST_STATE_SUCCESS
) 
 588 //--------------------------------------------------------------------------- 
 589 // wxGStreamerMediaBackend::Stop 
 591 // Pauses the stream and sets the position to 0 
 592 //--------------------------------------------------------------------------- 
 593 bool wxGStreamerMediaBackend::Stop() 
 595     if (gst_element_set_state (m_player
, 
 596                     GST_STATE_PAUSED
)    != GST_STATE_SUCCESS
) 
 598     return wxGStreamerMediaBackend::SetPosition(0); 
 601 //--------------------------------------------------------------------------- 
 602 // wxGStreamerMediaBackend::GetState 
 604 // Gets the state of the stream 
 605 //--------------------------------------------------------------------------- 
 606 wxMediaState 
wxGStreamerMediaBackend::GetState() 
 608     switch(GST_STATE(m_player
)) 
 610         case GST_STATE_PLAYING
: 
 611             return wxMEDIASTATE_PLAYING
; 
 612         case GST_STATE_PAUSED
: 
 613             if (m_nPausedPos 
== 0) 
 614                 return wxMEDIASTATE_STOPPED
; 
 616                 return wxMEDIASTATE_PAUSED
; 
 617         default://case GST_STATE_READY: 
 618             return wxMEDIASTATE_STOPPED
; 
 622 //--------------------------------------------------------------------------- 
 623 // wxGStreamerMediaBackend::GetPosition 
 625 // If paused, returns our marked position - otherwise it queries the  
 626 // GStreamer playbin for the position and returns that 
 629 //TODO: In lue of the last big TODO, when you pause and seek gstreamer 
 630 //TODO: doesn't update the position sometimes, so we need to keep track of whether     
 631 //TODO: we have paused or not and keep track of the time after the pause 
 632 //TODO: and whenever the user seeks while paused 
 634 //--------------------------------------------------------------------------- 
 635 wxLongLong 
wxGStreamerMediaBackend::GetPosition() 
 637     if(GetState() != wxMEDIASTATE_PLAYING
) 
 642         GstFormat fmtTime 
= GST_FORMAT_TIME
; 
 644         if (!gst_element_query (m_player
, GST_QUERY_POSITION
, &fmtTime
, &pos
)) 
 646         return pos 
/ GST_MSECOND 
; 
 650 //--------------------------------------------------------------------------- 
 651 // wxGStreamerMediaBackend::SetPosition 
 653 // Sets the position of the stream 
 654 // Note that GST_MSECOND is 1000000 (GStreamer uses nanoseconds - so 
 655 // there is 1000000 nanoseconds in a millisecond) 
 657 // If paused marks where we seeked to  
 658 //--------------------------------------------------------------------------- 
 659 bool wxGStreamerMediaBackend::SetPosition(wxLongLong where
) 
 661     if( gst_element_seek (m_player
, (GstSeekType
) (GST_SEEK_METHOD_SET 
| 
 662             GST_FORMAT_TIME 
| GST_SEEK_FLAG_FLUSH
), 
 663             where
.GetValue() * GST_MSECOND 
) ) 
 665         if (GetState() != wxMEDIASTATE_PLAYING
) 
 666             m_nPausedPos 
= where
; 
 674 //--------------------------------------------------------------------------- 
 675 // wxGStreamerMediaBackend::GetDuration 
 677 // Obtains the total time of our stream 
 678 //--------------------------------------------------------------------------- 
 679 wxLongLong 
wxGStreamerMediaBackend::GetDuration() 
 682     GstFormat fmtTime 
= GST_FORMAT_TIME
; 
 684     if(!gst_element_query(m_player
, GST_QUERY_TOTAL
, &fmtTime
, &length
)) 
 686     return length 
/ GST_MSECOND 
; 
 689 //--------------------------------------------------------------------------- 
 690 // wxGStreamerMediaBackend::Move 
 692 // Called when the window is moved - GStreamer takes care of this 
 693 // for us so nothing is needed 
 694 //--------------------------------------------------------------------------- 
 695 void wxGStreamerMediaBackend::Move(int x
, int y
, int w
, int h
) 
 699 //--------------------------------------------------------------------------- 
 700 // wxGStreamerMediaBackend::GetVideoSize 
 702 // Returns our cached video size from Load/OnVideoCapsReady 
 703 //--------------------------------------------------------------------------- 
 704 wxSize 
wxGStreamerMediaBackend::GetVideoSize() const 
 709 //--------------------------------------------------------------------------- 
 710 // wxGStreamerMediaBackend::GetPlaybackRate 
 711 // wxGStreamerMediaBackend::SetPlaybackRate 
 713 // Obtains/Sets the playback rate of the stream 
 715 //TODO: PlaybackRate not currently supported via playbin directly - 
 716 //TODO: Ronald S. Bultje noted on gstreamer-devel: 
 718 //TODO: Like "play at twice normal speed"? Or "play at 25 fps and 44,1 kHz"? As 
 719 //TODO: for the first, yes, we have elements for that, btu they"re not part of 
 720 //TODO: playbin. You can create a bin (with a ghost pad) containing the actual 
 721 //TODO: video/audiosink and the speed-changing element for this, and set that 
 722 //TODO: element as video-sink or audio-sink property in playbin. The 
 723 //TODO: audio-element is called "speed", the video-element is called "videodrop" 
 724 //TODO: (although that appears to be deprecated in favour of "videorate", which 
 725 //TODO: again cannot do this, so this may not work at all in the end). For 
 726 //TODO: forcing frame/samplerates, see audioscale and videorate. Audioscale is 
 727 //TODO: part of playbin. 
 728 //--------------------------------------------------------------------------- 
 729 double wxGStreamerMediaBackend::GetPlaybackRate() 
 731     //not currently supported via playbin 
 735 bool wxGStreamerMediaBackend::SetPlaybackRate(double dRate
) 
 737     //not currently supported via playbin 
 741 #endif //wxUSE_GSTREAMER 
 743 //in source file that contains stuff you don't directly use 
 744 #include <wx/html/forcelnk.h> 
 745 FORCE_LINK_ME(basewxmediabackends
); 
 747 #endif //wxUSE_MEDIACTRL