/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** 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 https://www.qt.io/terms-conditions. For further ** information use the contact form at https://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 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "avfcamerafocuscontrol.h" #include "avfcamerautility.h" #include "avfcameraservice.h" #include "avfcamerasession.h" #include "avfcameradebug.h" #include #include QT_BEGIN_NAMESPACE namespace { bool qt_focus_mode_supported(QCameraFocus::FocusModes mode) { // Check if QCameraFocus::FocusMode has counterpart in AVFoundation. // AVFoundation has 'Manual', 'Auto' and 'Continuous', // where 'Manual' is actually 'Locked' + writable property 'lensPosition'. // Since Qt does not provide an API to manipulate a lens position, 'Maual' mode // (at the moment) is not supported. return mode == QCameraFocus::AutoFocus || mode == QCameraFocus::ContinuousFocus; } bool qt_focus_point_mode_supported(QCameraFocus::FocusPointMode mode) { return mode == QCameraFocus::FocusPointAuto || mode == QCameraFocus::FocusPointCustom || mode == QCameraFocus::FocusPointCenter; } AVCaptureFocusMode avf_focus_mode(QCameraFocus::FocusModes requestedMode) { if (requestedMode == QCameraFocus::AutoFocus) return AVCaptureFocusModeAutoFocus; return AVCaptureFocusModeContinuousAutoFocus; } } AVFCameraFocusControl::AVFCameraFocusControl(AVFCameraService *service) : m_session(service->session()), m_focusMode(QCameraFocus::ContinuousFocus), m_focusPointMode(QCameraFocus::FocusPointAuto), m_customFocusPoint(0.5f, 0.5f), m_actualFocusPoint(m_customFocusPoint) { Q_ASSERT(m_session); connect(m_session, SIGNAL(stateChanged(QCamera::State)), SLOT(cameraStateChanged())); } QCameraFocus::FocusModes AVFCameraFocusControl::focusMode() const { return m_focusMode; } void AVFCameraFocusControl::setFocusMode(QCameraFocus::FocusModes mode) { if (m_focusMode == mode) return; AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice) { if (qt_focus_mode_supported(mode)) { m_focusMode = mode; Q_EMIT focusModeChanged(m_focusMode); } else { qDebugCamera() << Q_FUNC_INFO << "focus mode not supported"; } return; } if (isFocusModeSupported(mode)) { const AVFConfigurationLock lock(captureDevice); if (!lock) { qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration"; return; } captureDevice.focusMode = avf_focus_mode(mode); m_focusMode = mode; } else { qDebugCamera() << Q_FUNC_INFO << "focus mode not supported"; return; } Q_EMIT focusModeChanged(m_focusMode); } bool AVFCameraFocusControl::isFocusModeSupported(QCameraFocus::FocusModes mode) const { AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice) return false; if (!qt_focus_mode_supported(mode)) return false; return [captureDevice isFocusModeSupported:avf_focus_mode(mode)]; } QCameraFocus::FocusPointMode AVFCameraFocusControl::focusPointMode() const { return m_focusPointMode; } void AVFCameraFocusControl::setFocusPointMode(QCameraFocus::FocusPointMode mode) { if (m_focusPointMode == mode) return; AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice) { if (qt_focus_point_mode_supported(mode)) { m_focusPointMode = mode; Q_EMIT focusPointModeChanged(mode); } return; } if (isFocusPointModeSupported(mode)) { const AVFConfigurationLock lock(captureDevice); if (!lock) { qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration"; return; } bool resetPOI = false; if (mode == QCameraFocus::FocusPointCenter || mode == QCameraFocus::FocusPointAuto) { if (m_actualFocusPoint != QPointF(0.5, 0.5)) { m_actualFocusPoint = QPointF(0.5, 0.5); resetPOI = true; } } else if (mode == QCameraFocus::FocusPointCustom) { if (m_actualFocusPoint != m_customFocusPoint) { m_actualFocusPoint = m_customFocusPoint; resetPOI = true; } } // else for any other mode in future. if (resetPOI) { const CGPoint focusPOI = CGPointMake(m_actualFocusPoint.x(), m_actualFocusPoint.y()); [captureDevice setFocusPointOfInterest:focusPOI]; } m_focusPointMode = mode; } else { qDebugCamera() << Q_FUNC_INFO << "focus point mode is not supported"; return; } Q_EMIT focusPointModeChanged(mode); } bool AVFCameraFocusControl::isFocusPointModeSupported(QCameraFocus::FocusPointMode mode) const { AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice) return false; if (!qt_focus_point_mode_supported(mode)) return false; return [captureDevice isFocusPointOfInterestSupported]; } QPointF AVFCameraFocusControl::customFocusPoint() const { return m_customFocusPoint; } void AVFCameraFocusControl::setCustomFocusPoint(const QPointF &point) { if (m_customFocusPoint == point) return; if (!QRectF(0.f, 0.f, 1.f, 1.f).contains(point)) { qDebugCamera() << Q_FUNC_INFO << "invalid focus point (out of range)"; return; } m_customFocusPoint = point; Q_EMIT customFocusPointChanged(m_customFocusPoint); AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice || m_focusPointMode != QCameraFocus::FocusPointCustom) return; if ([captureDevice isFocusPointOfInterestSupported]) { const AVFConfigurationLock lock(captureDevice); if (!lock) { qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration"; return; } m_actualFocusPoint = m_customFocusPoint; const CGPoint focusPOI = CGPointMake(point.x(), point.y()); [captureDevice setFocusPointOfInterest:focusPOI]; if (m_focusMode != QCameraFocus::ContinuousFocus) [captureDevice setFocusMode:AVCaptureFocusModeAutoFocus]; } else { qDebugCamera() << Q_FUNC_INFO << "focus point of interest not supported"; return; } } QCameraFocusZoneList AVFCameraFocusControl::focusZones() const { // Unsupported. return QCameraFocusZoneList(); } void AVFCameraFocusControl::cameraStateChanged() { if (m_session->state() != QCamera::ActiveState) return; AVCaptureDevice *captureDevice = m_session->videoCaptureDevice(); if (!captureDevice) { qDebugCamera() << Q_FUNC_INFO << "capture device is nil in 'active' state"; return; } const AVFConfigurationLock lock(captureDevice); if (m_customFocusPoint != m_actualFocusPoint && m_focusPointMode == QCameraFocus::FocusPointCustom) { if (![captureDevice isFocusPointOfInterestSupported]) { qDebugCamera() << Q_FUNC_INFO << "focus point of interest not supported"; return; } if (!lock) { qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration"; return; } m_actualFocusPoint = m_customFocusPoint; const CGPoint focusPOI = CGPointMake(m_customFocusPoint.x(), m_customFocusPoint.y()); [captureDevice setFocusPointOfInterest:focusPOI]; } if (m_focusMode != QCameraFocus::ContinuousFocus) { const AVCaptureFocusMode avMode = avf_focus_mode(m_focusMode); if (captureDevice.focusMode != avMode) { if (![captureDevice isFocusModeSupported:avMode]) { qDebugCamera() << Q_FUNC_INFO << "focus mode not supported"; return; } if (!lock) { qDebugCamera() << Q_FUNC_INFO << "failed to lock for configuration"; return; } [captureDevice setFocusMode:avMode]; } } } QT_END_NAMESPACE #include "moc_avfcamerafocuscontrol.cpp"