2006-12-20 01:31:52 +01:00
// Copyright (c) 2006, Rodrigo Braz Monteiro, Mike Matsnev
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
// * Neither the name of the Aegisub Group nor the names of its contributors
// may be used to endorse or promote products derived from this software
// without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
// -----------------------------------------------------------------------------
//
// AEGISUB
//
// Website: http://aegisub.cellosoft.com
// Contact: mailto:zeratul@cellosoft.com
//
///////////
// Headers
2008-01-16 19:29:29 +01:00
2007-12-31 07:46:22 +01:00
# ifdef WITH_DIRECTSHOW
2006-12-20 01:31:52 +01:00
# pragma warning(disable: 4995)
2008-03-13 20:12:55 +01:00
# pragma warning(disable: 4238)
2007-01-23 05:42:08 +01:00
# include <wx/wxprec.h>
# ifdef __WINDOWS__
# include <wx/image.h>
2007-01-21 08:12:47 +01:00
# include <dshow.h>
# include <atlbase.h>
# include <atlcom.h>
# include <atlstr.h>
# include <atlcoll.h>
2006-12-20 01:31:52 +01:00
# include <windows.h>
# include <tchar.h>
# include <initguid.h>
# include "utils.h"
# include "vfr.h"
2007-01-21 08:12:47 +01:00
# include "videosink.h"
2007-01-27 07:15:25 +01:00
# include "gl_wrap.h"
# include "options.h"
2008-03-05 03:05:01 +01:00
# include "video_provider_dshow.h"
2007-01-21 08:12:47 +01:00
2006-12-20 01:31:52 +01:00
///////////////
// Constructor
// Based on Haali's code for DirectShowSource2
2008-03-07 22:24:14 +01:00
DirectShowVideoProvider : : DirectShowVideoProvider ( Aegisub : : String _filename , double _fps ) {
2007-01-01 04:29:20 +01:00
fps = _fps ;
2007-01-21 07:30:19 +01:00
m_registered = false ;
2006-12-20 01:31:52 +01:00
m_hFrameReady = CreateEvent ( NULL , FALSE , FALSE , NULL ) ;
2007-01-21 07:30:19 +01:00
HRESULT hr = OpenVideo ( _filename ) ;
if ( FAILED ( hr ) ) throw _T ( " Failed opening DirectShow content. " ) ;
2006-12-20 01:31:52 +01:00
}
//////////////
// Destructor
DirectShowVideoProvider : : ~ DirectShowVideoProvider ( ) {
2006-12-31 06:01:01 +01:00
CloseVideo ( ) ;
2006-12-20 01:31:52 +01:00
}
////////////
// Get pin
// Code by Haali
# define ENUM_FILTERS(graph, var) for (CComPtr<IEnumFilters> __pEF__; !__pEF__ && SUCCEEDED(graph->EnumFilters(&__pEF__)); ) for (CComPtr<IBaseFilter> var; __pEF__->Next(1, &var, NULL) == S_OK; var.Release())
# define ENUM_PINS(filter, var) for (CComPtr<IEnumPins> __pEP__; !__pEP__ && SUCCEEDED(filter->EnumPins(&__pEP__)); ) for (CComPtr<IPin> var; __pEP__->Next(1, &var, NULL) == S_OK; var.Release())
class MTPtr {
AM_MEDIA_TYPE * pMT ;
MTPtr ( const MTPtr & ) ;
MTPtr & operator = ( const MTPtr & ) ;
public :
MTPtr ( ) : pMT ( NULL ) { }
~ MTPtr ( ) { DeleteMediaType ( pMT ) ; }
AM_MEDIA_TYPE * operator - > ( ) { return pMT ; }
const AM_MEDIA_TYPE * operator - > ( ) const { return pMT ; }
operator AM_MEDIA_TYPE * ( ) { return pMT ; }
AM_MEDIA_TYPE * * operator & ( ) { DeleteMediaType ( pMT ) ; pMT = NULL ; return & pMT ; }
static void FreeMediaType ( AM_MEDIA_TYPE * pMT ) {
if ( pMT = = NULL )
return ;
if ( pMT - > cbFormat > 0 ) {
CoTaskMemFree ( pMT - > pbFormat ) ;
pMT - > pbFormat = NULL ;
pMT - > cbFormat = 0 ;
}
if ( pMT - > pUnk ) {
pMT - > pUnk - > Release ( ) ;
pMT - > pUnk = NULL ;
}
}
static void DeleteMediaType ( AM_MEDIA_TYPE * pMT ) {
if ( pMT = = NULL )
return ;
if ( pMT - > cbFormat > 0 )
CoTaskMemFree ( pMT - > pbFormat ) ;
if ( pMT - > pUnk )
pMT - > pUnk - > Release ( ) ;
CoTaskMemFree ( pMT ) ;
}
} ;
# define ENUM_MT(pin, var) for (CComPtr<IEnumMediaTypes> __pEMT__; !__pEMT__ && SUCCEEDED(pin->EnumMediaTypes(&__pEMT__)); ) for (MTPtr var; __pEMT__->Next(1, &var, NULL) == S_OK; )
CComPtr < IPin > GetPin ( IBaseFilter * pF , bool include_connected , PIN_DIRECTION dir , const GUID * pMT = NULL ) {
if ( pF = = NULL )
return CComPtr < IPin > ( ) ;
ENUM_PINS ( pF , pP ) {
PIN_DIRECTION pd ;
if ( FAILED ( pP - > QueryDirection ( & pd ) ) )
continue ;
if ( pd = = dir ) {
if ( ! include_connected ) {
CComPtr < IPin > pQ ;
if ( SUCCEEDED ( pP - > ConnectedTo ( & pQ ) ) )
continue ;
}
if ( pMT = = NULL )
return pP ;
ENUM_MT ( pP , MT )
if ( MT - > majortype = = * pMT )
return pP ;
}
}
return CComPtr < IPin > ( ) ;
}
////////////////////
// More Haali stuff
void DirectShowVideoProvider : : RegROT ( ) {
if ( ! m_pGC | | m_registered )
return ;
CComPtr < IRunningObjectTable > rot ;
if ( FAILED ( GetRunningObjectTable ( 0 , & rot ) ) )
return ;
CStringA name ;
name . Format ( " FilterGraph %08p pid %08x (avss) " , m_pGC . p , GetCurrentProcessId ( ) ) ;
CComPtr < IMoniker > mk ;
if ( FAILED ( CreateItemMoniker ( L " ! " , CA2W ( name ) , & mk ) ) )
return ;
if ( SUCCEEDED ( rot - > Register ( ROTFLAGS_REGISTRATIONKEEPSALIVE , m_pGC , mk , & m_rot_cookie ) ) )
m_registered = true ;
}
void DirectShowVideoProvider : : UnregROT ( ) {
if ( ! m_registered )
return ;
CComPtr < IRunningObjectTable > rot ;
if ( FAILED ( GetRunningObjectTable ( 0 , & rot ) ) )
return ;
if ( SUCCEEDED ( rot - > Revoke ( m_rot_cookie ) ) )
m_registered = false ;
}
//////////////
// Open video
HRESULT DirectShowVideoProvider : : OpenVideo ( wxString _filename ) {
HRESULT hr ;
// Create an instance of the Filter Graph
CComPtr < IGraphBuilder > pG ;
if ( FAILED ( hr = pG . CoCreateInstance ( CLSID_FilterGraph ) ) ) return hr ;
// Create an Instance of the Video Sink
2007-01-21 07:30:19 +01:00
//CComPtr<IBaseFilter> pR;
//CLSID CLSID_VideoSink;
//CLSIDFromString(L"{F13D3732-96BD-4108-AFEB-E85F68FF64DC}",&CLSID_VideoSink);
//if (FAILED(hr = pR.CoCreateInstance(CLSID_VideoSink))) return hr;
CComPtr < IBaseFilter > pR ;
hr = CreateVideoSink ( & pR ) ;
2006-12-20 01:31:52 +01:00
// Add VideoSink to graph
pG - > AddFilter ( pR , L " VideoSink " ) ;
2007-01-22 04:39:36 +01:00
// Query interface from sink
2006-12-20 01:31:52 +01:00
CComQIPtr < IVideoSink > sink ( pR ) ;
if ( ! sink ) return E_NOINTERFACE ;
CComQIPtr < IVideoSink2 > sink2 ( pR ) ;
if ( ! sink2 ) return E_NOINTERFACE ;
// Set allowed types for sink
2007-01-27 07:15:25 +01:00
unsigned int types = IVS_RGB24 | IVS_RGB32 ;
2007-01-27 16:25:53 +01:00
if ( OpenGLWrapper : : UseShaders ( ) ) types = types | IVS_YV12 ;
2007-01-27 07:15:25 +01:00
sink - > SetAllowedTypes ( types ) ;
2006-12-20 01:31:52 +01:00
2006-12-31 06:01:01 +01:00
// Pass the event to sink, so it gets set when a frame is available
2006-12-20 01:31:52 +01:00
ResetEvent ( m_hFrameReady ) ;
sink2 - > NotifyFrame ( m_hFrameReady ) ;
// Create source filter and add it to graph
CComPtr < IBaseFilter > pS ;
if ( FAILED ( hr = pG - > AddSourceFilter ( _filename . wc_str ( ) , NULL , & pS ) ) ) return hr ;
// Property bag? The heck is this?
// Is this supposed to make it "interactive", enabling some actions?
// I have no clue.
CComQIPtr < IPropertyBag > pPB ( pS ) ;
if ( pPB ) pPB - > Write ( L " ui.interactive " , & CComVariant ( 0u , VT_UI4 ) ) ;
// Get source's output pin
CComPtr < IPin > pO ( GetPin ( pS , false , PINDIR_OUTPUT , & MEDIATYPE_Video ) ) ;
if ( ! pO ) pO = GetPin ( pS , false , PINDIR_OUTPUT , & MEDIATYPE_Stream ) ;
// Get sink's input pin
CComPtr < IPin > pI ( GetPin ( pR , false , PINDIR_INPUT ) ) ;
// Check if pins are ok
if ( ! pO | | ! pI ) return E_FAIL ;
// Connect pins
if ( FAILED ( hr = pG - > Connect ( pO , pI ) ) ) return hr ;
2006-12-31 06:01:01 +01:00
// Query the control interfaces from the graph
2006-12-20 01:31:52 +01:00
CComQIPtr < IMediaControl > mc ( pG ) ;
CComQIPtr < IMediaSeeking > ms ( pG ) ;
// See if they were created correctly
if ( ! mc | | ! ms ) return E_NOINTERFACE ;
// Run MediaControl, initiating the data flow through it
if ( FAILED ( hr = mc - > Run ( ) ) ) return hr ;
// Get state from media seeking (??)
OAFilterState fs ;
if ( FAILED ( hr = mc - > GetState ( 2000 , & fs ) ) ) return hr ;
// Wait up to 5 seconds for the first frame to arrive
if ( WaitForSingleObject ( m_hFrameReady , 5000 ) ! = WAIT_OBJECT_0 ) return E_FAIL ;
// Get frame format
unsigned type , arx , ary ;
if ( FAILED ( hr = sink2 - > GetFrameFormat ( & type , & width , & height , & arx , & ary , & defd ) ) ) return hr ;
// Get video duration
if ( FAILED ( hr = ms - > GetDuration ( & duration ) ) ) return hr ;
// Set pixel type
//switch (type) {
// case IVS_RGB32: m_vi.pixel_type = VideoInfo::CS_BGR32; break;
// case IVS_YUY2: m_vi.pixel_type = VideoInfo::CS_YUY2; break;
// case IVS_YV12: m_vi.pixel_type = VideoInfo::CS_YV12; break;
// default: return E_FAIL;
//}
2007-01-01 04:29:20 +01:00
// Set FPS and frame duration
if ( defd = = 0 ) defd = 417083 ;
2007-08-31 16:11:35 +02:00
if ( fps ! = 0.0 ) defd = int64_t ( 10000000.0 / fps ) + 1 ;
2007-01-29 06:47:29 +01:00
else fps = 10000000.0 / double ( + + defd ) ;
2007-01-01 04:29:20 +01:00
// Set number of frames
2007-01-24 04:54:32 +01:00
last_fnum = 0 ;
2006-12-20 01:31:52 +01:00
num_frames = duration / defd ;
// Store filters
2006-12-20 01:45:02 +01:00
m_pR = sink ;
m_pGC = mc ;
m_pGS = ms ;
2006-12-20 01:31:52 +01:00
// Flag frame as ready?
SetEvent ( m_hFrameReady ) ;
2006-12-31 06:01:01 +01:00
// Register graph with Running Objects Table for remote graphedit connection
2006-12-20 01:45:02 +01:00
RegROT ( ) ;
2006-12-20 01:31:52 +01:00
2008-07-15 02:08:05 +02:00
// Read keyframes and timecodes from MKV file
isVfr = false ;
FrameRate temp ;
double overFps = 0 ;
bool mkvOpen = MatroskaWrapper : : wrapper . IsOpen ( ) ;
KeyFrames . Clear ( ) ;
wxString extension = _filename . Right ( 4 ) . Lower ( ) ;
if ( extension = = _T ( " .mkv " ) | | mkvOpen ) {
// Parse mkv
if ( ! mkvOpen ) MatroskaWrapper : : wrapper . Open ( _filename ) ;
// Get keyframes
KeyFrames = MatroskaWrapper : : wrapper . GetKeyFrames ( ) ;
keyFramesLoaded = true ;
// Ask to override timecodes
int override = wxYES ;
if ( VFR_Output . IsLoaded ( ) ) override = wxMessageBox ( _ ( " You already have timecodes loaded. Replace them with the timecodes from the Matroska file? " ) , _ ( " Replace timecodes? " ) , wxYES_NO | wxICON_QUESTION ) ;
if ( override = = wxYES ) {
MatroskaWrapper : : wrapper . SetToTimecodes ( temp ) ;
isVfr = temp . GetFrameRateType ( ) = = VFR ;
if ( isVfr ) {
overFps = temp . GetCommonFPS ( ) ;
MatroskaWrapper : : wrapper . SetToTimecodes ( VFR_Input ) ;
MatroskaWrapper : : wrapper . SetToTimecodes ( VFR_Output ) ;
trueFrameRate = temp ;
}
}
// Close mkv
MatroskaWrapper : : wrapper . Close ( ) ;
}
else if ( extension = = _T ( " .avi " ) ) {
keyFramesLoaded = false ;
KeyFrames . Clear ( ) ;
KeyFrames = VFWWrapper : : GetKeyFrames ( _filename ) ;
keyFramesLoaded = true ;
}
// Check if the file is all keyframes
bool isAllKeyFrames = true ;
for ( unsigned int i = 1 ; i < KeyFrames . GetCount ( ) ; i + + ) {
// Is the last keyframe not this keyframe -1?
if ( KeyFrames [ i - 1 ] ! = ( int ) ( i - 1 ) ) {
// It's not all keyframes, go ahead
isAllKeyFrames = false ;
break ;
}
}
// If it is all keyframes, discard the keyframe info as it is useless
if ( isAllKeyFrames ) {
KeyFrames . Clear ( ) ;
keyFramesLoaded = false ;
}
2006-12-31 06:01:01 +01:00
//NextFrame();
2006-12-20 01:31:52 +01:00
// Set frame count
//m_f.SetCount(m_vi.num_frames);
return hr ;
}
2006-12-31 06:01:01 +01:00
///////////////
// Close video
void DirectShowVideoProvider : : CloseVideo ( ) {
2007-01-21 07:30:19 +01:00
rdf . frame . Clear ( ) ;
2006-12-31 06:01:01 +01:00
CComQIPtr < IVideoSink2 > pVS2 ( m_pR ) ;
if ( pVS2 ) pVS2 - > NotifyFrame ( NULL ) ;
UnregROT ( ) ;
m_pR . Release ( ) ;
m_pGC . Release ( ) ;
m_pGS . Release ( ) ;
2007-01-22 04:39:36 +01:00
ResetEvent ( m_hFrameReady ) ;
CloseHandle ( m_hFrameReady ) ;
2006-12-31 06:01:01 +01:00
}
/////////////////////////
// Read DirectShow frame
2007-08-31 16:11:35 +02:00
void DirectShowVideoProvider : : ReadFrame ( int64_t timestamp , unsigned format , unsigned bpp , const unsigned char * frame , unsigned width , unsigned height , int stride , unsigned arx , unsigned ary , void * arg ) {
2006-12-31 06:01:01 +01:00
// Set frame
DF * df = ( DF * ) arg ;
df - > timestamp = timestamp ;
2007-01-21 07:30:19 +01:00
// Create frame
const unsigned char * src = frame ;
if ( stride < 0 ) {
src + = stride * ( height - 1 ) ;
stride = - stride ;
df - > frame . flipped = true ;
}
else df - > frame . flipped = false ;
df - > frame . w = width ;
df - > frame . h = height ;
df - > frame . pitch [ 0 ] = stride ;
2007-03-28 00:29:35 +02:00
if ( format = = IVS_YV12 ) {
df - > frame . pitch [ 1 ] = stride / 2 ;
df - > frame . pitch [ 2 ] = stride / 2 ;
}
2007-01-21 07:30:19 +01:00
df - > frame . cppAlloc = false ;
df - > frame . invertChannels = true ;
2007-03-28 00:29:35 +02:00
// Set format
if ( format = = IVS_RGB24 ) df - > frame . format = FORMAT_RGB24 ;
else if ( format = = IVS_RGB32 ) df - > frame . format = FORMAT_RGB32 ;
else if ( format = = IVS_YV12 ) {
df - > frame . format = FORMAT_YV12 ;
df - > frame . invertChannels = true ;
2006-12-31 06:01:01 +01:00
}
2007-03-28 00:29:35 +02:00
else if ( format = = IVS_YUY2 ) df - > frame . format = FORMAT_YUY2 ;
2006-12-31 06:01:01 +01:00
2007-03-28 00:29:35 +02:00
// Allocate and copy data
df - > frame . Allocate ( ) ;
memcpy ( df - > frame . data [ 0 ] , src , df - > frame . pitch [ 0 ] * height + ( df - > frame . pitch [ 1 ] + df - > frame . pitch [ 2 ] ) * height / 2 ) ;
2006-12-31 06:01:01 +01:00
}
/////////////////////
// Get Next DS Frame
2007-01-21 07:30:19 +01:00
int DirectShowVideoProvider : : NextFrame ( DF & df , int & _fn ) {
2006-12-31 06:01:01 +01:00
// Keep reading until it gets a good frame
while ( true ) {
2007-01-01 04:29:20 +01:00
// Set object and receive data
if ( WaitForSingleObject ( m_hFrameReady , INFINITE ) ! = WAIT_OBJECT_0 ) return 1 ;
2006-12-31 06:01:01 +01:00
// Read frame
HRESULT hr = m_pR - > ReadFrame ( ReadFrame , & df ) ;
2007-01-21 07:30:19 +01:00
if ( FAILED ( hr ) ) {
//df.frame.Clear();
return 2 ;
}
2006-12-31 06:01:01 +01:00
// End of file
2007-01-21 07:30:19 +01:00
if ( hr = = S_FALSE ) {
//df.frame.Clear();
return 3 ;
}
2006-12-31 06:01:01 +01:00
// Valid timestamp
if ( df . timestamp > = 0 ) {
2007-01-01 04:29:20 +01:00
// CFR frame number
int frameno = - 1 ;
2008-03-07 22:24:14 +01:00
if ( frameTime . size ( ) = = 0 ) frameno = ( int ) ( ( double ) df . timestamp / defd + 0.5 ) ;
2007-01-01 04:29:20 +01:00
// VFR
else {
2008-03-07 22:24:14 +01:00
for ( unsigned int i = 0 ; i < frameTime . size ( ) ; i + + ) {
2007-08-31 16:11:35 +02:00
if ( df . timestamp < ( int64_t ) frameTime [ i ] * 10000 ) {
2007-01-01 04:29:20 +01:00
frameno = i - 1 ;
break ;
}
}
2008-03-07 22:24:14 +01:00
if ( frameno = = - 1 ) frameno = frameTime . size ( ) - 1 ;
2007-01-01 04:29:20 +01:00
}
2006-12-31 06:01:01 +01:00
// Got a good one
2007-01-01 04:29:20 +01:00
if ( frameno > = 0 ) {
2006-12-31 06:01:01 +01:00
_fn = frameno ;
2007-01-21 07:30:19 +01:00
//_df = df;
2007-01-01 04:29:20 +01:00
return 0 ;
2006-12-31 06:01:01 +01:00
}
}
2007-01-21 07:30:19 +01:00
//df.frame.Clear();
2006-12-31 06:01:01 +01:00
}
}
/////////////
// Get frame
2008-03-07 01:17:03 +01:00
const AegiVideoFrame DirectShowVideoProvider : : GetFrame ( int n , int formatMask ) {
2006-12-31 06:36:20 +01:00
// Normalize frame number
if ( n > = ( signed ) num_frames ) n = num_frames - 1 ;
2007-01-01 04:29:20 +01:00
if ( n < 0 ) n = 0 ;
2006-12-31 06:36:20 +01:00
2006-12-31 06:01:01 +01:00
// Variables
2007-01-21 07:30:19 +01:00
//DF df;
2006-12-31 06:01:01 +01:00
int fn ;
2007-01-01 04:29:20 +01:00
// Time to seek to
REFERENCE_TIME cur ;
cur = defd * n + 10001 ;
2008-03-07 22:24:14 +01:00
if ( frameTime . size ( ) > ( unsigned ) n ) cur = frameTime [ n ] * 10000 + 10001 ;
2006-12-31 06:36:20 +01:00
if ( cur < 0 ) cur = 0 ;
2006-12-31 06:01:01 +01:00
// Is next
2007-01-24 04:54:32 +01:00
if ( n = = ( signed ) last_fnum + 1 ) {
2007-01-21 07:30:19 +01:00
//rdf.frame.Clear();
NextFrame ( rdf , fn ) ;
2006-12-31 06:01:01 +01:00
last_fnum = n ;
2007-01-21 07:30:19 +01:00
return rdf . frame ;
2006-12-31 06:01:01 +01:00
}
2006-12-31 06:36:20 +01:00
// Not the next, reset and seek first
seek :
ResetEvent ( m_hFrameReady ) ;
// Seek
2007-01-21 07:30:19 +01:00
if ( FAILED ( m_pGS - > SetPositions ( & cur , AM_SEEKING_AbsolutePositioning , NULL , AM_SEEKING_NoPositioning ) ) ) return AegiVideoFrame ( width , height ) ;
2006-12-31 06:36:20 +01:00
// Set time
2007-01-21 07:30:19 +01:00
REFERENCE_TIME timestamp = - 1 ;
2006-12-31 06:36:20 +01:00
2006-12-31 06:01:01 +01:00
// Actually get data
while ( true ) {
// Get frame
2007-01-01 04:29:20 +01:00
int fn = - 1 ;
2007-01-21 07:30:19 +01:00
int result = NextFrame ( rdf , fn ) ;
2006-12-31 06:01:01 +01:00
// Preroll
2007-01-21 07:30:19 +01:00
if ( result = = 0 & & fn < n ) {
continue ;
}
2006-12-31 06:01:01 +01:00
// Right frame
2007-01-01 04:29:20 +01:00
else if ( fn = = n ) {
2006-12-31 06:01:01 +01:00
// we want this frame, compare timestamps to account for decimation
// we see this for the first time
2007-01-21 07:30:19 +01:00
if ( timestamp < 0 ) timestamp = rdf . timestamp ;
2006-12-31 06:01:01 +01:00
// early, ignore
2007-01-21 07:30:19 +01:00
if ( rdf . timestamp < timestamp ) {
continue ;
}
2006-12-31 06:01:01 +01:00
// this is the frame we want
2007-01-21 07:30:19 +01:00
last_fnum = n ;
//rdf.frame.Clear();
//rdf.frame = df.frame;
return rdf . frame ;
2006-12-31 06:01:01 +01:00
}
2007-01-01 04:29:20 +01:00
// Passed or end of file, seek back and try again
else if ( result = = 0 | | result = = 3 ) {
2006-12-31 06:36:20 +01:00
cur - = defd ;
goto seek ;
2007-01-01 04:29:20 +01:00
}
// Failed
else {
2007-01-21 07:30:19 +01:00
return AegiVideoFrame ( width , height ) ;
2006-12-31 06:01:01 +01:00
}
}
}
2006-12-20 01:31:52 +01:00
////////////////
// Refresh subs
void DirectShowVideoProvider : : RefreshSubtitles ( ) {
}
///////////////////
// Get float frame
void DirectShowVideoProvider : : GetFloatFrame ( float * Buffer , int n ) {
}
2007-01-01 04:29:20 +01:00
////////////////////////
// Override frame times
2008-03-07 22:24:14 +01:00
void DirectShowVideoProvider : : OverrideFrameTimeList ( Aegisub : : IntArray list ) {
2007-01-01 04:29:20 +01:00
frameTime = list ;
2008-03-07 22:24:14 +01:00
num_frames = frameTime . size ( ) ;
2007-01-01 04:29:20 +01:00
}
2007-01-23 05:42:08 +01:00
# endif
2007-12-31 07:46:22 +01:00
# endif // WITH_DIRECTSHOW