Skip to content

Commit

Permalink
Cleanup GPS fix status handling, remove duplicate code
Browse files Browse the repository at this point in the history
  • Loading branch information
nyalldawson committed Oct 31, 2022
1 parent bf44964 commit f79eaf6
Show file tree
Hide file tree
Showing 11 changed files with 290 additions and 68 deletions.
17 changes: 17 additions & 0 deletions python/core/auto_additions/qgis.py
Expand Up @@ -850,6 +850,23 @@
Qgis.GpsConnectionType.__doc__ = 'GPS connection types.\n\n.. versionadded:: 3.30\n\n' + '* ``Automatic``: ' + Qgis.GpsConnectionType.Automatic.__doc__ + '\n' + '* ``Internal``: ' + Qgis.GpsConnectionType.Internal.__doc__ + '\n' + '* ``Serial``: ' + Qgis.GpsConnectionType.Serial.__doc__ + '\n' + '* ``Gpsd``: ' + Qgis.GpsConnectionType.Gpsd.__doc__
# --
Qgis.GpsConnectionType.baseClass = Qgis
QgsGpsInformation.FixStatus = Qgis.GpsFixStatus
# monkey patching scoped based enum
QgsGpsInformation.NoData = Qgis.GpsFixStatus.NoData
QgsGpsInformation.NoData.is_monkey_patched = True
QgsGpsInformation.NoData.__doc__ = "No fix data available"
QgsGpsInformation.NoFix = Qgis.GpsFixStatus.NoFix
QgsGpsInformation.NoFix.is_monkey_patched = True
QgsGpsInformation.NoFix.__doc__ = "GPS is not fixed"
QgsGpsInformation.Fix2D = Qgis.GpsFixStatus.Fix2D
QgsGpsInformation.Fix2D.is_monkey_patched = True
QgsGpsInformation.Fix2D.__doc__ = "2D fix"
QgsGpsInformation.Fix3D = Qgis.GpsFixStatus.Fix3D
QgsGpsInformation.Fix3D.is_monkey_patched = True
QgsGpsInformation.Fix3D.__doc__ = "3D fix"
Qgis.GpsFixStatus.__doc__ = 'GPS fix status.\n\n.. note::\n\n Prior to QGIS 3.30 this was available as :py:class:`QgsGpsInformation`.FixStatus\n\n.. versionadded:: 3.30\n\n' + '* ``NoData``: ' + Qgis.GpsFixStatus.NoData.__doc__ + '\n' + '* ``NoFix``: ' + Qgis.GpsFixStatus.NoFix.__doc__ + '\n' + '* ``Fix2D``: ' + Qgis.GpsFixStatus.Fix2D.__doc__ + '\n' + '* ``Fix3D``: ' + Qgis.GpsFixStatus.Fix3D.__doc__
# --
Qgis.GpsFixStatus.baseClass = Qgis
# monkey patching scoped based enum
Qgis.GpsQualityIndicator.Unknown.__doc__ = "Unknown"
Qgis.GpsQualityIndicator.Invalid.__doc__ = "Invalid"
Expand Down
11 changes: 2 additions & 9 deletions python/core/auto_generated/gps/qgsgpsconnection.sip.in
Expand Up @@ -56,14 +56,6 @@ Encapsulates information relating to a GPS position fix.
%End
public:

enum FixStatus
{
NoData,
NoFix,
Fix2D,
Fix3D
};

double latitude;

double longitude;
Expand Down Expand Up @@ -117,7 +109,7 @@ Returns whether the connection information is valid
.. versionadded:: 3.10
%End

FixStatus fixStatus() const;
Qgis::GpsFixStatus fixStatus() const;
%Docstring
Returns the fix status

Expand Down Expand Up @@ -213,6 +205,7 @@ Parse available data source content
%End
};


