使用DirectShow驱动摄像头

对于小摄像头的驱动有几种方法,通过使用DirectShow来驱动摄像头灵活性比较好,有简单的方法也有比较复杂但更有效的方法,本文只介绍简单方法,希望与大家交流!

       用DirectShow来使用摄像头,一般要求摄像头的驱动是WDM格式的,当然,一些比较老的驱动格式DirectShow也可支持。DirectShow通过图形过滤管理器(Filter Graph Manager)来与上层应用程序和下层的驱动进行联系。DirectShow通过一种叫作捕获过滤器(Capture Filter)的东东来支持对摄像头的捕获,一个捕获过滤器有多个插口(pin),其中的预览(preview)插口可用来进行显示祯图象。

       DirectShow通过几个COM接口来对视频捕获的全过程进行控制,其中IGraphBuilder 用于建立过滤器,ICaptureGraphBuilder2用于与下层的驱动程序建立联系,IVideoWindow,IMediaControl,IMediaEventEx分别对整个过程的视频窗口,播放过程和事件响应进行控制,

下面是例程:

CComQIPtr<IVideoWindow>   m_pVW;
    CComQIPtr<IMediaControl>  m_pMC;
    CComQIPtr<IMediaEventEx>  m_pME;
    CComPtr<IGraphBuilder>  m_pGraph;
    CComPtr<ICaptureGraphBuilder2>  m_pCapture;
    DWORD m_dwGraphRegister;
bool bInit(HWND hWnd)
{
    HRESULT hr;
//获得接口
    hr=CoCreateInstance (CLSID_FilterGraph, NULL, CLSCTX_INPROC,
                           IID_IGraphBuilder, (void **) &m_pGraph);
    if (FAILED(hr))
        return false;
    hr=CoCreateInstance (CLSID_CaptureGraphBuilder2 , NULL, CLSCTX_INPROC,
                          IID_ICaptureGraphBuilder2, (void **) &m_pCapture);
    if (FAILED(hr))
        return false;
    m_pMC=m_pGraph;
    m_pVW=m_pGraph;
    m_pME=m_pGraph;
    //取得消息
    m_pME->SetNotifyWindow((OAHWND)(m_hWnd=hWnd), WM_GRAPHNOTIFY, 0);
    //将过滤和捕获进行连接
    m_pCapture->SetFiltergraph(m_pGraph);
//设备联接
    //枚举设备
    CComPtr <ICreateDevEnum> pCde=NULL;
    CComPtr <IEnumMoniker> pEm=NULL;
    hr=CoCreateInstance (CLSID_SystemDeviceEnum, NULL, CLSCTX_INPROC,
                           IID_ICreateDevEnum, (void **) &pCde);
    if (FAILED(hr))
        return false;
    pCde->CreateClassEnumerator(CLSID_VideoInputDeviceCategory, &pEm, 0);
    if(pEm==NULL)
        return false;
    CComPtr <IMoniker> pM=NULL;
    ULONG cFetched;
    CComPtr<IBaseFilter> pBf=NULL;
    if(pEm->Next(1,&pM,&cFetched)==S_OK)
    {
        pM->BindToObject(0,0,IID_IBaseFilter, (void**)&pBf);
        pM.Release();
    }else{
        return false;
    }
    //将设备添加到graph
    hr = m_pGraph->AddFilter(pBf, L”Video Capture”);
    if (FAILED(hr))
        return false;
    //连接一个源插口
    hr=m_pCapture->RenderStream(&PIN_CATEGORY_PREVIEW,&MEDIATYPE_Video ,pBf,NULL,NULL);
    if (FAILED(hr))
        return false;
    pBf.Release();
//设定视频窗口
    //设定视频窗口为主窗口的一个子窗口
    hr=m_pVW->put_Owner((OAHWND)hWnd);
    if (FAILED(hr))
        return false;
    //设定窗口样式
    m_pVW->put_WindowStyle(WS_CHILD | WS_CLIPCHILDREN);
    if (FAILED(hr))
        return false;
    //设定窗口大小
    CRect rectClient;
    GetClientRect(hWnd,rectClient);
    m_pVW->SetWindowPosition(0, 0, 320, 240);
    //设定可视
    hr=m_pVW->put_Visible(OATRUE);
    if (FAILED(hr))
        return false;
    //将对象加入到运行对象列表中
    CComPtr<IRunningObjectTable> pROT;
    GetRunningObjectTable(0,&pROT);
    WCHAR c[128];
    wsprintfW(c, L”FilterGraph %08x pid %08x″, (DWORD_PTR)m_pGraph.p, GetCurrentProcessId());
    hr = CreateItemMoniker(L”!”,c,&pM);
    if (FAILED(hr))
        return false;
    hr = pROT->Register(ROTFLAGS_REGISTRATIONKEEPSALIVE,m_pGraph, pM,&m_dwGraphRegister);
    pM.Release();
    return false;
}


转载自VC项目库-Vistal C++ Project Base

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
//指定视频采集设备的友好名字,为它创建一个Filter IBaseFilter * CTestPreviewDlg::CreateVideoDevice(const char * inFriendlyName) { return CreateHardwareFilter(CLSID_VideoInputDeviceCategory,inFriendlyName); } //根据设备的友好名字,创建一个代表该设备的Filter IBaseFilter * CTestPreviewDlg::CreateHardwareFilter(GUID inCategory,const char * inFriendlyName) { //创建一个系统枚举组件对象 ICreateDevEnum * enumHardware = NULL; HRESULT hr = CoCreateInstance(CLSID_SystemDeviceEnum,NULL,CLSCTX_ALL, IID_ICreateDevEnum,(void**)&enumHardware); if(FAILED(hr)) { return NULL; } IBaseFilter * hardwareFilter = NULL; IEnumMoniker * enumMoniker = NULL; //为指定的目录创建枚举器 hr = enumHardware->CreateClassEnumerator(inCategory,&enumMoniker,0); if(enumMoniker) { enumMoniker->Reset(); ULONG fetched = 0; IMoniker * moniker = NULL; char friendlyName[256]; //枚举得到该目录下所有的设备,逐个进行名字匹配 while(!hardwareFilter && SUCCEEDED(enumMoniker->Next(1,&moniker, &fetched)) && fetched) { if(moniker) { IPropertyBag * propertyBag = NULL; VARIANT name; friendlyName[0] = 0; hr = moniker->BindToStorage(0,0,IID_IPropertyBag,(void**)&propertyBag); //读取设备的友好名字 if(SUCCEEDED(hr)) { name.vt = VT_BSTR; hr = propertyBag->Read(L"Friendlyname",&name,NULL); } if(SUCCEEDED(hr)) { WideCharToMultiByte(CP_ACP,0,name.bstrVal,-1, friendlyName,256,NULL,NULL); //如果当前设备的友好名字与用户指定的设备名字相同, //则将当前设备标识绑定为Filter形式 if(strcmp(friendlyName,inFriendlyName) == 0) { moniker->BindToObject(0,0,IID_IBaseFilter, (void**)&hardwareFilter); } } //释放使用过的接口 if(propertyBag) { propertyBag->Release(); propertyBag = NULL; } moniker->Release(); } } enumMoniker->Release(); } enumHardware->Release(); return hardwareFilter; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值