mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-24 17:40:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			1524 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1524 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| // Copyright (C) 2023 Jon Beniston, M7RCE <jon@beniston.com>                     //
 | |
| // Copyright (C) 2023 Edouard Griffiths, F4EXB <f4exb06@gmail.com>               //
 | |
| //                                                                               //
 | |
| // 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/>.          //
 | |
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #include <QDockWidget>
 | |
| #include <QMainWindow>
 | |
| #include <QDebug>
 | |
| #include <QAction>
 | |
| #include <QClipboard>
 | |
| #include <QFileDialog>
 | |
| #include <QScrollBar>
 | |
| #include <QMenu>
 | |
| 
 | |
| #include "SWGMapItem.h"
 | |
| 
 | |
| #include "ilsdemodgui.h"
 | |
| 
 | |
| #include "device/deviceset.h"
 | |
| #include "device/deviceuiset.h"
 | |
| #include "dsp/dspengine.h"
 | |
| #include "dsp/dspcommands.h"
 | |
| #include "dsp/morsedemod.h"
 | |
| #include "ui_ilsdemodgui.h"
 | |
| #include "plugin/pluginapi.h"
 | |
| #include "util/db.h"
 | |
| #include "util/units.h"
 | |
| #include "util/morse.h"
 | |
| #include "gui/basicchannelsettingsdialog.h"
 | |
| #include "dsp/dspengine.h"
 | |
| #include "dsp/glscopesettings.h"
 | |
| #include "dsp/spectrumvis.h"
 | |
| #include "gui/crightclickenabler.h"
 | |
| #include "gui/dialogpositioner.h"
 | |
| #include "gui/audioselectdialog.h"
 | |
| #include "channel/channelwebapiutils.h"
 | |
| #include "feature/featurewebapiutils.h"
 | |
| #include "feature/feature.h"
 | |
| #include "maincore.h"
 | |
| 
 | |
| #include "ilsdemod.h"
 | |
| 
 | |
| MESSAGE_CLASS_DEFINITION(ILSDemodGUI::MsgGSAngle, Message)
 | |
| 
 | |
| // 3.1.3.2 - ILS can use one or two carrier frequencies
 | |
| // If one is used, frequency tolerance is +/- 0.005% -> 5.5kHz
 | |
| // If two are used, frequency tolerance is +/- 0.002% and they should be symmetric about the assigned frequency
 | |
| // Frequency separation should be within [5kHz,14kHz]
 | |
| // Called "offset carrier" https://www.pprune.org/tech-log/108843-vhf-offset-carrier.html
 | |
| 
 | |
| // ICAO Anntex 10 Volume 1 - 3.1.6.1
 | |
| const QStringList ILSDemodGUI::m_locFrequencies = {
 | |
|     "108.10", "108.15", "108.30", "108.35", "108.50", "108.55", "108.70", "108.75",
 | |
|     "108.90", "108.95", "109.10", "109.15", "109.30", "109.35", "109.50", "109.55",
 | |
|     "109.70", "109.75", "109.90", "109.95", "110.10", "110.15", "110.30", "110.35",
 | |
|     "110.50", "110.55", "110.70", "110.75", "110.90", "110.95", "111.10", "111.15",
 | |
|     "111.30", "111.35", "111.50", "111.55", "111.70", "111.75", "111.90", "111.95",
 | |
| };
 | |
| 
 | |
| const QStringList ILSDemodGUI::m_gsFrequencies = {
 | |
|     "334.70", "334.55", "334.10", "333.95", "329.90", "329.75", "330.50", "330.35",
 | |
|     "329.30", "329.15", "331.40", "331.25", "332.00", "331.85", "332.60", "332.45",
 | |
|     "333.20", "333.05", "333.80", "333.65", "334.40", "334.25", "335.00", "334.85",
 | |
|     "329.60", "329.45", "330.20", "330.05", "330.80", "330.65", "331.70", "331.55",
 | |
|     "332.30", "332.15", "332.90", "332.75", "333.50", "333.35", "331.10", "330.95",
 | |
| };
 | |
| 
 | |
| // UK data from NATS: https://nats-uk.ead-it.com/cms-nats/opencms/en/Publications/AIP/
 | |
| // Note that AIP data is more accurate in tables than on charts
 | |
| // Some values tweaked from AIP to better align with 3D Map
 | |
| const QList<ILSDemodGUI::ILS> ILSDemodGUI::m_ils = {
 | |
|     {"EGKB", "IBGH", "21",  109350000, 205.71, 3.0, 51.338272, 0.038065,  569, 15.25, 1840, 1.13},
 | |
|     {"EGKK", "IWW",  "26L", 110900000, 257.59, 3.0, 51.150680, -0.171943, 212, 15.5,  3060, 0.06},
 | |
|     {"EGKK", "IGG",  "08R", 110900000, 77.63,  3.0, 51.145872, -0.206807, 212, 15.5,  3140, -0.06},
 | |
|     {"EGLL", "ILL",  "27L", 109500000, 269.72, 3.0, 51.464960, -0.434108, 93,  17.1,  3960, 0.0},
 | |
|     {"EGLL", "IRR",  "27R", 110300000, 269.71, 3.0, 51.477678, -0.433291, 99,  17.7,  4190, 0.0},
 | |
|     {"EGLL", "IAA",  "09L", 110300000, 89.67,  3.0, 51.477503, -0.484984, 99,  15.5,  4020, 0.0},
 | |
|     {"EGLL", "IBB",  "09R", 109500000, 89.68,  3.0, 51.464795, -0.482305, 93,  15.25, 3750, 0.0},
 | |
|     {"EGLC", "ILST", "09",  111150000, 92.87,  5.5, 51.505531, 0.045766,  48,  10.7,  1510, 0.0},
 | |
|     {"EGLC", "ILSR", "27",  111150000, 272.89, 5.5, 51.504927, 0.064960,  48,  10.7,  1580, 0.0},
 | |
|     {"EGSS", "ISX",  "22",  110500000, 222.78, 3.0, 51.895165, 0.250051,  352, 14.9,  3430, 0.0},
 | |
|     {"EGSS", "ISED", "04",  110500000, 42.78,  3.0, 51.877054, 0.222887,  352, 16.2,  3130, 0.0},
 | |
|     {"KGYH", "IGYH", "5",   108300000, 40.00,  3.0, 34.749987, -82.384983,850, 15.84, 2750, -0.6},
 | |
| };
 | |
| 
 | |
| ILSDemodGUI* ILSDemodGUI::create(PluginAPI* pluginAPI, DeviceUISet *deviceUISet, BasebandSampleSink *rxChannel)
 | |
