Manipulando a perda de dispositivo de vídeo
Este tópico descreve como detectar a perda de dispositivo ao usar um dispositivo de captura de vídeo. Ele contém as seções a seguir:
- Registrar-se para notificação de dispositivo
- Obter o link simbólico do dispositivo
- Manipular WM_DEVICECHANGE
- Cancelar o registro para notificação
- Tópicos relacionados
Registrar-se para notificação de dispositivo
Antes de começar a capturar do dispositivo, chame a função RegisterDeviceNotification para se registrar para notificações de dispositivo. Registre-se para a classe de dispositivo KSCATEGORY_CAPTURE , conforme mostrado no código a seguir.
#include <Dbt.h>
#include <ks.h>
#include <ksmedia.h>
HDEVNOTIFY g_hdevnotify = NULL;
BOOL RegisterForDeviceNotification(HWND hwnd)
{
DEV_BROADCAST_DEVICEINTERFACE di = { 0 };
di.dbcc_size = sizeof(di);
di.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
di.dbcc_classguid = KSCATEGORY_CAPTURE;
g_hdevnotify = RegisterDeviceNotification(
hwnd,
&di,
DEVICE_NOTIFY_WINDOW_HANDLE
);
if (g_hdevnotify == NULL)
{
return FALSE;
}
return TRUE;
}
Obter o link simbólico do dispositivo
Enumerar os dispositivos de vídeo no sistema, conforme descrito em Enumerando Dispositivos de Captura de Vídeo. Escolha um dispositivo na lista e, em seguida, consulte o objeto de ativação para o atributo MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_SYMBOLIC_LINK , conforme mostrado no código a seguir.
WCHAR *g_pwszSymbolicLink = NULL;
UINT32 g_cchSymbolicLink = 0;
HRESULT GetSymbolicLink(IMFActivate *pActivate)
{
return pActivate->GetAllocatedString(
MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_SYMBOLIC_LINK,
&g_pwszSymbolicLink,
&g_cchSymbolicLink
);
}
Manipular WM_DEVICECHANGE
No loop de mensagens, ouça WM_DEVICECHANGE mensagens. O parâmetro de mensagem lParam é um ponteiro para uma estrutura DEV_BROADCAST_HDR .
case WM_DEVICECHANGE:
if (lParam != 0)
{
HRESULT hr = S_OK;
BOOL bDeviceLost = FALSE;
hr = CheckDeviceLost((PDEV_BROADCAST_HDR)lParam, &bDeviceLost);
if (FAILED(hr) || bDeviceLost)
{
CloseDevice();
MessageBox(hwnd, L"Lost the capture device.", NULL, MB_OK);
}
}
return TRUE;
Em seguida, compare a mensagem de notificação do dispositivo com o link simbólico do dispositivo, da seguinte maneira:
- Verifique o membro dbch_devicetype da estrutura DEV_BROADCAST_HDR . Se o valor for DBT_DEVTYP_DEVICEINTERFACE, converta o ponteiro da estrutura em uma estrutura DEV_BROADCAST_DEVICEINTERFACE .
- Compare o membro dbcc_name dessa estrutura com o link simbólico do dispositivo.
HRESULT CheckDeviceLost(DEV_BROADCAST_HDR *pHdr, BOOL *pbDeviceLost)
{
DEV_BROADCAST_DEVICEINTERFACE *pDi = NULL;
if (pbDeviceLost == NULL)
{
return E_POINTER;
}
*pbDeviceLost = FALSE;
if (g_pSource == NULL)
{
return S_OK;
}
if (pHdr == NULL)
{
return S_OK;
}
if (pHdr->dbch_devicetype != DBT_DEVTYP_DEVICEINTERFACE)
{
return S_OK;
}
// Compare the device name with the symbolic link.
pDi = (DEV_BROADCAST_DEVICEINTERFACE*)pHdr;
if (g_pwszSymbolicLink)
{
if (_wcsicmp(g_pwszSymbolicLink, pDi->dbcc_name) == 0)
{
*pbDeviceLost = TRUE;
}
}
return S_OK;
}
Cancelar o registro para notificação
Antes que o aplicativo seja encerrado, chame UnregisterDeviceNotification para cancelar o registro de notificações do dispositivo/
void OnClose(HWND /*hwnd*/)
{
if (g_hdevnotify)
{
UnregisterDeviceNotification(g_hdevnotify);
}
PostQuitMessage(0);
}
Tópicos relacionados