/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtPositioning module 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include #include "qgeosatelliteinfosource_android_p.h" #include "jnipositioning.h" Q_DECLARE_METATYPE(QGeoSatelliteInfo) Q_DECLARE_METATYPE(QList) #define UPDATE_FROM_COLD_START 2*60*1000 QGeoSatelliteInfoSourceAndroid::QGeoSatelliteInfoSourceAndroid(QObject *parent) : QGeoSatelliteInfoSource(parent), m_error(NoError), updatesRunning(false) { qRegisterMetaType< QGeoSatelliteInfo >(); qRegisterMetaType< QList >(); androidClassKeyForUpdate = AndroidPositioning::registerPositionInfoSource(this); androidClassKeyForSingleRequest = AndroidPositioning::registerPositionInfoSource(this); requestTimer.setSingleShot(true); QObject::connect(&requestTimer, SIGNAL(timeout()), this, SLOT(requestTimeout())); } QGeoSatelliteInfoSourceAndroid::~QGeoSatelliteInfoSourceAndroid() { stopUpdates(); if (requestTimer.isActive()) { requestTimer.stop(); AndroidPositioning::stopUpdates(androidClassKeyForSingleRequest); } AndroidPositioning::unregisterPositionInfoSource(androidClassKeyForUpdate); AndroidPositioning::unregisterPositionInfoSource(androidClassKeyForSingleRequest); } void QGeoSatelliteInfoSourceAndroid::setUpdateInterval(int msec) { int previousInterval = updateInterval(); msec = (((msec > 0) && (msec < minimumUpdateInterval())) || msec < 0)? minimumUpdateInterval() : msec; if (msec == previousInterval) return; QGeoSatelliteInfoSource::setUpdateInterval(msec); if (updatesRunning) reconfigureRunningSystem(); } int QGeoSatelliteInfoSourceAndroid::minimumUpdateInterval() const { return 1000; } QGeoSatelliteInfoSource::Error QGeoSatelliteInfoSourceAndroid::error() const { return m_error; } void QGeoSatelliteInfoSourceAndroid::startUpdates() { if (updatesRunning) return; updatesRunning = true; QGeoSatelliteInfoSource::Error error = AndroidPositioning::startSatelliteUpdates( androidClassKeyForUpdate, false, updateInterval()); if (error != QGeoSatelliteInfoSource::NoError) { updatesRunning = false; m_error = error; emit QGeoSatelliteInfoSource::error(m_error); } } void QGeoSatelliteInfoSourceAndroid::stopUpdates() { if (!updatesRunning) return; updatesRunning = false; AndroidPositioning::stopUpdates(androidClassKeyForUpdate); } void QGeoSatelliteInfoSourceAndroid::requestUpdate(int timeout) { if (requestTimer.isActive()) return; if (timeout != 0 && timeout < minimumUpdateInterval()) { emit requestTimeout(); return; } if (timeout == 0) timeout = UPDATE_FROM_COLD_START; requestTimer.start(timeout); // if updates already running with interval equal or less then timeout // then we wait for next update coming through // assume that a single update will not be quicker than regular updates anyway if (updatesRunning && updateInterval() <= timeout) return; QGeoSatelliteInfoSource::Error error = AndroidPositioning::startSatelliteUpdates( androidClassKeyForSingleRequest, true, timeout); if (error != QGeoSatelliteInfoSource::NoError) { requestTimer.stop(); m_error = error; emit QGeoSatelliteInfoSource::error(m_error); } } void QGeoSatelliteInfoSourceAndroid::processSatelliteUpdateInView(const QList &satsInView, bool isSingleUpdate) { if (!isSingleUpdate) { //if requested while regular updates were running if (requestTimer.isActive()) requestTimer.stop(); emit QGeoSatelliteInfoSource::satellitesInViewUpdated(satsInView); return; } m_satsInView = satsInView; } void QGeoSatelliteInfoSourceAndroid::processSatelliteUpdateInUse(const QList &satsInUse, bool isSingleUpdate) { if (!isSingleUpdate) { //if requested while regular updates were running if (requestTimer.isActive()) requestTimer.stop(); emit QGeoSatelliteInfoSource::satellitesInUseUpdated(satsInUse); return; } m_satsInUse = satsInUse; } void QGeoSatelliteInfoSourceAndroid::requestTimeout() { AndroidPositioning::stopUpdates(androidClassKeyForSingleRequest); const int count = m_satsInView.count(); if (!count) { emit requestTimeout(); return; } emit QGeoSatelliteInfoSource::satellitesInViewUpdated(m_satsInView); emit QGeoSatelliteInfoSource::satellitesInUseUpdated(m_satsInUse); m_satsInUse.clear(); m_satsInView.clear(); } /* Updates the system assuming that updateInterval and/or preferredPositioningMethod have changed. */ void QGeoSatelliteInfoSourceAndroid::reconfigureRunningSystem() { if (!updatesRunning) return; stopUpdates(); startUpdates(); } void QGeoSatelliteInfoSourceAndroid::locationProviderDisabled() { m_error = QGeoSatelliteInfoSource::ClosedError; emit QGeoSatelliteInfoSource::error(m_error); }