Image encoder control - version for OS X/iOS
QImageEncoderControl - implementation for AVFoundation plugin (OS X/iOS, at the moment iOS >= 7.0). Change-Id: Ibc2c3ae48252dd4698e263f5abca5c328482d5e7 Reviewed-by: Yoann Lopes <yoann.lopes@theqtcompany.com>
This commit is contained in:
committed by
Yoann Lopes
parent
985ee3261b
commit
33b27c3c15
@@ -57,6 +57,7 @@ class AVFCameraExposureControl;
|
||||
class AVFCameraZoomControl;
|
||||
class AVFCameraViewfinderSettingsControl2;
|
||||
class AVFCameraViewfinderSettingsControl;
|
||||
class AVFImageEncoderControl;
|
||||
|
||||
class AVFCameraService : public QMediaService
|
||||
{
|
||||
@@ -81,6 +82,7 @@ public:
|
||||
AVFCameraRendererControl *videoOutput() const {return m_videoOutput; }
|
||||
AVFCameraViewfinderSettingsControl2 *viewfinderSettingsControl2() const {return m_viewfinderSettingsControl2; }
|
||||
AVFCameraViewfinderSettingsControl *viewfinderSettingsControl() const {return m_viewfinderSettingsControl; }
|
||||
AVFImageEncoderControl *imageEncoderControl() const {return m_imageEncoderControl; }
|
||||
|
||||
private:
|
||||
AVFCameraSession *m_session;
|
||||
@@ -97,6 +99,7 @@ private:
|
||||
AVFCameraZoomControl *m_cameraZoomControl;
|
||||
AVFCameraViewfinderSettingsControl2 *m_viewfinderSettingsControl2;
|
||||
AVFCameraViewfinderSettingsControl *m_viewfinderSettingsControl;
|
||||
AVFImageEncoderControl *m_imageEncoderControl;
|
||||
};
|
||||
|
||||
QT_END_NAMESPACE
|
||||
|
||||
@@ -51,6 +51,7 @@
|
||||
#include "avfcamerafocuscontrol.h"
|
||||
#include "avfcameraexposurecontrol.h"
|
||||
#include "avfcameraviewfindersettingscontrol.h"
|
||||
#include "avfimageencodercontrol.h"
|
||||
|
||||
#ifdef Q_OS_IOS
|
||||
#include "avfcamerazoomcontrol.h"
|
||||
@@ -87,6 +88,7 @@ AVFCameraService::AVFCameraService(QObject *parent):
|
||||
#endif
|
||||
m_viewfinderSettingsControl2 = new AVFCameraViewfinderSettingsControl2(this);
|
||||
m_viewfinderSettingsControl = new AVFCameraViewfinderSettingsControl(this);
|
||||
m_imageEncoderControl = new AVFImageEncoderControl(this);
|
||||
}
|
||||
|
||||
AVFCameraService::~AVFCameraService()
|
||||
@@ -112,6 +114,7 @@ AVFCameraService::~AVFCameraService()
|
||||
#endif
|
||||
delete m_viewfinderSettingsControl2;
|
||||
delete m_viewfinderSettingsControl;
|
||||
delete m_imageEncoderControl;
|
||||
|
||||
delete m_session;
|
||||
}
|
||||
@@ -152,6 +155,9 @@ QMediaControl *AVFCameraService::requestControl(const char *name)
|
||||
if (qstrcmp(name, QCameraViewfinderSettingsControl_iid) == 0)
|
||||
return m_viewfinderSettingsControl;
|
||||
|
||||
if (qstrcmp(name, QImageEncoderControl_iid) == 0)
|
||||
return m_imageEncoderControl;
|
||||
|
||||
if (qstrcmp(name,QMediaVideoProbeControl_iid) == 0) {
|
||||
AVFMediaVideoProbeControl *videoProbe = 0;
|
||||
videoProbe = new AVFMediaVideoProbeControl(this);
|
||||
|
||||
@@ -98,6 +98,7 @@ Q_SIGNALS:
|
||||
private:
|
||||
static void updateCameraDevices();
|
||||
void attachInputDevices();
|
||||
void applyImageEncoderSettings();
|
||||
void applyViewfinderSettings();
|
||||
|
||||
static QByteArray m_defaultCameraDevice;
|
||||
|
||||
@@ -40,6 +40,7 @@
|
||||
#include "avfaudioinputselectorcontrol.h"
|
||||
#include "avfmediavideoprobecontrol.h"
|
||||
#include "avfcameraviewfindersettingscontrol.h"
|
||||
#include "avfimageencodercontrol.h"
|
||||
|
||||
#include <CoreFoundation/CoreFoundation.h>
|
||||
#include <Foundation/Foundation.h>
|
||||
@@ -276,6 +277,7 @@ void AVFCameraSession::setState(QCamera::State newState)
|
||||
Q_EMIT readyToConfigureConnections();
|
||||
[m_captureSession commitConfiguration];
|
||||
[m_captureSession startRunning];
|
||||
applyImageEncoderSettings();
|
||||
applyViewfinderSettings();
|
||||
}
|
||||
|
||||
@@ -366,12 +368,27 @@ void AVFCameraSession::attachInputDevices()
|
||||
}
|
||||
}
|
||||
|
||||
void AVFCameraSession::applyImageEncoderSettings()
|
||||
{
|
||||
if (AVFImageEncoderControl *control = m_service->imageEncoderControl())
|
||||
control->applySettings();
|
||||
}
|
||||
|
||||
void AVFCameraSession::applyViewfinderSettings()
|
||||
{
|
||||
if (AVFCameraViewfinderSettingsControl2 *control = m_service->viewfinderSettingsControl2()) {
|
||||
QCameraViewfinderSettings settings(control->requestedSettings());
|
||||
// TODO: Adjust the resolution (from image encoder control), updating 'settings'.
|
||||
control->setViewfinderSettings(settings);
|
||||
if (AVFCameraViewfinderSettingsControl2 *vfControl = m_service->viewfinderSettingsControl2()) {
|
||||
QCameraViewfinderSettings vfSettings(vfControl->requestedSettings());
|
||||
if (AVFImageEncoderControl *imControl = m_service->imageEncoderControl()) {
|
||||
const QSize imageResolution(imControl->imageSettings().resolution());
|
||||
if (!imageResolution.isNull() && imageResolution.isValid()) {
|
||||
vfSettings.setResolution(imageResolution);
|
||||
vfControl->setViewfinderSettings(vfSettings);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (!vfSettings.isNull())
|
||||
vfControl->applySettings();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -56,6 +56,7 @@ public:
|
||||
|
||||
QCameraImageCapture::DriveMode driveMode() const { return QCameraImageCapture::SingleImageCapture; }
|
||||
void setDriveMode(QCameraImageCapture::DriveMode ) {}
|
||||
AVCaptureStillImageOutput *stillImageOutput() const {return m_stillImageOutput;}
|
||||
|
||||
int capture(const QString &fileName);
|
||||
void cancelCapture();
|
||||
|
||||
77
src/plugins/avfoundation/camera/avfimageencodercontrol.h
Normal file
77
src/plugins/avfoundation/camera/avfimageencodercontrol.h
Normal file
@@ -0,0 +1,77 @@
|
||||
/****************************************************************************
|
||||
**
|
||||
** Copyright (C) 2015 The Qt Company Ltd.
|
||||
** Contact: http://www.qt.io/licensing/
|
||||
**
|
||||
** This file is part of the Qt Toolkit.
|
||||
**
|
||||
** $QT_BEGIN_LICENSE:LGPL21$
|
||||
** Commercial License Usage
|
||||
** Licensees holding valid commercial Qt licenses may use this file in
|
||||
** accordance with the commercial license agreement provided with the
|
||||
** Software or, alternatively, in accordance with the terms contained in
|
||||
** a written agreement between you and The Qt Company. For licensing terms
|
||||
** and conditions see http://www.qt.io/terms-conditions. For further
|
||||
** information use the contact form at http://www.qt.io/contact-us.
|
||||
**
|
||||
** GNU Lesser General Public License Usage
|
||||
** Alternatively, this file may be used under the terms of the GNU Lesser
|
||||
** General Public License version 2.1 or version 3 as published by the Free
|
||||
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||
** following information to ensure the GNU Lesser General Public License
|
||||
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||
**
|
||||
** As a special exception, The Qt Company gives you certain additional
|
||||
** rights. These rights are described in The Qt Company LGPL Exception
|
||||
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||
**
|
||||
** $QT_END_LICENSE$
|
||||
**
|
||||
****************************************************************************/
|
||||
|
||||
#ifndef AVFIMAGEENCODERCONTROL_H
|
||||
#define AVFIMAGEENCODERCONTROL_H
|
||||
|
||||
#include <QtMultimedia/qmediaencodersettings.h>
|
||||
#include <QtMultimedia/qimageencodercontrol.h>
|
||||
|
||||
#include <QtCore/qglobal.h>
|
||||
#include <QtCore/qstring.h>
|
||||
#include <QtCore/qlist.h>
|
||||
|
||||
@class AVCaptureDeviceFormat;
|
||||
|
||||
QT_BEGIN_NAMESPACE
|
||||
|
||||
class AVFCameraService;
|
||||
|
||||
class AVFImageEncoderControl : public QImageEncoderControl
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
friend class AVFCameraSession;
|
||||
public:
|
||||
AVFImageEncoderControl(AVFCameraService *service);
|
||||
|
||||
QStringList supportedImageCodecs() const Q_DECL_OVERRIDE;
|
||||
QString imageCodecDescription(const QString &codecName) const Q_DECL_OVERRIDE;
|
||||
QList<QSize> supportedResolutions(const QImageEncoderSettings &settings,
|
||||
bool *continuous) const Q_DECL_OVERRIDE;
|
||||
QImageEncoderSettings imageSettings() const Q_DECL_OVERRIDE;
|
||||
void setImageSettings(const QImageEncoderSettings &settings) Q_DECL_OVERRIDE;
|
||||
|
||||
private:
|
||||
AVFCameraService *m_service;
|
||||
QImageEncoderSettings m_settings;
|
||||
|
||||
void applySettings();
|
||||
bool videoCaptureDeviceIsValid() const;
|
||||
};
|
||||
|
||||
QSize qt_image_high_resolution(AVCaptureDeviceFormat *fomat);
|
||||
|
||||
QT_END_NAMESPACE
|
||||
|
||||
#endif
|
||||
273
src/plugins/avfoundation/camera/avfimageencodercontrol.mm
Normal file
273
src/plugins/avfoundation/camera/avfimageencodercontrol.mm
Normal file
@@ -0,0 +1,273 @@
|
||||
/****************************************************************************
|
||||
**
|
||||
** Copyright (C) 2015 The Qt Company Ltd.
|
||||
** Contact: http://www.qt.io/licensing/
|
||||
**
|
||||
** This file is part of the Qt Toolkit.
|
||||
**
|
||||
** $QT_BEGIN_LICENSE:LGPL21$
|
||||
** Commercial License Usage
|
||||
** Licensees holding valid commercial Qt licenses may use this file in
|
||||
** accordance with the commercial license agreement provided with the
|
||||
** Software or, alternatively, in accordance with the terms contained in
|
||||
** a written agreement between you and The Qt Company. For licensing terms
|
||||
** and conditions see http://www.qt.io/terms-conditions. For further
|
||||
** information use the contact form at http://www.qt.io/contact-us.
|
||||
**
|
||||
** GNU Lesser General Public License Usage
|
||||
** Alternatively, this file may be used under the terms of the GNU Lesser
|
||||
** General Public License version 2.1 or version 3 as published by the Free
|
||||
** Software Foundation and appearing in the file LICENSE.LGPLv21 and
|
||||
** LICENSE.LGPLv3 included in the packaging of this file. Please review the
|
||||
** following information to ensure the GNU Lesser General Public License
|
||||
** requirements will be met: https://www.gnu.org/licenses/lgpl.html and
|
||||
** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
|
||||
**
|
||||
** As a special exception, The Qt Company gives you certain additional
|
||||
** rights. These rights are described in The Qt Company LGPL Exception
|
||||
** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
|
||||
**
|
||||
** $QT_END_LICENSE$
|
||||
**
|
||||
****************************************************************************/
|
||||
|
||||
#include "avfcameraviewfindersettingscontrol.h"
|
||||
#include "avfimageencodercontrol.h"
|
||||
#include "avfimagecapturecontrol.h"
|
||||
#include "avfcamerautility.h"
|
||||
#include "avfcamerasession.h"
|
||||
#include "avfcameraservice.h"
|
||||
#include "avfcameradebug.h"
|
||||
|
||||
#include <QtMultimedia/qmediaencodersettings.h>
|
||||
|
||||
#include <QtCore/qsysinfo.h>
|
||||
#include <QtCore/qdebug.h>
|
||||
|
||||
#include <AVFoundation/AVFoundation.h>
|
||||
|
||||
QT_BEGIN_NAMESPACE
|
||||
|
||||
QSize qt_image_high_resolution(AVCaptureDeviceFormat *format)
|
||||
{
|
||||
Q_ASSERT(format);
|
||||
QSize res;
|
||||
#if defined(Q_OS_IOS) && QT_IOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__IPHONE_8_0)
|
||||
if (QSysInfo::MacintoshVersion >= QSysInfo::MV_IOS_8_0) {
|
||||
const CMVideoDimensions hrDim(format.highResolutionStillImageDimensions);
|
||||
res.setWidth(hrDim.width);
|
||||
res.setHeight(hrDim.height);
|
||||
}
|
||||
#endif
|
||||
return res;
|
||||
}
|
||||
|
||||
AVFImageEncoderControl::AVFImageEncoderControl(AVFCameraService *service)
|
||||
: m_service(service)
|
||||
{
|
||||
Q_ASSERT(service);
|
||||
}
|
||||
|
||||
QStringList AVFImageEncoderControl::supportedImageCodecs() const
|
||||
{
|
||||
return QStringList() << QLatin1String("jpeg");
|
||||
}
|
||||
|
||||
QString AVFImageEncoderControl::imageCodecDescription(const QString &codecName) const
|
||||
{
|
||||
if (codecName == QLatin1String("jpeg"))
|
||||
return tr("JPEG image");
|
||||
|
||||
return QString();
|
||||
}
|
||||
|
||||
QList<QSize> AVFImageEncoderControl::supportedResolutions(const QImageEncoderSettings &settings,
|
||||
bool *continuous) const
|
||||
{
|
||||
Q_UNUSED(settings)
|
||||
|
||||
QList<QSize> resolutions;
|
||||
|
||||
if (!videoCaptureDeviceIsValid())
|
||||
return resolutions;
|
||||
|
||||
#if QT_MAC_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_7, __IPHONE_7_0)
|
||||
if (QSysInfo::MacintoshVersion >= qt_OS_limit(QSysInfo::MV_10_7, QSysInfo::MV_IOS_7_0)) {
|
||||
AVCaptureDevice *captureDevice = m_service->session()->videoCaptureDevice();
|
||||
for (AVCaptureDeviceFormat *format in captureDevice.formats) {
|
||||
if (qt_is_video_range_subtype(format))
|
||||
continue;
|
||||
|
||||
const QSize res(qt_device_format_resolution(format));
|
||||
if (!res.isNull() && res.isValid())
|
||||
resolutions << res;
|
||||
#if defined(Q_OS_IOS) && QT_IOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__IPHONE_8_0)
|
||||
if (QSysInfo::MacintoshVersion >= QSysInfo::MV_IOS_8_0) {
|
||||
// From Apple's docs (iOS):
|
||||
// By default, AVCaptureStillImageOutput emits images with the same dimensions as
|
||||
// its source AVCaptureDevice instance’s activeFormat.formatDescription. However,
|
||||
// if you set this property to YES, the receiver emits still images at the capture
|
||||
// device’s highResolutionStillImageDimensions value.
|
||||
const QSize hrRes(qt_image_high_resolution(format));
|
||||
if (!hrRes.isNull() && hrRes.isValid())
|
||||
resolutions << res;
|
||||
}
|
||||
#endif
|
||||
}
|
||||
} else {
|
||||
#else
|
||||
{
|
||||
#endif
|
||||
// TODO: resolutions without AVCaptureDeviceFormat ...
|
||||
}
|
||||
|
||||
if (continuous)
|
||||
*continuous = false;
|
||||
|
||||
return resolutions;
|
||||
}
|
||||
|
||||
QImageEncoderSettings AVFImageEncoderControl::imageSettings() const
|
||||
{
|
||||
QImageEncoderSettings settings;
|
||||
|
||||
if (!videoCaptureDeviceIsValid())
|
||||
return settings;
|
||||
|
||||
#if QT_MAC_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_7, __IPHONE_7_0)
|
||||
if (QSysInfo::MacintoshVersion >= qt_OS_limit(QSysInfo::MV_10_7, QSysInfo::MV_IOS_7_0)) {
|
||||
AVCaptureDevice *captureDevice = m_service->session()->videoCaptureDevice();
|
||||
if (!captureDevice.activeFormat) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "no active format";
|
||||
return settings;
|
||||
}
|
||||
|
||||
QSize res(qt_device_format_resolution(captureDevice.activeFormat));
|
||||
#if defined(Q_OS_IOS) && QT_IOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__IPHONE_8_0)
|
||||
if (QSysInfo::MacintoshVersion >= QSysInfo::MV_IOS_8_0) {
|
||||
if (!m_service->imageCaptureControl() || !m_service->imageCaptureControl()->stillImageOutput()) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "no still image output";
|
||||
return settings;
|
||||
}
|
||||
|
||||
AVCaptureStillImageOutput *stillImageOutput = m_service->imageCaptureControl()->stillImageOutput();
|
||||
if (stillImageOutput.highResolutionStillImageOutputEnabled)
|
||||
res = qt_image_high_resolution(captureDevice.activeFormat);
|
||||
}
|
||||
#endif
|
||||
if (res.isNull() || !res.isValid()) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "failed to exctract the image resolution";
|
||||
return settings;
|
||||
}
|
||||
|
||||
settings.setResolution(res);
|
||||
} else {
|
||||
#else
|
||||
{
|
||||
#endif
|
||||
// TODO: resolution without AVCaptureDeviceFormat.
|
||||
}
|
||||
|
||||
settings.setCodec(QLatin1String("jpeg"));
|
||||
|
||||
return settings;
|
||||
}
|
||||
|
||||
void AVFImageEncoderControl::setImageSettings(const QImageEncoderSettings &settings)
|
||||
{
|
||||
if (m_settings == settings || settings.isNull())
|
||||
return;
|
||||
|
||||
m_settings = settings;
|
||||
|
||||
applySettings();
|
||||
}
|
||||
|
||||
void AVFImageEncoderControl::applySettings()
|
||||
{
|
||||
if (!videoCaptureDeviceIsValid())
|
||||
return;
|
||||
|
||||
AVFCameraSession *session = m_service->session();
|
||||
if (!session || (session->state() != QCamera::ActiveState
|
||||
&& session->state() != QCamera::LoadedState)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (!m_service->imageCaptureControl()
|
||||
|| !m_service->imageCaptureControl()->stillImageOutput()) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "no still image output";
|
||||
return;
|
||||
}
|
||||
|
||||
if (m_settings.codec().size()
|
||||
&& m_settings.codec() != QLatin1String("jpeg")) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "unsupported codec:" << m_settings.codec();
|
||||
return;
|
||||
}
|
||||
|
||||
QSize res(m_settings.resolution());
|
||||
if (res.isNull()) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "invalid resolution:" << res;
|
||||
return;
|
||||
}
|
||||
|
||||
if (!res.isValid()) {
|
||||
// Invalid == default value.
|
||||
// Here we could choose the best format available, but
|
||||
// activeFormat is already equal to 'preset high' by default,
|
||||
// which is good enough, otherwise we can end in some format with low framerates.
|
||||
return;
|
||||
}
|
||||
|
||||
#if QT_MAC_PLATFORM_SDK_EQUAL_OR_ABOVE(__MAC_10_7, __IPHONE_7_0)
|
||||
if (QSysInfo::MacintoshVersion >= qt_OS_limit(QSysInfo::MV_10_7, QSysInfo::MV_IOS_7_0)) {
|
||||
AVCaptureDevice *captureDevice = m_service->session()->videoCaptureDevice();
|
||||
AVCaptureDeviceFormat *match = qt_find_best_resolution_match(captureDevice, res);
|
||||
|
||||
if (!match) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "unsupported resolution:" << res;
|
||||
return;
|
||||
}
|
||||
|
||||
if (match != captureDevice.activeFormat) {
|
||||
const AVFConfigurationLock lock(captureDevice);
|
||||
if (!lock) {
|
||||
qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration";
|
||||
return;
|
||||
}
|
||||
captureDevice.activeFormat = match;
|
||||
}
|
||||
|
||||
#if defined(Q_OS_IOS) && QT_IOS_PLATFORM_SDK_EQUAL_OR_ABOVE(__IPHONE_8_0)
|
||||
if (QSysInfo::MacintoshVersion >= QSysInfo::MV_IOS_8_0) {
|
||||
AVCaptureStillImageOutput *imageOutput = m_service->imageCaptureControl()->stillImageOutput();
|
||||
if (res == qt_image_high_resolution(captureDevice.activeFormat))
|
||||
imageOutput.highResolutionStillImageOutputEnabled = YES;
|
||||
else
|
||||
imageOutput.highResolutionStillImageOutputEnabled = NO;
|
||||
}
|
||||
#endif
|
||||
} else {
|
||||
#else
|
||||
{
|
||||
#endif
|
||||
// TODO: resolution without capture device format ...
|
||||
}
|
||||
}
|
||||
|
||||
bool AVFImageEncoderControl::videoCaptureDeviceIsValid() const
|
||||
{
|
||||
if (!m_service->session() || !m_service->session()->videoCaptureDevice())
|
||||
return false;
|
||||
|
||||
AVCaptureDevice *captureDevice = m_service->session()->videoCaptureDevice();
|
||||
if (!captureDevice.formats || !captureDevice.formats.count)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
QT_END_NAMESPACE
|
||||
|
||||
#include "moc_avfimageencodercontrol.cpp"
|
||||
@@ -40,7 +40,8 @@ HEADERS += \
|
||||
avfcamerafocuscontrol.h \
|
||||
avfcameraexposurecontrol.h \
|
||||
avfcamerautility.h \
|
||||
avfcameraviewfindersettingscontrol.h
|
||||
avfcameraviewfindersettingscontrol.h \
|
||||
avfimageencodercontrol.h
|
||||
|
||||
OBJECTIVE_SOURCES += \
|
||||
avfcameraserviceplugin.mm \
|
||||
@@ -60,7 +61,8 @@ OBJECTIVE_SOURCES += \
|
||||
avfcamerafocuscontrol.mm \
|
||||
avfcameraexposurecontrol.mm \
|
||||
avfcamerautility.mm \
|
||||
avfcameraviewfindersettingscontrol.mm
|
||||
avfcameraviewfindersettingscontrol.mm \
|
||||
avfimageencodercontrol.mm
|
||||
|
||||
ios {
|
||||
|
||||
|
||||
Reference in New Issue
Block a user