23static constexpr GUID KSCATEGORY_SENSOR_CAMERA = {
24 0x24e552d7, 0x6523, 0x47f7, { 0xa6, 0x47, 0xd3, 0x46, 0x5b, 0xf1, 0xf5, 0xca }
32LRESULT QT_WIN_CALLBACK deviceNotificationWndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
34 if (message == WM_DEVICECHANGE) {
35 auto b = (PDEV_BROADCAST_HDR)lParam;
36 if (b && b->dbch_devicetype == DBT_DEVTYP_DEVICEINTERFACE) {
37 auto wmd =
reinterpret_cast<QWindowsVideoDevices *>(GetWindowLongPtr(hWnd, GWLP_USERDATA));
39 if (wParam == DBT_DEVICEARRIVAL || wParam == DBT_DEVICEREMOVECOMPLETE) {
40 wmd->onVideoInputsChanged();
65 wx.cbSize =
sizeof(WNDCLASSEX);
66 wx.lpfnWndProc = deviceNotificationWndProc;
67 wx.hInstance = GetModuleHandle(
nullptr);
68 wx.lpszClassName = getWindowsClassName();
70 if (!RegisterClassEx(&wx))
73 auto hwnd = CreateWindowEx(0, getWindowsClassName(), TEXT(
"Message"), 0, 0, 0, 0, 0,
74 HWND_MESSAGE,
nullptr,
nullptr,
nullptr);
76 UnregisterClass(getWindowsClassName(), GetModuleHandle(
nullptr));
86 : QPlatformVideoDevices(integration)
88 qt_win_ensureComInitializedOnThisThread();
90 m_videoDeviceMsgWindow = createMessageOnlyWindow();
91 if (m_videoDeviceMsgWindow) {
92 SetWindowLongPtr(m_videoDeviceMsgWindow, GWLP_USERDATA, (LONG_PTR)
this);
94 DEV_BROADCAST_DEVICEINTERFACE di = {};
95 di.dbcc_size =
sizeof(di);
96 di.dbcc_devicetype = DBT_DEVTYP_DEVICEINTERFACE;
97 di.dbcc_classguid = KSCATEGORY_VIDEO_CAMERA;
99 m_videoDeviceNotification =
100 RegisterDeviceNotification(m_videoDeviceMsgWindow, &di, DEVICE_NOTIFY_WINDOW_HANDLE);
101 if (!m_videoDeviceNotification) {
102 DestroyWindow(m_videoDeviceMsgWindow);
103 m_videoDeviceMsgWindow =
nullptr;
105 UnregisterClass(getWindowsClassName(), GetModuleHandle(
nullptr));
109 if (!m_videoDeviceNotification) {
110 qWarning() <<
"Video device change notification disabled";
128 GUID subtype = GUID_NULL;
129 if (FAILED(mediaFormat->GetGUID(MF_MT_SUBTYPE, &subtype)))
132 auto pixelFormat = QWindowsMultimediaUtils::pixelFormatFromMediaSubtype(subtype);
133 if (pixelFormat == QVideoFrameFormat::Format_Invalid)
136 UINT32 nominalRange = 0;
137 auto colorRange = QVideoFrameFormat::ColorRange_Unknown;
139 if (SUCCEEDED(mediaFormat->GetUINT32(MF_MT_VIDEO_NOMINAL_RANGE, &nominalRange)))
140 colorRange = QWindowsMultimediaUtils::colorRangeFromNominalRange(nominalRange);
142 UINT32 yuvMatrix = 0;
143 auto colorSpace = QVideoFrameFormat::ColorSpace_Undefined;
145 if (SUCCEEDED(mediaFormat->GetUINT32(MF_MT_YUV_MATRIX , &yuvMatrix)))
146 colorSpace = QWindowsMultimediaUtils::colorSpaceFromMatrix(yuvMatrix);
150 if (FAILED(MFGetAttributeSize(mediaFormat, MF_MT_FRAME_SIZE, &width, &height)))
152 QSize resolution{
int(width),
int(height) };
159 if (SUCCEEDED(MFGetAttributeRatio(mediaFormat, MF_MT_FRAME_RATE_RANGE_MIN, &num, &den)))
160 minFr =
float(num) /
float(den);
162 if (SUCCEEDED(MFGetAttributeRatio(mediaFormat, MF_MT_FRAME_RATE_RANGE_MAX, &num, &den)))
163 maxFr =
float(num) /
float(den);
165 auto *f =
new QCameraFormatPrivate{ QSharedData(), pixelFormat, resolution, minFr, maxFr , colorRange, colorSpace };
184 auto info = std::make_unique<QCameraDevicePrivate>();
185 info->description = getString(device, MF_DEVSOURCE_ATTRIBUTE_FRIENDLY_NAME);
186 info->id = getString(device, MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_SYMBOLIC_LINK).toUtf8();
188 IMFMediaSource *source = NULL;
189 HRESULT hr = device->ActivateObject(IID_PPV_ARGS(&source));
193 ComPtr<IMFSourceReader> reader;
194 hr = wmf.mfCreateSourceReaderFromMediaSource(source, NULL, reader.GetAddressOf());
198 QList<QSize> photoResolutions;
199 QList<QCameraFormat> videoFormats;
200 for (DWORD i = 0;; ++i) {
202 ComPtr<IMFMediaType> mediaFormat;
203 hr = reader->GetNativeMediaType((DWORD)MF_SOURCE_READER_FIRST_VIDEO_STREAM, i,
204 mediaFormat.GetAddressOf());
208 auto maybeCamera = createCameraFormat(mediaFormat.Get());
210 videoFormats << *maybeCamera;
211 photoResolutions << maybeCamera->resolution();
215 info->videoFormats = videoFormats;
216 info->photoResolutions = photoResolutions;
217 return info.release()->create();
223 QList<QCameraDevice> cameras;
225 IMFActivate **devicesRaw =
nullptr;
226 HRESULT hr = wmf.mfEnumDeviceSources(attr, &devicesRaw, &count);
228 QComTaskResource<IMFActivate *[], QComDeleter> devices(devicesRaw, count);
230 for (UINT32 i = 0; i < count; i++) {
231 IMFActivate *device = devices[i];
233 auto maybeCamera = createCameraDevice(wmf, device);
235 cameras << *maybeCamera;
247 QList<QCameraDevice> cameras;
249 ComPtr<IMFAttributes> attr;
250 HRESULT hr = m_wmf->mfCreateAttributes(attr.GetAddressOf(), 2);
254 hr = attr->SetGUID(MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE,
255 MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_GUID);
257 cameras << readCameraDevices(*m_wmf, attr.Get());
259 hr = attr->SetGUID(MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_CATEGORY,
260 KSCATEGORY_SENSOR_CAMERA);
262 cameras << readCameraDevices(*m_wmf, attr.Get());