使用DirectShow驱动摄像头

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

 

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

 

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

 

下面是例程:

 

    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/0", (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;

 

}

//指定视频采集设备的友好名字,为它创建一个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、付费专栏及课程。

余额充值