Files
inspectrum/spectrogramcontrols.cpp
Mike Walters d6115cb458 Use double for sample rate
Fixes #158
2019-12-28 15:13:22 +01:00

229 lines
7.7 KiB
C++

/*
* Copyright (C) 2015, Mike Walters <mike@flomp.net>
* Copyright (C) 2015, Jared Boone <jared@sharebrained.com>
*
* This file is part of inspectrum.
*
* 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, either 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 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 "spectrogramcontrols.h"
#include <QIntValidator>
#include <QFileDialog>
#include <QSettings>
#include <QLabel>
#include <cmath>
#include "util.h"
SpectrogramControls::SpectrogramControls(const QString & title, QWidget * parent)
: QDockWidget::QDockWidget(title, parent)
{
widget = new QWidget(this);
layout = new QFormLayout(widget);
fileOpenButton = new QPushButton("Open file...", widget);
layout->addRow(fileOpenButton);
sampleRate = new QLineEdit();
auto double_validator = new QDoubleValidator(this);
double_validator->setBottom(0.0);
sampleRate->setValidator(double_validator);
layout->addRow(new QLabel(tr("Sample rate:")), sampleRate);
// Spectrogram settings
layout->addRow(new QLabel()); // TODO: find a better way to add an empty row?
layout->addRow(new QLabel(tr("<b>Spectrogram</b>")));
fftSizeSlider = new QSlider(Qt::Horizontal, widget);
fftSizeSlider->setRange(4, 13);
fftSizeSlider->setPageStep(1);
layout->addRow(new QLabel(tr("FFT size:")), fftSizeSlider);
zoomLevelSlider = new QSlider(Qt::Horizontal, widget);
zoomLevelSlider->setRange(0, 10);
zoomLevelSlider->setPageStep(1);
layout->addRow(new QLabel(tr("Zoom:")), zoomLevelSlider);
powerMaxSlider = new QSlider(Qt::Horizontal, widget);
powerMaxSlider->setRange(-140, 10);
layout->addRow(new QLabel(tr("Power max:")), powerMaxSlider);
powerMinSlider = new QSlider(Qt::Horizontal, widget);
powerMinSlider->setRange(-140, 10);
layout->addRow(new QLabel(tr("Power min:")), powerMinSlider);
scalesCheckBox = new QCheckBox(widget);
scalesCheckBox->setCheckState(Qt::Checked);
layout->addRow(new QLabel(tr("Scales:")), scalesCheckBox);
// Time selection settings
layout->addRow(new QLabel()); // TODO: find a better way to add an empty row?
layout->addRow(new QLabel(tr("<b>Time selection</b>")));
cursorsCheckBox = new QCheckBox(widget);
layout->addRow(new QLabel(tr("Enable cursors:")), cursorsCheckBox);
cursorSymbolsSpinBox = new QSpinBox();
cursorSymbolsSpinBox->setMinimum(1);
cursorSymbolsSpinBox->setMaximum(9999);
layout->addRow(new QLabel(tr("Symbols:")), cursorSymbolsSpinBox);
rateLabel = new QLabel();
layout->addRow(new QLabel(tr("Rate:")), rateLabel);
periodLabel = new QLabel();
layout->addRow(new QLabel(tr("Period:")), periodLabel);
symbolRateLabel = new QLabel();
layout->addRow(new QLabel(tr("Symbol rate:")), symbolRateLabel);
symbolPeriodLabel = new QLabel();
layout->addRow(new QLabel(tr("Symbol period:")), symbolPeriodLabel);
widget->setLayout(layout);
setWidget(widget);
connect(fftSizeSlider, &QSlider::valueChanged, this, &SpectrogramControls::fftSizeChanged);
connect(zoomLevelSlider, &QSlider::valueChanged, this, &SpectrogramControls::zoomLevelChanged);
connect(fileOpenButton, &QPushButton::clicked, this, &SpectrogramControls::fileOpenButtonClicked);
connect(cursorsCheckBox, &QCheckBox::stateChanged, this, &SpectrogramControls::cursorsStateChanged);
connect(powerMinSlider, &QSlider::valueChanged, this, &SpectrogramControls::powerMinChanged);
connect(powerMaxSlider, &QSlider::valueChanged, this, &SpectrogramControls::powerMaxChanged);
}
void SpectrogramControls::clearCursorLabels()
{
periodLabel->setText("");
rateLabel->setText("");
symbolPeriodLabel->setText("");
symbolRateLabel->setText("");
}
void SpectrogramControls::cursorsStateChanged(int state)
{
if (state == Qt::Unchecked) {
clearCursorLabels();
}
}
void SpectrogramControls::setDefaults()
{
fftOrZoomChanged();
cursorsCheckBox->setCheckState(Qt::Unchecked);
cursorSymbolsSpinBox->setValue(1);
// Try to set the sample rate from the last-used value
QSettings settings;
int savedSampleRate = settings.value("SampleRate", 8000000).toInt();
sampleRate->setText(QString::number(savedSampleRate));
fftSizeSlider->setValue(settings.value("FFTSize", 9).toInt());
powerMaxSlider->setValue(settings.value("PowerMax", 0).toInt());
powerMinSlider->setValue(settings.value("PowerMin", -100).toInt());
zoomLevelSlider->setValue(settings.value("ZoomLevel", 0).toInt());
}
void SpectrogramControls::fftOrZoomChanged(void)
{
int fftSize = pow(2, fftSizeSlider->value());
int zoomLevel = std::min(fftSize, (int)pow(2, zoomLevelSlider->value()));
emit fftOrZoomChanged(fftSize, zoomLevel);
}
void SpectrogramControls::fftSizeChanged(int value)
{
QSettings settings;
settings.setValue("FFTSize", value);
fftOrZoomChanged();
}
void SpectrogramControls::zoomLevelChanged(int value)
{
QSettings settings;
settings.setValue("ZoomLevel", value);
fftOrZoomChanged();
}
void SpectrogramControls::powerMinChanged(int value)
{
QSettings settings;
settings.setValue("PowerMin", value);
}
void SpectrogramControls::powerMaxChanged(int value)
{
QSettings settings;
settings.setValue("PowerMax", value);
}
void SpectrogramControls::fileOpenButtonClicked()
{
QSettings settings;
QString fileName;
QFileDialog fileSelect(this);
fileSelect.setNameFilter(tr("All files (*);;"
"complex<float> file (*.cfile *.cf32 *.fc32);;"
"complex<int8> HackRF file (*.cs8 *.sc8 *.c8);;"
"complex<int16> Fancy file (*.cs16 *.sc16 *.c16);;"
"complex<uint8> RTL-SDR file (*.cu8 *.uc8)"));
// Try and load a saved state
{
QByteArray savedState = settings.value("OpenFileState").toByteArray();
fileSelect.restoreState(savedState);
// Filter doesn't seem to be considered part of the saved state
QString lastUsedFilter = settings.value("OpenFileFilter").toString();
if(lastUsedFilter.size())
fileSelect.selectNameFilter(lastUsedFilter);
}
if(fileSelect.exec())
{
fileName = fileSelect.selectedFiles()[0];
// Remember the state of the dialog for the next time
QByteArray dialogState = fileSelect.saveState();
settings.setValue("OpenFileState", dialogState);
settings.setValue("OpenFileFilter", fileSelect.selectedNameFilter());
}
if (!fileName.isEmpty())
emit openFile(fileName);
}
void SpectrogramControls::timeSelectionChanged(float time)
{
if (cursorsCheckBox->checkState() == Qt::Checked) {
periodLabel->setText(QString::fromStdString(formatSIValue(time)) + "s");
rateLabel->setText(QString::fromStdString(formatSIValue(1 / time)) + "Hz");
int symbols = cursorSymbolsSpinBox->value();
symbolPeriodLabel->setText(QString::fromStdString(formatSIValue(time / symbols)) + "s");
symbolRateLabel->setText(QString::fromStdString(formatSIValue(symbols / time)) + "Bd");
}
}
void SpectrogramControls::zoomIn()
{
zoomLevelSlider->setValue(zoomLevelSlider->value() + 1);
}
void SpectrogramControls::zoomOut()
{
zoomLevelSlider->setValue(zoomLevelSlider->value() - 1);
}