/************************************************************************
* This file has been generated automatically from *
* *
Expand Down
8 changes: 8 additions & 0 deletions python/core/auto_generated/qgis.sip.in
Expand Up @@ -572,6 +572,14 @@ The development version
Gpsd,
};

enum class GpsFixStatus
{
NoData,
NoFix,
Fix2D,
Fix3D
};

enum class GpsQualityIndicator
{
Unknown,
Expand Down
42 changes: 10 additions & 32 deletions src/app/gps/qgsgpsinformationwidget.cpp
Expand Up @@ -276,7 +276,7 @@ QgsGpsInformationWidget::QgsGpsInformationWidget( QgsAppGpsConnection *connectio
mBtnDebug->setVisible( mySettings.value( QStringLiteral( "showDebug" ), "false", QgsSettings::Gps ).toBool() ); // use a registry setting to control - power users/devs could set it

// status = unknown
setStatusIndicator( NoData );
setStatusIndicator( Qgis::GpsFixStatus::NoData );

//SLM - added functionality
mLogFile = nullptr;
Expand Down Expand Up @@ -652,7 +652,7 @@ void QgsGpsInformationWidget::gpsDisconnected()
mConnectButton->setChecked( false );
mConnectButton->setText( tr( "&Connect" ) );

setStatusIndicator( NoData );
setStatusIndicator( Qgis::GpsFixStatus::NoData );
}

void QgsGpsInformationWidget::displayGPSInformation( const QgsGpsInformation &info )
Expand All @@ -662,32 +662,8 @@ void QgsGpsInformationWidget::displayGPSInformation( const QgsGpsInformation &in

QVector<QPointF> data;

// set validity flag and status from GPS data
// based on GGA, GSA and RMC sentences - the logic does not require all
bool validFlag = false; // true if GPS indicates position fix
FixStatus fixStatus = NoData;

// no fix if any of the three report bad; default values are invalid values and won't be changed if the corresponding NMEA msg is not received
if ( info.status == 'V' || info.fixType == NMEA_FIX_BAD || info.qualityIndicator == Qgis::GpsQualityIndicator::Invalid ) // some sources say that 'V' indicates position fix, but is below acceptable quality
{
fixStatus = NoFix;
}
else if ( info.fixType == NMEA_FIX_2D ) // 2D indication (from GGA)
{
fixStatus = Fix2D;
validFlag = true;
}
else if ( info.status == 'A' || info.fixType == NMEA_FIX_3D || ( info.qualityIndicator != Qgis::GpsQualityIndicator::Invalid ) ) // good
{
fixStatus = Fix3D;
validFlag = true;
}
else // unknown status (not likely)
{

}

// set visual status indicator -- do only on change of state
const Qgis::GpsFixStatus fixStatus = info.fixStatus();
if ( fixStatus != mLastFixStatus )
{
setStatusIndicator( fixStatus );
Expand Down Expand Up @@ -783,6 +759,8 @@ void QgsGpsInformationWidget::displayGPSInformation( const QgsGpsInformation &in
mpSatellitesWidget->replot();
} //satellites
#endif

bool validFlag = info.isValid();
if ( validFlag )
{
validFlag = info.longitude >= -180.0 && info.longitude <= 180.0 && info.latitude >= -90.0 && info.latitude <= 90.0;
Expand Down Expand Up @@ -1380,23 +1358,23 @@ void QgsGpsInformationWidget::layerEditStateChanged()
updateCloseFeatureButton( mLastLayer );
}

void QgsGpsInformationWidget::setStatusIndicator( const FixStatus statusValue )
void QgsGpsInformationWidget::setStatusIndicator( Qgis::GpsFixStatus statusValue )
{
mLastFixStatus = statusValue;
// the pixmap will be expanded to the size of the label
QPixmap status( 4, 4 );
switch ( statusValue )
{
case NoFix:
case Qgis::GpsFixStatus::NoFix:
status.fill( Qt::red );
break;
case Fix2D:
case Qgis::GpsFixStatus::Fix2D:
status.fill( Qt::yellow );
break;
case Fix3D:
case Qgis::GpsFixStatus::Fix3D:
status.fill( Qt::green );
break;
case NoData:
case Qgis::GpsFixStatus::NoData:
status.fill( Qt::darkGray );
}
mLblStatusIndicator->setPixmap( status );
Expand Down
10 changes: 3 additions & 7 deletions src/app/gps/qgsgpsinformationwidget.h
Expand Up @@ -19,7 +19,6 @@

#include "ui_qgsgpsinformationwidgetbase.h"
#include "qgis_app.h"
#include "gmath.h"
#include "info.h"
#include "nmeatime.h"
#include "qgsgpsmarker.h"
Expand Down Expand Up @@ -111,12 +110,9 @@ class APP_EXPORT QgsGpsInformationWidget: public QgsPanelWidget, public QgsMapCa
void gpsConnected();

private:
enum FixStatus //GPS status
{
NoData, NoFix, Fix2D, Fix3D
};

void addVertex();
void setStatusIndicator( FixStatus statusValue );
void setStatusIndicator( Qgis::GpsFixStatus statusValue );
void showStatusBarMessage( const QString &msg );
void updateTimeZones();
QVariant timestamp( QgsVectorLayer *vlayer, int idx );
Expand Down Expand Up @@ -147,7 +143,7 @@ class APP_EXPORT QgsGpsInformationWidget: public QgsPanelWidget, public QgsMapCa
QgsPointXY mSecondLastGpsPosition;
QVector<QgsPoint> mCaptureList;
double mLastElevation = 0.0;
FixStatus mLastFixStatus;
Qgis::GpsFixStatus mLastFixStatus = Qgis::GpsFixStatus::NoData;
QString mDateTimeFormat; // user specified format string in registry (no UI presented)
QPointer< QgsVectorLayer > mLastLayer;
QFile *mLogFile = nullptr;
Expand Down
16 changes: 9 additions & 7 deletions src/core/gps/qgsgpsconnection.cpp
Expand Up @@ -44,22 +44,22 @@ bool QgsGpsInformation::isValid() const
return valid;
}

QgsGpsInformation::FixStatus QgsGpsInformation::fixStatus() const
Qgis::GpsFixStatus QgsGpsInformation::fixStatus() const
{
FixStatus fixStatus = NoData;
Qgis::GpsFixStatus fixStatus = Qgis::GpsFixStatus::NoData;

// no fix if any of the three report bad; default values are invalid values and won't be changed if the corresponding NMEA msg is not received
if ( status == 'V' || fixType == NMEA_FIX_BAD || qualityIndicator == Qgis::GpsQualityIndicator::Invalid ) // some sources say that 'V' indicates position fix, but is below acceptable quality
{
fixStatus = NoFix;
fixStatus = Qgis::GpsFixStatus::NoFix;
}
else if ( fixType == NMEA_FIX_2D ) // 2D indication (from GGA)
{
fixStatus = Fix2D;
fixStatus = Qgis::GpsFixStatus::Fix2D;
}
else if ( status == 'A' || fixType == NMEA_FIX_3D || qualityIndicator != Qgis::GpsQualityIndicator::Invalid ) // good
{
fixStatus = Fix3D;
fixStatus = Qgis::GpsFixStatus::Fix3D;
}
return fixStatus;
}
Expand Down Expand Up @@ -105,8 +105,8 @@ QgsGpsConnection::QgsGpsConnection( QIODevice *dev )
: QObject( nullptr )
, mSource( dev )
{
clearLastGPSInformation();
QObject::connect( dev, &QIODevice::readyRead, this, &QgsGpsConnection::parseData );
if ( mSource )
QObject::connect( mSource.get(), &QIODevice::readyRead, this, &QgsGpsConnection::parseData );
}

QgsGpsConnection::~QgsGpsConnection()
Expand Down Expand Up @@ -153,6 +153,8 @@ void QgsGpsConnection::setSource( QIODevice *source )
{
cleanupSource();
mSource.reset( source );
QObject::connect( mSource.get(), &QIODevice::readyRead, this, &QgsGpsConnection::parseData );

clearLastGPSInformation();
}

Expand Down
16 changes: 3 additions & 13 deletions src/core/gps/qgsgpsconnection.h
Expand Up @@ -122,18 +122,6 @@ class CORE_EXPORT QgsGpsInformation
{
public:

/**
* GPS fix status
* \since QGIS 3.10
*/
enum FixStatus
{
NoData,
NoFix,
Fix2D,
Fix3D
};

