/////////////////////////////////////////////////////////////////////////////////// // 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 . // /////////////////////////////////////////////////////////////////////////////////// #ifndef INCLUDE_VORDEMODGUI_H #define INCLUDE_VORDEMODGUI_H #include #include #include #include #include #include #include #include #include #include #include #include "channel/channelgui.h" #include "dsp/channelmarker.h" #include "dsp/movingaverage.h" #include "util/messagequeue.h" #include "util/httpdownloadmanager.h" #include "util/azel.h" #include "vordemodsettings.h" #include "navaid.h" class PluginAPI; class DeviceUISet; class BasebandSampleSink; class VORDemod; class VORDemodGUI; namespace Ui { class VORDemodGUI; } class VORDemodGUI; // Table items for each VOR class VORGUI : public QObject { Q_OBJECT public: NavAid *m_navAid; QVariantList m_coordinates; VORDemodGUI *m_gui; QTableWidgetItem *m_nameItem; QTableWidgetItem *m_frequencyItem; QTableWidgetItem *m_offsetItem; 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, VORDemodGUI *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(VORDemodGUI *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 { 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() { beginRemoveRows(QModelIndex(), 0, m_vors.count()); m_vors.clear(); m_selected.clear(); m_radials.clear(); m_vorGUIs.clear(); endRemoveRows(); } QHash roleNames() const { QHash 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: VORDemodGUI *m_gui; bool m_radialsVisible; QList m_vors; QList m_selected; QList m_radials; QList m_vorGUIs; }; class VORDemodGUI : public ChannelGUI { Q_OBJECT public: static VORDemodGUI* create(PluginAPI* pluginAPI, DeviceUISet *deviceUISet, BasebandSampleSink *rxChannel); 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); public slots: void channelMarkerChangedByCursor(); void channelMarkerHighlightedByCursor(); private: friend class VORGUI; friend class VORModel; Ui::VORDemodGUI* ui; PluginAPI* m_pluginAPI; DeviceUISet* m_deviceUISet; ChannelMarker m_channelMarker; VORDemodSettings m_settings; bool m_doApplySettings; VORDemod* m_vorDemod; 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; int m_countryIndex; VORModel m_vorModel; QHash *m_vors; QHash m_selectedVORs; AzEl m_azEl; // Position of station QIcon m_muteIcon; explicit VORDemodGUI(PluginAPI* pluginAPI, DeviceUISet *deviceUISet, BasebandSampleSink *rxChannel, QWidget* parent = 0); virtual ~VORDemodGUI(); void blockApplySettings(bool block); void applySettings(bool force = false); void displaySettings(); void displayStreamIndex(); bool handleMessage(const Message& message); void leaveEvent(QEvent*); void enterEvent(QEvent*); void resizeTable(); QAction *createCheckableItem(QString& text, int idx, bool checked); void calculateFreqOffset(VORGUI *vorGUI); void calculateFreqOffsets(); void updateVORs(); QString getOpenAIPVORDBURL(int i); QString getOpenAIPVORDBFilename(int i); QString getVORDBFilename(); void readNavAids(); // Move to util QString getDataDir(); qint64 fileAgeInDays(QString filename); bool confirmDownload(QString filename); private slots: void on_thresh_valueChanged(int value); void on_volume_valueChanged(int value); void on_squelch_valueChanged(int value); void on_audioMute_toggled(bool checked); void on_getOurAirportsVORDB_clicked(bool checked = false); void on_getOpenAIPVORDB_clicked(bool checked = false); void on_magDecAdjust_clicked(bool checked = false); 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 updateDownloadProgress(qint64 bytesRead, qint64 totalBytes); void downloadFinished(const QString& filename, bool success); void handleInputMessages(); void audioSelect(); void tick(); }; #endif // INCLUDE_VORDEMODGUI_H