Tratamento da perda de dispositivos de vídeo
Este tópico descreve como detetar a perda de dispositivos ao usar um dispositivo de captura de vídeo. Contém as seguintes secções:
- Registar para notificação do dispositivo
- Obter o link simbólico do dispositivo
- Manipular WM_DEVICECHANGE
- Desinscrever-se das notificações
- Tópicos relacionados
Registre-se para notificação de dispositivo
Antes de começar a capturar a partir do dispositivo, chame a função RegisterDeviceNotification para registar as notificações do 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
Enumere 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
);
}
Gerir WM_DEVICECHANGE
No loop de mensagens, ouça WM_DEVICECHANGE mensagens. O parâmetro lParam message é 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 seu 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 para uma DEV_BROADCAST_DEVICEINTERFACE.
- Compare o membro dbcc_name desta estrutura com a ligação simbólica 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 registro para notificação
Antes de o aplicativo sair, ligue UnregisterDeviceNotification para cancelar o registro para notificações de dispositivo/
void OnClose(HWND /*hwnd*/)
{
if (g_hdevnotify)
{
UnregisterDeviceNotification(g_hdevnotify);
}
PostQuitMessage(0);
}
Tópicos relacionados