Enroute Flight Navigation
A navigation app for VFR pilots
WeatherDataProvider.h
1/***************************************************************************
2 * Copyright (C) 2019-2025 by Stefan Kebekus *
3 * stefan.kebekus@gmail.com *
4 * *
5 * This program is free software; you can redistribute it and/or modify *
6 * it under the terms of the GNU General Public License as published by *
7 * the Free Software Foundation; either version 3 of the License, or *
8 * (at your option) any later version. *
9 * *
10 * This program is distributed in the hope that it will be useful, *
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
13 * GNU General Public License for more details. *
14 * *
15 * You should have received a copy of the GNU General Public License *
16 * along with this program; if not, write to the *
17 * Free Software Foundation, Inc., *
18 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
19 ***************************************************************************/
20
21#pragma once
22
23#include <QGeoRectangle>
24#include <QProperty>
25#include <QTimer>
26
27#include "GlobalObject.h"
28#include "geomaps/Waypoint.h"
29#include "navigation/Atmosphere.h"
30#include "weather/METAR.h"
31#include "weather/TAF.h"
32
33class QNetworkReply;
34
35
36namespace Weather {
37
54class WeatherDataProvider : public QObject {
55 Q_OBJECT
56 QML_ELEMENT
57 QML_SINGLETON
58
59 class updateLogEntry;
60 friend QDataStream& operator<<(QDataStream& stream, const updateLogEntry& ule);
61 friend QDataStream& operator>>(QDataStream& stream, updateLogEntry& ule);
62
63public:
68 explicit WeatherDataProvider(QObject *parent = nullptr);
69
70 // No default constructor, important for QML singleton
71 explicit WeatherDataProvider() = delete;
72
73 // factory function for QML singleton
74 static Weather::WeatherDataProvider* create(QQmlEngine* /*unused*/, QJSEngine* /*unused*/)
75 {
77 }
78
79
80 //
81 // Properties
82 //
83
89 Q_PROPERTY(bool downloading READ downloading NOTIFY downloadingChanged)
90
91
96 Q_PROPERTY(Units::Pressure QNH READ QNH NOTIFY QNHInfoChanged)
97
104
112 Q_PROPERTY(QString QNHInfo READ QNHInfo NOTIFY QNHInfoChanged)
113
121 Q_PROPERTY(QString sunInfo READ sunInfo NOTIFY sunInfoChanged)
122
124 Q_PROPERTY(QMap<QString,Weather::METAR> METARs READ METARs BINDABLE bindableMETARs)
125
127 Q_PROPERTY(QMap<QString,Weather::TAF> TAFs READ TAFs BINDABLE bindableTAFs)
128
129
130 //
131 // Getter Methods
132 //
133
138 [[nodiscard]] bool downloading() const;
139
144 QMap<QString, Weather::METAR> METARs() {return m_METARs.value();}
145
150 QBindable<QMap<QString, Weather::METAR>> bindableMETARs() {return &m_METARs;}
151
156 [[nodiscard]] Units::Pressure QNH() const;
157
163
168 [[nodiscard]] QString QNHInfo() const;
169
174 static QString sunInfo();
175
180 QMap<QString, Weather::TAF> TAFs() {return m_TAFs.value();}
181
186 QBindable<QMap<QString, Weather::TAF>> bindableTAFs() {return &m_TAFs;}
187
188
189 //
190 // Methods
191 //
192
203 Q_INVOKABLE void requestUpdate();
204
217 Q_INVOKABLE void requestUpdate4Waypoint(const GeoMaps::Waypoint& wp);
218
219signals:
222
230 void error(QString message);
231
234
237
238private slots:
239 // Called when a download is finished
240 void downloadFinished();
241
242 // Check for expired METARs and TAFs and delete them. This also deletes
243 // weather stations if they are no longer in use.
244 void deleteExpiredMesages();
245
246 // Name says it all. This method is called from the constructor, but with a
247 // little lag to avoid conflicts in the initialisation of static objects.
248 void deferredInitialization();
249
250 // This method is called by requestUpdate and requestUpdate4Waypoint. It
251 // does the actual network request.
252 void startDownload(const QGeoRectangle& bBox);
253
254private:
255 Q_DISABLE_COPY_MOVE(WeatherDataProvider)
256
257 // Update interval is 30 mins, or 5 mins if update failed
258 static const int updateIntervalNormal_ms = 30*60*1000;
259 static const int updateIntervalOnError_ms = 5*60*1000;
260
261 // This method loads METAR/TAFs from a file "weather.dat" in
262 // QStandardPaths::AppDataLocation. There is locking to ensure that no two
263 // processes access the file. The method will fail silently on error.
264 // Returns true on success and false on failure.
265 bool load();
266
267 // This method saves all METAR/TAFs that are valid and not yet expired to a
268 // file "weather.dat" in QStandardPaths::AppDataLocation. There is locking
269 // to ensure that no two processes access the file. The method will fail
270 // silently on error.
271 void save();
272
273 // List of replies from aviationweather.com
274 QList<QPointer<QNetworkReply>> m_networkReplies;
275
276 // A timer used for auto-updating the weather reports every 30 minutes
277 QTimer m_updateTimer;
278
279 // A timer used for deleting expired weather reports ever 11 minutes
280 QTimer m_deleteExiredMessagesTimer;
281
282 // METARs and TAFs by ICAO Code
283 QProperty<QMap<QString, Weather::METAR>> m_METARs;
284 QProperty<QMap<QString, Weather::TAF>> m_TAFs;
285
286 // Time and BBox of the last succesful METAR update for the current region and flight route
287 struct updateLogEntry
288 {
289 QDateTime m_time;
290 QGeoRectangle m_bBox;
291 };
292 QList<updateLogEntry> updateLog;
293};
294
295QDataStream& operator<<(QDataStream& stream, const WeatherDataProvider::updateLogEntry& ule);
296
297QDataStream& operator>>(QDataStream& stream, WeatherDataProvider::updateLogEntry& ule);
298
299} // namespace Weather
Waypoint, such as an airfield, a navaid station or a reporting point.
Definition Waypoint.h:41
static Q_INVOKABLE Weather::WeatherDataProvider * weatherDataProvider()
Pointer to appplication-wide static WeatherDataProvider instance.
static Q_INVOKABLE Units::Distance height(Units::Density d)
Computation of height as a function of density.
Convenience class for distance computations.
Definition Distance.h:35
Convenience class for pressure computations.
Definition Pressure.h:35
METAR report.
Definition METAR.h:47
TAF forecast.
Definition TAF.h:42
WeatherDataProvider, weather service manager.
QBindable< QMap< QString, Weather::TAF > > bindableTAFs()
Getter method for property of the same name.
WeatherDataProvider(QObject *parent=nullptr)
Standard constructor.
QMap< QString, Weather::TAF > TAFs()
Getter method for property of the same name.
Q_INVOKABLE void requestUpdate4Waypoint(const GeoMaps::Waypoint &wp)
Request update.
void downloadingChanged()
Notifier signal.
void QNHInfoChanged()
Notifier signal.
static QString sunInfo()
Getter method for property of the same name.
bool downloading
Downloading flag.
QString QNHInfo() const
Getter method for property of the same name.
void sunInfoChanged()
Notifier signal.
void error(QString message)
Signal emitted when a network error occurs.
Units::Pressure QNH() const
Getter method for property of the same name.
Q_INVOKABLE void requestUpdate()
Request update.
QBindable< QMap< QString, Weather::METAR > > bindableMETARs()
Getter method for property of the same name.
QMap< QString, Weather::METAR > METARs
List of METARs.
Units::Distance QNHPressureAltitude() const
Getter method for property of the same name.
Units::Distance QNHPressureAltitude
QNHPressureAltitude.
QMap< QString, Weather::TAF > TAFs
List of TAFs.
Conversion between units used in aviation.
Definition Angle.h:34