directshow捕获摄像头的数据并显示

#include "stdafx.h"
#include <DShow.h>
#include <Guiddef.h>
#include <strmif.h>  #define CHECK_HR(s) if (FAILED(s)) {return 1;}
#define SAFE_RELEASE(p)     do { if ((p)) { (p)->Release(); (p) = NULL; } } while(0)  LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam);#pragma comment(lib, "Dxguid.lib")
#pragma comment(lib, "Strmiids.lib")  int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{CoInitialize(NULL);int nRes = 0;WNDCLASSEX wcex;wcex.cbSize = sizeof(WNDCLASSEX);wcex.style = CS_HREDRAW | CS_VREDRAW;wcex.lpfnWndProc = WndProc;wcex.cbClsExtra = 0;wcex.cbWndExtra = 0;wcex.hInstance = hInstance;wcex.hIcon = LoadIcon(hInstance, MAKEINTRESOURCE(IDI_APPLICATION));wcex.hCursor = LoadCursor(NULL, IDC_ARROW);wcex.hbrBackground = (HBRUSH)(COLOR_WINDOW + 1);wcex.lpszMenuName = NULL;wcex.lpszClassName = _T("dshow capture");wcex.hIconSm = LoadIcon(wcex.hInstance, MAKEINTRESOURCE(IDI_APPLICATION));if (!RegisterClassEx(&wcex)){MessageBox(NULL, _T("Call to RegisterClassEx failed!"), _T("Win32 Guided Tour"), NULL);return 1;}HWND m_hWnd = ::CreateWindowA("STATIC", "ds_video_preview", WS_POPUP, 100, 100, 500, 500, NULL, NULL, NULL, NULL);ShowWindow(m_hWnd, nCmdShow);UpdateWindow(m_hWnd);if (m_hWnd == NULL){nRes = 11;}//----------------------用dshow获取摄像头---start--------------------------IGraphBuilder         *m_pGraph;//filter总图表管理器ICaptureGraphBuilder2 *m_pBuild;//捕获图表管理器IVideoWindow          *m_pVidWin;//窗口接口HRESULT hr = CoCreateInstance(CLSID_FilterGraph, NULL, CLSCTX_INPROC_SERVER, IID_IGraphBuilder, (void**)&m_pGraph);CHECK_HR(1);hr = CoCreateInstance(CLSID_CaptureGraphBuilder2, NULL, CLSCTX_INPROC_SERVER, IID_ICaptureGraphBuilder2, (void**)&m_pBuild);CHECK_HR(2);hr = m_pBuild->SetFiltergraph(m_pGraph);//给捕获图表管理器指定一个可用的图表管理器来进行使用CHECK_HR(3);hr = m_pGraph->QueryInterface(IID_IVideoWindow, (void **)&m_pVidWin);//通过此函数来查询某个组件是否支持某个特定的接口,如果支持就返回这些接口的指针CHECK_HR(4);ICreateDevEnum *pDevEnum = NULL;IEnumMoniker *pClsEnum = NULL;IMoniker *pMoniker = NULL;//创建设备枚举COM对象  hr = CoCreateInstance(CLSID_SystemDeviceEnum, NULL, CLSCTX_INPROC, IID_ICreateDevEnum, (void **)&pDevEnum);CHECK_HR(5);//创建视频采集设备枚举COM对象  hr = pDevEnum->CreateClassEnumerator(CLSID_VideoInputDeviceCategory, &pClsEnum, 0);CHECK_HR(6);int i = 0;while (i <= 0){hr = pClsEnum->Next(1, &pMoniker, NULL);++i;}CHECK_HR(7);IBaseFilter           *m_pSrc;hr = pMoniker->BindToObject(0, 0, IID_IBaseFilter, (void **)&m_pSrc);//就是这句获得Filter  CHECK_HR(8);SAFE_RELEASE(pMoniker);SAFE_RELEASE(pClsEnum);SAFE_RELEASE(pDevEnum);//将设备添加到filter管理器graph  hr = m_pGraph->AddFilter(m_pSrc, L"Video Capture");CHECK_HR(9);hr = m_pBuild->RenderStream(&PIN_CATEGORY_PREVIEW, &MEDIATYPE_Video, m_pSrc, NULL, NULL);CHECK_HR(10);m_pVidWin->put_Owner((OAHWND)m_hWnd);m_pVidWin->SetWindowPosition(100, 100, 400, 300);m_pVidWin->put_WindowStyle(WS_CHILD | WS_CLIPSIBLINGS);//IMediaControl接口,用来控制流媒体在Filter Graph中的流动,例如流媒体的启动和停止IMediaControl         *m_pMediaControl;hr = m_pGraph->QueryInterface(IID_IMediaControl, (void **)&m_pMediaControl);CHECK_HR(12);hr = m_pMediaControl->Run();//----------------------用dshow获取摄像头---end--------------------------MSG msg;while (GetMessage(&msg, NULL, 0, 0)){TranslateMessage(&msg);DispatchMessage(&msg);}return (int)msg.wParam;}LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{PAINTSTRUCT ps;HDC hdc;TCHAR greeting[] = _T("Hello, World!");switch (message){case WM_PAINT:hdc = BeginPaint(hWnd, &ps);TextOut(hdc, 5, 5, greeting, _tcslen(greeting));EndPaint(hWnd, &ps);break;case WM_DESTROY:PostQuitMessage(0);break;default:return DefWindowProc(hWnd, message, wParam, lParam);break;}return 0;
}

