mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-31 04:50:29 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			3553 lines
		
	
	
		
			134 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			3553 lines
		
	
	
		
			134 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| // Copyright (C) 2012 maintech GmbH, Otto-Hahn-Str. 15, 97204 Hoechberg, Germany //
 | |
| // written by Christian Daniel                                                   //
 | |
| // Copyright (C) 2014 John Greb <hexameron@spam.no>                              //
 | |
| // Copyright (C) 2015-2023 Edouard Griffiths, F4EXB <f4exb06@gmail.com>          //
 | |
| // Copyright (C) 2017 Ziga S <ziga.svetina@gmail.com>                            //
 | |
| // Copyright (C) 2018 beta-tester <alpha-beta-release@gmx.net>                   //
 | |
| // Copyright (C) 2019 Vort <vvort@yandex.ru>                                     //
 | |
| // Copyright (C) 2019 Davide Gerhard <rainbow@irh.it>                            //
 | |
| // Copyright (C) 2019 Stefan Biereigel <stefan@biereigel.de>                     //
 | |
| // Copyright (C) 2020-2023 Jon Beniston, M7RCE <jon@beniston.com>                //
 | |
| // Copyright (C) 2022 CRD716 <crd716@gmail.com>                                  //
 | |
| // Copyright (C) 2023 Mohamed <mohamedadlyi@github.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 <QInputDialog>
 | |
| #include <QMessageBox>
 | |
| #include <QProgressDialog>
 | |
| #include <QLabel>
 | |
| #include <QToolButton>
 | |
| #include <QFile>
 | |
| #include <QFileInfo>
 | |
| #include <QFileDialog>
 | |
| #include <QTextStream>
 | |
| #include <QDateTime>
 | |
| #include <QSysInfo>
 | |
| #include <QKeyEvent>
 | |
| #include <QResource>
 | |
| #include <QFontDatabase>
 | |
| #include <QStandardPaths>
 | |
| #include <QDesktopServices>
 | |
| #include <QProcess>
 | |
| #include <QDirIterator>
 | |
| #include <QAction>
 | |
| #include <QMenuBar>
 | |
| #include <QStatusBar>
 | |
| #include <QScreen>
 | |
| 
 | |
| #include "device/devicegui.h"
 | |
| #include "device/deviceapi.h"
 | |
| #include "device/deviceuiset.h"
 | |
| #include "device/deviceset.h"
 | |
| #include "device/deviceenumerator.h"
 | |
| #include "channel/channelapi.h"
 | |
| #include "channel/channelgui.h"
 | |
| #include "channel/channelwebapiutils.h"
 | |
| #include "feature/featureuiset.h"
 | |
| #include "feature/featureset.h"
 | |
| #include "feature/feature.h"
 | |
| #include "feature/featuregui.h"
 | |
| #include "mainspectrum/mainspectrumgui.h"
 | |
| #include "commands/commandkeyreceiver.h"
 | |
| #include "gui/presetitem.h"
 | |
| #include "gui/pluginsdialog.h"
 | |
| #include "gui/aboutdialog.h"
 | |
| #include "gui/audiodialog.h"
 | |
| #include "gui/graphicsdialog.h"
 | |
| #include "gui/loggingdialog.h"
 | |
| #include "gui/deviceuserargsdialog.h"
 | |
| #include "gui/sdrangelsplash.h"
 | |
| #include "gui/mdiutils.h"
 | |
| #include "gui/mypositiondialog.h"
 | |
| #include "gui/fftdialog.h"
 | |
| #include "gui/fftwisdomdialog.h"
 | |
| #include "gui/workspace.h"
 | |
| #include "gui/featurepresetsdialog.h"
 | |
| #include "gui/devicesetpresetsdialog.h"
 | |
| #include "gui/commandsdialog.h"
 | |
| #include "gui/configurationsdialog.h"
 | |
| #include "gui/dialogpositioner.h"
 | |
| #include "gui/welcomedialog.h"
 | |
| #include "gui/profiledialog.h"
 | |
| #include "dsp/dspengine.h"
 | |
| #include "dsp/spectrumvis.h"
 | |
| #include "dsp/dspcommands.h"
 | |
| #include "dsp/devicesamplesource.h"
 | |
| #include "dsp/devicesamplesink.h"
 | |
| #include "dsp/devicesamplemimo.h"
 | |
| #include "dsp/dspdevicesourceengine.h"
 | |
| #include "dsp/dspdevicesinkengine.h"
 | |
| #include "dsp/dspdevicemimoengine.h"
 | |
| #include "plugin/pluginapi.h"
 | |
| #include "gui/glspectrum.h"
 | |
| #include "loggerwithfile.h"
 | |
| #include "webapi/webapirequestmapper.h"
 | |
| #include "webapi/webapiserver.h"
 | |
| #include "webapi/webapiadapter.h"
 | |
| #include "commands/command.h"
 | |
| #include "settings/serializableinterface.h"
 | |
| #ifdef ANDROID
 | |
| #include "util/android.h"
 | |
| #endif
 | |
| 
 | |
| #include "mainwindow.h"
 | |
| 
 | |
| #include <audio/audiodevicemanager.h>
 | |
| 
 | |
| //#include "ui_mainwindow.h"
 | |
| #include <QtWidgets/QApplication>
 | |
| 
 | |
| #include <string>
 | |
| #include <QDebug>
 | |
| #include <QSplashScreen>
 | |
| #include <QProgressDialog>
 | |
| 
 | |
| #include "gui/accessiblevaluedial.h"
 | |
| #include "gui/accessiblevaluedialz.h"
 | |
| 
 | |
| MainWindow *MainWindow::m_instance = nullptr;
 | |
| 
 | |
| MainWindow::MainWindow(qtwebapp::LoggerWithFile *logger, const MainParser& parser, QWidget* parent) :
 | |
| 	QMainWindow(parent),
 | |
| 	// ui(new Ui::MainWindow),
 | |
|     m_currentWorkspace(nullptr),
 | |
|     m_mainCore(MainCore::instance()),
 | |
| 	m_dspEngine(DSPEngine::instance()),
 | |
| 	m_lastEngineState(DeviceAPI::StNotStarted),
 | |
|     m_dateTimeWidget(nullptr),
 | |
|     m_showSystemWidget(nullptr),
 | |
|     m_commandKeyReceiver(nullptr),
 | |
|     m_profileDialog(nullptr),
 | |
| #if QT_CONFIG(process)
 | |
|     m_fftWisdomProcess(nullptr),
 | |
| #endif
 | |
|     m_settingsSaved(false)
 | |
| {
 | |
|     QAccessible::installFactory(AccessibleValueDial::factory);
 | |
|     QAccessible::installFactory(AccessibleValueDialZ::factory);
 | |
| 
 | |
| 	qDebug() << "MainWindow::MainWindow: start";
 | |
|     setWindowTitle("SDRangel");
 | |
| 
 | |
|     QApplication::setOverrideCursor(Qt::WaitCursor);
 | |
| 
 | |
|     m_instance = this;
 | |
|     m_mainCore->m_logger = logger;
 | |
|     m_mainCore->m_masterTabIndex = 0;
 | |
|     m_mainCore->m_mainMessageQueue = &m_inputMessageQueue;
 | |
| 	m_mainCore->m_settings.setAudioDeviceManager(m_dspEngine->getAudioDeviceManager());
 | |
| 
 | |
|     QFontDatabase::addApplicationFont(":/LiberationSans-Regular.ttf");
 | |
|     QFontDatabase::addApplicationFont(":/LiberationMono-Regular.ttf");
 | |
| 
 | |
|     QFont font("Liberation Sans");
 | |
|     font.setPointSize(9);
 | |
|     qApp->setFont(font);
 | |
| 
 | |
|     QPixmap logoPixmap(":/sdrangel_logo.png");
 | |
|     auto *splash = new SDRangelSplash(logoPixmap);
 | |
|     splash->setMessageRect(QRect(10, 80, 350, 16));
 | |
|     splash->show();
 | |
|     splash->showStatusMessage("starting...", Qt::white);
 | |
|     splash->showStatusMessage("starting...", Qt::white);
 | |
| 
 | |
|     setWindowIcon(QIcon(":/sdrangel_icon.png"));
 | |
| #ifndef ANDROID
 | |
|     // To save screen space on Android, don't have menu bar. Instead menus are accessed via toolbar button
 | |
|     createMenuBar(nullptr);
 | |
|     createStatusBar();
 | |
| #endif
 | |
| 
 | |
| #ifdef ANDROID
 | |
|     if (screen()->isLandscape(screen()->primaryOrientation())) {
 | |
|         setTabPosition(Qt::LeftDockWidgetArea, QTabWidget::West);
 | |
|     } else {
 | |
|         setTabPosition(Qt::LeftDockWidgetArea, QTabWidget::South);
 | |
|     }
 | |
| #else
 | |
|     setTabPosition(Qt::LeftDockWidgetArea, QTabWidget::West);
 | |
| #endif
 | |
|     setTabPosition(Qt::RightDockWidgetArea, QTabWidget::East);
 | |
| 	setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea);
 | |
| 	setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);
 | |
| 	setCorner(Qt::TopRightCorner, Qt::RightDockWidgetArea);
 | |
| 	setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);
 | |
| 
 | |
| 	connect(&m_inputMessageQueue, SIGNAL(messageEnqueued()), this, SLOT(handleMessages()), Qt::QueuedConnection);
 | |
| 
 | |
|     connect(screen(), &QScreen::orientationChanged, this, &MainWindow::orientationChanged);
 | |
|     #if (QT_VERSION < QT_VERSION_CHECK(6, 0, 0))
 | |
|     screen()->setOrientationUpdateMask(Qt::PortraitOrientation
 | |
|         | Qt::LandscapeOrientation
 | |
|         | Qt::InvertedPortraitOrientation
 | |
|         | Qt::InvertedLandscapeOrientation);
 | |
|     #endif
 | |
| 
 | |
| 	connect(&m_statusTimer, SIGNAL(timeout()), this, SLOT(updateStatus()));
 | |
| 	m_statusTimer.start(1000);
 | |
| 
 | |
|     splash->showStatusMessage("load settings...", Qt::white);
 | |
|     qDebug() << "MainWindow::MainWindow: load settings...";
 | |
| 
 | |
|     loadSettings();
 | |
| 
 | |
|     splash->showStatusMessage("allocate FFTs...", Qt::white);
 | |
| 
 | |
|     if (parser.getFFTWFWisdomFileName().length() != 0)
 | |
|     {
 | |
|         m_dspEngine->createFFTFactory(parser.getFFTWFWisdomFileName());
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         QString filePath = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation);
 | |
|         filePath += QDir::separator();
 | |
|         filePath += "fftw-wisdom";
 | |
|         auto fileInfo = QFileInfo(filePath);
 | |
| 
 | |
|         if (fileInfo.exists()) {
 | |
|             m_dspEngine->createFFTFactory(filePath);
 | |
|         } else {
 | |
|             m_dspEngine->createFFTFactory("");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     m_dspEngine->preAllocateFFTs();
 | |
| 
 | |
|     splash->showStatusMessage("load plugins...", Qt::white);
 | |
|     qDebug() << "MainWindow::MainWindow: load plugins...";
 | |
| 
 | |
|     m_pluginManager = new PluginManager(this);
 | |
|     m_mainCore->m_pluginManager = m_pluginManager;
 | |
|     m_pluginManager->setEnableSoapy(parser.getSoapy());
 | |
|     m_pluginManager->loadPlugins(QString("plugins"));
 | |
|     m_pluginManager->loadPluginsNonDiscoverable(m_mainCore->m_settings.getDeviceUserArgs());
 | |
| 
 | |
|     splash->showStatusMessage("Add command key receiver...", Qt::white);
 | |
| 	m_commandKeyReceiver = new CommandKeyReceiver();
 | |
| 	m_commandKeyReceiver->setRelease(true);
 | |
| 	this->installEventFilter(m_commandKeyReceiver);
 | |
| 
 | |
|     splash->showStatusMessage("Add unique feature set...", Qt::white);
 | |
|     addFeatureSet(); // Create the uniuefeature set
 | |
| 	m_apiAdapter = new WebAPIAdapter();
 | |
| 
 | |
|     splash->showStatusMessage("finishing...", Qt::white);
 | |
| 
 | |
| 	QString applicationDirPath = qApp->applicationDirPath();
 | |
| 
 | |
| #ifdef _MSC_VER
 | |
|     if (QResource::registerResource(applicationDirPath + "/sdrbase.rcc")) {
 | |
|         qDebug("MainWindow::MainWindow: registered resource file %s/%s", qPrintable(applicationDirPath), "sdrbase.rcc");
 | |
|     } else {
 | |
|         qWarning("MainWindow::MainWindow: could not register resource file %s/%s", qPrintable(applicationDirPath), "sdrbase.rcc");
 | |
|     }
 | |
| #endif
 | |
| 
 | |
| 	m_requestMapper = new WebAPIRequestMapper(this);
 | |
| 	m_requestMapper->setAdapter(m_apiAdapter);
 | |
| 	m_apiHost = parser.getServerAddress();
 | |
| 	m_apiPort = parser.getServerPort();
 | |
| 	m_apiServer = new WebAPIServer(m_apiHost, (uint16_t) m_apiPort, m_requestMapper);
 | |
| 	m_apiServer->start();
 | |
| 
 | |
|     m_dspEngine->setMIMOSupport(true);
 | |
| 
 | |
|     // Restore window size and position
 | |
|     QSettings s;
 | |
|     restoreGeometry(qUncompress(QByteArray::fromBase64(s.value("mainWindowGeometry").toByteArray())));
 | |
|     restoreState(qUncompress(QByteArray::fromBase64(s.value("mainWindowState").toByteArray())));
 | |
| 
 | |
|     // Load initial configuration
 | |
|     InitFSM *fsm = new InitFSM(this, splash, !parser.getScratch());
 | |
|     connect(fsm, &InitFSM::finished, fsm, &InitFSM::deleteLater);
 | |
|     connect(fsm, &InitFSM::finished, splash, &SDRangelSplash::deleteLater);
 | |
|     fsm->start();
 | |
| 
 | |
|     qDebug() << "MainWindow::MainWindow: end";
 | |
| }
 | |
| 
 | |
| MainWindow::~MainWindow()
 | |
| {
 | |
| 	qDebug() << "MainWindow::~MainWindow";
 | |
| 
 | |
|     m_statusTimer.stop();
 | |
|     m_apiServer->stop();
 | |
|     delete m_apiServer;
 | |
|     delete m_requestMapper;
 | |
|     delete m_apiAdapter;
 | |
| 
 | |
|     delete m_pluginManager;
 | |
| 	delete m_dateTimeWidget;
 | |
| 	delete m_showSystemWidget;
 | |
| 
 | |
|     removeAllFeatureSets();
 | |
| 
 | |
| 	delete m_commandKeyReceiver;
 | |
|     delete m_profileDialog;
 | |
| 
 | |
|     for (const auto& workspace : m_workspaces) {
 | |
|         delete workspace;
 | |
|     }
 | |
| 
 | |
| 	qDebug() << "MainWindow::~MainWindow: end";
 | |
| }
 | |
| 
 | |
| MainWindowFSM::MainWindowFSM(MainWindow *mainWindow, QObject *parent) :
 | |
|     QStateMachine(parent),
 | |
|     m_mainWindow(mainWindow),
 | |
|     m_finalState(nullptr)
 | |
| {
 | |
| }
 | |
| 
 | |
| void MainWindowFSM::createStates(int states)
 | |
| {
 | |
|     for (int i = 0; i < states - 1; i++) {
 | |
|         m_states.append(new QState());
 | |
|     }
 | |
|     m_finalState = new QFinalState();
 | |
|     for (int i = 0; i < m_states.size(); i++) {
 | |
|         addState(m_states[i]);
 | |
|     }
 | |
|     addState(m_finalState);
 | |
|     setInitialState(m_states[0]);
 | |
| }
 | |
| 
 | |
| AddSampleSourceFSM::AddSampleSourceFSM(MainWindow *mainWindow, Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex, bool loadDefaults, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_deviceWorkspace(deviceWorkspace),
 | |
|     m_spectrumWorkspace(spectrumWorkspace),
 | |
|     m_deviceIndex(deviceIndex),
 | |
|     m_loadDefaults(loadDefaults),
 | |
|     m_deviceSetIndex(-1),
 | |
|     m_deviceAPI(nullptr),
 | |
|     m_deviceUISet(nullptr)
 | |
| {
 | |
|     // Create source engine
 | |
|     addEngine();
 | |
| 
 | |
|     // Create FSM
 | |
|     createStates(3);
 | |
| 
 | |
|     m_states[0]->addTransition(m_dspDeviceSourceEngine, &DSPDeviceSourceEngine::sampleSet, m_states[1]);
 | |
|     m_states[1]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &AddSampleSourceFSM::addDevice);
 | |
|     connect(m_states[1], &QState::entered, this, &AddSampleSourceFSM::addDeviceUI);
 | |
| }
 | |
| 
 | |
| void AddSampleSourceFSM::addEngine()
 | |
| {
 | |
|     // Create the source engine
 | |
|     m_dspDeviceSourceEngine = m_mainWindow->m_dspEngine->addDeviceSourceEngine();
 | |
| }
 | |
| 
 | |
| void AddSampleSourceFSM::addDevice()
 | |
