mirror of
				https://github.com/f4exb/sdrangel.git
				synced 2025-10-24 17:40:24 -04:00 
			
		
		
		
	
		
			
				
	
	
		
			505 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			505 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////////
 | |
| // Copyright (C) 2012 maintech GmbH, Otto-Hahn-Str. 15, 97204 Hoechberg, Germany //
 | |
| // written by Christian Daniel                                                   //
 | |
| //                                                                               //
 | |
| // 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                  //
 | |
| //                                                                               //
 | |
| // 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 <QLabel>
 | |
| #include "mainwindow.h"
 | |
| #include "ui_mainwindow.h"
 | |
| #include "audio/audiodeviceinfo.h"
 | |
| #include "gui/indicator.h"
 | |
| #include "gui/presetitem.h"
 | |
| #include "gui/addpresetdialog.h"
 | |
| #include "gui/pluginsdialog.h"
 | |
| #include "gui/preferencesdialog.h"
 | |
| #include "gui/aboutdialog.h"
 | |
| #include "gui/rollupwidget.h"
 | |
| #include "dsp/dspengine.h"
 | |
| #include "dsp/spectrumvis.h"
 | |
| #include "dsp/dspcommands.h"
 | |
| #include "plugin/plugingui.h"
 | |
| #include "plugin/pluginapi.h"
 | |
| #include "plugin/plugingui.h"
 | |
| 
 | |
| #include <iostream>
 | |
| 
 | |
| MainWindow::MainWindow(QWidget* parent) :
 | |
| 	QMainWindow(parent),
 | |
| 	ui(new Ui::MainWindow),
 | |
| 	m_audioDeviceInfo(new AudioDeviceInfo),
 | |
| 	m_messageQueue(new MessageQueue),
 | |
| 	m_settings(),
 | |
| 	m_dspEngine(new DSPEngine(m_messageQueue)),
 | |
| 	m_lastEngineState((DSPEngine::State)-1),
 | |
| 	m_startOsmoSDRUpdateAfterStop(false),
 | |
| 	m_inputGUI(0),
 | |
| 	m_sampleRate(0),
 | |
| 	m_centerFrequency(0),
 | |
| 	m_pluginManager(new PluginManager(this, m_dspEngine))
 | |
| {
 | |
| 	ui->setupUi(this);
 | |
| 	delete ui->mainToolBar;
 | |
| 	createStatusBar();
 | |
| 
 | |
| 	setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea);
 | |
| 	setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea);
 | |
| 	setCorner(Qt::TopRightCorner, Qt::RightDockWidgetArea);
 | |
| 	setCorner(Qt::BottomRightCorner, Qt::RightDockWidgetArea);
 | |
| 
 | |
| 	// work around broken Qt dock widget ordering
 | |
| 	removeDockWidget(ui->inputDock);
 | |
| 	removeDockWidget(ui->processingDock);
 | |
| 	removeDockWidget(ui->presetDock);
 | |
| 	removeDockWidget(ui->channelDock);
 | |
| 	addDockWidget(Qt::LeftDockWidgetArea, ui->inputDock);
 | |
| 	addDockWidget(Qt::LeftDockWidgetArea, ui->processingDock);
 | |
| 	addDockWidget(Qt::LeftDockWidgetArea, ui->presetDock);
 | |
| 	addDockWidget(Qt::RightDockWidgetArea, ui->channelDock);
 | |
| 	ui->inputDock->show();
 | |
| 	ui->processingDock->show();
 | |
| 	ui->presetDock->show();
 | |
| 	ui->channelDock->show();
 | |
| 
 | |
| 	ui->menu_Window->addAction(ui->inputDock->toggleViewAction());
 | |
| 	ui->menu_Window->addAction(ui->processingDock->toggleViewAction());
 | |
| 	ui->menu_Window->addAction(ui->presetDock->toggleViewAction());
 | |
| 	ui->menu_Window->addAction(ui->channelDock->toggleViewAction());
 | |
| 
 | |
| 	connect(m_messageQueue, SIGNAL(messageEnqueued()), this, SLOT(handleMessages()), Qt::QueuedConnection);
 | |
| 
 | |
| 	connect(&m_statusTimer, SIGNAL(timeout()), this, SLOT(updateStatus()));
 | |
| 	m_statusTimer.start(500);
 | |
| 
 | |
| 	m_pluginManager->loadPlugins();
 | |
| 	bool sampleSourceSignalsBlocked = ui->sampleSource->blockSignals(true);
 | |