/**
* Latitude in decimal degrees, using the WGS84 datum. A positive value indicates the Northern Hemisphere, and
* a negative value indicates the Southern Hemisphere.
Expand Down Expand Up @@ -279,7 +267,7 @@ class CORE_EXPORT QgsGpsInformation
* Returns the fix status
* \since QGIS 3.10
*/
FixStatus fixStatus() const;
Qgis::GpsFixStatus fixStatus() const;

/**
* Returns a descriptive string for the signal quality.
Expand Down Expand Up @@ -389,4 +377,6 @@ class CORE_EXPORT QgsGpsConnection : public QObject
virtual void parseData() = 0;
};

Q_DECLARE_METATYPE( QgsGpsInformation )

#endif // QGSGPSCONNECTION_H
16 changes: 16 additions & 0 deletions src/core/qgis.h
Expand Up @@ -932,6 +932,22 @@ class CORE_EXPORT Qgis
};
Q_ENUM( GpsConnectionType )

/**
* GPS fix status.
*
* \note Prior to QGIS 3.30 this was available as QgsGpsInformation::FixStatus
*
* \since QGIS 3.30
*/
enum class GpsFixStatus SIP_MONKEYPATCH_SCOPEENUM_UNNEST( QgsGpsInformation, FixStatus ) : int
{
NoData, //!< No fix data available
NoFix, //!< GPS is not fixed
Fix2D, //!< 2D fix
Fix3D //!< 3D fix
};
Q_ENUM( GpsFixStatus );

/**
* GPS signal quality indicator
*
Expand Down
2 changes: 2 additions & 0 deletions src/core/qgsapplication.cpp
Expand Up @@ -83,6 +83,7 @@
#include "qgsunsetattributevalue.h"
#include "qgscolorrampimpl.h"
#include "qgsinterval.h"
#include "qgsgpsconnection.h"

#include "gps/qgsgpsconnectionregistry.h"
#include "processing/qgsprocessingregistry.h"
Expand Down Expand Up @@ -323,6 +324,7 @@ void QgsApplication::init( QString profileFolder )
qRegisterMetaType<QMap<QNetworkRequest::KnownHeaders, QVariant>>( "QMap<QNetworkRequest::KnownHeaders,QVariant>" );
qRegisterMetaType<QList<QNetworkReply::RawHeaderPair>>( "QList<QNetworkReply::RawHeaderPair>" );
qRegisterMetaType< QAuthenticator * >( "QAuthenticator*" );
qRegisterMetaType< QgsGpsInformation >( "QgsGpsInformation" );
} );

( void ) resolvePkgPath();
Expand Down
1 change: 1 addition & 0 deletions tests/src/core/CMakeLists.txt
Expand Up @@ -128,6 +128,7 @@ set(TESTS
testqgsnetworkaccessmanager.cpp
testqgsnetworkcontentfetcher.cpp
testqgsnewsfeedparser.cpp
testqgsnmeaconnection.cpp
testqgsofflineediting.cpp
testqgsogcutils.cpp
testqgsogrprovider.cpp
Expand Down

0 comments on commit f79eaf6

Please sign in to comment.