| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // Copyright (C) 2016 Edouard Griffiths, F4EXB                                   //
 | 
					
						
							|  |  |  | // Copyright (C) 2020 Jon Beniston, M7RCE                                        //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is free software; you can redistribute it and/or modify          //
 | 
					
						
							|  |  |  | // it under the terms of the GNU General Public License as published by          //
 | 
					
						
							|  |  |  | // the Free Software Foundation as version 3 of the License, or                  //
 | 
					
						
							|  |  |  | // (at your option) any later version.                                           //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // This program is distributed in the hope that it will be useful,               //
 | 
					
						
							|  |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of                //
 | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the                  //
 | 
					
						
							|  |  |  | // GNU General Public License V3 for more details.                               //
 | 
					
						
							|  |  |  | //                                                                               //
 | 
					
						
							|  |  |  | // You should have received a copy of the GNU General Public License             //
 | 
					
						
							|  |  |  | // along with this program. If not, see <http://www.gnu.org/licenses/>.          //
 | 
					
						
							|  |  |  | ///////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef INCLUDE_VORLOCALIZERGUI_H
 | 
					
						
							|  |  |  | #define INCLUDE_VORLOCALIZERGUI_H
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <QIcon>
 | 
					
						
							|  |  |  | #include <QAbstractListModel>
 | 
					
						
							|  |  |  | #include <QModelIndex>
 | 
					
						
							|  |  |  | #include <QProgressDialog>
 | 
					
						
							|  |  |  | #include <QGeoLocation>
 | 
					
						
							|  |  |  | #include <QGeoCoordinate>
 | 
					
						
							|  |  |  | #include <QTableWidgetItem>
 | 
					
						
							|  |  |  | #include <QPushButton>
 | 
					
						
							|  |  |  | #include <QToolButton>
 | 
					
						
							|  |  |  | #include <QHBoxLayout>
 | 
					
						
							|  |  |  | #include <QMenu>
 | 
					
						
							|  |  |  | #include <QTimer>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "feature/featuregui.h"
 | 
					
						
							|  |  |  | #include "dsp/movingaverage.h"
 | 
					
						
							|  |  |  | #include "util/messagequeue.h"
 | 
					
						
							|  |  |  | #include "util/httpdownloadmanager.h"
 | 
					
						
							|  |  |  | #include "util/azel.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-03 13:44:09 +01:00
										 |  |  | #include "util/openaip.h"
 | 
					
						
							| 
									
										
										
										
											2022-01-09 05:27:12 +01:00
										 |  |  | #include "settings/rollupstate.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | #include "vorlocalizersettings.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class PluginAPI; | 
					
						
							|  |  |  | class FeatureUISet; | 
					
						
							|  |  |  | class Feature; | 
					
						
							|  |  |  | class BasebandSampleSink; | 
					
						
							|  |  |  | class VORLocalizer; | 
					
						
							|  |  |  | class VORLocalizerGUI; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Ui { | 
					
						
							|  |  |  |     class VORLocalizerGUI; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | class VORLocalizerGUI; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Table items for each VOR
 | 
					
						
							|  |  |  | class VORGUI : public QObject { | 
					
						
							|  |  |  |     Q_OBJECT | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     NavAid *m_navAid; | 
					
						
							|  |  |  |     QVariantList m_coordinates; | 
					
						
							|  |  |  |     VORLocalizerGUI *m_gui; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QTableWidgetItem *m_nameItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_frequencyItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_identItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_morseItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_radialItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_rxIdentItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_rxMorseItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_varMagItem; | 
					
						
							|  |  |  |     QTableWidgetItem *m_refMagItem; | 
					
						
							|  |  |  |     QWidget *m_muteItem; | 
					
						
							|  |  |  |     QToolButton *m_muteButton; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VORGUI(NavAid *navAid, VORLocalizerGUI *gui); | 
					
						
							|  |  |  | private slots: | 
					
						
							|  |  |  |     void on_audioMute_toggled(bool checked); | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // VOR model used for each VOR on the map
 | 
					
						
							|  |  |  | class VORModel : public QAbstractListModel { | 
					
						
							|  |  |  |     Q_OBJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using QAbstractListModel::QAbstractListModel; | 
					
						
							|  |  |  |     enum MarkerRoles{ | 
					
						
							|  |  |  |         positionRole = Qt::UserRole + 1, | 
					
						
							|  |  |  |         vorDataRole = Qt::UserRole + 2, | 
					
						
							|  |  |  |         vorImageRole = Qt::UserRole + 3, | 
					
						
							|  |  |  |         vorRadialRole = Qt::UserRole + 4, | 
					
						
							|  |  |  |         bubbleColourRole = Qt::UserRole + 5, | 
					
						
							|  |  |  |         selectedRole = Qt::UserRole + 6 | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VORModel(VORLocalizerGUI *gui) : | 
					
						
							|  |  |  |         m_gui(gui), | 
					
						
							|  |  |  |         m_radialsVisible(true) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Q_INVOKABLE void addVOR(NavAid *vor) { | 
					
						
							|  |  |  |         beginInsertRows(QModelIndex(), rowCount(), rowCount()); | 
					
						
							|  |  |  |         m_vors.append(vor); | 
					
						
							|  |  |  |         m_selected.append(false); | 
					
						
							|  |  |  |         m_radials.append(-1.0f); | 
					
						
							|  |  |  |         m_vorGUIs.append(nullptr); | 
					
						
							|  |  |  |         endInsertRows(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int rowCount(const QModelIndex &parent = QModelIndex()) const override { | 
					
						
							|  |  |  |         Q_UNUSED(parent) | 
					
						
							|  |  |  |         return m_vors.count(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool setData(const QModelIndex &index, const QVariant& value, int role = Qt::EditRole) override; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Qt::ItemFlags flags(const QModelIndex &index) const override { | 
					
						
							|  |  |  |         (void) index; | 
					
						
							|  |  |  |         return Qt::ItemIsEnabled | Qt::ItemIsSelectable | Qt::ItemIsEditable; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void allVORUpdated() { | 
					
						
							|  |  |  |         for (int i = 0; i < m_vors.count(); i++) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             QModelIndex idx = index(i); | 
					
						
							|  |  |  |             emit dataChanged(idx, idx); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void removeVOR(NavAid *vor) { | 
					
						
							|  |  |  |         int row = m_vors.indexOf(vor); | 
					
						
							|  |  |  |         if (row >= 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             beginRemoveRows(QModelIndex(), row, row); | 
					
						
							|  |  |  |             m_vors.removeAt(row); | 
					
						
							|  |  |  |             m_selected.removeAt(row); | 
					
						
							|  |  |  |             m_radials.removeAt(row); | 
					
						
							|  |  |  |             m_vorGUIs.removeAt(row); | 
					
						
							|  |  |  |             endRemoveRows(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void removeAllVORs() { | 
					
						
							| 
									
										
										
										
											2021-04-13 10:14:54 +01:00
										 |  |  |         if (m_vors.count() > 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             beginRemoveRows(QModelIndex(), 0, m_vors.count() - 1); | 
					
						
							|  |  |  |             m_vors.clear(); | 
					
						
							|  |  |  |             m_selected.clear(); | 
					
						
							|  |  |  |             m_radials.clear(); | 
					
						
							|  |  |  |             m_vorGUIs.clear(); | 
					
						
							|  |  |  |             endRemoveRows(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QHash<int, QByteArray> roleNames() const { | 
					
						
							|  |  |  |         QHash<int, QByteArray> roles; | 
					
						
							|  |  |  |         roles[positionRole] = "position"; | 
					
						
							|  |  |  |         roles[vorDataRole] = "vorData"; | 
					
						
							|  |  |  |         roles[vorImageRole] = "vorImage"; | 
					
						
							|  |  |  |         roles[vorRadialRole] = "vorRadial"; | 
					
						
							|  |  |  |         roles[bubbleColourRole] = "bubbleColour"; | 
					
						
							|  |  |  |         roles[selectedRole] = "selected"; | 
					
						
							|  |  |  |         return roles; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void setRadialsVisible(bool radialsVisible) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         m_radialsVisible = radialsVisible; | 
					
						
							|  |  |  |         allVORUpdated(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void setRadial(int id, bool valid, Real radial) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         for (int i = 0; i < m_vors.count(); i++) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |             if (m_vors[i]->m_id == id) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if (valid) | 
					
						
							|  |  |  |                     m_radials[i] = radial; | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     m_radials[i] = -1; // -1 to indicate invalid
 | 
					
						
							|  |  |  |                 QModelIndex idx = index(i); | 
					
						
							|  |  |  |                 emit dataChanged(idx, idx); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool findIntersection(float &lat, float &lon); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     VORLocalizerGUI *m_gui; | 
					
						
							|  |  |  |     bool m_radialsVisible; | 
					
						
							|  |  |  |     QList<NavAid *> m_vors; | 
					
						
							|  |  |  |     QList<bool> m_selected; | 
					
						
							|  |  |  |     QList<Real> m_radials; | 
					
						
							|  |  |  |     QList<VORGUI *> m_vorGUIs; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class VORLocalizerGUI : public FeatureGUI { | 
					
						
							|  |  |  |     Q_OBJECT | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     static VORLocalizerGUI* create(PluginAPI* pluginAPI, FeatureUISet *featureUISet, Feature *feature); | 
					
						
							|  |  |  |     virtual void destroy(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void resetToDefaults(); | 
					
						
							|  |  |  |     QByteArray serialize() const; | 
					
						
							|  |  |  |     bool deserialize(const QByteArray& data); | 
					
						
							|  |  |  |     virtual MessageQueue *getInputMessageQueue() { return &m_inputMessageQueue; } | 
					
						
							|  |  |  |     void selectVOR(VORGUI *vorGUI, bool selected); | 
					
						
							| 
									
										
										
										
											2022-05-13 22:24:48 +02:00
										 |  |  |     virtual void setWorkspaceIndex(int index); | 
					
						
							| 
									
										
										
										
											2022-04-05 16:26:57 +02:00
										 |  |  |     virtual int getWorkspaceIndex() const { return m_settings.m_workspaceIndex; } | 
					
						
							|  |  |  |     virtual void setGeometryBytes(const QByteArray& blob) { m_settings.m_geometryBytes = blob; } | 
					
						
							|  |  |  |     virtual QByteArray getGeometryBytes() const { return m_settings.m_geometryBytes; } | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     friend class VORGUI; | 
					
						
							|  |  |  |     friend class VORModel; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Ui::VORLocalizerGUI* ui; | 
					
						
							|  |  |  |     PluginAPI* m_pluginAPI; | 
					
						
							|  |  |  |     FeatureUISet* m_featureUISet; | 
					
						
							|  |  |  |     VORLocalizerSettings m_settings; | 
					
						
							| 
									
										
										
										
											2022-11-30 22:00:26 +01:00
										 |  |  |     QList<QString> m_settingsKeys; | 
					
						
							| 
									
										
										
										
											2022-01-09 05:27:12 +01:00
										 |  |  |     RollupState m_rollupState; | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  |     bool m_doApplySettings; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     VORLocalizer* m_vorLocalizer; | 
					
						
							|  |  |  |     bool m_squelchOpen; | 
					
						
							|  |  |  |     int m_basebandSampleRate; | 
					
						
							|  |  |  |     uint32_t m_tickCount; | 
					
						
							|  |  |  |     MessageQueue m_inputMessageQueue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMenu *menu;                        // Column select context menu
 | 
					
						
							|  |  |  |     HttpDownloadManager m_dlm; | 
					
						
							|  |  |  |     QProgressDialog *m_progressDialog; | 
					
						
							| 
									
										
										
										
											2022-05-03 13:44:09 +01:00
										 |  |  |     OpenAIP m_openAIP; | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  |     int m_countryIndex; | 
					
						
							|  |  |  |     VORModel m_vorModel; | 
					
						
							| 
									
										
										
										
											2022-05-03 13:44:09 +01:00
										 |  |  |     QList<NavAid *> m_vors; | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  |     QHash<int, VORGUI *> m_selectedVORs; | 
					
						
							|  |  |  |     AzEl m_azEl;                        // Position of station
 | 
					
						
							|  |  |  |     QIcon m_muteIcon; | 
					
						
							|  |  |  | 	QTimer m_statusTimer; | 
					
						
							|  |  |  | 	int m_lastFeatureState; | 
					
						
							| 
									
										
										
										
											2020-12-08 08:28:55 +01:00
										 |  |  |     int m_rrSecondsCount; | 
					
						
							| 
									
										
										
										
											2022-05-03 14:53:56 +01:00
										 |  |  |     QTimer m_redrawMapTimer; | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     explicit VORLocalizerGUI(PluginAPI* pluginAPI, FeatureUISet *featureUISet, Feature *feature, QWidget* parent = nullptr); | 
					
						
							|  |  |  |     virtual ~VORLocalizerGUI(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void blockApplySettings(bool block); | 
					
						
							|  |  |  |     void applySettings(bool force = false); | 
					
						
							|  |  |  |     void displaySettings(); | 
					
						
							|  |  |  |     bool handleMessage(const Message& message); | 
					
						
							| 
									
										
										
										
											2022-05-03 14:53:56 +01:00
										 |  |  |     void redrawMap(); | 
					
						
							| 
									
										
										
										
											2022-04-04 10:23:52 +02:00
										 |  |  |     void makeUIConnections(); | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void resizeTable(); | 
					
						
							|  |  |  |     QAction *createCheckableItem(QString& text, int idx, bool checked); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void calculateFreqOffset(VORGUI *vorGUI); | 
					
						
							|  |  |  |     void calculateFreqOffsets(); | 
					
						
							|  |  |  |     void updateVORs(); | 
					
						
							|  |  |  |     void readNavAids(); | 
					
						
							|  |  |  |     void updateChannelList(); | 
					
						
							| 
									
										
										
										
											2022-09-30 15:59:51 +01:00
										 |  |  |     void applyMapSettings(); | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | private slots: | 
					
						
							|  |  |  |     void on_startStop_toggled(bool checked); | 
					
						
							|  |  |  |     void on_getOpenAIPVORDB_clicked(); | 
					
						
							| 
									
										
										
										
											2020-12-07 19:32:43 +01:00
										 |  |  |     void on_magDecAdjust_toggled(bool checked); | 
					
						
							|  |  |  |     void on_rrTime_valueChanged(int value); | 
					
						
							|  |  |  |     void on_centerShift_valueChanged(int value); | 
					
						
							| 
									
										
										
										
											2022-05-03 14:53:56 +01:00
										 |  |  |     void channelsRefresh(); | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  |     void vorData_sectionMoved(int logicalIndex, int oldVisualIndex, int newVisualIndex); | 
					
						
							|  |  |  |     void vorData_sectionResized(int logicalIndex, int oldSize, int newSize); | 
					
						
							|  |  |  |     void columnSelectMenu(QPoint pos); | 
					
						
							|  |  |  |     void columnSelectMenuChecked(bool checked = false); | 
					
						
							|  |  |  |     void onWidgetRolled(QWidget* widget, bool rollDown); | 
					
						
							|  |  |  |     void onMenuDialogCalled(const QPoint& p); | 
					
						
							|  |  |  |     void handleInputMessages(); | 
					
						
							|  |  |  |     void updateStatus(); | 
					
						
							|  |  |  |     void tick(); | 
					
						
							| 
									
										
										
										
											2022-05-03 13:44:09 +01:00
										 |  |  |     void downloadingURL(const QString& url); | 
					
						
							|  |  |  |     void downloadError(const QString& error); | 
					
						
							|  |  |  |     void downloadNavAidsFinished(); | 
					
						
							| 
									
										
										
										
											2022-05-03 13:51:39 +01:00
										 |  |  |     void preferenceChanged(int elementType); | 
					
						
							| 
									
										
										
										
											2022-05-03 14:53:56 +01:00
										 |  |  |     virtual void showEvent(QShowEvent *event); | 
					
						
							|  |  |  |     virtual bool eventFilter(QObject *obj, QEvent *event); | 
					
						
							| 
									
										
										
										
											2020-11-30 01:30:18 +01:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif // INCLUDE_VORLOCALIZERGUI_H
 |