| 	m_pluginManager->fillSampleSourceSelector(ui->sampleSource);
 | |
| 	ui->sampleSource->blockSignals(sampleSourceSignalsBlocked);
 | |
| 
 | |
| 	m_dspEngine->start();
 | |
| 
 | |
| 	m_spectrumVis = new SpectrumVis(ui->glSpectrum);
 | |
| 	m_dspEngine->addSink(m_spectrumVis);
 | |
| 
 | |
| 	ui->glSpectrumGUI->setBuddies(m_dspEngine->getMessageQueue(), m_spectrumVis, ui->glSpectrum);
 | |
| 
 | |
| 	loadSettings();
 | |
| 
 | |
| 	int sampleSourceIndex = m_pluginManager->selectSampleSource(m_settings.getCurrent()->getSource()); // select SampleSource from settings
 | |
| 	if(sampleSourceIndex >= 0) {
 | |
| 		bool sampleSourceSignalsBlocked = ui->sampleSource->blockSignals(true);
 | |
| 		ui->sampleSource->setCurrentIndex(sampleSourceIndex);
 | |
| 		ui->sampleSource->blockSignals(sampleSourceSignalsBlocked);
 | |
| 	}
 | |
| 
 | |
| 	loadSettings(m_settings.getCurrent());
 | |
| 
 | |
| 	applySettings();
 | |
| 	updatePresets();
 | |
| }
 | |
| 
 | |
| MainWindow::~MainWindow()
 | |
| {
 | |
| 	m_dspEngine->stopAcquistion();
 | |
| 
 | |
| 	saveSettings();
 | |
| 
 | |
| 	m_pluginManager->freeAll();
 | |
| 
 | |
| 	m_dspEngine->removeSink(m_spectrumVis);
 | |
| 	delete m_spectrumVis;
 | |
| 	delete m_pluginManager;
 | |
| 
 | |
| 	m_dspEngine->stop();
 | |
| 
 | |
| 	delete m_dspEngine;
 | |
| 	delete m_messageQueue;
 | |
| 	delete ui;
 | |
| }
 | |
| 
 | |
| void MainWindow::addChannelCreateAction(QAction* action)
 | |
| {
 | |
| 	ui->menu_Channels->addAction(action);
 | |
| }
 | |
| 
 | |
| void MainWindow::addChannelRollup(QWidget* widget)
 | |
| {
 | |
| 	((ChannelWindow*)ui->channelDock->widget())->addRollupWidget(widget);
 | |
| 	ui->channelDock->show();
 | |
| 	ui->channelDock->raise();
 | |
| }
 | |
| 
 | |
| void MainWindow::addViewAction(QAction* action)
 | |
| {
 | |
| 	ui->menu_Window->addAction(action);
 | |
| }
 | |
| 
 | |
| void MainWindow::addChannelMarker(ChannelMarker* channelMarker)
 | |
| {
 | |
| 	ui->glSpectrum->addChannelMarker(channelMarker);
 | |
| }
 | |
| 
 | |
| void MainWindow::removeChannelMarker(ChannelMarker* channelMarker)
 | |
| {
 | |
| 	ui->glSpectrum->removeChannelMarker(channelMarker);
 | |
| }
 | |
| 
 | |
| void MainWindow::setInputGUI(QWidget* gui)
 | |
| {
 | |
| 	if(m_inputGUI != 0)
 | |
| 		ui->inputDock->widget()->layout()->removeWidget(m_inputGUI);
 | |
| 	if(gui != 0)
 | |
| 		ui->inputDock->widget()->layout()->addWidget(gui);
 | |
| 	m_inputGUI = gui;
 | |
| }
 | |
| 
 | |
| void MainWindow::loadSettings()
 | |
| {
 | |
| 	std::cerr << "MainWindow::loadSettings" << std::endl;
 | |
|     m_settings.load();
 | |
| 
 | |
|     for(int i = 0; i < m_settings.getPresetCount(); ++i)
 | |
|     {
 | |
|     	addPresetToTree(m_settings.getPreset(i));
 | |
|     }
 | |
| 
 | |
|     Preset* current = m_settings.getCurrent();
 | |
| 
 | |
|     loadSettings(current);
 | |
| }
 | |
| 
 | |
| void MainWindow::loadSettings(const Preset* preset)
 | |