这是直接捕获并绘制到窗口上,一般的应用是获取流然后和音频一起编码,做直播和网络传输。那么有没有这种方式,进行传输呢,其实directshow 提供了ISampleGrabber,可以实现直接得到缓冲,或者异步回调数据。

1  从ISampleGrabber实例化一个sample grabber,充当一个transform filter

CComPtr<ISampleGrabber> m_pSampleGrabber;

2 设置相关参数,将Grabber注册并加入到filter graph中

//Add SampleGrabber
// create a sample grabber
//
hr = m_pSampleGrabber.CoCreateInstance( CLSID_SampleGrabber );
if( !m_pSampleGrabber )
{
MessageBox(NULL, L"Could not create SampleGrabber (is qedit.dll registered?)", L"", MB_OK);
return hr;
}CComQIPtr< IBaseFilter, &IID_IBaseFilter > pGrabBase( m_pSampleGrabber );
// force it to connect to video, 24 bit
//
CMediaType VideoType;
VideoType.SetType( &MEDIATYPE_Video );
VideoType.SetSubtype( &MEDIASUBTYPE_RGB24 );
VideoType.SetFormatType(&FORMAT_VideoInfo);
hr = m_pSampleGrabber->SetMediaType( &VideoType ); // shouldn't fail
if (FAILED(hr))
{
MessageBox(NULL, L"不能初始化SampleGrabber媒体类型。", L"", MB_OK);
return hr;
}
hr = m_pGB->AddFilter( pGrabBase, L"Grabber" );
if( FAILED( hr ) )
{
MessageBox(NULL, L"Could not put sample grabber in graph", L"", MB_OK);
return hr;
}

3 连接成功以后,将Sample Grabber作为一个中间filter,利用CSampleGrabber的实例调用回调函数BufferCB,从源filter中得到

摄像头设备捕获的每一帧数据,作为视频预览

