#include "receiverwidget.h" #include "logcategories.h" #include "rigidentities.h" receiverWidget::receiverWidget(bool scope, uchar receiver, uchar vfo, QWidget *parent) : QGroupBox{parent}, receiver(receiver), numVFO(vfo) { // Not sure if this should actually be used? Q_UNUSED(scope) QMutexLocker locker(&mutex); this->setObjectName("Spectrum Scope"); this->setTitle("Band"); this->defaultStyleSheet = this->styleSheet(); queue = cachingQueue::getInstance(); rigCaps = queue->getRigCaps(); mainLayout = new QHBoxLayout(this); layout = new QVBoxLayout(); mainLayout->addLayout(layout); splitter = new QSplitter(this); layout->addWidget(splitter); splitter->setOrientation(Qt::Vertical); originalParent = parent; displayLayout = new QHBoxLayout(); vfoSelectButton=new QPushButton(tr("VFO A"),this); vfoSelectButton->setHidden(true); vfoSelectButton->setCheckable(true); vfoSelectButton->setFocusPolicy(Qt::NoFocus); connect(vfoSelectButton, &QPushButton::clicked, this, [=](bool en) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,false); queue->add(priorityImmediate,queueItem(funcSelectVFO,QVariant::fromValue(en),false,t.receiver)); queue->add(priorityHighest,t.freqFunc,false,t.receiver); queue->add(priorityHighest,t.modeFunc,false,t.receiver); t = queue->getVfoCommand(vfoB,receiver,false); queue->add(priorityHighest,t.freqFunc,false,t.receiver); queue->add(priorityHighest,t.modeFunc,false,t.receiver); if (en) vfoSelectButton->setText(tr("VFO B")); else vfoSelectButton->setText(tr("VFO A")); selectedVFO = uchar(en); }); vfoSwapButton=new QPushButton(tr("A<>B"),this); vfoSwapButton->setHidden(true); vfoSwapButton->setFocusPolicy(Qt::NoFocus); connect(vfoSwapButton, &QPushButton::clicked, this, [=]() { vfoSwap(); vfoCommandType t = queue->getVfoCommand(vfoA,receiver,false); queue->add(priorityHighest,t.modeFunc,false,t.receiver); queue->add(priorityHighest,t.freqFunc,false,t.receiver); }); vfoEqualsButton=new QPushButton(tr("A=B"),this); vfoEqualsButton->setHidden(true); vfoEqualsButton->setFocusPolicy(Qt::NoFocus); connect(vfoEqualsButton, &QPushButton::clicked, this, [=]() { vfoCommandType t = queue->getVfoCommand(vfoB,receiver,false); queue->add(priorityImmediate,funcVFOEqualAB,false,0); queue->add(priorityHighest,t.modeFunc,false,t.receiver); queue->add(priorityHighest,t.freqFunc,false,t.receiver); }); vfoMemoryButton=new QPushButton(tr("V/M"),this); vfoMemoryButton->setHidden(true); vfoMemoryButton->setCheckable(true); vfoMemoryButton->setFocusPolicy(Qt::NoFocus); connect(vfoMemoryButton, &QPushButton::clicked, this, [=](bool en) { if (en) queue->add(priorityImmediate,queueItem(funcSelectVFO,QVariant::fromValue(vfoA),false,0)); else queue->add(priorityImmediate,queueItem(funcSplitStatus,QVariant::fromValue(vfoMem),false,0)); }); satelliteButton=new QPushButton(tr("SAT"),this); satelliteButton->setHidden(true); satelliteButton->setCheckable(true); satelliteButton->setFocusPolicy(Qt::NoFocus); connect(satelliteButton, &QPushButton::clicked, this, [=](bool en) { queue->add(priorityImmediate,queueItem(funcSatelliteMode,QVariant::fromValue(en),false,0)); }); splitButton=new QPushButton(tr("SPLIT"),this); splitButton->setHidden(true); splitButton->setFocusPolicy(Qt::NoFocus); splitButton->setCheckable(true); connect(splitButton, &QPushButton::clicked, this, [=](bool en) { queue->add(priorityImmediate,queueItem(funcSplitStatus,QVariant::fromValue(en),false,0)); }); for (uchar i=0;isetMinimumSize(280,30); fr->setMaximumSize(280,30); displayLayout->addWidget(fr); // Add the VFO buttons here. if (numVFO > 1) { vfoSelectButton->setHidden(false); displayLayout->addWidget(vfoSelectButton); displayLSpacer = new QSpacerItem(0,0,QSizePolicy::Expanding,QSizePolicy::Fixed); displayLayout->addSpacerItem(displayLSpacer); if (!receiver) { if (rigCaps->commands.contains(funcVFOEqualAB)) { vfoSwapButton->setHidden(false); displayLayout->addWidget(vfoSwapButton); } if (rigCaps->commands.contains(funcVFOEqualAB)) { vfoEqualsButton->setHidden(false); displayLayout->addWidget(vfoEqualsButton); } if(rigCaps->commands.contains(funcMemoryMode)) { vfoMemoryButton->setHidden(false); displayLayout->addWidget(vfoMemoryButton); } if(rigCaps->commands.contains(funcSatelliteMode)) { satelliteButton->setHidden(false); displayLayout->addWidget(satelliteButton); } displayMSpacer = new QSpacerItem(0,0,QSizePolicy::Expanding,QSizePolicy::Fixed); displayLayout->addSpacerItem(displayMSpacer); if (rigCaps->commands.contains(funcSplitStatus)) { splitButton->setHidden(false); displayLayout->addWidget(splitButton); } } } displayRSpacer = new QSpacerItem(0,0,QSizePolicy::Expanding,QSizePolicy::Fixed); displayLayout->addSpacerItem(displayRSpacer); } else { fr->setMinimumSize(180,20); fr->setMaximumSize(180,20); if (!rigCaps->hasCommand29 && receiver == 1) { fr->setVisible(false); } displayLayout->addWidget(fr); } connect(fr, &freqCtrl::newFrequency, this, [=](const qint64 &freq) { this->newFrequency(freq,i); }); freqDisplay.append(fr); } controlLayout = new QHBoxLayout(); detachButton = new QPushButton(tr("Detach")); detachButton->setCheckable(true); detachButton->setToolTip(tr("Detach/re-attach scope from main window")); detachButton->setChecked(false); //scopeModeLabel = new QLabel("Spectrum Mode:"); scopeModeCombo = new QComboBox(); scopeModeCombo->setAccessibleDescription(tr("Spectrum Mode")); //spanLabel = new QLabel("Span:"); spanCombo = new QComboBox(); spanCombo->setAccessibleDescription(tr("Spectrum Span")); //edgeLabel = new QLabel("Edge:"); edgeCombo = new QComboBox(); edgeCombo->setAccessibleDescription(tr("Spectrum Edge")); edgeButton = new QPushButton(tr("Custom Edge")); edgeButton->setToolTip(tr("Define a custom (fixed) scope edge")); toFixedButton = new QPushButton(tr("To Fixed")); toFixedButton->setToolTip(tr("<html><head/><body><p>Press button to convert center mode spectrum to fixed mode, preserving the range. This allows you to tune without the spectrum moving, in the same currently-visible range that you see now. </p><p><br/></p><p>The currently-selected edge slot will be overridden.</p></body></html>")); holdButton = new QPushButton("HOLD"); holdButton->setCheckable(true); holdButton->setFocusPolicy(Qt::NoFocus); controlSpacer = new QSpacerItem(0,0,QSizePolicy::Expanding,QSizePolicy::Fixed); midSpacer = new QSpacerItem(0,0,QSizePolicy::Expanding,QSizePolicy::Fixed); clearPeaksButton = new QPushButton("Clear Peaks"); confButton = new QPushButton("<"); confButton->setAccessibleName(tr("Configure Scope")); confButton->setAccessibleDescription(tr("Change various settings of the current Scope")); confButton->setToolTip(tr("Configure Scope")); confButton->setFocusPolicy(Qt::NoFocus); modeCombo = new QComboBox(); modeCombo->setToolTip(tr("Select current radio mode")); modeCombo->setAccessibleDescription(tr("Change the current radio mode for this receiver")); dataCombo = new QComboBox(); dataCombo->setToolTip(tr("Select data mode (if supported)")); dataCombo->setAccessibleDescription(tr("Change the current data mode (if supported)")); filterCombo = new QComboBox(); filterCombo->setToolTip(tr("Select current filter")); dataCombo->setAccessibleDescription(tr("Change the current filter")); filterShapeCombo = new QComboBox(); filterShapeCombo->setToolTip(tr("Select current filter shape")); filterShapeCombo->setAccessibleDescription(tr("Change the current filter shape")); if (!rigCaps->commands.contains(funcFilterShape)) { filterShapeCombo->hide(); } else { filterShapeCombo->addItem("Sharp",0); if (rigCaps->manufacturer == manufKenwood) { filterShapeCombo->addItem("Medium",1); filterShapeCombo->addItem("Soft",2); } else { filterShapeCombo->addItem("Soft",1); } } roofingCombo = new QComboBox(); roofingCombo->setToolTip(tr("Select roofing filter")); roofingCombo->setAccessibleDescription(tr("Change the current selected roofing filter")); if (!rigCaps->commands.contains(funcRoofingFilter)) roofingCombo->hide(); spanCombo->setVisible(false); edgeCombo->setVisible(false); edgeButton->setVisible(false); toFixedButton->setVisible(false); spectrum = new QCustomPlot(); spectrum->xAxis->axisRect()->setAutoMargins(QCP::MarginSide::msBottom); spectrum->yAxis->axisRect()->setAutoMargins(QCP::MarginSide::msBottom); spectrum->xAxis->setPadding(0); spectrum->yAxis->setPadding(0); waterfall = new QCustomPlot(); waterfall->xAxis->axisRect()->setAutoMargins(QCP::MarginSide::msNone); waterfall->yAxis->axisRect()->setAutoMargins(QCP::MarginSide::msNone); waterfall->xAxis->setPadding(0); waterfall->yAxis->setPadding(0); splitter->addWidget(spectrum); splitter->addWidget(waterfall); splitter->setHandleWidth(5); spectrum->axisRect()->setMargins(QMargins(30,0,0,0)); waterfall->axisRect()->setMargins(QMargins(30,0,0,0)); layout->addLayout(displayLayout); layout->addLayout(controlLayout); controlLayout->addWidget(detachButton); controlLayout->addWidget(scopeModeCombo); controlLayout->addWidget(spanCombo); controlLayout->addWidget(edgeCombo); controlLayout->addWidget(edgeButton); controlLayout->addWidget(toFixedButton); controlLayout->addWidget(holdButton); controlLayout->addSpacerItem(controlSpacer); controlLayout->addWidget(modeCombo); controlLayout->addWidget(dataCombo); controlLayout->addWidget(filterCombo); controlLayout->addWidget(filterShapeCombo); controlLayout->addWidget(roofingCombo); controlLayout->addSpacerItem(midSpacer); controlLayout->addWidget(clearPeaksButton); controlLayout->addWidget(confButton); this->layout->setContentsMargins(5,5,5,5); scopeModeCombo->addItem(tr("Center Mode"), (spectrumMode_t)spectModeCenter); scopeModeCombo->addItem(tr("Fixed Mode"), (spectrumMode_t)spectModeFixed); scopeModeCombo->addItem(tr("Scroll-C"), (spectrumMode_t)spectModeScrollC); scopeModeCombo->addItem(tr("Scroll-F"), (spectrumMode_t)spectModeScrollF); scopeModeCombo->setSizeAdjustPolicy(QComboBox::AdjustToContents); //edgeCombo->insertItems(0, QStringList({tr("Fixed Edge 1"),tr("Fixed Edge 2"),tr("Fixed Edge 3"),tr("Fixed Edge 4")})); //edgeCombo->setSizeAdjustPolicy(QComboBox::AdjustToContentsOnFirstShow); auto it = rigCaps->commands.find(funcScopeEdge); if (it != rigCaps->commands.end()) { for (int i=it->minVal; i<=it->maxVal; i++) { edgeCombo->addItem(QString("Fixed Edge %0").arg(i),QVariant::fromValue(i)); } } // Spectrum Plot setup passbandIndicator = new QCPItemRect(spectrum); passbandIndicator->setAntialiased(true); passbandIndicator->setPen(QPen(Qt::red)); passbandIndicator->setBrush(QBrush(Qt::red)); passbandIndicator->setSelectable(true); pbtIndicator = new QCPItemRect(spectrum); pbtIndicator->setAntialiased(true); pbtIndicator->setPen(QPen(Qt::red)); pbtIndicator->setBrush(QBrush(Qt::red)); pbtIndicator->setSelectable(true); pbtIndicator->setVisible(false); freqIndicatorLine = new QCPItemLine(spectrum); freqIndicatorLine->setAntialiased(true); freqIndicatorLine->setPen(QPen(Qt::blue)); oorIndicator = new QCPItemText(spectrum); oorIndicator->setVisible(false); oorIndicator->setAntialiased(true); oorIndicator->setPen(QPen(Qt::red)); oorIndicator->setBrush(QBrush(Qt::red)); oorIndicator->setFont(QFont(font().family(), 14)); oorIndicator->setPositionAlignment(Qt::AlignVCenter | Qt::AlignHCenter); oorIndicator->position->setType(QCPItemPosition::ptAxisRectRatio); // Positioned relative to the current plot rect oorIndicator->setText(tr("SCOPE OUT OF RANGE")); oorIndicator->position->setCoords(0.5f,0.5f); ovfIndicator = new QCPItemText(spectrum); ovfIndicator->setVisible(false); ovfIndicator->setAntialiased(true); ovfIndicator->setPen(QPen(Qt::red)); ovfIndicator->setColor(Qt::red); ovfIndicator->setFont(QFont(font().family(), 10)); ovfIndicator->setPositionAlignment(Qt::AlignLeft | Qt::AlignTop); ovfIndicator->position->setType(QCPItemPosition::ptAxisRectRatio); // Positioned relative to the current plot rect ovfIndicator->setText(tr(" OVF ")); ovfIndicator->position->setCoords(0.01f,0.0f); redrawSpeed = new QCPItemText(spectrum); redrawSpeed->setVisible(true); redrawSpeed->setColor(Qt::gray); redrawSpeed->setFont(QFont(font().family(), 8)); redrawSpeed->setPositionAlignment(Qt::AlignRight | Qt::AlignTop); redrawSpeed->position->setType(QCPItemPosition::ptAxisRectRatio); redrawSpeed->setText(""); redrawSpeed->position->setCoords(1.0f,0.0f); spectrum->addGraph(); // primary spectrum->addGraph(0, 0); // secondary, peaks, same axis as first. spectrum->addLayer( "Top Layer", spectrum->layer("main")); spectrum->graph(0)->setLayer("Top Layer"); spectrumPlasma.resize(spectrumPlasmaSizeMax); for(unsigned int p=0; p < spectrumPlasmaSizeMax; p++) { spectrumPlasma[p] = 0; } QColor color(20+200/4.0*1,70*(1.6-1/4.0), 150, 150); spectrum->graph(1)->setLineStyle(QCPGraph::lsLine); spectrum->graph(1)->setPen(QPen(color.lighter(200))); spectrum->graph(1)->setBrush(QBrush(color)); freqIndicatorLine->start->setCoords(0.5, 0); freqIndicatorLine->end->setCoords(0.5, 160); passbandIndicator->topLeft->setCoords(0.5, 0); passbandIndicator->bottomRight->setCoords(0.5, 160); pbtIndicator->topLeft->setCoords(0.5, 0); pbtIndicator->bottomRight->setCoords(0.5, 160); // Waterfall setup waterfall->addGraph(); colorMap = new QCPColorMap(waterfall->xAxis, waterfall->yAxis); colorMapData = NULL; #if QCUSTOMPLOT_VERSION < 0x020001 this->addPlottable(colorMap); #endif // Config Screen rhsLayout = new QVBoxLayout(); rhsTopSpacer = new QSpacerItem(0,0,QSizePolicy::Fixed,QSizePolicy::Expanding); rhsLayout->addSpacerItem(rhsTopSpacer); configGroup = new QGroupBox(); rhsLayout->addWidget(configGroup); configLayout = new QFormLayout(); configLayout->setHorizontalSpacing(0); configGroup->setLayout(configLayout); mainLayout->addLayout(rhsLayout); rhsBottomSpacer = new QSpacerItem(0,0,QSizePolicy::Fixed,QSizePolicy::Expanding); rhsLayout->addSpacerItem(rhsBottomSpacer); QFont font = configGroup->font(); configGroup->setStyleSheet(QString("QGroupBox{border:1px solid gray;} *{padding: 0px 0px 0px 0px; margin: 0px 0px 0px 0px; font-size: %0px;}").arg(font.pointSize()-1)); configGroup->setMaximumWidth(240); configRef = new QSlider(Qt::Orientation::Horizontal); configRef->setTickInterval(50); configRef->setSingleStep(20); configRef->setValue(0); configRef->setAccessibleName(tr("Scope display reference")); configRef->setAccessibleDescription(tr("Selects the display reference for the Scope display")); configRef->setToolTip(tr("Select display reference of scope")); configLayout->addRow(tr("Ref"),configRef); configLength = new QSlider(Qt::Orientation::Horizontal); configLength->setRange(100,1024); configLayout->addRow(tr("Length"),configLength); configTop = new QSlider(Qt::Orientation::Horizontal); configTop->setRange(0,rigCaps->spectAmpMax+10); configTop->setValue(plotCeiling); configTop->setAccessibleName(tr("Scope display ceiling")); configTop->setAccessibleDescription(tr("Selects the display ceiling for the Scope display")); configTop->setToolTip(tr("Select display ceiling of scope")); configLayout->addRow(tr("Ceiling"),configTop); configBottom = new QSlider(Qt::Orientation::Horizontal); configBottom->setRange(0,rigCaps->spectAmpMax+10); configBottom->setValue(plotFloor); configBottom->setAccessibleName(tr("Scope display floor")); configBottom->setAccessibleDescription(tr("Selects the display floor for the Scope display")); configBottom->setToolTip(tr("Select display floor of scope")); configLayout->addRow(tr("Floor"),configBottom); configSpeed = new QComboBox(); configSpeed->addItem(tr("Speed Fast"),QVariant::fromValue(uchar(0))); configSpeed->addItem(tr("Speed Mid"),QVariant::fromValue(uchar(1))); configSpeed->addItem(tr("Speed Slow"),QVariant::fromValue(uchar(2))); configSpeed->setCurrentIndex(configSpeed->findData(currentSpeed)); configSpeed->setAccessibleName(tr("Waterfall display speed")); configSpeed->setAccessibleDescription(tr("Selects the speed for the waterfall display")); configSpeed->setToolTip(tr("Waterfall Speed")); configLayout->addRow(tr("Speed"),configSpeed); configTheme = new QComboBox(); configTheme->setAccessibleName(tr("Waterfall display color theme")); configTheme->setAccessibleDescription(tr("Selects the color theme for the waterfall display")); configTheme->setToolTip(tr("Waterfall color theme")); configTheme->addItem("Jet", QCPColorGradient::gpJet); configTheme->addItem("Cold", QCPColorGradient::gpCold); configTheme->addItem("Hot", QCPColorGradient::gpHot); configTheme->addItem("Therm", QCPColorGradient::gpThermal); configTheme->addItem("Night", QCPColorGradient::gpNight); configTheme->addItem("Ion", QCPColorGradient::gpIon); configTheme->addItem("Gray", QCPColorGradient::gpGrayscale); configTheme->addItem("Geo", QCPColorGradient::gpGeography); configTheme->addItem("Hues", QCPColorGradient::gpHues); configTheme->addItem("Polar", QCPColorGradient::gpPolar); configTheme->addItem("Spect", QCPColorGradient::gpSpectrum); configTheme->addItem("Candy", QCPColorGradient::gpCandy); configTheme->setSizeAdjustPolicy(QComboBox::AdjustToContents); configLayout->addRow(tr("Theme"),configTheme); configPbtInner = new QSlider(Qt::Orientation::Horizontal); configPbtInner->setRange(0,255); configLayout->addRow(tr("PBT Inner"),configPbtInner); configPbtOuter = new QSlider(Qt::Orientation::Horizontal); configPbtOuter->setRange(0,255); configLayout->addRow(tr("PBT Outer"),configPbtOuter); configIfLayout = new QHBoxLayout(); configIfShift = new QSlider(Qt::Orientation::Horizontal); configIfShift->setRange(0,255); if (rigCaps != Q_NULLPTR && !rigCaps->commands.contains(funcIFShift)){ configIfShift->setEnabled(true); configIfShift->setValue(128); } else { configIfShift->setEnabled(false); } configResetIf = new QPushButton("R"); configIfLayout->addWidget(configIfShift); configIfLayout->addWidget(configResetIf); configLayout->addRow(tr("IF Shift"),configIfLayout); configFilterWidth = new QSlider(Qt::Orientation::Horizontal); configFilterWidth->setRange(0,10000); configLayout->addRow(tr("Fill Width"),configFilterWidth); configScopeEnabled = new QCheckBox(tr("Scope Enabled")); configScopeEnabled->setChecked(true); configLayout->addRow(configScopeEnabled); frequencyNotificationLockoutTimer = new QTimer(); connect(frequencyNotificationLockoutTimer, SIGNAL(timeout()), this, SLOT(freqNoteLockTimerSlot())); frequencyNotificationLockoutTimer->setInterval(200); frequencyNotificationLockoutTimer->setSingleShot(true); connect(configLength, &QSlider::valueChanged, this, [=](const int &val) { //prepareWf(val); changeWfLength(val); emit updateSettings(receiver,currentTheme,wfLength,plotFloor,plotCeiling); }); connect(configBottom, &QSlider::valueChanged, this, [=](const int &val) { this->plotFloor = val; this->wfFloor = val; this->setRange(plotFloor,plotCeiling); emit updateSettings(receiver,currentTheme,wfLength,plotFloor,plotCeiling); }); connect(configTop, &QSlider::valueChanged, this, [=](const int &val) { this->plotCeiling = val; this->wfCeiling = val; this->setRange(plotFloor,plotCeiling); emit updateSettings(receiver,currentTheme,wfLength,plotFloor,plotCeiling); }); if (rigCaps->commands.contains(funcScopeRef)) { auto v = rigCaps->commands.find(funcScopeRef); configRef->setRange(v.value().minVal,v.value().maxVal); connect(configRef, &QSlider::valueChanged, this, [=](const int &val) { currentRef = (val/5) * 5; // rounded to "nearest 5" vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcScopeRef,QVariant::fromValue(currentRef),false,t.receiver)); }); } connect(configSpeed, QOverload::of(&QComboBox::currentIndexChanged), this, [=](const int &val) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcScopeSpeed,QVariant::fromValue(configSpeed->itemData(val)),false,t.receiver)); }); connect(configTheme, QOverload::of(&QComboBox::currentIndexChanged), this, [=](const int &val) { Q_UNUSED(val) currentTheme = configTheme->currentData().value(); colorMap->setGradient(currentTheme); emit updateSettings(receiver,currentTheme,wfLength,plotFloor,plotCeiling); }); if (rigCaps->commands.contains(funcPBTInner)) { auto v = rigCaps->commands.find(funcPBTInner); configPbtInner->setRange(v.value().minVal,v.value().maxVal); connect(configPbtInner, &QSlider::valueChanged, this, [=](const int &val) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTInner,QVariant::fromValue(val),false,t.receiver)); }); } if (rigCaps->commands.contains(funcPBTOuter)) { auto v = rigCaps->commands.find(funcPBTOuter); configPbtOuter->setRange(v.value().minVal,v.value().maxVal); connect(configPbtOuter, &QSlider::valueChanged, this, [=](const int &val) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTOuter,QVariant::fromValue(val),false,t.receiver)); }); } connect(configIfShift, &QSlider::valueChanged, this, [=](const int &val) { if (rigCaps != Q_NULLPTR && rigCaps->commands.contains(funcIFShift)) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcIFShift,QVariant::fromValue(val),false,t.receiver)); } else { static int previousIFShift=128; // Default value unsigned char inner = configPbtInner->value(); unsigned char outer = configPbtOuter->value(); int shift = val - previousIFShift; inner = qMax( 0, qMin(255,int (inner + shift)) ); outer = qMax( 0, qMin(255,int (outer + shift)) ); configPbtInner->setValue(inner); configPbtOuter->setValue(outer); previousIFShift = val; } }); connect(configResetIf, &QPushButton::clicked, this, [=](const bool &val) { Q_UNUSED(val) double pbFreq = (pbtDefault / passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTInner,QVariant::fromValue(newFreq),false,t.receiver)); queue->addUnique(priorityImmediate,queueItem(funcPBTOuter,QVariant::fromValue(newFreq),false,t.receiver)); configIfShift->blockSignals(true); configIfShift->setValue(128); configIfShift->blockSignals(false); }); if (rigCaps->commands.contains(funcFilterWidth)) { connect(configFilterWidth, &QSlider::valueChanged, this, [=](const int &val) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcFilterWidth,QVariant::fromValue(val),false,t.receiver)); }); } #if (QT_VERSION < QT_VERSION_CHECK(6,7,0)) connect(configScopeEnabled, &QCheckBox::stateChanged, this, [=](const int &val) { #else connect(configScopeEnabled, &QCheckBox::checkStateChanged, this, [=](const Qt::CheckState &val) { #endif vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityHighest,queueItem(funcScopeOnOff,QVariant::fromValue(val),false,t.receiver)); qInfo() << "Queueing command to set scope to:" << bool(val); }); configGroup->setVisible(false); // Connections connect(detachButton,SIGNAL(toggled(bool)), this, SLOT(detachScope(bool))); connect(scopeModeCombo, QOverload::of(&QComboBox::currentIndexChanged), this, [=](int val){ spectrumMode_t s = scopeModeCombo->itemData(val).value(); vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcScopeMode,QVariant::fromValue(s),false,t.receiver)); showHideControls(s); currentScopeMode = s; }); connect(spanCombo, QOverload::of(&QComboBox::currentIndexChanged), this, [=](int val){ vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcScopeSpan,spanCombo->itemData(val),false,t.receiver)); }); connect(confButton,SIGNAL(clicked()), this, SLOT(configPressed()),Qt::QueuedConnection); connect(toFixedButton,SIGNAL(clicked()), this, SLOT(toFixedPressed())); connect(edgeCombo, QOverload::of(&QComboBox::currentIndexChanged), this, [=](int val){ vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcScopeEdge,edgeCombo->itemData(val),false,t.receiver)); }); connect(edgeButton,SIGNAL(clicked()), this, SLOT(customSpanPressed())); connect(holdButton, &QPushButton::toggled, this, [=](const bool &val) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->add(priorityImmediate,queueItem(funcScopeHold,QVariant::fromValue(val),false,t.receiver)); }); connect(modeCombo,SIGNAL(currentIndexChanged(int)),this,SLOT(updatedMode(int))); connect(filterCombo,SIGNAL(currentIndexChanged(int)),this,SLOT(updatedMode(int))); connect(filterShapeCombo, QOverload::of(&QComboBox::currentIndexChanged), this, [=](int val){ vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); uchar f = uchar(filterShapeCombo->itemData(val).toInt() + (filterCombo->currentData().toInt() * 10)); queue->addUnique(priorityImmediate,queueItem(funcFilterShape,QVariant::fromValue(f),false,t.receiver)); }); connect(roofingCombo, QOverload::of(&QComboBox::currentIndexChanged), this, [=](int val){ vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); uchar f = uchar(roofingCombo->itemData(val).toInt() + (filterCombo->currentData().toInt() * 10)); queue->addUnique(priorityImmediate,queueItem(funcRoofingFilter,QVariant::fromValue(f),false,t.receiver)); }); connect(dataCombo,SIGNAL(currentIndexChanged(int)),this,SLOT(updatedMode(int))); connect(clearPeaksButton,SIGNAL(clicked()), this, SLOT(clearPeaks())); connect(spectrum, SIGNAL(mouseDoubleClick(QMouseEvent*)), this, SLOT(doubleClick(QMouseEvent*))); connect(waterfall, SIGNAL(mouseDoubleClick(QMouseEvent*)), this, SLOT(doubleClick(QMouseEvent*))); connect(spectrum, SIGNAL(mousePress(QMouseEvent*)), this, SLOT(scopeClick(QMouseEvent*))); connect(waterfall, SIGNAL(mousePress(QMouseEvent*)), this, SLOT(waterfallClick(QMouseEvent*))); connect(spectrum, SIGNAL(mouseRelease(QMouseEvent*)), this, SLOT(scopeMouseRelease(QMouseEvent*))); connect(spectrum, SIGNAL(mouseMove(QMouseEvent*)), this, SLOT(scopeMouseMove(QMouseEvent *))); connect(waterfall, SIGNAL(mouseWheel(QWheelEvent*)), this, SLOT(scroll(QWheelEvent*))); connect(spectrum, SIGNAL(mouseWheel(QWheelEvent*)), this, SLOT(scroll(QWheelEvent*))); showHideControls(spectrumMode_t::spectModeCenter); lastData.start(); } receiverWidget::~receiverWidget(){ QMutableVectorIterator it(bandIndicators); while (it.hasNext()) { auto band = it.next(); spectrum->removeItem(band.line); spectrum->removeItem(band.text); it.remove(); } QMutableMapIterator sp(clusterSpots); while (sp.hasNext()) { auto spot = sp.next(); spectrum->removeItem(spot.value()->text); delete spot.value(); sp.remove(); } if(colorMapData != Q_NULLPTR) { delete colorMapData; } } void receiverWidget::setSeparators(QChar gsep, QChar dsep) { for (const auto &disp : freqDisplay) { qDebug(logRig()) << "Configuring separators:" << gsep << "and" << dsep; disp->setSeparators(gsep,dsep); } } void receiverWidget::prepareScope(uint maxAmp, uint spectWidth) { this->spectWidth = spectWidth; this->maxAmp = maxAmp; } void receiverWidget::changeWfLength(uint wf) { if(!scopePrepared) return; QMutexLocker locker(&mutex); this->wfLength = wf; colorMap->data()->clear(); colorMap->data()->setValueRange(QCPRange(0, wfLength-1)); colorMap->data()->setKeyRange(QCPRange(0, spectWidth-1)); colorMap->setDataRange(QCPRange(plotFloor, plotCeiling)); colorMap->setGradient(static_cast(currentTheme)); if(colorMapData != Q_NULLPTR) { delete colorMapData; } colorMapData = new QCPColorMapData(spectWidth, wfLength, QCPRange(0, spectWidth-1), QCPRange(0, wfLength-1)); colorMap->setData(colorMapData,true); // Copy the colorMap so deleting it won't result in crash! waterfall->yAxis->setRange(0,wfLength - 1); //waterfall->replot(); } bool receiverWidget::prepareWf(uint wf) { QMutexLocker locker(&mutex); bool ret=true; this->wfLength = wf; configLength->blockSignals(true); configLength->setValue(this->wfLength); configLength->blockSignals(false); this->wfLengthMax = 1024; // Initialize before use! QByteArray empty((int)spectWidth, '\x01'); spectrumPeaks = QByteArray( (int)spectWidth, '\x01' ); //unsigned int oldSize = wfimage.size(); if (!wfimage.empty() && wfimage.first().size() != spectWidth) { // Width of waterfall has changed! wfimage.clear(); } for(unsigned int i = wfimage.size(); i < wfLengthMax; i++) { wfimage.append(empty); } //wfimage.remove(wfLength, wfimage.size()-wfLength); wfimage.squeeze(); colorMap->data()->clear(); colorMap->data()->setValueRange(QCPRange(0, wfLength-1)); colorMap->data()->setKeyRange(QCPRange(0, spectWidth-1)); colorMap->setDataRange(QCPRange(plotFloor, plotCeiling)); colorMap->setGradient(static_cast(currentTheme)); if(colorMapData != Q_NULLPTR) { delete colorMapData; } colorMapData = new QCPColorMapData(spectWidth, wfLength, QCPRange(0, spectWidth-1), QCPRange(0, wfLength-1)); colorMap->setData(colorMapData,true); // Copy the colorMap so deleting it won't result in crash! waterfall->yAxis->setRangeReversed(true); waterfall->xAxis->setVisible(false); clearPeaks(); clearPlasma(); scopePrepared = true; return ret; } void receiverWidget::setRange(int floor, int ceiling) { plotFloor = floor; plotCeiling = ceiling; wfFloor = floor; wfCeiling = ceiling; maxAmp = ceiling; if (spectrum != Q_NULLPTR) spectrum->yAxis->setRange(QCPRange(floor, ceiling)); if (colorMap != Q_NULLPTR) colorMap->setDataRange(QCPRange(floor,ceiling)); configBottom->blockSignals(true); configBottom->setValue(floor); configBottom->blockSignals(false); configTop->blockSignals(true); configTop->setValue(ceiling); configTop->blockSignals(false); // Redraw band lines and eventually memory markers! for (auto &b: bandIndicators) { b.line->start->setCoords(b.line->start->coords().x(), spectrum->yAxis->range().upper-5); b.line->end->setCoords(b.line->end->coords().x(), spectrum->yAxis->range().upper-5); b.text->position->setCoords(b.text->position->coords().x(), spectrum->yAxis->range().upper-10); } } void receiverWidget::colorPreset(colorPrefsType *cp) { if (cp == Q_NULLPTR) { return; } colors = *cp; spectrum->setBackground(cp->plotBackground); spectrum->xAxis->grid()->setPen(cp->gridColor); spectrum->yAxis->grid()->setPen(cp->gridColor); spectrum->legend->setTextColor(cp->textColor); spectrum->legend->setBorderPen(cp->gridColor); spectrum->legend->setBrush(cp->gridColor); spectrum->xAxis->setTickLabelColor(cp->textColor); spectrum->xAxis->setLabelColor(cp->gridColor); spectrum->yAxis->setTickLabelColor(cp->textColor); spectrum->yAxis->setLabelColor(cp->gridColor); spectrum->xAxis->setBasePen(cp->axisColor); spectrum->xAxis->setTickPen(cp->axisColor); spectrum->yAxis->setBasePen(cp->axisColor); spectrum->yAxis->setTickPen(cp->axisColor); freqIndicatorLine->setPen(QPen(cp->tuningLine)); passbandIndicator->setPen(QPen(cp->passband)); passbandIndicator->setBrush(QBrush(cp->passband)); pbtIndicator->setPen(QPen(cp->pbt)); pbtIndicator->setBrush(QBrush(cp->pbt)); spectrum->graph(0)->setPen(QPen(cp->spectrumLine)); if(cp->useSpectrumFillGradient) { spectrumGradient.setStart(QPointF(0,1)); spectrumGradient.setFinalStop(QPointF(0,0)); spectrumGradient.setCoordinateMode(QLinearGradient::StretchToDeviceMode); //spectrumGradient.setColorAt(0, cp->spectrumFillBot); spectrumGradient.setColorAt(0.1, cp->spectrumFillBot); spectrumGradient.setColorAt(1, cp->spectrumFillTop); spectrum->graph(0)->setBrush(QBrush(spectrumGradient)); } else { spectrum->graph(0)->setBrush(QBrush(cp->spectrumFill)); } spectrum->graph(1)->setPen(QPen(cp->underlayLine)); if(cp->useUnderlayFillGradient) { underlayGradient.setStart(QPointF(0,1)); underlayGradient.setFinalStop(QPointF(0,0)); underlayGradient.setCoordinateMode(QLinearGradient::StretchToDeviceMode); //underlayGradient.setColorAt(0, cp->underlayFillBot); underlayGradient.setColorAt(0.1, cp->underlayFillBot); underlayGradient.setColorAt(1, cp->underlayFillTop); spectrum->graph(1)->setBrush(QBrush(underlayGradient)); } else { spectrum->graph(1)->setBrush(QBrush(cp->underlayFill)); } waterfall->yAxis->setBasePen(cp->wfAxis); waterfall->yAxis->setTickPen(cp->wfAxis); waterfall->xAxis->setBasePen(cp->wfAxis); waterfall->xAxis->setTickPen(cp->wfAxis); waterfall->xAxis->setLabelColor(cp->wfGrid); waterfall->yAxis->setLabelColor(cp->wfGrid); waterfall->xAxis->setTickLabelColor(cp->wfText); waterfall->yAxis->setTickLabelColor(cp->wfText); waterfall->setBackground(cp->wfBackground); holdButton->setStyleSheet(QString("QPushButton {background-color: %0;} QPushButton:checked {background-color: %1;border:1px solid;}") .arg(cp->buttonOff.name(QColor::HexArgb),cp->buttonOn.name(QColor::HexArgb))); splitButton->setStyleSheet(QString("QPushButton {background-color: %0;} QPushButton:checked {background-color: %1;border:1px solid;}") .arg(cp->buttonOff.name(QColor::HexArgb),cp->buttonOn.name(QColor::HexArgb))); satelliteButton->setStyleSheet(QString("QPushButton {background-color: %0;} QPushButton:checked {background-color: %1;border:1px solid;}") .arg(cp->buttonOff.name(QColor::HexArgb),cp->buttonOn.name(QColor::HexArgb))); vfoSelectButton->setStyleSheet(QString("QPushButton {background-color: %0;} QPushButton:checked {background-color: %1;border:1px solid;}") .arg(cp->buttonOff.name(QColor::HexArgb),cp->buttonOn.name(QColor::HexArgb))); } bool receiverWidget::updateScope(scopeData data) { if (!scopePrepared ) { return false; } if (!lastData.isValid()) { lastData.start(); } if (!bandIndicatorsVisible) { showBandIndicators(true); } //qint64 spectime = 0; bool updateRange = false; if (data.startFreq != lowerFreq || data.endFreq != upperFreq) { if(underlayMode == underlayPeakHold) { // TODO: create non-button function to do this // This will break if the button is ever moved or renamed. clearPeaks(); } // Inform other threads (cluster) that the frequency range has changed. emit frequencyRange(receiver, data.startFreq, data.endFreq); } lowerFreq = data.startFreq; upperFreq = data.endFreq; //qInfo(logSystem()) << "start: " << data.startFreq << " end: " << data.endFreq; quint16 specLen = data.data.length(); if (specLen != spectWidth) { qWarning(logSystem()) << "Spectrum length error, expected" << spectWidth << "got" << specLen << "(one can be ignored for USB connection)"; return false; } QVector x(spectWidth), y(spectWidth), y2(spectWidth); for(int i=0; i < spectWidth; i++) { x[i] = (i * (data.endFreq-data.startFreq)/spectWidth) + data.startFreq; } for(int i=0; i< specLen; i++) { y[i] = (quint8)data.data[i]; if(underlayMode == underlayPeakHold) { if((quint8)data.data[i] > (quint8)spectrumPeaks[i]) { spectrumPeaks[i] = data.data[i]; } y2[i] = (quint8)spectrumPeaks[i]; } } plasmaMutex.lock(); spectrumPlasma[spectrumPlasmaPosition] = data.data; spectrumPlasmaPosition = (spectrumPlasmaPosition+1) % spectrumPlasmaSizeCurrent; //spectrumPlasma.push_front(data.data); // if(spectrumPlasma.size() > (int)spectrumPlasmaSizeCurrent) // { // // If we have pushed_front more than spectrumPlasmaSize, // // then we cut one off the back. // spectrumPlasma.pop_back(); // } plasmaMutex.unlock(); mutex.lock(); if ((plotFloor != oldPlotFloor) || (plotCeiling != oldPlotCeiling)){ updateRange = true; } #if QCUSTOMPLOT_VERSION < 0x020000 spectrum->graph(0)->setData(x, y); #else spectrum->graph(0)->setData(x, y, true); #endif if((freq.MHzDouble < data.endFreq) && (freq.MHzDouble > data.startFreq)) { freqIndicatorLine->start->setCoords(freq.MHzDouble, 0); freqIndicatorLine->end->setCoords(freq.MHzDouble, maxAmp); double pbStart = 0.0; double pbEnd = 0.0; switch (mode.mk) { case modeLSB: case modeRTTY_R: case modePSK_R: if (rigCaps->manufacturer == manufKenwood) { pbStart = freq.MHzDouble-passbandWidth; pbEnd = freq.MHzDouble; } else { pbStart = freq.MHzDouble - passbandCenterFrequency - (passbandWidth / 2); pbEnd = freq.MHzDouble - passbandCenterFrequency + (passbandWidth / 2); } break; case modeCW: if (passbandWidth < 0.0006) { pbStart = freq.MHzDouble - (passbandWidth / 2); pbEnd = freq.MHzDouble + (passbandWidth / 2); } else { pbStart = freq.MHzDouble + passbandCenterFrequency - passbandWidth; pbEnd = freq.MHzDouble + passbandCenterFrequency; } break; case modeCW_R: if (passbandWidth < 0.0006) { pbStart = freq.MHzDouble - (passbandWidth / 2); pbEnd = freq.MHzDouble + (passbandWidth / 2); } else { pbStart = freq.MHzDouble - passbandCenterFrequency; pbEnd = freq.MHzDouble + passbandWidth - passbandCenterFrequency; } break; default: if (rigCaps->manufacturer == manufKenwood) { pbStart = freq.MHzDouble; pbEnd = freq.MHzDouble+passbandWidth; } else { pbStart = freq.MHzDouble + passbandCenterFrequency - (passbandWidth / 2); pbEnd = freq.MHzDouble + passbandCenterFrequency + (passbandWidth / 2); } break; } passbandIndicator->topLeft->setCoords(pbStart, 0); passbandIndicator->bottomRight->setCoords(pbEnd, maxAmp); if ((mode.mk == modeCW || mode.mk == modeCW_R) && passbandWidth > 0.0006) { pbtDefault = round((passbandWidth - (cwPitch / 1000000.0)) * 200000.0) / 200000.0; } else { pbtDefault = 0.0; } if ((PBTInner - pbtDefault || PBTOuter - pbtDefault) && passbandAction != passbandResizing && mode.mk != modeFM && mode.mk != modeWFM) { pbtIndicator->setVisible(true); } else { pbtIndicator->setVisible(false); } /* pbtIndicator displays the intersection between PBTInner and PBTOuter */ if (mode.mk == modeLSB || mode.mk == modeCW || mode.mk == modeRTTY) { pbtIndicator->topLeft->setCoords(qMax(pbStart - (PBTInner / 2) + (pbtDefault / 2), pbStart - (PBTOuter / 2) + (pbtDefault / 2)), 0); pbtIndicator->bottomRight->setCoords(qMin(pbStart - (PBTInner / 2) + (pbtDefault / 2) + passbandWidth, pbStart - (PBTOuter / 2) + (pbtDefault / 2) + passbandWidth), maxAmp); } else { pbtIndicator->topLeft->setCoords(qMax(pbStart + (PBTInner / 2) - (pbtDefault / 2), pbStart + (PBTOuter / 2) - (pbtDefault / 2)), 0); pbtIndicator->bottomRight->setCoords(qMin(pbStart + (PBTInner / 2) - (pbtDefault / 2) + passbandWidth, pbStart + (PBTOuter / 2) - (pbtDefault / 2) + passbandWidth), maxAmp); } //qDebug() << "Default" << pbtDefault << "Inner" << PBTInner << "Outer" << PBTOuter << "Pass" << passbandWidth << "Center" << passbandCenterFrequency << "CW" << cwPitch; } #if QCUSTOMPLOT_VERSION < 0x020000 if (underlayMode == underlayPeakHold) { spectrum->graph(1)->setData(x, y2); // peaks } else if (underlayMode != underlayNone) { computePlasma(); spectrum->graph(1)->setData(x, spectrumPlasmaLine); } else { spectrum->graph(1)->setData(x, y2); // peaks, but probably cleared out } #else if (underlayMode == underlayPeakHold) { spectrum->graph(1)->setData(x, y2, true); // peaks } else if (underlayMode != underlayNone) { computePlasma(); spectrum->graph(1)->setData(x, spectrumPlasmaLine, true); } else { spectrum->graph(1)->setData(x, y2, true); // peaks, but probably cleared out } #endif if(updateRange) spectrum->yAxis->setRange(plotFloor, plotCeiling); spectrum->xAxis->setRange(data.startFreq, data.endFreq); if(specLen == spectWidth) { wfimage.prepend(data.data); wfimage.pop_back(); QByteArray wfRow; // Waterfall: for(int row = 0; row < wfLength; row++) { wfRow = wfimage[row]; for(int col = 0; col < spectWidth; col++) { colorMap->data()->setCell( col, row, (quint8)wfRow[col]); } } if(updateRange) { colorMap->setDataRange(QCPRange(wfFloor, wfCeiling)); } waterfall->yAxis->setRange(0,wfLength - 1); waterfall->xAxis->setRange(0, spectWidth-1); } oldPlotFloor = plotFloor; oldPlotCeiling = plotCeiling; if (data.oor && !oorIndicator->visible()) { oorIndicator->setVisible(true); qInfo(logSystem()) << "Scope out of range"; } else if (!data.oor && oorIndicator->visible()) { oorIndicator->setVisible(false); } #if QCUSTOMPLOT_VERSION >= 0x020100 /* * Hopefully temporary workaround for large scopes taking longer to replot than the time * between scope data packets arriving from the radio, which were causing the UI to freeze * * Long term we should rewrite the scope/waterfall code to either use custom widgets, or * convert to Qwt (or other suitable package) with better performance than QCP. * */ if (lastData.elapsed() > (spectrum->replotTime(false)+waterfall->replotTime(false))) { spectrum->replot(); waterfall->replot(); lastData.restart(); redrawSpeed->setText(" "); } else { redrawSpeed->setText("*"); } emit spectrumTime(spectrum->replotTime(false)); emit waterfallTime(waterfall->replotTime(false)); #else spectrum->replot(); waterfall->replot(); lastData.restart(); #endif mutex.unlock(); emit sendScopeImage(receiver); return true; } // Plasma functions void receiverWidget::resizePlasmaBuffer(int size) { // QMutexLocker locker(&plasmaMutex); qDebug() << "Resizing plasma buffer via parameter, from oldsize " << spectrumPlasmaSizeCurrent << " to new size: " << size; spectrumPlasmaSizeCurrent = size; return; } void receiverWidget::clearPeaks() { // Clear the spectrum peaks as well as the plasma buffer spectrumPeaks = QByteArray( (int)spectWidth, '\x01' ); //clearPlasma(); } void receiverWidget::clearPlasma() { // Clear the buffer of spectrum used for peak and average computation. // This is only needed one time, when the VFO is created with spectrum size info. QMutexLocker locker(&plasmaMutex); QByteArray empty((int)spectWidth, '\x01'); int pSize = spectrumPlasma.size(); for(int i=0; i < pSize; i++) { spectrumPlasma[i] = empty; } } void receiverWidget::computePlasma() { QMutexLocker locker(&plasmaMutex); // Spec PlasmaLine is a single line of spectrum, ~~600 pixels or however many the radio provides. // This changes width only when we connect to a new radio. if(spectrumPlasmaLine.size() != spectWidth) { spectrumPlasmaLine.clear(); spectrumPlasmaLine.resize(spectWidth); } // spectrumPlasma is the bufffer of spectrum lines to use when computing the average or peak. int specPlasmaSize = spectrumPlasmaSizeCurrent; // go only this far in if(underlayMode == underlayAverageBuffer) { for(int col=0; col < spectWidth; col++) { for(int pos=0; pos < specPlasmaSize; pos++) { spectrumPlasmaLine[col] += (quint8)spectrumPlasma[pos][col]; } spectrumPlasmaLine[col] = spectrumPlasmaLine[col] / specPlasmaSize; } } else if (underlayMode == underlayPeakBuffer){ // peak mode, running peak display for(int col=0; col < spectWidth; col++) { spectrumPlasmaLine[col] = spectrumPlasma[0][col]; // initial value for(int pos=0; pos < specPlasmaSize; pos++) { if((double)((quint8)spectrumPlasma[pos][col]) > spectrumPlasmaLine[col]) spectrumPlasmaLine[col] = (quint8)spectrumPlasma[pos][col]; } } } } void receiverWidget::showHideControls(spectrumMode_t mode) { if(currentScopeMode == mode) { return; } if (!rigCaps->hasSpectrum) { spectrum->hide(); waterfall->hide(); splitter->hide(); scopeModeCombo->hide(); edgeCombo->hide(); edgeButton->hide(); holdButton->hide(); toFixedButton->hide(); spanCombo->hide(); clearPeaksButton->hide(); } else { spectrum->show(); waterfall->show(); splitter->show(); scopeModeCombo->show(); clearPeaksButton->show(); switch (mode) { case spectModeCenter: case spectModeScrollC: edgeCombo->hide(); edgeButton->hide(); toFixedButton->show(); spanCombo->show(); break; case spectModeFixed: case spectModeScrollF: toFixedButton->hide(); spanCombo->hide(); edgeCombo->show(); edgeButton->show(); break; case spectModeUnknown: break; } } detachButton->show(); if (rigCaps->hasSpectrum || rigCaps->commands.contains(funcIFShift) || rigCaps->commands.contains(funcPBTInner)) { confButton->show(); } else { confButton->hide(); } configSpeed->setEnabled(rigCaps->hasSpectrum); configBottom->setEnabled(rigCaps->hasSpectrum); configLength->setEnabled(rigCaps->hasSpectrum); configRef->setEnabled(rigCaps->hasSpectrum); configScopeEnabled->setEnabled(rigCaps->hasSpectrum); configTheme->setEnabled(rigCaps->hasSpectrum); configPbtInner->setEnabled(rigCaps->commands.contains(funcPBTInner)); configPbtOuter->setEnabled(rigCaps->commands.contains(funcPBTOuter)); configIfShift->setEnabled(rigCaps->commands.contains(funcIFShift) || rigCaps->commands.contains(funcPBTInner)); filterCombo->setVisible(rigCaps->filters.size()); dataCombo->setVisible(rigCaps->inputs.size()); } void receiverWidget::displayScope(bool en) { this->splitter->setVisible(en || rigCaps->hasCommand29); // Hide these controls if disabled if (!en) { this->edgeCombo->setVisible(en); this->edgeButton->setVisible(en); this->toFixedButton->setVisible(en); this->spanCombo->setVisible(en); QTimer::singleShot(0, [this]{ this->resize(this->minimumSizeHint()); }); } this->clearPeaksButton->setVisible(en && rigCaps->hasSpectrum); this->holdButton->setVisible(en && rigCaps->commands.contains(funcScopeHold)); } void receiverWidget::setScopeMode(spectrumMode_t m) { if (m != currentScopeMode) { scopeModeCombo->blockSignals(true); scopeModeCombo->setCurrentIndex(scopeModeCombo->findData(m)); scopeModeCombo->blockSignals(false); showHideControls(m); currentScopeMode = m; } } void receiverWidget::setSpan(centerSpanData s) { spanCombo->blockSignals(true); spanCombo->setCurrentIndex(spanCombo->findText(s.name)); spanCombo->blockSignals(false); } void receiverWidget::updatedMode(int index) { Q_UNUSED(index) // We don't know where it came from! modeInfo mi = modeCombo->currentData().value(); mi.filter = filterCombo->currentData().toInt(); if (mi.mk == modeCW || mi.mk == modeCW_R || mi.mk == modeRTTY || mi.mk == modeRTTY_R || mi.mk == modePSK || mi.mk == modePSK_R) { mi.data = 0; dataCombo->setEnabled(false); } else { mi.data = dataCombo->currentIndex(); dataCombo->setEnabled(true); } vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(t.modeFunc,QVariant::fromValue(mi),false,t.receiver)); if (t.modeFunc == funcModeSet) { queue->addUnique(priorityImmediate,queueItem(funcDataModeWithFilter,QVariant::fromValue(mi),false,t.receiver)); } // Request current filtershape/roofing if (rigCaps->manufacturer == manufIcom) { queue->addUnique(priorityHighest,funcFilterShape,false,t.receiver); queue->addUnique(priorityHighest,funcRoofingFilter,false,t.receiver); } } void receiverWidget::setEdge(uchar index) { edgeCombo->blockSignals(true); edgeCombo->setCurrentIndex(edgeCombo->findData(index)); edgeCombo->blockSignals(false); } void receiverWidget::setRoofing(uchar index) { roofingCombo->blockSignals(true); roofingCombo->setCurrentIndex(roofingCombo->findData(index)); roofingCombo->blockSignals(false); } void receiverWidget::setFilterShape(uchar index) { filterShapeCombo->blockSignals(true); filterShapeCombo->setCurrentIndex(filterShapeCombo->findData(index)); filterShapeCombo->blockSignals(false); } void receiverWidget::toFixedPressed() { int currentEdge = edgeCombo->currentIndex(); bool dialogOk = false; bool numOk = false; QStringList edges; edges << "1" << "2" << "3" << "4"; QString item = QInputDialog::getItem(this, "Select Edge", "Edge to replace:", edges, currentEdge, false, &dialogOk); if(dialogOk) { int edge = QString(item).toInt(&numOk,10); if(numOk) { edgeCombo->blockSignals(true); edgeCombo->setCurrentIndex(edgeCombo->findData(edge)); edgeCombo->blockSignals(false); queue->addUnique(priorityImmediate,queueItem(funcScopeSpeed,QVariant::fromValue(spectrumBounds(lowerFreq, upperFreq, edge)),false,receiver)); queue->addUnique(priorityImmediate,queueItem(funcScopeSpeed,QVariant::fromValue(spectrumMode_t::spectModeFixed),false,receiver)); } } } void receiverWidget::customSpanPressed() { float maxSpan = 0.0; float minSpan = 1.0; for (const auto &span: rigCaps->scopeCenterSpans) { if (double(span.freq / 1000000.0) > maxSpan) maxSpan = double(span.freq / 1000000.0); if (double(span.freq / 1000000.0) < minSpan) minSpan = double(span.freq / 1000000.0); } maxSpan = maxSpan * 2; minSpan = minSpan * 2; QDialog* dialog = new QDialog(this); dialog->setToolTip(tr("Please enter the lower and upper frequencies (in MHz) for the currently selected Scope Fixed edge")); dialog->setWindowTitle(tr("Scope Edges")); QVBoxLayout* layout = new QVBoxLayout(dialog); layout->setSizeConstraint(QLayout::SetFixedSize); QHBoxLayout* header = new QHBoxLayout(); layout->addLayout(header); QLabel* lowLabel = new QLabel(); lowLabel->setText(tr("Start Freq (MHz)")); lowLabel->setAlignment(Qt::AlignCenter); header->addWidget(lowLabel); QLabel* highLabel = new QLabel(); highLabel->setText(tr("End Freq (MHz)")); highLabel->setAlignment(Qt::AlignCenter); header->addWidget(highLabel); QHBoxLayout* spins = new QHBoxLayout(); layout->addLayout(spins); QDoubleSpinBox* low = new QDoubleSpinBox(); low->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Expanding); low->setValue(lowerFreq); low->setDecimals(3); low->setSingleStep(minSpan); low->setRange(minFreqMhz,maxFreqMhz); low->setAlignment(Qt::AlignCenter); low->setToolTip(tr("Fixed edge start frequency")); spins->addWidget(low); low->setMinimumHeight(low->minimumSizeHint().height()*2); QDoubleSpinBox* high = new QDoubleSpinBox(); high->setSizePolicy(QSizePolicy::Fixed,QSizePolicy::Expanding); high->setValue(upperFreq); high->setDecimals(3); high->setSingleStep(minSpan); high->setMinimumHeight(high->minimumSizeHint().height()*2); high->setRange(minFreqMhz,maxFreqMhz); high->setAlignment(Qt::AlignCenter); high->setToolTip(tr("Fixed edge end frequency")); spins->addWidget(high); QHBoxLayout* buttons = new QHBoxLayout(); layout->addLayout(buttons); QPushButton *ok = new QPushButton("OK"); QPushButton *cancel = new QPushButton("Cancel"); buttons->addWidget(ok); buttons->addWidget(cancel); connect(ok, &QPushButton::clicked, this, [=]() { // Here we need to attempt to update the fixed edge if (high->value() - low->value() > maxSpan) { high->setValue(low->value()+maxSpan); } else if (high->value() < low->value() || high->value() - low->value() < minSpan) { high->setValue(low->value()+minSpan); } else { qDebug(logGui()) << "setting edge to: " << low->value() << ", " << high->value() << ", edge num: " << edgeCombo->currentData().toUInt(); queue->addUnique(priorityImmediate,queueItem(funcScopeFixedEdgeFreq,QVariant::fromValue(spectrumBounds(low->value(), high->value(), edgeCombo->currentData().toUInt())),false,receiver)); dialog->close(); } }); connect(cancel, &QPushButton::clicked, this, [=]() { dialog->close(); }); dialog->exec(); } void receiverWidget::doubleClick(QMouseEvent *me) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); if (me->button() == Qt::LeftButton) { double x; freqt freqGo; if (!freqLock) { //y = plot->yAxis->pixelToCoord(me->pos().y()); x = spectrum->xAxis->pixelToCoord(me->pos().x()); freqGo.Hz = x * 1E6; freqGo.Hz = roundFrequency(freqGo.Hz, stepSize); freqGo.MHzDouble = (float)freqGo.Hz / 1E6; emit sendTrack(freqGo.Hz-this->freq.Hz); setFrequency(freqGo); queue->addUnique(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(freqGo),false,t.receiver)); } } else if (me->button() == Qt::RightButton) { QCPAbstractItem* item = spectrum->itemAt(me->pos(), true); double pbFreq = (pbtDefault / passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; queue->addUnique(priorityImmediate,queueItem(funcPBTInner,QVariant::fromValue(newFreq),false,t.receiver)); queue->addUnique(priorityImmediate,queueItem(funcPBTOuter,QVariant::fromValue(newFreq),false,t.receiver)); QCPItemRect* rectItem = dynamic_cast (item); if (rectItem != nullptr) { } } } void receiverWidget::scopeClick(QMouseEvent* me) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); QCPAbstractItem* item = spectrum->itemAt(me->pos(), true); QCPItemText* textItem = dynamic_cast (item); QCPItemRect* rectItem = dynamic_cast (item); #if QCUSTOMPLOT_VERSION < 0x020000 int leftPix = (int)passbandIndicator->left->pixelPoint().x(); int rightPix = (int)passbandIndicator->right->pixelPoint().x(); int pbtLeftPix = (int)pbtIndicator->left->pixelPoint().x(); int pbtRightPix = (int)pbtIndicator->right->pixelPoint().x(); #else int leftPix = (int)passbandIndicator->left->pixelPosition().x(); int rightPix = (int)passbandIndicator->right->pixelPosition().x(); int pbtLeftPix = (int)pbtIndicator->left->pixelPosition().x(); int pbtRightPix = (int)pbtIndicator->right->pixelPosition().x(); #endif int pbtCenterPix = pbtLeftPix + ((pbtRightPix - pbtLeftPix) / 2); int cursor = me->pos().x(); if (me->button() == Qt::LeftButton) { this->mousePressFreq = spectrum->xAxis->pixelToCoord(cursor); if (textItem != nullptr) { QMap::iterator spot = clusterSpots.find(textItem->text()); if (spot != clusterSpots.end() && spot.key() == textItem->text() && !freqLock) { qInfo(logGui()) << "Clicked on spot:" << textItem->text(); freqt freqGo; freqGo.Hz = (spot.value()->frequency) * 1E6; freqGo.MHzDouble = spot.value()->frequency; emit sendTrack(freqGo.Hz-this->freq.Hz); setFrequency(freqGo); queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(freqGo),false,t.receiver)); } } else if (passbandAction == passbandStatic && rectItem != nullptr) { if ((cursor <= leftPix && cursor > leftPix - 10) || (cursor >= rightPix && cursor < rightPix + 10)) { passbandAction = passbandResizing; } } // TODO clickdragtuning and sending messages to statusbar else if (clickDragTuning) { emit showStatusBarText(QString("Selected %1 MHz").arg(this->mousePressFreq)); } else { emit showStatusBarText(QString("Selected %1 MHz").arg(this->mousePressFreq)); } } else if (me->button() == Qt::RightButton) { if (textItem != nullptr) { QMap::iterator spot = clusterSpots.find(textItem->text()); if (spot != clusterSpots.end() && spot.key() == textItem->text()) { // parent and children are destroyed on close QDialog* spotDialog = new QDialog(); QVBoxLayout* vlayout = new QVBoxLayout; //spotDialog->setFixedSize(240, 100); spotDialog->setBaseSize(1, 1); spotDialog->setWindowTitle(spot.value()->dxcall); QLabel* dxcall = new QLabel(QString("DX:%1").arg(spot.value()->dxcall)); QLabel* spotter = new QLabel(QString("Spotter:%1").arg(spot.value()->spottercall)); QLabel* frequency = new QLabel(QString("Frequency:%1 MHz").arg(spot.value()->frequency)); QLabel* comment = new QLabel(QString("Comment:%1").arg(spot.value()->comment)); QAbstractButton* bExit = new QPushButton("Close"); vlayout->addWidget(dxcall); vlayout->addWidget(spotter); vlayout->addWidget(frequency); vlayout->addWidget(comment); vlayout->addWidget(bExit); spotDialog->setLayout(vlayout); spotDialog->show(); spotDialog->connect(bExit, SIGNAL(clicked()), spotDialog, SLOT(close())); } } else if (passbandAction == passbandStatic && rectItem != nullptr) { if (cursor <= pbtLeftPix && cursor > pbtLeftPix - 10) { passbandAction = pbtInnerMove; } else if (cursor >= pbtRightPix && cursor < pbtRightPix + 10) { passbandAction = pbtOuterMove; } else if (cursor > pbtCenterPix - 20 && cursor < pbtCenterPix + 20) { passbandAction = pbtMoving; } this->mousePressFreq = spectrum->xAxis->pixelToCoord(cursor); } } } void receiverWidget::scopeMouseRelease(QMouseEvent* me) { QCPAbstractItem* item = spectrum->itemAt(me->pos(), true); QCPItemText* textItem = dynamic_cast (item); if (textItem == nullptr && clickDragTuning) { this->mouseReleaseFreq = spectrum->xAxis->pixelToCoord(me->pos().x()); double delta = mouseReleaseFreq - mousePressFreq; qInfo(logGui()) << "Mouse release delta: " << delta; } if (passbandAction != passbandStatic) { passbandAction = passbandStatic; } } void receiverWidget::scopeMouseMove(QMouseEvent* me) { QCPAbstractItem* item = spectrum->itemAt(me->pos(), true); QCPItemText* textItem = dynamic_cast (item); QCPItemRect* rectItem = dynamic_cast (item); #if QCUSTOMPLOT_VERSION < 0x020000 int leftPix = (int)passbandIndicator->left->pixelPoint().x(); int rightPix = (int)passbandIndicator->right->pixelPoint().x(); int pbtLeftPix = (int)pbtIndicator->left->pixelPoint().x(); int pbtRightPix = (int)pbtIndicator->right->pixelPoint().x(); #else int leftPix = (int)passbandIndicator->left->pixelPosition().x(); int rightPix = (int)passbandIndicator->right->pixelPosition().x(); int pbtLeftPix = (int)pbtIndicator->left->pixelPosition().x(); int pbtRightPix = (int)pbtIndicator->right->pixelPosition().x(); #endif int pbtCenterPix = pbtLeftPix + ((pbtRightPix - pbtLeftPix) / 2); int cursor = me->pos().x(); double movedFrequency = spectrum->xAxis->pixelToCoord(cursor) - mousePressFreq; if (passbandAction == passbandStatic && rectItem != nullptr) { if ((cursor <= leftPix && cursor > leftPix - 10) || (cursor >= rightPix && cursor < rightPix + 10) || (cursor <= pbtLeftPix && cursor > pbtLeftPix - 10) || (cursor >= pbtRightPix && cursor < pbtRightPix + 10)) { setCursor(Qt::SizeHorCursor); } else if (cursor > pbtCenterPix - 20 && cursor < pbtCenterPix + 20) { setCursor(Qt::OpenHandCursor); } } else if (passbandAction == passbandResizing) { static double lastFreq = movedFrequency; if (lastFreq - movedFrequency > 0.000049 || movedFrequency - lastFreq > 0.000049) { // We are currently resizing the passband. double pb = 0.0; double origin = passbandCenterFrequency; if (mode.mk == modeLSB) { origin = - passbandCenterFrequency; } if (spectrum->xAxis->pixelToCoord(cursor) >= freq.MHzDouble + origin) { pb = spectrum->xAxis->pixelToCoord(cursor) - passbandIndicator->topLeft->coords().x(); } else { pb = passbandIndicator->bottomRight->coords().x() - spectrum->xAxis->pixelToCoord(cursor); } if (mode.bwMax != 0 && mode.bwMin != 0) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcFilterWidth,QVariant::fromValue(pb * 1000000),false,t.receiver)); } //qInfo() << "New passband" << uint(pb * 1000000); lastFreq = movedFrequency; } } else if (passbandAction == pbtMoving) { //qint16 shift = (qint16)(level - 128); //TPBFInner = (double)(shift / 127.0) * (passbandWidth); // Only move if more than 100Hz static double lastFreq = movedFrequency; if (lastFreq - movedFrequency > 0.000049 || movedFrequency - lastFreq > 0.000049) { double innerFreq = ((double)(PBTInner + movedFrequency) / passbandWidth) * 127.0; double outerFreq = ((double)(PBTOuter + movedFrequency) / passbandWidth) * 127.0; qint16 newInFreq = innerFreq + 128; qint16 newOutFreq = outerFreq + 128; if (newInFreq >= 0 && newInFreq <= 255 && newOutFreq >= 0 && newOutFreq <= 255 && mode.bwMax != 0) { qDebug() << QString("Moving passband by %1 Hz (Inner %2) (Outer %3) Mode:%4").arg((qint16)(movedFrequency * 1000000)) .arg(newInFreq).arg(newOutFreq).arg(mode.mk); vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTInner,QVariant::fromValue(newInFreq),false,t.receiver)); queue->addUnique(priorityImmediate,queueItem(funcPBTOuter,QVariant::fromValue(newInFreq),false,t.receiver)); } lastFreq = movedFrequency; } } else if (passbandAction == pbtInnerMove) { static double lastFreq = movedFrequency; if (lastFreq - movedFrequency > 0.000049 || movedFrequency - lastFreq > 0.000049) { double pbFreq = ((double)(PBTInner + movedFrequency) / passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; if (newFreq >= 0 && newFreq <= 255 && mode.bwMax != 0) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTInner,QVariant::fromValue(newFreq),false,t.receiver)); } lastFreq = movedFrequency; } } else if (passbandAction == pbtOuterMove) { static double lastFreq = movedFrequency; if (lastFreq - movedFrequency > 0.000049 || movedFrequency - lastFreq > 0.000049) { double pbFreq = ((double)(PBTOuter + movedFrequency) / passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; if (newFreq >= 0 && newFreq <= 255 && mode.bwMax != 0) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcPBTOuter,QVariant::fromValue(newFreq),false,t.receiver)); } lastFreq = movedFrequency; } } else if (passbandAction == passbandStatic && me->buttons() == Qt::LeftButton && textItem == nullptr && clickDragTuning ) { double delta = spectrum->xAxis->pixelToCoord(cursor) - mousePressFreq; qDebug(logGui()) << "Mouse moving delta: " << delta; if( (( delta < -0.0001 ) || (delta > 0.0001)) && ((delta < 0.501) && (delta > -0.501)) && !freqLock) { freqt freqGo; freqGo.Hz = (freq.MHzDouble + delta) * 1E6; freqGo.Hz = roundFrequency(freqGo.Hz, stepSize); freqGo.MHzDouble = (float)freqGo.Hz / 1E6; emit sendTrack(freqGo.Hz-this->freq.Hz); setFrequency(freqGo); vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(freqGo),false,t.receiver)); } } else { setCursor(Qt::ArrowCursor); } } void receiverWidget::waterfallClick(QMouseEvent *me) { double x = spectrum->xAxis->pixelToCoord(me->pos().x()); emit showStatusBarText(QString("Selected %1 MHz").arg(x)); } void receiverWidget::scroll(QWheelEvent *we) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); // angleDelta is supposed to be eights of a degree of mouse wheel rotation int deltaY = we->angleDelta().y(); int deltaX = we->angleDelta().x(); int delta = (deltaX==0)?deltaY:deltaX; //qreal offset = qreal(delta) / qreal((deltaX==0)?scrollYperClick:scrollXperClick); //qreal offset = qreal(delta) / 120; qreal numDegrees = delta / 8; qreal offset = numDegrees / 15; qreal stepsToScroll = QApplication::wheelScrollLines() * offset; // Did we change direction? if( (scrollWheelOffsetAccumulated > 0) && (offset > 0) ) { scrollWheelOffsetAccumulated += stepsToScroll; } else if ((scrollWheelOffsetAccumulated < 0) && (offset < 0)) { scrollWheelOffsetAccumulated += stepsToScroll; } else { // Changed direction, zap the old accumulation: scrollWheelOffsetAccumulated = stepsToScroll; //qInfo() << "Scroll changed direction"; } int clicks = int(scrollWheelOffsetAccumulated); if (!clicks) { // qInfo() << "Rejecting minor scroll motion, too small. Wheel Clicks: " << clicks << ", angleDelta: " << delta; // qInfo() << "Accumulator value: " << scrollWheelOffsetAccumulated; // qInfo() << "stepsToScroll: " << stepsToScroll; // qInfo() << "Storing scroll motion for later use."; return; } else { // qInfo() << "Accepted scroll motion. Wheel Clicks: " << clicks << ", angleDelta: " << delta; // qInfo() << "Accumulator value: " << scrollWheelOffsetAccumulated; // qInfo() << "stepsToScroll: " << stepsToScroll; } // If we made it this far, it's time to scroll and to ultimately // clear the scroll accumulator. unsigned int stepsHz = stepSize; Qt::KeyboardModifiers key = we->modifiers(); if ((key == Qt::ShiftModifier) && (stepsHz != 1)) { stepsHz /= 10; } else if (key == Qt::ControlModifier) { stepsHz *= 10; } if (!freqLock) { freqt f; f.Hz = roundFrequency(freq.Hz, clicks, stepsHz); f.MHzDouble = f.Hz / (double)1E6; emit sendTrack(f.Hz-this->freq.Hz); // Where does this go? setFrequencyLocally(f); queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(f),false,receiver)); tempLockAcceptFreqData(); //qInfo() << "Moving to freq:" << f.Hz << "step" << stepsHz; } scrollWheelOffsetAccumulated = 0; } void receiverWidget::receiveMode(modeInfo m, uchar vfo) { // Update mode information if mode/filter/data has changed. // Not all rigs send data so this "might" need to be updated independantly? if (vfo > 0) { if (m.mk != modeUnknown) unselectedMode=m; return; } if (m.filter != 0xff && this->mode.filter != m.filter) { filterCombo->blockSignals(true); filterCombo->setCurrentIndex(filterCombo->findData(m.filter)); filterCombo->blockSignals(false); mode.filter=m.filter; } if (m.data != 0xff && this->mode.data != m.data) { emit dataChanged(m); // Signal wfmain that the data mode has been changed. dataCombo->blockSignals(true); dataCombo->setCurrentIndex(m.data); dataCombo->blockSignals(false); mode.data=m.data; } if (m.mk != modeUnknown && mode.mk != m.mk) { qInfo(logSystem()) << __func__ << QString("Received new mode for %0: %1 (%2) filter:%3 data:%4") .arg((receiver?"Sub":"Main")).arg(QString::number(m.mk,16)).arg(m.name).arg(m.filter).arg(m.data) ; if (this->mode.mk != m.mk) { for (int i=0;icount();i++) { modeInfo mi = modeCombo->itemData(i).value(); if (mi.mk == m.mk) { modeCombo->blockSignals(true); modeCombo->setCurrentIndex(i); modeCombo->blockSignals(false); break; } } } if (m.mk == modeCW || m.mk == modeCW_R || m.mk == modeRTTY || m.mk == modeRTTY_R || m.mk == modePSK || m.mk == modePSK_R) { dataCombo->setEnabled(false); } else { dataCombo->setEnabled(true); } if (m.mk != mode.mk) { // We have changed mode so "may" need to change regular commands passbandCenterFrequency = 0.0; vfoCommandType t = queue->getVfoCommand(vfoA,receiver,false); // Make sure the filterWidth range is within limits. // If new mode doesn't allow bandwidth control, disable filterwidth and pbt. configFilterWidth->blockSignals(true); configFilterWidth->setRange(m.bwMin,m.bwMax); configFilterWidth->setValue(m.bwMax); configFilterWidth->blockSignals(false); if (m.bwMin > 0 || m.bwMax > 0) { // Set config specific options) if (rigCaps->manufacturer == manufKenwood) { if (m.mk == modeCW || m.mk == modeCW_R || m.mk == modePSK || m.mk == modePSK_R) { queue->addUnique(priorityHigh,funcFilterWidth,true,t.receiver); queue->del(funcPBTInner,t.receiver); queue->del(funcPBTOuter,t.receiver); configPbtInner->setEnabled(false); configPbtOuter->setEnabled(false); configIfShift->setEnabled(false); //configFilterWidth->setEnabled(true); } else if (m.mk == modeUSB || m.mk == modeLSB || m.mk == modeAM || m.mk == modeFM) { queue->addUnique(priorityHigh,funcPBTInner,true,t.receiver); queue->addUnique(priorityHigh,funcPBTOuter,true,t.receiver); queue->del(funcFilterWidth,t.receiver); configPbtInner->setEnabled(true && rigCaps->commands.contains(funcPBTInner)); configPbtOuter->setEnabled(true && rigCaps->commands.contains(funcPBTOuter)); configIfShift->setEnabled(true && (rigCaps->commands.contains(funcIFShift) || rigCaps->commands.contains(funcPBTInner))); //configFilterWidth->setEnabled(false); } } else { queue->addUnique(priorityHigh,funcPBTInner,true,t.receiver); queue->addUnique(priorityHigh,funcPBTOuter,true,t.receiver); queue->addUnique(priorityHigh,funcFilterWidth,true,t.receiver); configPbtInner->setEnabled(true); configPbtOuter->setEnabled(true); configFilterWidth->setEnabled(true); } } else{ queue->del(funcPBTInner,t.receiver); queue->del(funcPBTOuter,t.receiver); queue->del(funcFilterWidth,t.receiver); configPbtInner->setEnabled(false); configPbtOuter->setEnabled(false); configIfShift->setEnabled(false); configFilterWidth->setEnabled(false); passbandWidth = double(m.pass/1000000.0); } if (m.mk == modeFM) { queue->addUnique(priorityMediumHigh,funcToneFreq,true,t.receiver); queue->addUnique(priorityMediumHigh,funcTSQLFreq,true,t.receiver); queue->addUnique(priorityMediumHigh,funcRepeaterTone,true,t.receiver); queue->addUnique(priorityMediumHigh,funcRepeaterTSQL,true,t.receiver); } else { queue->del(funcToneFreq,t.receiver); queue->del(funcTSQLFreq,t.receiver); queue->del(funcRepeaterTone,t.receiver); queue->del(funcRepeaterTSQL,t.receiver); } if (m.mk == modeDD || m.mk == modeDV) { t = queue->getVfoCommand(vfoB,receiver,false); queue->del(t.freqFunc,t.receiver); queue->del(t.modeFunc,t.receiver); } else if (queue->getState().vfoMode == vfoModeVfo) { t = queue->getVfoCommand(vfoB,receiver,false); queue->addUnique(priorityHigh,t.freqFunc,true,t.receiver); queue->addUnique(priorityHigh,t.modeFunc,true,t.receiver); } if (m.mk == modeCW || m.mk == modeCW_R) { queue->addUnique(priorityMediumHigh,funcCwPitch,true,t.receiver); queue->addUnique(priorityMediumHigh,funcDashRatio,true,t.receiver); queue->addUnique(priorityMediumHigh,funcKeySpeed,true,t.receiver); queue->addUnique(priorityMediumHigh,funcBreakIn,true,t.receiver); } else { queue->del(funcCwPitch,t.receiver); queue->del(funcDashRatio,t.receiver); queue->del(funcKeySpeed,t.receiver); queue->del(funcBreakIn,t.receiver); } #if defined __GNUC__ #pragma GCC diagnostic push #pragma GCC diagnostic ignored "-Wimplicit-fallthrough" #endif switch (m.mk) { // M0VSE this needs to be replaced with 1/2 the "actual" width of the RTTY signal+the mark freq. case modeRTTY: case modeRTTY_R: passbandCenterFrequency = 0.00008925; break; case modeLSB: case modeUSB: case modePSK: case modePSK_R: if (rigCaps->manufacturer == manufIcom) passbandCenterFrequency = 0.0015; else passbandCenterFrequency = 0.0; case modeAM: case modeCW: case modeCW_R: break; default: break; } #if defined __GNUC__ #pragma GCC diagnostic pop #endif } this->mode = m; } } quint64 receiverWidget::roundFrequency(quint64 frequency, unsigned int tsHz) { return roundFrequency(frequency, 0, tsHz); } quint64 receiverWidget::roundFrequency(quint64 frequency, int steps, unsigned int tsHz) { if(steps > 0) { frequency = frequency + (quint64)(steps*tsHz); } else if (steps < 0) { frequency = frequency - std::min((quint64)(abs(steps)*tsHz), frequency); } quint64 rounded = frequency; if(tuningFloorZeros) { rounded = ((frequency % tsHz) > tsHz/2) ? frequency + tsHz - frequency%tsHz : frequency - frequency%tsHz; } return rounded; } void receiverWidget::receiveCwPitch(quint16 pitch) { if (mode.mk == modeCW || mode.mk == modeCW_R) { if (pitch != this->cwPitch) { passbandCenterFrequency = pitch / 2000000.0; qDebug(logSystem()) << QString("%0 Received new CW Pitch %1 Hz was %2 (center freq %3 MHz)").arg((receiver?"Sub":"Main")).arg(pitch).arg(cwPitch).arg(passbandCenterFrequency); this->cwPitch = pitch; } } } void receiverWidget::receivePassband(quint16 pass) { double pb = (double)(pass / 1000000.0); if (passbandWidth != pb) { passbandWidth = pb; //trxadj->updatePassband(pass); qDebug(logSystem()) << QString("%0 Received new IF Filter/Passband %1 Hz").arg(receiver?"Sub":"Main").arg(pass); emit showStatusBarText(QString("%0 IF filter width %1 Hz (%2 MHz)").arg(receiver?"Sub":"Main").arg(pass).arg(passbandWidth)); configFilterWidth->blockSignals(true); configFilterWidth->setValue(pass); configFilterWidth->blockSignals(false); } } void receiverWidget::selected(bool en) { isActive = en; // Hide scope if we are not selected and don't have command 29. this->displayScope(en || rigCaps->hasCommand29); this->setEnabled(en || rigCaps->hasCommand29); // If this scope is not visible set to visible if selected. this->setVisible(en || this->isVisible()); if (en) { this->setStyleSheet("QGroupBox { border:1px solid red;}"); } else { this->setStyleSheet(defaultStyleSheet); } } void receiverWidget::setHold(bool h) { this->holdButton->blockSignals(true); this->holdButton->setChecked(h); this->holdButton->blockSignals(false); } void receiverWidget::setSpeed(uchar s) { this->currentSpeed = s; configSpeed->setCurrentIndex(configSpeed->findData(currentSpeed)); } void receiverWidget::receiveSpots(uchar receiver, QList spots) { if (receiver != this->receiver) { return; } //QElapsedTimer timer; //timer.start(); bool current = false; if (clusterSpots.size() > 0) { current=clusterSpots.begin().value()->current; } for(const auto &s: spots) { bool found = false; QMap::iterator spot = clusterSpots.find(s.dxcall); while (spot != clusterSpots.end() && spot.key() == s.dxcall && spot.value()->frequency == s.frequency) { spot.value()->current = !current; found = true; ++spot; } if (!found) { QCPRange xrange=spectrum->xAxis->range(); QCPRange yrange=spectrum->yAxis->range(); double left = s.frequency; double top = yrange.upper-2.0; spotData* sp = new spotData(s); sp->current = !current; sp->text = new QCPItemText(spectrum); sp->text->setAntialiased(true); sp->text->setColor(colors.clusterSpots); sp->text->setText(sp->dxcall); sp->text->setFont(QFont(font().family(), 10)); sp->text->setPositionAlignment(Qt::AlignTop | Qt::AlignLeft); sp->text->position->setType(QCPItemPosition::ptPlotCoords); //QMargins margin; //int width = (sp->text->right - sp->text->left) / 8; //margin.setLeft(width); //margin.setRight(width); //sp->text->setPadding(margin); sp->text->position->setCoords(left, top); qDebug(logCluster()) << "ADD:" << sp->dxcall; bool conflict = true; while (conflict) { #if QCUSTOMPLOT_VERSION < 0x020000 QCPItemText* item = dynamic_cast(spectrum->itemAt(QPointF(spectrum->xAxis->coordToPixel(left),spectrum->yAxis->coordToPixel(top)), true)); #else QCPItemText* item = dynamic_cast(spectrum->itemAt(QPointF(spectrum->xAxis->coordToPixel(left),spectrum->yAxis->coordToPixel(top)), true)); #endif //QCPItemText* item = dynamic_cast (absitem); if (item != nullptr) { //qInfo() << "Spot found at top:" << top << "left:" << left << "(" << spectrum->xAxis->coordToPixel(top) <<"," << spectrum->xAxis->coordToPixel(left) << ") spot:" << item->text(); top = top - 5.0; if (top < 10.0) { top = yrange.upper-2.0; left = left + (xrange.size()/20); } } else { conflict = false; } } sp->text->position->setCoords(left, top); sp->text->setSelectable(true); sp->text->setVisible(true); clusterSpots.insert(sp->dxcall, sp); } } QMutableMapIterator sp(clusterSpots); while (sp.hasNext()) { auto spot = sp.next(); if (spot.value()->current == current) { spectrum->removeItem(spot.value()->text); delete spot.value(); sp.remove(); } } //qDebug(logCluster()) << "Processing took" << timer.nsecsElapsed() / 1000 << "us"; } void receiverWidget::configPressed() { this->configGroup->setVisible(!this->configGroup->isVisible()); //QTimer::singleShot(200, this, [this]() { if (this->configGroup->isVisible()) this->confButton->setText(">"); else this->confButton->setText("<"); //}); } void receiverWidget::freqNoteLockTimerSlot() { this->allowAcceptFreqData = true; } void receiverWidget::tempLockAcceptFreqData() { this->allowAcceptFreqData = false; frequencyNotificationLockoutTimer->start(); } void receiverWidget::wfTheme(int num) { currentTheme = QCPColorGradient::GradientPreset(num); colorMap->setGradient(static_cast(num)); configTheme->setCurrentIndex(configTheme->findData(currentTheme)); } void receiverWidget::setPBTInner (uchar val) { if (rigCaps->manufacturer == manufKenwood) { ushort width=0; if (this->mode.mk == modeLSB || this->mode.mk == modeUSB) { if (val < 25) width = (val+6) * 100; else if (val == 25) width = 3400; else if (val == 26) width = 4000; else if (val == 27) width = 5000; } if (double(width)/1000000.0 != this->PBTInner) { this->PBTInner = double(width)/1000000.0; configPbtInner->blockSignals(true); configPbtInner->setValue(val); configPbtInner->blockSignals(false); } } else { qint16 shift = (qint16)(val - 128); double tempVar = ceil((shift / 127.0) * passbandWidth * 20000.0) / 20000.0; // tempVar now contains value to the nearest 50Hz If CW mode, add/remove the cwPitch. double pitch = 0.0; if ((this->mode.mk == modeCW || this->mode.mk == modeCW_R) && this->passbandWidth > 0.0006) { pitch = (600.0 - cwPitch) / 1000000.0; } double newPBT = round((tempVar + pitch) * 200000.0) / 200000.0; // Nearest 5Hz. if (newPBT != this->PBTInner) { this->PBTInner = newPBT; double pbFreq = ((double)(this->PBTInner) / this->passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; if (newFreq >= 0 && newFreq <= 255) { configPbtInner->blockSignals(true); configPbtInner->setValue(newFreq); configPbtInner->blockSignals(false); } } } } void receiverWidget::setPBTOuter (uchar val) { if (rigCaps->manufacturer == manufKenwood) { ushort width=0; if (this->mode.mk == modeLSB || this->mode.mk == modeUSB) { if (val == 1) width = 50; else if (val > 1 && val < 22) width = (val-1) * 100; } if (double(width)/1000000.0 != this->PBTOuter) { this->PBTOuter = double(width)/1000000.0; configPbtOuter->blockSignals(true); configPbtOuter->setValue(val); configPbtOuter->blockSignals(false); } } else { qint16 shift = (qint16)(val - 128); double tempVar = ceil((shift / 127.0) * this->passbandWidth * 20000.0) / 20000.0; // tempVar now contains value to the nearest 50Hz If CW mode, add/remove the cwPitch. double pitch = 0.0; if ((this->mode.mk == modeCW || this->mode.mk == modeCW_R) && this->passbandWidth > 0.0006) { pitch = (600.0 - cwPitch) / 1000000.0; } double newPBT = round((tempVar + pitch) * 200000.0) / 200000.0; // Nearest 5Hz. if (newPBT != this->PBTOuter) { this->PBTOuter = newPBT; double pbFreq = ((double)(this->PBTOuter) / this->passbandWidth) * 127.0; qint16 newFreq = pbFreq + 128; if (newFreq >= 0 && newFreq <= 255) { configPbtOuter->blockSignals(true); configPbtOuter->setValue(newFreq); configPbtOuter->blockSignals(false); } } } } void receiverWidget::setIFShift(uchar val) { configIfShift->setEnabled(true); if (val != this->ifShift) { configIfShift->blockSignals(true); configIfShift->setValue(val); configIfShift->blockSignals(false); this->ifShift = val; } } void receiverWidget::setFrequencyLocally(freqt f, uchar vfo) { // This function is a duplicate of the regular setFrequency // function, but without the locking mechanism. // It is separate because we may wish to do additional things // differently for locally-generated frequency changes. if (vfo < numVFO) { freqDisplay[vfo]->blockSignals(true); freqDisplay[vfo]->setFrequency(f.Hz); freqDisplay[vfo]->blockSignals(false); } if (vfo==0) { this->freq = f; for (const auto &b: rigCaps->bands) { // Highest frequency band is always first! if (f.Hz >= b.lowFreq && f.Hz <= b.highFreq) { // This frequency is contained within this band! if (currentBand.band != b.band) { currentBand = b; emit bandChanged(this->receiver,b); } break; } } } else if (vfo==1) { this->unselectedFreq=f; } } void receiverWidget::setFrequency(freqt f, uchar vfo) { //qInfo() << "receiver:" << receiver << "Setting Frequency vfo=" << vfo << "Freq:" << f.Hz; if(!allowAcceptFreqData) return; if (vfo < numVFO) { freqDisplay[vfo]->blockSignals(true); freqDisplay[vfo]->setFrequency(f.Hz); freqDisplay[vfo]->blockSignals(false); } if (vfo==0) { this->freq = f; for (const auto &b: rigCaps->bands) { // Highest frequency band is always first! if (f.Hz >= b.lowFreq && f.Hz <= b.highFreq) { // This frequency is contained within this band! if (currentBand.band != b.band) { currentBand = b; emit bandChanged(this->receiver,b); } break; } } } else if (vfo==1) { this->unselectedFreq=f; } } void receiverWidget::showBandIndicators(bool en) { for (auto &bi: bandIndicators) { bi.line->setVisible(en); bi.text->setVisible(en); } if (!bandIndicators.empty()) { bandIndicatorsVisible=en; } } void receiverWidget::setBandIndicators(bool show, QString region, std::vector* bands) { this->currentRegion = region; QMutableVectorIterator it(bandIndicators); while (it.hasNext()) { bandIndicator band = it.next(); spectrum->removeItem(band.line); spectrum->removeItem(band.text); it.remove(); } // Step through the bands and add all indicators! if (show) { for (auto &band: *bands) { if (band.region == "" || band.region == region) { // Add band line to current scope! bandIndicator b; b.line = new QCPItemLine(spectrum); b.line->setHead(QCPLineEnding::esLineArrow); b.line->setTail(QCPLineEnding::esLineArrow); b.line->setVisible(false); b.line->setPen(QPen(band.color)); b.line->start->setCoords(double(band.lowFreq/1000000.0), spectrum->yAxis->range().upper-5); b.line->end->setCoords(double(band.highFreq/1000000.0), spectrum->yAxis->range().upper-5); b.text = new QCPItemText(spectrum); b.text->setVisible(false); b.text->setAntialiased(true); b.text->setColor(band.color); b.text->setFont(QFont(font().family(), 8)); b.text->setPositionAlignment(Qt::AlignTop); b.text->setText(band.name); b.text->position->setCoords(double(band.lowFreq/1000000.0), spectrum->yAxis->range().upper-10); bandIndicators.append(b); } } } bandIndicatorsVisible=false; } void receiverWidget::displaySettings(int numDigits, qint64 minf, qint64 maxf, int minStep,FctlUnit unit, std::vector* bands) { this->minFreqMhz = minf / 1000000.0; this->maxFreqMhz = maxf / 1000000.0; for (uchar i=0;isetup(numDigits, minf, maxf, minStep, unit, bands); } void receiverWidget::setUnit(FctlUnit unit) { for (uchar i=0;isetUnit(unit); } void receiverWidget::newFrequency(qint64 freq,uchar vfo) { freqt f; f.Hz = freq; f.MHzDouble = f.Hz / (double)1E6; if (f.Hz > 0 && !freqLock) { emit sendTrack(f.Hz-this->freq.Hz); vfoCommandType t = queue->getVfoCommand(vfo_t(vfo),receiver,true); queue->addUnique(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(f),false,t.receiver)); setFrequencyLocally(f, vfo); tempLockAcceptFreqData(); //qInfo() << "Setting new frequency for rx:" << receiver << "vfo:" << vfo << "freq:" << f.Hz << "using:" << funcString[t.freqFunc]; } } void receiverWidget::setRef(int ref) { configRef->setValue(ref); } void receiverWidget::setRefLimits(int lower, int upper) { configRef->setRange(lower,upper); } void receiverWidget::detachScope(bool state) { if (state) { windowLabel = new QLabel(); detachButton->setText("Attach"); qInfo(logGui()) << "Detaching scope" << (receiver?"Sub":"Main"); this->parentWidget()->layout()->replaceWidget(this,windowLabel); QTimer::singleShot(1, this, [&](){ if(originalParent) { this->originalParent->resize(1,1); } }); this->parentWidget()->resize(1,1); this->setParent(NULL); this-> setWindowFlags(Qt::Window | Qt::WindowMinimizeButtonHint | Qt::WindowMaximizeButtonHint); this->move(screen()->geometry().center() - frameGeometry().center()); } else { detachButton->setText("Detach"); qInfo(logGui()) << "Attaching scope" << (receiver?"Sub":"Main"); windowLabel->parentWidget()->layout()->replaceWidget(windowLabel,this); QTimer::singleShot(1, this, [&](){ if(originalParent) { this->originalParent->resize(1,1); } }); windowLabel->setParent(NULL); delete windowLabel; } // Force a redraw? this->show(); } void receiverWidget::changeSpan(qint8 val) { if ((val > 0 && spanCombo->currentIndex() < spanCombo->count()-val) || (val < char(0) && spanCombo->currentIndex() > 0)) { spanCombo->setCurrentIndex(spanCombo->currentIndex() + val); } else { if (val < char(0)) spanCombo->setCurrentIndex(spanCombo->count()-1); else spanCombo->setCurrentIndex(0); } } void receiverWidget::updateBSR(std::vector* bands) { // Send a new BSR value for the current frequency. // Not currently used. for (auto &b: *bands) { if (quint64(freqDisplay[0]->getFrequency()) >= b.lowFreq && quint64(freqDisplay[0]->getFrequency()) <= b.highFreq) { if(b.bsr != 0) { bandStackType bs(b.bsr,1); bs.data=dataCombo->currentIndex(); bs.filter=filterCombo->currentData().toInt(); bs.freq.Hz = freqDisplay[0]->getFrequency(); bs.freq.MHzDouble=bs.freq.Hz/1000000.0; bs.mode=scopeModeCombo->currentData().toInt(); bs.sql=0; bs.tone.tone=770; bs.tsql.tone=770; vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->addUnique(priorityImmediate,queueItem(funcBandStackReg,QVariant::fromValue(bs),false,t.receiver)); } break; } } } void receiverWidget::memoryMode(bool en) { this->memMode=en; qInfo(logRig) << "Receiver" << receiver << "Memory mode:" << en; if (en) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,false); queue->del(t.freqFunc,t.receiver); queue->del(t.modeFunc,t.receiver); } } QImage receiverWidget::getSpectrumImage() { QMutexLocker locker(&mutex); return spectrum->toPixmap().toImage(); } QImage receiverWidget::getWaterfallImage() { QMutexLocker locker(&mutex); return waterfall->toPixmap().toImage(); } void receiverWidget::vfoSwap() { if (!tracking) { vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); if (rigCaps->commands.contains(funcVFOSwapAB)) { queue->add(priorityImmediate,funcVFOSwapAB,false,t.receiver); } else { // Manufacture a VFO Swap (should be fun!) queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(unselectedFreq),false,t.receiver)); queue->add(priorityImmediate,queueItem(t.modeFunc,QVariant::fromValue(unselectedMode),false,t.receiver)); t = queue->getVfoCommand(vfoB,receiver,true); queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(freq),false,t.receiver)); queue->add(priorityImmediate,queueItem(t.modeFunc,QVariant::fromValue(mode),false,t.receiver)); } } } void receiverWidget::receiveTrack(int f) { return; // Disabled for now. qInfo(logRig) << "Got tracking for rx" << receiver<< "amount" << f << "Hz"; if (tracking && receiver) { // OK I am the sub receiver so lets try this. freqt freqGo; freqGo.Hz=this->freq.Hz+f; freqGo.MHzDouble = (float)freqGo.Hz / 1E6; freqGo.VFO = activeVFO; vfoCommandType t = queue->getVfoCommand(vfoA,receiver,true); queue->add(priorityImmediate,queueItem(t.freqFunc,QVariant::fromValue(freqGo),false,t.receiver)); freq=freqGo; } } // This function will request an update of all freq/mode info from the rig void receiverWidget::updateInfo() { // If we are not the active receiver, delete selected/unselected commands vfoCommandType t = queue->getVfoCommand(vfoA,receiver,false); queue->add(priorityImmediate,t.freqFunc,false,t.receiver); queue->add(priorityImmediate,t.modeFunc,false,t.receiver); if ((rigCaps->hasCommand29 || !receiver) && numVFO > 1) { t = queue->getVfoCommand(vfoB,receiver,false); queue->add(priorityImmediate,t.freqFunc,false,t.receiver); queue->add(priorityImmediate,t.modeFunc,false,t.receiver); } }