| {
 | |
|     m_deviceSetIndex = (int) m_mainWindow->m_deviceUIs.size();
 | |
|     m_mainWindow->m_mainCore->appendDeviceSet(0);
 | |
| 
 | |
|     DeviceSet *deviceSet = m_mainWindow->m_mainCore->getDeviceSets().back();
 | |
|     m_deviceUISet = new DeviceUISet(m_deviceSetIndex, deviceSet);
 | |
|     m_mainWindow->m_deviceUIs.push_back(m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceSourceEngine = m_dspDeviceSourceEngine;
 | |
|     deviceSet->m_deviceSourceEngine = m_dspDeviceSourceEngine;
 | |
|     m_deviceUISet->m_deviceSinkEngine = nullptr;
 | |
|     deviceSet->m_deviceSinkEngine = nullptr;
 | |
|     m_deviceUISet->m_deviceMIMOEngine = nullptr;
 | |
|     deviceSet->m_deviceMIMOEngine = nullptr;
 | |
| 
 | |
|     m_deviceAPI = new DeviceAPI(DeviceAPI::StreamSingleRx, m_deviceSetIndex, m_dspDeviceSourceEngine, nullptr, nullptr);
 | |
| 
 | |
|     m_deviceUISet->m_deviceAPI = m_deviceAPI;
 | |
|     deviceSet->m_deviceAPI = m_deviceAPI;
 | |
|     QList<QString> channelNames;
 | |
|     m_mainWindow->m_pluginManager->listRxChannels(channelNames);
 | |
|     m_deviceUISet->setNumberOfAvailableRxChannels(channelNames.size());
 | |
| 
 | |
|     m_dspDeviceSourceEngine->addSink(m_deviceUISet->m_spectrumVis);
 | |
| 
 | |
|     // Create a file source instance by default if requested device was not enumerated (index = -1)
 | |
|     if (m_deviceIndex < 0) {
 | |
|         m_deviceIndex = DeviceEnumerator::instance()->getFileInputDeviceIndex();
 | |
|     }
 | |
| 
 | |
|     m_mainWindow->sampleSourceCreate(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| }
 | |
| 
 | |
| void AddSampleSourceFSM::addDeviceUI()
 | |
| {
 | |
|     m_mainWindow->sampleSourceCreateUI(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceGUI->setWorkspaceIndex(m_deviceWorkspace->getIndex());
 | |
|     m_deviceUISet->m_mainSpectrumGUI->setWorkspaceIndex(m_spectrumWorkspace->getIndex());
 | |
|     MainSpectrumGUI *mainSpectrumGUI = m_deviceUISet->m_mainSpectrumGUI;
 | |
| 
 | |
|     QObject::connect(
 | |
|         mainSpectrumGUI,
 | |
|         &MainSpectrumGUI::moveToWorkspace,
 | |
|         m_mainWindow,
 | |
|         [m_mainWindow=m_mainWindow, mainSpectrumGUI](int wsIndexDest){ m_mainWindow->mainSpectrumMove(mainSpectrumGUI, wsIndexDest); }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceUISet->m_deviceGUI,
 | |
|         &DeviceGUI::addChannelEmitted,
 | |
|         m_mainWindow,
 | |
|         [m_mainWindow=m_mainWindow, m_deviceWorkspace=m_deviceWorkspace, m_deviceSetIndex=m_deviceSetIndex](int channelPluginIndex){
 | |
|             m_mainWindow->channelAddClicked(m_deviceWorkspace, m_deviceSetIndex, channelPluginIndex);
 | |
|         }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         mainSpectrumGUI,
 | |
|         &MainSpectrumGUI::requestCenterFrequency,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::mainSpectrumRequestDeviceCenterFrequency
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceAPI,
 | |
|         &DeviceAPI::stateChanged,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::deviceStateChanged
 | |
|     );
 | |
| 
 | |
|     m_deviceWorkspace->addToMdiArea(m_deviceUISet->m_deviceGUI);
 | |
|     m_spectrumWorkspace->addToMdiArea(m_deviceUISet->m_mainSpectrumGUI);
 | |
|     if (m_loadDefaults) {
 | |
|     m_mainWindow->loadDefaultPreset(m_deviceAPI->getSamplingDeviceId(), m_deviceUISet);
 | |
|     }
 | |
|     emit m_mainWindow->m_mainCore->deviceSetAdded(m_deviceSetIndex, m_deviceAPI);
 | |
| 
 | |
| #ifdef ANDROID
 | |
|     // Seemingly needed on some versions of Android, otherwise the new windows aren't always displayed??
 | |
|     m_deviceWorkspace->repaint();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| AddSampleSinkFSM::AddSampleSinkFSM(MainWindow *mainWindow, Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex, bool loadDefaults, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_deviceWorkspace(deviceWorkspace),
 | |
|     m_spectrumWorkspace(spectrumWorkspace),
 | |
|     m_deviceIndex(deviceIndex),
 | |
|     m_loadDefaults(loadDefaults),
 | |
|     m_deviceSetIndex(-1),
 | |
|     m_deviceAPI(nullptr),
 | |
|     m_deviceUISet(nullptr)
 | |
| {
 | |
|     // Create source engine
 | |
|     addEngine();
 | |
| 
 | |
|     // Create FSM
 | |
|     createStates(3);
 | |
| 
 | |
|     m_states[0]->addTransition(m_dspDeviceSinkEngine, &DSPDeviceSinkEngine::sampleSet, m_states[1]);
 | |
|     m_states[1]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &AddSampleSinkFSM::addDevice);
 | |
|     connect(m_states[1], &QState::entered, this, &AddSampleSinkFSM::addDeviceUI);
 | |
| }
 | |
| 
 | |
| void AddSampleSinkFSM::addEngine()
 | |
| {
 | |
|     // Create the source engine
 | |
|     m_dspDeviceSinkEngine = m_mainWindow->m_dspEngine->addDeviceSinkEngine();
 | |
| }
 | |
| 
 | |
| void AddSampleSinkFSM::addDevice()
 | |
| {
 | |
|     m_deviceSetIndex = (int) m_mainWindow->m_deviceUIs.size();
 | |
|     m_mainWindow->m_mainCore->appendDeviceSet(1);
 | |
| 
 | |
|     DeviceSet *deviceSet = m_mainWindow->m_mainCore->getDeviceSets().back();
 | |
|     m_deviceUISet = new DeviceUISet(m_deviceSetIndex, deviceSet);
 | |
|     m_mainWindow->m_deviceUIs.push_back(m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceSourceEngine = nullptr;
 | |
|     deviceSet->m_deviceSourceEngine = nullptr;
 | |
|     m_deviceUISet->m_deviceSinkEngine = m_dspDeviceSinkEngine;
 | |
|     deviceSet->m_deviceSinkEngine = m_dspDeviceSinkEngine;
 | |
|     m_deviceUISet->m_deviceMIMOEngine = nullptr;
 | |
|     deviceSet->m_deviceMIMOEngine = nullptr;
 | |
| 
 | |
|     m_deviceAPI = new DeviceAPI(DeviceAPI::StreamSingleTx, m_deviceSetIndex, nullptr, m_dspDeviceSinkEngine, nullptr);
 | |
| 
 | |
|     m_deviceUISet->m_deviceAPI = m_deviceAPI;
 | |
|     deviceSet->m_deviceAPI = m_deviceAPI;
 | |
|     QList<QString> channelNames;
 | |
|     m_mainWindow->m_pluginManager->listTxChannels(channelNames);
 | |
|     m_deviceUISet->setNumberOfAvailableTxChannels(channelNames.size());
 | |
| 
 | |
|     m_dspDeviceSinkEngine->addSpectrumSink(m_deviceUISet->m_spectrumVis);
 | |
|     m_deviceUISet->m_spectrum->setDisplayedStream(false, 0);
 | |
| 
 | |
|     // Create a file sink instance by default if requested device was not enumerated (index = -1)
 | |
|     if (m_deviceIndex < 0) {
 | |
|         m_deviceIndex = DeviceEnumerator::instance()->getFileOutputDeviceIndex();
 | |
|     }
 | |
| 
 | |
|     m_mainWindow->sampleSinkCreate(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| }
 | |
| 
 | |
| void AddSampleSinkFSM::addDeviceUI()
 | |
| {
 | |
|     m_mainWindow->sampleSinkCreateUI(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceGUI->setWorkspaceIndex(m_deviceWorkspace->getIndex());
 | |
|     m_deviceUISet->m_mainSpectrumGUI->setWorkspaceIndex(m_spectrumWorkspace->getIndex());
 | |
|     MainSpectrumGUI *mainSpectrumGUI = m_deviceUISet->m_mainSpectrumGUI;
 | |
| 
 | |
|     QObject::connect(
 | |
|         mainSpectrumGUI,
 | |
|         &MainSpectrumGUI::moveToWorkspace,
 | |
|         m_mainWindow,
 | |
|         [m_mainWindow=m_mainWindow, mainSpectrumGUI](int wsIndexDest){ m_mainWindow->mainSpectrumMove(mainSpectrumGUI, wsIndexDest); }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceUISet->m_deviceGUI,
 | |
|         &DeviceGUI::addChannelEmitted,
 | |
|         m_mainWindow,
 | |
|         [m_mainWindow=m_mainWindow, m_deviceWorkspace=m_deviceWorkspace, m_deviceSetIndex=m_deviceSetIndex](int channelPluginIndex){
 | |
|             m_mainWindow->channelAddClicked(m_deviceWorkspace, m_deviceSetIndex, channelPluginIndex);
 | |
|         }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         mainSpectrumGUI,
 | |
|         &MainSpectrumGUI::requestCenterFrequency,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::mainSpectrumRequestDeviceCenterFrequency
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceAPI,
 | |
|         &DeviceAPI::stateChanged,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::deviceStateChanged
 | |
|     );
 | |
| 
 | |
|     m_deviceWorkspace->addToMdiArea(m_deviceUISet->m_deviceGUI);
 | |
|     m_spectrumWorkspace->addToMdiArea(m_deviceUISet->m_mainSpectrumGUI);
 | |
|     if (m_loadDefaults) {
 | |
|     m_mainWindow->loadDefaultPreset(m_deviceAPI->getSamplingDeviceId(), m_deviceUISet);
 | |
|     }
 | |
|     emit m_mainWindow->m_mainCore->deviceSetAdded(m_deviceSetIndex, m_deviceAPI);
 | |
| 
 | |
| #ifdef ANDROID
 | |
|     // Seemingly needed on some versions of Android, otherwise the new windows aren't always displayed??
 | |
|     m_deviceWorkspace->repaint();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| AddSampleMIMOFSM::AddSampleMIMOFSM(MainWindow *mainWindow, Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex, bool loadDefaults, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_deviceWorkspace(deviceWorkspace),
 | |
|     m_spectrumWorkspace(spectrumWorkspace),
 | |
|     m_deviceIndex(deviceIndex),
 | |
|     m_loadDefaults(loadDefaults),
 | |
|     m_deviceSetIndex(-1),
 | |
|     m_deviceAPI(nullptr),
 | |
|     m_deviceUISet(nullptr)
 | |
| {
 | |
|     // Create source engine
 | |
|     addEngine();
 | |
| 
 | |
|     // Create FSM
 | |
|     createStates(3);
 | |
| 
 | |
|     m_states[0]->addTransition(m_dspDeviceMIMOEngine, &DSPDeviceMIMOEngine::sampleSet, m_states[1]);
 | |
|     m_states[1]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &AddSampleMIMOFSM::addDevice);
 | |
|     connect(m_states[1], &QState::entered, this, &AddSampleMIMOFSM::addDeviceUI);
 | |
| }
 | |
| 
 | |
| void AddSampleMIMOFSM::addEngine()
 | |
| {
 | |
|     // Create the source engine
 | |
|     m_dspDeviceMIMOEngine = m_mainWindow->m_dspEngine->addDeviceMIMOEngine();
 | |
| }
 | |
| 
 | |
| void AddSampleMIMOFSM::addDevice()
 | |
| {
 | |
|     m_deviceSetIndex = (int) m_mainWindow->m_deviceUIs.size();
 | |
|     m_mainWindow->m_mainCore->appendDeviceSet(2);
 | |
| 
 | |
|     DeviceSet *deviceSet = m_mainWindow->m_mainCore->getDeviceSets().back();
 | |
|     m_deviceUISet = new DeviceUISet(m_deviceSetIndex, deviceSet);
 | |
|     m_mainWindow->m_deviceUIs.push_back(m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceSourceEngine = nullptr;
 | |
|     deviceSet->m_deviceSourceEngine = nullptr;
 | |
|     m_deviceUISet->m_deviceSinkEngine = nullptr;
 | |
|     deviceSet->m_deviceSinkEngine = nullptr;
 | |
|     m_deviceUISet->m_deviceMIMOEngine = m_dspDeviceMIMOEngine;
 | |
|     deviceSet->m_deviceMIMOEngine = m_dspDeviceMIMOEngine;
 | |
| 
 | |
|     m_deviceAPI = new DeviceAPI(DeviceAPI::StreamMIMO, m_deviceSetIndex, nullptr, nullptr, m_dspDeviceMIMOEngine);
 | |
| 
 | |
|     m_deviceUISet->m_deviceAPI = m_deviceAPI;
 | |
|     deviceSet->m_deviceAPI = m_deviceAPI;
 | |
|     QList<QString> mimoChannelNames;
 | |
|     m_mainWindow->m_pluginManager->listMIMOChannels(mimoChannelNames);
 | |
|     m_deviceUISet->setNumberOfAvailableMIMOChannels(mimoChannelNames.size());
 | |
|     // Add Rx channels
 | |
|     QList<QString> rxChannelNames;
 | |
|     m_mainWindow->m_pluginManager->listRxChannels(rxChannelNames);
 | |
|     m_deviceUISet->setNumberOfAvailableRxChannels(rxChannelNames.size());
 | |
|     // Add Tx channels
 | |
|     QList<QString> txChannelNames;
 | |
|     m_mainWindow->m_pluginManager->listTxChannels(txChannelNames);
 | |
|     m_deviceUISet->setNumberOfAvailableTxChannels(txChannelNames.size());
 | |
| 
 | |
|     m_dspDeviceMIMOEngine->addSpectrumSink(m_deviceUISet->m_spectrumVis);
 | |
| 
 | |
|     // Create a Test MIMO instance by default if requested device was not enumerated (index = -1)
 | |
|     if (m_deviceIndex < 0) {
 | |
|         m_deviceIndex = DeviceEnumerator::instance()->getTestMIMODeviceIndex();
 | |
|     }
 | |
| 
 | |
|     m_mainWindow->sampleMIMOCreate(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| }
 | |
| 
 | |
| void AddSampleMIMOFSM::addDeviceUI()
 | |
| {
 | |
|     m_mainWindow->sampleMIMOCreateUI(m_deviceSetIndex, m_deviceIndex, m_deviceUISet);
 | |
| 
 | |
|     m_deviceUISet->m_deviceGUI->setWorkspaceIndex(m_deviceWorkspace->getIndex());
 | |
|     m_deviceUISet->m_mainSpectrumGUI->setWorkspaceIndex(m_spectrumWorkspace->getIndex());
 | |
|     MainSpectrumGUI *mainSpectrumGUI = m_deviceUISet->m_mainSpectrumGUI;
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceUISet->m_deviceGUI,
 | |
|         &DeviceGUI::addChannelEmitted,
 | |
|         m_mainWindow,
 | |
|         [m_mainWindow=m_mainWindow, m_deviceWorkspace=m_deviceWorkspace, m_deviceSetIndex=m_deviceSetIndex](int channelPluginIndex){
 | |
|             m_mainWindow->channelAddClicked(m_deviceWorkspace, m_deviceSetIndex, channelPluginIndex);
 | |
|         }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         mainSpectrumGUI,
 | |
|         &MainSpectrumGUI::requestCenterFrequency,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::mainSpectrumRequestDeviceCenterFrequency
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_deviceAPI,
 | |
|         &DeviceAPI::stateChanged,
 | |
|         m_mainWindow,
 | |
|         &MainWindow::deviceStateChanged
 | |
|     );
 | |
| 
 | |
|     m_deviceWorkspace->addToMdiArea(m_deviceUISet->m_deviceGUI);
 | |
|     m_spectrumWorkspace->addToMdiArea(m_deviceUISet->m_mainSpectrumGUI);
 | |
|     if (m_loadDefaults) {
 | |
|     m_mainWindow->loadDefaultPreset(m_deviceAPI->getSamplingDeviceId(), m_deviceUISet);
 | |
|     }
 | |
|     emit m_mainWindow->m_mainCore->deviceSetAdded(m_deviceSetIndex, m_deviceAPI);
 | |
| 
 | |
| #ifdef ANDROID
 | |
|     // Seemingly needed on some versions of Android, otherwise the new windows aren't always displayed??
 | |
|     m_deviceWorkspace->repaint();
 | |
| #endif
 | |
| }
 | |
| 
 | |
| RemoveDeviceSetFSM::RemoveDeviceSetFSM(MainWindow *mainWindow, int deviceSetIndex, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_deviceSetIndex(deviceSetIndex),
 | |
|     m_deviceSourceEngine(nullptr),
 | |
|     m_deviceSinkEngine(nullptr),
 | |
|     m_deviceMIMOEngine(nullptr),
 | |
|     m_t1(nullptr),
 | |
|     m_t2(nullptr)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(6);
 | |
| 
 | |
|     m_deviceUISet = m_mainWindow->m_deviceUIs[m_deviceSetIndex];
 | |
|     if (m_deviceUISet->m_deviceSourceEngine)
 | |
|     {
 | |
|         m_deviceSourceEngine = m_deviceUISet->m_deviceSourceEngine;
 | |
|         m_t1 = new QSignalTransition(m_deviceSourceEngine, &DSPDeviceSourceEngine::acquistionStopped, m_states[0]);
 | |
|         m_t1->setTargetState(m_states[1]);
 | |
|         m_t2 = new QSignalTransition(m_deviceSourceEngine, &DSPDeviceSourceEngine::sinkRemoved, m_states[1]);
 | |
|         m_t2->setTargetState(m_states[2]);
 | |
|     }
 | |
|     else if (m_deviceUISet->m_deviceSinkEngine)
 | |
|     {
 | |
|         m_deviceSinkEngine = m_deviceUISet->m_deviceSinkEngine;
 | |
|         m_t1 = new QSignalTransition(m_deviceSinkEngine, &DSPDeviceSinkEngine::generationStopped, m_states[0]);
 | |
|         m_t1->setTargetState(m_states[1]);
 | |
|         m_t2 = new QSignalTransition(m_deviceSinkEngine, &DSPDeviceSinkEngine::spectrumSinkRemoved, m_states[1]);
 | |
|         m_t2->setTargetState(m_states[2]);
 | |
|     }
 | |
|     else if (m_deviceUISet->m_deviceMIMOEngine)
 | |
|     {
 | |
|         m_deviceMIMOEngine = m_deviceUISet->m_deviceMIMOEngine;
 | |
|         m_t1 = new QSignalTransition(m_deviceMIMOEngine, &DSPDeviceMIMOEngine::acquisitionStopped, m_states[0]);
 | |
|         m_t1->setTargetState(m_states[1]);
 | |
|         m_t2 = new QSignalTransition(m_deviceMIMOEngine, &DSPDeviceMIMOEngine::spectrumSinkRemoved, m_states[1]);
 | |
|         m_t2->setTargetState(m_states[2]);
 | |
|     }
 | |
|     m_states[2]->addTransition(m_states[3]);
 | |
|     m_states[3]->addTransition(m_mainWindow, &MainWindow::engineStopped, m_states[4]);
 | |
|     m_states[4]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &RemoveDeviceSetFSM::stopAcquisition);
 | |
|     connect(m_states[1], &QState::entered, this, &RemoveDeviceSetFSM::removeSink);
 | |
|     connect(m_states[2], &QState::entered, this, &RemoveDeviceSetFSM::removeUI);
 | |
|     connect(m_states[3], &QState::entered, this, &RemoveDeviceSetFSM::stopEngine);
 | |
|     connect(m_states[4], &QState::entered, this, &RemoveDeviceSetFSM::removeDeviceSet);
 | |
| }
 | |
| 
 | |
| void RemoveDeviceSetFSM::stopAcquisition()
 | |
| {
 | |
|     qDebug() << "RemoveDeviceSetFSM::stopAcquisition";
 | |
|     if (m_deviceSourceEngine)
 | |
|     {
 | |
|         m_deviceSourceEngine->stopAcquistion();
 | |
|     }
 | |
|     else if (m_deviceSinkEngine)
 | |
|     {
 | |
| 	    m_deviceSinkEngine->stopGeneration();
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         m_deviceMIMOEngine->stopProcess(1); // Tx side
 | |
|         m_deviceMIMOEngine->stopProcess(0); // Rx side
 | |
|     }
 | |
| }
 | |
| 
 | |
| void RemoveDeviceSetFSM::removeSink()
 | |
| {
 | |
|     qDebug() << "RemoveDeviceSetFSM::removeSink";
 | |
|     if (m_deviceSourceEngine) {
 | |
|         m_deviceSourceEngine->removeSink(m_deviceUISet->m_spectrumVis, false);
 | |
|     } else if (m_deviceSinkEngine) {
 | |
|         m_deviceSinkEngine->removeSpectrumSink(m_deviceUISet->m_spectrumVis);
 | |
|     } else {
 | |
|         m_deviceMIMOEngine->removeSpectrumSink(m_deviceUISet->m_spectrumVis);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void RemoveDeviceSetFSM::removeUI()
 | |
| {
 | |
|     qDebug() << "RemoveDeviceSetFSM::removeUI";
 | |
| 
 | |
|     // Remove transitions, as we will delete the object (DSPDevice*Engine) that is the source of these signals
 | |
|     m_states[0]->removeTransition(m_t1);
 | |
|     delete m_t1;
 | |
|     m_t1 = nullptr;
 | |
|     m_states[1]->removeTransition(m_t2);
 | |
|     delete m_t2;
 | |
|     m_t2 = nullptr;
 | |
| 
 | |
|     m_deviceUISet->freeChannels();      // destroys the channel instances
 | |
|     if (m_deviceSourceEngine) {
 | |
|         m_deviceUISet->m_deviceAPI->getSampleSource()->setMessageQueueToGUI(nullptr); // have source stop sending messages to the GUI
 | |
|     } else if (m_deviceSinkEngine) {
 | |
|         m_deviceUISet->m_deviceAPI->getSampleSink()->setMessageQueueToGUI(nullptr); // have sink stop sending messages to the GUI
 | |
|     } else {
 | |
|         m_deviceUISet->m_deviceAPI->getSampleMIMO()->setMessageQueueToGUI(nullptr); // have sink stop sending messages to the GUI
 | |
|     }
 | |
|     delete m_deviceUISet->m_deviceGUI;
 | |
|     m_deviceUISet->m_deviceAPI->resetSamplingDeviceId();
 | |
|     if (!m_deviceMIMOEngine) {
 | |
|         m_deviceUISet->m_deviceAPI->clearBuddiesLists(); // clear old API buddies lists
 | |
|     }
 | |
| }
 | |
| 
 | |
| void RemoveDeviceSetFSM::stopEngine()
 | |
| {
 | |
|     qDebug() << "RemoveDeviceSetFSM::stopEngine";
 | |
|     QThread *thread = m_mainWindow->m_dspEngine->getDeviceEngineThread(m_deviceSetIndex);
 | |
| 
 | |
|     if (thread)
 | |
|     {
 | |
|         bool finished = thread->isFinished();
 | |
| 
 | |
|         if (!finished) {
 | |
|             connect(thread, &QThread::finished, m_mainWindow, &MainWindow::engineStopped);
 | |
|         }
 | |
|         m_mainWindow->m_dspEngine->removeDeviceEngineAt(m_deviceSetIndex);
 | |
|         if (finished) {
 | |
|             emit m_mainWindow->engineStopped();
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         emit m_mainWindow->engineStopped();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void RemoveDeviceSetFSM::removeDeviceSet()
 | |
| {
 | |
|     if (m_deviceSourceEngine) {
 | |
|         DeviceEnumerator::instance()->removeRxSelection(m_deviceSetIndex);
 | |
|     } else if (m_deviceSinkEngine) {
 | |
|         DeviceEnumerator::instance()->removeTxSelection(m_deviceSetIndex);
 | |
|     } else {
 | |
|         DeviceEnumerator::instance()->removeMIMOSelection(m_deviceSetIndex);
 | |
|     }
 | |
| 
 | |
|     DeviceAPI *deviceAPI = m_deviceUISet->m_deviceAPI;
 | |
|     delete m_deviceUISet;
 | |
|     if (m_deviceSourceEngine)
 | |
|     {
 | |
|         delete deviceAPI->getSampleSource();
 | |
|         delete m_deviceSourceEngine;
 | |
|     }
 | |
|     else if (m_deviceSinkEngine)
 | |
|     {
 | |
|         delete deviceAPI->getSampleSink();
 | |
|         delete m_deviceSinkEngine;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         delete deviceAPI->getSampleMIMO();
 | |
|         delete m_deviceMIMOEngine;
 | |
|     }
 | |
|     delete deviceAPI;
 | |
| 
 | |
|     m_mainWindow->m_deviceUIs.erase(m_mainWindow->m_deviceUIs.begin() + m_deviceSetIndex);
 | |
|     m_mainWindow->m_mainCore->removeDeviceSet(m_deviceSetIndex);
 | |
|     DeviceEnumerator::instance()->renumeratetabIndex(m_deviceSetIndex);
 | |
| 
 | |
|     // Renumerate
 | |
|     for (int i = 0; i < (int) m_mainWindow->m_deviceUIs.size(); i++)
 | |
|     {
 | |
|         DeviceUISet *xDeviceUISet = m_mainWindow->m_deviceUIs[i];
 | |
|         xDeviceUISet->setIndex(i);
 | |
|         const DeviceGUI *deviceGUI = m_mainWindow->m_deviceUIs[i]->m_deviceGUI;
 | |
|         Workspace *deviceWorkspace = m_mainWindow->m_workspaces[deviceGUI->getWorkspaceIndex()];
 | |
| 
 | |
|         QObject::disconnect(
 | |
|             deviceGUI,
 | |
|             &DeviceGUI::addChannelEmitted,
 | |
|             this,
 | |
|             nullptr
 | |
|         );
 | |
|         QObject::connect(
 | |
|             deviceGUI,
 | |
|             &DeviceGUI::addChannelEmitted,
 | |
|             this,
 | |
|             [this, deviceWorkspace, i](int channelPluginIndex){ m_mainWindow->channelAddClicked(deviceWorkspace, i, channelPluginIndex); }
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     emit m_mainWindow->m_mainCore->deviceSetRemoved(m_deviceSetIndex);
 | |
| }
 | |
| 
 | |
| RemoveAllDeviceSetsFSM::RemoveAllDeviceSetsFSM(MainWindow *mainWindow, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(2);
 | |
| 
 | |
|     m_states[0]->addTransition(m_mainWindow->m_mainCore, &MainCore::deviceSetRemoved, m_states[0]);
 | |
|     m_states[0]->addTransition(m_mainWindow, &MainWindow::allDeviceSetsRemoved, m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &RemoveAllDeviceSetsFSM::removeNext);
 | |
| }
 | |
| 
 | |
| void RemoveAllDeviceSetsFSM::removeNext()
 | |
| {
 | |
|     qDebug() << "RemoveAllDeviceSetsFSM::removeNext";
 | |
|     if (m_mainWindow->m_deviceUIs.size() > 0) {
 | |
|         m_mainWindow->removeDeviceSet(m_mainWindow->m_deviceUIs.size() - 1);
 | |
|     } else {
 | |
|         emit m_mainWindow->allDeviceSetsRemoved();
 | |
|     }
 | |
| }
 | |
| 
 | |
| RemoveAllWorkspacesFSM::RemoveAllWorkspacesFSM(MainWindow *mainWindow, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(3);
 | |
| 
 | |
|     m_removeAllDeviceSetsFSM = new RemoveAllDeviceSetsFSM(m_mainWindow, this);
 | |
| 
 | |
|     m_states[0]->addTransition(m_removeAllDeviceSetsFSM, &RemoveAllDeviceSetsFSM::finished, m_states[1]);
 | |
|     m_states[1]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &RemoveAllWorkspacesFSM::removeDeviceSets);
 | |
|     connect(m_states[1], &QState::entered, this, &RemoveAllWorkspacesFSM::removeWorkspaces);
 | |
| }
 | |
| 
 | |
| void RemoveAllWorkspacesFSM::removeDeviceSets()
 | |
| {
 | |
|     qDebug() << "RemoveAllWorkspacesFSM::removeDeviceSets";
 | |
|     m_removeAllDeviceSetsFSM->start();
 | |
| }
 | |
| 
 | |
| void RemoveAllWorkspacesFSM::removeWorkspaces()
 | |
| {
 | |
|     qDebug() << "RemoveAllWorkspacesFSM::removeWorkspaces";
 | |
|     // Features
 | |
|     m_mainWindow->m_featureUIs[0]->freeFeatures();
 | |
|     // Workspaces
 | |
|     for (const auto& workspace : m_mainWindow->m_workspaces) {
 | |
|         delete workspace;
 | |
|     }
 | |
|     m_mainWindow->m_workspaces.clear();
 | |
|     qDebug() << "RemoveAllWorkspacesFSM::removeWorkspaces done";
 | |
| }
 | |
| 
 | |
| LoadConfigurationFSM::LoadConfigurationFSM(MainWindow *mainWindow, const Configuration *configuration, QProgressDialog *waitBox, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_configuration(configuration),
 | |
|     m_waitBox(waitBox)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(7);
 | |
| 
 | |
|     m_removeAllWorkspacesFSM = new RemoveAllWorkspacesFSM(m_mainWindow, this);
 | |
| 
 | |
|     m_states[0]->addTransition(m_removeAllWorkspacesFSM, &RemoveAllWorkspacesFSM::finished, m_states[1]);
 | |
|     m_states[1]->addTransition(m_states[2]);
 | |
|     m_states[2]->addTransition(m_mainWindow, &MainWindow::allDeviceSetsAdded, m_states[3]);
 | |
|     m_states[3]->addTransition(m_states[4]);
 | |
|     m_states[4]->addTransition(m_states[5]);
 | |
|     m_states[5]->addTransition(m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &LoadConfigurationFSM::clearWorkspace);
 | |
|     connect(m_states[1], &QState::entered, this, &LoadConfigurationFSM::createWorkspaces);
 | |
|     connect(m_states[2], &QState::entered, this, &LoadConfigurationFSM::loadDeviceSets);
 | |
|     connect(m_states[3], &QState::entered, this, &LoadConfigurationFSM::loadDeviceSetSettings);
 | |
|     connect(m_states[4], &QState::entered, this, &LoadConfigurationFSM::loadFeatureSets);
 | |
|     connect(m_states[5], &QState::entered, this, &LoadConfigurationFSM::restoreGeometry);
 | |
| }
 | |
| 
 | |
| void LoadConfigurationFSM::clearWorkspace()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::clearWorkspace";
 | |
| 
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setLabelText("Deleting existing...");
 | |
|         m_waitBox->setValue(5);
 | |
|     }
 | |
| 
 | |
|     m_removeAllWorkspacesFSM->start();
 | |
| }
 | |
| 
 | |
| void LoadConfigurationFSM::createWorkspaces()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::createWorkspaces";
 | |
| 
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setLabelText("Creating workspaces...");
 | |
|         m_waitBox->setValue(20);
 | |
|     }
 | |
| 
 | |
|     // Workspaces
 | |
|     for (int i = 0; i < m_configuration->getNumberOfWorkspaceGeometries(); i++)
 | |
|     {
 | |
|         m_mainWindow->addWorkspace();
 | |
|         m_mainWindow->m_workspaces[i]->setAutoStackOption(m_configuration->getWorkspaceAutoStackOptions()[i]);
 | |
|         m_mainWindow->m_workspaces[i]->setTabSubWindowsOption(m_configuration->getWorkspaceTabSubWindowsOptions()[i]);
 | |
|     }
 | |
|     }
 | |
| 
 | |
| void LoadConfigurationFSM::loadDeviceSets()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::loadDeviceSets";
 | |
| 
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setLabelText("Loading device sets...");
 | |
|         m_waitBox->setValue(25);
 | |
|     }
 | |
| 
 | |
|     const QList<Preset>& deviceSetPresets = m_configuration->getDeviceSetPresets();
 | |
| 
 | |
|     // State machine that runs all of the individual AddSampleSourceFSM, AddSampleSinkFSM, AddSampleMIMOFSMs
 | |
|     QStateMachine *m_addDevicesFSM = new QStateMachine();
 | |
|     QState *sInitial = nullptr;
 | |
|     QState *sPrev = nullptr;
 | |
|     QFinalState *sFinal = new QFinalState();
 | |
| 
 | |
|     connect(m_addDevicesFSM, &QStateMachine::finished, this, [=](){emit m_mainWindow->allDeviceSetsAdded();});
 | |
|     connect(m_addDevicesFSM, &QStateMachine::finished, m_addDevicesFSM, &QStateMachine::deleteLater);
 | |
| 
 | |
|     for (const auto& deviceSetPreset : deviceSetPresets)
 | |
|     {
 | |
|         MainWindowFSM *fsm = nullptr;
 | |
|         int deviceWorkspaceIndex = deviceSetPreset.getDeviceWorkspaceIndex() < m_mainWindow->m_workspaces.size() ?
 | |
|             deviceSetPreset.getDeviceWorkspaceIndex() :
 | |
|             0;
 | |
|         int spectrumWorkspaceIndex = deviceSetPreset.getSpectrumWorkspaceIndex() < m_mainWindow->m_workspaces.size() ?
 | |
|             deviceSetPreset.getSpectrumWorkspaceIndex() :
 | |
|             deviceWorkspaceIndex;
 | |
| 
 | |
|         if (deviceSetPreset.isSourcePreset())
 | |
|         {
 | |
|             int bestDeviceIndex = DeviceEnumerator::instance()->getBestRxSamplingDeviceIndex(
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceId,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSerial,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSequence,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceItemIndex
 | |
|             );
 | |
|             qDebug("MainWindow::loadConfiguration: add source %s in workspace %d spectrum in %d",
 | |
|                 qPrintable(deviceSetPreset.getSelectedDevice().m_deviceId),
 | |
|                 deviceSetPreset.getDeviceWorkspaceIndex(),
 | |
|                 deviceSetPreset.getSpectrumWorkspaceIndex());
 | |
| 
 | |
|             fsm = new AddSampleSourceFSM(m_mainWindow,
 | |
|                 m_mainWindow->m_workspaces[deviceWorkspaceIndex],
 | |
|                 m_mainWindow->m_workspaces[spectrumWorkspaceIndex],
 | |
|                 bestDeviceIndex, false, m_addDevicesFSM);
 | |
|         }
 | |
|         else if (deviceSetPreset.isSinkPreset())
 | |
|         {
 | |
|             int bestDeviceIndex = DeviceEnumerator::instance()->getBestTxSamplingDeviceIndex(
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceId,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSerial,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSequence,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceItemIndex
 | |
|             );
 | |
|             qDebug("MainWindow::loadConfiguration: add sink %s in workspace %d spectrum in %d",
 | |
|                 qPrintable(deviceSetPreset.getSelectedDevice().m_deviceId),
 | |
|                 deviceSetPreset.getDeviceWorkspaceIndex(),
 | |
|                 deviceSetPreset.getSpectrumWorkspaceIndex());
 | |
| 
 | |
|             fsm = new AddSampleSinkFSM(m_mainWindow,
 | |
|                 m_mainWindow->m_workspaces[deviceWorkspaceIndex],
 | |
|                 m_mainWindow->m_workspaces[spectrumWorkspaceIndex],
 | |
|                 bestDeviceIndex, false, m_addDevicesFSM);
 | |
|         }
 | |
|         else if (deviceSetPreset.isMIMOPreset())
 | |
|         {
 | |
|             int bestDeviceIndex = DeviceEnumerator::instance()->getBestMIMOSamplingDeviceIndex(
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceId,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSerial,
 | |
|                 deviceSetPreset.getSelectedDevice().m_deviceSequence
 | |
|             );
 | |
|             qDebug("MainWindow::loadConfiguration: add MIMO %s in workspace %d spectrum in %d",
 | |
|                 qPrintable(deviceSetPreset.getSelectedDevice().m_deviceId),
 | |
|                 deviceSetPreset.getDeviceWorkspaceIndex(),
 | |
|                 deviceSetPreset.getSpectrumWorkspaceIndex());
 | |
| 
 | |
|             fsm = new AddSampleMIMOFSM(m_mainWindow,
 | |
|                 m_mainWindow->m_workspaces[deviceWorkspaceIndex],
 | |
|                 m_mainWindow->m_workspaces[spectrumWorkspaceIndex],
 | |
|                 bestDeviceIndex, false, m_addDevicesFSM);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             qDebug() << "MainWindow::loadConfiguration: Unknown preset type: " << deviceSetPreset.getPresetType();
 | |
|         }
 | |
| 
 | |
|         // Chain the FSMs together
 | |
|         if (fsm)
 | |
|         {
 | |
|             m_addDevicesFSM->addState(fsm);
 | |
| 
 | |
|             if (!sInitial) {
 | |
|                 sInitial = fsm;
 | |
|             }
 | |
|             if (sPrev) {
 | |
|                 sPrev->addTransition(sPrev, &QState::finished, fsm);
 | |
|             }
 | |
|             sPrev = fsm;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     m_addDevicesFSM->addState(sFinal);
 | |
|     if (sPrev)
 | |
|     {
 | |
|         m_addDevicesFSM->setInitialState(sInitial);
 | |
|         sPrev->addTransition(sPrev, &QState::finished, sFinal);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         m_addDevicesFSM->setInitialState(sFinal);
 | |
|     }
 | |
|     // Run all the FSMs
 | |
|     m_addDevicesFSM->start();
 | |
| }
 | |
| 
 | |
| void LoadConfigurationFSM::loadDeviceSetSettings()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::loadDeviceSetSettings";
 | |
| 
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setLabelText("Loading device set settings...");
 | |
|         m_waitBox->setValue(70);
 | |
|     }
 | |
| 
 | |
|     const QList<Preset>& deviceSetPresets = m_configuration->getDeviceSetPresets();
 | |
| 
 | |
|     int deviceSetIndex = 0;
 | |
|     for (const auto& deviceSetPreset : deviceSetPresets)
 | |
|     {
 | |
|         if (((int) m_mainWindow->m_deviceUIs.size()) > deviceSetIndex)
 | |
|         {
 | |
|             MDIUtils::restoreMDIGeometry(m_mainWindow->m_deviceUIs[deviceSetIndex]->m_deviceGUI, deviceSetPreset.getDeviceGeometry());
 | |
|             MDIUtils::restoreMDIGeometry(m_mainWindow->m_deviceUIs[deviceSetIndex]->m_mainSpectrumGUI, deviceSetPreset.getSpectrumGeometry());
 | |
|             m_mainWindow->m_deviceUIs[deviceSetIndex]->loadDeviceSetSettings(&deviceSetPreset, m_mainWindow->m_pluginManager->getPluginAPI(), &m_mainWindow->m_workspaces, nullptr);
 | |
|             deviceSetIndex++;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void LoadConfigurationFSM::loadFeatureSets()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::loadFeatureSets";
 | |
| 
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setLabelText("Loading feature sets...");
 | |
|         m_waitBox->setValue(75);
 | |
|     }
 | |
| 
 | |
|     m_mainWindow->m_featureUIs[0]->loadFeatureSetSettings(
 | |
|         &m_configuration->getFeatureSetPreset(),
 | |
|         m_mainWindow->m_pluginManager->getPluginAPI(),
 | |
|         m_mainWindow->m_apiAdapter,
 | |
|         &m_mainWindow->m_workspaces,
 | |
|         nullptr
 | |
|     );
 | |
| 
 | |
|     for (int i = 0; i < m_mainWindow->m_featureUIs[0]->getNumberOfFeatures(); i++)
 | |
|     {
 | |
|         FeatureGUI *gui = m_mainWindow->m_featureUIs[0]->getFeatureGuiAt(i);
 | |
|         QObject::connect(
 | |
|             gui,
 | |
|             &FeatureGUI::moveToWorkspace,
 | |
|             this,
 | |
|             [m_mainWindow=m_mainWindow, gui](int wsIndexDest){ m_mainWindow->featureMove(gui, wsIndexDest); }
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void LoadConfigurationFSM::restoreGeometry()
 | |
| {
 | |
|     qDebug() << "LoadConfigurationFSM::restoreGeometry";
 | |
| 
 | |
|     // Lastly restore workspaces geometry
 | |
|     if (m_waitBox)
 | |
|     {
 | |
|         m_waitBox->setValue(90);
 | |
|         m_waitBox->setLabelText("Finalizing...");
 | |
|     }
 | |
| 
 | |
|     for (int i = 0; i < m_configuration->getNumberOfWorkspaceGeometries(); i++)
 | |
|     {
 | |
|         m_mainWindow->m_workspaces[i]->restoreGeometry(m_configuration->getWorkspaceGeometries()[i]);
 | |
|         m_mainWindow->m_workspaces[i]->restoreGeometry(m_configuration->getWorkspaceGeometries()[i]);
 | |
|         m_mainWindow->m_workspaces[i]->adjustSubWindowsAfterRestore();
 | |
| #ifdef ANDROID
 | |
|         // On Android, workspaces seem to be restored to 0,20, rather than 0,0
 | |
|         m_mainWindow->m_workspaces[i]->move(m_mainWindow->m_workspaces[i]->pos().x(), 0);
 | |
|         // Need to call updateGeometry, otherwise sometimes the layout is corrupted
 | |
|         m_mainWindow->m_workspaces[i]->updateGeometry();
 | |
| #endif
 | |
|         if (m_mainWindow->m_workspaces[i]->getAutoStackOption()) {
 | |
|             m_mainWindow->m_workspaces[i]->layoutSubWindows();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (m_waitBox) {
 | |
|         m_waitBox->setValue(100);
 | |
|     }
 | |
| }
 | |
| 
 | |
| InitFSM::InitFSM(MainWindow *mainWindow, SDRangelSplash *splash, bool loadDefault, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent),
 | |
|     m_splash(splash)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(2);
 | |
| 
 | |
|     if (loadDefault)
 | |
|     {
 | |
|     m_loadConfigurationFSM = new LoadConfigurationFSM(m_mainWindow, m_mainWindow->m_mainCore->getMutableSettings().getWorkingConfiguration(), nullptr, this);
 | |
| 
 | |
|     m_states[0]->addTransition(m_loadConfigurationFSM, &LoadConfigurationFSM::finished, m_finalState);
 | |
| 
 | |
|     connect(m_states[0], &QState::entered, this, &InitFSM::loadDefaultConfiguration);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         m_states[0]->addTransition(m_finalState);
 | |
|     }
 | |
|     connect(m_finalState, &QState::entered, this, &InitFSM::showDefaultConfigurations);
 | |
| }
 | |
| 
 | |
| void InitFSM::loadDefaultConfiguration()
 | |
| {
 | |
|     m_splash->showStatusMessage("load current configuration...", Qt::white);
 | |
|     qDebug() << "MainWindow::MainWindow: load current configuration...";
 | |
|     m_loadConfigurationFSM->start();
 | |
| }
 | |
| 
 | |
| void InitFSM::showDefaultConfigurations()
 | |
| {
 | |
|     QApplication::restoreOverrideCursor();
 | |
|     //m_mainWindow->show();
 | |
| 
 | |
|     if (m_mainWindow->m_workspaces.size() == 0)
 | |
|     {
 | |
|         qDebug() << "MainWindow::MainWindow: no or empty current configuration, creating empty workspace...";
 | |
|         m_mainWindow->addWorkspace();
 | |
| 
 | |
|         // If no configurations, load some basic examples
 | |
|         if (m_mainWindow->m_mainCore->getMutableSettings().getConfigurations()->size() == 0) {
 | |
|             m_mainWindow->loadDefaultConfigurations();
 | |
|         }
 | |
| #if defined(ANDROID) || defined(__EMSCRIPTEN__)
 | |
|         // Show welcome dialog
 | |
|         m_mainWindow->on_action_Welcome_triggered();
 | |
| #endif
 | |
|         // Show configurations
 | |
|         m_mainWindow->openConfigurationDialog(true);
 | |
| }
 | |
| }
 | |
| 
 | |
| CloseFSM::CloseFSM(MainWindow *mainWindow, QObject *parent) :
 | |
|     MainWindowFSM(mainWindow, parent)
 | |
| {
 | |
|     // Create FSM
 | |
|     createStates(2);
 | |
| 
 | |
|     m_states[0]->addTransition(m_mainWindow, &MainWindow::allDeviceSetsRemoved, m_finalState);
 | |
| 
 | |
|     connect(this, &QStateMachine::started, this, &CloseFSM::on_started);
 | |
|     connect(this, &QStateMachine::finished, this, &CloseFSM::on_finished);
 | |
| }
 | |
| 
 | |
| void CloseFSM::on_started()
 | |
| {
 | |
|     m_mainWindow->removeAllDeviceSets();
 | |
| }
 | |
| 
 | |
| void CloseFSM::on_finished()
 | |
| {
 | |
|     m_mainWindow->close();
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSourceAdd(Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex)
 | |
| {
 | |
|     AddSampleSourceFSM *fsm = new AddSampleSourceFSM(this, deviceWorkspace, spectrumWorkspace, deviceIndex, true);
 | |
|     connect(fsm, &AddSampleSourceFSM::finished, fsm, &AddSampleSourceFSM::deleteLater);
 | |
|     fsm->start();
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSourceCreate(
 | |
|     int deviceSetIndex,
 | |
|     int& deviceIndex,
 | |
|     DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     DeviceEnumerator::instance()->changeRxSelection(deviceSetIndex, deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getRxSamplingDevice(deviceIndex);
 | |
|     deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|     deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|     deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|     deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|     deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|     deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|     deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|     deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getRxPluginInterface(deviceIndex));
 | |
| 
 | |
|     qDebug() << "MainWindow::sampleSourceCreate:"
 | |
|         << "deviceSetIndex:" << deviceSetIndex
 | |
|         << "deviceIndex:" << deviceIndex
 | |
|         << "hardwareId:" << samplingDevice->hardwareId
 | |
|         << "sequence:" << samplingDevice->sequence
 | |
|         << "id:" << samplingDevice->id
 | |
|         << "serial:" << samplingDevice->serial
 | |
|         << "displayedName:" << samplingDevice->displayedName;
 | |
| 
 | |
|     if (deviceAPI->getSamplingDeviceId().size() == 0) // non existent device => replace by default
 | |
|     {
 | |
|         qDebug("MainWindow::sampleSourceCreate: non existent device replaced by File Input");
 | |
|         int fileInputDeviceIndex = DeviceEnumerator::instance()->getFileInputDeviceIndex();
 | |
|         deviceIndex = fileInputDeviceIndex;
 | |
|         samplingDevice = DeviceEnumerator::instance()->getRxSamplingDevice(fileInputDeviceIndex);
 | |
|         deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|         deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|         deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|         deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|         deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|         deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|         deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|         deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getRxPluginInterface(fileInputDeviceIndex));
 | |
|     }
 | |
| 
 | |
|     QString userArgs = m_mainCore->m_settings.getDeviceUserArgs().findUserArgs(samplingDevice->hardwareId, samplingDevice->sequence);
 | |
| 
 | |
|     if (userArgs.size() > 0) {
 | |
|         deviceAPI->setHardwareUserArguments(userArgs);
 | |
|     }
 | |
| 
 | |
|     // add to buddies list
 | |
|     auto it = m_deviceUIs.begin();
 | |
|     int nbOfBuddies = 0;
 | |
| 
 | |
|     for (; it != m_deviceUIs.end(); ++it)
 | |
|     {
 | |
|         if ((*it != deviceUISet) && // do not add to itself
 | |
|             (deviceUISet->m_deviceAPI->getHardwareId() == (*it)->m_deviceAPI->getHardwareId()) &&
 | |
|             (deviceUISet->m_deviceAPI->getSamplingDeviceSerial() == (*it)->m_deviceAPI->getSamplingDeviceSerial()))
 | |
|         {
 | |
|             (*it)->m_deviceAPI->addBuddy(deviceUISet->m_deviceAPI);
 | |
|             nbOfBuddies++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (nbOfBuddies == 0) {
 | |
|         deviceUISet->m_deviceAPI->setBuddyLeader(true);
 | |
|     }
 | |
| 
 | |
|     // constructs new GUI and input object
 | |
|     DeviceSampleSource *source = deviceAPI->getPluginInterface()->createSampleSourcePluginInstance(
 | |
|             deviceAPI->getSamplingDeviceId(), deviceAPI);
 | |
|     deviceAPI->setSampleSource(source);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSourceCreateUI(
 | |
|     int deviceSetIndex,
 | |
|     int deviceIndex,
 | |
|     DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     QWidget *gui;
 | |
|     DeviceGUI *deviceGUI = deviceAPI->getPluginInterface()->createSampleSourcePluginInstanceGUI(
 | |
|             deviceAPI->getSamplingDeviceId(),
 | |
|             &gui,
 | |
|             deviceUISet
 | |
|         );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::moveToWorkspace,
 | |
|         this,
 | |
|         [this, deviceGUI](int wsIndexDest){ this->deviceMove(deviceGUI, wsIndexDest); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceChange,
 | |
|         this,
 | |
|         [this, deviceGUI](int newDeviceIndex){ this->samplingDeviceChangeHandler(deviceGUI, newDeviceIndex); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showSpectrum,
 | |
|         this,
 | |
|         &MainWindow::mainSpectrumShow
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showAllChannels,
 | |
|         this,
 | |
|         &MainWindow::showAllChannels
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::closing,
 | |
|         this,
 | |
|         [this, deviceGUI](){ this->removeDeviceSet(deviceGUI->getIndex()); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceSetPresetsDialogRequested,
 | |
|         this,
 | |
|         &MainWindow::openDeviceSetPresetsDialog
 | |
|     );
 | |
| 
 | |
|     deviceAPI->getSampleSource()->setMessageQueueToGUI(deviceGUI->getInputMessageQueue());
 | |
|     deviceUISet->m_deviceGUI = deviceGUI;
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getRxSamplingDevice(deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *selectedDevice = DeviceEnumerator::instance()->getRxSamplingDevice(deviceIndex); // FIXME: Why not use samplingDevice?
 | |
|     deviceUISet->m_selectedDeviceId = selectedDevice->id;
 | |
|     deviceUISet->m_selectedDeviceSerial = selectedDevice->serial;
 | |
|     deviceUISet->m_selectedDeviceSequence = selectedDevice->sequence;
 | |
|     deviceUISet->m_selectedDeviceItemImdex = selectedDevice->deviceItemIndex;
 | |
|     deviceUISet->m_deviceAPI->getSampleSource()->init();
 | |
|     // Finalize GUI setup and add it to workspace MDI
 | |
|     deviceGUI->setDeviceType(DeviceGUI::DeviceRx);
 | |
|     deviceGUI->setIndex(deviceSetIndex);
 | |
|     deviceGUI->setToolTip(samplingDevice->displayedName);
 | |
|     deviceGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
|     deviceGUI->setCurrentDeviceIndex(deviceIndex);
 | |
|     QStringList channelNames;
 | |
|     m_pluginManager->listRxChannels(channelNames);
 | |
|     deviceGUI->setChannelNames(channelNames);
 | |
|     MainSpectrumGUI *mainSpectrumGUI = deviceUISet->m_mainSpectrumGUI;
 | |
|     mainSpectrumGUI->setDeviceType(MainSpectrumGUI::DeviceRx);
 | |
|     mainSpectrumGUI->setIndex(deviceSetIndex);
 | |
|     mainSpectrumGUI->setToolTip(samplingDevice->displayedName);
 | |
|     mainSpectrumGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSinkAdd(Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex)
 | |
| {
 | |
|     AddSampleSinkFSM *fsm = new AddSampleSinkFSM(this, deviceWorkspace, spectrumWorkspace, deviceIndex, true);
 | |
|     connect(fsm, &AddSampleSinkFSM::finished, fsm, &AddSampleSinkFSM::deleteLater);
 | |
|     fsm->start();
 | |
|     }
 | |
| 
 | |
| void MainWindow::sampleSinkCreate(
 | |
|         int deviceSetIndex,
 | |
|         int& deviceIndex,
 | |
|         DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     DeviceEnumerator::instance()->changeTxSelection(deviceSetIndex, deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getTxSamplingDevice(deviceIndex);
 | |
|     deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|     deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|     deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|     deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|     deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|     deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|     deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|     deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getTxPluginInterface(deviceIndex));
 | |
| 
 | |
|     qDebug() << "MainWindow::sampleSinkCreate:"
 | |
|         << "deviceSetIndex:" << deviceSetIndex
 | |
|         << "newDeviceIndex:" << deviceIndex
 | |
|         << "hardwareId:" << samplingDevice->hardwareId
 | |
|         << "sequence:" << samplingDevice->sequence
 | |
|         << "id:" << samplingDevice->id
 | |
|         << "serial:" << samplingDevice->serial
 | |
|         << "displayedName:" << samplingDevice->displayedName;
 | |
| 
 | |
|     if (deviceAPI->getSamplingDeviceId().size() == 0) // non existent device => replace by default
 | |
|     {
 | |
|         qDebug("MainWindow::sampleSinkCreate: non existent device replaced by File Sink");
 | |
|         int fileSinkDeviceIndex = DeviceEnumerator::instance()->getFileOutputDeviceIndex();
 | |
|         deviceIndex = fileSinkDeviceIndex;
 | |
|         samplingDevice = DeviceEnumerator::instance()->getTxSamplingDevice(fileSinkDeviceIndex);
 | |
|         deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|         deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|         deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|         deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|         deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|         deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|         deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|         deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getTxPluginInterface(fileSinkDeviceIndex));
 | |
|     }
 | |
| 
 | |
|     QString userArgs = m_mainCore->m_settings.getDeviceUserArgs().findUserArgs(samplingDevice->hardwareId, samplingDevice->sequence);
 | |
| 
 | |
|     if (userArgs.size() > 0) {
 | |
|         deviceAPI->setHardwareUserArguments(userArgs);
 | |
|     }
 | |
| 
 | |
|     // add to buddies list
 | |
|     auto it = m_deviceUIs.begin();
 | |
|     int nbOfBuddies = 0;
 | |
| 
 | |
|     for (; it != m_deviceUIs.end(); ++it)
 | |
|     {
 | |
|         if ((*it != deviceUISet) && // do not add to itself
 | |
|             (deviceAPI->getHardwareId() == (*it)->m_deviceAPI->getHardwareId()) &&
 | |
|             (deviceAPI->getSamplingDeviceSerial() == (*it)->m_deviceAPI->getSamplingDeviceSerial()))
 | |
|         {
 | |
|             (*it)->m_deviceAPI->addBuddy(deviceAPI);
 | |
|             nbOfBuddies++;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     if (nbOfBuddies == 0) {
 | |
|         deviceAPI->setBuddyLeader(true);
 | |
|     }
 | |
| 
 | |
|     // constructs new GUI and output object
 | |
|     DeviceSampleSink *sink = deviceAPI->getPluginInterface()->createSampleSinkPluginInstance(
 | |
|             deviceAPI->getSamplingDeviceId(), deviceAPI);
 | |
|     deviceAPI->setSampleSink(sink);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSinkCreateUI(
 | |
|         int deviceSetIndex,
 | |
|         int deviceIndex,
 | |
|         DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     QWidget *gui;
 | |
|     DeviceGUI *deviceGUI = deviceAPI->getPluginInterface()->createSampleSinkPluginInstanceGUI(
 | |
|             deviceAPI->getSamplingDeviceId(),
 | |
|             &gui,
 | |
|             deviceUISet
 | |
|         );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::moveToWorkspace,
 | |
|         this,
 | |
|         [this, deviceGUI](int wsIndexDest){ this->deviceMove(deviceGUI, wsIndexDest); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceChange,
 | |
|         this,
 | |
|         [this, deviceGUI](int newDeviceIndex){ this->samplingDeviceChangeHandler(deviceGUI, newDeviceIndex); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showSpectrum,
 | |
|         this,
 | |
|         &MainWindow::mainSpectrumShow
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showAllChannels,
 | |
|         this,
 | |
|         &MainWindow::showAllChannels
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::closing,
 | |
|         this,
 | |
|         [this, deviceGUI](){ this->removeDeviceSet(deviceGUI->getIndex()); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceSetPresetsDialogRequested,
 | |
|         this,
 | |
|         &MainWindow::openDeviceSetPresetsDialog
 | |
|     );
 | |
| 
 | |
|     deviceAPI->getSampleSink()->setMessageQueueToGUI(deviceGUI->getInputMessageQueue());
 | |
|     deviceUISet->m_deviceGUI = deviceGUI;
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getTxSamplingDevice(deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *selectedDevice = DeviceEnumerator::instance()->getRxSamplingDevice(deviceIndex); // FIXME: Why getRxSamplingDevice?
 | |
|     deviceUISet->m_selectedDeviceId = selectedDevice->id;
 | |
|     deviceUISet->m_selectedDeviceSerial = selectedDevice->serial;
 | |
|     deviceUISet->m_selectedDeviceSequence = selectedDevice->sequence;
 | |
|     deviceUISet->m_selectedDeviceItemImdex = selectedDevice->deviceItemIndex;
 | |
|     deviceUISet->m_deviceAPI->getSampleSink()->init();
 | |
|     // Finalize GUI setup and add it to workspace MDI
 | |
|     deviceGUI->setDeviceType(DeviceGUI::DeviceTx);
 | |
|     deviceGUI->setIndex(deviceSetIndex);
 | |
|     deviceGUI->setToolTip(samplingDevice->displayedName);
 | |
|     deviceGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
|     deviceGUI->setCurrentDeviceIndex(deviceIndex);
 | |
|     QStringList channelNames;
 | |
|     m_pluginManager->listTxChannels(channelNames);
 | |
|     deviceGUI->setChannelNames(channelNames);
 | |
|     MainSpectrumGUI *spectrumGUI = deviceUISet->m_mainSpectrumGUI;
 | |
|     spectrumGUI->setDeviceType(MainSpectrumGUI::DeviceTx);
 | |
|     spectrumGUI->setIndex(deviceSetIndex);
 | |
|     spectrumGUI->setToolTip(samplingDevice->displayedName);
 | |
|     spectrumGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleMIMOAdd(Workspace *deviceWorkspace, Workspace *spectrumWorkspace, int deviceIndex)
 | |
| {
 | |
|     AddSampleMIMOFSM *fsm = new AddSampleMIMOFSM(this, deviceWorkspace, spectrumWorkspace, deviceIndex, true);
 | |
|     connect(fsm, &AddSampleMIMOFSM::finished, fsm, &AddSampleSinkFSM::deleteLater);
 | |
|     fsm->start();
 | |
|     }
 | |
| 
 | |
| void MainWindow::sampleMIMOCreate(
 | |
|     int deviceSetIndex,
 | |
|     int& deviceIndex,
 | |
|     DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     DeviceEnumerator::instance()->changeMIMOSelection(deviceSetIndex, deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getMIMOSamplingDevice(deviceIndex);
 | |
|     deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|     deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|     deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|     deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|     deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|     deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|     deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|     deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getMIMOPluginInterface(deviceIndex));
 | |
| 
 | |
|     qDebug() << "MainWindow::sampleMIMOCreate:"
 | |
|         << "deviceSetIndex" << deviceSetIndex
 | |
|         << "newDeviceIndex:" << deviceIndex
 | |
|         << "hardwareId:" << samplingDevice->hardwareId
 | |
|         << "sequence:" << samplingDevice->sequence
 | |
|         << "id:" << samplingDevice->id
 | |
|         << "serial:" << samplingDevice->serial
 | |
|         << "displayedName:" << samplingDevice->displayedName;
 | |
| 
 | |
|     if (deviceAPI->getSamplingDeviceId().size() == 0) // non existent device => replace by default
 | |
|     {
 | |
|         qDebug("MainWindow::sampleMIMOCreate: non existent device replaced by Test MIMO");
 | |
|         int testMIMODeviceIndex = DeviceEnumerator::instance()->getTestMIMODeviceIndex();
 | |
|         deviceIndex = testMIMODeviceIndex;
 | |
|         samplingDevice = DeviceEnumerator::instance()->getMIMOSamplingDevice(testMIMODeviceIndex);
 | |
|         deviceAPI->setSamplingDeviceSequence(samplingDevice->sequence);
 | |
|         deviceAPI->setDeviceNbItems(samplingDevice->deviceNbItems);
 | |
|         deviceAPI->setDeviceItemIndex(samplingDevice->deviceItemIndex);
 | |
|         deviceAPI->setHardwareId(samplingDevice->hardwareId);
 | |
|         deviceAPI->setSamplingDeviceId(samplingDevice->id);
 | |
|         deviceAPI->setSamplingDeviceSerial(samplingDevice->serial);
 | |
|         deviceAPI->setSamplingDeviceDisplayName(samplingDevice->displayedName);
 | |
|         deviceAPI->setSamplingDevicePluginInterface(DeviceEnumerator::instance()->getMIMOPluginInterface(testMIMODeviceIndex));
 | |
|     }
 | |
| 
 | |
|     QString userArgs = m_mainCore->m_settings.getDeviceUserArgs().findUserArgs(samplingDevice->hardwareId, samplingDevice->sequence);
 | |
| 
 | |
|     if (userArgs.size() > 0) {
 | |
|         deviceAPI->setHardwareUserArguments(userArgs);
 | |
|     }
 | |
| 
 | |
|     // constructs new GUI and output object
 | |
|     DeviceSampleMIMO *mimo = deviceAPI->getPluginInterface()->createSampleMIMOPluginInstance(
 | |
|             deviceAPI->getSamplingDeviceId(), deviceAPI);
 | |
|     deviceAPI->setSampleMIMO(mimo);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleMIMOCreateUI(
 | |
|     int deviceSetIndex,
 | |
|     int deviceIndex,
 | |
|     DeviceUISet *deviceUISet
 | |
| )
 | |
| {
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
|     QWidget *gui;
 | |
|     DeviceGUI *deviceGUI = deviceAPI->getPluginInterface()->createSampleMIMOPluginInstanceGUI(
 | |
|             deviceAPI->getSamplingDeviceId(),
 | |
|             &gui,
 | |
|             deviceUISet
 | |
|         );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::moveToWorkspace,
 | |
|         this,
 | |
|         [this, deviceGUI](int wsIndexDest){ this->deviceMove(deviceGUI, wsIndexDest); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceChange,
 | |
|         this,
 | |
|         [this, deviceGUI](int newDeviceIndex){ this->samplingDeviceChangeHandler(deviceGUI, newDeviceIndex); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showSpectrum,
 | |
|         this,
 | |
|         &MainWindow::mainSpectrumShow
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::showAllChannels,
 | |
|         this,
 | |
|         &MainWindow::showAllChannels
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::closing,
 | |
|         this,
 | |
|         [this, deviceGUI](){ this->removeDeviceSet(deviceGUI->getIndex()); }
 | |
|     );
 | |
|     QObject::connect(
 | |
|         deviceGUI,
 | |
|         &DeviceGUI::deviceSetPresetsDialogRequested,
 | |
|         this,
 | |
|         &MainWindow::openDeviceSetPresetsDialog
 | |
|     );
 | |
| 
 | |
|     deviceAPI->getSampleMIMO()->setMessageQueueToGUI(deviceGUI->getInputMessageQueue());
 | |
|     deviceUISet->m_deviceGUI = deviceGUI;
 | |
|     const PluginInterface::SamplingDevice *samplingDevice = DeviceEnumerator::instance()->getMIMOSamplingDevice(deviceIndex);
 | |
|     const PluginInterface::SamplingDevice *selectedDevice = DeviceEnumerator::instance()->getRxSamplingDevice(deviceIndex); // FIXME: Why getRxSamplingDevice?
 | |
|     deviceUISet->m_selectedDeviceId = selectedDevice->id;
 | |
|     deviceUISet->m_selectedDeviceSerial = selectedDevice->serial;
 | |
|     deviceUISet->m_selectedDeviceSequence = selectedDevice->sequence;
 | |
|     deviceUISet->m_selectedDeviceItemImdex = selectedDevice->deviceItemIndex;
 | |
|     deviceUISet->m_deviceAPI->getSampleMIMO()->init();
 | |
|     // Finalize GUI setup and add it to workspace MDI
 | |
|     deviceGUI->setDeviceType(DeviceGUI::DeviceMIMO);
 | |
|     deviceGUI->setIndex(deviceSetIndex);
 | |
|     deviceGUI->setToolTip(samplingDevice->displayedName);
 | |
|     deviceGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
|     deviceGUI->setCurrentDeviceIndex(deviceIndex);
 | |
|     QStringList channelNames;
 | |
|     QStringList tmpChannelNames;
 | |
|     m_pluginManager->listMIMOChannels(channelNames);
 | |
|     m_pluginManager->listRxChannels(tmpChannelNames);
 | |
|     channelNames.append(tmpChannelNames);
 | |
|     m_pluginManager->listTxChannels(tmpChannelNames);
 | |
|     channelNames.append(tmpChannelNames);
 | |
|     deviceGUI->setChannelNames(channelNames);
 | |
|     MainSpectrumGUI *spectrumGUI = deviceUISet->m_mainSpectrumGUI;
 | |
|     spectrumGUI->setDeviceType(MainSpectrumGUI::DeviceMIMO);
 | |
|     spectrumGUI->setIndex(deviceSetIndex);
 | |
|     spectrumGUI->setToolTip(samplingDevice->displayedName);
 | |
|     spectrumGUI->setTitle(samplingDevice->displayedName.split(" ")[0]);
 | |
| }
 | |
| 
 | |
| void MainWindow::removeDeviceSet(int deviceSetIndex)
 | |
| {
 | |
|     qDebug("MainWindow::removeDeviceSet: index: %d", deviceSetIndex);
 | |
|     if (deviceSetIndex >= (int) m_deviceUIs.size()) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     RemoveDeviceSetFSM *fsm = new RemoveDeviceSetFSM(this, deviceSetIndex);
 | |
|     connect(fsm, &RemoveDeviceSetFSM::finished, fsm, &RemoveDeviceSetFSM::deleteLater);
 | |
|     fsm->start();
 | |
|     }
 | |
| 
 | |
| void MainWindow::removeAllDeviceSets()
 | |
|     {
 | |
|     qDebug() << "MainWindow::removeAllDeviceSets";
 | |
|     RemoveAllDeviceSetsFSM *fsm = new RemoveAllDeviceSetsFSM(this);
 | |
|     connect(fsm, &RemoveAllDeviceSetsFSM::finished, fsm, &RemoveAllDeviceSetsFSM::deleteLater);
 | |
|     fsm->start();
 | |
|     }
 | |
| 
 | |
| void MainWindow::removeLastDeviceSet()
 | |
| {
 | |
|     if (m_deviceUIs.size() > 0)
 | |
|     {
 | |
|     qDebug("MainWindow::removeLastDeviceSet: %s", qPrintable(m_deviceUIs.back()->m_deviceAPI->getHardwareId()));
 | |
|         removeDeviceSet(m_deviceUIs.size() - 1);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::addFeatureSet()
 | |
| {
 | |
|     auto newFeatureSetIndex = (int) m_featureUIs.size();
 | |
| 
 | |
|     if (newFeatureSetIndex != 0)
 | |
|     {
 | |
|         qWarning("MainWindow::addFeatureSet: attempt to add more than one feature set (%d)", newFeatureSetIndex);
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_mainCore->appendFeatureSet();
 | |
|     m_featureUIs.push_back(new FeatureUISet(newFeatureSetIndex, m_mainCore->m_featureSets[newFeatureSetIndex]));
 | |
|     emit m_mainCore->featureSetAdded(newFeatureSetIndex);
 | |
| }
 | |
| 
 | |
| void MainWindow::removeFeatureSet(unsigned int tabIndex)
 | |
| {
 | |
|     if (tabIndex < m_featureUIs.size())
 | |
|     {
 | |
|         delete m_featureUIs[tabIndex];
 | |
|         m_featureUIs.pop_back();
 | |
|         m_mainCore->removeFeatureSet(tabIndex);
 | |
|         emit m_mainCore->featureSetRemoved(tabIndex);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::removeAllFeatureSets()
 | |
| {
 | |
|     while (!m_featureUIs.empty())
 | |
|     {
 | |
|         delete m_featureUIs.back();
 | |
|         m_featureUIs.pop_back();
 | |
|         m_mainCore->removeLastFeatureSet();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::deleteChannel(int deviceSetIndex, int channelIndex)
 | |
| {
 | |
|     if ((deviceSetIndex >= 0) && (deviceSetIndex < (int) m_deviceUIs.size()))
 | |
|     {
 | |
|         DeviceUISet *deviceSet = m_deviceUIs[deviceSetIndex];
 | |
|         deviceSet->deleteChannel(channelIndex);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::loadSettings()
 | |
| {
 | |
| 	qDebug() << "MainWindow::loadSettings";
 | |
| 
 | |
|     m_mainCore->m_settings.load();
 | |
|     m_mainCore->m_settings.sortPresets();
 | |
|     m_mainCore->m_settings.sortCommands();
 | |
|     if (m_mainCore->m_logger) {
 | |
|         m_mainCore->setLoggingOptions();
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::loadDeviceSetPresetSettings(const Preset* preset, int deviceSetIndex)
 | |
| {
 | |
| 	qDebug("MainWindow::loadDeviceSetPresetSettings: preset [%s | %s]",
 | |
| 		qPrintable(preset->getGroup()),
 | |
| 		qPrintable(preset->getDescription()));
 | |
| 
 | |
| 	if (deviceSetIndex >= 0)
 | |
| 	{
 | |
|         DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|         deviceUISet->loadDeviceSetSettings(preset, m_pluginManager->getPluginAPI(), &m_workspaces, nullptr);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::saveDeviceSetPresetSettings(Preset* preset, int deviceSetIndex)
 | |
| {
 | |
| 	qDebug("MainWindow::saveDeviceSetPresetSettings: preset [%s | %s]",
 | |
| 		qPrintable(preset->getGroup()),
 | |
| 		qPrintable(preset->getDescription()));
 | |
| 
 | |
|     // Save from currently selected source tab
 | |
|     //int currentSourceTabIndex = ui->tabInputsView->currentIndex();
 | |
|     const DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|     deviceUISet->saveDeviceSetSettings(preset);
 | |
| }
 | |
| 
 | |
| void MainWindow::loadFeatureSetPresetSettings(const FeatureSetPreset* preset, int featureSetIndex, Workspace *workspace)
 | |
| {
 | |
| 	qDebug("MainWindow::loadFeatureSetPresetSettings: preset [%s | %s]",
 | |
| 		qPrintable(preset->getGroup()),
 | |
| 		qPrintable(preset->getDescription()));
 | |
| 
 | |
| 	if (featureSetIndex >= 0)
 | |
| 	{
 | |
|         FeatureUISet *featureSetUI = m_featureUIs[featureSetIndex];
 | |
|         qDebug("MainWindow::loadFeatureSetPresetSettings: m_apiAdapter: %p", m_apiAdapter);
 | |
|         featureSetUI->loadFeatureSetSettings(preset, m_pluginManager->getPluginAPI(), m_apiAdapter, &m_workspaces, workspace);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::saveFeatureSetPresetSettings(FeatureSetPreset* preset, int featureSetIndex)
 | |
| {
 | |
| 	qDebug("MainWindow::saveFeatureSetPresetSettings: preset [%s | %s]",
 | |
| 		qPrintable(preset->getGroup()),
 | |
| 		qPrintable(preset->getDescription()));
 | |
| 
 | |
|     // Save from currently selected source tab
 | |
|     //int currentSourceTabIndex = ui->tabInputsView->currentIndex();
 | |
|     FeatureUISet *featureUI = m_featureUIs[featureSetIndex];
 | |
| 
 | |
|     preset->clearFeatures();
 | |
|     featureUI->saveFeatureSetSettings(preset);
 | |
| }
 | |
| 
 | |
| void MainWindow::loadDefaultConfigurations() const
 | |
| {
 | |
|     QDirIterator configurationsIt(":configurations", QDirIterator::Subdirectories);
 | |
|     while (configurationsIt.hasNext())
 | |
|     {
 | |
|         QString group = configurationsIt.next();
 | |
|         QDirIterator groupIt(group, {"*.cfgx"}, QDir::Files);
 | |
|         while (groupIt.hasNext())
 | |
|         {
 | |
|             QFile file(groupIt.next());
 | |
| 			if (file.open(QIODevice::ReadOnly | QIODevice::Text))
 | |
| 			{
 | |
| 				QByteArray base64Str;
 | |
| 				QTextStream instream(&file);
 | |
| 				instream >> base64Str;
 | |
| 				file.close();
 | |
| 
 | |
| 				Configuration* configuration = MainCore::instance()->m_settings.newConfiguration("", "");
 | |
| 				configuration->deserialize(QByteArray::fromBase64(base64Str));
 | |
| 			}
 | |
|             else
 | |
|             {
 | |
|                 qDebug() << "MainWindow::loadDefaultConfigurations: Failed to open configuration " << file.fileName();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     QDirIterator presetIt(":presets", QDirIterator::Subdirectories);
 | |
|     while (presetIt.hasNext())
 | |
|     {
 | |
|         QString group = presetIt.next();
 | |
|         QDirIterator groupIt(group, {"*.prex"}, QDir::Files);
 | |
|         while (groupIt.hasNext())
 | |
|         {
 | |
|             QFile file(groupIt.next());
 | |
| 			if (file.open(QIODevice::ReadOnly | QIODevice::Text))
 | |
| 			{
 | |
| 				QByteArray base64Str;
 | |
| 				QTextStream instream(&file);
 | |
| 				instream >> base64Str;
 | |
| 				file.close();
 | |
| 
 | |
| 				Preset* preset = MainCore::instance()->m_settings.newPreset("", "");
 | |
| 				preset->deserialize(QByteArray::fromBase64(base64Str));
 | |
| 			}
 | |
|             else
 | |
|             {
 | |
|                 qDebug() << "MainWindow::loadDefaultConfigurations: Failed to open preset " << file.fileName();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::loadConfiguration(const Configuration *configuration, bool fromDialog)
 | |
| {
 | |
| 	qDebug("MainWindow::loadConfiguration: configuration [%s | %s] %d workspace(s) - %d device set(s) - %d feature(s)",
 | |
| 		qPrintable(configuration->getGroup()),
 | |
| 		qPrintable(configuration->getDescription()),
 | |
|         (int)configuration->getNumberOfWorkspaceGeometries(),
 | |
|         (int)configuration->getDeviceSetPresets().size(),
 | |
|         (int)configuration->getFeatureSetPreset().getFeatureCount()
 | |
|     );
 | |
| 
 | |
|     QProgressDialog *waitBox = nullptr;
 | |
| 
 | |
|     if (fromDialog)
 | |
|     {
 | |
|         waitBox = new QProgressDialog("Loading configuration...", "", 0, 100, this);
 | |
|         waitBox->setWindowModality(Qt::WindowModal);
 | |
|         waitBox->setAttribute(Qt::WA_DeleteOnClose, true);
 | |
|         waitBox->setMinimumDuration(0);
 | |
|         waitBox->setCancelButton(nullptr);
 | |
|         waitBox->setValue(1);
 | |
|         QApplication::processEvents();
 | |
|     }
 | |
| 
 | |
|     LoadConfigurationFSM *fsm = new LoadConfigurationFSM(this, configuration, waitBox);
 | |
|     connect(fsm, &LoadConfigurationFSM::finished, fsm, &LoadConfigurationFSM::deleteLater);
 | |
|     fsm->start();
 | |
|     }
 | |
| 
 | |
| void MainWindow::saveConfiguration(Configuration *configuration)
 | |
| {
 | |
| 	qDebug("MainWindow::saveConfiguration: configuration [%s | %s] %d workspaces",
 | |
| 		qPrintable(configuration->getGroup()),
 | |
| 		qPrintable(configuration->getDescription()),
 | |
|         (int)m_workspaces.size()
 | |
|     );
 | |
| 
 | |
|     configuration->clearData();
 | |
|     QList<Preset>& deviceSetPresets = configuration->getDeviceSetPresets();
 | |
| 
 | |
|     for (const auto& deviceUISet : m_deviceUIs)
 | |
|     {
 | |
|         deviceSetPresets.push_back(Preset());
 | |
|         deviceUISet->saveDeviceSetSettings(&deviceSetPresets.back());
 | |
|         deviceSetPresets.back().setSpectrumGeometry(MDIUtils::saveMDIGeometry(deviceUISet->m_mainSpectrumGUI));
 | |
|         deviceSetPresets.back().setSpectrumWorkspaceIndex(deviceUISet->m_mainSpectrumGUI->getWorkspaceIndex());
 | |
|         deviceSetPresets.back().setDeviceGeometry(MDIUtils::saveMDIGeometry(deviceUISet->m_deviceGUI));
 | |
|         deviceSetPresets.back().setDeviceWorkspaceIndex(deviceUISet->m_deviceGUI->getWorkspaceIndex());
 | |
|         qDebug("MainWindow::saveConfiguration: %s device in workspace %d spectrum in %d",
 | |
|             qPrintable(deviceUISet->m_deviceAPI->getSamplingDeviceId()),
 | |
|             deviceUISet->m_deviceGUI->getWorkspaceIndex(),
 | |
|             deviceUISet->m_mainSpectrumGUI->getWorkspaceIndex());
 | |
|     }
 | |
| 
 | |
|     m_featureUIs[0]->saveFeatureSetSettings(&configuration->getFeatureSetPreset());
 | |
| 
 | |
|     for (const auto& workspace : m_workspaces)
 | |
|     {
 | |
|         configuration->getWorkspaceGeometries().push_back(workspace->saveGeometry());
 | |
|         configuration->getWorkspaceAutoStackOptions().push_back(workspace->getAutoStackOption());
 | |
|         configuration->getWorkspaceTabSubWindowsOptions().push_back(workspace->getTabSubWindowsOption());
 | |
|     }
 | |
| }
 | |
| 
 | |
| QString MainWindow::openGLVersion() const
 | |
| {
 | |
|     const QOpenGLContext *glCurrentContext = QOpenGLContext::globalShareContext();
 | |
|     if (glCurrentContext)
 | |
|     {
 | |
|         if (glCurrentContext->isValid())
 | |
|         {
 | |
|             int major = glCurrentContext->format().majorVersion();
 | |
|             int minor = glCurrentContext->format().minorVersion();
 | |
|             bool es = glCurrentContext->isOpenGLES();
 | |
|             QString version = QString("%1.%2%3").arg(major).arg(minor).arg(es ? " ES" : "");
 | |
|             // Waterfall doesn't work if version is less than 2.1, so display in red
 | |
|             if ((major < 2) || ((major == 2) && (minor == 0))) {
 | |
|                 version = "<span style=\"color:red\">" + version + "</span>";
 | |
|             }
 | |
|             return version;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return "N/A";
 | |
|         }
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         return "N/A";
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::createMenuBar(QToolButton *button) const
 | |
| {
 | |
|     QMenu *fileMenu;
 | |
|     QMenu *viewMenu;
 | |
|     QMenu *workspacesMenu;
 | |
|     QMenu *preferencesMenu;
 | |
|     QMenu *helpMenu;
 | |
| 
 | |
|     if (button == nullptr)
 | |
|     {
 | |
|         QMenuBar *menuBar = this->menuBar();
 | |
|         fileMenu = menuBar->addMenu("&File");
 | |
|         viewMenu = menuBar->addMenu("&View");
 | |
|         workspacesMenu = menuBar->addMenu("&Workspaces");
 | |
|         preferencesMenu = menuBar->addMenu("&Preferences");
 | |
|         helpMenu = menuBar->addMenu("&Help");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         auto *menu = new QMenu();
 | |
|         fileMenu = new QMenu("&File");
 | |
|         menu->addMenu(fileMenu);
 | |
|         viewMenu = new QMenu("&View");
 | |
|         menu->addMenu(viewMenu);
 | |
|         workspacesMenu = new QMenu("&Workspaces");
 | |
|         menu->addMenu(workspacesMenu);
 | |
|         preferencesMenu = new QMenu("&Preferences");
 | |
|         menu->addMenu(preferencesMenu);
 | |
|         helpMenu = new QMenu("&Help");
 | |
|         menu->addMenu(helpMenu);
 | |
|         button->setMenu(menu);
 | |
|     }
 | |
| 
 | |
|     QAction *exitAction = fileMenu->addAction("E&xit");
 | |
|     exitAction->setShortcut(QKeySequence(Qt::CTRL + Qt::Key_Q));
 | |
|     exitAction->setToolTip("Exit");
 | |
|     QObject::connect(exitAction, &QAction::triggered, this, &QMainWindow::close);
 | |
| 
 | |
|     QAction *fullscreenAction = viewMenu->addAction("&Fullscreen");
 | |
|     fullscreenAction->setShortcut(QKeySequence(Qt::Key_F11));
 | |
|     fullscreenAction->setToolTip("Toggle fullscreen view");
 | |
|     fullscreenAction->setCheckable(true);
 | |
|     QObject::connect(fullscreenAction, &QAction::triggered, this, &MainWindow::on_action_View_Fullscreen_toggled);
 | |
| #ifdef ANDROID
 | |
|     QAction *keepscreenonAction = viewMenu->addAction("&Keep screen on");
 | |
|     keepscreenonAction->setToolTip("Prevent screen from switching off");
 | |
|     keepscreenonAction->setCheckable(true);
 | |
|     QObject::connect(keepscreenonAction, &QAction::triggered, this, &MainWindow::on_action_View_KeepScreenOn_toggled);
 | |
| #endif
 | |
| #ifdef ENABLE_PROFILER
 | |
|     QAction *profileAction = viewMenu->addAction("&Profile data...");
 | |
|     profileAction->setToolTip("View profile data");
 | |
|     QObject::connect(profileAction, &QAction::triggered, this, &MainWindow::on_action_Profile_triggered);
 | |
| #endif
 | |
| 
 | |
|     QAction *newWorkspaceAction = workspacesMenu->addAction("&New");
 | |
|     newWorkspaceAction->setToolTip("Add a new workspace");
 | |
|     QObject::connect(newWorkspaceAction, &QAction::triggered, this, &MainWindow::addWorkspace);
 | |
|     QAction *viewAllWorkspacesAction = workspacesMenu->addAction("&View all");
 | |
|     viewAllWorkspacesAction->setToolTip("View all workspaces");
 | |
|     QObject::connect(viewAllWorkspacesAction, &QAction::triggered, this, &MainWindow::viewAllWorkspaces);
 | |
|     QAction *removeEmptyWorkspacesAction = workspacesMenu->addAction("&Remove empty");
 | |
|     removeEmptyWorkspacesAction->setToolTip("Remove empty workspaces");
 | |
|     QObject::connect(removeEmptyWorkspacesAction, &QAction::triggered, this, &MainWindow::removeEmptyWorkspaces);
 | |
| 
 | |
|     QAction *configurationsAction = preferencesMenu->addAction("&Configurations...");
 | |
|     configurationsAction->setToolTip("Manage configurations");
 | |
|     QObject::connect(configurationsAction, &QAction::triggered, this, &MainWindow::on_action_Configurations_triggered);
 | |
|     QAction *audioAction = preferencesMenu->addAction("&Audio...");
 | |
|     audioAction->setToolTip("Audio preferences");
 | |
|     QObject::connect(audioAction, &QAction::triggered, this, &MainWindow::on_action_Audio_triggered);
 | |
|     QAction *graphicsAction = preferencesMenu->addAction("&Graphics...");
 | |
|     graphicsAction->setToolTip("Graphics preferences");
 | |
|     QObject::connect(graphicsAction, &QAction::triggered, this, &MainWindow::on_action_Graphics_triggered);
 | |
|     QAction *loggingAction = preferencesMenu->addAction("&Logging...");
 | |
|     loggingAction->setToolTip("Logging preferences");
 | |
|     QObject::connect(loggingAction, &QAction::triggered, this, &MainWindow::on_action_Logging_triggered);
 | |
|     QAction *myPositionAction = preferencesMenu->addAction("My &Position...");
 | |
|     myPositionAction->setToolTip("Set station position");
 | |
|     QObject::connect(myPositionAction, &QAction::triggered, this, &MainWindow::on_action_My_Position_triggered);
 | |
|     QAction *fftAction = preferencesMenu->addAction("&FFT...");
 | |
|     fftAction->setToolTip("Set FFT preferences");
 | |
|     QObject::connect(fftAction, &QAction::triggered, this, &MainWindow::on_action_FFT_triggered);
 | |
| #if QT_CONFIG(process)
 | |
|     QAction *fftWisdomAction = preferencesMenu->addAction("&FFTW Wisdom...");
 | |
|     fftWisdomAction->setToolTip("Set FFTW cache");
 | |
|     QObject::connect(fftWisdomAction, &QAction::triggered, this, &MainWindow::on_action_FFTWisdom_triggered);
 | |
| #endif
 | |
|     QMenu *devicesMenu = preferencesMenu->addMenu("&Devices");
 | |
|     QAction *userArgumentsAction = devicesMenu->addAction("&User arguments...");
 | |
|     userArgumentsAction->setToolTip("Device custom user arguments");
 | |
|     QObject::connect(userArgumentsAction, &QAction::triggered, this, &MainWindow::on_action_DeviceUserArguments_triggered);
 | |
| #if QT_CONFIG(process)
 | |
|     QAction *commandsAction = preferencesMenu->addAction("C&ommands...");
 | |
|     commandsAction->setToolTip("External commands dialog");
 | |
|     QObject::connect(commandsAction, &QAction::triggered, this, &MainWindow::on_action_commands_triggered);
 | |
| #endif
 | |
|     QAction *saveAllAction = preferencesMenu->addAction("&Save all");
 | |
|     saveAllAction->setToolTip("Save all current settings");
 | |
|     QObject::connect(saveAllAction, &QAction::triggered, this, &MainWindow::on_action_saveAll_triggered);
 | |
| 
 | |
| #if defined(ANDROID) || defined(__EMSCRIPTEN__)
 | |
|     QAction *welcomeAction = helpMenu->addAction("&Welcome...");
 | |
|     welcomeAction->setToolTip("Show welcome dialog");
 | |
|     QObject::connect(welcomeAction, &QAction::triggered, this, &MainWindow::on_action_Welcome_triggered);
 | |
| #endif
 | |
|     QAction *quickStartAction = helpMenu->addAction("&Quick start...");
 | |
|     quickStartAction->setToolTip("Instructions for quick start");
 | |
|     QObject::connect(quickStartAction, &QAction::triggered, this, &MainWindow::on_action_Quick_Start_triggered);
 | |
|     QAction *mainWindowAction = helpMenu->addAction("&Main Window...");
 | |
|     mainWindowAction->setToolTip("Help on main window details");
 | |
|     QObject::connect(mainWindowAction, &QAction::triggered, this, &MainWindow::on_action_Main_Window_triggered);
 | |
|     QAction *loadedPluginsAction = helpMenu->addAction("Loaded &Plugins...");
 | |
|     loadedPluginsAction->setToolTip("List available plugins");
 | |
|     QObject::connect(loadedPluginsAction, &QAction::triggered, this, &MainWindow::on_action_Loaded_Plugins_triggered);
 | |
|     QAction *aboutAction = helpMenu->addAction("&About SDRangel...");
 | |
|     aboutAction->setToolTip("SDRangel application details");
 | |
|     QObject::connect(aboutAction, &QAction::triggered, this, &MainWindow::on_action_About_triggered);
 | |
| }
 | |
| 
 | |
| void MainWindow::createStatusBar()
 | |
| {
 | |
|     QString qtVersionStr = QString("Qt %1 ").arg(QT_VERSION_STR);
 | |
|     QString openGLVersionStr = QString("OpenGL %1 ").arg(openGLVersion());
 | |
| #if QT_VERSION >= QT_VERSION_CHECK(5, 4, 0)
 | |
|     m_showSystemWidget = new QLabel("SDRangel " + qApp->applicationVersion() + " " + qtVersionStr + openGLVersionStr
 | |
|             + QSysInfo::currentCpuArchitecture() + " " + QSysInfo::prettyProductName(), this);
 | |
| #else
 | |
|     m_showSystemWidget = new QLabel("SDRangel " + qApp->applicationVersion() + " " + qtVersionStr + openGLVersionStr, this);
 | |
| #endif
 | |
|     statusBar()->addPermanentWidget(m_showSystemWidget);
 | |
| 
 | |
| 	m_dateTimeWidget = new QLabel(tr("Date"), this);
 | |
| 	m_dateTimeWidget->setToolTip(tr("Current date/time"));
 | |
| 	statusBar()->addPermanentWidget(m_dateTimeWidget);
 | |
| }
 | |
| 
 | |
| void MainWindow::closeEvent(QCloseEvent *closeEvent)
 | |
| {
 | |
|     qDebug("MainWindow::closeEvent");
 | |
| 
 | |
|     if (!m_settingsSaved)
 | |
|     {
 | |
|     // Save window size and position
 | |
|     QSettings s;
 | |
|     s.setValue("mainWindowGeometry", qCompress(saveGeometry()).toBase64());
 | |
|     s.setValue("mainWindowState", qCompress(saveState()).toBase64());
 | |
| 
 | |
|     saveConfiguration(m_mainCore->m_settings.getWorkingConfiguration());
 | |
|     m_mainCore->m_settings.save();
 | |
| 
 | |
|         m_settingsSaved = true;
 | |
|     }
 | |
| 
 | |
|     if (m_deviceUIs.size() > 0)
 | |
|     {
 | |
|         CloseFSM *fsm = new CloseFSM(this);
 | |
|         connect(fsm, &CloseFSM::finished, fsm, &CloseFSM::deleteLater);
 | |
|         fsm->start();
 | |
|         closeEvent->ignore(); // CloseFSM will call close again later once all devices closed
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     if (m_profileDialog) {
 | |
|         m_profileDialog->close();
 | |
|     }
 | |
| 
 | |
|     closeEvent->accept();
 | |
| }
 | |
| 
 | |
| void MainWindow::applySettings()
 | |
| {
 | |
|     loadConfiguration(m_mainCore->m_settings.getWorkingConfiguration());
 | |
| 
 | |
|     m_mainCore->m_settings.sortPresets();
 | |
|     m_mainCore->setLoggingOptions();
 | |
| }
 | |
| 
 | |
| bool MainWindow::handleMessage(const Message& cmd)
 | |
| {
 | |
|     if (MainCore::MsgLoadPreset::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgLoadPreset&) cmd;
 | |
|         int deviceSetIndex =  notif.getDeviceSetIndex();
 | |
|         const Preset *preset = notif.getPreset();
 | |
| 
 | |
|         if (deviceSetIndex < (int) m_deviceUIs.size())
 | |
|         {
 | |
|             DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|             deviceUISet->loadDeviceSetSettings(preset, m_pluginManager->getPluginAPI(), &m_workspaces, nullptr);
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgSavePreset::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgSavePreset&) cmd;
 | |
|         saveDeviceSetPresetSettings(notif.getPreset(), notif.getDeviceSetIndex());
 | |
|         m_mainCore->m_settings.sortPresets();
 | |
|         m_mainCore->m_settings.save();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgLoadFeatureSetPreset::match(cmd))
 | |
|     {
 | |
|         if (!m_workspaces.empty())
 | |
|         {
 | |
|             auto& notif = (const MainCore::MsgLoadFeatureSetPreset&) cmd;
 | |
|             loadFeatureSetPresetSettings(notif.getPreset(), notif.getFeatureSetIndex(), m_workspaces[0]);
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgSaveFeatureSetPreset::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgSaveFeatureSetPreset&) cmd;
 | |
|         saveFeatureSetPresetSettings(notif.getPreset(), notif.getFeatureSetIndex());
 | |
|         m_mainCore->m_settings.sortFeatureSetPresets();
 | |
|         m_mainCore->m_settings.save();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeletePreset::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgDeletePreset&) cmd;
 | |
|         const Preset *presetToDelete = notif.getPreset();
 | |
|         // remove preset from settings
 | |
|         m_mainCore->m_settings.deletePreset(presetToDelete);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgLoadConfiguration::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgLoadConfiguration&) cmd;
 | |
|         const Configuration *configuration = notif.getConfiguration();
 | |
|         loadConfiguration(configuration, false);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgSaveConfiguration::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgSaveConfiguration&) cmd;
 | |
|         Configuration *configuration = notif.getConfiguration();
 | |
|         saveConfiguration(configuration);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeleteConfiguration::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgDeleteConfiguration&) cmd;
 | |
|         const Configuration *configurationToDelete = notif.getConfiguration();
 | |
|         // remove configuration from settings
 | |
|         m_mainCore->m_settings.deleteConfiguration(configurationToDelete);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeleteEmptyWorkspaces::match(cmd))
 | |
|     {
 | |
|         removeEmptyWorkspaces();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgAddWorkspace::match(cmd))
 | |
|     {
 | |
|         addWorkspace();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeleteFeatureSetPreset::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgDeleteFeatureSetPreset&) cmd;
 | |
|         const FeatureSetPreset *presetToDelete = notif.getPreset();
 | |
|         // remove preset from settings
 | |
|         m_mainCore->m_settings.deleteFeatureSetPreset(presetToDelete);
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgAddDeviceSet::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgAddDeviceSet&) cmd;
 | |
|         int direction = notif.getDirection();
 | |
| 
 | |
|         if (!m_workspaces.empty())
 | |
|         {
 | |
|             if (direction == 1) { // Single stream Tx
 | |
|                 sampleSinkAdd(m_workspaces[0], m_workspaces[0], -1); // create with file output device by default
 | |
|             } else if (direction == 0) { // Single stream Rx
 | |
|                 sampleSourceAdd(m_workspaces[0], m_workspaces[0], -1); // create with file input device by default
 | |
|             } else if (direction == 2) { // MIMO
 | |
|                 sampleMIMOAdd(m_workspaces[0], m_workspaces[0], -1); // create with testMI MIMO device by default
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgRemoveLastDeviceSet::match(cmd))
 | |
|     {
 | |
|         if (!m_deviceUIs.empty()) {
 | |
|             removeLastDeviceSet();
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgSetDevice::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgSetDevice&) cmd;
 | |
|         int deviceSetIndex = notif.getDeviceSetIndex();
 | |
| 
 | |
|         if ((deviceSetIndex >= 0) && (deviceSetIndex < (int) m_deviceUIs.size()))
 | |
|         {
 | |
|             Workspace *workspace = m_workspaces[m_deviceUIs[deviceSetIndex]->m_deviceGUI->getWorkspaceIndex()];
 | |
|             sampleDeviceChange(notif.getDeviceType(), notif.getDeviceSetIndex(), notif.getDeviceIndex(), workspace);
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgAddChannel::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgAddChannel&) cmd;
 | |
|         int deviceSetIndex = notif.getDeviceSetIndex();
 | |
| 
 | |
|         if ((deviceSetIndex >= 0) && (deviceSetIndex < (int) m_deviceUIs.size()))
 | |
|         {
 | |
|             const DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|             int deviceWorkspaceIndex = deviceUISet->m_deviceGUI->getWorkspaceIndex();
 | |
|             deviceWorkspaceIndex = deviceWorkspaceIndex < m_workspaces.size() ? deviceWorkspaceIndex : 0;
 | |
|             int channelRegistrationIndex;
 | |
| 
 | |
|             if (deviceUISet->m_deviceMIMOEngine)
 | |
|             {
 | |
|                 int nbMIMOChannels = deviceUISet->getNumberOfAvailableMIMOChannels();
 | |
|                 int nbRxChannels = deviceUISet->getNumberOfAvailableRxChannels();
 | |
|                 int direction = notif.getDirection();
 | |
| 
 | |
|                 if (direction == 2) {
 | |
|                     channelRegistrationIndex = notif.getChannelRegistrationIndex();
 | |
|                 } else if (direction == 0) {
 | |
|                     channelRegistrationIndex = nbMIMOChannels + notif.getChannelRegistrationIndex();
 | |
|                 } else {
 | |
|                     channelRegistrationIndex = nbMIMOChannels + nbRxChannels + notif.getChannelRegistrationIndex();
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 channelRegistrationIndex = notif.getChannelRegistrationIndex();
 | |
|             }
 | |
| 
 | |
|             channelAddClicked(m_workspaces[deviceWorkspaceIndex], deviceSetIndex, channelRegistrationIndex);
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeleteChannel::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgDeleteChannel&) cmd;
 | |
|         deleteChannel(notif.getDeviceSetIndex(), notif.getChannelIndex());
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgAddFeature::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgAddFeature&) cmd;
 | |
| 
 | |
|         if (!m_workspaces.empty()) {
 | |
|             featureAddClicked(m_workspaces[0], notif.getFeatureRegistrationIndex());
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDeleteFeature::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgDeleteFeature&) cmd;
 | |
|         deleteFeature(0, notif.getFeatureIndex());
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgMoveDeviceUIToWorkspace::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgMoveDeviceUIToWorkspace&) cmd;
 | |
|         int deviceSetIndex = notif.getDeviceSetIndex();
 | |
| 
 | |
|         if (deviceSetIndex < (int) m_deviceUIs.size())
 | |
|         {
 | |
|             DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|             DeviceGUI *gui = deviceUISet->m_deviceGUI;
 | |
|             deviceMove(gui, notif.getWorkspaceIndex());
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgMoveMainSpectrumUIToWorkspace::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgMoveMainSpectrumUIToWorkspace&) cmd;
 | |
|         int deviceSetIndex = notif.getDeviceSetIndex();
 | |
| 
 | |
|         if (deviceSetIndex < (int) m_deviceUIs.size())
 | |
|         {
 | |
|             DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|             MainSpectrumGUI *gui = deviceUISet->m_mainSpectrumGUI;
 | |
|             mainSpectrumMove(gui, notif.getWorkspaceIndex());
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgMoveFeatureUIToWorkspace::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgMoveFeatureUIToWorkspace&) cmd;
 | |
|         int featureIndex = notif.getFeatureIndex();
 | |
| 
 | |
|         if (featureIndex < m_featureUIs[0]->getNumberOfFeatures())
 | |
|         {
 | |
|             FeatureGUI *gui = m_featureUIs[0]->getFeatureGuiAt(featureIndex);
 | |
|             featureMove(gui, notif.getWorkspaceIndex());
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgMoveChannelUIToWorkspace::match(cmd))
 | |
|     {
 | |
|         auto& notif = (const MainCore::MsgMoveChannelUIToWorkspace&) cmd;
 | |
|         int deviceSetIndex = notif.getDeviceSetIndex();
 | |
| 
 | |
|         if (deviceSetIndex < (int) m_deviceUIs.size())
 | |
|         {
 | |
|             int channelIndex = notif.getChannelIndex();
 | |
|             DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
| 
 | |
|             if (channelIndex < deviceUISet->getNumberOfChannels())
 | |
|             {
 | |
|                 ChannelGUI *gui = deviceUISet->getChannelGUIAt(channelIndex);
 | |
|                 channelMove(gui, notif.getWorkspaceIndex());
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgApplySettings::match(cmd))
 | |
|     {
 | |
|         applySettings();
 | |
|         return true;
 | |
|     }
 | |
|     else if (MainCore::MsgDVSerial::match(cmd))
 | |
|     {
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void MainWindow::handleMessages()
 | |
| {
 | |
| 	Message* message;
 | |
| 
 | |
| 	while ((message = m_inputMessageQueue.pop()) != nullptr)
 | |
| 	{
 | |
| 		qDebug("MainWindow::handleMessages: message: %s", message->getIdentifier());
 | |
| 		handleMessage(*message);
 | |
| 		delete message;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::handleWorkspaceVisibility(Workspace *workspace, bool visibility)
 | |
| {
 | |
|     qDebug("MainWindow::handleWorkspaceHasFocus: index: %d %s",
 | |
|         workspace->getIndex(), visibility ? "visible" : "non visible");
 | |
|     m_currentWorkspace = workspace;
 | |
| }
 | |
| 
 | |
| void MainWindow::addWorkspace()
 | |
| {
 | |
|     int workspaceIndex = m_workspaces.size();
 | |
|     auto *workspace = new Workspace(workspaceIndex);
 | |
|     m_workspaces.push_back(workspace);
 | |
|     if (workspace->getMenuButton()) {
 | |
|         createMenuBar(workspace->getMenuButton());
 | |
|     }
 | |
|     QStringList featureNames;
 | |
|     m_pluginManager->listFeatures(featureNames);
 | |
|     m_workspaces.back()->addAvailableFeatures(featureNames);
 | |
|     this->addDockWidget(Qt::LeftDockWidgetArea, m_workspaces.back());
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::addRxDevice,
 | |
|         this,
 | |
|         [this](Workspace *inWorkspace, int deviceIndex) { this->sampleSourceAdd(inWorkspace, inWorkspace, deviceIndex); }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::addTxDevice,
 | |
|         this,
 | |
|         [this](Workspace *inWorkspace, int deviceIndex) { this->sampleSinkAdd(inWorkspace, inWorkspace, deviceIndex); }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::addMIMODevice,
 | |
|         this,
 | |
|         [this](Workspace *inWorkspace, int deviceIndex) { this->sampleMIMOAdd(inWorkspace, inWorkspace, deviceIndex); }
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::addFeature,
 | |
|         this,
 | |
|         &MainWindow::featureAddClicked
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::featurePresetsDialogRequested,
 | |
|         this,
 | |
|         &MainWindow::openFeaturePresetsDialog
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::configurationPresetsDialogRequested,
 | |
|         this,
 | |
|         &MainWindow::on_action_Configurations_triggered
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::startAllDevices,
 | |
|         this,
 | |
|         &MainWindow::startAllDevices
 | |
|     );
 | |
| 
 | |
|     QObject::connect(
 | |
|         m_workspaces.back(),
 | |
|         &Workspace::stopAllDevices,
 | |
|         this,
 | |
|         &MainWindow::stopAllDevices
 | |
|     );
 | |
| 
 | |
|     if (m_workspaces.size() > 1)
 | |
|     {
 | |
|         for (int i = 1; i < m_workspaces.size(); i++) {
 | |
|             tabifyDockWidget(m_workspaces[0], m_workspaces[i]);
 | |
|         }
 | |
| 
 | |
|         m_workspaces.back()->show();
 | |
|         m_workspaces.back()->raise();
 | |
|     }
 | |
|  }
 | |
| 
 | |
| void MainWindow::viewAllWorkspaces() const
 | |
| {
 | |
|     for (const auto& workspace : m_workspaces)
 | |
|     {
 | |
|         if (workspace->isHidden()) {
 | |
|             workspace->show();
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::removeEmptyWorkspaces()
 | |
| {
 | |
|     auto it = m_workspaces.begin();
 | |
| 
 | |
|     while (it != m_workspaces.end())
 | |
|     {
 | |
|         if ((*it)->getNumberOfSubWindows() == 0)
 | |
|         {
 | |
|             removeDockWidget(*it);
 | |
|             it = m_workspaces.erase(it);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             ++it;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Renumerate
 | |
|     for (int i = 0; i < m_workspaces.size(); i++)
 | |
|     {
 | |
|         Workspace *workspace = m_workspaces[i];
 | |
|         workspace->setIndex(i);
 | |
|         QList<QMdiSubWindow *> subWindows = workspace->getSubWindowList();
 | |
| 
 | |
|         for (auto& subWindow : subWindows)
 | |
|         {
 | |
|             if (qobject_cast<DeviceGUI*>(subWindow)) {
 | |
|                 qobject_cast<DeviceGUI*>(subWindow)->setWorkspaceIndex(i);
 | |
|             }
 | |
| 
 | |
|             if (qobject_cast<MainSpectrumGUI*>(subWindow)) {
 | |
|                 qobject_cast<MainSpectrumGUI*>(subWindow)->setWorkspaceIndex(i);
 | |
|             }
 | |
| 
 | |
|             if (qobject_cast<ChannelGUI*>(subWindow)) {
 | |
|                 qobject_cast<ChannelGUI*>(subWindow)->setWorkspaceIndex(i);
 | |
|             }
 | |
| 
 | |
|             if (qobject_cast<FeatureGUI*>(subWindow)) {
 | |
|                 qobject_cast<FeatureGUI*>(subWindow)->setWorkspaceIndex(i);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
| #ifdef ANDROID
 | |
|     // Need at least one workspace on Android, as no menus without
 | |
|     if (m_workspaces.size() == 0) {
 | |
|         addWorkspace();
 | |
|     }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #ifdef ANDROID
 | |
| void MainWindow::on_action_View_KeepScreenOn_toggled(bool checked)
 | |
| {
 | |
|     if (checked) {
 | |
|         Android::acquireScreenLock();
 | |
|     } else {
 | |
|         Android::releaseScreenLock();
 | |
|     }
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void MainWindow::on_action_View_Fullscreen_toggled(bool checked)
 | |
| {
 | |
| 	if(checked) {
 | |
| 		showFullScreen();
 | |
| 	} else {
 | |
| 	    showNormal();
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Profile_triggered()
 | |
| {
 | |
|     if (m_profileDialog == nullptr)
 | |
|     {
 | |
|         m_profileDialog = new ProfileDialog();
 | |
|         new DialogPositioner(m_profileDialog, true);
 | |
|     }
 | |
|     m_profileDialog->show();
 | |
|     m_profileDialog->raise();
 | |
| }
 | |
| 
 | |
| void MainWindow::commandKeysConnect(const QObject *object, const char *slot)
 | |
| {
 | |
|     setFocus();
 | |
|     connect(
 | |
|         m_commandKeyReceiver,
 | |
|         SIGNAL(capturedKey(Qt::Key, Qt::KeyboardModifiers, bool)),
 | |
|         object,
 | |
|         slot
 | |
|     );
 | |
| }
 | |
| 
 | |
| void MainWindow::commandKeysDisconnect(const QObject *object, const char *slot) const
 | |
| {
 | |
|     disconnect(
 | |
|         m_commandKeyReceiver,
 | |
|         SIGNAL(capturedKey(Qt::Key, Qt::KeyboardModifiers, bool)),
 | |
|         object,
 | |
|         slot
 | |
|     );
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_saveAll_triggered()
 | |
| {
 | |
|     saveConfiguration(m_mainCore->m_settings.getWorkingConfiguration());
 | |
|     m_mainCore->m_settings.save();
 | |
|     QMessageBox::information(this, tr("Done"), tr("All current settings saved"));
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Welcome_triggered()
 | |
| {
 | |
|     // Show welcome dialog
 | |
|     WelcomeDialog welcomeDialog(this);
 | |
|     new DialogPositioner(&welcomeDialog, true);
 | |
|     welcomeDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Quick_Start_triggered() const
 | |
| {
 | |
|     QDesktopServices::openUrl(QUrl("https://github.com/f4exb/sdrangel/wiki/Quick-start"));
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Main_Window_triggered() const
 | |
| {
 | |
|     QDesktopServices::openUrl(QUrl("https://github.com/f4exb/sdrangel/blob/master/sdrgui/readme.md"));
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Loaded_Plugins_triggered()
 | |
| {
 | |
|     PluginsDialog pluginsDialog(m_pluginManager, this);
 | |
|     pluginsDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Configurations_triggered()
 | |
| {
 | |
|     openConfigurationDialog(false);
 | |
| }
 | |
| 
 | |
| void MainWindow::openConfigurationDialog(bool openOnly)
 | |
| {
 | |
|     ConfigurationsDialog dialog(openOnly, this);
 | |
|     dialog.setConfigurations(m_mainCore->m_settings.getConfigurations());
 | |
|     dialog.populateTree();
 | |
|     QObject::connect(
 | |
|         &dialog,
 | |
|         &ConfigurationsDialog::saveConfiguration,
 | |
|         this,
 | |
|         &MainWindow::saveConfiguration
 | |
|     );
 | |
|     QObject::connect(
 | |
|         &dialog,
 | |
|         &ConfigurationsDialog::loadConfiguration,
 | |
|         this,
 | |
|         [this](const Configuration* configuration) { this->loadConfiguration(configuration, true); }
 | |
|     );
 | |
|     new DialogPositioner(&dialog, true);
 | |
|     dialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Audio_triggered()
 | |
| {
 | |
| 	AudioDialogX audioDialog(m_dspEngine->getAudioDeviceManager(), this);
 | |
|     new DialogPositioner(&audioDialog, true);
 | |
| 	audioDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Graphics_triggered()
 | |
| {
 | |
|     GraphicsDialog graphicsDialog(m_mainCore->m_settings, this);
 | |
|     new DialogPositioner(&graphicsDialog, true);
 | |
|     graphicsDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Logging_triggered()
 | |
| {
 | |
|     LoggingDialog loggingDialog(m_mainCore->m_settings, this);
 | |
|     new DialogPositioner(&loggingDialog, true);
 | |
|     loggingDialog.exec();
 | |
|     m_mainCore->setLoggingOptions();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_My_Position_triggered()
 | |
| {
 | |
| 	MyPositionDialog myPositionDialog(m_mainCore->m_settings, this);
 | |
|     new DialogPositioner(&myPositionDialog, true);
 | |
| 	myPositionDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_DeviceUserArguments_triggered()
 | |
| {
 | |
|     qDebug("MainWindow::on_action_DeviceUserArguments_triggered");
 | |
|     DeviceUserArgsDialog deviceUserArgsDialog(DeviceEnumerator::instance(), m_mainCore->m_settings.getDeviceUserArgs(), this);
 | |
|     new DialogPositioner(&deviceUserArgsDialog, true);
 | |
|     deviceUserArgsDialog.exec();
 | |
| }
 | |
| 
 | |
| #if QT_CONFIG(process)
 | |
| void MainWindow::on_action_commands_triggered()
 | |
| {
 | |
|     qDebug("MainWindow::on_action_commands_triggered");
 | |
|     CommandsDialog commandsDialog(this);
 | |
|     commandsDialog.setApiHost(m_apiServer->getHost());
 | |
|     commandsDialog.setApiPort(m_apiServer->getPort());
 | |
|     commandsDialog.setCommandKeyReceiver(m_commandKeyReceiver);
 | |
|     commandsDialog.populateTree();
 | |
|     new DialogPositioner(&commandsDialog, true);
 | |
|     commandsDialog.exec();
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void MainWindow::on_action_FFT_triggered()
 | |
| {
 | |
|     qDebug("MainWindow::on_action_FFT_triggered");
 | |
|     FFTDialog fftDialog(m_mainCore->m_settings, this);
 | |
|     new DialogPositioner(&fftDialog, true);
 | |
|     fftDialog.exec();
 | |
| }
 | |
| 
 | |
| #if QT_CONFIG(process)
 | |
| void MainWindow::on_action_FFTWisdom_triggered()
 | |
| {
 | |
|     qDebug("MainWindow::on_action_FFTWisdom_triggered");
 | |
| 
 | |
|     if (m_fftWisdomProcess)
 | |
|     {
 | |
|         QMessageBox::information(this, "FFTW Wisdom", QString("Process %1 is already running").arg(m_fftWisdomProcess->processId()));
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_fftWisdomProcess = new QProcess(this);
 | |
|     connect(m_fftWisdomProcess,
 | |
|         SIGNAL(finished(int, QProcess::ExitStatus)),
 | |
|         this,
 | |
|         SLOT(fftWisdomProcessFinished(int, QProcess::ExitStatus)));
 | |
|     FFTWisdomDialog fftWisdomDialog(m_fftWisdomProcess, this);
 | |
|     new DialogPositioner(&fftWisdomDialog, true);
 | |
| 
 | |
|     if (fftWisdomDialog.exec() == QDialog::Rejected)
 | |
|     {
 | |
|         disconnect(m_fftWisdomProcess,
 | |
|             SIGNAL(finished(int, QProcess::ExitStatus)),
 | |
|             this,
 | |
|             SLOT(fftWisdomProcessFinished(int, QProcess::ExitStatus)));
 | |
|         delete m_fftWisdomProcess;
 | |
|         m_fftWisdomProcess = nullptr;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         QMessageBox::information(this, "FFTW Wisdom", QString("Process %1 started").arg(m_fftWisdomProcess->processId()));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::fftWisdomProcessFinished(int exitCode, QProcess::ExitStatus exitStatus)
 | |
| {
 | |
|     qDebug("MainWindow::fftWisdomProcessFinished: process finished rc=%d (%d)", exitCode, (int) exitStatus);
 | |
| 
 | |
|     if ((exitCode != 0) || (exitStatus != QProcess::NormalExit))
 | |
|     {
 | |
|         QMessageBox::critical(this, "FFTW Wisdom", "fftwf-widdsom program failed");
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         QString log = m_fftWisdomProcess->readAllStandardOutput();
 | |
|         QMessageBox::information(this, "FFTW Wisdom", QString("Success\n%1").arg(log));
 | |
| 
 | |
|     }
 | |
| 
 | |
|     delete m_fftWisdomProcess;
 | |
|     m_fftWisdomProcess = nullptr;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void MainWindow::samplingDeviceChangeHandler(const DeviceGUI *deviceGUI, int newDeviceIndex)
 | |
| {
 | |
|     auto deviceType = (int) deviceGUI->getDeviceType();
 | |
|     int deviceSetIndex = deviceGUI->getIndex();
 | |
|     Workspace *workspace = m_workspaces[deviceGUI->getWorkspaceIndex()];
 | |
|     sampleDeviceChange(deviceType, deviceSetIndex, newDeviceIndex, workspace);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleDeviceChange(int deviceType, int deviceSetIndex, int newDeviceIndex, Workspace *workspace)
 | |
| {
 | |
|     qDebug("MainWindow::sampleDeviceChange: deviceType: %d deviceSetIndex: %d newDeviceIndex: %d",
 | |
|         deviceType, deviceSetIndex, newDeviceIndex);
 | |
|     if (deviceType == 0) {
 | |
|         sampleSourceChange(deviceSetIndex, newDeviceIndex, workspace);
 | |
|     } else if (deviceType == 1) {
 | |
|         sampleSinkChange(deviceSetIndex, newDeviceIndex, workspace);
 | |
|     } else if (deviceType == 2) {
 | |
|         sampleMIMOChange(deviceSetIndex, newDeviceIndex, workspace);
 | |
|     }
 | |
| 
 | |
|     emit MainCore::instance()->deviceChanged(deviceSetIndex);
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSourceChange(int deviceSetIndex, int newDeviceIndex, Workspace *workspace)
 | |
| {
 | |
|     if (deviceSetIndex >= 0)
 | |
|     {
 | |
|         qDebug("MainWindow::sampleSourceChange: deviceSet %d workspace: %d", deviceSetIndex, workspace->getIndex());
 | |
|         DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|         QPoint p = deviceUISet->m_deviceGUI->pos();
 | |
|         workspace->removeFromMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceAPI->stopDeviceEngine();
 | |
| 
 | |
|         // deletes old UI and input object
 | |
|         deviceUISet->m_deviceAPI->getSampleSource()->setMessageQueueToGUI(nullptr); // have source stop sending messages to the GUI
 | |
| 
 | |
|         delete deviceUISet->m_deviceGUI;
 | |
|         deviceUISet->m_deviceAPI->resetSamplingDeviceId();
 | |
|         deviceUISet->m_deviceAPI->getPluginInterface()->deleteSampleSourcePluginInstanceInput(deviceUISet->m_deviceAPI->getSampleSource());
 | |
|         deviceUISet->m_deviceAPI->clearBuddiesLists(); // clear old API buddies lists
 | |
| 
 | |
|         sampleSourceCreate(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         sampleSourceCreateUI(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         deviceUISet->m_deviceGUI->setWorkspaceIndex(workspace->getIndex());
 | |
|         workspace->addToMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceGUI->move(p);
 | |
| 
 | |
|         QObject::connect(
 | |
|             deviceUISet->m_deviceGUI,
 | |
|             &DeviceGUI::addChannelEmitted,
 | |
|             this,
 | |
|             [this, workspace, deviceSetIndex](int channelPluginIndex){ this->channelAddClicked(workspace, deviceSetIndex, channelPluginIndex); }
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleSinkChange(int deviceSetIndex, int newDeviceIndex, Workspace *workspace)
 | |
| {
 | |
|     if (deviceSetIndex >= 0)
 | |
|     {
 | |
|         qDebug("MainWindow::sampleSinkChange: deviceSet %d workspace: %d", deviceSetIndex, workspace->getIndex());
 | |
|         DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|         QPoint p = deviceUISet->m_deviceGUI->pos();
 | |
|         workspace->removeFromMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceAPI->saveSamplingDeviceSettings(m_mainCore->m_settings.getWorkingPreset()); // save old API settings
 | |
|         deviceUISet->m_deviceAPI->stopDeviceEngine();
 | |
| 
 | |
|         // deletes old UI and output object
 | |
|         deviceUISet->m_deviceAPI->getSampleSink()->setMessageQueueToGUI(nullptr); // have sink stop sending messages to the GUI
 | |
|         delete m_deviceUIs[deviceSetIndex]->m_deviceGUI;
 | |
|         deviceUISet->m_deviceAPI->resetSamplingDeviceId();
 | |
|         deviceUISet->m_deviceAPI->getPluginInterface()->deleteSampleSinkPluginInstanceOutput(deviceUISet->m_deviceAPI->getSampleSink());
 | |
|         deviceUISet->m_deviceAPI->clearBuddiesLists(); // clear old API buddies lists
 | |
| 
 | |
|         sampleSinkCreate(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         sampleSinkCreateUI(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         deviceUISet->m_deviceGUI->setWorkspaceIndex(workspace->getIndex());
 | |
|         workspace->addToMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceGUI->move(p);
 | |
| 
 | |
|         QObject::connect(
 | |
|             deviceUISet->m_deviceGUI,
 | |
|             &DeviceGUI::addChannelEmitted,
 | |
|             this,
 | |
|             [this, workspace, deviceSetIndex](int channelPluginIndex){ this->channelAddClicked(workspace, deviceSetIndex, channelPluginIndex); }
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::sampleMIMOChange(int deviceSetIndex, int newDeviceIndex, Workspace *workspace)
 | |
| {
 | |
|     if (deviceSetIndex >= 0)
 | |
|     {
 | |
|         qDebug("MainWindow::sampleSinkChange: deviceSet %d workspace: %d", deviceSetIndex, workspace->getIndex());
 | |
|         DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|         QPoint p = deviceUISet->m_deviceGUI->pos();
 | |
|         workspace->removeFromMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceAPI->saveSamplingDeviceSettings(m_mainCore->m_settings.getWorkingPreset()); // save old API settings
 | |
|         deviceUISet->m_deviceAPI->stopDeviceEngine();
 | |
| 
 | |
|         // deletes old UI and output object
 | |
|         deviceUISet->m_deviceAPI->getSampleMIMO()->setMessageQueueToGUI(nullptr); // have sink stop sending messages to the GUI
 | |
|         delete deviceUISet->m_deviceGUI;
 | |
|         deviceUISet->m_deviceAPI->resetSamplingDeviceId();
 | |
|         deviceUISet->m_deviceAPI->getPluginInterface()->deleteSampleMIMOPluginInstanceMIMO(deviceUISet->m_deviceAPI->getSampleMIMO());
 | |
| 
 | |
|         sampleMIMOCreate(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         sampleMIMOCreateUI(deviceSetIndex, newDeviceIndex, deviceUISet);
 | |
|         deviceUISet->m_deviceGUI->setWorkspaceIndex(workspace->getIndex());
 | |
|         workspace->addToMdiArea(deviceUISet->m_deviceGUI);
 | |
|         deviceUISet->m_deviceGUI->move(p);
 | |
| 
 | |
|         QObject::connect(
 | |
|             deviceUISet->m_deviceGUI,
 | |
|             &DeviceGUI::addChannelEmitted,
 | |
|             this,
 | |
|             [this, workspace, deviceSetIndex](int channelPluginIndex){ this->channelAddClicked(workspace, deviceSetIndex, channelPluginIndex); }
 | |
|         );
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::channelMoveToDeviceSet(ChannelGUI *gui, int dsIndexDestination)
 | |
| {
 | |
|     int deviceSetIndex = gui->getDeviceSetIndex();
 | |
|     int channelIndex = gui->getIndex();
 | |
|     qDebug("MainWindow::channelMoveToDeviceSet: %s at %d:%d to %d",
 | |
|         qPrintable(gui->getTitle()), deviceSetIndex, channelIndex, dsIndexDestination);
 | |
| 
 | |
|     if ((deviceSetIndex < (int) m_deviceUIs.size()) && (dsIndexDestination < (int) m_deviceUIs.size()))
 | |
|     {
 | |
|         DeviceUISet *deviceUI = m_deviceUIs[deviceSetIndex];
 | |
|         DeviceUISet *destDeviceUI = m_deviceUIs[dsIndexDestination];
 | |
|         ChannelAPI *channelAPI = deviceUI->getChannelAt(channelIndex);
 | |
|         deviceUI->unregisterChannelInstanceAt(channelIndex);
 | |
| 
 | |
|         if (deviceUI->m_deviceSourceEngine) // source devices
 | |
|         {
 | |
|             destDeviceUI->registerRxChannelInstance(channelAPI, gui);
 | |
|         }
 | |
|         else if (deviceUI->m_deviceSinkEngine) // sink devices
 | |
|         {
 | |
|             destDeviceUI->registerTxChannelInstance(channelAPI, gui);
 | |
|         }
 | |
|         else if (deviceUI->m_deviceMIMOEngine) // MIMO devices
 | |
|         {
 | |
|             destDeviceUI->registerChannelInstance(channelAPI, gui);
 | |
|         }
 | |
| 
 | |
|         gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|         gui->setDeviceSetIndex(dsIndexDestination);
 | |
|         DeviceAPI *destDeviceAPI = destDeviceUI->m_deviceAPI;
 | |
|         gui->setIndexToolTip(destDeviceAPI->getSamplingDeviceDisplayName());
 | |
|         channelAPI->setDeviceAPI(destDeviceAPI);
 | |
|         deviceUI->removeChannelMarker(&gui->getChannelMarker());
 | |
|         destDeviceUI->addChannelMarker(&gui->getChannelMarker());
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::channelDuplicate(const ChannelGUI *sourceChannelGUI)
 | |
| {
 | |
|     channelDuplicateToDeviceSet(sourceChannelGUI, sourceChannelGUI->getDeviceSetIndex()); // Duplicate in same device set
 | |
| }
 | |
| 
 | |
| void MainWindow::channelDuplicateToDeviceSet(const ChannelGUI *sourceChannelGUI, int dsIndexDestination)
 | |
| {
 | |
|     int dsIndexSource = sourceChannelGUI->getDeviceSetIndex();
 | |
|     int sourceChannelIndex = sourceChannelGUI->getIndex();
 | |
| 
 | |
|     qDebug("MainWindow::channelDuplicateToDeviceSet: %s at %d:%d to %d in workspace %d",
 | |
|         qPrintable(sourceChannelGUI->getTitle()), dsIndexSource, sourceChannelIndex, dsIndexDestination, sourceChannelGUI->getWorkspaceIndex());
 | |
| 
 | |
|     if ((dsIndexSource < (int) m_deviceUIs.size()) && (dsIndexDestination < (int) m_deviceUIs.size()))
 | |
|     {
 | |
|         DeviceUISet *sourceDeviceUI = m_deviceUIs[dsIndexSource];
 | |
|         const ChannelAPI *sourceChannelAPI = sourceDeviceUI->getChannelAt(sourceChannelIndex);
 | |
|         ChannelGUI *destChannelGUI = nullptr;
 | |
|         DeviceUISet *destDeviceUI = m_deviceUIs[dsIndexDestination];
 | |
| 
 | |
|         if (destDeviceUI->m_deviceSourceEngine) // source device => Rx channels
 | |
|         {
 | |
|             const PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getRxChannelRegistrations();
 | |
|             const PluginInterface *pluginInterface = nullptr;
 | |
| 
 | |
|             for (const auto& channelRegistration : *channelRegistrations)
 | |
|             {
 | |
|                 if (channelRegistration.m_channelIdURI == sourceChannelAPI->getURI())
 | |
|                 {
 | |
|                     pluginInterface = channelRegistration.m_plugin;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (pluginInterface)
 | |
|             {
 | |
|                 ChannelAPI *channelAPI;
 | |
|                 BasebandSampleSink *rxChannel = nullptr;
 | |
|                 pluginInterface->createRxChannel(destDeviceUI->m_deviceAPI, &rxChannel, &channelAPI);
 | |
|                 destChannelGUI = pluginInterface->createRxChannelGUI(destDeviceUI, rxChannel);
 | |
|                 destDeviceUI->registerRxChannelInstance(channelAPI, destChannelGUI);
 | |
|                 destChannelGUI->setDeviceType(ChannelGUI::DeviceType::DeviceRx);
 | |
|                 destChannelGUI->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 QByteArray b = sourceChannelGUI->serialize();
 | |
|                 destChannelGUI->deserialize(b);
 | |
|             }
 | |
|         }
 | |
|         else if (destDeviceUI->m_deviceSinkEngine) // sink device => Tx channels
 | |
|         {
 | |
|             const PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getTxChannelRegistrations(); // Available channel plugins
 | |
|             const PluginInterface *pluginInterface = nullptr;
 | |
| 
 | |
|             for (const auto& channelRegistration : *channelRegistrations)
 | |
|             {
 | |
|                 if (channelRegistration.m_channelIdURI == sourceChannelAPI->getURI())
 | |
|                 {
 | |
|                     pluginInterface = channelRegistration.m_plugin;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (pluginInterface)
 | |
|             {
 | |
|                 ChannelAPI *channelAPI;
 | |
|                 BasebandSampleSource *txChannel = nullptr;
 | |
|                 pluginInterface->createTxChannel(destDeviceUI->m_deviceAPI, &txChannel, &channelAPI);
 | |
|                 destChannelGUI = pluginInterface->createTxChannelGUI(destDeviceUI, txChannel);
 | |
|                 destDeviceUI->registerTxChannelInstance(channelAPI, destChannelGUI);
 | |
|                 destChannelGUI->setDeviceType(ChannelGUI::DeviceType::DeviceTx);
 | |
|                 destChannelGUI->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 QByteArray b = sourceChannelGUI->serialize();
 | |
|                 destChannelGUI->deserialize(b);
 | |
|             }
 | |
|         }
 | |
|         else if (destDeviceUI->m_deviceMIMOEngine) // MIMO device => Any type of channel is possible
 | |
|         {
 | |
|             const PluginAPI::ChannelRegistrations *rxChannelRegistrations = m_pluginManager->getRxChannelRegistrations();
 | |
|             const PluginAPI::ChannelRegistrations *txChannelRegistrations = m_pluginManager->getTxChannelRegistrations();
 | |
|             const PluginAPI::ChannelRegistrations *mimoChannelRegistrations = m_pluginManager->getMIMOChannelRegistrations();
 | |
|             const PluginInterface *pluginInterface = nullptr;
 | |
| 
 | |
|             for (const auto& channelRegistration : *rxChannelRegistrations)
 | |
|             {
 | |
|                 if (channelRegistration.m_channelIdURI == sourceChannelAPI->getURI())
 | |
|                 {
 | |
|                     pluginInterface = channelRegistration.m_plugin;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (pluginInterface) // Rx channel
 | |
|             {
 | |
|                 ChannelAPI *channelAPI;
 | |
|                 BasebandSampleSink *rxChannel = nullptr;
 | |
|                 pluginInterface->createRxChannel(destDeviceUI->m_deviceAPI, &rxChannel, &channelAPI);
 | |
|                 destChannelGUI = pluginInterface->createRxChannelGUI(destDeviceUI, rxChannel);
 | |
|                 destDeviceUI->registerRxChannelInstance(channelAPI, destChannelGUI);
 | |
|                 destChannelGUI->setDeviceType(ChannelGUI::DeviceType::DeviceMIMO);
 | |
|                 destChannelGUI->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 QByteArray b = sourceChannelGUI->serialize();
 | |
|                 destChannelGUI->deserialize(b);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 for (const auto& channelRegistration : *txChannelRegistrations)
 | |
|                 {
 | |
|                     if (channelRegistration.m_channelIdURI == sourceChannelAPI->getURI())
 | |
|                     {
 | |
|                         pluginInterface = channelRegistration.m_plugin;
 | |
|                         break;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (pluginInterface) // Tx channel
 | |
|                 {
 | |
|                     ChannelAPI *channelAPI;
 | |
|                     BasebandSampleSource *txChannel = nullptr;
 | |
|                     pluginInterface->createTxChannel(destDeviceUI->m_deviceAPI, &txChannel, &channelAPI);
 | |
|                     destChannelGUI = pluginInterface->createTxChannelGUI(destDeviceUI, txChannel);
 | |
|                     destDeviceUI->registerTxChannelInstance(channelAPI, destChannelGUI);
 | |
|                     destChannelGUI->setDeviceType(ChannelGUI::DeviceType::DeviceMIMO);
 | |
|                     destChannelGUI->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                     QByteArray b = sourceChannelGUI->serialize();
 | |
|                     destChannelGUI->deserialize(b);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     for (const auto& channelRegistration : *mimoChannelRegistrations)
 | |
|                     {
 | |
|                         if (channelRegistration.m_channelIdURI == sourceChannelAPI->getURI())
 | |
|                         {
 | |
|                             pluginInterface = channelRegistration.m_plugin;
 | |
|                             break;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if (pluginInterface)
 | |
|                     {
 | |
|                         ChannelAPI *channelAPI;
 | |
|                         MIMOChannel *mimoChannel = nullptr;
 | |
|                         pluginInterface->createMIMOChannel(destDeviceUI->m_deviceAPI, &mimoChannel, &channelAPI);
 | |
|                         destChannelGUI = pluginInterface->createMIMOChannelGUI(destDeviceUI, mimoChannel);
 | |
|                         destDeviceUI->registerChannelInstance(channelAPI, destChannelGUI);
 | |
|                         destChannelGUI->setDeviceType(ChannelGUI::DeviceType::DeviceMIMO);
 | |
|                         destChannelGUI->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                         QByteArray b = sourceChannelGUI->serialize();
 | |
|                         destChannelGUI->deserialize(b);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         const DeviceAPI *destDeviceAPI = destDeviceUI->m_deviceAPI;
 | |
|         int workspaceIndex = sourceChannelGUI->getWorkspaceIndex();
 | |
|         Workspace *workspace = workspaceIndex < m_workspaces.size() ? m_workspaces[sourceChannelGUI->getWorkspaceIndex()] : nullptr;
 | |
| 
 | |
|         if (destChannelGUI && workspace)
 | |
|         {
 | |
|             QObject::connect(
 | |
|                 destChannelGUI,
 | |
|                 &ChannelGUI::moveToWorkspace,
 | |
|                 this,
 | |
|                 [this, destChannelGUI](int wsIndexDest){ this->channelMove(destChannelGUI, wsIndexDest); }
 | |
|             );
 | |
|             QObject::connect(
 | |
|                 destChannelGUI,
 | |
|                 &ChannelGUI::duplicateChannelEmitted,
 | |
|                 this,
 | |
|                 [this, destChannelGUI](){ this->channelDuplicate(destChannelGUI); }
 | |
|             );
 | |
|             QObject::connect(
 | |
|                 destChannelGUI,
 | |
|                 &ChannelGUI::moveToDeviceSet,
 | |
|                 this,
 | |
|                 [this, destChannelGUI](int dsIndexDest){ this->channelMoveToDeviceSet(destChannelGUI, dsIndexDest); }
 | |
|             );
 | |
| 
 | |
|             destChannelGUI->setDeviceSetIndex(dsIndexDestination);
 | |
|             destChannelGUI->setIndexToolTip(destDeviceAPI->getSamplingDeviceDisplayName());
 | |
|             destChannelGUI->setWorkspaceIndex(workspace->getIndex());
 | |
|             qDebug("MainWindow::channelDuplicate: adding %s to workspace #%d",
 | |
|                 qPrintable(destChannelGUI->getTitle()), workspace->getIndex());
 | |
|             workspace->addToMdiArea((QMdiSubWindow*) destChannelGUI);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::channelAddClicked(Workspace *workspace, int deviceSetIndex, int channelPluginIndex)
 | |
| {
 | |
|     if (deviceSetIndex < (int) m_deviceUIs.size())
 | |
|     {
 | |
|         DeviceUISet *deviceUI = m_deviceUIs[deviceSetIndex];
 | |
|         ChannelGUI *gui;
 | |
|         ChannelAPI *channelAPI;
 | |
|         const DeviceAPI *deviceAPI = deviceUI->m_deviceAPI;
 | |
| 
 | |
|         if (deviceUI->m_deviceSourceEngine) // source device => Rx channels
 | |
|         {
 | |
|             PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getRxChannelRegistrations(); // Available channel plugins
 | |
|             const PluginInterface *pluginInterface = (*channelRegistrations)[channelPluginIndex].m_plugin;
 | |
|             BasebandSampleSink *rxChannel = nullptr;
 | |
|             pluginInterface->createRxChannel(deviceUI->m_deviceAPI, &rxChannel, &channelAPI);
 | |
|             gui = pluginInterface->createRxChannelGUI(deviceUI, rxChannel);
 | |
|             deviceUI->registerRxChannelInstance(channelAPI, gui);
 | |
|             gui->setDeviceType(ChannelGUI::DeviceType::DeviceRx);
 | |
|             gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|             gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|         }
 | |
|         else if (deviceUI->m_deviceSinkEngine) // sink device => Tx channels
 | |
|         {
 | |
|             PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getTxChannelRegistrations(); // Available channel plugins
 | |
|             const PluginInterface *pluginInterface = (*channelRegistrations)[channelPluginIndex].m_plugin;
 | |
|             BasebandSampleSource *txChannel = nullptr;
 | |
|             pluginInterface->createTxChannel(deviceUI->m_deviceAPI, &txChannel, &channelAPI);
 | |
|             gui = pluginInterface->createTxChannelGUI(deviceUI, txChannel);
 | |
|             deviceUI->registerTxChannelInstance(channelAPI, gui);
 | |
|             gui->setDeviceType(ChannelGUI::DeviceType::DeviceTx);
 | |
|             gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|             gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|         }
 | |
|         else if (deviceUI->m_deviceMIMOEngine) // MIMO device => all possible channels. Depends on index range
 | |
|         {
 | |
|             int nbMIMOChannels = deviceUI->getNumberOfAvailableMIMOChannels();
 | |
|             int nbRxChannels = deviceUI->getNumberOfAvailableRxChannels();
 | |
|             int nbTxChannels = deviceUI->getNumberOfAvailableTxChannels();
 | |
|             qDebug("MainWindow::channelAddClicked: MIMO: dev %d : nbMIMO: %d nbRx: %d nbTx: %d selected: %d",
 | |
|                 deviceSetIndex, nbMIMOChannels, nbRxChannels, nbTxChannels, channelPluginIndex);
 | |
| 
 | |
|             if (channelPluginIndex < nbMIMOChannels)
 | |
|             {
 | |
|                 PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getMIMOChannelRegistrations(); // Available channel plugins
 | |
|                 const PluginInterface *pluginInterface = (*channelRegistrations)[channelPluginIndex].m_plugin;
 | |
|                 MIMOChannel *mimoChannel = nullptr;
 | |
|                 pluginInterface->createMIMOChannel(deviceUI->m_deviceAPI, &mimoChannel, &channelAPI);
 | |
|                 gui = pluginInterface->createMIMOChannelGUI(deviceUI, mimoChannel);
 | |
|                 deviceUI->registerChannelInstance(channelAPI, gui);
 | |
|                 gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|             }
 | |
|             else if (channelPluginIndex < nbMIMOChannels + nbRxChannels) // Rx
 | |
|             {
 | |
|                 PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getRxChannelRegistrations(); // Available channel plugins
 | |
|                 const PluginInterface *pluginInterface = (*channelRegistrations)[channelPluginIndex - nbMIMOChannels].m_plugin;
 | |
|                 BasebandSampleSink *rxChannel = nullptr;
 | |
|                 pluginInterface->createRxChannel(deviceUI->m_deviceAPI, &rxChannel, &channelAPI);
 | |
|                 gui = pluginInterface->createRxChannelGUI(deviceUI, rxChannel);
 | |
|                 deviceUI->registerRxChannelInstance(channelAPI, gui);
 | |
|                 gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|             }
 | |
|             else if (channelPluginIndex < nbMIMOChannels + nbRxChannels + nbTxChannels)
 | |
|             {
 | |
|                 PluginAPI::ChannelRegistrations *channelRegistrations = m_pluginManager->getTxChannelRegistrations(); // Available channel plugins
 | |
|                 const PluginInterface *pluginInterface = (*channelRegistrations)[channelPluginIndex - nbMIMOChannels - nbRxChannels].m_plugin;
 | |
|                 BasebandSampleSource *txChannel = nullptr;
 | |
|                 pluginInterface->createTxChannel(deviceUI->m_deviceAPI, &txChannel, &channelAPI);
 | |
|                 gui = pluginInterface->createTxChannelGUI(deviceUI, txChannel);
 | |
|                 deviceUI->registerTxChannelInstance(channelAPI, gui);
 | |
|                 gui->setIndex(channelAPI->getIndexInDeviceSet());
 | |
|                 gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             gui->setDeviceType(ChannelGUI::DeviceType::DeviceMIMO);
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (gui)
 | |
|         {
 | |
|             QObject::connect(
 | |
|                 gui,
 | |
|                 &ChannelGUI::moveToWorkspace,
 | |
|                 this,
 | |
|                 [this, gui](int wsIndexDest){ this->channelMove(gui, wsIndexDest); }
 | |
|             );
 | |
|             QObject::connect(
 | |
|                 gui,
 | |
|                 &ChannelGUI::duplicateChannelEmitted,
 | |
|                 this,
 | |
|                 [this, gui](){ this->channelDuplicate(gui); }
 | |
|             );
 | |
|             QObject::connect(
 | |
|                 gui,
 | |
|                 &ChannelGUI::moveToDeviceSet,
 | |
|                 this,
 | |
|                 [this, gui](int dsIndexDest){ this->channelMoveToDeviceSet(gui, dsIndexDest); }
 | |
|             );
 | |
| 
 | |
|             gui->setDeviceSetIndex(deviceSetIndex);
 | |
|             gui->setIndexToolTip(deviceAPI->getSamplingDeviceDisplayName());
 | |
|             gui->setWorkspaceIndex(workspace->getIndex());
 | |
|             qDebug("MainWindow::channelAddClicked: adding %s to workspace #%d",
 | |
|                 qPrintable(gui->getTitle()), workspace->getIndex());
 | |
|             workspace->addToMdiArea((QMdiSubWindow*) gui);
 | |
|             loadDefaultPreset(channelAPI->getURI(), gui);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::featureAddClicked(Workspace *workspace, int featureIndex)
 | |
| {
 | |
|     qDebug("MainWindow::featureAddClicked: W%d feature at %d", workspace->getIndex(), featureIndex);
 | |
| 
 | |
|     int currentFeatureSetIndex = 0; // Do it in the unique set
 | |
|     FeatureUISet *featureUISet = m_featureUIs[currentFeatureSetIndex];
 | |
|     qDebug("MainWindow::featureAddClicked: m_apiAdapter: %p", m_apiAdapter);
 | |
|     PluginAPI::FeatureRegistrations *featureRegistrations = m_pluginManager->getFeatureRegistrations(); // Available feature plugins
 | |
|     const PluginInterface *pluginInterface = (*featureRegistrations)[featureIndex].m_plugin;
 | |
|     Feature *feature = pluginInterface->createFeature(m_apiAdapter);
 | |
|     FeatureGUI *gui = pluginInterface->createFeatureGUI(featureUISet, feature);
 | |
|     featureUISet->registerFeatureInstance(gui, feature);
 | |
|     gui->setIndex(feature->getIndexInFeatureSet());
 | |
|     gui->setWorkspaceIndex(workspace->getIndex());
 | |
|     gui->setDisplayedame(pluginInterface->getPluginDescriptor().displayedName);
 | |
|     workspace->addToMdiArea((QMdiSubWindow*) gui);
 | |
|     loadDefaultPreset(feature->getURI(), gui);
 | |
| 
 | |
|     QObject::connect(
 | |
|         gui,
 | |
|         &FeatureGUI::moveToWorkspace,
 | |
|         this,
 | |
|         [this, gui](int wsIndexDest){ this->featureMove(gui, wsIndexDest); }
 | |
|     );
 | |
| }
 | |
| 
 | |
| void MainWindow::featureMove(FeatureGUI *gui, int wsIndexDestnation)
 | |
| {
 | |
|     int wsIndexOrigin = gui->getWorkspaceIndex();
 | |
| 
 | |
|     if (wsIndexOrigin == wsIndexDestnation) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_workspaces[wsIndexOrigin]->removeFromMdiArea(gui);
 | |
|     gui->setWorkspaceIndex(wsIndexDestnation);
 | |
|     m_workspaces[wsIndexDestnation]->addToMdiArea(gui);
 | |
| }
 | |
| 
 | |
| void MainWindow::deviceMove(DeviceGUI *gui, int wsIndexDestnation)
 | |
| {
 | |
|     int wsIndexOrigin = gui->getWorkspaceIndex();
 | |
|     qDebug("MainWindow::deviceMove: %s from %d to %d",
 | |
|         qPrintable(gui->getTitle()), wsIndexOrigin, wsIndexDestnation);
 | |
| 
 | |
|     if (wsIndexOrigin == wsIndexDestnation) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_workspaces[wsIndexOrigin]->removeFromMdiArea(gui);
 | |
|     gui->setWorkspaceIndex(wsIndexDestnation);
 | |
|     m_workspaces[wsIndexDestnation]->addToMdiArea(gui);
 | |
| }
 | |
| 
 | |
| void MainWindow::channelMove(ChannelGUI *gui, int wsIndexDestnation)
 | |
| {
 | |
|     int wsIndexOrigin = gui->getWorkspaceIndex();
 | |
| 
 | |
|     if (wsIndexOrigin == wsIndexDestnation) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_workspaces[wsIndexOrigin]->removeFromMdiArea(gui);
 | |
|     gui->setWorkspaceIndex(wsIndexDestnation);
 | |
|     m_workspaces[wsIndexDestnation]->addToMdiArea(gui);
 | |
| }
 | |
| 
 | |
| void MainWindow::mainSpectrumMove(MainSpectrumGUI *gui, int wsIndexDestnation)
 | |
| {
 | |
|     int wsIndexOrigin = gui->getWorkspaceIndex();
 | |
|     qDebug("MainWindow::mainSpectrumMove: %s from %d to %d",
 | |
|         qPrintable(gui->getTitle()), wsIndexOrigin, wsIndexDestnation);
 | |
| 
 | |
|     if (wsIndexOrigin == wsIndexDestnation) {
 | |
|         return;
 | |
|     }
 | |
| 
 | |
|     m_workspaces[wsIndexOrigin]->removeFromMdiArea(gui);
 | |
|     gui->setWorkspaceIndex(wsIndexDestnation);
 | |
|     m_workspaces[wsIndexDestnation]->addToMdiArea(gui);
 | |
| }
 | |
| 
 | |
| void MainWindow::mainSpectrumShow(int deviceSetIndex)
 | |
| {
 | |
|     DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|     deviceUISet->m_mainSpectrumGUI->show();
 | |
|     deviceUISet->m_mainSpectrumGUI->raise();
 | |
| }
 | |
| 
 | |
| void MainWindow::mainSpectrumRequestDeviceCenterFrequency(int deviceSetIndex, qint64 deviceCenterFrequency)
 | |
| {
 | |
|     DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
|     DeviceAPI *deviceAPI = deviceUISet->m_deviceAPI;
 | |
| 
 | |
|     if (deviceAPI->getSampleSource()) {
 | |
|         deviceAPI->getSampleSource()->setCenterFrequency(deviceCenterFrequency);
 | |
|     } else if (deviceAPI->getSampleSink()) {
 | |
|         deviceAPI->getSampleSink()->setCenterFrequency(deviceCenterFrequency);
 | |
|     }
 | |
|     // Not implemented for MIMO
 | |
| }
 | |
| 
 | |
| void MainWindow::showAllChannels(int deviceSetIndex)
 | |
| {
 | |
|     DeviceUISet *deviceUISet = m_deviceUIs[deviceSetIndex];
 | |
| 
 | |
|     for (int i = 0; i < deviceUISet->getNumberOfChannels(); i++)
 | |
|     {
 | |
|         deviceUISet->getChannelGUIAt(i)->show();
 | |
|         deviceUISet->getChannelGUIAt(i)->raise();
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Start all devices in the workspace
 | |
| void MainWindow::startAllDevices(const Workspace *workspace) const
 | |
| {
 | |
|     int workspaceIndex = workspace->getIndex();
 | |
|     for (auto deviceUI : m_deviceUIs)
 | |
|     {
 | |
|         if (deviceUI->m_deviceAPI->getWorkspaceIndex() == workspaceIndex)
 | |
|         {
 | |
|             // We use WebAPI rather than call deviceUI->m_deviceAPI->startDeviceEngine();
 | |
|             // so that the start/stop button in the Device GUI is correctly updated
 | |
|             int deviceIndex = deviceUI->m_deviceAPI->getDeviceSetIndex();
 | |
|             ChannelWebAPIUtils::run(deviceIndex);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Stop all devices in the workspace
 | |
| void MainWindow::stopAllDevices(const Workspace *workspace) const
 | |
| {
 | |
|     int workspaceIndex = workspace->getIndex();
 | |
|     for (auto deviceUI : m_deviceUIs)
 | |
|     {
 | |
|         if (deviceUI->m_deviceAPI->getWorkspaceIndex() == workspaceIndex)
 | |
|         {
 | |
|             int deviceIndex = deviceUI->m_deviceAPI->getDeviceSetIndex();
 | |
|             ChannelWebAPIUtils::stop(deviceIndex);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::deviceStateChanged(DeviceAPI *deviceAPI)
 | |
| {
 | |
|     emit m_mainCore->deviceStateChanged(deviceAPI->getDeviceSetIndex(), deviceAPI);
 | |
| }
 | |
| 
 | |
| void MainWindow::openFeaturePresetsDialog(QPoint p, Workspace *workspace)
 | |
| {
 | |
|     FeaturePresetsDialog dialog;
 | |
|     dialog.setFeatureUISet(m_featureUIs[0]);
 | |
|     dialog.setPresets(m_mainCore->m_settings.getFeatureSetPresets());
 | |
|     dialog.setPluginAPI(m_pluginManager->getPluginAPI());
 | |
|     dialog.setWebAPIAdapter(m_apiAdapter);
 | |
|     dialog.setCurrentWorkspace(workspace);
 | |
|     dialog.setWorkspaces(&m_workspaces);
 | |
|     dialog.populateTree();
 | |
|     dialog.move(p);
 | |
|     new DialogPositioner(&dialog, true);
 | |
|     dialog.exec();
 | |
| 
 | |
|     if (dialog.wasPresetLoaded())
 | |
|     {
 | |
|         for (int i = 0; i < m_featureUIs[0]->getNumberOfFeatures(); i++)
 | |
|         {
 | |
|             FeatureGUI *gui = m_featureUIs[0]->getFeatureGuiAt(i);
 | |
|             QObject::connect(
 | |
|                 gui,
 | |
|                 &FeatureGUI::moveToWorkspace,
 | |
|                 this,
 | |
|                 [this, gui](int wsIndexDest){ this->featureMove(gui, wsIndexDest); }
 | |
|             );
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::openDeviceSetPresetsDialog(QPoint p, const DeviceGUI *deviceGUI)
 | |
| {
 | |
|     Workspace *workspace = m_workspaces[deviceGUI->getWorkspaceIndex()];
 | |
|     DeviceUISet *deviceUISet = m_deviceUIs[deviceGUI->getIndex()];
 | |
| 
 | |
|     DeviceSetPresetsDialog dialog;
 | |
|     dialog.setDeviceUISet(deviceUISet);
 | |
|     dialog.setPresets(m_mainCore->m_settings.getPresets());
 | |
|     dialog.setPluginAPI(m_pluginManager->getPluginAPI());
 | |
|     dialog.setCurrentWorkspace(workspace);
 | |
|     dialog.setWorkspaces(&m_workspaces);
 | |
|     dialog.populateTree((int) deviceGUI->getDeviceType());
 | |
|     dialog.move(p);
 | |
|     new DialogPositioner(&dialog, true);
 | |
|     dialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::deleteFeature(int featureSetIndex, int featureIndex)
 | |
| {
 | |
|     if ((featureSetIndex >= 0) && (featureSetIndex < (int) m_featureUIs.size()))
 | |
|     {
 | |
|         FeatureUISet *featureUISet = m_featureUIs[featureSetIndex];
 | |
|         featureUISet->deleteFeature(featureIndex);
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Look for and load a preset named Defaults/Default for the given plugin id
 | |
| void MainWindow::loadDefaultPreset(const QString& pluginId, SerializableInterface *serializableInterface)
 | |
| {
 | |
|     const QList<PluginPreset*>* presets = m_mainCore->m_settings.getPluginPresets();
 | |
| 
 | |
|     for (const auto preset : *presets)
 | |
|     {
 | |
|         if (preset->getGroup() == "Defaults"
 | |
|             && preset->getDescription() == "Default"
 | |
|             && preset->getPluginIdURI() == pluginId)
 | |
|         {
 | |
|             qDebug() << "MainWindow::loadDefaultPreset: Loading " << preset->getGroup() << preset->getDescription() << "for" << pluginId;
 | |
|             serializableInterface->deserialize(preset->getConfig());
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_About_triggered()
 | |
| {
 | |
| 	AboutDialog dlg(m_apiHost.isEmpty() ? "0.0.0.0" : m_apiHost, m_apiPort, m_mainCore->m_settings, this);
 | |
| 	dlg.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::updateStatus()
 | |
| {
 | |
|     if (m_dateTimeWidget) {
 | |
|         m_dateTimeWidget->setText(QDateTime::currentDateTime().toString("yyyy-MM-dd HH:mm:ss t"));
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::commandKeyPressed(Qt::Key key, Qt::KeyboardModifiers keyModifiers, bool release) const
 | |
| {
 | |
|     qDebug("MainWindow::commandKeyPressed: key: %x mod: %x %s", (int) key, (int) keyModifiers, release ? "release" : "press");
 | |
|     int currentDeviceSetIndex = 0;
 | |
| 
 | |
|     for (int i = 0; i < m_mainCore->m_settings.getCommandCount(); ++i)
 | |
|     {
 | |
|         const Command* command = m_mainCore->m_settings.getCommand(i);
 | |
| 
 | |
|         if (command->getAssociateKey()
 | |
|                 && (command->getRelease() == release)
 | |
|                 && (command->getKey() == key)
 | |
|                 && (command->getKeyModifiers() == keyModifiers))
 | |
|         {
 | |
|             auto* command_mod = const_cast<Command*>(command);
 | |
|             command_mod->run(m_apiServer->getHost(), m_apiServer->getPort(), currentDeviceSetIndex);
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::keyPressEvent(QKeyEvent* event)
 | |
| {
 | |
| #ifdef ANDROID
 | |
|     if (event->key() == Qt::Key_Back)
 | |
|     {
 | |
|         // On Android, we don't want to exit when back key is pressed, just run in the background
 | |
|         Android::moveTaskToBack();
 | |
|     }
 | |
|     else
 | |
| #endif
 | |
|     {
 | |
|         QMainWindow::keyPressEvent(event);
 | |
|     }
 | |
| }
 | |
| 
 | |
| void MainWindow::orientationChanged(Qt::ScreenOrientation orientation)
 | |
| {
 | |
| #ifdef ANDROID
 | |
|     // Adjust workspace tab position, to leave max space for MDI windows
 | |
|     if ((orientation == Qt::LandscapeOrientation) || (orientation == Qt::InvertedLandscapeOrientation)) {
 | |
|         setTabPosition(Qt::LeftDockWidgetArea, QTabWidget::West);
 | |
|     } else {
 | |
|         setTabPosition(Qt::LeftDockWidgetArea, QTabWidget::South);
 | |
|     }
 | |
| #else
 | |
|     (void) orientation;
 | |
| #endif
 | |
| }
 |