mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-30 20:40:20 -04:00 
			
		
		
		
	screen when screen orientation changes. Add DialPopup to allow dials to adjusted with a slider, which is easier on touchscreens. Add TableTapAndHold to support table context menus on touchscreens.
		
			
				
	
	
		
			2245 lines
		
	
	
		
			74 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			2245 lines
		
	
	
		
			74 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| // Copyright (C) 2021 Jon Beniston, M7RCE                                        //
 | |
| // Copyright (C) 2020 Edouard Griffiths, F4EXB                                   //
 | |
| //                                                                               //
 | |
| // 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 <cmath>
 | |
| #include <algorithm>
 | |
| #include <QMessageBox>
 | |
| #include <QLineEdit>
 | |
| #include <QRegExp>
 | |
| #include <QNetworkAccessManager>
 | |
| #include <QNetworkReply>
 | |
| #include <QGraphicsScene>
 | |
| 
 | |
| #include <QtCharts/QChartView>
 | |
| #include <QtCharts/QLineSeries>
 | |
| #include <QtCharts/QDateTimeAxis>
 | |
| #include <QtCharts/QValueAxis>
 | |
| 
 | |
| #include "SWGStarTrackerDisplaySettings.h"
 | |
| #include "SWGStarTrackerDisplayLoSSettings.h"
 | |
| 
 | |
| #include "feature/featureuiset.h"
 | |
| #include "feature/featurewebapiutils.h"
 | |
| #include "gui/basicfeaturesettingsdialog.h"
 | |
| #include "gui/dmsspinbox.h"
 | |
| #include "gui/graphicsviewzoom.h"
 | |
| #include "gui/dialogpositioner.h"
 | |
| #include "mainwindow.h"
 | |
| #include "device/deviceuiset.h"
 | |
| #include "util/units.h"
 | |
| #include "util/astronomy.h"
 | |
| #include "util/interpolation.h"
 | |
| #include "util/png.h"
 | |
| 
 | |
| #include "ui_startrackergui.h"
 | |
| #include "startracker.h"
 | |
| #include "startrackergui.h"
 | |
| #include "startrackerreport.h"
 | |
| #include "startrackersettingsdialog.h"
 | |
| 
 | |
| StarTrackerGUI* StarTrackerGUI::create(PluginAPI* pluginAPI, FeatureUISet *featureUISet, Feature *feature)
 | |