hr = m_pCapture->RenderStream(&PIN_CATEGORY_PREVIEW, &MEDIATYPE_Video, m_pBF, pGrabBase, m_pVMR);if (FAILED(hr))
{
MessageBox(NULL, L"Could not put sample into sample grabber", L"", MB_OK);
return hr;
}//Set mCB
AM_MEDIA_TYPE mt;
hr = m_pSampleGrabber->GetConnectedMediaType( &mt );
if ( FAILED( hr) )
{
MessageBox( NULL, TEXT("Could not read the connected media type"),L"", MB_OK);
return hr;
}VIDEOINFOHEADER * vih = (VIDEOINFOHEADER*) mt.pbFormat;
mCB.IWidth = vih->bmiHeader.biWidth;
mCB.IHeight = vih->bmiHeader.biHeight;
FreeMediaType( mt );hr = RenderFileList(m_pGB,m_pVMR );// don't buffer the samples as they pass through
//
hr = m_pSampleGrabber->SetBufferSamples( TRUE);// only grab one at a time, stop stream after
// grabbing one sample
//
hr = m_pSampleGrabber->SetOneShot( FALSE );// set the callback, so we can grab the one sample
//
hr = m_pSampleGrabber->SetCallback( &mCB, 1 );
if (FAILED(hr))
{
MessageBox(NULL, L"Could not call callback method BufferCB().", L"", MB_OK);
return hr;
}
pGrabBase.Release();

4 实现接口类ISampleGrabber,用于捕获视频帧的sample,从BufferCB中返回图像数据,用户传输,之前已设定了

图像的采样格式,这里不再赘述。

