1
0
mirror of https://github.com/f4exb/sdrangel.git synced 2024-11-22 08:04:49 -05:00
sdrangel/sdrgui/mainwindow.cpp
2024-10-21 17:05:19 +01:00

3536 lines
133 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);
} 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->removeDeviceEngineAt(m_deviceSetIndex);
if (thread && !thread->isFinished()) // FIXME: Is there a race condition here? We might need to connect before calling thread->exit
{
connect(thread, &QThread::finished, m_mainWindow, &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();
} else if (m_deviceSinkEngine) {
delete deviceAPI->getSampleSink();
} else {
delete deviceAPI->getSampleMIMO();
}
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
}