| {
 | |
|     StarTrackerGUI* gui = new StarTrackerGUI(pluginAPI, featureUISet, feature);
 | |
|     return gui;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::destroy()
 | |
| {
 | |
|     qDeleteAll(m_lineOfSightMarkers);
 | |
|     delete this;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::resetToDefaults()
 | |
| {
 | |
|     m_settings.resetToDefaults();
 | |
|     displaySettings();
 | |
|     applySettings(true);
 | |
| }
 | |
| 
 | |
| QByteArray StarTrackerGUI::serialize() const
 | |
| {
 | |
|     return m_settings.serialize();
 | |
| }
 | |
| 
 | |
| bool StarTrackerGUI::deserialize(const QByteArray& data)
 | |
| {
 | |
|     if (m_settings.deserialize(data))
 | |
|     {
 | |
|         m_feature->setWorkspaceIndex(m_settings.m_workspaceIndex);
 | |
|         displaySettings();
 | |
|         applySettings(true);
 | |
|         return true;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         resetToDefaults();
 | |
|         return false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool StarTrackerGUI::handleMessage(const Message& message)
 | |
| {
 | |
|     if (StarTracker::MsgConfigureStarTracker::match(message))
 | |
|     {
 | |
|         qDebug("StarTrackerGUI::handleMessage: StarTracker::MsgConfigureStarTracker");
 | |
|         const StarTracker::MsgConfigureStarTracker& cfg = (StarTracker::MsgConfigureStarTracker&) message;
 | |
| 
 | |
|         if (cfg.getForce()) {
 | |
|             m_settings = cfg.getSettings();
 | |
|         } else {
 | |
|             m_settings.applySettings(cfg.getSettingsKeys(), cfg.getSettings());
 | |
|         }
 | |
| 
 | |
|         blockApplySettings(true);
 | |
|         displaySettings();
 | |
|         blockApplySettings(false);
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (StarTrackerReport::MsgReportAzAl::match(message))
 | |
|     {
 | |
|         StarTrackerReport::MsgReportAzAl& azAl = (StarTrackerReport::MsgReportAzAl&) message;
 | |
|         blockApplySettings(true);
 | |
|         ui->azimuth->setValue(azAl.getAzimuth());
 | |
|         ui->elevation->setValue(azAl.getElevation());
 | |
|         blockApplySettings(false);
 | |
|         return true;
 | |
|     }
 | |
|     else if (StarTrackerReport::MsgReportRADec::match(message))
 | |
|     {
 | |
|         StarTrackerReport::MsgReportRADec& raDec = (StarTrackerReport::MsgReportRADec&) message;
 | |
|         QString target = raDec.getTarget();
 | |
|         if (target == "target")
 | |
|         {
 | |
|             m_settings.m_ra = Units::decimalHoursToHoursMinutesAndSeconds(raDec.getRA());
 | |
|             m_settings.m_dec = Units::decimalDegreesToDegreeMinutesAndSeconds(raDec.getDec());
 | |
|             ui->rightAscension->setText(m_settings.m_ra);
 | |
|             ui->declination->setText(m_settings.m_dec);
 | |
|         }
 | |
|         else if (target == "sun")
 | |
|         {
 | |
|             m_sunRA = raDec.getRA();
 | |
|             m_sunDec = raDec.getDec();
 | |
|         }
 | |
|         else if (target == "moon")
 | |
|         {
 | |
|             m_moonRA = raDec.getRA();
 | |
|             m_moonDec = raDec.getDec();
 | |
|         }
 | |
|         raDecChanged();
 | |
|         return true;
 | |
|     }
 | |
|     else if (StarTrackerReport::MsgReportGalactic::match(message))
 | |
|     {
 | |
|         StarTrackerReport::MsgReportGalactic& galactic = (StarTrackerReport::MsgReportGalactic&) message;
 | |
|         blockApplySettings(true);
 | |
|         ui->galacticLongitude->setValue(galactic.getL());
 | |
|         ui->galacticLatitude->setValue(galactic.getB());
 | |
|         blockApplySettings(false);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgStarTrackerDisplaySettings::match(message))
 | |
|     {
 | |
|         if (m_settings.m_link)
 | |
|         {
 | |
|             MainCore::MsgStarTrackerDisplaySettings& settings = (MainCore::MsgStarTrackerDisplaySettings&) message;
 | |
|             SWGSDRangel::SWGStarTrackerDisplaySettings *swgSettings = settings.getSWGStarTrackerDisplaySettings();
 | |
|             ui->dateTimeSelect->setCurrentText("Custom");
 | |
|             QDateTime dt = QDateTime::fromString(*swgSettings->getDateTime(), Qt::ISODateWithMs);
 | |
|             ui->dateTime->setDateTime(dt);
 | |
|             ui->target->setCurrentText("Custom Az/El");
 | |
|             ui->azimuth->setValue(swgSettings->getAzimuth());
 | |
|             ui->elevation->setValue(swgSettings->getElevation());
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgStarTrackerDisplayLoSSettings::match(message))
 | |
|     {
 | |
|         MainCore::MsgStarTrackerDisplayLoSSettings& settings = (MainCore::MsgStarTrackerDisplayLoSSettings&) message;
 | |
|         SWGSDRangel::SWGStarTrackerDisplayLoSSettings *swgSettings = settings.getSWGStarTrackerDisplayLoSSettings();
 | |
|         bool found = false;
 | |
|         for (int i = 0; i < m_lineOfSightMarkers.size(); i++)
 | |
|         {
 | |
|             if (m_lineOfSightMarkers[i]->m_name == *swgSettings->getName())
 | |
|             {
 | |
|                 if (swgSettings->getD() == 0.0)
 | |
|                 {
 | |
|                     // Delete
 | |
|                     ui->image->scene()->removeItem(m_lineOfSightMarkers[i]->m_text);
 | |
|                     delete m_lineOfSightMarkers[i]->m_text;
 | |
|                     delete m_lineOfSightMarkers[i];
 | |
|                     m_lineOfSightMarkers.removeAt(i);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     // Update
 | |
|                     m_lineOfSightMarkers[i]->m_l = swgSettings->getL();
 | |
|                     m_lineOfSightMarkers[i]->m_b = swgSettings->getB();
 | |
|                     m_lineOfSightMarkers[i]->m_d = swgSettings->getD();
 | |
|                     plotGalacticMarker(m_lineOfSightMarkers[i]);
 | |
|                 }
 | |
|                 found = true;
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
|         if (!found && (swgSettings->getD() != 0.0))
 | |
|         {
 | |
|             // Create new
 | |
|             LoSMarker* marker = new LoSMarker();
 | |
|             marker->m_name = *swgSettings->getName();
 | |
|             marker->m_l = swgSettings->getL();
 | |
|             marker->m_b = swgSettings->getB();
 | |
|             marker->m_d = swgSettings->getD();
 | |
|             marker->m_text = ui->image->scene()->addText(marker->m_name);
 | |
|             m_lineOfSightMarkers.append(marker);
 | |
|             plotGalacticMarker(marker);
 | |
|         }
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::handleInputMessages()
 | |
| {
 | |
|     Message* message;
 | |
| 
 | |
|     while ((message = getInputMessageQueue()->pop()))
 | |
|     {
 | |
|         if (handleMessage(*message)) {
 | |
|             delete message;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::onWidgetRolled(QWidget* widget, bool rollDown)
 | |
| {
 | |
|     (void) widget;
 | |
|     (void) rollDown;
 | |
| 
 | |
|     RollupContents *rollupContents = getRollupContents();
 | |
| 
 | |
|     rollupContents->saveState(m_rollupState);
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| StarTrackerGUI::StarTrackerGUI(PluginAPI* pluginAPI, FeatureUISet *featureUISet, Feature *feature, QWidget* parent) :
 | |
|     FeatureGUI(parent),
 | |
|     ui(new Ui::StarTrackerGUI),
 | |
|     m_pluginAPI(pluginAPI),
 | |
|     m_featureUISet(featureUISet),
 | |
|     m_doApplySettings(true),
 | |
|     m_lastFeatureState(0),
 | |
|     m_azElLineChart(nullptr),
 | |
|     m_azElPolarChart(nullptr),
 | |
|     m_solarFluxChart(nullptr),
 | |
|     m_networkManager(nullptr),
 | |
|     m_solarFlux(0.0),
 | |
|     m_solarFluxesValid(false),
 | |
|     m_images{QImage(":/startracker/startracker/150mhz_ra_dec.png"),
 | |
|         QImage(":/startracker/startracker/150mhz_galactic.png"),
 | |
|         QImage(":/startracker/startracker/408mhz_ra_dec.png"),
 | |
|         QImage(":/startracker/startracker/408mhz_galactic.png"),
 | |
|         QImage(":/startracker/startracker/1420mhz_ra_dec.png"),
 | |
|         QImage(":/startracker/startracker/1420mhz_galactic.png")},
 | |
|     m_milkyWayImages{QPixmap(":/startracker/startracker/milkyway.png"),
 | |
|         QPixmap(":/startracker/startracker/milkywayannotated.png")},
 | |
|     m_sunRA(0.0),
 | |
|     m_sunDec(0.0),
 | |
|     m_moonRA(0.0),
 | |
|     m_moonDec(0.0)
 | |
| {
 | |
|     m_feature = feature;
 | |
|     setAttribute(Qt::WA_DeleteOnClose, true);
 | |
|     m_helpURL = "plugins/feature/startracker/readme.md";
 | |
|     RollupContents *rollupContents = getRollupContents();
 | |
| 	ui->setupUi(rollupContents);
 | |
|     rollupContents->arrangeRollups();
 | |
| 	connect(rollupContents, SIGNAL(widgetRolled(QWidget*,bool)), this, SLOT(onWidgetRolled(QWidget*,bool)));
 | |
| 
 | |
|     m_starTracker = reinterpret_cast<StarTracker*>(feature);
 | |
|     m_starTracker->setMessageQueueToGUI(&m_inputMessageQueue);
 | |
| 
 | |
|     m_settings.setRollupState(&m_rollupState);
 | |
| 
 | |
|     connect(this, SIGNAL(customContextMenuRequested(const QPoint &)), this, SLOT(onMenuDialogCalled(const QPoint &)));
 | |
|     connect(getInputMessageQueue(), SIGNAL(messageEnqueued()), this, SLOT(handleInputMessages()));
 | |
| 
 | |
|     connect(&m_dlm, &HttpDownloadManager::downloadComplete, this, &StarTrackerGUI::downloadFinished);
 | |
| 
 | |
|     connect(&m_statusTimer, SIGNAL(timeout()), this, SLOT(updateStatus()));
 | |
|     m_statusTimer.start(1000);
 | |
| 
 | |
|     connect(ui->azimuth, SIGNAL(valueChanged(double)), this, SLOT(on_azimuth_valueChanged(double)));
 | |
|     ui->azimuth->setRange(0, 360.0);
 | |
|     ui->elevation->setRange(-90.0, 90.0);
 | |
|     ui->galacticLongitude->setRange(0, 360.0);
 | |
|     ui->galacticLatitude->setRange(-90.0, 90.0);
 | |
|     ui->galacticLatitude->setText("");
 | |
|     ui->galacticLongitude->setText("");
 | |
| 
 | |
|     // Intialise chart
 | |
|     m_chart.legend()->hide();
 | |
|     ui->chart->setChart(&m_chart);
 | |
|     ui->chart->setRenderHint(QPainter::Antialiasing);
 | |
|     m_chart.addAxis(&m_chartXAxis, Qt::AlignBottom);
 | |
|     m_chart.addAxis(&m_chartYAxis, Qt::AlignLeft);
 | |
|     m_chart.layout()->setContentsMargins(0, 0, 0, 0);
 | |
|     m_chart.setMargins(QMargins(1, 1, 1, 1));
 | |
| 
 | |
|     // Create axes that are static
 | |
| 
 | |
|     m_skyTempGalacticLXAxis.setTitleText(QString("Galactic longitude (%1)").arg(QChar(0xb0)));
 | |
|     m_skyTempGalacticLXAxis.setMin(0);
 | |
|     m_skyTempGalacticLXAxis.setMax(360);
 | |
|     m_skyTempGalacticLXAxis.append("180", 0);
 | |
|     m_skyTempGalacticLXAxis.append("90", 90);
 | |
|     m_skyTempGalacticLXAxis.append("0/360", 180);
 | |
|     m_skyTempGalacticLXAxis.append("270", 270);
 | |
|     //m_skyTempGalacticLXAxis.append("180", 360); // Note - labels need to be unique, so can't have 180 at start and end
 | |
|     m_skyTempGalacticLXAxis.setLabelsPosition(QCategoryAxis::AxisLabelsPositionOnValue);
 | |
|     m_skyTempGalacticLXAxis.setGridLineVisible(false);
 | |
| 
 | |
|     m_skyTempRAXAxis.setTitleText(QString("Right ascension (hours)"));
 | |
|     m_skyTempRAXAxis.setMin(0);
 | |
|     m_skyTempRAXAxis.setMax(24);
 | |
|     m_skyTempRAXAxis.append("12", 0);
 | |
|     m_skyTempRAXAxis.append("9", 3);
 | |
|     m_skyTempRAXAxis.append("6", 6);
 | |
|     m_skyTempRAXAxis.append("3", 9);
 | |
|     m_skyTempRAXAxis.append("0", 12);
 | |
|     m_skyTempRAXAxis.append("21", 15);
 | |
|     m_skyTempRAXAxis.append("18", 18);
 | |
|     m_skyTempRAXAxis.append("15", 21);
 | |
|     //m_skyTempRAXAxis.append("12", 24); // Note - labels need to be unique, so can't have 12 at start and end
 | |
|     m_skyTempRAXAxis.setLabelsPosition(QCategoryAxis::AxisLabelsPositionOnValue);
 | |
|     m_skyTempRAXAxis.setGridLineVisible(false);
 | |
| 
 | |
|     m_skyTempYAxis.setGridLineVisible(false);
 | |
|     m_skyTempYAxis.setRange(-90.0, 90.0);
 | |
|     m_skyTempYAxis.setGridLineVisible(false);
 | |
| 
 | |
|     ui->dateTime->setDateTime(QDateTime::currentDateTime());
 | |
|     displaySettings();
 | |
|     applySettings(true);
 | |
|     makeUIConnections();
 | |
| 
 | |
|     // Populate subchart menu
 | |
|     on_chartSelect_currentIndexChanged(0);
 | |
| 
 | |
|     connect(&m_chart, SIGNAL(plotAreaChanged(QRectF)), this, SLOT(plotAreaChanged(QRectF)));
 | |
| 
 | |
|     // Use My Position from preferences, if none set
 | |
|     if ((m_settings.m_latitude == 0.0) && (m_settings.m_longitude == 0.0)) {
 | |
|         on_useMyPosition_clicked();
 | |
|     }
 | |
| 
 | |
| /*
 | |
|     printf("saemundsson=[");
 | |
|     for (int i = 0; i <= 90; i+= 5)
 | |
|         printf("%f ", Astronomy::refractionSaemundsson(i, m_settings.m_pressure, m_settings.m_temperature));
 | |
|     printf("];\n");
 | |
|     printf("palRadio=[");
 | |
|     for (int i = 0; i <= 90; i+= 5)
 | |
|         printf("%f ", Astronomy::refractionPAL(i, m_settings.m_pressure, m_settings.m_temperature, m_settings.m_humidity,
 | |
|                                                 100000000, m_settings.m_latitude, m_settings.m_heightAboveSeaLevel,
 | |
|                                                 m_settings.m_temperatureLapseRate));
 | |
|     printf("];\n");
 | |
|     printf("palLight=[");
 | |
|     for (int i = 0; i <= 90; i+= 5)
 | |
|         printf("%f ",Astronomy::refractionPAL(i, m_settings.m_pressure, m_settings.m_temperature, m_settings.m_humidity,
 | |
|                                                 7.5e14, m_settings.m_latitude, m_settings.m_heightAboveSeaLevel,
 | |
|                                                 m_settings.m_temperatureLapseRate));
 | |
|     printf("];\n");
 | |
| */
 | |
| 
 | |
|     m_networkManager = new QNetworkAccessManager();
 | |
|     QObject::connect(
 | |
|         m_networkManager,
 | |
|         &QNetworkAccessManager::finished,
 | |
|         this,
 | |
|         &StarTrackerGUI::networkManagerFinished
 | |
|     );
 | |
| 
 | |
|     readSolarFlux();
 | |
|     connect(&m_solarFluxTimer, SIGNAL(timeout()), this, SLOT(autoUpdateSolarFlux()));
 | |
|     m_solarFluxTimer.start(1000*60*60*24); // Update every 24hours
 | |
|     autoUpdateSolarFlux();
 | |
| 
 | |
|     createGalacticLineOfSightScene();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| StarTrackerGUI::~StarTrackerGUI()
 | |
| {
 | |
|     QObject::disconnect(
 | |
|         m_networkManager,
 | |
|         &QNetworkAccessManager::finished,
 | |
|         this,
 | |
|         &StarTrackerGUI::networkManagerFinished
 | |
|     );
 | |
|     delete m_networkManager;
 | |
|     delete ui;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::setWorkspaceIndex(int index)
 | |
| {
 | |
|     m_settings.m_workspaceIndex = index;
 | |
|     m_settingsKeys.append("workspaceIndex");
 | |
|     m_feature->setWorkspaceIndex(index);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::blockApplySettings(bool block)
 | |
| {
 | |
|     m_doApplySettings = !block;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::displaySettings()
 | |
| {
 | |
|     setTitleColor(m_settings.m_rgbColor);
 | |
|     setWindowTitle(m_settings.m_title);
 | |
|     setTitle(m_settings.m_title);
 | |
|     blockApplySettings(true);
 | |
|     ui->darkTheme->setChecked(m_settings.m_chartsDarkTheme);
 | |
| 
 | |
|     if (m_solarFluxChart) {
 | |
|         m_solarFluxChart->setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     }
 | |
| 
 | |
|     m_chart.setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     ui->drawSun->setChecked(m_settings.m_drawSunOnSkyTempChart);
 | |
|     ui->drawMoon->setChecked(m_settings.m_drawMoonOnSkyTempChart);
 | |
|     ui->link->setChecked(m_settings.m_link);
 | |
|     ui->latitude->setValue(m_settings.m_latitude);
 | |
|     ui->longitude->setValue(m_settings.m_longitude);
 | |
|     ui->target->setCurrentIndex(ui->target->findText(m_settings.m_target));
 | |
|     ui->azimuth->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|     ui->elevation->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|     ui->galacticLatitude->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|     ui->galacticLongitude->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|     ui->azimuthOffset->setValue(m_settings.m_azOffset);
 | |
|     ui->elevationOffset->setValue(m_settings.m_elOffset);
 | |
| 
 | |
|     if (m_settings.m_target == "Custom RA/Dec")
 | |
|     {
 | |
|         ui->rightAscension->setText(m_settings.m_ra);
 | |
|         ui->declination->setText(m_settings.m_dec);
 | |
|     }
 | |
|     else if (m_settings.m_target == "Custom Az/El")
 | |
|     {
 | |
|         ui->azimuth->setValue(m_settings.m_az);
 | |
|         ui->elevation->setValue(m_settings.m_el);
 | |
|     }
 | |
|     else if ((m_settings.m_target == "Custom l/b")
 | |
|         || (m_settings.m_target == "S7")
 | |
|         || (m_settings.m_target == "S8")
 | |
|         || (m_settings.m_target == "S9")
 | |
|     )
 | |
|     {
 | |
|         ui->galacticLatitude->setValue(m_settings.m_b);
 | |
|         ui->galacticLongitude->setValue(m_settings.m_l);
 | |
|     }
 | |
|     if (m_settings.m_dateTime == "")
 | |
|     {
 | |
|         ui->dateTimeSelect->setCurrentIndex(0);
 | |
|         ui->dateTime->setVisible(false);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ui->dateTime->setDateTime(QDateTime::fromString(m_settings.m_dateTime, Qt::ISODateWithMs));
 | |
|         ui->dateTime->setVisible(true);
 | |
|         ui->dateTimeSelect->setCurrentIndex(1);
 | |
|     }
 | |
| 
 | |
|     if ((m_settings.m_solarFluxData != StarTrackerSettings::DRAO_2800) && !m_solarFluxesValid) {
 | |
|         autoUpdateSolarFlux();
 | |
|     }
 | |
| 
 | |
|     ui->frequency->setValue(m_settings.m_frequency/1000000.0);
 | |
|     ui->beamwidth->setValue(m_settings.m_beamwidth);
 | |
|     updateForTarget();
 | |
|     getRollupContents()->restoreState(m_rollupState);
 | |
|     plotChart();
 | |
|     blockApplySettings(false);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::onMenuDialogCalled(const QPoint &p)
 | |
| {
 | |
|     if (m_contextMenuType == ContextMenuChannelSettings)
 | |
|     {
 | |
|         BasicFeatureSettingsDialog dialog(this);
 | |
|         dialog.setTitle(m_settings.m_title);
 | |
|         dialog.setUseReverseAPI(m_settings.m_useReverseAPI);
 | |
|         dialog.setReverseAPIAddress(m_settings.m_reverseAPIAddress);
 | |
|         dialog.setReverseAPIPort(m_settings.m_reverseAPIPort);
 | |
|         dialog.setReverseAPIFeatureSetIndex(m_settings.m_reverseAPIFeatureSetIndex);
 | |
|         dialog.setReverseAPIFeatureIndex(m_settings.m_reverseAPIFeatureIndex);
 | |
|         dialog.setDefaultTitle(m_displayedName);
 | |
| 
 | |
|         dialog.move(p);
 | |
|         new DialogPositioner(&dialog, false);
 | |
|         dialog.exec();
 | |
| 
 | |
|         m_settings.m_title = dialog.getTitle();
 | |
|         m_settings.m_useReverseAPI = dialog.useReverseAPI();
 | |
|         m_settings.m_reverseAPIAddress = dialog.getReverseAPIAddress();
 | |
|         m_settings.m_reverseAPIPort = dialog.getReverseAPIPort();
 | |
|         m_settings.m_reverseAPIFeatureSetIndex = dialog.getReverseAPIFeatureSetIndex();
 | |
|         m_settings.m_reverseAPIFeatureIndex = dialog.getReverseAPIFeatureIndex();
 | |
| 
 | |
|         setTitle(m_settings.m_title);
 | |
|         setTitleColor(m_settings.m_rgbColor);
 | |
| 
 | |
|         m_settingsKeys.append("title");
 | |
|         m_settingsKeys.append("rgbColor");
 | |
|         m_settingsKeys.append("useReverseAPI");
 | |
|         m_settingsKeys.append("reverseAPIAddress");
 | |
|         m_settingsKeys.append("reverseAPIPort");
 | |
|         m_settingsKeys.append("reverseAPIFeatureSetIndex");
 | |
|         m_settingsKeys.append("reverseAPIFeatureIndex");
 | |
| 
 | |
|         applySettings();
 | |
|     }
 | |
| 
 | |
|     resetContextMenuType();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_startStop_toggled(bool checked)
 | |
| {
 | |
|     if (m_doApplySettings)
 | |
|     {
 | |
|         StarTracker::MsgStartStop *message = StarTracker::MsgStartStop::create(checked);
 | |
|         m_starTracker->getInputMessageQueue()->push(message);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_latitude_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_latitude = value;
 | |
|     m_settingsKeys.append("latitude");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_longitude_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_longitude = value;
 | |
|     m_settingsKeys.append("longitude");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_rightAscension_editingFinished()
 | |
| {
 | |
|     m_settings.m_ra = ui->rightAscension->text();
 | |
|     m_settingsKeys.append("ra");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_declination_editingFinished()
 | |
| {
 | |
|     m_settings.m_dec = ui->declination->text();
 | |
|     m_settingsKeys.append("dec");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_azimuth_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_az = value;
 | |
|     m_settingsKeys.append("az");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_elevation_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_el = value;
 | |
|     m_settingsKeys.append("el");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_azimuthOffset_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_azOffset = value;
 | |
|     m_settingsKeys.append("azOffset");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_elevationOffset_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_elOffset = value;
 | |
|     m_settingsKeys.append("elOffset");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_galacticLatitude_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_b = value;
 | |
|     m_settingsKeys.append("b");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_galacticLongitude_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_l = value;
 | |
|     m_settingsKeys.append("l");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::updateForTarget()
 | |
| {
 | |
|     if (m_settings.m_target == "Sun")
 | |
|     {
 | |
|         ui->rightAscension->setReadOnly(true);
 | |
|         ui->declination->setReadOnly(true);
 | |
|         ui->rightAscension->setText("");
 | |
|         ui->declination->setText("");
 | |
|     }
 | |
|     else if (m_settings.m_target == "Moon")
 | |
|     {
 | |
|         ui->rightAscension->setReadOnly(true);
 | |
|         ui->declination->setReadOnly(true);
 | |
|         ui->rightAscension->setText("");
 | |
|         ui->declination->setText("");
 | |
|     }
 | |
|     else if (m_settings.m_target == "Custom RA/Dec")
 | |
|     {
 | |
|         ui->rightAscension->setReadOnly(false);
 | |
|         ui->declination->setReadOnly(false);
 | |
|     }
 | |
|     else if (m_settings.m_target == "S7")
 | |
|     {
 | |
|         ui->galacticLatitude->setValue(-1.0);
 | |
|         ui->galacticLongitude->setValue(132.0);
 | |
|     }
 | |
|     else if (m_settings.m_target == "S8")
 | |
|     {
 | |
|         ui->galacticLatitude->setValue(-15.0);
 | |
|         ui->galacticLongitude->setValue(207.0);
 | |
|     }
 | |
|     else if (m_settings.m_target == "S9")
 | |
|     {
 | |
|         ui->galacticLatitude->setValue(-4.0);
 | |
|         ui->galacticLongitude->setValue(356.0);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ui->rightAscension->setReadOnly(true);
 | |
|         ui->declination->setReadOnly(true);
 | |
|         if (m_settings.m_target == "PSR B0329+54")
 | |
|         {
 | |
|             ui->rightAscension->setText("03h32m59.35s");
 | |
|             ui->declination->setText(QString("54%0134'45.05\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "PSR B0833-45")
 | |
|         {
 | |
|             ui->rightAscension->setText("08h35m20.66s");
 | |
|             ui->declination->setText(QString("-45%0110'35.15\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "Sagittarius A")
 | |
|         {
 | |
|             ui->rightAscension->setText("17h45m40.04s");
 | |
|             ui->declination->setText(QString("-29%0100'28.17\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "Cassiopeia A")
 | |
|         {
 | |
|             ui->rightAscension->setText("23h23m24s");
 | |
|             ui->declination->setText(QString("58%0148'54\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "Cygnus A")
 | |
|         {
 | |
|             ui->rightAscension->setText("19h59m28.36s");
 | |
|             ui->declination->setText(QString("40%0144'02.1\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "Taurus A (M1)")
 | |
|         {
 | |
|             ui->rightAscension->setText("05h34m31.94s");
 | |
|             ui->declination->setText(QString("22%0100'52.2\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         else if (m_settings.m_target == "Virgo A (M87)")
 | |
|         {
 | |
|             ui->rightAscension->setText("12h30m49.42s");
 | |
|             ui->declination->setText(QString("12%0123'28.04\"").arg(QChar(0xb0)));
 | |
|         }
 | |
|         on_rightAscension_editingFinished();
 | |
|         on_declination_editingFinished();
 | |
|     }
 | |
|     if (m_settings.m_target != "Custom Az/El")
 | |
|     {
 | |
|         ui->azimuth->setReadOnly(true);
 | |
|         ui->elevation->setReadOnly(true);
 | |
|         // Clear as no longer valid when target has changed
 | |
|         ui->azimuth->setText("");
 | |
|         ui->elevation->setText("");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ui->rightAscension->setReadOnly(true);
 | |
|         ui->declination->setReadOnly(true);
 | |
|         ui->azimuth->setReadOnly(false);
 | |
|         ui->elevation->setReadOnly(false);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_target_currentTextChanged(const QString &text)
 | |
| {
 | |
|     m_settings.m_target = text;
 | |
|     m_settingsKeys.append("target");
 | |
|     applySettings();
 | |
|     updateForTarget();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::updateLST()
 | |
| {
 | |
|     QDateTime dt;
 | |
| 
 | |
|     if (m_settings.m_dateTime.isEmpty()) {
 | |
|         dt = QDateTime::currentDateTime();
 | |
|     } else {
 | |
|         dt = QDateTime::fromString(m_settings.m_dateTime, Qt::ISODateWithMs);
 | |
|     }
 | |
| 
 | |
|     double lst = Astronomy::localSiderealTime(dt, m_settings.m_longitude);
 | |
|     ui->lst->setText(Units::decimalHoursToHoursMinutesAndSeconds(lst/15.0, 0));
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::updateStatus()
 | |
| {
 | |
|     int state = m_starTracker->getState();
 | |
| 
 | |
|     if (m_lastFeatureState != state)
 | |
|     {
 | |
|         // We set checked state of start/stop button, in case it was changed via API
 | |
|         bool oldState;
 | |
|         switch (state)
 | |
|         {
 | |
|             case Feature::StNotStarted:
 | |
|                 ui->startStop->setStyleSheet("QToolButton { background:rgb(79,79,79); }");
 | |
|                 break;
 | |
|             case Feature::StIdle:
 | |
|                 oldState = ui->startStop->blockSignals(true);
 | |
|                 ui->startStop->setChecked(false);
 | |
|                 ui->startStop->blockSignals(oldState);
 | |
|                 ui->startStop->setStyleSheet("QToolButton { background-color : blue; }");
 | |
|                 break;
 | |
|             case Feature::StRunning:
 | |
|                 oldState = ui->startStop->blockSignals(true);
 | |
|                 ui->startStop->setChecked(true);
 | |
|                 ui->startStop->blockSignals(oldState);
 | |
|                 ui->startStop->setStyleSheet("QToolButton { background-color : green; }");
 | |
|                 break;
 | |
|             case Feature::StError:
 | |
|                 ui->startStop->setStyleSheet("QToolButton { background-color : red; }");
 | |
|                 QMessageBox::information(this, tr("Message"), m_starTracker->getErrorMessage());
 | |
|                 break;
 | |
|             default:
 | |
|                 break;
 | |
|         }
 | |
| 
 | |
|         m_lastFeatureState = state;
 | |
|     }
 | |
| 
 | |
|     updateLST();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::applySettings(bool force)
 | |
| {
 | |
|     if (m_doApplySettings)
 | |
|     {
 | |
|         StarTracker::MsgConfigureStarTracker* message = StarTracker::MsgConfigureStarTracker::create(m_settings, m_settingsKeys, force);
 | |
|         m_starTracker->getInputMessageQueue()->push(message);
 | |
|     }
 | |
| 
 | |
|     m_settingsKeys.clear();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_link_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_link = checked;
 | |
|     m_settingsKeys.append("link");
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_useMyPosition_clicked(bool checked)
 | |
| {
 | |
|     (void) checked;
 | |
|     double stationLatitude = MainCore::instance()->getSettings().getLatitude();
 | |
|     double stationLongitude = MainCore::instance()->getSettings().getLongitude();
 | |
|     double stationAltitude = MainCore::instance()->getSettings().getAltitude();
 | |
| 
 | |
|     ui->latitude->setValue(stationLatitude);
 | |
|     ui->longitude->setValue(stationLongitude);
 | |
|     m_settings.m_heightAboveSeaLevel = stationAltitude;
 | |
|     m_settingsKeys.append("heightAboveSeaLevel");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| // Show settings dialog
 | |
| void StarTrackerGUI::on_displaySettings_clicked()
 | |
| {
 | |
|     StarTrackerSettingsDialog dialog(&m_settings, m_settingsKeys);
 | |
| 
 | |
|     if (dialog.exec() == QDialog::Accepted)
 | |
|     {
 | |
|         applySettings();
 | |
|         ui->elevation->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|         ui->azimuth->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|         ui->galacticLatitude->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|         ui->galacticLongitude->setUnits((DMSSpinBox::DisplayUnits)m_settings.m_azElUnits);
 | |
|         displaySolarFlux();
 | |
| 
 | |
|         if (ui->chartSelect->currentIndex() == 1) {
 | |
|             plotChart();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_dateTimeSelect_currentTextChanged(const QString &text)
 | |
| {
 | |
|     if (text == "Now")
 | |
|     {
 | |
|         m_settings.m_dateTime = "";
 | |
|         ui->dateTime->setVisible(false);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         m_settings.m_dateTime = ui->dateTime->dateTime().toString(Qt::ISODateWithMs);
 | |
|         ui->dateTime->setVisible(true);
 | |
|     }
 | |
| 
 | |
|     m_settingsKeys.append("dateTime");
 | |
|     applySettings();
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_dateTime_dateTimeChanged(const QDateTime &datetime)
 | |
| {
 | |
|     (void) datetime;
 | |
| 
 | |
|     if (ui->dateTimeSelect->currentIndex() == 1)
 | |
|     {
 | |
|         m_settings.m_dateTime = ui->dateTime->dateTime().toString(Qt::ISODateWithMs);
 | |
|         m_settingsKeys.append("dateTime");
 | |
|         applySettings();
 | |
|         plotChart();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotChart()
 | |
| {
 | |
|     if (ui->chartSelect->currentIndex() == 0)
 | |
|     {
 | |
|         if (ui->chartSubSelect->currentIndex() == 0) {
 | |
|             plotElevationLineChart();
 | |
|         } else {
 | |
|             plotElevationPolarChart();
 | |
|         }
 | |
|     }
 | |
|     else if (ui->chartSelect->currentIndex() == 1)
 | |
|     {
 | |
|         plotSolarFluxChart();
 | |
|     }
 | |
|     else if (ui->chartSelect->currentIndex() == 2)
 | |
|     {
 | |
|         plotSkyTemperatureChart();
 | |
|     }
 | |
|     else if (ui->chartSelect->currentIndex() == 3)
 | |
|     {
 | |
|         plotGalacticLineOfSight();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::raDecChanged()
 | |
| {
 | |
|     if (ui->chartSelect->currentIndex() == 2) {
 | |
|         plotSkyTemperatureChart();
 | |
|     } else if (ui->chartSelect->currentIndex() == 3) {
 | |
|         plotGalacticLineOfSight();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_frequency_valueChanged(int value)
 | |
| {
 | |
|     m_settings.m_frequency = value*1000000.0;
 | |
|     m_settingsKeys.append("frequency");
 | |
|     applySettings();
 | |
| 
 | |
|     if (ui->chartSelect->currentIndex() != 0)
 | |
|     {
 | |
|         updateChartSubSelect();
 | |
|         plotChart();
 | |
|     }
 | |
| 
 | |
|     displaySolarFlux();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_beamwidth_valueChanged(double value)
 | |
| {
 | |
|     m_settings.m_beamwidth = value;
 | |
|     m_settingsKeys.append("beamwidth");
 | |
|     applySettings();
 | |
|     updateChartSubSelect();
 | |
| 
 | |
|     if (ui->chartSelect->currentIndex() == 2) {
 | |
|         plotChart();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotSolarFluxChart()
 | |
| {
 | |
|     ui->chart->setVisible(true);
 | |
|     ui->image->setVisible(false);
 | |
|     ui->drawSun->setVisible(false);
 | |
|     ui->drawMoon->setVisible(false);
 | |
|     ui->darkTheme->setVisible(true);
 | |
|     ui->zoomIn->setVisible(false);
 | |
|     ui->zoomOut->setVisible(false);
 | |
|     ui->addAnimationFrame->setVisible(false);
 | |
|     ui->clearAnimation->setVisible(false);
 | |
|     ui->saveAnimation->setVisible(false);
 | |
| 
 | |
|     QChart *oldChart = m_solarFluxChart;
 | |
| 
 | |
|     m_solarFluxChart = new QChart();
 | |
| 
 | |
|     if (m_solarFluxesValid)
 | |
|     {
 | |
| 
 | |
|         m_solarFluxChart->setTitle("");
 | |
|         m_solarFluxChart->legend()->hide();
 | |
|         m_solarFluxChart->layout()->setContentsMargins(0, 0, 0, 0);
 | |
|         m_solarFluxChart->setMargins(QMargins(1, 1, 1, 1));
 | |
|         m_solarFluxChart->setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
| 
 | |
|         double maxValue = -std::numeric_limits<double>::infinity();
 | |
|         double minValue = std::numeric_limits<double>::infinity();
 | |
|         QLineSeries *series = new QLineSeries();
 | |
|         for (int i = 0; i < 8; i++)
 | |
|         {
 | |
|             double value = convertSolarFluxUnits(m_solarFluxes[i]);
 | |
|             series->append(m_solarFluxFrequencies[i], value);
 | |
|             maxValue = std::max(value, maxValue);
 | |
|             minValue = std::min(value, minValue);
 | |
|         }
 | |
|         series->setPointLabelsVisible(true);
 | |
|         series->setPointLabelsFormat("@yPoint");
 | |
|         series->setPointLabelsClipping(false);
 | |
|         m_solarFluxChart->addSeries(series);
 | |
| 
 | |
|         QLogValueAxis *chartSolarFluxXAxis = new QLogValueAxis();
 | |
|         QValueAxis *chartSolarFluxYAxis = new QValueAxis();
 | |
| 
 | |
|         chartSolarFluxXAxis->setTitleText(QString("Frequency (MHz)"));
 | |
|         chartSolarFluxXAxis->setMinorTickCount(-1);
 | |
| 
 | |
|         chartSolarFluxYAxis->setTitleText(QString("Solar flux density (%1)").arg(solarFluxUnit()));
 | |
|         chartSolarFluxYAxis->setMinorTickCount(-1);
 | |
|         if (m_settings.m_solarFluxUnits == StarTrackerSettings::SFU)
 | |
|         {
 | |
|             chartSolarFluxYAxis->setLabelFormat("%d");
 | |
|             chartSolarFluxYAxis->setRange(0.0, ((((int)maxValue)+99)/100)*100);
 | |
|         }
 | |
|         else if (m_settings.m_solarFluxUnits == StarTrackerSettings::JANSKY)
 | |
|         {
 | |
|             chartSolarFluxYAxis->setLabelFormat("%.2g");
 | |
|             chartSolarFluxYAxis->setRange(0, ((((int)maxValue)+999999)/100000)*100000);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             chartSolarFluxYAxis->setLabelFormat("%.2g");
 | |
|             // Bug in QtCharts for values < ~1e-12  https://bugreports.qt.io/browse/QTBUG-95304
 | |
|             // Set range to 0-1 here, then real range after axis have been attached
 | |
|             chartSolarFluxYAxis->setRange(0.0, 1.0);
 | |
|         }
 | |
| 
 | |
|         m_solarFluxChart->addAxis(chartSolarFluxXAxis, Qt::AlignBottom);
 | |
|         m_solarFluxChart->addAxis(chartSolarFluxYAxis, Qt::AlignLeft);
 | |
|         series->attachAxis(chartSolarFluxXAxis);
 | |
|         series->attachAxis(chartSolarFluxYAxis);
 | |
| 
 | |
|         if (m_settings.m_solarFluxUnits == StarTrackerSettings::WATTS_M_HZ) {
 | |
|             chartSolarFluxYAxis->setRange(minValue, maxValue);
 | |
|         }
 | |
| 
 | |
|     }
 | |
|     else
 | |
|         m_solarFluxChart->setTitle("Press download Solar flux density data to view");
 | |
| 
 | |
|     ui->chart->setChart(m_solarFluxChart);
 | |
| 
 | |
|     delete oldChart;
 | |
| }
 | |
| 
 | |
| QList<QLineSeries*> StarTrackerGUI::createDriftScan(bool galactic)
 | |
| {
 | |
|     QList<QLineSeries *>list;
 | |
|     QLineSeries *series = new QLineSeries();
 | |
|     list.append(series);
 | |
| 
 | |
|     QDateTime dt;
 | |
| 
 | |
|     // Get date and time to calculate position at
 | |
|     if (m_settings.m_dateTime == "") {
 | |
|         dt = QDateTime::currentDateTime();
 | |
|     } else {
 | |
|         dt = QDateTime::fromString(m_settings.m_dateTime, Qt::ISODateWithMs);
 | |
|     }
 | |
| 
 | |
|     // Create a list of RA/Dec points of drift scan path
 | |
|     AzAlt aa;
 | |
|     aa.alt = m_settings.m_el;
 | |
|     aa.az = m_settings.m_az;
 | |
|     double prevX;
 | |
|     // Plot every 30min over a day
 | |
|     for (int i = 0; i <= 24*2; i++)
 | |
|     {
 | |
|         dt = dt.addSecs(30*60);
 | |
|         RADec rd = Astronomy::azAltToRaDec(aa, m_settings.m_latitude, m_settings.m_longitude, dt);
 | |
|         double x, y;
 | |
|         mapRaDec(rd.ra, rd.dec, galactic, x, y);
 | |
|         if (i == 0)
 | |
|         {
 | |
|             series->append(x, y);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             // Check for crossing edge of chart
 | |
|             if (galactic)
 | |
|             {
 | |
|                 if (((prevX < 90.0) && (x > 270.0)) || ((prevX > 270.0) && (x < 90.0)))
 | |
|                 {
 | |
|                     // Start new series, so we don't have lines crossing across the chart
 | |
|                     series = new QLineSeries();
 | |
|                     list.append(series);
 | |
|                 }
 | |
|             }
 | |
|             series->append(x, y);
 | |
|         }
 | |
|         prevX = x;
 | |
|     }
 | |
| 
 | |
|     return list;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::mapRaDec(double ra, double dec, bool galactic, double& x, double& y)
 | |
| {
 | |
|     if (galactic)
 | |
|     {
 | |
|         // Convert to category coordinates
 | |
|         double l, b;
 | |
|         Astronomy::equatorialToGalactic(ra, dec, l, b);
 | |
|         // Map to linear axis
 | |
|         double lAxis;
 | |
|         if (l < 180.0) {
 | |
|             lAxis = 180.0 - l;
 | |
|         } else {
 | |
|             lAxis = 360.0 - l + 180.0;
 | |
|         }
 | |
| 
 | |
|         x = lAxis;
 | |
|         y = b;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Map to category axis
 | |
|         double raAxis;
 | |
|         if (ra <= 12.0) {
 | |
|             raAxis = 12.0 - ra;
 | |
|         } else {
 | |
|             raAxis = 24 - ra + 12;
 | |
|         }
 | |
| 
 | |
|         x = raAxis;
 | |
|         y = dec;
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Is there a way to get this from the theme? Got these values from the source
 | |
| QColor StarTrackerGUI::getSeriesColor(int series)
 | |
| {
 | |
|     if (m_settings.m_chartsDarkTheme)
 | |
|     {
 | |
|         if (series == 0) {
 | |
|             return QColor(0x38ad6b);
 | |
|         } else if (series == 1) {
 | |
|             return QColor(0x3c84a7);
 | |
|         } else {
 | |
|             return QColor(0xeb8817);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         if (series == 0) {
 | |
|             return QColor(0x209fdf);
 | |
|         } else if (series == 1) {
 | |
|             return QColor(0x99ca53);
 | |
|         } else {
 | |
|             return QColor(0xf6a625);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::createGalacticLineOfSightScene()
 | |
| {
 | |
|     m_zoom = new GraphicsViewZoom(ui->image); // Deleted automatically when view is deleted
 | |
| 
 | |
|     QGraphicsScene *scene = new QGraphicsScene(ui->image);
 | |
|     scene->setBackgroundBrush(QBrush(Qt::black));
 | |
| 
 | |
|     // Milkyway images
 | |
|     for (int i = 0; i < m_milkyWayImages.size(); i++)
 | |
|     {
 | |
|         m_milkyWayItems.append(scene->addPixmap(m_milkyWayImages[i]));
 | |
|         m_milkyWayItems[i]->setPos(0, 0);
 | |
|         m_milkyWayItems[i]->setVisible(i == 0);
 | |
|     }
 | |
| 
 | |
|     // Line of sight
 | |
|     QPen pen(QColor(255, 0, 0), 4, Qt::SolidLine);
 | |
|     m_lineOfSight = scene->addLine(511, 708, 511, 708, pen);
 | |
| 
 | |
|     ui->image->setScene(scene);
 | |
|     ui->image->show();
 | |
| 
 | |
|     ui->image->setDragMode(QGraphicsView::ScrollHandDrag);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotGalacticLineOfSight()
 | |
| {
 | |
|     if (!ui->image->isVisible())
 | |
|     {
 | |
|         // Start zoomed out
 | |
|         ui->image->fitInView(m_milkyWayItems[0], Qt::KeepAspectRatio);
 | |
|     }
 | |
| 
 | |
|     // Draw top-down image of Milky Way
 | |
|     ui->chart->setVisible(false);
 | |
|     ui->image->setVisible(true);
 | |
|     ui->drawSun->setVisible(false);
 | |
|     ui->drawMoon->setVisible(false);
 | |
|     ui->darkTheme->setVisible(false);
 | |
|     ui->zoomIn->setVisible(true);
 | |
|     ui->zoomOut->setVisible(true);
 | |
|     ui->addAnimationFrame->setVisible(true);
 | |
|     ui->clearAnimation->setVisible(true);
 | |
|     ui->saveAnimation->setVisible(true);
 | |
| 
 | |
|     // Select which Milky Way image to show
 | |
|     int imageIdx = ui->chartSubSelect->currentIndex();
 | |
|     for (int i = 0; i < m_milkyWayItems.size(); i++) {
 | |
|         m_milkyWayItems[i]->setVisible(i == imageIdx);
 | |
|     }
 | |
| 
 | |
|     // Calculate Galactic longitude we're observing
 | |
|     float ra = Astronomy::raToDecimal(m_settings.m_ra);
 | |
|     float dec = Astronomy::decToDecimal(m_settings.m_dec);
 | |
|     double l, b;
 | |
|     Astronomy::equatorialToGalactic(ra, dec, l, b);
 | |
| 
 | |
|     //l = ui->azimuth->value(); // For testing, just use azimuth
 | |
| 
 | |
|     // Calculate length of line, as Sun is not at centre
 | |
|     // we assume end point lies on an ellipse, with Sun at foci
 | |
|     // See https://en.wikipedia.org/wiki/Ellipse Polar form relative to focus
 | |
|     QPointF sun(511, 708); // Location of Sun on Milky Way image
 | |
|     float a = sun.x() - 112; // semi-major axis
 | |
|     float c = sun.y() - sun.x(); // linear eccentricity
 | |
|     float e = c / a; // eccentricity
 | |
|     float r = a * (1.0f - e*e) / (1.0f - e * cos(Units::degreesToRadians(l)));
 | |
| 
 | |
|     // Draw line from Sun along observation galactic longitude
 | |
|     QTransform rotation = QTransform().translate(sun.x(), -sun.y()).rotate(l).translate(-sun.x(), sun.y());  // Flip Y
 | |
|     QPointF point = rotation.map(QPointF(511, -sun.y() + r));
 | |
|     m_lineOfSight->setLine(sun.x(), sun.y(), point.x(), -point.y());
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotGalacticMarker(LoSMarker* marker)
 | |
| {
 | |
|     QPointF sun(511, 708); // Location of Sun on Milky Way image
 | |
|     double pixelsPerKPC = 564.0/22.995;  // 75,000ly = 23kpc
 | |
|     QTransform rotation = QTransform().translate(sun.x(), -sun.y()).rotate(marker->m_l).translate(-sun.x(), sun.y());  // Flip Y
 | |
|     QPointF point = rotation.map(QPointF(511, -sun.y() + pixelsPerKPC*marker->m_d));
 | |
|     marker->m_text->setPos(point.x(), -point.y());
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_zoomIn_clicked()
 | |
| {
 | |
|     m_zoom->gentleZoom(1.25);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_zoomOut_clicked()
 | |
| {
 | |
|     m_zoom->gentleZoom(0.75);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_addAnimationFrame_clicked()
 | |
| {
 | |
|     QImage image(ui->image->size(), QImage::Format_ARGB32);
 | |
|     image.fill(Qt::black);
 | |
|     QPainter painter(&image);
 | |
|     ui->image->render(&painter);
 | |
|     m_animationImages.append(image);
 | |
| 
 | |
|     ui->saveAnimation->setEnabled(true);
 | |
|     ui->clearAnimation->setEnabled(true);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_clearAnimation_clicked()
 | |
| {
 | |
|     m_animationImages.clear();
 | |
|     ui->saveAnimation->setEnabled(false);
 | |
|     ui->clearAnimation->setEnabled(false);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_saveAnimation_clicked()
 | |
| {
 | |
|     // Get filename of animation file
 | |
|     QFileDialog fileDialog(nullptr, "Select file to save animation to", "", "*.png");
 | |
|     fileDialog.setAcceptMode(QFileDialog::AcceptSave);
 | |
|     if (fileDialog.exec())
 | |
|     {
 | |
|         QStringList fileNames = fileDialog.selectedFiles();
 | |
|         if (fileNames.size() > 0)
 | |
|         {
 | |
|             APNG apng(m_animationImages.size());
 | |
|             for (int i = 0; i < m_animationImages.size(); i++) {
 | |
|                 apng.addImage(m_animationImages[i]);
 | |
|             }
 | |
|             if (!apng.save(fileNames[0])) {
 | |
|                 QMessageBox::critical(this, "Star Tracker", QString("Failed to write to file %1").arg(fileNames[0]));
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotSkyTemperatureChart()
 | |
| {
 | |
|     ui->chart->setVisible(true);
 | |
|     ui->image->setVisible(false);
 | |
|     ui->drawSun->setVisible(true);
 | |
|     ui->drawMoon->setVisible(true);
 | |
|     ui->darkTheme->setVisible(false);
 | |
|     ui->zoomIn->setVisible(false);
 | |
|     ui->zoomOut->setVisible(false);
 | |
|     ui->addAnimationFrame->setVisible(false);
 | |
|     ui->clearAnimation->setVisible(false);
 | |
|     ui->saveAnimation->setVisible(false);
 | |
| 
 | |
|     bool galactic = (ui->chartSubSelect->currentIndex() & 1) == 1;
 | |
| 
 | |
|     m_chart.removeAllSeries();
 | |
|     removeAllAxes();
 | |
| 
 | |
|     // Plot drift scan path
 | |
|     QList<QLineSeries*> lineSeries;
 | |
|     if (m_settings.m_target == "Custom Az/El")
 | |
|     {
 | |
|         lineSeries = createDriftScan(galactic);
 | |
|         QPen pen(getSeriesColor(1), 2, Qt::SolidLine);
 | |
| 
 | |
|         for (int i = 0; i < lineSeries.length(); i++) {
 | |
|             lineSeries[i]->setPen(pen);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     QScatterSeries *series = new QScatterSeries();
 | |
|     float ra = Astronomy::raToDecimal(m_settings.m_ra);
 | |
|     float dec = Astronomy::decToDecimal(m_settings.m_dec);
 | |
| 
 | |
|     double beamWidth = m_settings.m_beamwidth;
 | |
|     // Ellipse not supported, so draw circle on shorter axis
 | |
|     double degPerPixelW = 360.0/m_chart.plotArea().width();
 | |
|     double degPerPixelH = 180.0/m_chart.plotArea().height();
 | |
|     double degPerPixel = std::min(degPerPixelW, degPerPixelH);
 | |
|     double markerSize;
 | |
| 
 | |
|     double x, y;
 | |
|     mapRaDec(ra, dec, galactic, x, y);
 | |
|     series->append(x, y);
 | |
| 
 | |
|     // Get temperature
 | |
|     int idx = ui->chartSubSelect->currentIndex();
 | |
| 
 | |
|     if ((idx == 6) || (idx == 7))
 | |
|     {
 | |
|         double temp;
 | |
| 
 | |
|         if (m_starTracker->calcSkyTemperature(m_settings.m_frequency, m_settings.m_beamwidth, ra, dec, temp))
 | |
|         {
 | |
|             series->setPointLabelsVisible(true);
 | |
|             series->setPointLabelsColor(Qt::red);
 | |
|             series->setPointLabelsFormat(QString("%1 K").arg(std::round(temp)));
 | |
| 
 | |
|             // Scale marker size by beamwidth
 | |
|             markerSize = std::max((int)round(beamWidth * degPerPixel), 5);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         // Read temperature from selected FITS file at target RA/Dec
 | |
|         QImage *img = &m_images[idx];
 | |
|         const FITS *fits = m_starTracker->getTempFITS(idx/2);
 | |
|         double x;
 | |
| 
 | |
|         if (ra <= 12.0) {
 | |
|             x = (12.0 - ra) / 24.0;
 | |
|         } else {
 | |
|             x = (24 - ra + 12) / 24.0;
 | |
|         }
 | |
| 
 | |
|         int imgX = x * (img->width() - 1);
 | |
| 
 | |
|         if (imgX >= img->width()) {
 | |
|             imgX = img->width() - 1;
 | |
|         }
 | |
| 
 | |
|         int imgY = (90.0-dec)/180.0 * (img->height() - 1);
 | |
| 
 | |
|         if (imgY >= img->height()) {
 | |
|             imgY = img->height() - 1;
 | |
|         }
 | |
| 
 | |
|         if (fits->valid())
 | |
|         {
 | |
|             double temp = fits->scaledValue(imgX, imgY);
 | |
|             series->setPointLabelsVisible(true);
 | |
|             series->setPointLabelsColor(Qt::red);
 | |
|             series->setPointLabelsFormat(QString("%1 K").arg(std::round(temp)));
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             qDebug() << "FITS not valid";
 | |
|         }
 | |
| 
 | |
|         // Temperature from just one pixel, but need to make marker visbile
 | |
|         markerSize = 5;
 | |
|     }
 | |
| 
 | |
|     series->setMarkerSize(markerSize);
 | |
|     series->setColor(getSeriesColor(0));
 | |
|     m_chart.setTitle("");
 | |
| 
 | |
|     // We want scatter (for the beam) to be on top of line, but same color even when other series
 | |
|     for (int i = 0; i < lineSeries.length(); i++) {
 | |
|         m_chart.addSeries(lineSeries[i]);
 | |
|     }
 | |
| 
 | |
|     // Draw Sun on chart if requested
 | |
|     QScatterSeries *sunSeries = nullptr;
 | |
|     if (m_settings.m_drawSunOnSkyTempChart)
 | |
|     {
 | |
|         sunSeries = new QScatterSeries();
 | |
|         mapRaDec(m_sunRA, m_sunDec, galactic, x, y);
 | |
|         sunSeries->append(x, y);
 | |
|         sunSeries->setMarkerSize((int)std::max(std::round(0.53 * degPerPixel), 5.0));
 | |
|         sunSeries->setColor(Qt::yellow);
 | |
|         sunSeries->setBorderColor(Qt::yellow);
 | |
| 
 | |
|         if (m_settings.m_target != "Sun") // Avoid labels on top of each other
 | |
|         {
 | |
|             sunSeries->setPointLabelsVisible(true);
 | |
|             sunSeries->setPointLabelsColor(Qt::red);
 | |
|             sunSeries->setPointLabelsFormat("Sun");
 | |
|         }
 | |
| 
 | |
|         m_chart.addSeries(sunSeries);
 | |
|     }
 | |
| 
 | |
|     // Draw moon on chart if requested
 | |
|     QScatterSeries *moonSeries = nullptr;
 | |
|     if (m_settings.m_drawMoonOnSkyTempChart)
 | |
|     {
 | |
|         moonSeries = new QScatterSeries();
 | |
|         mapRaDec(m_moonRA, m_moonDec, galactic, x, y);
 | |
|         moonSeries->append(x, y);
 | |
|         moonSeries->setMarkerSize((int)std::max(std::round(0.53 * degPerPixel), 5.0));
 | |
|         moonSeries->setColor(qRgb(150, 150, 150));
 | |
|         moonSeries->setBorderColor(qRgb(150, 150, 150));
 | |
| 
 | |
|         if (m_settings.m_target != "Moon") // Avoid labels on top of each other
 | |
|         {
 | |
|             moonSeries->setPointLabelsVisible(true);
 | |
|             moonSeries->setPointLabelsColor(Qt::red);
 | |
|             moonSeries->setPointLabelsFormat("Moon");
 | |
|         }
 | |
| 
 | |
|         m_chart.addSeries(moonSeries);
 | |
|     }
 | |
| 
 | |
|     m_chart.addSeries(series);
 | |
| 
 | |
|     if (galactic)
 | |
|     {
 | |
|         m_chart.addAxis(&m_skyTempGalacticLXAxis, Qt::AlignBottom);
 | |
|         series->attachAxis(&m_skyTempGalacticLXAxis);
 | |
| 
 | |
|         if (sunSeries) {
 | |
|             sunSeries->attachAxis(&m_skyTempGalacticLXAxis);
 | |
|         }
 | |
|         if (moonSeries) {
 | |
|             moonSeries->attachAxis(&m_skyTempGalacticLXAxis);
 | |
|         }
 | |
| 
 | |
|         m_skyTempYAxis.setTitleText(QString("Galactic latitude (%1)").arg(QChar(0xb0)));
 | |
|         m_chart.addAxis(&m_skyTempYAxis, Qt::AlignLeft);
 | |
|         series->attachAxis(&m_skyTempYAxis);
 | |
| 
 | |
|         if (sunSeries) {
 | |
|             sunSeries->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
|         if (moonSeries) {
 | |
|             moonSeries->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
| 
 | |
|         for (int i = 0; i < lineSeries.length(); i++)
 | |
|         {
 | |
|             lineSeries[i]->attachAxis(&m_skyTempGalacticLXAxis);
 | |
|             lineSeries[i]->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         m_chart.addAxis(&m_skyTempRAXAxis, Qt::AlignBottom);
 | |
|         series->attachAxis(&m_skyTempRAXAxis);
 | |
| 
 | |
|         if (sunSeries) {
 | |
|             sunSeries->attachAxis(&m_skyTempRAXAxis);
 | |
|         }
 | |
|         if (moonSeries) {
 | |
|             moonSeries->attachAxis(&m_skyTempRAXAxis);
 | |
|         }
 | |
| 
 | |
|         m_skyTempYAxis.setTitleText(QString("Declination (%1)").arg(QChar(0xb0)));
 | |
|         m_chart.addAxis(&m_skyTempYAxis, Qt::AlignLeft);
 | |
|         series->attachAxis(&m_skyTempYAxis);
 | |
| 
 | |
|         if (sunSeries) {
 | |
|             sunSeries->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
|         if (moonSeries) {
 | |
|             moonSeries->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
| 
 | |
|         for (int i = 0; i < lineSeries.length(); i++)
 | |
|         {
 | |
|             lineSeries[i]->attachAxis(&m_skyTempRAXAxis);
 | |
|             lineSeries[i]->attachAxis(&m_skyTempYAxis);
 | |
|         }
 | |
|     }
 | |
|     ui->chart->setChart(&m_chart);
 | |
|     plotAreaChanged(m_chart.plotArea());
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::plotAreaChanged(const QRectF &plotArea)
 | |
| {
 | |
|     int width = static_cast<int>(plotArea.width());
 | |
|     int height = static_cast<int>(plotArea.height());
 | |
|     int viewW = static_cast<int>(ui->chart->width());
 | |
|     int viewH = static_cast<int>(ui->chart->height());
 | |
| 
 | |
|     // Scale the image to fit plot area
 | |
|     int imageIdx = ui->chartSubSelect->currentIndex();
 | |
| 
 | |
|     if (imageIdx == -1) {
 | |
|         return;
 | |
|     } else if (imageIdx == 6) {
 | |
|         imageIdx = 2;
 | |
|     } else if (imageIdx == 7) {
 | |
|         imageIdx = 3;
 | |
|     }
 | |
| 
 | |
|     QImage image = m_images[imageIdx].scaled(QSize(width, height), Qt::IgnoreAspectRatio);
 | |
|     QImage translated(viewW, viewH, QImage::Format_ARGB32);
 | |
|     translated.fill(Qt::white);
 | |
|     QPainter painter(&translated);
 | |
|     painter.drawImage(plotArea.topLeft(), image);
 | |
| 
 | |
|     m_chart.setPlotAreaBackgroundBrush(translated);
 | |
|     m_chart.setPlotAreaBackgroundVisible(true);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::removeAllAxes()
 | |
| {
 | |
|     QList<QAbstractAxis *> axes;
 | |
|     axes = m_chart.axes(Qt::Horizontal);
 | |
| 
 | |
|     for (QAbstractAxis *axis : axes) {
 | |
|         m_chart.removeAxis(axis);
 | |
|     }
 | |
| 
 | |
|     axes = m_chart.axes(Qt::Vertical);
 | |
| 
 | |
|     for (QAbstractAxis *axis : axes) {
 | |
|         m_chart.removeAxis(axis);
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Plot target elevation angle over the day
 | |
| void StarTrackerGUI::plotElevationLineChart()
 | |
| {
 | |
|     ui->chart->setVisible(true);
 | |
|     ui->image->setVisible(false);
 | |
|     ui->drawSun->setVisible(false);
 | |
|     ui->drawMoon->setVisible(false);
 | |
|     ui->darkTheme->setVisible(true);
 | |
|     ui->zoomIn->setVisible(false);
 | |
|     ui->zoomOut->setVisible(false);
 | |
|     ui->addAnimationFrame->setVisible(false);
 | |
|     ui->clearAnimation->setVisible(false);
 | |
|     ui->saveAnimation->setVisible(false);
 | |
| 
 | |
|     QChart *oldChart = m_azElLineChart;
 | |
| 
 | |
|     m_azElLineChart = new QChart();
 | |
|     m_azElLineChart->setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     QDateTimeAxis *xAxis = new QDateTimeAxis();
 | |
|     QValueAxis *yLeftAxis = new QValueAxis();
 | |
|     QValueAxis *yRightAxis = new QValueAxis();
 | |
| 
 | |
|     m_azElLineChart->legend()->hide();
 | |
| 
 | |
|     m_azElLineChart->layout()->setContentsMargins(0, 0, 0, 0);
 | |
|     m_azElLineChart->setMargins(QMargins(1, 1, 1, 1));
 | |
| 
 | |
|     double maxElevation = -90.0;
 | |
| 
 | |
|     QLineSeries *elSeries = new QLineSeries();
 | |
|     QList<QLineSeries *> azSeriesList;
 | |
|     QLineSeries *azSeries = new QLineSeries();
 | |
|     azSeriesList.append(azSeries);
 | |
|     QPen pen(getSeriesColor(1), 2, Qt::SolidLine);
 | |
|     azSeries->setPen(pen);
 | |
| 
 | |
|     QDateTime dt;
 | |
| 
 | |
|     if (m_settings.m_dateTime.isEmpty()) {
 | |
|         dt = QDateTime::currentDateTime();
 | |
|     } else {
 | |
|         dt = QDateTime::fromString(m_settings.m_dateTime, Qt::ISODateWithMs);
 | |
|     }
 | |
| 
 | |
|     dt.setTime(QTime(0,0));
 | |
|     QDateTime startTime = dt;
 | |
|     QDateTime endTime = dt;
 | |
|     double prevAz;
 | |
|     int timestep = 10*60;
 | |
| 
 | |
|     for (int step = 0; step <= 24*60*60/timestep; step++)
 | |
|     {
 | |
|         AzAlt aa;
 | |
|         RADec rd;
 | |
| 
 | |
|         // Calculate elevation of desired object
 | |
|         if (m_settings.m_target == "Sun")
 | |
|         {
 | |
|             Astronomy::sunPosition(aa, rd, m_settings.m_latitude, m_settings.m_longitude, dt);
 | |
|         }
 | |
|         else if (m_settings.m_target == "Moon")
 | |
|         {
 | |
|             Astronomy::moonPosition(aa, rd, m_settings.m_latitude, m_settings.m_longitude, dt);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             rd.ra = Astronomy::raToDecimal(m_settings.m_ra);
 | |
|             rd.dec = Astronomy::decToDecimal(m_settings.m_dec);
 | |
|             aa = Astronomy::raDecToAzAlt(rd, m_settings.m_latitude, m_settings.m_longitude, dt, !m_settings.m_jnow);
 | |
|         }
 | |
| 
 | |
|         if (aa.alt > maxElevation) {
 | |
|             maxElevation = aa.alt;
 | |
|         }
 | |
| 
 | |
|         // Adjust for refraction
 | |
|         if (m_settings.m_refraction == "Positional Astronomy Library")
 | |
|         {
 | |
|             aa.alt += Astronomy::refractionPAL(
 | |
|                 aa.alt,
 | |
|                 m_settings.m_pressure,
 | |
|                 m_settings.m_temperature,
 | |
|                 m_settings.m_humidity,
 | |
|                 m_settings.m_frequency,
 | |
|                 m_settings.m_latitude,
 | |
|                 m_settings.m_heightAboveSeaLevel,
 | |
|                 m_settings.m_temperatureLapseRate
 | |
|             );
 | |
| 
 | |
|             if (aa.alt > 90.0) {
 | |
|                 aa.alt = 90.0f;
 | |
|             }
 | |
|         }
 | |
|         else if (m_settings.m_refraction == "Saemundsson")
 | |
|         {
 | |
|             aa.alt += Astronomy::refractionSaemundsson(aa.alt, m_settings.m_pressure, m_settings.m_temperature);
 | |
| 
 | |
|             if (aa.alt > 90.0) {
 | |
|                 aa.alt = 90.0f;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (step == 0) {
 | |
|             prevAz = aa.az;
 | |
|         }
 | |
| 
 | |
|         if (((prevAz >= 270) && (aa.az < 90)) || ((prevAz < 90) && (aa.az >= 270)))
 | |
|         {
 | |
|             azSeries = new QLineSeries();
 | |
|             azSeriesList.append(azSeries);
 | |
|             azSeries->setPen(pen);
 | |
|         }
 | |
| 
 | |
|         elSeries->append(dt.toMSecsSinceEpoch(), aa.alt);
 | |
|         azSeries->append(dt.toMSecsSinceEpoch(), aa.az);
 | |
|         endTime = dt;
 | |
|         prevAz = aa.az;
 | |
|         dt = dt.addSecs(timestep); // addSecs accounts for daylight savings jumps
 | |
|     }
 | |
| 
 | |
|     m_azElLineChart->addAxis(xAxis, Qt::AlignBottom);
 | |
|     m_azElLineChart->addAxis(yLeftAxis, Qt::AlignLeft);
 | |
|     m_azElLineChart->addAxis(yRightAxis, Qt::AlignRight);
 | |
|     m_azElLineChart->addSeries(elSeries);
 | |
| 
 | |
|     for (int i = 0; i < azSeriesList.size(); i++)
 | |
|     {
 | |
|         m_azElLineChart->addSeries(azSeriesList[i]);
 | |
|         azSeriesList[i]->attachAxis(xAxis);
 | |
|         azSeriesList[i]->attachAxis(yRightAxis);
 | |
|     }
 | |
| 
 | |
|     elSeries->attachAxis(xAxis);
 | |
|     elSeries->attachAxis(yLeftAxis);
 | |
|     xAxis->setTitleText(QString("%1 %2").arg(startTime.date().toString()).arg(startTime.timeZoneAbbreviation()));
 | |
|     xAxis->setFormat("hh");
 | |
|     xAxis->setTickCount(7);
 | |
|     xAxis->setRange(startTime, endTime);
 | |
|     yLeftAxis->setRange(0.0, 90.0);
 | |
|     yLeftAxis->setTitleText(QString("Elevation (%1)").arg(QChar(0xb0)));
 | |
|     yRightAxis->setRange(0.0, 360.0);
 | |
|     yRightAxis->setTitleText(QString("Azimuth (%1)").arg(QChar(0xb0)));
 | |
| 
 | |
|     if (maxElevation < 0) {
 | |
|         m_azElLineChart->setTitle("Not visible from this latitude");
 | |
|     } else {
 | |
|         m_azElLineChart->setTitle("");
 | |
|     }
 | |
| 
 | |
|     ui->chart->setChart(m_azElLineChart);
 | |
|     delete oldChart;
 | |
| }
 | |
| 
 | |
| // Plot target elevation angle over the day
 | |
| void StarTrackerGUI::plotElevationPolarChart()
 | |
| {
 | |
|     ui->chart->setVisible(true);
 | |
|     ui->image->setVisible(false);
 | |
|     ui->drawSun->setVisible(false);
 | |
|     ui->drawMoon->setVisible(false);
 | |
|     ui->darkTheme->setVisible(true);
 | |
|     ui->zoomIn->setVisible(false);
 | |
|     ui->zoomOut->setVisible(false);
 | |
|     ui->addAnimationFrame->setVisible(false);
 | |
|     ui->clearAnimation->setVisible(false);
 | |
|     ui->saveAnimation->setVisible(false);
 | |
| 
 | |
|     QChart *oldChart = m_azElPolarChart;
 | |
| 
 | |
|     m_azElPolarChart = new QPolarChart();
 | |
|     m_azElPolarChart->setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     QValueAxis *angularAxis = new QValueAxis();
 | |
|     QCategoryAxis *radialAxis = new QCategoryAxis();
 | |
| 
 | |
|     angularAxis->setTickCount(9);
 | |
|     angularAxis->setMinorTickCount(1);
 | |
|     angularAxis->setLabelFormat("%d");
 | |
|     angularAxis->setRange(0, 360);
 | |
| 
 | |
|     radialAxis->setMin(0);
 | |
|     radialAxis->setMax(90);
 | |
|     radialAxis->append("90", 0);
 | |
|     radialAxis->append("60", 30);
 | |
|     radialAxis->append("30", 60);
 | |
|     radialAxis->append("0", 90);
 | |
|     radialAxis->setLabelsPosition(QCategoryAxis::AxisLabelsPositionOnValue);
 | |
| 
 | |
|     m_azElPolarChart->addAxis(angularAxis, QPolarChart::PolarOrientationAngular);
 | |
|     m_azElPolarChart->addAxis(radialAxis, QPolarChart::PolarOrientationRadial);
 | |
|     m_azElPolarChart->legend()->hide();
 | |
|     m_azElPolarChart->layout()->setContentsMargins(0, 0, 0, 0);
 | |
|     m_azElPolarChart->setMargins(QMargins(1, 1, 1, 1));
 | |
| 
 | |
|     double maxElevation = -90.0;
 | |
| 
 | |
|     QLineSeries *polarSeries = new QLineSeries();
 | |
|     QDateTime dt;
 | |
| 
 | |
|     if (m_settings.m_dateTime.isEmpty()) {
 | |
|         dt = QDateTime::currentDateTime();
 | |
|     } else {
 | |
|         dt = QDateTime::fromString(m_settings.m_dateTime, Qt::ISODateWithMs);
 | |
|     }
 | |
| 
 | |
|     dt.setTime(QTime(0,0));
 | |
|     QDateTime startTime = dt;
 | |
|     QDateTime endTime = dt;
 | |
|     QDateTime riseTime;
 | |
|     QDateTime setTime;
 | |
|     int riseIdx = -1;
 | |
|     int setIdx = -1;
 | |
|     int idx = 0;
 | |
|     int timestep = 10*60; // Rise/set times accurate to nearest 10 minutes
 | |
|     double prevAlt;
 | |
| 
 | |
|     for (int step = 0; step <= 24*60*60/timestep; step++)
 | |
|     {
 | |
|         AzAlt aa;
 | |
|         RADec rd;
 | |
| 
 | |
|         // Calculate elevation of desired object
 | |
|         if (m_settings.m_target == "Sun")
 | |
|         {
 | |
|             Astronomy::sunPosition(aa, rd, m_settings.m_latitude, m_settings.m_longitude, dt);
 | |
|         }
 | |
|         else if (m_settings.m_target == "Moon")
 | |
|         {
 | |
|             Astronomy::moonPosition(aa, rd, m_settings.m_latitude, m_settings.m_longitude, dt);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             rd.ra = Astronomy::raToDecimal(m_settings.m_ra);
 | |
|             rd.dec = Astronomy::decToDecimal(m_settings.m_dec);
 | |
|             aa = Astronomy::raDecToAzAlt(rd, m_settings.m_latitude, m_settings.m_longitude, dt, !m_settings.m_jnow);
 | |
|         }
 | |
| 
 | |
|         if (aa.alt > maxElevation) {
 | |
|             maxElevation = aa.alt;
 | |
|         }
 | |
| 
 | |
|         // Adjust for refraction
 | |
|         if (m_settings.m_refraction == "Positional Astronomy Library")
 | |
|         {
 | |
|             aa.alt += Astronomy::refractionPAL(
 | |
|                 aa.alt,
 | |
|                 m_settings.m_pressure,
 | |
|                 m_settings.m_temperature,
 | |
|                 m_settings.m_humidity,
 | |
|                 m_settings.m_frequency,
 | |
|                 m_settings.m_latitude,
 | |
|                 m_settings.m_heightAboveSeaLevel,
 | |
|                 m_settings.m_temperatureLapseRate
 | |
|             );
 | |
| 
 | |
|             if (aa.alt > 90.0) {
 | |
|                 aa.alt = 90.0f;
 | |
|             }
 | |
|         }
 | |
|         else if (m_settings.m_refraction == "Saemundsson")
 | |
|         {
 | |
|             aa.alt += Astronomy::refractionSaemundsson(aa.alt, m_settings.m_pressure, m_settings.m_temperature);
 | |
| 
 | |
|             if (aa.alt > 90.0) {
 | |
|                 aa.alt = 90.0f;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (idx == 0) {
 | |
|             prevAlt = aa.alt;
 | |
|         }
 | |
| 
 | |
|         // We can have set before rise in a day, if the object starts > 0
 | |
|         if ((aa.alt >= 0.0) && (prevAlt < 0.0))
 | |
|         {
 | |
|             riseTime = dt;
 | |
|             riseIdx = idx;
 | |
|         }
 | |
| 
 | |
|         if ((aa.alt < 0.0) && (prevAlt >= 0.0))
 | |
|         {
 | |
|             setTime = endTime;
 | |
|             setIdx = idx;
 | |
|         }
 | |
| 
 | |
|         polarSeries->append(aa.az, 90 - aa.alt);
 | |
|         idx++;
 | |
|         endTime = dt;
 | |
|         prevAlt = aa.alt;
 | |
|         dt = dt.addSecs(timestep); // addSecs accounts for daylight savings jumps
 | |
|     }
 | |
| 
 | |
|     // Polar charts can't handle points that are more than 180 degrees apart, so
 | |
|     // we need to split passes that cross from 359 -> 0 degrees (or the reverse)
 | |
|     QList<QLineSeries *> series;
 | |
|     series.append(new QLineSeries());
 | |
|     QLineSeries *s = series.first();
 | |
|     QPen pen(getSeriesColor(0), 2, Qt::SolidLine);
 | |
|     s->setPen(pen);
 | |
| 
 | |
|     qreal prevAz = polarSeries->at(0).x();
 | |
|     qreal prevEl = polarSeries->at(0).y();
 | |
| 
 | |
|     for (int i = 1; i < polarSeries->count(); i++)
 | |
|     {
 | |
|         qreal az = polarSeries->at(i).x();
 | |
|         qreal el = polarSeries->at(i).y();
 | |
| 
 | |
|         if ((prevAz > 270.0) && (az <= 90.0))
 | |
|         {
 | |
|             double elMid = Interpolation::interpolate(prevAz, prevEl, az+360.0, el, 360.0);
 | |
|             s->append(360.0, elMid);
 | |
|             series.append(new QLineSeries());
 | |
|             s = series.last();
 | |
|             s->setPen(pen);
 | |
|             s->append(0.0, elMid);
 | |
|             s->append(az, el);
 | |
|         }
 | |
|         else if ((prevAz <= 90.0) && (az > 270.0))
 | |
|         {
 | |
|             double elMid = Interpolation::interpolate(prevAz, prevEl, az-360.0, el, 0.0);
 | |
|             s->append(0.0, elMid);
 | |
|             series.append(new QLineSeries());
 | |
|             s = series.last();
 | |
|             s->setPen(pen);
 | |
|             s->append(360.0, elMid);
 | |
|             s->append(az, el);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             s->append(polarSeries->at(i));
 | |
|         }
 | |
| 
 | |
|         prevAz = az;
 | |
|         prevEl = el;
 | |
|     }
 | |
| 
 | |
|     for (int i = 0; i < series.length(); i++)
 | |
|     {
 | |
|         m_azElPolarChart->addSeries(series[i]);
 | |
|         series[i]->attachAxis(angularAxis);
 | |
|         series[i]->attachAxis(radialAxis);
 | |
|     }
 | |
| 
 | |
|     // Create series with single point, so we can plot time of rising
 | |
|     if (riseTime.isValid())
 | |
|     {
 | |
|         QLineSeries *riseSeries = new QLineSeries();
 | |
|         riseSeries->append(polarSeries->at(riseIdx));
 | |
|         riseSeries->setPointLabelsFormat(QString("Rise %1").arg(riseTime.time().toString("hh:mm")));
 | |
|         riseSeries->setPointLabelsVisible(true);
 | |
|         riseSeries->setPointLabelsClipping(false);
 | |
|         m_azElPolarChart->addSeries(riseSeries);
 | |
|         riseSeries->attachAxis(angularAxis);
 | |
|         riseSeries->attachAxis(radialAxis);
 | |
|     }
 | |
| 
 | |
|     // Create series with single point, so we can plot time of setting
 | |
|     if (setTime.isValid())
 | |
|     {
 | |
|         QLineSeries *setSeries = new QLineSeries();
 | |
|         setSeries->append(polarSeries->at(setIdx));
 | |
|         setSeries->setPointLabelsFormat(QString("Set %1").arg(setTime.time().toString("hh:mm")));
 | |
|         setSeries->setPointLabelsVisible(true);
 | |
|         setSeries->setPointLabelsClipping(false);
 | |
|         m_azElPolarChart->addSeries(setSeries);
 | |
|         setSeries->attachAxis(angularAxis);
 | |
|         setSeries->attachAxis(radialAxis);
 | |
|     }
 | |
| 
 | |
|     if (maxElevation < 0) {
 | |
|         m_azElPolarChart->setTitle("Not visible from this latitude");
 | |
|     } else {
 | |
|         m_azElPolarChart->setTitle("");
 | |
|     }
 | |
| 
 | |
|     ui->chart->setChart(m_azElPolarChart);
 | |
| 
 | |
|     delete polarSeries;
 | |
|     delete oldChart;
 | |
| }
 | |
| 
 | |
| // Find target on the Map
 | |
| void StarTrackerGUI::on_viewOnMap_clicked()
 | |
| {
 | |
|     QString target = m_settings.m_target == "Sun" || m_settings.m_target == "Moon" ? m_settings.m_target : "Star";
 | |
|     FeatureWebAPIUtils::mapFind(target);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::updateChartSubSelect()
 | |
| {
 | |
|     if (ui->chartSelect->currentIndex() == 2)
 | |
|     {
 | |
|         ui->chartSubSelect->setItemText(6, QString("%1 MHz %2%3 Equatorial")
 | |
|                                         .arg((int)std::round(m_settings.m_frequency/1e6))
 | |
|                                         .arg((int)std::round(m_settings.m_beamwidth))
 | |
|                                         .arg(QChar(0xb0)));
 | |
|         ui->chartSubSelect->setItemText(7, QString("%1 MHz %2%3 Galactic")
 | |
|                                         .arg((int)std::round(m_settings.m_frequency/1e6))
 | |
|                                         .arg((int)std::round(m_settings.m_beamwidth))
 | |
|                                         .arg(QChar(0xb0)));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_chartSelect_currentIndexChanged(int index)
 | |
| {
 | |
|     bool oldState = ui->chartSubSelect->blockSignals(true);
 | |
|     ui->chartSubSelect->clear();
 | |
| 
 | |
|     if (index == 0)
 | |
|     {
 | |
|         ui->chartSubSelect->addItem("Az/El vs time");
 | |
|         ui->chartSubSelect->addItem("Polar");
 | |
|     }
 | |
|     else if (index == 2)
 | |
|     {
 | |
|         ui->chartSubSelect->addItem(QString("150 MHz 5%1 Equatorial").arg(QChar(0xb0)));
 | |
|         ui->chartSubSelect->addItem(QString("150 MHz 5%1 Galactic").arg(QChar(0xb0)));
 | |
|         ui->chartSubSelect->addItem("408 MHz 51' Equatorial");
 | |
|         ui->chartSubSelect->addItem("408 MHz 51' Galactic");
 | |
|         ui->chartSubSelect->addItem("1420 MHz 35' Equatorial");
 | |
|         ui->chartSubSelect->addItem("1420 MHz 35' Galactic");
 | |
|         ui->chartSubSelect->addItem("Custom Equatorial");
 | |
|         ui->chartSubSelect->addItem("Custom Galactic");
 | |
|         ui->chartSubSelect->setCurrentIndex(2);
 | |
|         updateChartSubSelect();
 | |
|     }
 | |
|     else if (index == 3)
 | |
|     {
 | |
|         ui->chartSubSelect->addItem("Milky Way");
 | |
|         ui->chartSubSelect->addItem("Milky Way annotated");
 | |
|     }
 | |
| 
 | |
|     ui->chartSubSelect->blockSignals(oldState);
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_chartSubSelect_currentIndexChanged(int index)
 | |
| {
 | |
|     (void) index;
 | |
|     plotChart();
 | |
| }
 | |
| 
 | |
| double StarTrackerGUI::convertSolarFluxUnits(double sfu)
 | |
| {
 | |
|     switch (m_settings.m_solarFluxUnits)
 | |
|     {
 | |
|     case StarTrackerSettings::SFU:
 | |
|         return sfu;
 | |
|     case StarTrackerSettings::JANSKY:
 | |
|         return Units::solarFluxUnitsToJansky(sfu);
 | |
|     case StarTrackerSettings::WATTS_M_HZ:
 | |
|         return Units::solarFluxUnitsToWattsPerMetrePerHertz(sfu);
 | |
|     }
 | |
|     return 0.0;
 | |
| }
 | |
| 
 | |
| QString StarTrackerGUI::solarFluxUnit()
 | |
| {
 | |
|     switch (m_settings.m_solarFluxUnits)
 | |
|     {
 | |
|     case StarTrackerSettings::SFU:
 | |
|         return "sfu";
 | |
|     case StarTrackerSettings::JANSKY:
 | |
|         return "Jy";
 | |
|     case StarTrackerSettings::WATTS_M_HZ:
 | |
|         return "Wm^-2Hz^-1";
 | |
|     }
 | |
|     return "";
 | |
| }
 | |
| 
 | |
| // Calculate solar flux at given frequency in SFU
 | |
| double StarTrackerGUI::calcSolarFlux(double freqMhz)
 | |
| {
 | |
|     if (m_solarFluxesValid)
 | |
|     {
 | |
|         double solarFlux;
 | |
|         const int fluxes = sizeof(m_solarFluxFrequencies)/sizeof(*m_solarFluxFrequencies);
 | |
|         int i;
 | |
| 
 | |
|         for (i = 0; i < fluxes; i++)
 | |
|         {
 | |
|             if (freqMhz < m_solarFluxFrequencies[i]) {
 | |
|                 break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if (i == 0)
 | |
|         {
 | |
|             solarFlux = Interpolation::extrapolate(
 | |
|                 (double)m_solarFluxFrequencies[0],
 | |
|                 (double)m_solarFluxes[0],
 | |
|                 (double)m_solarFluxFrequencies[1],
 | |
|                 (double)m_solarFluxes[1],
 | |
|                 freqMhz
 | |
|             );
 | |
|         }
 | |
|         else if (i == fluxes)
 | |
|         {
 | |
|             solarFlux = Interpolation::extrapolate(
 | |
|                 (double)m_solarFluxFrequencies[fluxes-2],
 | |
|                 (double)m_solarFluxes[fluxes-2],
 | |
|                 (double)m_solarFluxFrequencies[fluxes-1],
 | |
|                 (double)m_solarFluxes[fluxes-1],
 | |
|                 freqMhz
 | |
|             );
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             solarFlux = Interpolation::interpolate(
 | |
|                 (double)m_solarFluxFrequencies[i-1],
 | |
|                 (double)m_solarFluxes[i-1],
 | |
|                 (double)m_solarFluxFrequencies[i],
 | |
|                 (double)m_solarFluxes[i],
 | |
|                 freqMhz
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         return solarFlux;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         return 0.0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::displaySolarFlux()
 | |
| {
 | |
|     if (((m_settings.m_solarFluxData == StarTrackerSettings::DRAO_2800) && (m_solarFlux == 0.0))
 | |
|      || ((m_settings.m_solarFluxData != StarTrackerSettings::DRAO_2800) && !m_solarFluxesValid))
 | |
|     {
 | |
|         ui->solarFlux->setText("");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         double solarFlux;
 | |
|         double freqMhz = m_settings.m_frequency/1000000.0;
 | |
| 
 | |
|         if (m_settings.m_solarFluxData == StarTrackerSettings::DRAO_2800)
 | |
|         {
 | |
|             solarFlux = m_solarFlux;
 | |
|             ui->solarFlux->setToolTip(QString("Solar flux density at 2800 MHz"));
 | |
|         }
 | |
|         else if (m_settings.m_solarFluxData == StarTrackerSettings::TARGET_FREQ)
 | |
|         {
 | |
|             solarFlux = calcSolarFlux(freqMhz);
 | |
|             ui->solarFlux->setToolTip(QString("Solar flux density interpolated to %1 MHz").arg(freqMhz));
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             int idx = m_settings.m_solarFluxData-StarTrackerSettings::L_245;
 | |
|             solarFlux = m_solarFluxes[idx];
 | |
|             ui->solarFlux->setToolTip(QString("Solar flux density at %1 MHz").arg(m_solarFluxFrequencies[idx]));
 | |
|         }
 | |
| 
 | |
|         ui->solarFlux->setText(QString("%1 %2").arg(convertSolarFluxUnits(solarFlux)).arg(solarFluxUnit()));
 | |
|         ui->solarFlux->setCursorPosition(0);
 | |
| 
 | |
|         // Calculate value at target frequency in Jy for Radio Astronomy plugin
 | |
|         m_starTracker->getInputMessageQueue()->push(StarTracker::MsgSetSolarFlux::create(Units::solarFluxUnitsToJansky(calcSolarFlux(freqMhz))));
 | |
|     }
 | |
| }
 | |
| 
 | |
| bool StarTrackerGUI::readSolarFlux()
 | |
| {
 | |
|     QFile file(getSolarFluxFilename());
 | |
|     QDateTime lastModified = file.fileTime(QFileDevice::FileModificationTime);
 | |
| 
 | |
|     if (QDateTime::currentDateTime().secsTo(lastModified) >= -(60*60*24))
 | |
|     {
 | |
|         if (file.open(QIODevice::ReadOnly | QIODevice::Text))
 | |
|         {
 | |
|             QByteArray bytes = file.readLine();
 | |
|             QString string(bytes);
 | |
|             // HHMMSS 245    410     610    1415   2695   4995   8800  15400   Mhz
 | |
|             // 000000 000019 000027 000037 000056 000073 000116 000202 000514  sfu
 | |
|             // Occasionally, file will contain ////// in a column, presumably to indicate no data
 | |
|             QRegExp re("([0-9]{2})([0-9]{2})([0-9]{2}) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+) ([0-9\\/]+)");
 | |
| 
 | |
|             if (re.indexIn(string) != -1)
 | |
|             {
 | |
|                 for (int i = 0; i < 8; i++)
 | |
|                     m_solarFluxes[i] = re.capturedTexts()[i+4].toInt();
 | |
|                 m_solarFluxesValid = true;
 | |
|                 displaySolarFlux();
 | |
|                 plotChart();
 | |
|                 return true;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 qDebug() << "StarTrackerGUI::readSolarFlux: No match for " << string;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         qDebug() << "StarTrackerGUI::readSolarFlux: Solar flux data is more than 1 day old";
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::networkManagerFinished(QNetworkReply *reply)
 | |
| {
 | |
|     ui->solarFlux->setText(""); // Don't show obsolete data
 | |
|     QNetworkReply::NetworkError replyError = reply->error();
 | |
| 
 | |
|     if (replyError)
 | |
|     {
 | |
|         qWarning() << "StarTrackerGUI::networkManagerFinished:"
 | |
|             << " error(" << (int) replyError
 | |
|             << "): " << replyError
 | |
|             << ": " << reply->errorString();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         QString answer = reply->readAll();
 | |
|         QRegExp re("\\<th\\>Observed Flux Density\\<\\/th\\>\\<td\\>([0-9]+(\\.[0-9]+)?)\\<\\/td\\>");
 | |
| 
 | |
|         if (re.indexIn(answer) != -1)
 | |
|         {
 | |
|             m_solarFlux = re.capturedTexts()[1].toDouble();
 | |
|             displaySolarFlux();
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             qDebug() << "StarTrackerGUI::networkManagerFinished - No Solar flux found: " << answer;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     reply->deleteLater();
 | |
| }
 | |
| 
 | |
| QString StarTrackerGUI::getSolarFluxFilename()
 | |
| {
 | |
|     return HttpDownloadManager::downloadDir() + "/solar_flux.srd";
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::updateSolarFlux(bool all)
 | |
| {
 | |
|     qDebug() << "StarTrackerGUI: Updating Solar flux data";
 | |
| 
 | |
|     if ((m_settings.m_solarFluxData != StarTrackerSettings::DRAO_2800) || all)
 | |
|     {
 | |
|         QDate today = QDateTime::currentDateTimeUtc().date();
 | |
|         QString solarFluxFile = getSolarFluxFilename();
 | |
| 
 | |
|         if (m_dlm.confirmDownload(solarFluxFile, nullptr, 0))
 | |
|         {
 | |
|             QString urlString = QString("https://www.sws.bom.gov.au/Category/World Data Centre/Data Display and Download/Solar Radio/station/learmonth/SRD/%1/L%2.SRD")
 | |
|                                     .arg(today.year()).arg(today.toString("yyMMdd"));
 | |
|             m_dlm.download(QUrl(urlString), solarFluxFile, this);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if ((m_settings.m_solarFluxData == StarTrackerSettings::DRAO_2800) || all)
 | |
|     {
 | |
|         m_networkRequest.setUrl(QUrl("https://www.spaceweather.gc.ca/forecast-prevision/solar-solaire/solarflux/sx-4-en.php"));
 | |
|         m_networkManager->get(m_networkRequest);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::autoUpdateSolarFlux()
 | |
| {
 | |
|     updateSolarFlux(false);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_downloadSolarFlux_clicked()
 | |
| {
 | |
|     updateSolarFlux(true);
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_darkTheme_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_chartsDarkTheme = checked;
 | |
| 
 | |
|     if (m_solarFluxChart) {
 | |
|         m_solarFluxChart->setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     }
 | |
| 
 | |
|     m_chart.setTheme(m_settings.m_chartsDarkTheme ? QChart::ChartThemeDark : QChart::ChartThemeLight);
 | |
|     plotChart();
 | |
|     m_settingsKeys.append("chartsDarkTheme");
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_drawSun_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_drawSunOnSkyTempChart = checked;
 | |
|     plotChart();
 | |
|     m_settingsKeys.append("drawSunOnSkyTempChart");
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::on_drawMoon_clicked(bool checked)
 | |
| {
 | |
|     m_settings.m_drawMoonOnSkyTempChart = checked;
 | |
|     plotChart();
 | |
|     m_settingsKeys.append("drawMoonOnSkyTempChart");
 | |
|     applySettings();
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::downloadFinished(const QString& filename, bool success)
 | |
| {
 | |
|     (void) filename;
 | |
| 
 | |
|     if (success) {
 | |
|         readSolarFlux();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void StarTrackerGUI::makeUIConnections()
 | |
| {
 | |
|     QObject::connect(ui->startStop, &ButtonSwitch::toggled, this, &StarTrackerGUI::on_startStop_toggled);
 | |
|     QObject::connect(ui->link, &ButtonSwitch::clicked, this, &StarTrackerGUI::on_link_clicked);
 | |
|     QObject::connect(ui->useMyPosition, &QToolButton::clicked, this, &StarTrackerGUI::on_useMyPosition_clicked);
 | |
|     QObject::connect(ui->latitude, qOverload<double>(&QDoubleSpinBox::valueChanged), this, &StarTrackerGUI::on_latitude_valueChanged);
 | |
|     QObject::connect(ui->longitude, qOverload<double>(&QDoubleSpinBox::valueChanged), this, &StarTrackerGUI::on_longitude_valueChanged);
 | |
|     QObject::connect(ui->rightAscension, &QLineEdit::editingFinished, this, &StarTrackerGUI::on_rightAscension_editingFinished);
 | |
|     QObject::connect(ui->declination, &QLineEdit::editingFinished, this, &StarTrackerGUI::on_declination_editingFinished);
 | |
|     QObject::connect(ui->azimuth, &DMSSpinBox::valueChanged, this, &StarTrackerGUI::on_azimuth_valueChanged);
 | |
|     QObject::connect(ui->elevation, &DMSSpinBox::valueChanged, this, &StarTrackerGUI::on_elevation_valueChanged);
 | |
|     QObject::connect(ui->azimuthOffset, qOverload<double>(&QDoubleSpinBox::valueChanged), this, &StarTrackerGUI::on_azimuthOffset_valueChanged);
 | |
|     QObject::connect(ui->elevationOffset, qOverload<double>(&QDoubleSpinBox::valueChanged), this, &StarTrackerGUI::on_elevationOffset_valueChanged);
 | |
|     QObject::connect(ui->galacticLatitude, &DMSSpinBox::valueChanged, this, &StarTrackerGUI::on_galacticLatitude_valueChanged);
 | |
|     QObject::connect(ui->galacticLongitude, &DMSSpinBox::valueChanged, this, &StarTrackerGUI::on_galacticLongitude_valueChanged);
 | |
|     QObject::connect(ui->frequency, qOverload<int>(&QSpinBox::valueChanged), this, &StarTrackerGUI::on_frequency_valueChanged);
 | |
|     QObject::connect(ui->beamwidth, qOverload<double>(&QDoubleSpinBox::valueChanged), this, &StarTrackerGUI::on_beamwidth_valueChanged);
 | |
|     QObject::connect(ui->target, &QComboBox::currentTextChanged, this, &StarTrackerGUI::on_target_currentTextChanged);
 | |
|     QObject::connect(ui->displaySettings, &QToolButton::clicked, this, &StarTrackerGUI::on_displaySettings_clicked);
 | |
|     QObject::connect(ui->dateTimeSelect, &QComboBox::currentTextChanged, this, &StarTrackerGUI::on_dateTimeSelect_currentTextChanged);
 | |
|     QObject::connect(ui->dateTime, &WrappingDateTimeEdit::dateTimeChanged, this, &StarTrackerGUI::on_dateTime_dateTimeChanged);
 | |
|     QObject::connect(ui->viewOnMap, &QToolButton::clicked, this, &StarTrackerGUI::on_viewOnMap_clicked);
 | |
|     QObject::connect(ui->chartSelect, qOverload<int>(&QComboBox::currentIndexChanged), this, &StarTrackerGUI::on_chartSelect_currentIndexChanged);
 | |
|     QObject::connect(ui->chartSubSelect, qOverload<int>(&QComboBox::currentIndexChanged), this, &StarTrackerGUI::on_chartSubSelect_currentIndexChanged);
 | |
|     QObject::connect(ui->downloadSolarFlux, &QToolButton::clicked, this, &StarTrackerGUI::on_downloadSolarFlux_clicked);
 | |
|     QObject::connect(ui->darkTheme, &QToolButton::clicked, this, &StarTrackerGUI::on_darkTheme_clicked);
 | |
|     QObject::connect(ui->zoomIn, &QToolButton::clicked, this, &StarTrackerGUI::on_zoomIn_clicked);
 | |
|     QObject::connect(ui->zoomOut, &QToolButton::clicked, this, &StarTrackerGUI::on_zoomOut_clicked);
 | |
|     QObject::connect(ui->addAnimationFrame, &QToolButton::clicked, this, &StarTrackerGUI::on_addAnimationFrame_clicked);
 | |
|     QObject::connect(ui->clearAnimation, &QToolButton::clicked, this, &StarTrackerGUI::on_clearAnimation_clicked);
 | |
|     QObject::connect(ui->saveAnimation, &QToolButton::clicked, this, &StarTrackerGUI::on_saveAnimation_clicked);
 | |
|     QObject::connect(ui->drawSun, &QToolButton::clicked, this, &StarTrackerGUI::on_drawSun_clicked);
 | |
|     QObject::connect(ui->drawMoon, &QToolButton::clicked, this, &StarTrackerGUI::on_drawMoon_clicked);
 | |
| }
 |