//SampleGrabber
// Global data
#define WM_CAPTURE_BITMAP   WM_APP + 1
BOOL g_bOneShot=TRUE;// Structures
typedef struct _callbackinfo
{
double dblSampleTime;
long lBufferSize;
BYTE *pBuffer;
BITMAPINFOHEADER bih;} CALLBACKINFO;CALLBACKINFO cb={0};class CSampleGrabberCB : public ISampleGrabberCB
{
private:
LPBITMAPFILEHEADER m_pFileHeader ;    // Bmp文件头
LPBITMAPINFOHEADER m_pBmpInfo ;     // Bmp信息头指针BYTE* m_pImgFileData;                //Bmp文件数据BOOL m_bViladImage ;LPBYTE m_pBitmapHeader ;
LPVOID m_pvColorTable ;               //调色板指针
public:
// these will get set by the main thread below. We need to
// know this in order to write out the bmp
long IWidth;
long IHeight;CSampleGrabberCB( )
{
m_pFileHeader = new BITMAPFILEHEADER ;
m_pBitmapHeader = new BYTE[sizeof(BITMAPINFOHEADER)] ;
}   // fake out any COM ref counting
//
STDMETHODIMP_(ULONG) AddRef() { return 2; }
STDMETHODIMP_(ULONG) Release() { return 1; }// fake out any COM QI'ing
//
STDMETHODIMP QueryInterface(REFIID riid, void ** ppv)
{
if( riid == IID_ISampleGrabberCB || riid == IID_IUnknown )
{
*ppv = (void *) static_cast<ISampleGrabberCB*> ( this );
return NOERROR;
}
return E_NOINTERFACE;
}// we don't implement this interface for this example
//
STDMETHODIMP SampleCB( double SampleTime, IMediaSample * pSample )
{
return 0;
}
// As a workaround, copy the bitmap data during the callback,
// post a message to our app, and write the data later.
//
STDMETHODIMP BufferCB( double dblSampleTime, BYTE * pBuffer, long lBufferSize )
{
// this flag will get set to true in order to take a picture
//
if( !g_bOneShot )
return 0;if (!pBuffer)
{
return E_POINTER;
}if( cb.lBufferSize < lBufferSize )
{
delete [] cb.pBuffer;
cb.pBuffer = NULL;
cb.lBufferSize = 0;
}// Since we can't access Windows API functions in this callback, just
// copy the bitmap data to a global structure for later reference.
cb.dblSampleTime = dblSampleTime;// If we haven't yet allocated the data buffer, do it now.
// Just allocate what we need to store the new bitmap.
if (!cb.pBuffer)
{
cb.pBuffer = new BYTE[lBufferSize];
cb.lBufferSize = lBufferSize;
}if( !cb.pBuffer )
{
cb.lBufferSize = 0;
return E_OUTOFMEMORY;
}//Get bmp information
BITMAPINFOHEADER bih;
memset( &bih, 0, sizeof( bih ) );
bih.biSize = sizeof( bih );
bih.biWidth = IWidth;
bih.biHeight = IHeight;
bih.biPlanes = 1;
bih.biBitCount = 24;
memcpy(&(cb.bih), &bih, sizeof(bih));// Copy the bitmap data into our global buffer
memcpy(cb.pBuffer, pBuffer, lBufferSize);
// Post a message to our application, telling it to come back
// and write the saved data to a bitmap file on the user's disk.
SendMessage(m_hWnd, WM_CAPTURE_BITMAP, 0, 0L);
return 0;
}BOOL FormatImage( BYTE *lpImageData, int nBitCount, int nWidth, int nHeight )
{m_bViladImage = FALSE ;int nKlsBmpBitCount ;
int nImgWidth = nWidth ;
int nImgHeight = nHeight ;
if (nBitCount == 8 || nBitCount == 24 || nBitCount == 32)
{
nKlsBmpBitCount = nBitCount;
}
else
{
return m_bViladImage ;
}int nDataWidth = nKlsBmpBitCount / 8 * nWidth ;nDataWidth = ( nDataWidth % 4 == 0 ) ? nDataWidth : ( ( nDataWidth / 4 + 1 ) * 4 ) ;//m_pFileHeader = new BITMAPFILEHEADER ;m_pFileHeader->bfType      = 0x4d42 ;
m_pFileHeader->bfSize      = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) + nDataWidth*nImgHeight ;
m_pFileHeader->bfReserved1 = 0 ;
m_pFileHeader->bfReserved2 = 0 ;if ( nBitCount == 8 )
{
int nBmpInfoSize = sizeof(BITMAPFILEHEADER)
+ sizeof(BITMAPINFOHEADER)
+ 256 * 4 ;m_pFileHeader->bfOffBits = nBmpInfoSize ;m_pBitmapHeader = new BYTE[nBmpInfoSize] ;m_pBmpInfo = (LPBITMAPINFOHEADER)m_pBitmapHeader ;m_pvColorTable = m_pBitmapHeader + sizeof(BITMAPINFOHEADER) ;LPRGBQUAD pDibQuad = (LPRGBQUAD)(m_pvColorTable) ;for ( int c=0; c<256; ++c )
{
pDibQuad[c].rgbRed = c ;
pDibQuad[c].rgbGreen = c ;
pDibQuad[c].rgbBlue = c ;
pDibQuad[c].rgbReserved = 0 ;
}
}
else
{
m_pFileHeader->bfOffBits = sizeof(BITMAPFILEHEADER) + sizeof(BITMAPINFOHEADER) ;//m_pBitmapHeader = new BYTE[sizeof(BITMAPINFOHEADER)] ;m_pBmpInfo = (LPBITMAPINFOHEADER)m_pBitmapHeader ;
m_pvColorTable = NULL ;
}m_pBmpInfo->biBitCount      = nKlsBmpBitCount ;
m_pBmpInfo->biWidth         = nImgWidth ;
m_pBmpInfo->biHeight        = nImgHeight ;
m_pBmpInfo->biPlanes        = 1 ;
m_pBmpInfo->biSize          = sizeof(BITMAPINFOHEADER) ;
m_pBmpInfo->biSizeImage     = nImgWidth * nImgHeight * nKlsBmpBitCount / 8 ;
m_pBmpInfo->biClrImportant  = 0 ;
m_pBmpInfo->biClrUsed       = 0 ;
m_pBmpInfo->biCompression   = 0 ;
m_pBmpInfo->biXPelsPerMeter = 0 ;
m_pBmpInfo->biYPelsPerMeter = 0 ; //m_pImgFileData = new BYTE[nDataWidth*nImgHeight] ;
SetImgFileData(nDataWidth, nImgHeight);if ( nBitCount == 8 )
{
if ( nImgWidth % 4 == 0 )
{
memset( m_pImgFileData, 0, nDataWidth*nImgHeight ) ;
memcpy( m_pImgFileData, lpImageData, cb.lBufferSize) ;
}
else
{
memset( m_pImgFileData, 0, nDataWidth*nImgHeight ) ;for ( int i=0; i<nImgHeight; i++ )
{
memcpy( m_pImgFileData + i*nDataWidth, lpImageData + i*nImgWidth, nImgWidth ) ;
}
}
}
else if ( nBitCount == 24 )
{
if ( nImgWidth % 4 == 0 )
{
memset( m_pImgFileData, 0, nDataWidth*nImgHeight ) ;
memcpy( m_pImgFileData, lpImageData, cb.lBufferSize ) ;
}
else
{
memset( m_pImgFileData, 0, nDataWidth*nImgHeight ) ;for ( int i=0; i<nImgHeight; i++ )
{
memcpy( m_pImgFileData + i*nDataWidth, lpImageData + i*3*nImgWidth, 3*nImgWidth ) ;
}
}
}
else if ( nBitCount == 32 )
{
memcpy( m_pImgFileData, lpImageData,cb.lBufferSize ) ;
}m_bViladImage = TRUE ;return m_bViladImage ;
}BYTE* GetImgFileData()
{
return m_pImgFileData;
}void SetImgFileData(int dataWidth, int dataHeight)
{
if (!m_pImgFileData)
{
m_pImgFileData = new BYTE[dataWidth*dataHeight];
}
else
{
delete [] m_pImgFileData;
m_pImgFileData = NULL;
m_pImgFileData = new BYTE[dataWidth*dataHeight];
}}
};//
//
// This semi-COM object will receive sample callbacks for us
//
//
CSampleGrabberCB mCB;
//SampleGrabber

