http://blog.youkuaiyun.com/saloon_yuan/article/details/9259505
相对DirectShow,AviCap算是视频采集框架里面的老前辈了,虽然已经有点力不从心,但对一些要求不高的场合还是很实用的,网上一番搜索之后(借鉴了很多现成的轮子),实现了一个基于AviCap的摄像头采集程序(C sharp实现):
先实现一个CapVideo的类,主要是引入avicap提供的接口,结构体及一些方法的定义
其中主要用到了Avicap32.dll中提供的以下三个函数:
capCreateCaptureWindowA:创建一个视频显示窗口
capGetDriverDescriptionA:获取视频设备描述符
capGetVideoFormat:获取视频格式
- public class CapVideo
- {
- #region DLL Import Method
- [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("avicap32.dll")]
- public static extern int capGetVideoFormat(IntPtr hWnd, IntPtr psVideoFormat, int wSize);
- [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, 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 int SetWindowPos(IntPtr hWnd, int hWndInsertAfter, int x, int y, int cx, int cy, int wFlags);
- #endregion
- #region public const Fields
- 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 + 5;
- 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_GRAB_FRAME_NOSTOP = WM_USER + 61;
- public const int WM_CAP_GRAB_FRAME = WM_USER + 60;
- public const int WM_CAP_FILE_SAVEDIBA = WM_USER + 25;
- #endregion
- #region Structures
- [StructLayout(LayoutKind.Sequential)]
- public struct VIDEOHDR
- {
- [MarshalAs(UnmanagedType.I4)]
- public int lpData;
- [MarshalAs(UnmanagedType.I4)]
- public int dwBufferLength;
- [MarshalAs(UnmanagedType.I4)]
- public int dwBytesUsed;
- [MarshalAs(UnmanagedType.I4)]
- public int dwTimeCaptured;
- [MarshalAs(UnmanagedType.I4)]
- public int dwUser;
- [MarshalAs(UnmanagedType.I4)]
- public int dwFlags;
- [MarshalAs(UnmanagedType.ByValArray, SizeConst = 4)]
- public int[] dwReserved;
- }
- [StructLayout(LayoutKind.Sequential)]
- public struct BITMAPINFOHEADER
- {
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biSize;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biWidth;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biHeight;
- [MarshalAs(UnmanagedType.I2)]
- public short biPlanes;
- [MarshalAs(UnmanagedType.I2)]
- public short biBitCount;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biCompression;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biSizeImage;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biXPelsPerMeter;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biYPelsPerMeter;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biClrUsed;
- [MarshalAs(UnmanagedType.I4)]
- public Int32 biClrImportant;
- }
- [StructLayout(LayoutKind.Sequential)]
- public struct BITMAPINFO
- {
- [MarshalAs(UnmanagedType.Struct, SizeConst = 40)]
- public BITMAPINFOHEADER bmiHeader;
- [MarshalAs(UnmanagedType.ByValArray, SizeConst = 1024)]
- public Int32[] bmiColors;
- }
- #endregion
- #region Public Method
- public delegate void FrameEventHandler(IntPtr lwnd, IntPtr lpVHdr);
- 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, 0, 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
- }
接着实现一个Camera类,主要实现摄像头采集开始,停止,截图等功能。
- public class Camera
- {
- public Camera(IntPtr handle, int width, int height)
- {
- mControlPtr = handle;
- mWidth = width;
- mHeight = height;
- }
- //定义委托
- public delegate void RecievedFrameEventHandler(byte[] data);
- public event RecievedFrameEventHandler RecievedFrame;
- private IntPtr lwndC;
- private IntPtr mControlPtr;
- private int mWidth;
- private int mHeight;
- private CapVideo.FrameEventHandler mFrameEventHandler;
- #region Camera Operation
- //开始采集
- public bool StartCamera()
- {
- byte[] lpszName = new byte[100];
- byte[] lpszVer = new byte[100];
- //获取设备信息
- CapVideo.capGetDriverDescriptionA(0, lpszName, 100, lpszVer, 100);
- //创建捕获窗口
- this.lwndC = CapVideo.capCreateCaptureWindowA(lpszName, CapVideo.WS_VISIBLE + CapVideo.WS_CHILD, 0, 0, mWidth, mHeight, mControlPtr, 0);
- //连接设备
- if (this.capDriverConnect(this.lwndC, 0))
- {
- //设置为预览模式,初始帧率为60
- this.capPreview(this.lwndC, true);
- this.capPreviewRate(this.lwndC, 60);
- //BitMap及其头信息
- CapVideo.BITMAPINFO bitmapinfo = new CapVideo.BITMAPINFO();
- bitmapinfo.bmiHeader.biSize = CapVideo.SizeOf(bitmapinfo.bmiHeader);
- bitmapinfo.bmiHeader.biWidth = 400;
- bitmapinfo.bmiHeader.biHeight = 200;
- bitmapinfo.bmiHeader.biPlanes = 1;
- bitmapinfo.bmiHeader.biBitCount = 24;
- //设置视频参数
- this.capSetVideoFormat(this.lwndC, ref bitmapinfo, CapVideo.SizeOf(bitmapinfo));
- //注册回调
- this.mFrameEventHandler = new CapVideo.FrameEventHandler(FrameCallBack);
- this.capSetCallbackOnFrame(this.lwndC, this.mFrameEventHandler);
- //设置窗口
- CapVideo.SetWindowPos(this.lwndC, 0, 0, 0, mWidth, mHeight, 6);
- return true;
- }
- else
- {
- return false;
- }
- }
- //停止采集
- public void CloseCamera()
- {
- this.capDriverDisconnect(this.lwndC);
- }
- //快照
- public bool SnapShot(string str)
- {
- bool blRes = CapVideo.SendMessage(this.lwndC, CapVideo.WM_CAP_GRAB_FRAME_NOSTOP, 0, 0);
- if ( !blRes )
- {
- return blRes;
- }
- IntPtr ptr = Marshal.StringToHGlobalAnsi(str);
- return CapVideo.SendMessage(this.lwndC,CapVideo.WM_CAP_FILE_SAVEDIBA,0,ptr.ToInt32());
- Marshal.FreeHGlobal(ptr);
- }
- //收到帧数据时会执行此回调函数
- private void FrameCallBack(IntPtr lwnd, IntPtr lpVHdr)
- {
- CapVideo.VIDEOHDR videoHeader = new CapVideo.VIDEOHDR();
- byte[] VideoData;
- //获取帧头信息图像数据地址
- videoHeader = (CapVideo.VIDEOHDR)CapVideo.GetStructure(lpVHdr, videoHeader);
- VideoData = new byte[videoHeader.dwBytesUsed];
- //复制图像数据
- CapVideo.Copy(videoHeader.lpData, VideoData);
- if (this.RecievedFrame != null)
- this.RecievedFrame(VideoData);
- }
- //连接设备
- private bool capDriverConnect(IntPtr lwnd, short i)
- {
- return CapVideo.SendMessage(lwnd, CapVideo.WM_CAP_DRIVER_CONNECT, i, 0);
- }
- //断开连接
- private bool capDriverDisconnect(IntPtr lwnd)
- {
- return CapVideo.SendMessage(lwnd, CapVideo.WM_CAP_DRIVER_DISCONNECT, 0, 0);
- }
- //设置为预览模式
- private bool capPreview(IntPtr lwnd, bool f)
- {
- return CapVideo.SendMessage(lwnd, CapVideo.WM_CAP_SET_PREVIEW, f, 0);
- }
- //设置预览帧率
- private bool capPreviewRate(IntPtr lwnd, short wMS)
- {
- return CapVideo.SendMessage(lwnd, CapVideo.WM_CAP_SET_PREVIEWRATE, wMS, 0);
- }
- //设置回调函数
- private bool capSetCallbackOnFrame(IntPtr lwnd, CapVideo.FrameEventHandler lpProc)
- {
- return CapVideo.SendMessage(lwnd, CapVideo.WM_CAP_SET_CALLBACK_FRAME, 0, lpProc);
- }
- //设置视频格式
- private bool capSetVideoFormat(IntPtr hCapWnd, ref CapVideo.BITMAPINFO BmpFormat, int CapFormatSize)
- {
- return CapVideo.SendMessage(hCapWnd, CapVideo.WM_CAP_SET_VIDEOFORMAT, CapFormatSize, ref BmpFormat);
- }
- #endregion
- }
接下来就是演示例子,编写一个简单的WPF程序,利用以上代码实现摄像头采集,运行结果如下,画质不是很清楚,比起摄像头自带的软件,完全下降了两个档次,具体原因有待调查,不过利用AviCap来实现还是很节省资源的,采集时看了下CPU使用率,不超过5%。整个工程已上传至我的资源里,需要参考的朋友可以点击这里下载,另外后续想将摄像头分辨率调节的功能添加进去。