| {
 | |
| 	std::cerr << "MainWindow::loadSettings(preset): " << preset->getSource().toStdString() << std::endl;
 | |
| 
 | |
| 	ui->glSpectrumGUI->deserialize(preset->getSpectrumConfig());
 | |
| 	ui->dcOffset->setChecked(preset->getDCOffsetCorrection());
 | |
| 	ui->iqImbalance->setChecked(preset->getIQImbalanceCorrection());
 | |
| 
 | |
| 	m_pluginManager->loadSettings(preset);
 | |
| 
 | |
| 	// has to be last step
 | |
| 	restoreState(preset->getLayout());
 | |
| }
 | |
| 
 | |
| void MainWindow::saveSettings()
 | |
| {
 | |
| 	std::cerr << "MainWindow::saveSettings" << std::endl;
 | |
| 
 | |
| 	saveSettings(m_settings.getCurrent());
 | |
| 	m_settings.save();
 | |
| 
 | |
| }
 | |
| 
 | |
| void MainWindow::saveSettings(Preset* preset)
 | |
| {
 | |
| 	std::cerr << "MainWindow::saveSettings(preset): " << preset->getSource().toStdString() << std::endl;
 | |
| 
 | |
| 	preset->setSpectrumConfig(ui->glSpectrumGUI->serialize());
 | |
|     preset->clearChannels();
 | |
|     m_pluginManager->saveSettings(preset);
 | |
| 
 | |
|     preset->setLayout(saveState());
 | |
| }
 | |
| 
 | |
| void MainWindow::createStatusBar()
 | |
| {
 | |
| 	m_sampleRateWidget = new QLabel(tr("Rate: 0 kHz"), this);
 | |
| 	m_sampleRateWidget->setToolTip(tr("Sample Rate"));
 | |
| 	statusBar()->addPermanentWidget(m_sampleRateWidget);
 | |
| 
 | |
| 	m_engineIdle = new Indicator(tr("Idle"), this);
 | |
| 	m_engineIdle->setToolTip(tr("DSP engine is idle"));
 | |
| 	statusBar()->addPermanentWidget(m_engineIdle);
 | |
| 
 | |
| 	m_engineRunning = new Indicator(tr("Run"), this);
 | |
| 	m_engineRunning->setToolTip(tr("DSP engine is running"));
 | |
| 	statusBar()->addPermanentWidget(m_engineRunning);
 | |
| 
 | |
| 	m_engineError = new Indicator(tr("Err"), this);
 | |
| 	m_engineError->setToolTip(tr("DSP engine failed"));
 | |
| 	statusBar()->addPermanentWidget(m_engineError);
 | |
| }
 | |
| 
 | |
| void MainWindow::closeEvent(QCloseEvent*)
 | |
| {
 | |
| }
 | |
| 
 | |
| void MainWindow::updateCenterFreqDisplay()
 | |
| {
 | |
| 	ui->glSpectrum->setCenterFrequency(m_centerFrequency);
 | |
| }
 | |
| 
 | |
| void MainWindow::updateSampleRate()
 | |
| {
 | |
| 	ui->glSpectrum->setSampleRate(m_sampleRate);
 | |
| 	m_sampleRateWidget->setText(tr("Rate: %1 kHz").arg((float)m_sampleRate / 1000));
 | |
| }
 | |
| 
 | |
| void MainWindow::updatePresets()
 | |