得到sample之后,放入到回调函数BufferCB中进行处理,从而得到数据块,并存入相应的结构体

//From Step 3
// set the callback, so we can grab the one sample
//
hr = m_pSampleGrabber->SetCallback( &mCB, 1 );//From BufferCB
//Get bmp information
BITMAPINFOHEADER bih;
memset( &bih, 0, sizeof( bih ) );
bih.biSize = sizeof( bih );
bih.biWidth = IWidth;
bih.biHeight = IHeight;
bih.biPlanes = 1;
bih.biBitCount = 24;
memcpy(&(cb.bih), &bih, sizeof(bih));// Copy the bitmap data into our global buffer
memcpy(cb.pBuffer, pBuffer, lBufferSize);

采样部分进行到这里,已经完成大部分工作了,下一步就是直接进行编解码并做音频同步,然后网络传输,这个将在之后的文章中陆续分享出来

directshow捕获摄像头数据相关推荐

  1. FFMPEG(一) 从V4L2捕获摄像头数据

    系列相关博文: FFMPEG(一) 从V4L2捕获摄像头数据 FFMPEG(二) v4l2 数据格式装换 FFMPEG(三) v4l2 数据编码H264 最近在学习FFMPEG,发现网上的很多例子都是 ...

  2. python捕获摄像头帧_Xuggler教程:帧捕获和视频创建

    python捕获摄像头帧 注意:这是我们的" Xuggler开发教程 "系列的一部分. 到目前为止,在我们的Xuggler教程系列中,我们已经对视频处理的Xuggler进行了介绍, ...

  3. 使用TensorFlow.js进行人脸触摸检测第1部分:将实时网络摄像头数据与深度学习配合使用

    目录 起点 将HTML5网络摄像头API与TensorFlow.js结合使用 检测脸部触摸 技术脚注 终点线 下一步是什么?我们是否可以在未经培训的情况下检测到面部触摸? 下载TensorFlowJS ...

  4. ffmpeg推流摄像头数据至公网服务器

    完整的推流代码已经托管到个人的Gitee,如有需要请自取 https://gitee.com/MonsterAKALei/push_video.git ffmpeg推流摄像头数据 昨天实现用API分别 ...

  5. QT 显示USB摄像头数据

    QT 显示USB摄像头数据 一.简述       记--简单的将USB摄像头捕获的图像数据显示到窗口上.       例子1打包:链接: https://pan.baidu.com/s/1U3CW7s ...

  6. C#通过引用AForge获取摄像头数据

    一.概述 实现了读取摄像头数据,而且当电脑接有多个摄像头设备时,可以选择连接的摄像头设备.把摄像头的数据进行了实时显示,且具有截图(拍照)的功能,可对截取的图片进行保存,以便进行后续的图像处理. 二. ...

  7. opencv-python 实时获取摄像头数据并实时显示。

    opencv-python 实时获取摄像头数据并实时显示. 目录 opencv-python 实时获取摄像头数据并实时显示. 代码: 一.关于 cv2.VideoCapture(source) 函数 ...

  8. CenterFusion:融合雷达与摄像头数据的高精度3D目标检测

    点击上方"3D视觉工坊",选择"星标" 干货第一时间送达 编辑丨我爱计算机视觉 介绍一篇新出的论文 CenterFusion: Center-based Rad ...

  9. java 捕获数据包,缓存从pcap捕获的数据包

    我想要完成的事情: functionA:使用pcap捕获数据包 . 修改源/目标地址 . 重新计算校验和 . 注入pcap . functionB:创建两个线程 . 线程1发送一个魔术包以唤醒睡眠客户 ...

