1 // ------------------------------------------------------------------------- 
   4 // Author:     Guilhem Lavaux 
   7 // Copyright:  (C) 1997, 1998, 1999 Guilhem Lavaux 
   8 // License:    wxWindows license 
   9 // ------------------------------------------------------------------------- 
  11 #include "wx/wxprec.h" 
  19 #include <wx/gtk/win_gtk.h> 
  23 #include <X11/Intrinsic.h> 
  25 #include <gtk/gtkwidget.h> 
  26 #include <gtk/gtkwindow.h> 
  28 #include <gdk/gdkprivate.h> 
  31 #include <wx/filefn.h> 
  32 #include <wx/wfstream.h> 
  33 #include <wx/datstrm.h> 
  34 #include <wx/tokenzr.h> 
  36 #define WXMMEDIA_INTERNAL 
  37 #include "wx/mmedia/vidbase.h" 
  38 #include "wx/mmedia/vidxanm.h" 
  40 IMPLEMENT_DYNAMIC_CLASS(wxVideoXANIM
, wxVideoBaseDriver
) 
  42 // ------------------------------------------------------------------------- 
  43 // End process detector 
  45 class wxVideoXANIMProcess
: public wxProcess 
{ 
  47     wxVideoXANIMProcess(wxVideoXANIM 
*xanim
); 
  49     void OnTerminate(int pid
, int status
); 
  52     wxVideoXANIM 
*m_vid_xanim
; 
  55 class wxVideoXANIMOutput
: public wxProcess 
{ 
  59     void OnTerminate(int pid
, int status
); 
  61     bool IsTerminated() const; 
  66 // ------------------------------------------------------------------------- 
  67 // XAnim video driver (process handling implementation) 
  69 wxVideoXANIMProcess::wxVideoXANIMProcess(wxVideoXANIM 
*xanim
) 
  74 void wxVideoXANIMProcess::OnTerminate(int WXUNUSED(pid
), int WXUNUSED(status
)) 
  76     m_vid_xanim
->m_xanim_started 
= false; 
  77     m_vid_xanim
->OnFinished(); 
  80 wxVideoXANIMOutput::wxVideoXANIMOutput() 
  81         : wxProcess(NULL
, wxID_ANY
) 
  87 bool wxVideoXANIMOutput::IsTerminated() const 
  92 void wxVideoXANIMOutput::OnTerminate(int pid
, int status
) 
  97 // ------------------------------------------------------------------------- 
  98 // XAnim video driver (implementation) 
 100 wxVideoXANIM::wxVideoXANIM() 
 101  : wxVideoBaseDriver() 
 103     m_internal       
= new wxXANIMinternal
; 
 104     m_xanim_detector 
= new wxVideoXANIMProcess(this); 
 105     m_xanim_started  
= false; 
 107     m_filename       
= wxEmptyString
; 
 108     m_remove_file    
= false; 
 111 wxVideoXANIM::wxVideoXANIM(wxInputStream
& str
) 
 112   : wxVideoBaseDriver(str
) 
 114     m_internal       
= new wxXANIMinternal
; 
 115     m_xanim_detector 
= new wxVideoXANIMProcess(this); 
 116     m_xanim_started  
= false; 
 121     m_filename       
= wxGetTempFileName(_T("vidxa")); 
 122     m_remove_file    
= true; 
 123     wxFileOutputStream 
fout(m_filename
); 
 130 wxVideoXANIM::wxVideoXANIM(const wxString
& filename
) 
 132     m_internal       
= new wxXANIMinternal
; 
 133     m_xanim_detector 
= new wxVideoXANIMProcess(this); 
 134     m_xanim_started  
= false; 
 137     m_filename       
= filename
; 
 138     m_remove_file    
= false; 
 145 wxVideoXANIM::~wxVideoXANIM() 
 150     delete m_xanim_detector
; 
 153         wxRemoveFile(m_filename
); 
 156 // ------------------------------------------------------------------------- 
 159 bool wxVideoXANIM::Play() 
 161     if (!m_paused 
&& m_xanim_started
) 
 163     if (!m_video_output
) { 
 164         wxVideoCreateFrame(this); 
 168     // The movie starts with xanim 
 169     if (RestartXANIM()) { 
 176 bool wxVideoXANIM::Pause() 
 178     if (!m_paused 
&& SendCommand(" ")) { 
 185 bool wxVideoXANIM::Resume() 
 187     if (m_paused 
&& SendCommand(" ")) { 
 194 bool wxVideoXANIM::Stop() 
 196     if (!m_xanim_started
) 
 201     // We are waiting for the termination of the subprocess. 
 202     while (m_xanim_started
) {  
 211 // ------------------------------------------------------------------------- 
 212 // Movie size controller 
 214 bool wxVideoXANIM::SetSize(wxSize size
) 
 219     m_video_output
->SetSize(size
.GetWidth(), size
.GetHeight()); 
 223 bool wxVideoXANIM::GetSize(wxSize
& size
) const 
 227     size
.Set(m_size
[0], m_size
[1]); 
 231 // ------------------------------------------------------------------------- 
 232 // Capabilities of XAnim 
 234 bool wxVideoXANIM::IsCapable(wxVideoType v_type
) const 
 236     if (v_type 
== wxVIDEO_MSAVI 
|| v_type 
== wxVIDEO_MPEG 
|| 
 237         v_type 
== wxVIDEO_QT 
|| v_type 
== wxVIDEO_GIF 
|| v_type 
== wxVIDEO_JMOV 
|| 
 238         v_type 
== wxVIDEO_FLI 
|| v_type 
== wxVIDEO_IFF 
|| v_type 
== wxVIDEO_SGI
) 
 244 // ------------------------------------------------------------------------- 
 247 wxString 
wxVideoXANIM::GetMovieCodec() const 
 254 wxString 
wxVideoXANIM::GetAudioCodec() const 
 261 wxUint32 
wxVideoXANIM::GetSampleRate() const 
 268 wxUint8 
wxVideoXANIM::GetChannels() const 
 275 wxUint8 
wxVideoXANIM::GetBPS() const 
 282 double wxVideoXANIM::GetFrameRate() const 
 289 wxUint32 
wxVideoXANIM::GetNbFrames() const 
 297 bool wxVideoXANIM::IsPaused() const 
 302 bool wxVideoXANIM::IsStopped() const 
 304     return !m_xanim_started
; 
 307 // ------------------------------------------------------------------------- 
 310 bool wxVideoXANIM::AttachOutput(wxWindow
& out
) 
 312     if (!wxVideoBaseDriver::AttachOutput(out
)) 
 318 void wxVideoXANIM::DetachOutput() 
 321     m_xanim_started 
= false; 
 324     wxVideoBaseDriver::DetachOutput(); 
 327 // ------------------------------------------------------------------------- 
 328 // Lowlevel XAnim controller 
 330 bool wxVideoXANIM::SendCommand(const char *command
, char **ret
, 
 333     if (!m_xanim_started
) 
 337     // Send a command to XAnim through X11 Property 
 338     XChangeProperty(m_internal
->xanim_dpy
, m_internal
->xanim_window
, 
 339             m_internal
->xanim_atom
, 
 340             XA_STRING
, 8, PropModeReplace
, (unsigned char *)command
, 
 342     XFlush(m_internal
->xanim_dpy
); 
 348     XGetWindowProperty(m_internal
->xanim_dpy
, m_internal
->xanim_window
, 
 349                m_internal
->xanim_ret
, 0, 16, True
, AnyPropertyType
, 
 350                &prop_type
, &prop_format
, (unsigned long *)size
, 
 351                &extra
, (unsigned char **)ret
); 
 356 bool wxVideoXANIM::CollectInfo() 
 358     wxVideoXANIMOutput 
*xanimProcess
; 
 359     wxString xanim_command
; 
 360     wxStringTokenizer tokenizer
; 
 362     xanimProcess 
= new wxVideoXANIMOutput
; 
 363     xanim_command 
= wxT("xanim +v +Zv -Ae "); 
 364     xanim_command 
+= m_filename
; 
 365     if (!wxExecute(xanim_command
, false, xanimProcess
)) 
 368     wxInputStream 
*infoStream 
= xanimProcess
->GetInputStream(); 
 369     wxString totalOutput
; 
 371     while (infoStream
->GetLastError() == wxSTREAM_NO_ERROR
) { 
 374         infoStream
->Read(line
, sizeof(line
)-1); 
 375         if (infoStream
->LastRead() == 0) 
 378         line
[infoStream
->LastRead()] = 0; 
 380         totalOutput 
+= wxString::FromAscii(line
);  
 383     // This is good for everything ... :-) 
 384     int position 
= totalOutput
.Find(wxT("Video Codec:")); 
 386     totalOutput
.Remove(0, position
+13); 
 388     position 
= totalOutput
.Find(wxT("depth=")); 
 389     m_movieCodec 
= totalOutput(0, position
); 
 391     totalOutput
.Remove(0, position
); 
 392     tokenizer
.SetString(totalOutput
, wxT("\n\r")); 
 394     // the rest of the line 
 395     wxString token 
= tokenizer
.GetNextToken(); 
 396     unsigned long my_long
; 
 399 totalOutput.ToULong(&my_long); \ 
 403     totalOutput 
= tokenizer
.GetString(); 
 404     totalOutput
.Remove(0, totalOutput
.Find(wxT(":"))+2); 
 406     position 
= totalOutput
.Find(wxT("Rate")); 
 407     m_audioCodec 
= totalOutput(0, position
-1); 
 410     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 411     GETINT(m_sampleRate
); 
 413     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 416     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 419     tokenizer
.Reinit(totalOutput
); 
 420     tokenizer
.GetNextToken(); 
 421     totalOutput 
= tokenizer
.GetString(); 
 422     totalOutput
.Remove(0, totalOutput
.Find(wxT(":"))+2); 
 424     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 427     totalOutput
.Remove(0,1); 
 430     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 433     totalOutput
.Remove(0, totalOutput
.Find(wxT("="))+1); 
 434     totalOutput
.ToDouble(&m_frameRate
); 
 436     // We wait for the conclusion 
 437     while (!xanimProcess
->IsTerminated()) 
 445 bool wxVideoXANIM::RestartXANIM() 
 447     wxString xanim_command
; 
 451     unsigned long nitems
; 
 456     if (!m_video_output 
|| m_xanim_started
) 
 459     // Check if we can change the size of the window dynamicly 
 460     xanim_chg_size 
= true; 
 461     // Get current display 
 463     m_internal
->xanim_dpy 
= gdk_display
; 
 464     GtkPizza 
*pizza 
= GTK_PIZZA( m_video_output
->m_wxwindow 
); 
 465     GdkWindow 
*window 
= pizza
->bin_window
; 
 467     m_internal
->xanim_window 
= GDK_WINDOW_XWINDOW(window
); 
 469     // Get the XANIM atom 
 470     m_internal
->xanim_atom 
= XInternAtom(m_internal
->xanim_dpy
, 
 471                      "XANIM_PROPERTY", False
); 
 474     xanim_command
.Printf( 
 475         wxT("xanim -Zr +Ze +Sr +f +W%d +f +q +Av70 %s %s"), 
 476         (int)m_internal
->xanim_window
, 
 477         (xanim_chg_size
) ? _T("") : _T(""),   // ??? why ??? 
 478         WXSTRINGCAST m_filename
); 
 481     if (!wxExecute(xanim_command
, false, m_xanim_detector
)) 
 484     // Wait for XAnim to be ready 
 486     m_xanim_started 
= true; 
 487     while (nitems 
== 0 && m_xanim_started
) { 
 488       ret 
= XGetWindowProperty(m_internal
->xanim_dpy
, m_internal
->xanim_window
, 
 489                    m_internal
->xanim_atom
, 
 490                    0, 4, False
, AnyPropertyType
, &prop_type
, 
 491                    &prop_format
, &nitems
, &extra
, 
 492                    (unsigned char **)&prop
); 
 498     vibrato_size 
= m_video_output
->GetSize(); 
 500     vibrato_size
.SetWidth(vibrato_size
.GetWidth()+1); 
 501     m_video_output
->SetSize(vibrato_size
); 
 502     vibrato_size
.SetWidth(vibrato_size
.GetWidth()-1); 
 503     m_video_output
->SetSize(vibrato_size
); 
 504     // Very useful ! Actually it "should" sends a SETSIZE event to XAnim 
 505     // FIXME: This event is not sent !!