| {
 | |
|     ILSDemodGUI* gui = new ILSDemodGUI(pluginAPI, deviceUISet, rxChannel);
 | |
|     return gui;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::destroy()
 | |
| {
 | |
|     delete this;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::resetToDefaults()
 | |
| {
 | |
|     m_settings.resetToDefaults();
 | |
|     displaySettings();
 | |
|     applySettings(true);
 | |
| }
 | |
| 
 | |
| QByteArray ILSDemodGUI::serialize() const
 | |
| {
 | |
|     return m_settings.serialize();
 | |
| }
 | |
| 
 | |
| bool ILSDemodGUI::deserialize(const QByteArray& data)
 | |
| {
 | |
|     if(m_settings.deserialize(data))
 | |
|     {
 | |
|         displaySettings();
 | |
|         applySettings(true);
 | |
|         return true;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         resetToDefaults();
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| QString ILSDemodGUI::formatAngleDirection(float angle) const
 | |
| {
 | |
|     QString text;
 | |
|     if (m_settings.m_mode == ILSDemodSettings::LOC)
 | |
|     {
 | |
|         if (angle == 0.0) {
 | |
|             text = "Centre";
 | |
|         } else if (angle > 0.0) {
 | |
|             text = "Left";
 | |
|         } else {
 | |
|             text = "Right";
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if (angle == 0.0) {
 | |
|             text = "On slope";
 | |
|         } else if (angle > 0.0) {
 | |
|             text = "Above";
 | |
|         } else {
 | |
|             text = "Below";
 | |
|         }
 | |
|     }
 | |
|     return text;
 | |
| }
 | |
| 
 | |
| bool ILSDemodGUI::handleMessage(const Message& message)
 | |
| {
 | |
|     if (ILSDemod::MsgConfigureILSDemod::match(message))
 | |
|     {
 | |
|         qDebug("ILSDemodGUI::handleMessage: ILSDemod::MsgConfigureILSDemod");
 | |
|         const ILSDemod::MsgConfigureILSDemod& cfg = (ILSDemod::MsgConfigureILSDemod&) message;
 | |
|         m_settings = cfg.getSettings();
 | |
|         blockApplySettings(true);
 | |
|         ui->scopeGUI->updateSettings();
 | |
|         m_channelMarker.updateSettings(static_cast<const ChannelMarker*>(m_settings.m_channelMarker));
 | |
|         displaySettings();
 | |
|         blockApplySettings(false);
 | |
|         return true;
 | |
|     }
 | |
|     else if (DSPSignalNotification::match(message))
 | |
|     {
 | |
|         DSPSignalNotification& notif = (DSPSignalNotification&) message;
 | |
|         m_deviceCenterFrequency = notif.getCenterFrequency();
 | |
|         m_basebandSampleRate = notif.getSampleRate();
 | |
|         ui->deltaFrequency->setValueRange(false, 7, -m_basebandSampleRate/2, m_basebandSampleRate/2);
 | |
|         ui->deltaFrequencyLabel->setToolTip(tr("Range %1 %L2 Hz").arg(QChar(0xB1)).arg(m_basebandSampleRate/2));
 | |
|         updateAbsoluteCenterFrequency();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MorseDemod::MsgReportIdent::match(message))
 | |
|     {
 | |
|         MorseDemod::MsgReportIdent& report = (MorseDemod::MsgReportIdent&) message;
 | |
| 
 | |
|         QString ident = report.getIdent();
 | |
|         QString identString = Morse::toString(ident); // Convert Morse to a string
 | |
| 
 | |
|         ui->morseIdent->setText(Morse::toSpacedUnicode(ident) + "   " + identString);
 | |
| 
 | |
|         // Check it matches ident setting
 | |
|         if (identString == m_settings.m_ident) {
 | |
|             ui->morseIdent->setStyleSheet("QLabel { color: white }");
 | |
|         } else {
 | |
|             ui->morseIdent->setStyleSheet("QLabel { color: red }");
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (ILSDemod::MsgAngleEstimate::match(message))
 | |
|     {
 | |
|         ILSDemod::MsgAngleEstimate& report = (ILSDemod::MsgAngleEstimate&) message;
 | |
|         ui->md90->setValue(report.getModDepth90());
 | |
|         ui->md150->setValue(report.getModDepth150());
 | |
|         float sdmPercent = report.getSDM() * 100.0f;
 | |
|         ui->sdm->setValue(sdmPercent);
 | |
|         // 3.1.3.5.3.6.1
 | |
|         if ((sdmPercent < 30.0f) || (sdmPercent > 60.0f)) {
 | |
|             ui->sdm->setStyleSheet("QLineEdit { background: rgb(255, 0, 0); }");
 | |
|         } else {
 | |
|             ui->sdm->setStyleSheet("");
 | |
|         }
 | |
|         ui->ddm->setText(formatDDM(report.getDDM()));
 | |
|         float angle = report.getAngle();
 | |
|         float absAngle = std::abs(angle);
 | |
|         ui->angle->setText(QString("%1").arg(absAngle, 0, 'f', 1));
 | |
|         ui->angleDirection->setText(formatAngleDirection(angle));
 | |
|         ui->pCarrier->setText(QString("%1").arg(report.getPowerCarrier(), 0, 'f', 1));
 | |
|         ui->p90->setText(QString("%1").arg(report.getPower90(), 0, 'f', 1));
 | |
|         ui->p150->setText(QString("%1").arg(report.getPower150(), 0, 'f', 1));
 | |
|         if (m_settings.m_mode == ILSDemodSettings::LOC) {
 | |
|             ui->cdi->setLocalizerDDM(report.getDDM());
 | |
|         } else {
 | |
|             ui->cdi->setGlideSlopeDDM(report.getDDM());
 | |
|         }
 | |
|         if (m_settings.m_mode == ILSDemodSettings::GS)
 | |
|         {
 | |
|             m_gsAngle = angle;
 | |
|             if (!sendToLOCChannel(angle)) {
 | |
|                 drawPath();
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             m_locAngle = angle;
 | |
|             drawPath();
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
|     else if (MsgGSAngle::match(message))
 | |
|     {
 | |
|         MsgGSAngle& report = (MsgGSAngle&) message;
 | |
|         m_gsAngle = report.getAngle();
 | |
|         drawPath();
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::handleInputMessages()
 | |
| {
 | |
|     Message* message;
 | |
| 
 | |
|     while ((message = getInputMessageQueue()->pop()) != 0)
 | |
|     {
 | |
|         if (handleMessage(*message)) {
 | |
|             delete message;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::channelMarkerChangedByCursor()
 | |
| {
 | |
|     ui->deltaFrequency->setValue(m_channelMarker.getCenterFrequency());
 | |
|     m_settings.m_inputFrequencyOffset = m_channelMarker.getCenterFrequency();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::channelMarkerHighlightedByCursor()
 | |
| {
 | |
|     setHighlighted(m_channelMarker.getHighlighted());
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_deltaFrequency_changed(qint64 value)
 | |
| {
 | |
|     m_channelMarker.setCenterFrequency(value);
 | |
|     m_settings.m_inputFrequencyOffset = m_channelMarker.getCenterFrequency();
 | |
|     updateAbsoluteCenterFrequency();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_rfBW_valueChanged(int value)
 | |
| {
 | |
|     float bw = value;
 | |
|     ui->rfBWText->setText(formatFrequency((int)bw));
 | |
|     m_channelMarker.setBandwidth(bw);
 | |
|     m_settings.m_rfBandwidth = bw;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_mode_currentIndexChanged(int index)
 | |
| {
 | |
|     int freqIdx = ui->frequency->currentIndex();
 | |
|     ui->frequency->clear();
 | |
|     m_settings.m_mode = (ILSDemodSettings::Mode)index;
 | |
|     if (m_settings.m_mode == ILSDemodSettings::LOC)
 | |
|     {
 | |
|         ui->cdi->setMode(CourseDeviationIndicator::LOC);
 | |
|         ui->frequency->setMinimumSize(60, 0);
 | |
|         for (const auto &freq : m_locFrequencies) {
 | |
|             ui->frequency->addItem(freq);
 | |
|         }
 | |
|         scanAvailableChannels();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ui->cdi->setMode(CourseDeviationIndicator::GS);
 | |
|         ui->frequency->setMinimumSize(110, 0);
 | |
|         for (int i = 0; i < m_locFrequencies.size(); i++)
 | |
|         {
 | |
|             QString text = m_locFrequencies[i] + "/" + m_gsFrequencies[i];
 | |
|             ui->frequency->addItem(text);
 | |
|         }
 | |
|         closePipes();
 | |
|     }
 | |
|     ui->frequency->setCurrentIndex(freqIdx);
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_frequency_currentIndexChanged(int index)
 | |
| {
 | |
|     m_settings.m_frequencyIndex = index;
 | |
|     if ((index >= 0) && (index < m_locFrequencies.size()))
 | |
|     {
 | |
|         QString text;
 | |
|         if (m_settings.m_mode == ILSDemodSettings::LOC) {
 | |
|             text = m_locFrequencies[index];
 | |
|         } else {
 | |
|             text = m_gsFrequencies[index];
 | |
|         }
 | |
|         double frequency = text.toDouble() * 1e6;
 | |
|         ChannelWebAPIUtils::setCenterFrequency(m_ilsDemod->getDeviceSetIndex(), frequency);
 | |
|     }
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_average_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_average = checked;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_volume_valueChanged(int value)
 | |
| {
 | |
|     ui->volumeText->setText(QString("%1").arg(value / 10.0, 0, 'f', 1));
 | |
|     m_settings.m_volume = value / 10.0;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_squelch_valueChanged(int value)
 | |
| {
 | |
|     ui->squelchText->setText(QString("%1 dB").arg(value));
 | |
|     m_settings.m_squelch = value;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_audioMute_toggled(bool checked)
 | |
| {
 | |
|     m_settings.m_audioMute = checked;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_thresh_valueChanged(int value)
 | |
| {
 | |
|     ui->threshText->setText(QString("%1").arg(value / 10.0, 0, 'f', 1));
 | |
|     m_settings.m_identThreshold = value / 10.0;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_ddmUnits_currentIndexChanged(int index)
 | |
| {
 | |
|     m_settings.m_ddmUnits = (ILSDemodSettings::DDMUnits) index;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_ident_editingFinished()
 | |
| {
 | |
|    m_settings.m_ident = ui->ident->currentText();
 | |
|    applySettings();
 | |
| }
 | |
| 
 | |
| // 3.1.3.7.3 Note 1
 | |
| float ILSDemodGUI::calcCourseWidth(int m_thresholdToLocalizer) const
 | |
| {
 | |
|     float width = 2.0f * Units::radiansToDegrees(atan(105.0f/m_thresholdToLocalizer));
 | |
|     width = std::min(6.0f, width);
 | |
|     return width;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_ident_currentIndexChanged(int index)
 | |
| {
 | |
|     m_settings.m_ident = ui->ident->currentText();
 | |
|     if ((index >= 0) && (index < m_ils.size()))
 | |
|     {
 | |
|         m_disableDrawILS = true;
 | |
|         ui->trueBearing->setValue(m_ils[index].m_trueBearing);
 | |
|         ui->latitude->setText(QString::number(m_ils[index].m_latitude, 'f', 8));
 | |
|         on_latitude_editingFinished();
 | |
|         ui->longitude->setText(QString::number(m_ils[index].m_longitude, 'f', 8));
 | |
|         on_longitude_editingFinished();
 | |
|         ui->elevation->setValue(m_ils[index].m_elevation);
 | |
|         ui->glidePath->setValue(m_ils[index].m_glidePath);
 | |
|         ui->height->setValue(m_ils[index].m_refHeight);
 | |
|         ui->courseWidth->setValue(calcCourseWidth(m_ils[index].m_thresholdToLocalizer));
 | |
|         ui->slope->setValue(m_ils[index].m_slope);
 | |
|         ui->runway->setText(QString("%1 %2").arg(m_ils[index].m_airportICAO).arg(m_ils[index].m_runway));
 | |
|         on_runway_editingFinished();
 | |
|         int frequency = m_ils[index].m_frequency;
 | |
|         QString freqText = QString("%1").arg(frequency / 1000000.0f, 0, 'f', 2);
 | |
|         int freqIndex = m_locFrequencies.indexOf(freqText);
 | |
|         ui->frequency->setCurrentIndex(freqIndex);
 | |
|         m_disableDrawILS = false;
 | |
|     }
 | |
|     drawILSOnMap();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_runway_editingFinished()
 | |
| {
 | |
|     m_settings.m_runway = ui->runway->text();
 | |
|     drawILSOnMap();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_findOnMap_clicked()
 | |
| {
 | |
|     QString pos = QString("%1,%2").arg(m_settings.m_latitude).arg(m_settings.m_longitude);
 | |
|     FeatureWebAPIUtils::mapFind(pos);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_addMarker_clicked()
 | |
| {
 | |
|     float stationLatitude = MainCore::instance()->getSettings().getLatitude();
 | |
|     float stationLongitude = MainCore::instance()->getSettings().getLongitude();
 | |
|     float stationAltitude = MainCore::instance()->getSettings().getAltitude();
 | |
| 
 | |
|     QList<ObjectPipe*> mapPipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", mapPipes);
 | |
|     for (const auto& pipe : mapPipes)
 | |
|     {
 | |
|         MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|         SWGSDRangel::SWGMapItem *swgMapItem = new SWGSDRangel::SWGMapItem();
 | |
| 
 | |
|         QString mode = m_settings.m_mode == ILSDemodSettings::LOC ? "LOC" : "GS";
 | |
|         QString name = QString("%1 M%2").arg(mode).arg(m_markerNo);
 | |
|         swgMapItem->setName(new QString(name));
 | |
| 
 | |
|         swgMapItem->setLatitude(stationLatitude);
 | |
|         swgMapItem->setLongitude(stationLongitude);
 | |
|         swgMapItem->setAltitude(stationAltitude);
 | |
|         swgMapItem->setAltitudeReference(1); // CLAMP_TO_GROUND
 | |
|         swgMapItem->setFixedPosition(true);
 | |
|         swgMapItem->setPositionDateTime(new QString(QDateTime::currentDateTime().toString(Qt::ISODateWithMs)));
 | |
| 
 | |
|         swgMapItem->setImage(new QString(QString("qrc:///aprs/aprs/aprs-symbols-24-0-06.png")));
 | |
|         QString text;
 | |
|         if (!ui->ddm->text().isEmpty())
 | |
|         {
 | |
|             text = QString("ILS %7 Marker %1\n90Hz: %2%\n150Hz: %3%\nDDM: %4\nAngle: %5%6 %8")
 | |
|                             .arg(m_markerNo)
 | |
|                             .arg(ui->md90->value())
 | |
|                             .arg(ui->md150->value())
 | |
|                             .arg(ui->ddm->text())
 | |
|                             .arg(ui->angle->text())
 | |
|                             .arg(QChar(0xb0))
 | |
|                             .arg(mode)
 | |
|                             .arg(ui->angleDirection->text());
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             text = QString("ILS %1 Marker %2\nNo data").arg(mode).arg(m_markerNo);
 | |
|         }
 | |
|         swgMapItem->setText(new QString(text));
 | |
| 
 | |
|         if (!m_mapMarkers.contains(name)) {
 | |
|             m_mapMarkers.insert(name, true);
 | |
|         }
 | |
| 
 | |
|         MainCore::MsgMapItem *msg = MainCore::MsgMapItem::create(m_ilsDemod, swgMapItem);
 | |
|         messageQueue->push(msg);
 | |
| 
 | |
|         m_markerNo++;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::removeFromMap(const QString& name)
 | |
| {
 | |
|     QList<ObjectPipe*> mapPipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", mapPipes);
 | |
| 
 | |
|     for (const auto& pipe : mapPipes)
 | |
|     {
 | |
|         MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|         SWGSDRangel::SWGMapItem *swgMapItem = new SWGSDRangel::SWGMapItem();
 | |
| 
 | |
|         swgMapItem->setName(new QString(name));
 | |
|         swgMapItem->setImage(new QString(""));
 | |
| 
 | |
|         MainCore::MsgMapItem *msg = MainCore::MsgMapItem::create(m_ilsDemod, swgMapItem);
 | |
|         messageQueue->push(msg);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_clearMarkers_clicked()
 | |
| {
 | |
|     QMutableHashIterator<QString, bool> itr(m_mapMarkers);
 | |
|     while (itr.hasNext())
 | |
|     {
 | |
|         itr.next();
 | |
|         removeFromMap(itr.key());
 | |
|         itr.remove();
 | |
|     }
 | |
|     m_markerNo = 0;
 | |
| }
 | |
| 
 | |
| // Lats and longs in decimal degrees. Distance in metres. Bearing in degrees.
 | |
| // https://www.movable-type.co.uk/scripts/latlong.html
 | |
| static void calcRadialEndPoint(float startLatitude, float startLongitude, float distance, float bearing, float &endLatitude, float &endLongitude)
 | |
| {
 | |
|     double startLatRad = startLatitude*M_PI/180.0;
 | |
|     double startLongRad = startLongitude*M_PI/180.0;
 | |
|     double theta = bearing*M_PI/180.0;
 | |
|     double earthRadius = 6378137.0; // At equator
 | |
|     double delta = distance/earthRadius;
 | |
|     double endLatRad = std::asin(sin(startLatRad)*cos(delta) + cos(startLatRad)*sin(delta)*cos(theta));
 | |
|     double endLongRad = startLongRad + std::atan2(sin(theta)*sin(delta)*cos(startLatRad), cos(delta) - sin(startLatRad)*sin(endLatRad));
 | |
|     endLatitude = endLatRad*180.0/M_PI;
 | |
|     endLongitude = endLongRad*180.0/M_PI;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::addPolygonToMap(const QString& name, const QString& label, const QList<QGeoCoordinate>& coordinates, QRgb color)
 | |
| {
 | |
|     // Send to Map feature
 | |
|     QList<ObjectPipe*> mapPipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", mapPipes);
 | |
| 
 | |
|     if (mapPipes.size() > 0)
 | |
|     {
 | |
|         if (!m_mapILS.contains(name)) {
 | |
|             m_mapILS.insert(name, true);
 | |
|         }
 | |
| 
 | |
|         for (const auto& pipe : mapPipes)
 | |
|         {
 | |
|             MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|             SWGSDRangel::SWGMapItem *swgMapItem = new SWGSDRangel::SWGMapItem();
 | |
| 
 | |
|             swgMapItem->setName(new QString(name));
 | |
|             swgMapItem->setLabel(new QString(label));
 | |
|             swgMapItem->setLatitude(coordinates[0].latitude());
 | |
|             swgMapItem->setLongitude(coordinates[0].longitude());
 | |
|             swgMapItem->setAltitude(coordinates[0].altitude());
 | |
|             QString image = QString("none");
 | |
|             swgMapItem->setImage(new QString(image));
 | |
|             swgMapItem->setImageRotation(0);
 | |
|             swgMapItem->setFixedPosition(true);
 | |
|             swgMapItem->setAltitudeReference(3); // 1 - CLAMP_TO_GROUND, 3 - CLIP_TO_GROUND
 | |
|             swgMapItem->setColorValid(true);
 | |
|             swgMapItem->setColor(color);
 | |
|             QList<SWGSDRangel::SWGMapCoordinate *> *coords = new QList<SWGSDRangel::SWGMapCoordinate *>();
 | |
| 
 | |
|             for (const auto& coord : coordinates)
 | |
|             {
 | |
|                 SWGSDRangel::SWGMapCoordinate* c = new SWGSDRangel::SWGMapCoordinate();
 | |
|                 c->setLatitude(coord.latitude());
 | |
|                 c->setLongitude(coord.longitude());
 | |
|                 c->setAltitude(coord.altitude());
 | |
|                 coords->append(c);
 | |
|             }
 | |
| 
 | |
|             swgMapItem->setCoordinates(coords);
 | |
|             swgMapItem->setType(2);
 | |
| 
 | |
|             MainCore::MsgMapItem *msg = MainCore::MsgMapItem::create(m_ilsDemod, swgMapItem);
 | |
|             messageQueue->push(msg);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::addLineToMap(const QString& name, const QString& label, float startLatitude, float startLongitude, float startAltitude, float endLatitude, float endLongitude, float endAltitude)
 | |
| {
 | |
|     // Send to Map feature
 | |
|     QList<ObjectPipe*> mapPipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", mapPipes);
 | |
| 
 | |
|     if (mapPipes.size() > 0)
 | |
|     {
 | |
|         if (!m_mapILS.contains(name)) {
 | |
|             m_mapILS.insert(name, true);
 | |
|         }
 | |
| 
 | |
|         for (const auto& pipe : mapPipes)
 | |
|         {
 | |
|             MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|             SWGSDRangel::SWGMapItem *swgMapItem = new SWGSDRangel::SWGMapItem();
 | |
| 
 | |
|             swgMapItem->setName(new QString(name));
 | |
|             swgMapItem->setLabel(new QString(label));
 | |
|             swgMapItem->setLatitude(startLatitude);
 | |
|             swgMapItem->setLongitude(startLongitude);
 | |
|             swgMapItem->setAltitude(startAltitude);
 | |
|             QString image = QString("none");
 | |
|             swgMapItem->setImage(new QString(image));
 | |
|             swgMapItem->setImageRotation(0);
 | |
|             swgMapItem->setFixedPosition(true);
 | |
|             swgMapItem->setAltitudeReference(3); // CLIP_TO_GROUND
 | |
|             QList<SWGSDRangel::SWGMapCoordinate *> *coords = new QList<SWGSDRangel::SWGMapCoordinate *>();
 | |
| 
 | |
|             SWGSDRangel::SWGMapCoordinate* c = new SWGSDRangel::SWGMapCoordinate();
 | |
|             c->setLatitude(startLatitude);
 | |
|             c->setLongitude(startLongitude);
 | |
|             c->setAltitude(startAltitude);
 | |
|             coords->append(c);
 | |
| 
 | |
|             c = new SWGSDRangel::SWGMapCoordinate();
 | |
|             c->setLatitude(endLatitude);
 | |
|             c->setLongitude(endLongitude);
 | |
|             c->setAltitude(endAltitude);
 | |
|             coords->append(c);
 | |
| 
 | |
|             swgMapItem->setCoordinates(coords);
 | |
|             swgMapItem->setType(3);
 | |
| 
 | |
|             MainCore::MsgMapItem *msg = MainCore::MsgMapItem::create(m_ilsDemod, swgMapItem);
 | |
|             messageQueue->push(msg);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::clearILSFromMap()
 | |
| {
 | |
|     QMutableHashIterator<QString, bool> itr(m_mapILS);
 | |
|     while (itr.hasNext())
 | |
|     {
 | |
|         itr.next();
 | |
|         removeFromMap(itr.key());
 | |
|         itr.remove();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::drawILSOnMap()
 | |
| {
 | |
|     m_ilsValid = false;
 | |
|     if (m_disableDrawILS) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     // Check and get all needed settings to display the ILS
 | |
|     float thresholdLatitude, thresholdLongitude;
 | |
|     if (!Units::stringToLatitudeAndLongitude(m_settings.m_latitude + " " + m_settings.m_longitude, thresholdLatitude, thresholdLongitude))
 | |
|     {
 | |
|         qDebug() << "ILSDemodGUI::drawILSOnMap: Lat/lon invalid: " << (m_settings.m_latitude + " " + m_settings.m_longitude);
 | |
|         return;
 | |
|     }
 | |
|     // Can't set this to 0 and use CLIP_TO_GROUND or CLAMP_TO_GROUND, as ground at GARP is below runway at somewhere like EGKB
 | |
|     // Perhaps Map needs API so we can query altitude in current terrain
 | |
|     m_altitude = Units::feetToMetres(m_settings.m_elevation);
 | |
|     float slope = m_settings.m_slope;
 | |
| 
 | |
|     // Estimate touchdown point, which is roughly where glide-path should intersect runway
 | |
|     float thresholdToTouchDown = m_settings.m_refHeight / tan(Units::degreesToRadians(m_settings.m_glidePath));
 | |
|     calcRadialEndPoint(thresholdLatitude, thresholdLongitude, thresholdToTouchDown, m_settings.m_trueBearing, m_tdLatitude, m_tdLongitude);
 | |
| 
 | |
|     // Estimate localizer reference point (can be localizer location, but may be behind - similar to GARP)
 | |
|     float thresholdToLoc = 105.0f / tan(Units::degreesToRadians(m_settings.m_courseWidth / 2.0f));
 | |
|     calcRadialEndPoint(thresholdLatitude, thresholdLongitude, thresholdToLoc, m_settings.m_trueBearing, m_locLatitude, m_locLongitude);
 | |
|     m_locAltitude = m_altitude + thresholdToLoc * (slope / 100.0f);
 | |
| 
 | |
|     // Update GPS angle
 | |
|     updateGPSAngle();
 | |
| 
 | |
|     // Check to see if there are any Maps open
 | |
|     QList<ObjectPipe*> pipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", pipes);
 | |
|     if (pipes.size() == 0) {
 | |
|         return;
 | |
|     }
 | |
|     m_hasDrawnILS = true;
 | |
| 
 | |
|     // Check to see if we have a LOC Demod open, if so, G/S Demod doesn't need to draw
 | |
|     if (m_settings.m_mode == ILSDemodSettings::GS)
 | |
|     {
 | |
|         MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "ilsdemod", pipes);
 | |
|         if (pipes.size() > 0) {
 | |
|             return;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     clearILSFromMap();
 | |
| 
 | |
|     // Check to see if map is using Ellipsoid for terrain
 | |
|     int featureSetIndex = -1;
 | |
|     int featureIndex = -1;
 | |
|     QString terrain;
 | |
|     Feature *feature = FeatureWebAPIUtils::getFeature(featureSetIndex, featureIndex, "sdrangel.feature.map");
 | |
|     if (feature && ChannelWebAPIUtils::getFeatureSetting(featureSetIndex, featureIndex, "terrain", terrain))
 | |
|     {
 | |
|         if (terrain == "Ellipsoid")
 | |
|         {
 | |
|             m_altitude = 0.0f;
 | |
|             slope = 0.0f;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     m_locDistance = Units::nauticalMilesToMetres(18); // 3.1.3.3.1 (Can be 25NM)
 | |
|     m_gsDistance = Units::nauticalMilesToMetres(10);
 | |
|     m_ilsValid = true;
 | |
| 
 | |
|      // Runway true bearing points towards runway (direction aircraft land) - calculate bearing towards aircraft
 | |
|     float bearing = fmod(m_settings.m_trueBearing - 180.0f, 360.0f);
 | |
| 
 | |
|    // Calculate course line (actually true bearing (rather than magnetic course) so we can plot on map)
 | |
|     float lineEndLatitude, lineEndLongitude;
 | |
|     calcRadialEndPoint(m_locLatitude, m_locLongitude, m_locDistance, bearing, lineEndLatitude, lineEndLongitude);
 | |
| 
 | |
|     QList<QGeoCoordinate> coords;
 | |
|     float midLatitude, midLongitude;
 | |
|     float endLatitude, endLongitude;
 | |
|     // Calculate sector lines
 | |
|     m_locToTouchdown = thresholdToLoc - thresholdToTouchDown;
 | |
|     float tdToEnd =  (m_locDistance - m_locToTouchdown);
 | |
|     float locEndAltitude = m_altitude + sin(Units::degreesToRadians(m_settings.m_glidePath)) * tdToEnd;
 | |
|     // Coverage should be +/- 10 degrees
 | |
|     // We plot course sector (i.e. +/- 0.155 DDM)
 | |
|     float bearingR = fmod(bearing - m_settings.m_courseWidth / 2.0, 360.0f);
 | |
|     calcRadialEndPoint(m_locLatitude, m_locLongitude, m_locToTouchdown, bearingR, midLatitude, midLongitude);
 | |
|     calcRadialEndPoint(midLatitude, midLongitude, tdToEnd, bearingR, endLatitude, endLongitude);
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(m_locLatitude, m_locLongitude, m_locAltitude));
 | |
|     coords.append(QGeoCoordinate(midLatitude, midLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     addPolygonToMap("ILS Localizer Sector 150Hz Runway", "", coords, QColor(0, 200, 0, 125).rgba());
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(midLatitude, midLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(endLatitude, endLongitude, locEndAltitude));
 | |
|     coords.append(QGeoCoordinate(lineEndLatitude, lineEndLongitude, locEndAltitude));
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     addPolygonToMap("ILS Localizer Sector 150Hz", "", coords, QColor(0, 200, 0, 125).rgba());
 | |
| 
 | |
|     float bearingL = fmod(bearing + m_settings.m_courseWidth / 2.0, 360.0f);
 | |
|     calcRadialEndPoint(m_locLatitude, m_locLongitude, m_locToTouchdown, bearingL, midLatitude, midLongitude);
 | |
|     calcRadialEndPoint(midLatitude, midLongitude, tdToEnd, bearingL, endLatitude, endLongitude);
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(m_locLatitude, m_locLongitude, m_locAltitude));
 | |
|     coords.append(QGeoCoordinate(midLatitude, midLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     addPolygonToMap("ILS Localizer Sector 90Hz Runway", "", coords, QColor(0, 150, 0, 125).rgba());
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(midLatitude, midLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(endLatitude, endLongitude, locEndAltitude));
 | |
|     coords.append(QGeoCoordinate(lineEndLatitude, lineEndLongitude, locEndAltitude));
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     addPolygonToMap("ILS Localizer Sector 90Hz", "", coords, QColor(0, 150, 0, 125).rgba());
 | |
| 
 | |
|     // Calculate start of glide path
 | |
|     // 1.75*theta and 0.45*theta are the required limits of glidepath coverage (Figure 10)
 | |
|     // We plot full scale deflection (i.e. +/- 0.175 DDM)
 | |
|     float gpEndLatitude, gpEndLongitude;
 | |
|     calcRadialEndPoint(m_tdLatitude, m_tdLongitude, m_gsDistance, bearing, gpEndLatitude, gpEndLongitude);
 | |
|     float endAltitudeA = m_altitude + sin(Units::degreesToRadians(m_settings.m_glidePath + 0.36f)) * m_gsDistance;
 | |
|     float endAltitudeB = m_altitude + sin(Units::degreesToRadians(m_settings.m_glidePath - 0.36f)) * m_gsDistance;
 | |
|     float gpEndAltitude = m_altitude + sin(Units::degreesToRadians(m_settings.m_glidePath)) * m_gsDistance;
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(gpEndLatitude, gpEndLongitude, endAltitudeA));
 | |
|     coords.append(QGeoCoordinate(gpEndLatitude, gpEndLongitude, gpEndAltitude));
 | |
|     addPolygonToMap("ILS Glide Path 90Hz", "", coords, QColor(150, 150, 0, 175).rgba());
 | |
| 
 | |
|     coords.clear();
 | |
|     coords.append(QGeoCoordinate(m_tdLatitude, m_tdLongitude, m_altitude));
 | |
|     coords.append(QGeoCoordinate(gpEndLatitude, gpEndLongitude, endAltitudeB));
 | |
|     coords.append(QGeoCoordinate(gpEndLatitude, gpEndLongitude, gpEndAltitude));
 | |
|     addPolygonToMap("ILS Glide Path 150Hz", "", coords, QColor(200, 200, 0, 175).rgba());
 | |
| 
 | |
|     drawPath();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::drawPath()
 | |
| {
 | |
|     if (!m_hasDrawnILS) {
 | |
|         drawILSOnMap();
 | |
|     }
 | |
|     if (!m_ilsValid) {
 | |
|         return;
 | |
|     }
 | |
|     float locAngle = std::isnan(m_locAngle) ? 0.0f : m_locAngle;
 | |
|     float gsAngle = std::isnan(m_gsAngle) ? m_settings.m_glidePath : (m_settings.m_glidePath + m_gsAngle);
 | |
| 
 | |
|     // Plot line at current estimated loc & g/s angles
 | |
|     float bearing = fmod(m_settings.m_trueBearing - 180.0f + locAngle, 360.0f);
 | |
|     float tdLatitude, tdLongitude;
 | |
|     float endLatitude, endLongitude;
 | |
|     float distance = m_locDistance - m_locToTouchdown;
 | |
| 
 | |
|     calcRadialEndPoint(m_locLatitude, m_locLongitude, m_locToTouchdown, bearing, tdLatitude, tdLongitude);
 | |
|     calcRadialEndPoint(tdLatitude, tdLongitude, distance, bearing, endLatitude, endLongitude);
 | |
|     float endAltitude = m_altitude + sin(Units::degreesToRadians(gsAngle)) * distance;
 | |
| 
 | |
|     QStringList runwayStrings = m_settings.m_runway.split(" ");
 | |
|     QString label;
 | |
|     if (runwayStrings.size() == 2) {
 | |
|         label = QString("%1 %2").arg(runwayStrings[1]).arg(m_settings.m_ident); // Assume first string is airport ICAO
 | |
|     } else if (!runwayStrings[0].isEmpty()) {
 | |
|         label = QString("%1 %2").arg(runwayStrings[0]).arg(m_settings.m_ident);
 | |
|     } else {
 | |
|         label = QString("%2%3T %1").arg(m_settings.m_ident).arg((int)std::round(m_settings.m_trueBearing)).arg(QChar(0xb0));
 | |
|     }
 | |
|     addLineToMap("ILS Radial Runway", label, m_locLatitude, m_locLongitude, m_locAltitude, tdLatitude, tdLongitude, m_altitude);
 | |
|     addLineToMap("ILS Radial", "", tdLatitude, tdLongitude, m_altitude, endLatitude, endLongitude, endAltitude);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::updateGPSAngle()
 | |
| {
 | |
|     // Get GPS position
 | |
|     float gpsLatitude = MainCore::instance()->getSettings().getLatitude();
 | |
|     float gpsLongitude = MainCore::instance()->getSettings().getLongitude();
 | |
|     float gpsAltitude = MainCore::instance()->getSettings().getAltitude();
 | |
|     QGeoCoordinate gpsPos(gpsLatitude, gpsLongitude, gpsAltitude);
 | |
| 
 | |
|     if (m_settings.m_mode == ILSDemodSettings::LOC)
 | |
|     {
 | |
|         // Calculate angle from localizer to GPS
 | |
|         QGeoCoordinate locPos(m_locLatitude, m_locLongitude, m_altitude);
 | |
|         qreal angle = gpsPos.azimuthTo(locPos);
 | |
|         angle = angle - m_settings.m_trueBearing;
 | |
|         ui->gpsAngle->setText(QString::number(std::abs(angle), 'f', 1));
 | |
|         ui->gpsAngleDirection->setText(formatAngleDirection(angle));
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Calculate angle from glide path runway intersection to GPS
 | |
|         QGeoCoordinate tdPos(m_tdLatitude, m_tdLongitude, m_altitude);
 | |
|         qreal d = tdPos.distanceTo(gpsPos);
 | |
|         float h = gpsAltitude - m_altitude;
 | |
|         float angle = Units::radiansToDegrees(atan(h/d)) - m_settings.m_glidePath;
 | |
|         ui->gpsAngle->setText(QString::number(std::abs(angle), 'f', 1));
 | |
|         ui->gpsAngleDirection->setText(formatAngleDirection(angle));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_trueBearing_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_trueBearing = (float) value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_elevation_valueChanged(int value)
 | |
| {
 | |
|     m_settings.m_elevation = value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_latitude_editingFinished()
 | |
| {
 | |
|     m_settings.m_latitude = ui->latitude->text();
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_longitude_editingFinished()
 | |
| {
 | |
|     m_settings.m_longitude = ui->longitude->text();
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_glidePath_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_glidePath = value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_height_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_refHeight = (float)value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_courseWidth_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_courseWidth = (float)value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_slope_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_slope = (float)value;
 | |
|     applySettings();
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_udpEnabled_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_udpEnabled = checked;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_udpAddress_editingFinished()
 | |
| {
 | |
|     m_settings.m_udpAddress = ui->udpAddress->text();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_udpPort_editingFinished()
 | |
| {
 | |
|     m_settings.m_udpPort = ui->udpPort->text().toInt();
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_channel1_currentIndexChanged(int index)
 | |
| {
 | |
|     m_settings.m_scopeCh1 = index;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_channel2_currentIndexChanged(int index)
 | |
| {
 | |
|     m_settings.m_scopeCh2 = index;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::onWidgetRolled(QWidget* widget, bool rollDown)
 | |
| {
 | |
|     (void) widget;
 | |
|     (void) rollDown;
 | |
| 
 | |
|     getRollupContents()->saveState(m_rollupState);
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::onMenuDialogCalled(const QPoint &p)
 | |
| {
 | |
|     if (m_contextMenuType == ContextMenuType::ContextMenuChannelSettings)
 | |
|     {
 | |
|         BasicChannelSettingsDialog dialog(&m_channelMarker, this);
 | |
|         dialog.setUseReverseAPI(m_settings.m_useReverseAPI);
 | |
|         dialog.setReverseAPIAddress(m_settings.m_reverseAPIAddress);
 | |
|         dialog.setReverseAPIPort(m_settings.m_reverseAPIPort);
 | |
|         dialog.setReverseAPIDeviceIndex(m_settings.m_reverseAPIDeviceIndex);
 | |
|         dialog.setReverseAPIChannelIndex(m_settings.m_reverseAPIChannelIndex);
 | |
|         dialog.setDefaultTitle(m_displayedName);
 | |
| 
 | |
|         if (m_deviceUISet->m_deviceMIMOEngine)
 | |
|         {
 | |
|             dialog.setNumberOfStreams(m_ilsDemod->getNumberOfDeviceStreams());
 | |
|             dialog.setStreamIndex(m_settings.m_streamIndex);
 | |
|         }
 | |
| 
 | |
|         dialog.move(p);
 | |
|         new DialogPositioner(&dialog, false);
 | |
|         dialog.exec();
 | |
| 
 | |
|         m_settings.m_rgbColor = m_channelMarker.getColor().rgb();
 | |
|         m_settings.m_title = m_channelMarker.getTitle();
 | |
|         m_settings.m_useReverseAPI = dialog.useReverseAPI();
 | |
|         m_settings.m_reverseAPIAddress = dialog.getReverseAPIAddress();
 | |
|         m_settings.m_reverseAPIPort = dialog.getReverseAPIPort();
 | |
|         m_settings.m_reverseAPIDeviceIndex = dialog.getReverseAPIDeviceIndex();
 | |
|         m_settings.m_reverseAPIChannelIndex = dialog.getReverseAPIChannelIndex();
 | |
| 
 | |
|         setWindowTitle(m_settings.m_title);
 | |
|         setTitle(m_channelMarker.getTitle());
 | |
|         setTitleColor(m_settings.m_rgbColor);
 | |
| 
 | |
|         if (m_deviceUISet->m_deviceMIMOEngine)
 | |
|         {
 | |
|             m_settings.m_streamIndex = dialog.getSelectedStreamIndex();
 | |
|             m_channelMarker.clearStreamIndexes();
 | |
|             m_channelMarker.addStreamIndex(m_settings.m_streamIndex);
 | |
|             updateIndexLabel();
 | |
|         }
 | |
| 
 | |
|         applySettings();
 | |
|     }
 | |
| 
 | |
|     resetContextMenuType();
 | |
| }
 | |
| 
 | |
| ILSDemodGUI::ILSDemodGUI(PluginAPI* pluginAPI, DeviceUISet *deviceUISet, BasebandSampleSink *rxChannel, QWidget* parent) :
 | |
|     ChannelGUI(parent),
 | |
|     ui(new Ui::ILSDemodGUI),
 | |
|     m_pluginAPI(pluginAPI),
 | |
|     m_deviceUISet(deviceUISet),
 | |
|     m_channelMarker(this),
 | |
|     m_deviceCenterFrequency(0),
 | |
|     m_doApplySettings(true),
 | |
|     m_squelchOpen(false),
 | |
|     m_tickCount(0),
 | |
|     m_markerNo(0),
 | |
|     m_disableDrawILS(false),
 | |
|     m_hasDrawnILS(false),
 | |
|     m_locAngle(NAN),
 | |
|     m_gsAngle(NAN)
 | |
| {
 | |
|     setAttribute(Qt::WA_DeleteOnClose, true);
 | |
|     m_helpURL = "plugins/channelrx/demodils/readme.md";
 | |
|     RollupContents *rollupContents = getRollupContents();
 | |
|     ui->setupUi(rollupContents);
 | |
|     setSizePolicy(rollupContents->sizePolicy());
 | |
|     rollupContents->arrangeRollups();
 | |
|     connect(rollupContents, SIGNAL(widgetRolled(QWidget*,bool)), this, SLOT(onWidgetRolled(QWidget*,bool)));
 | |
|     connect(this, SIGNAL(customContextMenuRequested(const QPoint &)), this, SLOT(onMenuDialogCalled(const QPoint &)));
 | |
| 
 | |
|     m_ilsDemod = reinterpret_cast<ILSDemod*>(rxChannel);
 | |
|     m_ilsDemod->setMessageQueueToGUI(getInputMessageQueue());
 | |
|     m_spectrumVis = m_ilsDemod->getSpectrumVis();
 | |
|     m_spectrumVis->setGLSpectrum(ui->glSpectrum);
 | |
| 
 | |
|     connect(&MainCore::instance()->getMasterTimer(), SIGNAL(timeout()), this, SLOT(tick())); // 50 ms
 | |
| 
 | |
|     CRightClickEnabler *audioMuteRightClickEnabler = new CRightClickEnabler(ui->audioMute);
 | |
|     connect(audioMuteRightClickEnabler, SIGNAL(rightClick(const QPoint &)), this, SLOT(audioSelect(const QPoint &)));
 | |
| 
 | |
|     ui->deltaFrequencyLabel->setText(QString("%1f").arg(QChar(0x94, 0x03)));
 | |
|     ui->deltaFrequency->setColorMapper(ColorMapper(ColorMapper::GrayGold));
 | |
|     ui->deltaFrequency->setValueRange(false, 7, -9999999, 9999999);
 | |
|     ui->channelPowerMeter->setColorTheme(LevelMeterSignalDB::ColorGreenAndBlue);
 | |
| 
 | |
|     m_scopeVis = m_ilsDemod->getScopeSink();
 | |
|     m_scopeVis->setGLScope(ui->glScope);
 | |
|     ui->glScope->connectTimer(MainCore::instance()->getMasterTimer());
 | |
|     ui->scopeGUI->setBuddies(m_scopeVis->getInputMessageQueue(), m_scopeVis, ui->glScope);
 | |
| 
 | |
|     // Scope settings to display the IQ waveforms
 | |
|     ui->scopeGUI->setPreTrigger(1);
 | |
|     GLScopeSettings::TraceData traceDataI, traceDataQ;
 | |
|     traceDataI.m_projectionType = Projector::ProjectionReal;
 | |
|     traceDataI.m_amp = 1.0;      // for -1 to +1
 | |
|     traceDataI.m_ofs = 0.0;      // vertical offset
 | |
|     traceDataQ.m_projectionType = Projector::ProjectionImag;
 | |
|     traceDataQ.m_amp = 1.0;
 | |
|     traceDataQ.m_ofs = 0.0;
 | |
|     ui->scopeGUI->changeTrace(0, traceDataI);
 | |
|     ui->scopeGUI->addTrace(traceDataQ);
 | |
|     ui->scopeGUI->setDisplayMode(GLScopeSettings::DisplayXYV);
 | |
|     ui->scopeGUI->focusOnTrace(0); // re-focus to take changes into account in the GUI
 | |
| 
 | |
|     GLScopeSettings::TriggerData triggerData;
 | |
|     triggerData.m_triggerLevel = 0.1;
 | |
|     triggerData.m_triggerLevelCoarse = 10;
 | |
|     triggerData.m_triggerPositiveEdge = true;
 | |
|     ui->scopeGUI->changeTrigger(0, triggerData);
 | |
|     ui->scopeGUI->focusOnTrigger(0); // re-focus to take changes into account in the GUI
 | |
| 
 | |
|     m_scopeVis->setLiveRate(ILSDemodSettings::ILSDEMOD_SPECTRUM_SAMPLE_RATE);
 | |
|     m_scopeVis->configure(500, 1, 0, 0, true);   // not working!
 | |
|     //m_scopeVis->setFreeRun(false); // FIXME: add method rather than call m_scopeVis->configure()
 | |
| 
 | |
|     ui->spectrumGUI->setBuddies(m_spectrumVis, ui->glSpectrum);
 | |
| 
 | |
|     ui->glSpectrum->setCenterFrequency(0);
 | |
|     ui->glSpectrum->setSampleRate(ILSDemodSettings::ILSDEMOD_SPECTRUM_SAMPLE_RATE);
 | |
|     ui->glSpectrum->setMeasurementParams(SpectrumSettings::MeasurementPeaks, 0, 1000, 90, 150, 1, 5, true, 1);
 | |
|     ui->glSpectrum->setMeasurementsVisible(true);
 | |
| 
 | |
|     m_channelMarker.setColor(Qt::yellow);
 | |
|     m_channelMarker.setBandwidth(m_settings.m_rfBandwidth);
 | |
|     m_channelMarker.setCenterFrequency(m_settings.m_inputFrequencyOffset);
 | |
|     m_channelMarker.setTitle("ILS Demodulator");
 | |
|     m_channelMarker.blockSignals(false);
 | |
|     m_channelMarker.setVisible(true); // activate signal on the last setting only
 | |
| 
 | |
|     setTitleColor(m_channelMarker.getColor());
 | |
|     m_settings.setChannelMarker(&m_channelMarker);
 | |
|     m_settings.setScopeGUI(ui->scopeGUI);
 | |
|     m_settings.setSpectrumGUI(ui->spectrumGUI);
 | |
|     m_settings.setRollupState(&m_rollupState);
 | |
| 
 | |
|     m_deviceUISet->addChannelMarker(&m_channelMarker);
 | |
| 
 | |
|     on_mode_currentIndexChanged(ui->mode->currentIndex()); // Populate frequencies
 | |
| 
 | |
|     connect(&m_channelMarker, SIGNAL(changedByCursor()), this, SLOT(channelMarkerChangedByCursor()));
 | |
|     connect(&m_channelMarker, SIGNAL(highlightedByCursor()), this, SLOT(channelMarkerHighlightedByCursor()));
 | |
|     connect(getInputMessageQueue(), SIGNAL(messageEnqueued()), this, SLOT(handleInputMessages()));
 | |
| 
 | |
|     for (const auto& ils : m_ils) {
 | |
|         ui->ident->addItem(ils.m_ident);
 | |
|     }
 | |
| 
 | |
|     // Get updated when position changes
 | |
|     connect(&MainCore::instance()->getSettings(), &MainSettings::preferenceChanged, this, &ILSDemodGUI::preferenceChanged);
 | |
| 
 | |
|     ui->scopeContainer->setVisible(false);
 | |
| 
 | |
|     displaySettings();
 | |
|     makeUIConnections();
 | |
|     applySettings(true);
 | |
|     m_resizer.enableChildMouseTracking();
 | |
|     drawILSOnMap();
 | |
| 
 | |
|     bool devMode = false;
 | |
|     ui->pCarrierLabel->setVisible(devMode);
 | |
|     ui->pCarrier->setVisible(devMode);
 | |
|     ui->pCarrierUnits->setVisible(devMode);
 | |
|     ui->p90Label->setVisible(devMode);
 | |
|     ui->p90->setVisible(devMode);
 | |
|     ui->p90Units->setVisible(devMode);
 | |
|     ui->p150Label->setVisible(devMode);
 | |
|     ui->p150->setVisible(devMode);
 | |
|     ui->p150Units->setVisible(devMode);
 | |
| 
 | |
|     SpectrumSettings spectrumSettings = m_spectrumVis->getSettings();
 | |
|     spectrumSettings.m_fftSize = 256;
 | |
|     spectrumSettings.m_fftWindow = FFTWindow::Flattop;  // To match what's used in sink
 | |
|     spectrumSettings.m_averagingMode = SpectrumSettings::AvgModeMoving;
 | |
|     spectrumSettings.m_averagingValue = 1;
 | |
|     spectrumSettings.m_displayWaterfall = true;
 | |
|     spectrumSettings.m_displayMaxHold = false;
 | |
|     spectrumSettings.m_ssb = false;
 | |
|     spectrumSettings.m_displayHistogram = false;
 | |
|     spectrumSettings.m_displayCurrent = true;
 | |
|     spectrumSettings.m_spectrumStyle = SpectrumSettings::Gradient;
 | |
|     SpectrumVis::MsgConfigureSpectrumVis *msg = SpectrumVis::MsgConfigureSpectrumVis::create(spectrumSettings, false);
 | |
|     m_spectrumVis->getInputMessageQueue()->push(msg);
 | |
| 
 | |
|     scanAvailableChannels();
 | |
|     QObject::connect(
 | |
|         MainCore::instance(),
 | |
|         &MainCore::channelAdded,
 | |
|         this,
 | |
|         &ILSDemodGUI::handleChannelAdded
 | |
|     );
 | |
| }
 | |
| 
 | |
| ILSDemodGUI::~ILSDemodGUI()
 | |
| {
 | |
|     QObject::disconnect(&MainCore::instance()->getMasterTimer(), &QTimer::timeout, this, &ILSDemodGUI::tick);
 | |
|     QObject::disconnect(MainCore::instance(), &MainCore::channelAdded, this, &ILSDemodGUI::handleChannelAdded);
 | |
|     on_clearMarkers_clicked();
 | |
|     clearILSFromMap();
 | |
|     delete ui;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::blockApplySettings(bool block)
 | |
| {
 | |
|     m_doApplySettings = !block;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::applySettings(bool force)
 | |
| {
 | |
|     if (m_doApplySettings)
 | |
|     {
 | |
|         ILSDemod::MsgConfigureILSDemod* message = ILSDemod::MsgConfigureILSDemod::create( m_settings, force);
 | |
|         m_ilsDemod->getInputMessageQueue()->push(message);
 | |
|     }
 | |
| }
 | |
| 
 | |
| QString ILSDemodGUI::formatDDM(float ddm) const
 | |
| {
 | |
|     switch (m_settings.m_ddmUnits)
 | |
|     {
 | |
|     case ILSDemodSettings::PERCENT:
 | |
|         return QString::number(ddm * 100.0f, 'f', 1);
 | |
|     case ILSDemodSettings::MICROAMPS:
 | |
|         if (m_settings.m_mode == ILSDemodSettings::LOC) {
 | |
|             return QString::number(ddm * 967.75f, 'f', 1);  // 0.155 -> 150uA
 | |
|         } else {
 | |
|             return QString::number(ddm * 857.125f, 'f', 1);
 | |
|         }
 | |
|     default:
 | |
|         return QString::number(ddm, 'f', 3);
 | |
|     }
 | |
| }
 | |
| 
 | |
| QString ILSDemodGUI::formatFrequency(int frequency) const
 | |
| {
 | |
|     QString suffix = "";
 | |
|     if (width() > 450) {
 | |
|         suffix = " Hz";
 | |
|     }
 | |
|     return QString("%1%2").arg(frequency).arg(suffix);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::displaySettings()
 | |
| {
 | |
|     m_channelMarker.blockSignals(true);
 | |
|     m_channelMarker.setBandwidth(m_settings.m_rfBandwidth);
 | |
|     m_channelMarker.setCenterFrequency(m_settings.m_inputFrequencyOffset);
 | |
|     m_channelMarker.setTitle(m_settings.m_title);
 | |
|     m_channelMarker.blockSignals(false);
 | |
|     m_channelMarker.setColor(m_settings.m_rgbColor); // activate signal on the last setting only
 | |
| 
 | |
|     setTitleColor(m_settings.m_rgbColor);
 | |
|     setWindowTitle(m_channelMarker.getTitle());
 | |
|     setTitle(m_channelMarker.getTitle());
 | |
| 
 | |
|     blockApplySettings(true);
 | |
| 
 | |
|     ui->deltaFrequency->setValue(m_channelMarker.getCenterFrequency());
 | |
| 
 | |
|     ui->rfBWText->setText(formatFrequency((int)m_settings.m_rfBandwidth));
 | |
|     ui->rfBW->setValue(m_settings.m_rfBandwidth);
 | |
| 
 | |
|     ui->volume->setValue(m_settings.m_volume * 10.0);
 | |
|     ui->volumeText->setText(QString("%1").arg(m_settings.m_volume, 0, 'f', 1));
 | |
| 
 | |
|     ui->squelch->setValue(m_settings.m_squelch);
 | |
|     ui->squelchText->setText(QString("%1 dB").arg(m_settings.m_squelch));
 | |
| 
 | |
|     ui->audioMute->setChecked(m_settings.m_audioMute);
 | |
|     ui->average->setChecked(m_settings.m_average);
 | |
| 
 | |
|     ui->thresh->setValue(m_settings.m_identThreshold * 10.0);
 | |
|     ui->threshText->setText(QString("%1").arg(m_settings.m_identThreshold, 0, 'f', 1));
 | |
| 
 | |
|     m_disableDrawILS = true;
 | |
|     ui->mode->setCurrentIndex((int) m_settings.m_mode);
 | |
|     ui->frequency->setCurrentIndex(m_settings.m_frequencyIndex);
 | |
|     ui->ident->setCurrentText(m_settings.m_ident);
 | |
|     ui->runway->setText(m_settings.m_runway);
 | |
|     on_runway_editingFinished();
 | |
|     ui->trueBearing->setValue(m_settings.m_trueBearing);
 | |
|     ui->height->setValue(m_settings.m_refHeight);
 | |
|     ui->courseWidth->setValue(m_settings.m_courseWidth);
 | |
|     ui->slope->setValue(m_settings.m_slope);
 | |
|     ui->latitude->setText(m_settings.m_latitude);
 | |
|     ui->longitude->setText(m_settings.m_longitude);
 | |
|     ui->elevation->setValue(m_settings.m_elevation);
 | |
|     ui->glidePath->setValue(m_settings.m_glidePath);
 | |
|     m_disableDrawILS = false;
 | |
| 
 | |
|     updateIndexLabel();
 | |
| 
 | |
|     ui->udpEnabled->setChecked(m_settings.m_udpEnabled);
 | |
|     ui->udpAddress->setText(m_settings.m_udpAddress);
 | |
|     ui->udpPort->setText(QString::number(m_settings.m_udpPort));
 | |
| 
 | |
|     ui->channel1->setCurrentIndex(m_settings.m_scopeCh1);
 | |
|     ui->channel2->setCurrentIndex(m_settings.m_scopeCh2);
 | |
| 
 | |
|     ui->logFilename->setToolTip(QString(".csv log filename: %1").arg(m_settings.m_logFilename));
 | |
|     ui->logEnable->setChecked(m_settings.m_logEnabled);
 | |
| 
 | |
|     getRollupContents()->restoreState(m_rollupState);
 | |
|     updateAbsoluteCenterFrequency();
 | |
|     blockApplySettings(false);
 | |
|     drawILSOnMap();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::leaveEvent(QEvent* event)
 | |
| {
 | |
|     m_channelMarker.setHighlighted(false);
 | |
|     ChannelGUI::leaveEvent(event);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::enterEvent(EnterEventType* event)
 | |
| {
 | |
|     m_channelMarker.setHighlighted(true);
 | |
|     ChannelGUI::enterEvent(event);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::audioSelect(const QPoint& p)
 | |
| {
 | |
|     qDebug("ILSDemodGUI::audioSelect");
 | |
|     AudioSelectDialog audioSelect(DSPEngine::instance()->getAudioDeviceManager(), m_settings.m_audioDeviceName);
 | |
|     audioSelect.move(p);
 | |
|     new DialogPositioner(&audioSelect, false);
 | |
|     audioSelect.exec();
 | |
| 
 | |
|     if (audioSelect.m_selected)
 | |
|     {
 | |
|         m_settings.m_audioDeviceName = audioSelect.m_audioDeviceName;
 | |
|         applySettings();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::tick()
 | |
| {
 | |
|     double magsqAvg, magsqPeak;
 | |
|     int nbMagsqSamples;
 | |
|     m_ilsDemod->getMagSqLevels(magsqAvg, magsqPeak, nbMagsqSamples);
 | |
|     double powDbAvg = CalcDb::dbPower(magsqAvg);
 | |
|     double powDbPeak = CalcDb::dbPower(magsqPeak);
 | |
|     ui->channelPowerMeter->levelChanged(
 | |
|             (100.0f + powDbAvg) / 100.0f,
 | |
|             (100.0f + powDbPeak) / 100.0f,
 | |
|             nbMagsqSamples);
 | |
| 
 | |
|     if (m_tickCount % 4 == 0) {
 | |
|         ui->channelPower->setText(QString::number(powDbAvg, 'f', 1));
 | |
|     }
 | |
| 
 | |
|     int audioSampleRate = m_ilsDemod->getAudioSampleRate();
 | |
|     bool squelchOpen = m_ilsDemod->getSquelchOpen();
 | |
| 
 | |
|     if (squelchOpen != m_squelchOpen)
 | |
|     {
 | |
|         if (audioSampleRate < 0) {
 | |
|             ui->audioMute->setStyleSheet("QToolButton { background-color : red; }");
 | |
|         } else if (squelchOpen) {
 | |
|             ui->audioMute->setStyleSheet("QToolButton { background-color : green; }");
 | |
|         } else {
 | |
|             ui->audioMute->setStyleSheet("QToolButton { background:rgb(79,79,79); }");
 | |
|         }
 | |
| 
 | |
|         m_squelchOpen = squelchOpen;
 | |
|     }
 | |
| 
 | |
|     if (!m_hasDrawnILS && (m_tickCount % 25 == 0))
 | |
|     {
 | |
|         // Check to see if there are any Maps open - Is there a signal we can use instead?
 | |
|         QList<ObjectPipe*> mapPipes;
 | |
|         MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "mapitems", mapPipes);
 | |
|         if (mapPipes.size() > 0) {
 | |
|             drawILSOnMap();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     m_tickCount++;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_logEnable_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_logEnabled = checked;
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::on_logFilename_clicked()
 | |
| {
 | |
|     // Get filename to save to
 | |
|     QFileDialog fileDialog(nullptr, "Select CSV file to log data to", "", "*.csv");
 | |
|     fileDialog.setAcceptMode(QFileDialog::AcceptSave);
 | |
|     if (fileDialog.exec())
 | |
|     {
 | |
|         QStringList fileNames = fileDialog.selectedFiles();
 | |
|         if (fileNames.size() > 0)
 | |
|         {
 | |
|             m_settings.m_logFilename = fileNames[0];
 | |
|             ui->logFilename->setToolTip(QString(".csv log filename: %1").arg(m_settings.m_logFilename));
 | |
|             applySettings();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::makeUIConnections()
 | |
| {
 | |
|     QObject::connect(ui->deltaFrequency, &ValueDialZ::changed, this, &ILSDemodGUI::on_deltaFrequency_changed);
 | |
|     QObject::connect(ui->rfBW, &QSlider::valueChanged, this, &ILSDemodGUI::on_rfBW_valueChanged);
 | |
|     QObject::connect(ui->mode, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_mode_currentIndexChanged);
 | |
|     QObject::connect(ui->frequency, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_frequency_currentIndexChanged);
 | |
|     QObject::connect(ui->average, &QPushButton::clicked, this, &ILSDemodGUI::on_average_clicked);
 | |
|     QObject::connect(ui->volume, &QDial::valueChanged, this, &ILSDemodGUI::on_volume_valueChanged);
 | |
|     QObject::connect(ui->squelch, &QDial::valueChanged, this, &ILSDemodGUI::on_squelch_valueChanged);
 | |
|     QObject::connect(ui->audioMute, &QToolButton::toggled, this, &ILSDemodGUI::on_audioMute_toggled);
 | |
|     QObject::connect(ui->thresh, &QDial::valueChanged, this, &ILSDemodGUI::on_thresh_valueChanged);
 | |
|     QObject::connect(ui->ddmUnits, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_ddmUnits_currentIndexChanged);
 | |
|     QObject::connect(ui->ident, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_ident_currentIndexChanged);
 | |
|     QObject::connect(ui->ident->lineEdit(), &QLineEdit::editingFinished, this, &ILSDemodGUI::on_ident_editingFinished);
 | |
|     QObject::connect(ui->runway, &QLineEdit::editingFinished, this, &ILSDemodGUI::on_runway_editingFinished);
 | |
|     QObject::connect(ui->trueBearing, QOverload<double>::of(&QDoubleSpinBox::valueChanged), this, &ILSDemodGUI::on_trueBearing_valueChanged);
 | |
|     QObject::connect(ui->elevation, QOverload<int>::of(&QSpinBox::valueChanged), this, &ILSDemodGUI::on_elevation_valueChanged);
 | |
|     QObject::connect(ui->latitude, &QLineEdit::editingFinished, this, &ILSDemodGUI::on_latitude_editingFinished);
 | |
|     QObject::connect(ui->longitude, &QLineEdit::editingFinished, this, &ILSDemodGUI::on_longitude_editingFinished);
 | |
|     QObject::connect(ui->glidePath, QOverload<double>::of(&QDoubleSpinBox::valueChanged), this, &ILSDemodGUI::on_glidePath_valueChanged);
 | |
|     QObject::connect(ui->height, QOverload<double>::of(&QDoubleSpinBox::valueChanged), this, &ILSDemodGUI::on_height_valueChanged);
 | |
|     QObject::connect(ui->courseWidth, QOverload<double>::of(&QDoubleSpinBox::valueChanged), this, &ILSDemodGUI::on_courseWidth_valueChanged);
 | |
|     QObject::connect(ui->slope, QOverload<double>::of(&QDoubleSpinBox::valueChanged), this, &ILSDemodGUI::on_slope_valueChanged);
 | |
|     QObject::connect(ui->findOnMap, &QPushButton::clicked, this, &ILSDemodGUI::on_findOnMap_clicked);
 | |
|     QObject::connect(ui->addMarker, &QPushButton::clicked, this, &ILSDemodGUI::on_addMarker_clicked);
 | |
|     QObject::connect(ui->clearMarkers, &QPushButton::clicked, this, &ILSDemodGUI::on_clearMarkers_clicked);
 | |
|     QObject::connect(ui->udpEnabled, &QCheckBox::clicked, this, &ILSDemodGUI::on_udpEnabled_clicked);
 | |
|     QObject::connect(ui->udpAddress, &QLineEdit::editingFinished, this, &ILSDemodGUI::on_udpAddress_editingFinished);
 | |
|     QObject::connect(ui->udpPort, &QLineEdit::editingFinished, this, &ILSDemodGUI::on_udpPort_editingFinished);
 | |
|     QObject::connect(ui->logEnable, &ButtonSwitch::clicked, this, &ILSDemodGUI::on_logEnable_clicked);
 | |
|     QObject::connect(ui->logFilename, &QToolButton::clicked, this, &ILSDemodGUI::on_logFilename_clicked);
 | |
|     QObject::connect(ui->channel1, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_channel1_currentIndexChanged);
 | |
|     QObject::connect(ui->channel2, QOverload<int>::of(&QComboBox::currentIndexChanged), this, &ILSDemodGUI::on_channel2_currentIndexChanged);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::updateAbsoluteCenterFrequency()
 | |
| {
 | |
|     setStatusFrequency(m_deviceCenterFrequency + m_settings.m_inputFrequencyOffset);
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::preferenceChanged(int elementType)
 | |
| {
 | |
|     Preferences::ElementType pref = (Preferences::ElementType)elementType;
 | |
|     if ((pref == Preferences::Latitude) || (pref == Preferences::Longitude) || (pref == Preferences::Altitude)) {
 | |
|         updateGPSAngle();
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool ILSDemodGUI::sendToLOCChannel(float angle)
 | |
| {
 | |
|     QList<ObjectPipe*> pipes;
 | |
|     MainCore::instance()->getMessagePipes().getMessagePipes(m_ilsDemod, "ilsdemod", pipes);
 | |
|     for (const auto& pipe : pipes)
 | |
|     {
 | |
|         MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|         MsgGSAngle *msg = MsgGSAngle::create(angle);
 | |
|         messageQueue->push(msg);
 | |
|     }
 | |
|     return pipes.size() > 0;
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::closePipes()
 | |
| {
 | |
|     for (const auto channel : m_availableChannels)
 | |
|     {
 | |
|         ObjectPipe *pipe = MainCore::instance()->getMessagePipes().unregisterProducerToConsumer(channel, m_ilsDemod, "ilsdemod");
 | |
|         if (pipe)
 | |
|         {
 | |
|             MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
| 
 | |
|             if (messageQueue) {
 | |
|                 disconnect(messageQueue, &MessageQueue::messageEnqueued, this, nullptr);  // Have to use nullptr, as slot is a lambda.
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::scanAvailableChannels()
 | |
| {
 | |
|     MainCore *mainCore = MainCore::instance();
 | |
|     MessagePipes& messagePipes = mainCore->getMessagePipes();
 | |
|     std::vector<DeviceSet*>& deviceSets = mainCore->getDeviceSets();
 | |
|     m_availableChannels.clear();
 | |
| 
 | |
|     for (const auto& deviceSet : deviceSets)
 | |
|     {
 | |
|         DSPDeviceSourceEngine *deviceSourceEngine =  deviceSet->m_deviceSourceEngine;
 | |
| 
 | |
|         if (deviceSourceEngine)
 | |
|         {
 | |
|             for (int chi = 0; chi < deviceSet->getNumberOfChannels(); chi++)
 | |
|             {
 | |
|                 ChannelAPI *channel = deviceSet->getChannelAt(chi);
 | |
| 
 | |
|                 if ((channel->getURI() == "sdrangel.channel.ilsdemod") && !m_availableChannels.contains(channel) && (m_ilsDemod != channel))
 | |
|                 {
 | |
|                     ObjectPipe *pipe = messagePipes.registerProducerToConsumer(channel, m_ilsDemod, "ilsdemod");
 | |
|                     MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|                     QObject::connect(
 | |
|                         messageQueue,
 | |
|                         &MessageQueue::messageEnqueued,
 | |
|                         this,
 | |
|                         [=](){ this->handleChannelMessageQueue(messageQueue); },
 | |
|                         Qt::QueuedConnection
 | |
|                     );
 | |
|                     QObject::connect(
 | |
|                         pipe,
 | |
|                         &ObjectPipe::toBeDeleted,
 | |
|                         this,
 | |
|                         &ILSDemodGUI::handleMessagePipeToBeDeleted
 | |
|                     );
 | |
|                     m_availableChannels.insert(channel);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::handleChannelAdded(int deviceSetIndex, ChannelAPI *channel)
 | |
| {
 | |
|     qDebug("ILSDemodGUI::handleChannelAdded: deviceSetIndex: %d:%d channel: %s (%p)",
 | |
|         deviceSetIndex, channel->getIndexInDeviceSet(), qPrintable(channel->getURI()), channel);
 | |
|     std::vector<DeviceSet*>& deviceSets = MainCore::instance()->getDeviceSets();
 | |
|     DeviceSet *deviceSet = deviceSets[deviceSetIndex];
 | |
|     DSPDeviceSourceEngine *deviceSourceEngine =  deviceSet->m_deviceSourceEngine;
 | |
| 
 | |
|     if (deviceSourceEngine && (channel->getURI() == "sdrangel.channel.ilsdemod"))
 | |
|     {
 | |
|         if (!m_availableChannels.contains(channel) && (m_ilsDemod != channel))
 | |
|         {
 | |
|             MessagePipes& messagePipes = MainCore::instance()->getMessagePipes();
 | |
|             ObjectPipe *pipe = messagePipes.registerProducerToConsumer(channel, m_ilsDemod, "ilsdemod");
 | |
|             MessageQueue *messageQueue = qobject_cast<MessageQueue*>(pipe->m_element);
 | |
|             QObject::connect(
 | |
|                 messageQueue,
 | |
|                 &MessageQueue::messageEnqueued,
 | |
|                 this,
 | |
|                 [=](){ this->handleChannelMessageQueue(messageQueue); },
 | |
|                 Qt::QueuedConnection
 | |
|             );
 | |
|             QObject::connect(
 | |
|                 pipe,
 | |
|                 &ObjectPipe::toBeDeleted,
 | |
|                 this,
 | |
|                 &ILSDemodGUI::handleMessagePipeToBeDeleted
 | |
|             );
 | |
|             m_availableChannels.insert(channel);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::handleMessagePipeToBeDeleted(int reason, QObject* object)
 | |
| {
 | |
|     if ((reason == 0) && m_availableChannels.contains((ChannelAPI*) object)) // producer (channel)
 | |
|     {
 | |
|         qDebug("ILSDemodGUI::handleMessagePipeToBeDeleted: removing channel at (%p)", object);
 | |
|         m_availableChannels.remove((ChannelAPI*) object);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void ILSDemodGUI::handleChannelMessageQueue(MessageQueue* messageQueue)
 | |
| {
 | |
|     Message* message;
 | |
| 
 | |
|     while ((message = messageQueue->pop()) != nullptr)
 | |
|     {
 | |
|         if (handleMessage(*message)) {
 | |
|             delete message;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 |