1
0
mirror of synced 2025-01-04 10:53:04 +01:00
OpenTaiko/FDK19/コード/04.グラフィック/CAvi.cs
2021-09-21 00:16:38 +02:00

307 lines
8.4 KiB
C#

using System;
using System.Collections.Generic;
using System.Text;
using System.Runtime.InteropServices;
using System.Drawing;
using SlimDX;
namespace FDK
{
public class CAvi : IDisposable
{
// プロパティ
public uint dwスケール
{
get;
set;
}
public uint dwレート
{
get;
set;
}
public uint nフレーム高さ
{
get;
set;
}
public uint nフレーム幅
{
get;
set;
}
// コンストラクタ
public CAvi( string filename )
{
if ( AVIFileOpen( out this.aviFile, filename, OpenFileFlags.OF_READ, IntPtr.Zero ) != 0 )
{
this.Release();
throw new Exception( "AVIFileOpen failed." );
}
if ( AVIFileGetStream( this.aviFile, out this.aviStream, streamtypeVIDEO, 0 ) != 0 )
{
this.Release();
throw new Exception( "AVIFileGetStream failed." );
}
var info = new AVISTREAMINFO();
AVIStreamInfo( this.aviStream, ref info, Marshal.SizeOf( info ) );
this.dwレート = info.dwRate;
this.dwスケール = info.dwScale;
this.nフレーム幅 = info.rcFrame.right - info.rcFrame.left;
this.nフレーム高さ = info.rcFrame.bottom - info.rcFrame.top;
try
{
this.frame = AVIStreamGetFrameOpen( this.aviStream, 0 );
}
catch
{
this.Release();
throw new Exception( "AVIStreamGetFrameOpen failed." );
}
}
// メソッド
public static void t初期化()
{
AVIFileInit();
}
public static void t終了()
{
AVIFileExit();
}
public Bitmap GetFrame( int no )
{
if( this.aviStream == IntPtr.Zero )
throw new InvalidOperationException();
return BitmapUtil.ToBitmap( AVIStreamGetFrame( this.frame, no ) );
}
public int GetFrameNoFromTime( int time )
{
return (int) ( time * ( ( (double) this.dwレート ) / ( 1000.0 * this.dwスケール ) ) );
}
public IntPtr GetFramePtr( int no )
{
if( this.aviStream == IntPtr.Zero )
throw new InvalidOperationException();
return AVIStreamGetFrame( this.frame, no );
}
public int GetMaxFrameCount()
{
if( this.aviStream == IntPtr.Zero )
throw new InvalidOperationException();
return AVIStreamLength( this.aviStream );
}
public unsafe void tBitmap24ToGraphicsStreamR5G6B5( BitmapUtil.BITMAPINFOHEADER* pBITMAPINFOHEADER, DataStream gs, int nWidth, int nHeight )
{
int nBmpWidth = pBITMAPINFOHEADER->biWidthビットマップの幅dot;
int nBmpHeight = pBITMAPINFOHEADER->biHeightビットマップの高さdot;
int nBmpLineByte = ( nBmpWidth * 3 ) + ( ( 4 - ( ( nBmpWidth * 3 ) % 4 ) ) % 4 );
ushort* pTexture = (ushort*) gs.DataPointer.ToPointer();
byte* pBitmap = (byte*) ( pBITMAPINFOHEADER + 1 );
for( int i = 0; i < nBmpHeight; i++ )
{
if( i >= nHeight )
break;
for( int j = 0; j < nBmpWidth; j++ )
{
if( j >= nWidth )
break;
ushort B = (ushort) ( ( *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 0 ) >> 3 ) & 0x1f );
ushort G = (ushort) ( ( *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 1 ) >> 2 ) & 0x3f );
ushort R = (ushort) ( ( *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 2 ) >> 3 ) & 0x1f );
*( pTexture + ( i * nWidth ) + j ) = (ushort) ( ( R << 11 ) | ( G << 5 ) | B );
}
}
}
public unsafe void tBitmap24ToGraphicsStreamX8R8G8B8( BitmapUtil.BITMAPINFOHEADER* pBITMAPINFOHEADER, DataStream ds, int nWidth, int nHeight )
{
int nBmpWidth = pBITMAPINFOHEADER->biWidthビットマップの幅dot;
int nBmpHeight = pBITMAPINFOHEADER->biHeightビットマップの高さdot;
int nBmpLineByte = ( nBmpWidth * 3 ) + ( ( 4 - ( ( nBmpWidth * 3 ) % 4 ) ) % 4 );
uint* pTexture = (uint*) ds.DataPointer.ToPointer();
byte* pBitmap = (byte*) ( pBITMAPINFOHEADER + 1 );
for( int i = 0; i < nBmpHeight; i++ )
{
if( i >= nHeight )
break;
for( int j = 0; j < nBmpWidth; j++ )
{
if( j >= nWidth )
break;
uint B = *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 0 );
uint G = *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 1 );
uint R = *( ( pBitmap + ( ( ( nBmpHeight - i ) - 1 ) * nBmpLineByte ) ) + ( j * 3 ) + 2 );
*( pTexture + ( i * nWidth ) + j ) = ( R << 16 ) | ( G << 8 ) | B;
}
}
}
#region [ Dispose-Finalize ]
//-----------------
public void Dispose()
{
this.Dispose( true );
GC.SuppressFinalize( this ); // 2011.8.19 from: 忘れてた。
}
protected void Dispose( bool disposeManagedObjects )
{
if( this.bDispose完了済み )
return;
if( disposeManagedObjects )
{
// (A) Managed リソースの解放
}
// (B) Unamanaged リソースの解放
if( this.frame != IntPtr.Zero )
AVIStreamGetFrameClose( this.frame );
this.Release();
this.bDispose完了済み = true;
}
~CAvi()
{
this.Dispose( false );
}
//-----------------
#endregion
// その他
#region [ Win32 AVI関連関数インポート ]
//-----------------
internal enum OpenFileFlags : uint
{
OF_CANCEL = 0x800,
OF_CREATE = 0x1000,
OF_DELETE = 0x200,
OF_EXIST = 0x4000,
OF_PARSE = 0x100,
OF_PROMPT = 0x2000,
OF_READ = 0,
OF_READWRITE = 2,
OF_REOPEN = 0x8000,
OF_SHARE_COMPAT = 0,
OF_SHARE_DENY_NONE = 0x40,
OF_SHARE_DENY_READ = 0x30,
OF_SHARE_DENY_WRITE = 0x20,
OF_SHARE_EXCLUSIVE = 0x10,
OF_VERIFY = 0x400,
OF_WRITE = 1
}
[StructLayout( LayoutKind.Sequential, Pack = 1 )]
internal struct AVISTREAMINFO
{
public uint fccType;
public uint fccHandler;
public uint dwFlags;
public uint dwCaps;
public ushort wPriority;
public ushort wLanguage;
public uint dwScale;
public uint dwRate;
public uint dwStart;
public uint dwLength;
public uint dwInitialFrames;
public uint dwSuggestedBufferSize;
public uint dwQuality;
public uint dwSampleSize;
public CAvi.RECT rcFrame;
public uint dwEditCount;
public uint dwFormatChangeCount;
[MarshalAs( UnmanagedType.ByValArray, SizeConst = 0x40 )]
public ushort[] szName;
}
[StructLayout( LayoutKind.Sequential, Pack = 1 )]
internal struct RECT
{
public uint left;
public uint top;
public uint right;
public uint bottom;
}
[DllImport( "AVIFIL32" )]
private static extern void AVIFileExit();
[DllImport( "AVIFIL32" )]
private static extern uint AVIFileGetStream( IntPtr pfile, out IntPtr ppavi, uint fccType, int lParam );
[DllImport( "AVIFIL32" )]
private static extern void AVIFileInit();
[DllImport( "AVIFIL32" )]
private static extern uint AVIFileOpen( out IntPtr ppfile, string szFile, OpenFileFlags mode, IntPtr pclsidHandler );
[DllImport( "AVIFIL32" )]
private static extern int AVIFileRelease( IntPtr pfile );
[DllImport( "AVIFIL32" )]
private static extern IntPtr AVIStreamGetFrame( IntPtr pgf, int lPos );
[DllImport( "AVIFIL32" )]
private static extern uint AVIStreamGetFrameClose( IntPtr pget );
[DllImport( "AVIFIL32" )]
private static extern IntPtr AVIStreamGetFrameOpen( IntPtr pavi, int lpbiWanted );
[DllImport( "AVIFIL32" )]
private static extern int AVIStreamInfo( IntPtr pavi, ref AVISTREAMINFO psi, int lSize );
[DllImport( "AVIFIL32" )]
private static extern int AVIStreamLength( IntPtr pavi );
[DllImport( "AVIFIL32" )]
private static extern int AVIStreamRelease( IntPtr pavi );
//-----------------
#endregion
#region [ private ]
//-----------------
private IntPtr aviFile = IntPtr.Zero;
private const string AVIFILE32 = "AVIFIL32";
private const int AVIGETFRAMEF_BESTDISPLAYFMT = 1;
private IntPtr aviStream = IntPtr.Zero;
private bool bDispose完了済み;
private IntPtr frame = IntPtr.Zero;
private static readonly uint streamtypeAUDIO = mmioFOURCC( 'a', 'u', 'd', 's' );
private static readonly uint streamtypeMIDI = mmioFOURCC( 'm', 'i', 'd', 's' );
private static readonly uint streamtypeTEXT = mmioFOURCC( 't', 'x', 't', 's' );
private static readonly uint streamtypeVIDEO = mmioFOURCC( 'v', 'i', 'd', 's' );
private static uint mmioFOURCC( char c0, char c1, char c2, char c3 )
{
return ( (uint) c3 << 0x18 ) | ( (uint) c2 << 0x10 ) | ( (uint) c1 << 0x08 ) | (uint) c0;
}
private void Release()
{
if( this.aviStream != IntPtr.Zero )
{
AVIStreamRelease( this.aviStream );
this.aviStream = IntPtr.Zero;
}
if( this.aviFile != IntPtr.Zero )
{
AVIFileRelease( this.aviFile );
this.aviFile = IntPtr.Zero;
}
}
//-----------------
#endregion
}
}