C#写的一个视频转换解码器
C#写的一个视频转换解码器
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Drawing;
using System.Runtime.InteropServices;
using System.Threading;
using System.Windows.Forms;
using System.IO; namespace Basic
{
public class VideoAPI //视频API类
{
// 视频API调用
[DllImport("avicap32.dll")]//包含了执行视频捕获的函数,它给AVI文件I/O和视频、音频设备驱动程序提供一个高级接口
public static extern IntPtr capCreateCaptureWindow(string lpszWindowName, int dwStyle, int x, int y, int nWidth, int nHeight, IntPtr hwndParent, int nID);
[DllImport("AVICAP32.dll", CharSet = CharSet.Unicode)]
public static extern bool capGetDriverDescription(int wDriverIndex, StringBuilder lpszName, int cbName, StringBuilder lpszVer, int cbVer);
[DllImport("avicap32.dll")]
public static extern IntPtr capCreateCaptureWindowA(byte[] lpszWindowName, int dwStyle, int x, int y, int nWidth, int nHeight, IntPtr hWndParent, int nID);
[DllImport("avicap32.dll")]
public static extern bool capGetDriverDescriptionA(short wDriver, byte[] lpszName, int cbName, byte[] lpszVer, int cbVer);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, bool wParam, int lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, short wParam, int lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, int wParam, int lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, short wParam, FrameEventHandler lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, int wParam, ref BITMAPINFO lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, int wParam, ref CAPDRIVERCAPS lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, int wParam, ref CAPTUREPARMS lParam);
[DllImport("User32.dll")]
public static extern bool SendMessage(IntPtr hWnd, int wMsg, int wParam, ref CAPSTATUS lParam);
[DllImport("User32.dll")]
public static extern int SetWindowPos(IntPtr hWnd, int hWndInsertAfter, int x, int y, int cx, int cy, int wFlags);
[DllImport("avicap32.dll")]
public static extern int capGetVideoFormat(IntPtr hWnd, IntPtr psVideoFormat, int wSize); // 常量
// public const int WM_USER = 0x400;
public const int WS_CHILD = 0x40000000;
public const int WS_VISIBLE = 0x10000000; public const int SWP_NOMOVE = 0x2;
public const int SWP_NOZORDER = 0x4;
// public const int WM_CAP_DRIVER_CONNECT = WM_USER + 10;
// public const int WM_CAP_DRIVER_DISCONNECT = WM_USER + 11;
public const int WM_CAP_SET_CALLBACK_FRAME = WM_USER + ;
// public const int WM_CAP_SET_PREVIEW = WM_USER + 50;
// public const int WM_CAP_SET_PREVIEWRATE = WM_USER + 52;
// public const int WM_CAP_SET_VIDEOFORMAT = WM_USER + 45;
// public const int WM_CAP_START = WM_USER;
public const int WM_CAP_SAVEDIB = WM_CAP_START + ; public const string avicap32 = "avicap32.dll";
public const int WM_USER = ;
/// <summary>
///WM_CAP_START=WM_USER=1024
/// </summary>
public const int WM_CAP_START = WM_USER; // start of unicode messages
/// <summary>
/// 开始 WM_USER + 100=1124
/// </summary>
public const int WM_CAP_UNICODE_START = WM_USER + ; //开始 1124
/// <summary>
/// /获得 CAPSTR EAMPTR
/// WM_CAP_START + 1=1025
/// </summary>
public const int WM_CAP_GET_CAPSTREAMPTR = (WM_CAP_START + ); //获得 CAPSTR EAMPTR
/// <summary>
/// 设置收回错误 WM_CAP_START + 2=1026
/// </summary>
public const int WM_CAP_SET_CALLBACK_ERROR = (WM_CAP_START + ); //设置收回错误
/// <summary>
/// 设置收回状态 WM_CAP_START + 3=1027
/// </summary>
public const int WM_CAP_SET_CALLBACK_STATUS = (WM_CAP_START + ); //设置收回状态
/// <summary>
/// 设置收回出产 WM_CAP_START + 4=1028
/// </summary>
public const int WM_CAP_SET_CALLBACK_YIELD = (WM_CAP_START + ); //设置收回出产
/// <summary>
/// 设置收回结构 WM_CAP_START + 5=1029
/// </summary>
public const int WM_CAP_SET_CALLBACK_FRame = (WM_CAP_START + ); //设置收回结构
/// <summary>
/// 设置收回视频流 WM_CAP_START + 6=1030
/// </summary>
public const int WM_CAP_SET_CALLBACK_VIDEOSTREAM = (WM_CAP_START + ); //设置收回视频流
/// <summary>
/// 设置收回视频波流 WM_CAP_START +7=1031
/// </summary>
public const int WM_CAP_SET_CALLBACK_WAVESTREAM = (WM_CAP_START + ); //设置收回视频波流
/// <summary>
/// 获得使用者数据 WM_CAP_START + 8=1032
/// </summary>
public const int WM_CAP_GET_USER_DATA = (WM_CAP_START + ); //获得使用者数据
/// <summary>
/// 设置使用者数据 WM_CAP_START + 9=1033
/// </summary>
public const int WM_CAP_SET_USER_DATA = (WM_CAP_START + ); //设置使用者数据
/// <summary>
/// 驱动程序连接 WM_CAP_START + 10=1034
/// </summary>
public const int WM_CAP_DRIVER_CONNECT = (WM_CAP_START + ); //驱动程序连接
/// <summary>
/// 断开启动程序连接 WM_CAP_START + 11=1035
/// </summary>
public const int WM_CAP_DRIVER_DISCONNECT = (WM_CAP_START + ); //断开启动程序连接
/// <summary>
/// 获得驱动程序名字 WM_CAP_START + 12=1036
/// </summary>
public const int WM_CAP_DRIVER_GET_NAME = (WM_CAP_START + ); //获得驱动程序名字
/// <summary>
/// 获得驱动程序版本 WM_CAP_START + 13=1037
/// </summary>
public const int WM_CAP_DRIVER_GET_VERSION = (WM_CAP_START + ); //获得驱动程序版本
/// <summary>
/// 获得驱动程序帽子 WM_CAP_START + 14=1038
/// </summary>
public const int WM_CAP_DRIVER_GET_CAPS = (WM_CAP_START + ); //获得驱动程序帽子
/// <summary>
/// 设置捕获文件 WM_CAP_START + 20=1044
/// </summary>
public const int WM_CAP_FILE_SET_CAPTURE_FILE = (WM_CAP_START + ); //设置捕获文件
/// <summary>
/// 获得捕获文件 WM_CAP_START + 21=1045
/// </summary>
public const int WM_CAP_FILE_GET_CAPTURE_FILE = (WM_CAP_START + ); //获得捕获文件
/// <summary>
/// 另存文件为 WM_CAP_START + 23=1047
/// </summary>
public const int WM_CAP_FILE_SAVEAS = (WM_CAP_START + ); //另存文件为
/// <summary>
/// 保存文件 WM_CAP_START + 25=1049
/// </summary>
public const int WM_CAP_FILE_SAVEDIB = (WM_CAP_START + ); //保存文件 // out of order to save on ifdefs
/// <summary>
/// 分派文件 WM_CAP_START + 22=1044
/// </summary>
public const int WM_CAP_FILE_ALLOCATE = (WM_CAP_START + ); //分派文件
/// <summary>
/// 设置开始文件 WM_CAP_START + 24=1046
/// </summary>
public const int WM_CAP_FILE_SET_INFOCHUNK = (WM_CAP_START + ); //设置开始文件
/// <summary>
/// 编辑复制 WM_CAP_START + 30=1054
/// </summary>
public const int WM_CAP_EDIT_COPY = (WM_CAP_START + ); //编辑复制
/// <summary>
/// 设置音频格式 WM_CAP_START + 35=1059
/// </summary>
public const int WM_CAP_SET_AUDIOFORMAT = (WM_CAP_START + ); //设置音频格式
/// <summary>
/// 捕获音频格式 WM_CAP_START + 36=1060
/// </summary>
public const int WM_CAP_GET_AUDIOFORMAT = (WM_CAP_START + ); //捕获音频格式
/// <summary>
/// 打开视频格式设置对话框 WM_CAP_START + 41=1065
/// </summary>
public const int WM_CAP_DLG_VIDEOFORMAT = (WM_CAP_START + ); //1065 打开视频格式设置对话框
/// <summary>
/// 打开属性设置对话框,设置对比度、亮度等 WM_CAP_START + 42=1066
/// </summary>
public const int WM_CAP_DLG_VIDEOSOURCE = (WM_CAP_START + ); //1066 打开属性设置对话框,设置对比度、亮度等。
/// <summary>
/// 打开视频显示 WM_CAP_START + 43=1067
/// </summary>
public const int WM_CAP_DLG_VIDEODISPLAY = (WM_CAP_START + ); //1067 打开视频显示
/// <summary>
/// 获得视频格式 WM_CAP_START + 44=1068
/// </summary>
public const int WM_CAP_GET_VIDEOFORMAT = (WM_CAP_START + ); //1068 获得视频格式
/// <summary>
/// 设置视频格式 WM_CAP_START + 45=1069
/// </summary>
public const int WM_CAP_SET_VIDEOFORMAT = (WM_CAP_START + ); //1069 设置视频格式
/// <summary>
/// 打开压缩设置对话框 WM_CAP_START + 46=1070
/// </summary>
public const int WM_CAP_DLG_VIDEOCOMPRESSION = (WM_CAP_START + ); //1070 打开压缩设置对话框
/// <summary>
/// 设置预览 WM_CAP_START + 50=1074
/// </summary>
public const int WM_CAP_SET_PREVIEW = (WM_CAP_START + ); //设置预览
/// <summary>
/// 设置覆盖 WM_CAP_START + 51=1075
/// </summary>
public const int WM_CAP_SET_OVERLAY = (WM_CAP_START + ); //设置覆盖
/// <summary>
/// 设置预览比例 WM_CAP_START + 52=1076
/// </summary>
public const int WM_CAP_SET_PREVIEWRATE = (WM_CAP_START + ); //设置预览比例
/// <summary>
/// 设置刻度 WM_CAP_START + 53=1077
/// </summary>
public const int WM_CAP_SET_SCALE = (WM_CAP_START + ); //设置刻度
/// <summary>
/// 获得状态 WM_CAP_START + 54=1078
/// </summary>
public const int WM_CAP_GET_STATUS = (WM_CAP_START + ); //获得状态
/// <summary>
/// 设置卷 WM_CAP_START + 55=1079
/// </summary>
public const int WM_CAP_SET_SCROLL = (WM_CAP_START + ); //设置卷
/// <summary>
/// 逮捕结构 WM_CAP_START + 60=1084
/// </summary>
public const int WM_CAP_GRAB_FRame = (WM_CAP_START + ); //逮捕结构
/// <summary>
/// 停止逮捕结构 WM_CAP_START + 61=1085
/// </summary>
public const int WM_CAP_GRAB_FRame_NOSTOP = (WM_CAP_START + ); //停止逮捕结构
/// <summary>
/// 次序 WM_CAP_START + 62=1086
/// </summary>
public const int WM_CAP_SEQUENCE = (WM_CAP_START + ); //次序
/// <summary>
/// 没有文件 WM_CAP_START + 63=1087
/// </summary>
public const int WM_CAP_SEQUENCE_NOFILE = (WM_CAP_START + ); //没有文件
/// <summary>
/// 设置安装次序 WM_CAP_START + 64=1088
/// </summary>
public const int WM_CAP_SET_SEQUENCE_SETUP = (WM_CAP_START + ); //设置安装次序
/// <summary>
/// 获得安装次序 WM_CAP_START + 65=1089
/// </summary>
public const int WM_CAP_GET_SEQUENCE_SETUP = (WM_CAP_START + ); //获得安装次序
/// <summary>
/// 设置媒体控制接口 WM_CAP_START + 66=1090
/// </summary>
public const int WM_CAP_SET_MCI_DEVICE = (WM_CAP_START + ); //设置媒体控制接口
/// <summary>
/// 获得媒体控制接口 WM_CAP_START + 67=1091
/// </summary>
public const int WM_CAP_GET_MCI_DEVICE = (WM_CAP_START + ); //获得媒体控制接口
/// <summary>
/// 停止 WM_CAP_START + 68=1092
/// </summary>
public const int WM_CAP_STOP = (WM_CAP_START + ); //停止
/// <summary>
/// 异常中断 WM_CAP_START + 69=1093
/// </summary>
public const int WM_CAP_ABORT = (WM_CAP_START + ); //异常中断
/// <summary>
/// 打开单一的结构 WM_CAP_START + 68=1094
/// </summary>
public const int WM_CAP_SINGLE_FRame_OPEN = (WM_CAP_START + ); //打开单一的结构
/// <summary>
/// 关闭单一的结构 WM_CAP_START + 71=1095
/// </summary>
public const int WM_CAP_SINGLE_FRame_CLOSE = (WM_CAP_START + ); //关闭单一的结构
/// <summary>
/// 单一的结构 WM_CAP_START + 72=1096
/// </summary>
public const int WM_CAP_SINGLE_FRame = (WM_CAP_START + ); //单一的结构
/// <summary>
/// 打开视频 WM_CAP_START + 80=1104
/// </summary>
public const int WM_CAP_PAL_OPEN = (WM_CAP_START + ); //打开视频
/// <summary>
/// 保存视频 WM_CAP_START + 81=1105
/// </summary>
public const int WM_CAP_PAL_SAVE = (WM_CAP_START + ); //保存视频
/// <summary>
/// 粘贴视频 WM_CAP_START + 82=1106
/// </summary>
public const int WM_CAP_PAL_PASTE = (WM_CAP_START + ); //粘贴视频
/// <summary>
/// 自动创造 WM_CAP_START + 83=1107
/// </summary>
public const int WM_CAP_PAL_AUTOCREATE = (WM_CAP_START + ); //自动创造
/// <summary>
/// 手动创造 WM_CAP_START + 84=1108
/// </summary>
public const int WM_CAP_PAL_MANUALCREATE = (WM_CAP_START + ); //手动创造 // Following added post VFW 1.1
/// <summary>
/// 设置收回的错误 WM_CAP_START + 85=1109
/// </summary>
public const int WM_CAP_SET_CALLBACK_CAPCONTROL = (WM_CAP_START + ); // 设置收回的错误 public const int WM_CAP_END = WM_CAP_SET_CALLBACK_CAPCONTROL; public delegate void FrameEventHandler(IntPtr lwnd, IntPtr lpVHdr); #region 公共函数
//公共函数
public static object GetStructure(IntPtr ptr, ValueType structure)
{
return Marshal.PtrToStructure(ptr, structure.GetType());
} public static object GetStructure(int ptr, ValueType structure)
{
return GetStructure(new IntPtr(ptr), structure);
} public static void Copy(IntPtr ptr, byte[] data)
{
Marshal.Copy(ptr, data, , data.Length);
} public static void Copy(int ptr, byte[] data)
{
Copy(new IntPtr(ptr), data);
} public static int SizeOf(object structure)
{
return Marshal.SizeOf(structure);
}
#endregion 公共函数
#region 结构 VIDEOHDR|BITMAPINFOHEADER|BITMAPINFO|CAPTUREPARMS|CAPDRIVERCAPS|CAPSTATUS
//========================================================VideoHdr 结构=====================================================================
//VideoHdr 结构 定义了视频数据块的头信息,在编写回调函数时常用到其数据成员lpData(指向数据缓存的指针)和dwBufferLength(数据缓存的大小)。
//视频帧到缓存的捕获则需要应用回调函数和相应的数据块结构 VIDEOHDR
[StructLayout(LayoutKind.Sequential)]
public struct VIDEOHDR
{
public IntPtr lpData; /* 指向数据缓存的指针 */
public int dwBufferLength; /* 数据缓存的大小 */
public int dwBytesUsed; /* Bytes actually used */
public int dwTimeCaptured; /* Milliseconds from start of stream */
public int dwUser; /* for client's use */
public int dwFlags; /* assorted flags (see defines) */
public int dwReserved; /* reserved for driver */
}
//=======================================================BitmapInfoHeader结构===================================================================
//BitmapInfoHeader定义了位图的头部信息
[StructLayout(LayoutKind.Sequential)]
public struct BITMAPINFOHEADER
{
public int biSize;
public int biWidth;
public int biHeight;
public short biPlanes;
public short biBitCount;
public int biCompression;
public int biSizeImage;
public int biXPelsPerMeter;
public int biYPelsPerMeter;
public int biClrUsed;
public int biClrImportant;
}
//========================================================================================================================================= //======================================================BitmapInfo结构=====================================================================
//BitmapInfo 位图信息
[StructLayout(LayoutKind.Sequential)]
public struct BITMAPINFO
{
public BITMAPINFOHEADER bmiHeader;
public int bmiColors;
} //=====================================================CAPTUREPARMS结构======================================================================
//CAPTUREPARMS 包含控制视频流捕获过程的参数,如捕获帧频、指定键盘或鼠标键以终止捕获、捕获时间限制等;
[StructLayout(LayoutKind.Sequential)]
public struct CAPTUREPARMS
{
public int dwRequestMicroSecPerFrame; // 期望的桢播放率,以毫秒为单位,默认为66667,相当于15桢每秒。
public bool fMakeUserHitOKToCapture; // Show "Hit OK to cap" dlg?开始捕获标志位,如果值为真,则在开始捕获前要产生一个询问对话框,默认为假。
public uint wPercentDropForError; //所允许的最大丢桢百分比,可以从0变化到100,默认值为10。
public bool fYield; /*另起线程标志位,如果为真,则程序重新启动一个线程用于视频流的捕获,默认值是假。
但是如果你是为了真,你必须要在程序中处理一些潜在的操作,因为当视频捕获时,其他操作并没有被屏蔽。*/
public int dwIndexSize; // 在AVI文件中所允许的最大数目的索引项(32K)
public uint wChunkGranularity; // AVI文件的逻辑尺寸,以字节为单位。如果值是0,则说明该尺寸渐增 在 Win32程序中无用。(2K)
public bool fUsingDOSMemory; // Use DOS buffers?
public uint wNumVideoRequested; // 所被允许分配的最大视频缓存
public bool fCaptureAudio; // 音频标志位,如果音频流正在捕获,则该值为真。
public uint wNumAudioRequested; // 最大数量的音频缓存,默认值为10。
public uint vKeyAbort; // 终止流捕获的虚拟键盘码,默认值为VK_ESCAPE
[MarshalAs(UnmanagedType.Bool)]
public bool fAbortLeftMouse; // 终止鼠标左键标志位,如果该值为真,则在流捕获过程中如果点击鼠标左键则该捕获终止,默认值为真。
public bool fAbortRightMouse; // Abort on right mouse?
public bool fLimitEnabled; // 捕获操作时间限制,如果为真,则时间到了以后捕获操作终止,默认为假
public uint wTimeLimit; // 具体终止时间,只有 fLimitEnabled是真时.该位才有效
public bool fMCIControl; // Use MCI video source?
public bool fStepMCIDevice; // Step MCI device?MCI 设备标志。
public int dwMCIStartTime; // Time to start in MS
public int dwMCIStopTime; // Time to stop in MS
public bool fStepCaptureAt2x; // Perform spatial averaging 2x
public int wStepCaptureAverageFrames; // 当基于平均采样来创建桢时,桢的采样时间,典型值是5
public int dwAudioBufferSize; // 音频缓存的尺寸,如果用默认值0,缓存尺寸是最大0.5秒,或10k字节。
public int fDisableWriteCache; // Attempt to disable write cache
public int AVStreamMaster; //音视频同步标志。
}
//========================================================================================================================================= //=================================================CAPDRIVERCAPS结构=======================================================================
//CAPDRIVERCAPS定义了捕获驱动器的能力,如有无视频叠加能力、有无控制视频源、视频格式的对话框等;
[StructLayout(LayoutKind.Sequential)]
public struct CAPDRIVERCAPS
{
[MarshalAs(UnmanagedType.U2)]
public UInt16 wDeviceIndex; //捕获驱动器的索引值,该值可以由0到9变化。
[MarshalAs(UnmanagedType.Bool)]
public bool fHasOverlay; // 视频叠加标志,如果设备支持视频叠加这该位是真。
[MarshalAs(UnmanagedType.Bool)]
public bool fHasDlgVideoSource; //视频资源对话框标志位,如果设备支持视频选择、控制对话框,该值为真。
[MarshalAs(UnmanagedType.Bool)]
public bool fHasDlgVideoFormat; //视频格式对话框标志位,如果设备支持对视频格式对话框的选择,该位真。
[MarshalAs(UnmanagedType.Bool)]
public bool fHasDlgVideoDisplay; //视频展示对话框标志位,如果设备支持对视频捕获缓存区的重新播放,该位是真。
[MarshalAs(UnmanagedType.Bool)]
public bool fCaptureInitialized; //捕获安装标志位,如果捕获驱动器已经成功连接,该值为真。
//[MarshalAs(UnmanagedType.Bool)]
public bool fDriverSuppliesPalettes; //驱动器调色板标志位,如果驱动器能创建调色板,则该位是真。
[MarshalAs(UnmanagedType.I4)]
public int hVideoIn;
[MarshalAs(UnmanagedType.I4)]
public int hVideoOut;
[MarshalAs(UnmanagedType.I4)]
public int hVideoExtIn;
[MarshalAs(UnmanagedType.I4)]
public int hVideoExtOut;
}
//========================================================================================================================================= //=====================================================CAPSTATUS结构========================================================================
//CAPSTATUS定义了捕获窗口的当前状态,如图像的宽、高等;
[StructLayout(LayoutKind.Sequential)]
public struct CAPSTATUS
{
public int uiImageWidth; //图像宽度
public int uiImageHeight; //图像高度
public bool fLiveWindow; //活动窗口标记,如果窗口正以预览的方式展示图像,那么该值为真
public bool fOverlayWindow; //叠加窗口标志位,如果正在使用硬件叠加,则该位是真。
public bool fScale; //输入所放标志位,如果窗口是正在缩放视频到客户区,那么该位是真。当使用硬件叠加时,改位无效。
public Point ptScroll; //被展示在窗口客户区左上角的那个象素的x、y坐标偏移量。
public bool fUsingDefaultPalette; //默认调色板标志位,如果捕获窗口正在使用当前默认调色板,该值为真
public bool fAudioHardware; // 音频硬件标志位,如果系统已经安装了音频硬件,该值为真。
public bool fCapFileExists; //捕获文件标志位,如果一个捕获文件已经被创建,该值为真
public int dwCurrentVideoFrame; // 当前或最近流捕获过程中,所处理的桢的数目。包括丢弃的桢。
public int dwCurrentVideoFramesDropped;//当前流捕获过程中丢弃的桢的数目。
public int dwCurrentWaveSamples; // # of wave samples cap'td
public int dwCurrentTimeElapsedMS; // 从当前流捕获开始计算,程序所用的时间,以毫秒为单位。
public IntPtr hPalCurrent; // 当前剪切板的句柄。
public bool fCapturingNow; // 捕获标志位,当捕获是正在进行时,改位是真
public int dwReturn; // 错误返回值,当你的应用程序不支持错误回调函数时可以应用改位
public int wNumVideoAllocated; // 被分配的视频缓存的数目。
public int wNumAudioAllocated; // 被分配的音频缓存的数目。
}
//========================================================================================================================================= #endregion 结构 VIDEOHDR|BITMAPINFOHEADER|BITMAPINFO|CAPTUREPARMS|CAPDRIVERCAPS|CAPSTATUS }
public class cVideo //视频类
{
public bool flag = true;
private IntPtr lwndC; //保存无符号句柄
private IntPtr mControlPtr; //保存管理指示器
private int mWidth;
private int mHeight;
public delegate void RecievedFrameEventHandler(byte[] data);
public event RecievedFrameEventHandler RecievedFrame; public VideoAPI.CAPTUREPARMS Capparms;
private VideoAPI.FrameEventHandler mFrameEventHandler;
public VideoAPI.CAPDRIVERCAPS CapDriverCAPS;//捕获驱动器的能力,如有无视频叠加能力、有无控制视频源、视频格式的对话框等;
public VideoAPI.CAPSTATUS CapStatus;//该结构用于保存视频设备捕获窗口的当前状态,如图像的宽、高等
string strFileName;
public cVideo(IntPtr handle, int width, int height)
{
CapDriverCAPS = new VideoAPI.CAPDRIVERCAPS();//捕获驱动器的能力,如有无视频叠加能力、有无控制视频源、视频格式的对话框等;
CapStatus = new VideoAPI.CAPSTATUS();//该结构用于保存视频设备捕获窗口的当前状态,如图像的宽、高等
mControlPtr = handle; //显示视频控件的句柄
mWidth = width; //视频宽度
mHeight = height; //视频高度
}
public bool StartWebCam()
{
return StartWebCam(mWidth, mHeight);
}
/// <summary>
/// 打开视频设备
/// </summary>
/// <param name="width">捕获窗口的宽度</param>
/// <param name="height">捕获窗口的高度</param>
/// <returns></returns>
public bool StartWebCam(int width,int height)
{
//byte[] lpszName = new byte[100];
//byte[] lpszVer = new byte[100];
//VideoAPI.capGetDriverDescriptionA(0, lpszName, 100, lpszVer, 100);
//this.lwndC = VideoAPI.capCreateCaptureWindowA(lpszName, VideoAPI.WS_CHILD | VideoAPI.WS_VISIBLE, 0, 0, mWidth, mHeight, mControlPtr, 0);
//if (VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_DRIVER_CONNECT, 0, 0))
//{
// VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_SET_PREVIEWRATE, 100, 0);
// VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_SET_PREVIEW, true, 0);
// return true;
//}
//else
//{
// return false;
//}
this.lwndC = VideoAPI.capCreateCaptureWindow("", VideoAPI.WS_CHILD | VideoAPI.WS_VISIBLE, , , mWidth, mHeight, mControlPtr, );//AVICap类的捕捉窗口
VideoAPI.FrameEventHandler FrameEventHandler = new VideoAPI.FrameEventHandler(FrameCallback);
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_ERROR, , );//注册错误回调函数
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_STATUS, , );//注册状态回调函数
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_VIDEOSTREAM, , );//注册视频流回调函数
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_FRAME, , FrameEventHandler);//注册帧回调函数 //if (!CapDriverCAPS.fCaptureInitialized)//判断当前设备是否被其他设备连接已经连接
//{ if (VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_DRIVER_CONNECT, , ))
{
//-----------------------------------------------------------------------
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_DRIVER_GET_CAPS, VideoAPI.SizeOf(CapDriverCAPS), ref CapDriverCAPS);//获得当前视频 CAPDRIVERCAPS定义了捕获驱动器的能力,如有无视频叠加能力、有无控制视频源、视频格式的对话框等;
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_GET_STATUS, VideoAPI.SizeOf(CapStatus), ref CapStatus);//获得当前视频流的尺寸 存入CapStatus结构 VideoAPI.BITMAPINFO bitmapInfo = new VideoAPI.BITMAPINFO();//设置视频格式 (height and width in pixels, bits per frame).
bitmapInfo.bmiHeader = new VideoAPI.BITMAPINFOHEADER();
bitmapInfo.bmiHeader.biSize = VideoAPI.SizeOf(bitmapInfo.bmiHeader);
bitmapInfo.bmiHeader.biWidth = mWidth;
bitmapInfo.bmiHeader.biHeight = mHeight;
bitmapInfo.bmiHeader.biPlanes = ;
bitmapInfo.bmiHeader.biBitCount = ;
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_PREVIEWRATE, , );//设置在PREVIEW模式下设定视频窗口的刷新率 设置每40毫秒显示一帧,即显示帧速为每秒25帧
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_SCALE, , );//打开预览视频的缩放比例
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_VIDEOFORMAT, VideoAPI.SizeOf(bitmapInfo), ref bitmapInfo); this.mFrameEventHandler = new VideoAPI.FrameEventHandler(FrameCallback);
this.capSetCallbackOnFrame(this.lwndC, this.mFrameEventHandler); VideoAPI.CAPTUREPARMS captureparms = new VideoAPI.CAPTUREPARMS();
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_GET_SEQUENCE_SETUP, VideoAPI.SizeOf(captureparms), ref captureparms);
if (CapDriverCAPS.fHasOverlay)
{
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_OVERLAY, , );//启用叠加 注:据说启用此项可以加快渲染速度
}
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_PREVIEW, , );//设置显示图像启动预览模式 PREVIEW
VideoAPI.SetWindowPos(this.lwndC, , , , width, height, VideoAPI.SWP_NOZORDER | VideoAPI.SWP_NOMOVE);//使捕获窗口与进来的视频流尺寸保持一致
return true;
}
else
{ flag = false;
return false;
}
}
public bool ReSizePic(int width, int height)
{
try
{
// CloseWebcam();
// StartWebCam();
// this.lwndC = VideoAPI.capCreateCaptureWindow("", VideoAPI.WS_CHILD | VideoAPI.WS_VISIBLE, 0, 0, mWidth, mHeight, mControlPtr, 0); // VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_GET_STATUS, VideoAPI.SizeOf(CapStatus), ref CapStatus);//获得当前视频流的尺寸 存入CapStatus结构
// VideoAPI.FrameEventHandler FrameEventHandler = new VideoAPI.FrameEventHandler(FrameCallback);
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_ERROR, 0, 0);//注册错误回调函数
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_STATUS, 0, 0);//注册状态回调函数
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_VIDEOSTREAM, 0, 0);//注册视频流回调函数
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_FRAME, 0, FrameEventHandler);//注册帧回调函数
//// if (VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_DRIVER_CONNECT, 0, 0))
// {
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_SCALE, 1, 0);
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_PREVIEWRATE, 40, 0);
// VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_PREVIEW, 1, 0); return VideoAPI.SetWindowPos(this.lwndC, , , , width, height, VideoAPI.SWP_NOZORDER | VideoAPI.SWP_NOMOVE) > ;
// }
// return false;
}
catch
{
return false;
}
}
public void get()
{
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_GET_SEQUENCE_SETUP, VideoAPI.SizeOf(Capparms), ref Capparms);
}
public void set()
{
VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_SEQUENCE_SETUP, VideoAPI.SizeOf(Capparms), ref Capparms);
}
private bool capSetCallbackOnFrame(IntPtr lwnd, VideoAPI.FrameEventHandler lpProc)
{
return VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SET_CALLBACK_FRAME, , lpProc);
}
/// <summary>
/// 关闭视频设备
/// </summary>
public void CloseWebcam()
{
VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_DRIVER_DISCONNECT, , );
}
/// <summary>
/// 拍照
/// </summary>
/// <param name="path">要保存bmp文件的路径</param>
public void GrabImage(IntPtr hWndC, string path)
{
IntPtr hBmp = Marshal.StringToHGlobalAnsi(path);
VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_SAVEDIB, , hBmp.ToInt32());
}
public bool StarKinescope(string path)
{
try
{
strFileName = path;
string dir = path.Remove(path.LastIndexOf("\\"));
if (!File.Exists(dir))
{
Directory.CreateDirectory(dir);
}
int hBmp = Marshal.StringToHGlobalAnsi(path).ToInt32();
bool b = VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_FILE_SET_CAPTURE_FILE, , hBmp);
b = b && VideoAPI.SendMessage(this.lwndC, VideoAPI.WM_CAP_SEQUENCE, , );
return b;
}
catch
{
return false;
}
}
/// <summary>
/// 停止录像
/// </summary>
public bool StopKinescope()
{
return VideoAPI.SendMessage(lwndC, VideoAPI.WM_CAP_STOP, , );
}
private void FrameCallback(IntPtr lwnd, IntPtr lpvhdr)
{
VideoAPI.VIDEOHDR videoHeader = new VideoAPI.VIDEOHDR();
byte[] VideoData;
videoHeader = (VideoAPI.VIDEOHDR)VideoAPI.GetStructure(lpvhdr, videoHeader);
VideoData = new byte[videoHeader.dwBytesUsed];
VideoAPI.Copy(videoHeader.lpData, VideoData);
if (this.RecievedFrame != null)
this.RecievedFrame(VideoData);
}
private Thread myThread; public void CompressVideoFfmpeg()
{
myThread = new Thread(new ThreadStart(testfn));
myThread.Start();
testfn();
}
private void testfn()
{
string file_name = strFileName;
string command_line = " -i " + file_name + " -vcodec libx264 -cqp 25 -y " + file_name.Replace(".avi", "_264") + ".avi";
System.Diagnostics.Process proc = new System.Diagnostics.Process();
proc.StartInfo.WorkingDirectory = Application.StartupPath;
proc.StartInfo.UseShellExecute = false; //use false if you want to hide the window
proc.StartInfo.CreateNoWindow = true;
proc.StartInfo.FileName = "ffmpeg";
proc.StartInfo.Arguments = command_line;
proc.Start();
proc.WaitForExit();
proc.Close();
Common.writeLog("视频"+strFileName+"压缩完毕");
// 删除原始avi文件
FileInfo file = new FileInfo(file_name);
if (file.Exists)
{
try
{
file.Delete(); //删除单个文件
}
catch (Exception e)
{
// Common.writeLog("删除视频文件“" + file_name + "”出错!准备" ); (new Thread(() =>
{
bool b = false;
for (int i = ; i < ; i++)
{
Thread.Sleep(*(i*i));
try
{
file.Delete();
b = true;
break;
}
catch
{
}
};
if (!b)
Common.writeLog("删除视频文件“" + file_name + "”出错");
}
)).Start();
}
}
//myThread.Abort();
}
}
}
C#写的一个视频转换解码器的更多相关文章
- sql server 关于表中只增标识问题 C# 实现自动化打开和关闭可执行文件(或 关闭停止与系统交互的可执行文件) ajaxfileupload插件上传图片功能,用MVC和aspx做后台各写了一个案例 将小写阿拉伯数字转换成大写的汉字, C# WinForm 中英文实现, 国际化实现的简单方法 ASP.NET Core 2 学习笔记(六)ASP.NET Core 2 学习笔记(三)
sql server 关于表中只增标识问题 由于我们系统时间用的过长,数据量大,设计是采用自增ID 我们插入数据的时候把ID也写进去,我们可以采用 关闭和开启自增标识 没有关闭的时候 ,提示一下错 ...
- 这两天自己模仿写的一个Asp.Net的显示分页方法 附加实体转换和存储过程
之前自己一直用Aspnetpager控件来显示项目中的分页,但是每次都要拖一个aspnetpager的控件进去,感觉很不舒服,因为现在自己写的webform都不用服务器控件了,所以自己仿照aspnet ...
- C# 利用ffmpeg 对视频转换系类操作 (1) 基本分析
最近公司做一个项目,开发一个视频站点.项目需求中有很多视频转换的需求,如:格式转换(flv,Mp4),视频水印,视频截图,视频合成,获取视频的基本信息(时间戳,视频大小等).经过网络的收集资料以及自己 ...
- Linux下视频转换工具:转换,切割,连接,
首先当然是需要安装ffmpeg软件包,可以直接从源中进行安装!但我安装后并不能成功执行后面所需要执行的转换命令,所以我只能重新从源码编译安装ffmpeg: (1)下载ffmpeg源码包,注意版本不能太 ...
- Java+Windows+ffmpeg实现视频转换
最近由于项目需要,研究了一下如何用Java实现视频转换,“着实”废了点心思,整理整理,写出给自己备忘下. 思路 由于之前没有没法过相关功能的经验,一开始来真不知道从哪里入手.当然,这个解决,googl ...
- 视频转换ffmpeg
使用yum在centos下安装ffmpeg ffmpeg -i IMG_1893.MOV -ab 56 -ar 22050 -b 500 -r 15 -s 640x480 test.mp4 说 ...
- <开源项目分析>Cisco的开源视频加解码器THOR(H.264解码)
原创博客,转载请联系博主! 题外话:自学了快两个月的Perl语言,本来打算写两篇基础介绍的博文来科普一下一些小技巧,但是仔细想想还是没有必要了吧,毕竟现在无论是在用Perl5还是Perl6的人都是小众 ...
- C#调用mmpeg进行各种视频转换的类实例
本文实例讲述了C#调用mmpeg进行各种视频转换的类.分享给大家供大家参考.具体如下: 这个C#类封装了视频转换所需的各种方法,基本上是围绕着如何通过mmpeg工具来进行视频转换 using Syst ...
- 2D到3D视频转换 三维重建
2D到3D视频转换(也称为2D到立体3D转换和立体转换)是将2D(“平面”)胶片转换为3D形式的过程,几乎在所有情况下都是立体声,因此它是创建图像的过程.每个眼睛来自一个2D图像. 内容 1概述 1. ...
随机推荐
- Android的简单应用(三)——为你的程序添加监听器
平时在写程序时经常会遇到监听器,比如按钮的click监听器,按键监听器等等.而android中的监听器和java中的回调函数是同一个概念,都是在底层代码中定义一个接口来调用高层的代码.那么什么是回调函 ...
- BAT 前端开发面经 —— 吐血总结
更好阅读,请移步这里 聊之前 最近暑期实习招聘已经开始,个人目前参加了阿里的内推及腾讯和百度的实习生招聘,在此总结一下 一是备忘.总结提升,二是希望给大家一些参考 其他面试及基础相关可以参考其他博文: ...
- spring boot jar 进程自动停止,自动终止,不能后台持续运行
第一次部署spring boot 到linux上,用命令java -jar **.jar,发现应用自动退出,进程停止了.后来发现要不挂断的执行命令,忽略所有的挂断信号,用以下命令解决 nohup ja ...
- Selenium2+python自动化73-定位的坑:class属性有空格【转载】
前言 有些class属性中间有空格,如果直接复制过来定位是会报错的InvalidSelectorException: Message: The given selector u-label f-dn ...
- KO工作原理及带来的好处
介绍 Knockout是一个以数据模型(data model)为基础的能够帮助你创建富文本,响应显示和编辑用户界面的JavaScript类库.任何时候如果你的UI需要自动更新(比如:更新依赖于用户的行 ...
- UESTC 30.最短路-最短路(Floyd or Spfa(链式前向星存图))
最短路 Time Limit: 3000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) 在每年的校赛里,所有进入决赛的同 ...
- (12)python 标准库
模块 如果模块和自己写的程序不在同一个目录,可以通过sys.path.append(路径)把程序引入 import sys sys.path.append('C:/abc')#注意 \ 的方向 意思是 ...
- DP(悬线法)【P1169】 [ZJOI2007]棋盘制作
顾z 你没有发现两个字里的blog都不一样嘛 qwq 题目描述-->p1169 棋盘制作 题目大意 给定一个01棋盘,求其中01交错的最大正方形与矩形. 解题思路: 动态规划---悬线法 以下内 ...
- 子域名/目录暴力工具Gobuster
子域名/目录暴力工具Gobuster Gobuster是Kali Linux默认安装的一款暴力扫描工具.它是使用Go语言编写的命令行工具,具备优异的执行效率和并发性能.该工具支持对子域名和Web目 ...
- [xsy2978]Product of Roots
$\newcommand{align}[1]{\begin{align*}#1\end{align*}}$题意:给出$f(x)=\prod\limits_{i=1}^n(a_ix+1)$和$g(x)= ...