| {
 | |
| 	ui->presetTree->resizeColumnToContents(0);
 | |
| 	if(ui->presetTree->currentItem() != 0) {
 | |
| 		ui->presetDelete->setEnabled(true);
 | |
| 		ui->presetLoad->setEnabled(true);
 | |
| 	} else {
 | |
| 		ui->presetDelete->setEnabled(false);
 | |
| 		ui->presetLoad->setEnabled(false);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| QTreeWidgetItem* MainWindow::addPresetToTree(const Preset* preset)
 | |
| {
 | |
| 	QTreeWidgetItem* group = 0;
 | |
| 	for(int i = 0; i < ui->presetTree->topLevelItemCount(); i++) {
 | |
| 		if(ui->presetTree->topLevelItem(i)->text(0) == preset->getGroup()) {
 | |
| 			group = ui->presetTree->topLevelItem(i);
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 	if(group == 0) {
 | |
| 		QStringList sl;
 | |
| 		sl.append(preset->getGroup());
 | |
| 		group = new QTreeWidgetItem(ui->presetTree, sl, PGroup);
 | |
| 		group->setFirstColumnSpanned(true);
 | |
| 		group->setExpanded(true);
 | |
| 		ui->presetTree->sortByColumn(0, Qt::AscendingOrder);
 | |
| 	}
 | |
| 	QStringList sl;
 | |
| 	sl.append(QString("%1 kHz").arg(preset->getCenterFrequency() / 1000));
 | |
| 	sl.append(preset->getDescription());
 | |
| 	PresetItem* item = new PresetItem(group, sl, preset->getCenterFrequency(), PItem);
 | |
| 	item->setTextAlignment(0, Qt::AlignRight);
 | |
| 	item->setData(0, Qt::UserRole, qVariantFromValue(preset));
 | |
| 	ui->presetTree->resizeColumnToContents(0);
 | |
| 
 | |
| 	updatePresets();
 | |
| 	return item;
 | |
| }
 | |
| 
 | |
| void MainWindow::applySettings()
 | |
| {
 | |
| 	updateCenterFreqDisplay();
 | |
| 	updateSampleRate();
 | |
| }
 | |
| 
 | |
| void MainWindow::handleMessages()
 | |
| {
 | |
| 	Message* message;
 | |
| 	while((message = m_messageQueue->accept()) != 0) {
 | |
| 		qDebug("Message: %s", message->getIdentifier());
 | |
| 		if(DSPEngineReport::match(message)) {
 | |
| 			DSPEngineReport* rep = (DSPEngineReport*)message;
 | |
| 			m_sampleRate = rep->getSampleRate();
 | |
| 			m_centerFrequency = rep->getCenterFrequency();
 | |
| 			qDebug("SampleRate:%d, CenterFrequency:%llu", rep->getSampleRate(), rep->getCenterFrequency());
 | |
| 			updateCenterFreqDisplay();
 | |
| 			updateSampleRate();
 | |
| 			message->completed();
 | |
| 		} else {
 | |
| 			if(!m_pluginManager->handleMessage(message))
 | |
| 				message->completed();
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::updateStatus()
 | |
| {
 | |
| 	int state = m_dspEngine->state();
 | |
| 	if(m_lastEngineState != state) {
 | |
| 		switch(state) {
 | |
| 			case DSPEngine::StNotStarted:
 | |
| 				m_engineIdle->setColor(Qt::gray);
 | |
| 				m_engineRunning->setColor(Qt::gray);
 | |
| 				m_engineError->setColor(Qt::gray);
 | |
| 				statusBar()->clearMessage();
 | |
| 				break;
 | |
| 
 | |
| 			case DSPEngine::StIdle:
 | |
| 				m_engineIdle->setColor(Qt::cyan);
 | |
| 				m_engineRunning->setColor(Qt::gray);
 | |
| 				m_engineError->setColor(Qt::gray);
 | |
| 				statusBar()->clearMessage();
 | |
| 				break;
 | |
| 
 | |
| 			case DSPEngine::StRunning:
 | |
| 				m_engineIdle->setColor(Qt::gray);
 | |
| 				m_engineRunning->setColor(Qt::green);
 | |
| 				m_engineError->setColor(Qt::gray);
 | |
| 				statusBar()->showMessage(tr("Sampling from %1").arg(m_dspEngine->deviceDescription()));
 | |
| 				break;
 | |
| 
 | |
| 			case DSPEngine::StError:
 | |
| 				m_engineIdle->setColor(Qt::gray);
 | |
| 				m_engineRunning->setColor(Qt::gray);
 | |
| 				m_engineError->setColor(Qt::red);
 | |
| 				statusBar()->showMessage(tr("Error: %1").arg(m_dspEngine->errorMessage()));
 | |
| 				break;
 | |
| 		}
 | |
| 		m_lastEngineState = state;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Start_triggered()
 | |
| {
 | |
| 	m_dspEngine->startAcquisition();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Stop_triggered()
 | |
| {
 | |
| 	m_dspEngine->stopAcquistion();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_dcOffset_toggled(bool checked)
 | |
| {
 | |
| 	m_settings.getCurrent()->setDCOffsetCorrection(checked);
 | |
| 	m_dspEngine->configureCorrections(m_settings.getCurrent()->getDCOffsetCorrection(), m_settings.getCurrent()->getIQImbalanceCorrection());
 | |
| }
 | |
| 
 | |
| void MainWindow::on_iqImbalance_toggled(bool checked)
 | |
| {
 | |
| 	m_settings.getCurrent()->setIQImbalanceCorrection(checked);
 | |
| 	m_dspEngine->configureCorrections(m_settings.getCurrent()->getDCOffsetCorrection(), m_settings.getCurrent()->getIQImbalanceCorrection());
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_View_Fullscreen_toggled(bool checked)
 | |
| {
 | |
| 	if(checked)
 | |
| 		showFullScreen();
 | |
| 	else showNormal();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetSave_clicked()
 | |
| {
 | |
| 	QStringList groups;
 | |
| 	QString group;
 | |
| 	QString description = "";
 | |
| 	for(int i = 0; i < ui->presetTree->topLevelItemCount(); i++)
 | |
| 		groups.append(ui->presetTree->topLevelItem(i)->text(0));
 | |
| 
 | |
| 	QTreeWidgetItem* item = ui->presetTree->currentItem();
 | |
| 	if(item != 0) {
 | |
| 		if(item->type() == PGroup)
 | |
| 			group = item->text(0);
 | |
| 		else if(item->type() == PItem) {
 | |
| 			group = item->parent()->text(0);
 | |
| 			description = item->text(0);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	AddPresetDialog dlg(groups, group, this);
 | |
| 
 | |
| 	if (description.length() > 0) {
 | |
| 		dlg.setDescription(description);
 | |
| 	}
 | |
| 
 | |
| 	if(dlg.exec() == QDialog::Accepted) {
 | |
| 		Preset* preset = m_settings.newPreset(dlg.group(), dlg.description());
 | |
| 		saveSettings(preset);
 | |
| 
 | |
| 		ui->presetTree->setCurrentItem(addPresetToTree(preset));
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetUpdate_clicked()
 | |
| {
 | |
| 	QTreeWidgetItem* item = ui->presetTree->currentItem();
 | |
| 	
 | |
| 	if(item != 0) {
 | |
| 		if(item->type() == PItem) {
 | |
| 			const Preset* preset = qvariant_cast<const Preset*>(item->data(0, Qt::UserRole));
 | |
| 			if (preset != 0) {
 | |
| 				Preset* preset_mod = const_cast<Preset*>(preset);
 | |
| 				saveSettings(preset_mod);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetLastLoad_clicked()
 | |
| {
 | |
| 	m_settings.load();
 | |
| 	applySettings();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetLoad_clicked()
 | |
| {
 | |
| 	QTreeWidgetItem* item = ui->presetTree->currentItem();
 | |
| 	if(item == 0) {
 | |
| 		updatePresets();
 | |
| 		return;
 | |
| 	}
 | |
| 	const Preset* preset = qvariant_cast<const Preset*>(item->data(0, Qt::UserRole));
 | |
| 	if(preset == 0) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	loadSettings(preset);
 | |
| 	applySettings();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetDelete_clicked()
 | |
| {
 | |
| 	QTreeWidgetItem* item = ui->presetTree->currentItem();
 | |
| 	if(item == 0) {
 | |
| 		updatePresets();
 | |
| 		return;
 | |
| 	}
 | |
| 	const Preset* preset = qvariant_cast<const Preset*>(item->data(0, Qt::UserRole));
 | |
| 	if(preset == 0)
 | |
| 		return;
 | |
| 
 | |
| 	if(QMessageBox::question(this, tr("Delete Preset"), tr("Do you want to delete preset '%1'?").arg(preset->getDescription()), QMessageBox::No | QMessageBox::Yes, QMessageBox::No) == QMessageBox::Yes) {
 | |
| 		delete item;
 | |
| 		m_settings.deletePreset(preset);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetTree_currentItemChanged(QTreeWidgetItem *current, QTreeWidgetItem *previous)
 | |
| {
 | |
| 	updatePresets();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_presetTree_itemActivated(QTreeWidgetItem *item, int column)
 | |
| {
 | |
| 	on_presetLoad_clicked();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Loaded_Plugins_triggered()
 | |
| {
 | |
| 	PluginsDialog pluginsDialog(m_pluginManager, this);
 | |
| 
 | |
| 	pluginsDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_Preferences_triggered()
 | |
| {
 | |
| 	PreferencesDialog preferencesDialog(m_audioDeviceInfo, this);
 | |
| 
 | |
| 	preferencesDialog.exec();
 | |
| }
 | |
| 
 | |
| void MainWindow::on_sampleSource_currentIndexChanged(int index)
 | |
| {
 | |
| 	m_pluginManager->selectSampleSource(ui->sampleSource->currentIndex());
 | |
| }
 | |
| 
 | |
| void MainWindow::on_action_About_triggered()
 | |
| {
 | |
| 	AboutDialog dlg(this);
 | |
| 	dlg.exec();
 | |
| }
 |