sdrangel/plugins/channelrx/demodatv/atvdemodgui.cpp

567 lines
16 KiB
C++
Raw Normal View History

2017-02-23 02:18:56 -05:00
///////////////////////////////////////////////////////////////////////////////////
// Copyright (C) 2017 F4HKW //
// for F4EXB / SDRAngel //
// //
// 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 <QDockWidget>
#include <QMainWindow>
#include "atvdemodgui.h"
#include "device/devicesourceapi.h"
#include "dsp/downchannelizer.h"
#include "dsp/threadedbasebandsamplesink.h"
#include "dsp/scopevisng.h"
2017-02-23 02:18:56 -05:00
#include "ui_atvdemodgui.h"
#include "plugin/pluginapi.h"
#include "util/simpleserializer.h"
#include "util/db.h"
#include "gui/basicchannelsettingswidget.h"
#include "dsp/dspengine.h"
#include "mainwindow.h"
#include "atvdemod.h"
const QString ATVDemodGUI::m_strChannelID = "sdrangel.channel.demodatv";
ATVDemodGUI* ATVDemodGUI::create(PluginAPI* objPluginAPI,
DeviceSourceAPI *objDeviceAPI)
2017-02-23 02:18:56 -05:00
{
ATVDemodGUI* gui = new ATVDemodGUI(objPluginAPI, objDeviceAPI);
return gui;
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::destroy()
{
delete this;
}
void ATVDemodGUI::setName(const QString& strName)
{
setObjectName(strName);
}
QString ATVDemodGUI::getName() const
{
return objectName();
2017-02-23 02:18:56 -05:00
}
qint64 ATVDemodGUI::getCenterFrequency() const
{
return m_objChannelMarker.getCenterFrequency();
}
void ATVDemodGUI::setCenterFrequency(qint64 intCenterFrequency)
{
m_objChannelMarker.setCenterFrequency(intCenterFrequency);
applySettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::resetToDefaults()
{
blockApplySettings(true);
2017-02-23 02:18:56 -05:00
//********** ATV Default values **********
ui->synchLevel->setValue(100);
ui->blackLevel->setValue(310);
ui->lineTime->setValue(640);
ui->topTime->setValue(3);
ui->modulation->setCurrentIndex(0);
ui->fps->setCurrentIndex(0);
ui->hSync->setChecked(true);
ui->vSync->setChecked(true);
ui->halfImage->setChecked(false);
2017-03-19 14:10:48 -04:00
ui->invertVideo->setChecked(false);
//********** RF Default values **********
ui->decimatorEnable->setChecked(false);
ui->rfFiltering->setChecked(false);
ui->rfBW->setValue(10);
ui->rfOppBW->setValue(10);
ui->bfo->setValue(0);
2017-02-23 02:18:56 -05:00
blockApplySettings(false);
applySettings();
2017-02-23 02:18:56 -05:00
}
QByteArray ATVDemodGUI::serialize() const
{
SimpleSerializer s(1);
2017-02-23 02:18:56 -05:00
s.writeS32(1, m_objChannelMarker.getCenterFrequency());
s.writeU32(2, m_objChannelMarker.getColor().rgb());
s.writeS32(3, ui->synchLevel->value());
s.writeS32(4, ui->blackLevel->value());
s.writeS32(5, ui->lineTime->value());
s.writeS32(6, ui->topTime->value());
s.writeS32(7, ui->modulation->currentIndex());
s.writeS32(8, ui->fps->currentIndex());
s.writeBool(9, ui->hSync->isChecked());
s.writeBool(10, ui->vSync->isChecked());
s.writeBool(11, ui->halfImage->isChecked());
s.writeS32(12, ui->rfBW->value());
s.writeS32(13, ui->rfOppBW->value());
s.writeS32(14, ui->bfo->value());
2017-03-19 14:10:48 -04:00
s.writeBool(15, ui->invertVideo->isChecked());
2017-02-23 02:18:56 -05:00
return s.final();
}
bool ATVDemodGUI::deserialize(const QByteArray& arrData)
{
SimpleDeserializer d(arrData);
if (!d.isValid())
2017-02-23 02:18:56 -05:00
{
resetToDefaults();
return false;
}
2017-02-23 02:18:56 -05:00
if (d.getVersion() == 1)
2017-02-23 02:18:56 -05:00
{
QByteArray bytetmp;
uint32_t u32tmp;
int tmp;
bool booltmp;
2017-02-23 02:18:56 -05:00
blockApplySettings(true);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.blockSignals(true);
d.readS32(1, &tmp, 0);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.setCenterFrequency(tmp);
// if (d.readU32(2, &u32tmp)) {
// m_objChannelMarker.setColor(u32tmp);
// } else {
// m_objChannelMarker.setColor(Qt::white);
// }
2017-02-23 02:18:56 -05:00
d.readS32(3, &tmp, 100);
ui->synchLevel->setValue(tmp);
d.readS32(4, &tmp, 310);
ui->blackLevel->setValue(tmp);
d.readS32(5, &tmp, 640);
ui->lineTime->setValue(tmp);
d.readS32(6, &tmp, 3);
ui->topTime->setValue(tmp);
d.readS32(7, &tmp, 0);
ui->modulation->setCurrentIndex(tmp);
d.readS32(8, &tmp, 0);
ui->fps->setCurrentIndex(tmp);
d.readBool(9, &booltmp, true);
ui->hSync->setChecked(booltmp);
d.readBool(10, &booltmp, true);
ui->vSync->setChecked(booltmp);
d.readBool(11, &booltmp, false);
ui->halfImage->setChecked(booltmp);
d.readS32(12, &tmp, 10);
ui->rfBW->setValue(tmp);
d.readS32(13, &tmp, 10);
ui->rfOppBW->setValue(tmp);
d.readS32(14, &tmp, 10);
ui->bfo->setValue(tmp);
2017-03-19 14:10:48 -04:00
d.readBool(15, &booltmp, true);
ui->invertVideo->setChecked(booltmp);
2017-02-23 02:18:56 -05:00
blockApplySettings(false);
m_objChannelMarker.blockSignals(false);
applySettings();
return true;
}
2017-02-23 02:18:56 -05:00
else
{
resetToDefaults();
return false;
}
2017-02-23 02:18:56 -05:00
}
bool ATVDemodGUI::handleMessage(const Message& objMessage)
{
if (ATVDemod::MsgReportEffectiveSampleRate::match(objMessage))
{
int sampleRate = ((ATVDemod::MsgReportEffectiveSampleRate&)objMessage).getSampleRate();
ui->channelSampleRateText->setText(tr("%1k").arg(sampleRate/1000.0f, 0, 'f', 0));
setRFFiltersSlidersRange(sampleRate);
return true;
}
else
{
return false;
}
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::viewChanged()
{
qDebug("ATVDemodGUI::viewChanged");
applySettings();
applyRFSettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::channelSampleRateChanged()
{
qDebug("ATVDemodGUI::channelSampleRateChanged");
applySettings();
applyRFSettings();
}
void ATVDemodGUI::handleSourceMessages()
{
Message* message;
while ((message = m_objATVDemod->getOutputMessageQueue()->pop()) != 0)
{
if (handleMessage(*message))
{
delete message;
}
}
}
2017-02-23 02:18:56 -05:00
void ATVDemodGUI::onWidgetRolled(QWidget* widget, bool rollDown)
{
}
void ATVDemodGUI::onMenuDoubleClicked()
{
if (!m_blnBasicSettingsShown)
2017-02-23 02:18:56 -05:00
{
m_blnBasicSettingsShown = true;
BasicChannelSettingsWidget* bcsw = new BasicChannelSettingsWidget(
&m_objChannelMarker, this);
2017-02-23 02:18:56 -05:00
bcsw->show();
}
}
ATVDemodGUI::ATVDemodGUI(PluginAPI* objPluginAPI, DeviceSourceAPI *objDeviceAPI,
QWidget* objParent) :
RollupWidget(objParent),
ui(new Ui::ATVDemodGUI),
m_objPluginAPI(objPluginAPI),
m_objDeviceAPI(objDeviceAPI),
m_objChannelMarker(this),
m_blnBasicSettingsShown(false),
m_blnDoApplySettings(true),
m_intTickCount(0)
2017-02-23 02:18:56 -05:00
{
ui->setupUi(this);
setAttribute(Qt::WA_DeleteOnClose, true);
connect(this, SIGNAL(widgetRolled(QWidget*,bool)), this, SLOT(onWidgetRolled(QWidget*,bool)));
connect(this, SIGNAL(menuDoubleClickEvent()), this, SLOT(onMenuDoubleClicked()));
2017-02-23 02:18:56 -05:00
m_objScopeVis = new ScopeVisNG(ui->glScope);
m_objATVDemod = new ATVDemod(m_objScopeVis);
m_objATVDemod->setATVScreen(ui->screenTV);
2017-02-23 02:18:56 -05:00
m_objChannelizer = new DownChannelizer(m_objATVDemod);
m_objThreadedChannelizer = new ThreadedBasebandSampleSink(m_objChannelizer, this);
2017-02-23 02:18:56 -05:00
m_objDeviceAPI->addThreadedSink(m_objThreadedChannelizer);
ui->glScope->connectTimer(m_objPluginAPI->getMainWindow()->getMasterTimer());
connect(&m_objPluginAPI->getMainWindow()->getMasterTimer(), SIGNAL(timeout()), this, SLOT(tick())); // 50 ms
2017-02-23 02:18:56 -05:00
2017-03-15 13:56:38 -04:00
ui->deltaFrequency->setColorMapper(ColorMapper(ColorMapper::ReverseGold));
ui->deltaFrequency->setValueRange(7, 0U, 9999999U);
2017-03-15 13:56:38 -04:00
connect(m_objChannelizer, SIGNAL(inputSampleRateChanged()), this, SLOT(channelSampleRateChanged()));
//m_objPluginAPI->addThreadedSink(m_objThreadedChannelizer);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.setColor(Qt::white);
m_objChannelMarker.setMovable(false);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.setBandwidth(6000000);
m_objChannelMarker.setCenterFrequency(0);
m_objChannelMarker.setVisible(true);
setTitleColor(m_objChannelMarker.getColor());
2017-02-23 02:18:56 -05:00
connect(&m_objChannelMarker, SIGNAL(changed()), this, SLOT(viewChanged()));
2017-02-23 02:18:56 -05:00
m_objDeviceAPI->registerChannelInstance(m_strChannelID, this);
m_objDeviceAPI->addChannelMarker(&m_objChannelMarker);
m_objDeviceAPI->addRollupWidget(this);
//ui->screenTV->connectTimer(m_objPluginAPI->getMainWindow()->getMasterTimer());
2017-02-23 02:18:56 -05:00
m_objMagSqAverage.resize(4, 1.0);
ui->scopeGUI->setBuddies(m_objScopeVis->getInputMessageQueue(), m_objScopeVis, ui->glScope);
resetToDefaults(); // does applySettings()
connect(m_objATVDemod->getOutputMessageQueue(), SIGNAL(messageEnqueued()), this, SLOT(handleSourceMessages()));
2017-02-23 02:18:56 -05:00
}
ATVDemodGUI::~ATVDemodGUI()
{
m_objDeviceAPI->removeChannelInstance(this);
m_objDeviceAPI->removeThreadedSink(m_objThreadedChannelizer);
delete m_objThreadedChannelizer;
delete m_objChannelizer;
delete m_objATVDemod;
delete m_objScopeVis;
2017-02-23 02:18:56 -05:00
delete ui;
}
void ATVDemodGUI::blockApplySettings(bool blnBlock)
{
m_blnDoApplySettings = !blnBlock;
}
void ATVDemodGUI::applySettings()
{
if (m_blnDoApplySettings)
{
2017-03-15 13:56:38 -04:00
ui->deltaFrequency->setValue(abs(m_objChannelMarker.getCenterFrequency()));
ui->deltaFrequencyMinus->setChecked(m_objChannelMarker.getCenterFrequency() < 0);
2017-02-23 02:18:56 -05:00
m_objChannelizer->configure(m_objChannelizer->getInputMessageQueue(),
m_objChannelizer->getInputSampleRate(), // always use maximum available bandwidth
m_objChannelMarker.getCenterFrequency());
2017-02-23 02:18:56 -05:00
m_objATVDemod->configure(m_objATVDemod->getInputMessageQueue(),
ui->lineTime->value() / 10.0f,
ui->topTime->value() * 1.0f,
(ui->fps->currentIndex() == 0) ? 25.0f : 30.0f,
(ui->halfImage->checkState() == Qt::Checked) ? 0.5f : 1.0f,
ui->synchLevel->value() / 1000.0f,
ui->blackLevel->value() / 1000.0f,
ui->hSync->isChecked(),
2017-03-19 14:10:48 -04:00
ui->vSync->isChecked(),
ui->invertVideo->isChecked());
2017-02-23 02:18:56 -05:00
qDebug() << "ATVDemodGUI::applySettings:"
<< " m_objChannelizer.inputSampleRate: " << m_objChannelizer->getInputSampleRate()
<< " m_objATVDemod.sampleRate: " << m_objATVDemod->getSampleRate();
}
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::applyRFSettings()
{
if (m_blnDoApplySettings)
{
m_objATVDemod->configureRF(m_objATVDemod->getInputMessageQueue(),
(ATVDemod::ATVModulation) ui->modulation->currentIndex(),
ui->rfBW->value() * 100000.0f,
ui->rfOppBW->value() * 100000.0f,
ui->rfFiltering->isChecked(),
ui->decimatorEnable->isChecked(),
ui->bfo->value() * 10.0f);
}
}
void ATVDemodGUI::setChannelMarkerBandwidth()
{
m_blnDoApplySettings = false; // avoid infinite recursion
if (ui->rfFiltering->isChecked()) // FFT filter
{
m_objChannelMarker.setBandwidth(ui->rfBW->value()*100000);
m_objChannelMarker.setOppositeBandwidth(ui->rfOppBW->value()*100000);
if (ui->modulation->currentIndex() == (int) ATVDemod::ATV_LSB) {
m_objChannelMarker.setSidebands(ChannelMarker::vlsb);
} else if (ui->modulation->currentIndex() == (int) ATVDemod::ATV_USB) {
m_objChannelMarker.setSidebands(ChannelMarker::vusb);
} else {
m_objChannelMarker.setSidebands(ChannelMarker::vusb);
}
}
else
{
if (ui->decimatorEnable->isChecked()) {
m_objChannelMarker.setBandwidth(ui->rfBW->value()*100000);
} else {
m_objChannelMarker.setBandwidth(m_objChannelizer->getInputSampleRate());
}
m_objChannelMarker.setSidebands(ChannelMarker::dsb);
}
m_blnDoApplySettings = true;
}
void ATVDemodGUI::setRFFiltersSlidersRange(int sampleRate)
{
// RF filters sliders range
if (ui->rfFiltering->isChecked())
{
ui->rfBW->setMaximum(sampleRate / 200000);
ui->rfOppBW->setMaximum(sampleRate / 200000);
}
else
{
ui->rfBW->setMaximum(sampleRate / 100000);
ui->rfOppBW->setMaximum(sampleRate / 100000);
}
}
2017-02-23 02:18:56 -05:00
void ATVDemodGUI::leaveEvent(QEvent*)
{
blockApplySettings(true);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.setHighlighted(false);
blockApplySettings(false);
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::enterEvent(QEvent*)
{
blockApplySettings(true);
2017-02-23 02:18:56 -05:00
m_objChannelMarker.setHighlighted(true);
blockApplySettings(false);
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::tick()
{
if (m_intTickCount < 4) // ~200 ms
{
m_intTickCount++;
}
else
{
if (m_objATVDemod)
{
m_objMagSqAverage.feed(m_objATVDemod->getMagSq());
double magSqDB = CalcDb::dbPower(m_objMagSqAverage.average() / (1<<30));
ui->channePowerText->setText(tr("%1 dB").arg(magSqDB, 0, 'f', 1));
if (m_objATVDemod->getBFOLocked()) {
ui->bfoLockedLabel->setStyleSheet("QLabel { background-color : green; }");
} else {
ui->bfoLockedLabel->setStyleSheet("QLabel { background:rgb(79,79,79); }");
}
}
m_intTickCount = 0;
}
return;
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_synchLevel_valueChanged(int value)
2017-02-23 02:18:56 -05:00
{
applySettings();
ui->synchLevelText->setText(QString("%1 mV").arg(value));
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_blackLevel_valueChanged(int value)
2017-02-23 02:18:56 -05:00
{
applySettings();
ui->blackLevelText->setText(QString("%1 mV").arg(value));
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_lineTime_valueChanged(int value)
2017-02-23 02:18:56 -05:00
{
ui->lineTimeText->setText(QString("%1 uS").arg(((float) value) / 10.0f));
applySettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_topTime_valueChanged(int value)
2017-02-23 02:18:56 -05:00
{
ui->topTimeText->setText(QString("%1 uS").arg(value));
applySettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_hSync_clicked()
2017-02-23 02:18:56 -05:00
{
applySettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_vSync_clicked()
{
applySettings();
2017-02-23 02:18:56 -05:00
}
2017-03-19 14:10:48 -04:00
void ATVDemodGUI::on_invertVideo_clicked()
{
applySettings();
}
void ATVDemodGUI::on_halfImage_clicked()
2017-02-23 02:18:56 -05:00
{
applySettings();
2017-02-23 02:18:56 -05:00
}
void ATVDemodGUI::on_fps_currentIndexChanged(int index)
2017-02-23 02:18:56 -05:00
{
applySettings();
}
void ATVDemodGUI::on_reset_clicked(bool checked)
{
resetToDefaults();
}
void ATVDemodGUI::on_modulation_currentIndexChanged(int index)
{
setRFFiltersSlidersRange(m_objATVDemod->getEffectiveSampleRate());
setChannelMarkerBandwidth();
applyRFSettings();
}
void ATVDemodGUI::on_rfBW_valueChanged(int value)
{
ui->rfBWText->setText(QString("%1 MHz").arg(value / 10.0, 0, 'f', 1));
setChannelMarkerBandwidth();
applyRFSettings();
}
void ATVDemodGUI::on_rfOppBW_valueChanged(int value)
{
ui->rfOppBWText->setText(QString("%1").arg(value / 10.0, 0, 'f', 1));
setChannelMarkerBandwidth();
applyRFSettings();
}
void ATVDemodGUI::on_rfFiltering_toggled(bool checked)
{
setRFFiltersSlidersRange(m_objATVDemod->getEffectiveSampleRate());
setChannelMarkerBandwidth();
applyRFSettings();
}
void ATVDemodGUI::on_decimatorEnable_toggled(bool checked)
{
setChannelMarkerBandwidth();
applyRFSettings();
}
void ATVDemodGUI::on_deltaFrequency_changed(quint64 value)
{
if (ui->deltaFrequencyMinus->isChecked()) {
m_objChannelMarker.setCenterFrequency(-value);
} else {
m_objChannelMarker.setCenterFrequency(value);
}
}
void ATVDemodGUI::on_deltaFrequencyMinus_toggled(bool minus)
{
int deltaFrequency = m_objChannelMarker.getCenterFrequency();
bool minusDelta = (deltaFrequency < 0);
if (minus ^ minusDelta) // sign change
{
m_objChannelMarker.setCenterFrequency(-deltaFrequency);
}
}
void ATVDemodGUI::on_bfo_valueChanged(int value)
{
ui->bfoText->setText(QString("%1").arg(value * 10.0, 0, 'f', 0));
applyRFSettings();
}