最新文章

  1. ACMNO.35 C语言-宏润年 给年份year,定义一个宏,以判别该年份是否闰年。提示:宏名可以定义为LEAP_YEAR,形参为y,既定义宏的形式为 #define LEAP_YEAR(y)
  2. python四大软件-Python未来可能面临的四大转折
  3. 关于图片延迟加载的解决方案(针对移动端)
  4. Java EE 6与Spring Framework:技术决策过程
  5. 【C++基础】自定义异常类与多重捕获
  6. spring AOP 之一:spring AOP功能介绍
  7. 背景图层和普通图层的区别_008Photoshop四赞图层(图层样式)
  8. 使用wordpress_为什么我使用WordPress进行教育
  9. 云效 > 产品简介 > 产品概述
  10. 对linux中多线程编程中pthread_join的理解
  11. Android中如何使按钮的背景变得透明
  12. 快手分享链接,实战解析无水印视频源码
  13. 诺基亚 8208支持java么_诺基亚8_诺基亚8208怎么上网_8208 诺基亚
  14. 如何利用花生壳动态域名解析服务搭建http服务。
  15. python自动化测试绕过手机验证码
  16. 不同类型的物联网产品、物联网产品的优势
  17. 计算机键盘锁不了怎么办,电脑键盘不能打字_电脑键盘锁住不能打字
  18. 解决客户 IE 浏览器“兼容性视图“设置带来的问题
  19. 从档案信息管理到档案知识管理
  20. Android系统永不熄屏和取消开机锁屏功能

热门文章

  1. 大三末之初级前端面试(二)
  2. Peercast源代码文件结构分析
  3. Peercast简介、分析及常见问题处理 (一)
  4. 钉钉机器人智能回复_契约机器人开发教程-新人入群智能回复|良心开发工作室...
  5. 有选择的人从来都谈不上落魄,过着没有选择,不得已的生活才叫落魄
  6. FFmpeg和Audacity噪声处理
  7. 《Java从入门到放弃》入门篇:hibernate查询——HQL
  8. IntelliJ IDEA 无法使用输入法,输入法失效怎么办?
  9. python 年月日时分秒_如何将年月日转换为年月日时分秒
  10. 【TP5源码】获取项目的根目录