2020-12-05 22:42:12 +01:00
|
|
|
#include <gui/widgets/waterfall.h>
|
2020-10-07 22:44:54 +02:00
|
|
|
#include <imgui.h>
|
|
|
|
#include <imgui_internal.h>
|
|
|
|
#include <GL/glew.h>
|
|
|
|
#include <imutils.h>
|
|
|
|
#include <algorithm>
|
2021-04-13 03:52:30 +02:00
|
|
|
#include <volk/volk.h>
|
2021-04-24 04:06:04 +02:00
|
|
|
#include <GLFW/glfw3.h>
|
2020-12-09 19:45:32 +01:00
|
|
|
#include <spdlog/spdlog.h>
|
2021-06-17 20:14:23 +02:00
|
|
|
#include <gui/gui.h>
|
2020-12-09 19:45:32 +01:00
|
|
|
|
2020-12-31 14:26:12 +01:00
|
|
|
float DEFAULT_COLOR_MAP[][3] = {
|
2020-07-11 21:15:10 +02:00
|
|
|
{0x00, 0x00, 0x20},
|
2020-06-10 04:13:56 +02:00
|
|
|
{0x00, 0x00, 0x30},
|
2020-07-11 21:15:10 +02:00
|
|
|
{0x00, 0x00, 0x50},
|
|
|
|
{0x00, 0x00, 0x91},
|
|
|
|
{0x1E, 0x90, 0xFF},
|
|
|
|
{0xFF, 0xFF, 0xFF},
|
|
|
|
{0xFF, 0xFF, 0x00},
|
|
|
|
{0xFE, 0x6D, 0x16},
|
|
|
|
{0xFF, 0x00, 0x00},
|
|
|
|
{0xC6, 0x00, 0x00},
|
|
|
|
{0x9F, 0x00, 0x00},
|
|
|
|
{0x75, 0x00, 0x00},
|
|
|
|
{0x4A, 0x00, 0x00}
|
2020-06-10 04:13:56 +02:00
|
|
|
};
|
|
|
|
|
2021-04-13 03:52:30 +02:00
|
|
|
inline void doZoom(int offset, int width, int outWidth, float* data, float* out, bool fast) {
|
2020-07-19 15:59:44 +02:00
|
|
|
// NOTE: REMOVE THAT SHIT, IT'S JUST A HACKY FIX
|
|
|
|
if (offset < 0) {
|
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
if (width > 65535) {
|
|
|
|
width = 65535;
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
float factor = (float)width / (float)outWidth;
|
2021-04-12 23:02:45 +02:00
|
|
|
|
|
|
|
if (fast) {
|
|
|
|
for (int i = 0; i < outWidth; i++) {
|
|
|
|
out[i] = data[(int)(offset + ((float)i * factor))];
|
|
|
|
}
|
|
|
|
}
|
2021-04-13 03:52:30 +02:00
|
|
|
else {
|
|
|
|
float sFactor = ceilf(factor);
|
|
|
|
float id = offset;
|
|
|
|
float val, maxVal;
|
|
|
|
uint32_t maxId;
|
|
|
|
for (int i = 0; i < outWidth; i++) {
|
|
|
|
maxVal = -INFINITY;
|
|
|
|
for (int j = 0; j < sFactor; j++) {
|
|
|
|
if (data[(int)id + j] > maxVal) { maxVal = data[(int)id + j]; }
|
|
|
|
}
|
|
|
|
out[i] = maxVal;
|
|
|
|
id += factor;
|
2021-04-12 23:02:45 +02:00
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
// TODO: Fix this hacky BS
|
|
|
|
|
|
|
|
double freq_ranges[] = {
|
|
|
|
1.0, 2.0, 2.5, 5.0,
|
|
|
|
10.0, 20.0, 25.0, 50.0,
|
|
|
|
100.0, 200.0, 250.0, 500.0,
|
|
|
|
1000.0, 2000.0, 2500.0, 5000.0,
|
|
|
|
10000.0, 20000.0, 25000.0, 50000.0,
|
|
|
|
100000.0, 200000.0, 250000.0, 500000.0,
|
|
|
|
1000000.0, 2000000.0, 2500000.0, 5000000.0,
|
|
|
|
10000000.0, 20000000.0, 25000000.0, 50000000.0
|
2020-07-11 21:15:10 +02:00
|
|
|
};
|
2020-06-15 15:53:45 +02:00
|
|
|
|
2021-04-16 19:53:47 +02:00
|
|
|
inline double findBestRange(double bandwidth, int maxSteps) {
|
2020-07-19 21:26:37 +02:00
|
|
|
for (int i = 0; i < 32; i++) {
|
2020-10-15 16:09:01 +02:00
|
|
|
if (bandwidth / freq_ranges[i] < (double)maxSteps) {
|
2020-07-11 21:15:10 +02:00
|
|
|
return freq_ranges[i];
|
|
|
|
}
|
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
return 50000000.0;
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
|
2021-04-16 19:53:47 +02:00
|
|
|
inline void printAndScale(double freq, char* buf) {
|
2021-02-20 15:27:43 +01:00
|
|
|
double freqAbs = fabs(freq);
|
|
|
|
if (freqAbs < 1000) {
|
|
|
|
sprintf(buf, "%.6g", freq);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2021-02-20 15:27:43 +01:00
|
|
|
else if (freqAbs < 1000000) {
|
|
|
|
sprintf(buf, "%.6lgK", freq / 1000.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2021-02-20 15:27:43 +01:00
|
|
|
else if (freqAbs < 1000000000) {
|
|
|
|
sprintf(buf, "%.6lgM", freq / 1000000.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2021-02-20 15:27:43 +01:00
|
|
|
else if (freqAbs < 1000000000000) {
|
|
|
|
sprintf(buf, "%.6lgG", freq / 1000000000.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
namespace ImGui {
|
2020-06-10 04:13:56 +02:00
|
|
|
WaterFall::WaterFall() {
|
2020-10-15 16:09:01 +02:00
|
|
|
fftMin = -70.0;
|
|
|
|
fftMax = 0.0;
|
|
|
|
waterfallMin = -70.0;
|
|
|
|
waterfallMax = 0.0;
|
2020-08-20 18:29:23 +02:00
|
|
|
FFTAreaHeight = 300;
|
|
|
|
newFFTAreaHeight = FFTAreaHeight;
|
|
|
|
fftHeight = FFTAreaHeight - 50;
|
2020-07-11 21:15:10 +02:00
|
|
|
dataWidth = 600;
|
|
|
|
lastWidgetPos.x = 0;
|
|
|
|
lastWidgetPos.y = 0;
|
|
|
|
lastWidgetSize.x = 0;
|
|
|
|
lastWidgetSize.y = 0;
|
|
|
|
latestFFT = new float[1];
|
|
|
|
waterfallFb = new uint32_t[1];
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
viewBandwidth = 1.0;
|
|
|
|
wholeBandwidth = 1.0;
|
2020-07-11 21:15:10 +02:00
|
|
|
|
2020-12-31 14:26:12 +01:00
|
|
|
updatePallette(DEFAULT_COLOR_MAP, 13);
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-10-22 03:16:11 +02:00
|
|
|
void WaterFall::init() {
|
|
|
|
glGenTextures(1, &textureId);
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::drawFFT() {
|
|
|
|
// Calculate scaling factor
|
2020-07-19 21:26:37 +02:00
|
|
|
float startLine = floorf(fftMax / vRange) * vRange;
|
2020-07-11 21:15:10 +02:00
|
|
|
float vertRange = fftMax - fftMin;
|
|
|
|
float scaleFactor = fftHeight / vertRange;
|
2020-06-10 04:13:56 +02:00
|
|
|
char buf[100];
|
2020-08-17 02:39:56 +02:00
|
|
|
|
|
|
|
ImU32 trace = ImGui::GetColorU32(ImGuiCol_PlotLines);
|
|
|
|
ImU32 shadow = ImGui::GetColorU32(ImGuiCol_PlotLines, 0.2);
|
2021-06-23 21:45:38 +02:00
|
|
|
ImU32 text = ImGui::GetColorU32(ImGuiCol_Text);
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-06-10 18:52:07 +02:00
|
|
|
// Vertical scale
|
2020-07-19 21:26:37 +02:00
|
|
|
for (float line = startLine; line > fftMin; line -= vRange) {
|
2020-07-11 21:15:10 +02:00
|
|
|
float yPos = widgetPos.y + fftHeight + 10 - ((line - fftMin) * scaleFactor);
|
2020-08-20 18:29:23 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(roundf(widgetPos.x + 50), roundf(yPos)),
|
|
|
|
ImVec2(roundf(widgetPos.x + dataWidth + 50), roundf(yPos)),
|
2020-10-15 16:09:01 +02:00
|
|
|
IM_COL32(50, 50, 50, 255), 1.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
sprintf(buf, "%d", (int)line);
|
2020-06-10 18:52:07 +02:00
|
|
|
ImVec2 txtSz = ImGui::CalcTextSize(buf);
|
2021-06-23 21:45:38 +02:00
|
|
|
window->DrawList->AddText(ImVec2(widgetPos.x + 40 - txtSz.x, roundf(yPos - (txtSz.y / 2.0))), text, buf);
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-06-10 18:52:07 +02:00
|
|
|
// Horizontal scale
|
2020-10-15 16:09:01 +02:00
|
|
|
double startFreq = ceilf(lowerFreq / range) * range;
|
|
|
|
double horizScale = (double)dataWidth / viewBandwidth;
|
|
|
|
for (double freq = startFreq; freq < upperFreq; freq += range) {
|
|
|
|
double xPos = widgetPos.x + 50 + ((freq - lowerFreq) * horizScale);
|
2020-07-11 21:15:10 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(roundf(xPos), widgetPos.y + 10),
|
|
|
|
ImVec2(roundf(xPos), widgetPos.y + fftHeight + 10),
|
2020-10-15 16:09:01 +02:00
|
|
|
IM_COL32(50, 50, 50, 255), 1.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(roundf(xPos), widgetPos.y + fftHeight + 10),
|
|
|
|
ImVec2(roundf(xPos), widgetPos.y + fftHeight + 17),
|
2021-06-23 21:45:38 +02:00
|
|
|
text, 1.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
printAndScale(freq, buf);
|
2020-06-10 18:52:07 +02:00
|
|
|
ImVec2 txtSz = ImGui::CalcTextSize(buf);
|
2021-06-23 21:45:38 +02:00
|
|
|
window->DrawList->AddText(ImVec2(roundf(xPos - (txtSz.x / 2.0)), widgetPos.y + fftHeight + 10 + txtSz.y), text, buf);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Data
|
2020-12-14 19:33:30 +01:00
|
|
|
if (latestFFT != NULL && fftLines != 0) {
|
|
|
|
for (int i = 1; i < dataWidth; i++) {
|
|
|
|
double aPos = widgetPos.y + fftHeight + 10 - ((latestFFT[i - 1] - fftMin) * scaleFactor);
|
|
|
|
double bPos = widgetPos.y + fftHeight + 10 - ((latestFFT[i] - fftMin) * scaleFactor);
|
|
|
|
aPos = std::clamp<double>(aPos, widgetPos.y + 10, widgetPos.y + fftHeight + 10);
|
|
|
|
bPos = std::clamp<double>(bPos, widgetPos.y + 10, widgetPos.y + fftHeight + 10);
|
|
|
|
window->DrawList->AddLine(ImVec2(widgetPos.x + 49 + i, roundf(aPos)),
|
|
|
|
ImVec2(widgetPos.x + 50 + i, roundf(bPos)), trace, 1.0);
|
|
|
|
window->DrawList->AddLine(ImVec2(widgetPos.x + 50 + i, roundf(bPos)),
|
|
|
|
ImVec2(widgetPos.x + 50 + i, widgetPos.y + fftHeight + 10), shadow, 1.0);
|
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2021-07-04 02:25:36 +02:00
|
|
|
FFTRedrawArgs args;
|
|
|
|
args.min = ImVec2(widgetPos.x + 50, widgetPos.y + 9);
|
|
|
|
args.max = ImVec2(widgetPos.x + dataWidth + 50, widgetPos.y + fftHeight + 10);
|
|
|
|
args.lowFreq = lowerFreq;
|
|
|
|
args.highFreq = upperFreq;
|
|
|
|
args.freqToPixelRatio = horizScale;
|
|
|
|
args.pixelToFreqRatio = viewBandwidth / (double)dataWidth;
|
|
|
|
args.window = window;
|
|
|
|
onFFTRedraw.emit(args);
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
// X Axis
|
|
|
|
window->DrawList->AddLine(ImVec2(widgetPos.x + 50, widgetPos.y + fftHeight + 10),
|
|
|
|
ImVec2(widgetPos.x + dataWidth + 50, widgetPos.y + fftHeight + 10),
|
2021-06-23 21:45:38 +02:00
|
|
|
text, 1.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
// Y Axis
|
2020-07-19 15:59:44 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(widgetPos.x + 50, widgetPos.y + 9),
|
|
|
|
ImVec2(widgetPos.x + 50, widgetPos.y + fftHeight + 9),
|
2021-06-23 21:45:38 +02:00
|
|
|
text, 1.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::drawWaterfall() {
|
|
|
|
if (waterfallUpdate) {
|
|
|
|
waterfallUpdate = false;
|
|
|
|
updateWaterfallTexture();
|
2020-06-10 18:52:07 +02:00
|
|
|
}
|
2021-02-09 02:11:40 +01:00
|
|
|
window->DrawList->AddImage((void*)(intptr_t)textureId, wfMin, wfMax);
|
|
|
|
ImVec2 mPos = ImGui::GetMousePos();
|
|
|
|
|
2021-07-12 16:53:59 +02:00
|
|
|
if (IS_IN_AREA(mPos, wfMin, wfMax) && !gui::mainWindow.lockWaterfallControls && !inputHandled) {
|
2021-02-09 02:11:40 +01:00
|
|
|
for (auto const& [name, vfo] : vfos) {
|
2021-05-04 20:41:23 +02:00
|
|
|
window->DrawList->AddRectFilled(vfo->wfRectMin, vfo->wfRectMax, vfo->color);
|
2021-07-04 16:41:46 +02:00
|
|
|
if (!vfo->lineVisible) { continue; }
|
2021-02-10 21:35:56 +01:00
|
|
|
window->DrawList->AddLine(vfo->wfLineMin, vfo->wfLineMax, (name == selectedVFO) ? IM_COL32(255, 0, 0, 255) : IM_COL32(255, 255, 0, 255));
|
2021-02-09 02:11:40 +01:00
|
|
|
}
|
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 18:52:07 +02:00
|
|
|
|
2020-08-10 02:30:25 +02:00
|
|
|
void WaterFall::drawVFOs() {
|
|
|
|
for (auto const& [name, vfo] : vfos) {
|
|
|
|
vfo->draw(window, name == selectedVFO);
|
|
|
|
}
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2020-08-11 18:33:42 +02:00
|
|
|
void WaterFall::selectFirstVFO() {
|
2020-12-08 04:36:37 +01:00
|
|
|
bool available = false;
|
2020-08-11 18:33:42 +02:00
|
|
|
for (auto const& [name, vfo] : vfos) {
|
2020-12-08 04:36:37 +01:00
|
|
|
available = true;
|
2020-08-11 18:33:42 +02:00
|
|
|
selectedVFO = name;
|
2021-04-24 19:26:22 +02:00
|
|
|
selectedVFOChanged = true;
|
2020-08-11 18:33:42 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-12-08 04:36:37 +01:00
|
|
|
if (!available) {
|
|
|
|
selectedVFO = "";
|
2021-04-24 19:26:22 +02:00
|
|
|
selectedVFOChanged = true;
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
2020-08-11 18:33:42 +02:00
|
|
|
}
|
|
|
|
|
2020-08-10 02:30:25 +02:00
|
|
|
void WaterFall::processInputs() {
|
2021-04-17 22:37:50 +02:00
|
|
|
// Pre calculate useful values
|
|
|
|
WaterfallVFO* selVfo = NULL;
|
2020-12-08 04:36:37 +01:00
|
|
|
if (selectedVFO != "") {
|
2021-04-17 22:37:50 +02:00
|
|
|
selVfo = vfos[selectedVFO];
|
2020-12-08 04:36:37 +01:00
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
ImVec2 mousePos = ImGui::GetMousePos();
|
|
|
|
ImVec2 drag = ImGui::GetMouseDragDelta(ImGuiMouseButton_Left);
|
|
|
|
ImVec2 dragOrigin(mousePos.x - drag.x, mousePos.y - drag.y);
|
2020-08-16 03:39:05 +02:00
|
|
|
|
|
|
|
bool mouseHovered, mouseHeld;
|
2021-04-17 22:37:50 +02:00
|
|
|
bool mouseClicked = ImGui::ButtonBehavior(ImRect(fftAreaMin, wfMax), GetID("WaterfallID"), &mouseHovered, &mouseHeld,
|
2020-08-16 03:39:05 +02:00
|
|
|
ImGuiButtonFlags_MouseButtonLeft | ImGuiButtonFlags_PressedOnClick);
|
2021-02-09 02:11:40 +01:00
|
|
|
|
2020-08-16 03:39:05 +02:00
|
|
|
bool draging = ImGui::IsMouseDragging(ImGuiMouseButton_Left) && ImGui::IsWindowFocused();
|
2021-04-24 01:24:27 +02:00
|
|
|
mouseInFreq = IS_IN_AREA(dragOrigin, freqAreaMin, freqAreaMax);
|
|
|
|
mouseInFFT = IS_IN_AREA(dragOrigin, fftAreaMin, fftAreaMax);
|
|
|
|
mouseInWaterfall = IS_IN_AREA(dragOrigin, wfMin, wfMax);
|
|
|
|
|
|
|
|
int mouseWheel = ImGui::GetIO().MouseWheel;
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2021-05-18 02:26:55 +02:00
|
|
|
bool mouseMoved = false;
|
|
|
|
if (mousePos.x != lastMousePos.x || mousePos.y != lastMousePos.y) { mouseMoved = true; }
|
|
|
|
lastMousePos = mousePos;
|
|
|
|
|
|
|
|
std::string hoveredVFOName = "";
|
|
|
|
for (auto const& [name, _vfo] : vfos) {
|
|
|
|
if (ImGui::IsMouseHoveringRect(_vfo->rectMin, _vfo->rectMax) || ImGui::IsMouseHoveringRect(_vfo->wfRectMin, _vfo->wfRectMax)) {
|
|
|
|
hoveredVFOName = name;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
// Deselect everything if the mouse is released
|
|
|
|
if (!ImGui::IsMouseDown(ImGuiMouseButton_Left)) {
|
|
|
|
freqScaleSelect = false;
|
|
|
|
vfoSelect = false;
|
|
|
|
vfoBorderSelect = false;
|
|
|
|
lastDrag = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If mouse was clicked, check what was clicked
|
2020-08-16 03:39:05 +02:00
|
|
|
if (mouseClicked) {
|
2021-04-17 22:37:50 +02:00
|
|
|
bool targetFound = false;
|
|
|
|
mouseDownPos = mousePos;
|
|
|
|
|
|
|
|
// First, check if a VFO border was selected
|
2020-08-11 18:33:42 +02:00
|
|
|
for (auto const& [name, _vfo] : vfos) {
|
2021-04-17 22:37:50 +02:00
|
|
|
if (_vfo->bandwidthLocked) { continue; }
|
2021-07-31 20:17:36 +02:00
|
|
|
if (_vfo->rectMax.x - _vfo->rectMin.x < 10) { continue; }
|
2021-04-17 22:37:50 +02:00
|
|
|
bool resizing = false;
|
|
|
|
if (_vfo->reference != REF_LOWER) {
|
|
|
|
if (IS_IN_AREA(mousePos, _vfo->lbwSelMin, _vfo->lbwSelMax)) { resizing = true; }
|
|
|
|
else if (IS_IN_AREA(mousePos, _vfo->wfLbwSelMin, _vfo->wfLbwSelMax)) { resizing = true; }
|
2020-08-11 18:33:42 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
if (_vfo->reference != REF_UPPER) {
|
|
|
|
if (IS_IN_AREA(mousePos, _vfo->rbwSelMin, _vfo->rbwSelMax)) { resizing = true; }
|
|
|
|
else if (IS_IN_AREA(mousePos, _vfo->wfRbwSelMin, _vfo->wfRbwSelMax)) { resizing = true; }
|
|
|
|
}
|
|
|
|
if (!resizing) { continue; }
|
|
|
|
relatedVfo = _vfo;
|
|
|
|
vfoBorderSelect = true;
|
|
|
|
targetFound = true;
|
|
|
|
break;
|
2020-08-11 18:33:42 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
|
|
|
|
// Next, check if a VFO was selected
|
2021-05-18 02:26:55 +02:00
|
|
|
if (!targetFound && hoveredVFOName != "") {
|
|
|
|
selectedVFO = hoveredVFOName;
|
|
|
|
selectedVFOChanged = true;
|
|
|
|
targetFound = true;
|
|
|
|
return;
|
2020-08-11 18:33:42 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
|
|
|
|
// Now, check frequency scale
|
|
|
|
if (!targetFound && mouseInFreq) {
|
|
|
|
freqScaleSelect = true;
|
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
// If a vfo border is selected, resize VFO accordingly
|
|
|
|
if (vfoBorderSelect) {
|
2021-04-18 15:59:37 +02:00
|
|
|
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW);
|
2021-04-18 00:47:23 +02:00
|
|
|
double dist = (relatedVfo->reference == REF_CENTER) ? fabsf(mousePos.x - relatedVfo->lineMin.x) : (mousePos.x - relatedVfo->lineMin.x);
|
|
|
|
if (relatedVfo->reference == REF_UPPER) { dist = -dist; }
|
2021-04-17 22:37:50 +02:00
|
|
|
double hzDist = dist * (viewBandwidth / (double)dataWidth);
|
|
|
|
if (relatedVfo->reference == REF_CENTER) {
|
|
|
|
hzDist *= 2.0;
|
2020-08-11 18:33:42 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
hzDist = std::clamp<double>(hzDist, relatedVfo->minBandwidth, relatedVfo->maxBandwidth);
|
|
|
|
relatedVfo->setBandwidth(hzDist);
|
2021-08-20 20:40:14 +02:00
|
|
|
relatedVfo->onUserChangedBandwidth.emit(hzDist);
|
2021-04-17 22:37:50 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
// If the frequency scale is selected, move it
|
|
|
|
if (freqScaleSelect) {
|
2021-04-18 15:59:37 +02:00
|
|
|
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW);
|
2020-10-15 16:09:01 +02:00
|
|
|
double deltax = drag.x - lastDrag;
|
2020-07-19 15:59:44 +02:00
|
|
|
lastDrag = drag.x;
|
2020-10-15 16:09:01 +02:00
|
|
|
double viewDelta = deltax * (viewBandwidth / (double)dataWidth);
|
2020-07-19 15:59:44 +02:00
|
|
|
|
|
|
|
viewOffset -= viewDelta;
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
if (viewOffset + (viewBandwidth / 2.0) > wholeBandwidth / 2.0) {
|
|
|
|
double freqOffset = (viewOffset + (viewBandwidth / 2.0)) - (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (wholeBandwidth / 2.0) - (viewBandwidth / 2.0);
|
2021-07-21 04:08:28 +02:00
|
|
|
if (!centerFrequencyLocked) {
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
if (viewOffset - (viewBandwidth / 2.0) < -(wholeBandwidth / 2.0)) {
|
|
|
|
double freqOffset = (viewOffset - (viewBandwidth / 2.0)) + (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (viewBandwidth / 2.0) - (wholeBandwidth / 2.0);
|
2021-07-21 04:08:28 +02:00
|
|
|
if (!centerFrequencyLocked) {
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
2021-04-24 01:24:27 +02:00
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
|
|
|
|
|
|
|
if (viewBandwidth != wholeBandwidth) {
|
|
|
|
updateAllVFOs();
|
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the mouse wheel is moved on the frequency scale
|
|
|
|
if (mouseWheel != 0 && mouseInFreq) {
|
2021-07-28 02:18:54 +02:00
|
|
|
viewOffset -= (double)mouseWheel * viewBandwidth / 20.0;
|
2021-04-24 01:24:27 +02:00
|
|
|
|
|
|
|
if (viewOffset + (viewBandwidth / 2.0) > wholeBandwidth / 2.0) {
|
|
|
|
double freqOffset = (viewOffset + (viewBandwidth / 2.0)) - (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (wholeBandwidth / 2.0) - (viewBandwidth / 2.0);
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
|
|
|
if (viewOffset - (viewBandwidth / 2.0) < -(wholeBandwidth / 2.0)) {
|
|
|
|
double freqOffset = (viewOffset - (viewBandwidth / 2.0)) + (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (viewBandwidth / 2.0) - (wholeBandwidth / 2.0);
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
2021-04-24 04:06:04 +02:00
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
|
|
|
|
|
|
|
if (viewBandwidth != wholeBandwidth) {
|
|
|
|
updateAllVFOs();
|
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the left and right keys are pressed while hovering the freq scale, move it too
|
2021-07-31 21:00:47 +02:00
|
|
|
bool leftKeyPressed = ImGui::IsKeyPressed(GLFW_KEY_LEFT);
|
|
|
|
if ((leftKeyPressed || ImGui::IsKeyPressed(GLFW_KEY_RIGHT)) && mouseInFreq) {
|
|
|
|
viewOffset += leftKeyPressed ? (viewBandwidth / 20.0) : (-viewBandwidth / 20.0);
|
2021-04-24 04:06:04 +02:00
|
|
|
|
|
|
|
if (viewOffset + (viewBandwidth / 2.0) > wholeBandwidth / 2.0) {
|
|
|
|
double freqOffset = (viewOffset + (viewBandwidth / 2.0)) - (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (wholeBandwidth / 2.0) - (viewBandwidth / 2.0);
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
|
|
|
if (viewOffset - (viewBandwidth / 2.0) < -(wholeBandwidth / 2.0)) {
|
|
|
|
double freqOffset = (viewOffset - (viewBandwidth / 2.0)) + (wholeBandwidth / 2.0);
|
|
|
|
viewOffset = (viewBandwidth / 2.0) - (wholeBandwidth / 2.0);
|
|
|
|
centerFreq += freqOffset;
|
|
|
|
centerFreqMoved = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
2020-10-15 16:09:01 +02:00
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
2020-12-12 05:34:58 +01:00
|
|
|
|
|
|
|
if (viewBandwidth != wholeBandwidth) {
|
|
|
|
updateAllVFOs();
|
2021-04-13 04:54:47 +02:00
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
return;
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
|
|
|
|
// Finally, if nothing else was selected, just move the VFO
|
2021-05-18 02:26:55 +02:00
|
|
|
if (ImGui::IsMouseDown(ImGuiMouseButton_Left) && (mouseInFFT|mouseInWaterfall) && (mouseMoved || hoveredVFOName == "")) {
|
2021-04-17 22:37:50 +02:00
|
|
|
if (selVfo != NULL) {
|
|
|
|
int refCenter = mousePos.x - (widgetPos.x + 50);
|
|
|
|
if (refCenter >= 0 && refCenter < dataWidth) {
|
|
|
|
double off = ((((double)refCenter / ((double)dataWidth / 2.0)) - 1.0) * (viewBandwidth / 2.0)) + viewOffset;
|
|
|
|
off += centerFreq;
|
|
|
|
off = (round(off / selVfo->snapInterval) * selVfo->snapInterval) - centerFreq;
|
|
|
|
selVfo->setOffset(off);
|
|
|
|
}
|
|
|
|
}
|
2021-04-29 23:29:40 +02:00
|
|
|
}
|
2021-05-18 02:26:55 +02:00
|
|
|
else if (!ImGui::IsMouseDown(ImGuiMouseButton_Left)) {
|
2021-04-29 23:29:40 +02:00
|
|
|
// Check if a VFO is hovered. If yes, show tooltip
|
|
|
|
for (auto const& [name, _vfo] : vfos) {
|
|
|
|
if (ImGui::IsMouseHoveringRect(_vfo->rectMin, _vfo->rectMax) || ImGui::IsMouseHoveringRect(_vfo->wfRectMin, _vfo->wfRectMax)) {
|
|
|
|
char buf[128];
|
|
|
|
ImGui::BeginTooltip();
|
|
|
|
|
|
|
|
ImGui::Text(name.c_str());
|
|
|
|
|
|
|
|
if (ImGui::IsKeyDown(GLFW_KEY_LEFT_CONTROL) || ImGui::IsKeyDown(GLFW_KEY_RIGHT_CONTROL)) {
|
|
|
|
ImGui::Separator();
|
|
|
|
printAndScale(_vfo->generalOffset + centerFreq, buf);
|
|
|
|
ImGui::Text("Frequency: %sHz", buf);
|
|
|
|
printAndScale(_vfo->bandwidth, buf);
|
|
|
|
ImGui::Text("Bandwidth: %sHz", buf);
|
|
|
|
ImGui::Text("Bandwidth Locked: %s", _vfo->bandwidthLocked ? "Yes" : "No");
|
2021-04-30 04:28:08 +02:00
|
|
|
|
|
|
|
float strength, snr;
|
2021-06-29 03:32:40 +02:00
|
|
|
if (calculateVFOSignalInfo(waterfallVisible ? &rawFFTs[currentFFTLine * rawFFTSize] : rawFFTs, _vfo, strength, snr)) {
|
2021-04-30 04:28:08 +02:00
|
|
|
ImGui::Text("Strength: %0.1fdBFS", strength);
|
|
|
|
ImGui::Text("SNR: %0.1fdB", snr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ImGui::Text("Strength: ---.-dBFS");
|
|
|
|
ImGui::Text("SNR: ---.-dB");
|
|
|
|
}
|
2021-04-29 23:29:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndTooltip();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2021-07-31 20:17:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Handle Page Up to cycle through VFOs
|
|
|
|
if (ImGui::IsKeyPressed(GLFW_KEY_PAGE_UP) && selVfo != NULL) {
|
|
|
|
std::string next = (--vfos.end())->first;
|
|
|
|
std::string lowest = "";
|
|
|
|
double lowestOffset = INFINITY;
|
|
|
|
double firstVfoOffset = selVfo->generalOffset;
|
|
|
|
double smallestDistance = INFINITY;
|
|
|
|
bool found = false;
|
|
|
|
for (auto& [_name, _vfo] : vfos) {
|
|
|
|
if (_vfo->generalOffset > firstVfoOffset && (_vfo->generalOffset - firstVfoOffset) < smallestDistance) {
|
|
|
|
next = _name;
|
|
|
|
smallestDistance = (_vfo->generalOffset - firstVfoOffset);
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
if (_vfo->generalOffset < lowestOffset) {
|
|
|
|
lowestOffset = _vfo->generalOffset;
|
|
|
|
lowest = _name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
selectedVFO = found ? next : lowest;
|
|
|
|
selectedVFOChanged = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle Page Down to cycle through VFOs
|
|
|
|
if (ImGui::IsKeyPressed(GLFW_KEY_PAGE_DOWN) && selVfo != NULL) {
|
|
|
|
std::string next = (--vfos.end())->first;
|
|
|
|
std::string highest = "";
|
|
|
|
double highestOffset = -INFINITY;
|
|
|
|
double firstVfoOffset = selVfo->generalOffset;
|
|
|
|
double smallestDistance = INFINITY;
|
|
|
|
bool found = false;
|
|
|
|
for (auto& [_name, _vfo] : vfos) {
|
|
|
|
if (_vfo->generalOffset < firstVfoOffset && (firstVfoOffset - _vfo->generalOffset) < smallestDistance) {
|
|
|
|
next = _name;
|
|
|
|
smallestDistance = (firstVfoOffset - _vfo->generalOffset);
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
if (_vfo->generalOffset > highestOffset) {
|
|
|
|
highestOffset = _vfo->generalOffset;
|
|
|
|
highest = _name;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
selectedVFO = found ? next : highest;
|
|
|
|
selectedVFOChanged = true;
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
|
|
|
|
2021-06-29 03:32:40 +02:00
|
|
|
bool WaterFall::calculateVFOSignalInfo(float* fftLine, WaterfallVFO* _vfo, float& strength, float& snr) {
|
|
|
|
if (fftLine == NULL || fftLines <= 0) { return false; }
|
2021-04-30 04:28:08 +02:00
|
|
|
|
|
|
|
// Calculate FFT index data
|
|
|
|
double vfoMinSizeFreq = _vfo->centerOffset - _vfo->bandwidth;
|
|
|
|
double vfoMinFreq = _vfo->centerOffset - (_vfo->bandwidth/2.0);
|
|
|
|
double vfoMaxFreq = _vfo->centerOffset + (_vfo->bandwidth/2.0);
|
|
|
|
double vfoMaxSizeFreq = _vfo->centerOffset + _vfo->bandwidth;
|
|
|
|
int vfoMinSideOffset = std::clamp<int>(((vfoMinSizeFreq / (wholeBandwidth/2.0)) * (double)(rawFFTSize/2)) + (rawFFTSize/2), 0, rawFFTSize);
|
|
|
|
int vfoMinOffset = std::clamp<int>(((vfoMinFreq / (wholeBandwidth/2.0)) * (double)(rawFFTSize/2)) + (rawFFTSize/2), 0, rawFFTSize);
|
|
|
|
int vfoMaxOffset = std::clamp<int>(((vfoMaxFreq / (wholeBandwidth/2.0)) * (double)(rawFFTSize/2)) + (rawFFTSize/2), 0, rawFFTSize);
|
|
|
|
int vfoMaxSideOffset = std::clamp<int>(((vfoMaxSizeFreq / (wholeBandwidth/2.0)) * (double)(rawFFTSize/2)) + (rawFFTSize/2), 0, rawFFTSize);
|
|
|
|
|
|
|
|
double avg = 0;
|
|
|
|
float max = -INFINITY;
|
|
|
|
int avgCount = 0;
|
|
|
|
|
|
|
|
// Calculate Left average
|
|
|
|
for (int i = vfoMinSideOffset; i < vfoMinOffset; i++) {
|
|
|
|
avg += fftLine[i];
|
|
|
|
avgCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate Right average
|
|
|
|
for (int i = vfoMaxOffset + 1; i < vfoMaxSideOffset; i++) {
|
|
|
|
avg += fftLine[i];
|
|
|
|
avgCount++;
|
|
|
|
}
|
|
|
|
|
|
|
|
avg /= (double)(avgCount);
|
|
|
|
|
|
|
|
// Calculate max
|
|
|
|
for (int i = vfoMinOffset; i <= vfoMaxOffset; i++) {
|
|
|
|
if (fftLine[i] > max) { max = fftLine[i]; }
|
|
|
|
}
|
|
|
|
|
|
|
|
strength = max;
|
|
|
|
snr = max - avg;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-12 23:02:45 +02:00
|
|
|
void WaterFall::setFastFFT(bool fastFFT) {
|
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
|
|
|
_fastFFT = fastFFT;
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::updateWaterfallFb() {
|
2020-12-12 05:34:58 +01:00
|
|
|
if (!waterfallVisible || rawFFTs == NULL) {
|
2020-08-20 18:29:23 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
double offsetRatio = viewOffset / (wholeBandwidth / 2.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
int drawDataSize;
|
|
|
|
int drawDataStart;
|
2020-12-12 05:34:58 +01:00
|
|
|
// TODO: Maybe put on the stack for faster alloc?
|
2020-07-11 21:15:10 +02:00
|
|
|
float* tempData = new float[dataWidth];
|
|
|
|
float pixel;
|
|
|
|
float dataRange = waterfallMax - waterfallMin;
|
2020-12-12 05:34:58 +01:00
|
|
|
int count = std::min<float>(waterfallHeight, fftLines);
|
2021-07-28 22:53:38 +02:00
|
|
|
if (rawFFTs != NULL && fftLines >= 0) {
|
2020-12-12 05:34:58 +01:00
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
drawDataSize = (viewBandwidth / wholeBandwidth) * rawFFTSize;
|
|
|
|
drawDataStart = (((double)rawFFTSize / 2.0) * (offsetRatio + 1)) - (drawDataSize / 2);
|
2021-04-12 23:02:45 +02:00
|
|
|
doZoom(drawDataStart, drawDataSize, dataWidth, &rawFFTs[((i + currentFFTLine) % waterfallHeight) * rawFFTSize], tempData, _fastFFT);
|
2020-12-12 05:34:58 +01:00
|
|
|
for (int j = 0; j < dataWidth; j++) {
|
|
|
|
pixel = (std::clamp<float>(tempData[j], waterfallMin, waterfallMax) - waterfallMin) / dataRange;
|
|
|
|
waterfallFb[(i * dataWidth) + j] = waterfallPallet[(int)(pixel * (WATERFALL_RESOLUTION - 1))];
|
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
2021-07-28 22:53:38 +02:00
|
|
|
|
|
|
|
for (int i = count; i < waterfallHeight; i++) {
|
|
|
|
for (int j = 0; j < dataWidth; j++) {
|
|
|
|
waterfallFb[(i * dataWidth) + j] = (uint32_t)255 << 24;
|
|
|
|
}
|
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
delete[] tempData;
|
|
|
|
waterfallUpdate = true;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-08-05 21:13:53 +02:00
|
|
|
void WaterFall::drawBandPlan() {
|
|
|
|
int count = bandplan->bands.size();
|
2020-10-15 16:09:01 +02:00
|
|
|
double horizScale = (double)dataWidth / viewBandwidth;
|
|
|
|
double start, end, center, aPos, bPos, cPos, width;
|
2020-08-05 21:13:53 +02:00
|
|
|
ImVec2 txtSz;
|
|
|
|
bool startVis, endVis;
|
|
|
|
uint32_t color, colorTrans;
|
2021-04-14 01:45:21 +02:00
|
|
|
|
|
|
|
float height = ImGui::CalcTextSize("0").y * 2.5f;
|
|
|
|
float bpBottom;
|
|
|
|
|
|
|
|
if (bandPlanPos == BANDPLAN_POS_BOTTOM) {
|
|
|
|
bpBottom = widgetPos.y + fftHeight + 10;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bpBottom = widgetPos.y + height + 10;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-08-05 21:13:53 +02:00
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
start = bandplan->bands[i].start;
|
|
|
|
end = bandplan->bands[i].end;
|
|
|
|
if (start < lowerFreq && end < lowerFreq) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (start > upperFreq && end > upperFreq) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
startVis = (start > lowerFreq);
|
|
|
|
endVis = (end < upperFreq);
|
2020-10-15 16:09:01 +02:00
|
|
|
start = std::clamp<double>(start, lowerFreq, upperFreq);
|
|
|
|
end = std::clamp<double>(end, lowerFreq, upperFreq);
|
|
|
|
center = (start + end) / 2.0;
|
2020-08-05 21:13:53 +02:00
|
|
|
aPos = widgetPos.x + 50 + ((start - lowerFreq) * horizScale);
|
|
|
|
bPos = widgetPos.x + 50 + ((end - lowerFreq) * horizScale);
|
|
|
|
cPos = widgetPos.x + 50 + ((center - lowerFreq) * horizScale);
|
|
|
|
width = bPos - aPos;
|
|
|
|
txtSz = ImGui::CalcTextSize(bandplan->bands[i].name.c_str());
|
|
|
|
if (bandplan::colorTable.find(bandplan->bands[i].type.c_str()) != bandplan::colorTable.end()) {
|
|
|
|
color = bandplan::colorTable[bandplan->bands[i].type].colorValue;
|
|
|
|
colorTrans = bandplan::colorTable[bandplan->bands[i].type].transColorValue;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
color = IM_COL32(255, 255, 255, 255);
|
|
|
|
colorTrans = IM_COL32(255, 255, 255, 100);
|
|
|
|
}
|
|
|
|
if (aPos <= widgetPos.x + 50) {
|
|
|
|
aPos = widgetPos.x + 51;
|
|
|
|
}
|
|
|
|
if (bPos <= widgetPos.x + 50) {
|
|
|
|
bPos = widgetPos.x + 51;
|
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
if (width >= 1.0) {
|
2021-04-14 01:45:21 +02:00
|
|
|
window->DrawList->AddRectFilled(ImVec2(roundf(aPos), bpBottom - height),
|
|
|
|
ImVec2(roundf(bPos), bpBottom), colorTrans);
|
2020-08-05 21:13:53 +02:00
|
|
|
if (startVis) {
|
2021-04-14 01:45:21 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(roundf(aPos), bpBottom - height - 1),
|
|
|
|
ImVec2(roundf(aPos), bpBottom - 1), color);
|
2020-08-05 21:13:53 +02:00
|
|
|
}
|
|
|
|
if (endVis) {
|
2021-04-14 01:45:21 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(roundf(bPos), bpBottom - height - 1),
|
|
|
|
ImVec2(roundf(bPos), bpBottom - 1), color);
|
2020-08-05 21:13:53 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (txtSz.x <= width) {
|
2021-04-14 01:45:21 +02:00
|
|
|
window->DrawList->AddText(ImVec2(cPos - (txtSz.x / 2.0), bpBottom - (height / 2.0f) - (txtSz.y / 2.0f)),
|
2020-08-05 21:13:53 +02:00
|
|
|
IM_COL32(255, 255, 255, 255), bandplan->bands[i].name.c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::updateWaterfallTexture() {
|
|
|
|
glBindTexture(GL_TEXTURE_2D, textureId);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
|
|
|
|
glPixelStorei(GL_UNPACK_ROW_LENGTH, 0);
|
|
|
|
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, dataWidth, waterfallHeight, 0, GL_RGBA, GL_UNSIGNED_BYTE, (uint8_t*)waterfallFb);
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::onPositionChange() {
|
2020-08-04 21:34:56 +02:00
|
|
|
// Nothing to see here...
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::onResize() {
|
2020-08-20 18:29:23 +02:00
|
|
|
// return if widget is too small
|
|
|
|
if (widgetSize.x < 100 || widgetSize.y < 100) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-12-12 05:34:58 +01:00
|
|
|
int lastWaterfallHeight = waterfallHeight;
|
|
|
|
|
2020-08-20 18:29:23 +02:00
|
|
|
if (waterfallVisible) {
|
|
|
|
FFTAreaHeight = std::min<int>(FFTAreaHeight, widgetSize.y - 50);
|
2021-08-04 00:14:55 +02:00
|
|
|
newFFTAreaHeight = FFTAreaHeight;
|
2020-08-20 18:29:23 +02:00
|
|
|
fftHeight = FFTAreaHeight - 50;
|
|
|
|
waterfallHeight = widgetSize.y - fftHeight - 52;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
fftHeight = widgetSize.y - 50;
|
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
dataWidth = widgetSize.x - 60.0;
|
2020-08-20 18:29:23 +02:00
|
|
|
|
2020-12-14 19:33:30 +01:00
|
|
|
if (waterfallVisible) {
|
|
|
|
// Raw FFT resize
|
|
|
|
fftLines = std::min<int>(fftLines, waterfallHeight) - 1;
|
|
|
|
if (rawFFTs != NULL) {
|
|
|
|
if (currentFFTLine != 0) {
|
|
|
|
float* tempWF = new float[currentFFTLine * rawFFTSize];
|
|
|
|
int moveCount = lastWaterfallHeight - currentFFTLine;
|
|
|
|
memcpy(tempWF, rawFFTs, currentFFTLine * rawFFTSize * sizeof(float));
|
|
|
|
memmove(rawFFTs, &rawFFTs[currentFFTLine * rawFFTSize], moveCount * rawFFTSize * sizeof(float));
|
|
|
|
memcpy(&rawFFTs[moveCount * rawFFTSize], tempWF, currentFFTLine * rawFFTSize * sizeof(float));
|
|
|
|
delete[] tempWF;
|
|
|
|
}
|
|
|
|
currentFFTLine = 0;
|
|
|
|
rawFFTs = (float*)realloc(rawFFTs, waterfallHeight * rawFFTSize * sizeof(float));
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
2020-12-14 19:33:30 +01:00
|
|
|
else {
|
|
|
|
rawFFTs = (float*)malloc(waterfallHeight * rawFFTSize * sizeof(float));
|
|
|
|
}
|
|
|
|
// ==============
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
2020-08-17 02:39:56 +02:00
|
|
|
|
2020-12-14 19:33:30 +01:00
|
|
|
if (latestFFT != NULL) {
|
|
|
|
delete[] latestFFT;
|
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
latestFFT = new float[dataWidth];
|
2020-12-14 19:33:30 +01:00
|
|
|
|
2020-08-20 18:29:23 +02:00
|
|
|
if (waterfallVisible) {
|
2020-12-12 05:34:58 +01:00
|
|
|
delete[] waterfallFb;
|
2020-08-20 18:29:23 +02:00
|
|
|
waterfallFb = new uint32_t[dataWidth * waterfallHeight];
|
|
|
|
memset(waterfallFb, 0, dataWidth * waterfallHeight * sizeof(uint32_t));
|
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
for (int i = 0; i < dataWidth; i++) {
|
2020-10-15 16:09:01 +02:00
|
|
|
latestFFT[i] = -1000.0; // Hide everything
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
|
|
|
|
fftAreaMin = ImVec2(widgetPos.x + 50, widgetPos.y + 9);
|
|
|
|
fftAreaMax = ImVec2(widgetPos.x + dataWidth + 50, widgetPos.y + fftHeight + 10);
|
|
|
|
freqAreaMin = ImVec2(widgetPos.x + 50, widgetPos.y + fftHeight + 11);
|
|
|
|
freqAreaMax = ImVec2(widgetPos.x + dataWidth + 50, widgetPos.y + fftHeight + 50);
|
2021-02-09 02:11:40 +01:00
|
|
|
wfMin = ImVec2(widgetPos.x + 50, widgetPos.y + fftHeight + 51);
|
|
|
|
wfMax = ImVec2(widgetPos.x + 50 + dataWidth, widgetPos.y + fftHeight + 51 + waterfallHeight);
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2020-08-16 03:39:05 +02:00
|
|
|
maxHSteps = dataWidth / (ImGui::CalcTextSize("000.000").x + 10);
|
|
|
|
maxVSteps = fftHeight / (ImGui::CalcTextSize("000.000").y);
|
2020-07-19 21:26:37 +02:00
|
|
|
|
|
|
|
range = findBestRange(viewBandwidth, maxHSteps);
|
|
|
|
vRange = findBestRange(fftMax - fftMin, maxVSteps);
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
updateWaterfallFb();
|
2020-08-10 02:30:25 +02:00
|
|
|
updateAllVFOs();
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::draw() {
|
|
|
|
buf_mtx.lock();
|
|
|
|
window = GetCurrentWindow();
|
2020-08-17 02:39:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
widgetPos = ImGui::GetWindowContentRegionMin();
|
2020-08-20 18:29:23 +02:00
|
|
|
widgetEndPos = ImGui::GetWindowContentRegionMax();
|
2020-07-11 21:15:10 +02:00
|
|
|
widgetPos.x += window->Pos.x;
|
|
|
|
widgetPos.y += window->Pos.y;
|
2020-08-17 02:39:56 +02:00
|
|
|
widgetEndPos.x += window->Pos.x - 4; // Padding
|
2020-07-11 21:15:10 +02:00
|
|
|
widgetEndPos.y += window->Pos.y;
|
|
|
|
widgetSize = ImVec2(widgetEndPos.x - widgetPos.x, widgetEndPos.y - widgetPos.y);
|
|
|
|
|
2020-12-08 04:36:37 +01:00
|
|
|
if (selectedVFO == "" && vfos.size() > 0) {
|
|
|
|
selectFirstVFO();
|
|
|
|
}
|
2020-08-17 02:39:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
if (widgetPos.x != lastWidgetPos.x || widgetPos.y != lastWidgetPos.y) {
|
|
|
|
lastWidgetPos = widgetPos;
|
|
|
|
onPositionChange();
|
|
|
|
}
|
|
|
|
if (widgetSize.x != lastWidgetSize.x || widgetSize.y != lastWidgetSize.y) {
|
|
|
|
lastWidgetSize = widgetSize;
|
|
|
|
onResize();
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2021-06-23 21:45:38 +02:00
|
|
|
//window->DrawList->AddRectFilled(widgetPos, widgetEndPos, IM_COL32( 0, 0, 0, 255 ));
|
|
|
|
ImU32 bg = ImGui::ColorConvertFloat4ToU32(gui::themeManager.waterfallBg);
|
|
|
|
window->DrawList->AddRectFilled(widgetPos, widgetEndPos, bg);
|
2020-07-11 21:15:10 +02:00
|
|
|
window->DrawList->AddRect(widgetPos, widgetEndPos, IM_COL32( 50, 50, 50, 255 ));
|
2020-10-15 16:09:01 +02:00
|
|
|
window->DrawList->AddLine(ImVec2(widgetPos.x, widgetPos.y + fftHeight + 50), ImVec2(widgetPos.x + widgetSize.x, widgetPos.y + fftHeight + 50), IM_COL32(50, 50, 50, 255), 1.0);
|
2020-06-10 18:52:07 +02:00
|
|
|
|
2021-07-04 16:41:46 +02:00
|
|
|
if (!gui::mainWindow.lockWaterfallControls) {
|
|
|
|
inputHandled = false;
|
|
|
|
InputHandlerArgs args;
|
|
|
|
args.fftRectMin = fftAreaMin;
|
|
|
|
args.fftRectMax = fftAreaMax;
|
|
|
|
args.freqScaleRectMin = freqAreaMin;
|
|
|
|
args.freqScaleRectMax = freqAreaMax;
|
|
|
|
args.waterfallRectMin = wfMin;
|
|
|
|
args.waterfallRectMax = wfMax;
|
|
|
|
args.lowFreq = lowerFreq;
|
|
|
|
args.highFreq = upperFreq;
|
|
|
|
args.freqToPixelRatio = (double)dataWidth / viewBandwidth;
|
|
|
|
args.pixelToFreqRatio = viewBandwidth / (double)dataWidth;
|
|
|
|
onInputProcess.emit(args);
|
|
|
|
if (!inputHandled) { processInputs(); }
|
|
|
|
}
|
2021-05-04 02:52:59 +02:00
|
|
|
|
|
|
|
updateAllVFOs(true);
|
2020-08-10 02:30:25 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
drawFFT();
|
2020-08-20 18:29:23 +02:00
|
|
|
if (waterfallVisible) {
|
|
|
|
drawWaterfall();
|
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
drawVFOs();
|
2020-09-25 14:25:36 +02:00
|
|
|
if (bandplan != NULL && bandplanVisible) {
|
2020-08-05 21:13:53 +02:00
|
|
|
drawBandPlan();
|
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2020-08-20 18:29:23 +02:00
|
|
|
if (!waterfallVisible) {
|
|
|
|
buf_mtx.unlock();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle fft resize
|
2021-07-12 16:53:59 +02:00
|
|
|
if (!gui::mainWindow.lockWaterfallControls && !inputHandled) {
|
2021-06-23 21:45:38 +02:00
|
|
|
ImVec2 winSize = ImGui::GetWindowSize();
|
|
|
|
ImVec2 mousePos = ImGui::GetMousePos();
|
|
|
|
mousePos.x -= widgetPos.x;
|
|
|
|
mousePos.y -= widgetPos.y;
|
|
|
|
bool click = ImGui::IsMouseClicked(ImGuiMouseButton_Left);
|
|
|
|
bool down = ImGui::IsMouseDown(ImGuiMouseButton_Left);
|
|
|
|
if (draggingFW) {
|
|
|
|
newFFTAreaHeight = mousePos.y;
|
|
|
|
newFFTAreaHeight = std::clamp<float>(newFFTAreaHeight, 150, widgetSize.y - 50);
|
|
|
|
ImGui::GetForegroundDrawList()->AddLine(ImVec2(widgetPos.x, newFFTAreaHeight + widgetPos.y), ImVec2(widgetEndPos.x, newFFTAreaHeight + widgetPos.y),
|
|
|
|
ImGui::GetColorU32(ImGuiCol_SeparatorActive));
|
|
|
|
}
|
|
|
|
if (mousePos.y >= newFFTAreaHeight - 2 && mousePos.y <= newFFTAreaHeight + 2 && mousePos.x > 0 && mousePos.x < widgetSize.x) {
|
|
|
|
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeNS);
|
|
|
|
if (click) {
|
|
|
|
draggingFW = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!down && draggingFW) {
|
|
|
|
draggingFW = false;
|
|
|
|
FFTAreaHeight = newFFTAreaHeight;
|
|
|
|
onResize();
|
2020-08-20 18:29:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
buf_mtx.unlock();
|
|
|
|
}
|
2020-06-10 18:52:07 +02:00
|
|
|
|
2020-12-12 05:34:58 +01:00
|
|
|
float* WaterFall::getFFTBuffer() {
|
|
|
|
if (rawFFTs == NULL) { return NULL; }
|
2020-07-11 21:15:10 +02:00
|
|
|
buf_mtx.lock();
|
2021-01-11 03:19:09 +01:00
|
|
|
if (waterfallVisible) {
|
|
|
|
currentFFTLine--;
|
|
|
|
fftLines++;
|
|
|
|
currentFFTLine = ((currentFFTLine + waterfallHeight) % waterfallHeight);
|
|
|
|
fftLines = std::min<float>(fftLines, waterfallHeight);
|
|
|
|
return &rawFFTs[currentFFTLine * rawFFTSize];
|
|
|
|
}
|
|
|
|
return rawFFTs;
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::pushFFT() {
|
|
|
|
if (rawFFTs == NULL) { return; }
|
2020-10-15 16:09:01 +02:00
|
|
|
double offsetRatio = viewOffset / (wholeBandwidth / 2.0);
|
2020-12-12 05:34:58 +01:00
|
|
|
int drawDataSize = (viewBandwidth / wholeBandwidth) * rawFFTSize;
|
|
|
|
int drawDataStart = (((double)rawFFTSize / 2.0) * (offsetRatio + 1)) - (drawDataSize / 2);
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2021-04-12 23:02:45 +02:00
|
|
|
// If in fast mode, apply IIR filtering
|
|
|
|
float* buf = &rawFFTs[currentFFTLine * rawFFTSize];
|
|
|
|
if (_fastFFT) {
|
|
|
|
float last = buf[0];
|
|
|
|
for (int i = 0; i < rawFFTSize; i++) {
|
|
|
|
last = (buf[i] * 0.1f) + (last * 0.9f);
|
|
|
|
buf[i] = last;
|
|
|
|
}
|
|
|
|
}
|
2020-06-15 15:53:45 +02:00
|
|
|
|
2020-08-20 18:29:23 +02:00
|
|
|
if (waterfallVisible) {
|
2021-04-12 23:02:45 +02:00
|
|
|
doZoom(drawDataStart, drawDataSize, dataWidth, &rawFFTs[currentFFTLine * rawFFTSize], latestFFT, _fastFFT);
|
2020-08-20 18:29:23 +02:00
|
|
|
memmove(&waterfallFb[dataWidth], waterfallFb, dataWidth * (waterfallHeight - 1) * sizeof(uint32_t));
|
|
|
|
float pixel;
|
|
|
|
float dataRange = waterfallMax - waterfallMin;
|
|
|
|
for (int j = 0; j < dataWidth; j++) {
|
|
|
|
pixel = (std::clamp<float>(latestFFT[j], waterfallMin, waterfallMax) - waterfallMin) / dataRange;
|
|
|
|
int id = (int)(pixel * (WATERFALL_RESOLUTION - 1));
|
|
|
|
waterfallFb[j] = waterfallPallet[id];
|
|
|
|
}
|
|
|
|
waterfallUpdate = true;
|
2020-06-15 15:53:45 +02:00
|
|
|
}
|
2021-01-11 03:19:09 +01:00
|
|
|
else {
|
2021-04-12 23:02:45 +02:00
|
|
|
doZoom(drawDataStart, drawDataSize, dataWidth, rawFFTs, latestFFT, _fastFFT);
|
2021-01-11 03:19:09 +01:00
|
|
|
fftLines = 1;
|
|
|
|
}
|
2021-04-30 04:28:08 +02:00
|
|
|
|
|
|
|
if (selectedVFO != "" && vfos.size() > 0) {
|
|
|
|
float dummy;
|
2021-06-29 03:32:40 +02:00
|
|
|
calculateVFOSignalInfo(waterfallVisible ? &rawFFTs[currentFFTLine * rawFFTSize] : rawFFTs, vfos[selectedVFO], dummy, selectedVFOSNR);
|
2021-04-30 04:28:08 +02:00
|
|
|
}
|
2020-08-20 18:29:23 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
buf_mtx.unlock();
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::updatePallette(float colors[][3], int colorCount) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-07-11 21:15:10 +02:00
|
|
|
for (int i = 0; i < WATERFALL_RESOLUTION; i++) {
|
|
|
|
int lowerId = floorf(((float)i / (float)WATERFALL_RESOLUTION) * colorCount);
|
|
|
|
int upperId = ceilf(((float)i / (float)WATERFALL_RESOLUTION) * colorCount);
|
2020-07-20 13:36:16 +02:00
|
|
|
lowerId = std::clamp<int>(lowerId, 0, colorCount - 1);
|
|
|
|
upperId = std::clamp<int>(upperId, 0, colorCount - 1);
|
2020-07-11 21:15:10 +02:00
|
|
|
float ratio = (((float)i / (float)WATERFALL_RESOLUTION) * colorCount) - lowerId;
|
2020-10-15 16:09:01 +02:00
|
|
|
float r = (colors[lowerId][0] * (1.0 - ratio)) + (colors[upperId][0] * (ratio));
|
|
|
|
float g = (colors[lowerId][1] * (1.0 - ratio)) + (colors[upperId][1] * (ratio));
|
|
|
|
float b = (colors[lowerId][2] * (1.0 - ratio)) + (colors[upperId][2] * (ratio));
|
2020-07-11 21:15:10 +02:00
|
|
|
waterfallPallet[i] = ((uint32_t)255 << 24) | ((uint32_t)b << 16) | ((uint32_t)g << 8) | (uint32_t)r;
|
|
|
|
}
|
2020-12-31 14:26:12 +01:00
|
|
|
updateWaterfallFb();
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::updatePalletteFromArray(float* colors, int colorCount) {
|
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
|
|
|
for (int i = 0; i < WATERFALL_RESOLUTION; i++) {
|
|
|
|
int lowerId = floorf(((float)i / (float)WATERFALL_RESOLUTION) * colorCount);
|
|
|
|
int upperId = ceilf(((float)i / (float)WATERFALL_RESOLUTION) * colorCount);
|
|
|
|
lowerId = std::clamp<int>(lowerId, 0, colorCount - 1);
|
|
|
|
upperId = std::clamp<int>(upperId, 0, colorCount - 1);
|
|
|
|
float ratio = (((float)i / (float)WATERFALL_RESOLUTION) * colorCount) - lowerId;
|
|
|
|
float r = (colors[(lowerId * 3) + 0] * (1.0 - ratio)) + (colors[(upperId * 3) + 0] * (ratio));
|
|
|
|
float g = (colors[(lowerId * 3) + 1] * (1.0 - ratio)) + (colors[(upperId * 3) + 1] * (ratio));
|
|
|
|
float b = (colors[(lowerId * 3) + 2] * (1.0 - ratio)) + (colors[(upperId * 3) + 2] * (ratio));
|
|
|
|
waterfallPallet[i] = ((uint32_t)255 << 24) | ((uint32_t)b << 16) | ((uint32_t)g << 8) | (uint32_t)r;
|
|
|
|
}
|
|
|
|
updateWaterfallFb();
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::autoRange() {
|
|
|
|
float min = INFINITY;
|
|
|
|
float max = -INFINITY;
|
|
|
|
for (int i = 0; i < dataWidth; i++) {
|
|
|
|
if (latestFFT[i] < min) {
|
|
|
|
min = latestFFT[i];
|
|
|
|
}
|
|
|
|
if (latestFFT[i] > max) {
|
|
|
|
max = latestFFT[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fftMin = min - 5;
|
|
|
|
fftMax = max + 5;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterFall::setCenterFrequency(double freq) {
|
2020-07-11 21:15:10 +02:00
|
|
|
centerFreq = freq;
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
updateAllVFOs();
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
double WaterFall::getCenterFrequency() {
|
2020-07-11 21:15:10 +02:00
|
|
|
return centerFreq;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterFall::setBandwidth(double bandWidth) {
|
|
|
|
double currentRatio = viewBandwidth / wholeBandwidth;
|
2020-07-11 21:15:10 +02:00
|
|
|
wholeBandwidth = bandWidth;
|
|
|
|
setViewBandwidth(bandWidth * currentRatio);
|
2020-12-05 22:42:12 +01:00
|
|
|
for (auto const& [name, vfo] : vfos) {
|
|
|
|
if (vfo->lowerOffset < -(bandWidth / 2)) {
|
|
|
|
vfo->setCenterOffset(-(bandWidth / 2));
|
|
|
|
}
|
|
|
|
if (vfo->upperOffset > (bandWidth / 2)) {
|
|
|
|
vfo->setCenterOffset(bandWidth / 2);
|
|
|
|
}
|
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
updateAllVFOs();
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
double WaterFall::getBandwidth() {
|
2020-07-11 21:15:10 +02:00
|
|
|
return wholeBandwidth;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterFall::setViewBandwidth(double bandWidth) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-07-11 21:15:10 +02:00
|
|
|
if (bandWidth == viewBandwidth) {
|
|
|
|
return;
|
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
if (abs(viewOffset) + (bandWidth / 2.0) > wholeBandwidth / 2.0) {
|
2020-07-11 21:15:10 +02:00
|
|
|
if (viewOffset < 0) {
|
2020-10-15 16:09:01 +02:00
|
|
|
viewOffset = (bandWidth / 2.0) - (wholeBandwidth / 2.0);
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
else {
|
2020-10-15 16:09:01 +02:00
|
|
|
viewOffset = (wholeBandwidth / 2.0) - (bandWidth / 2.0);
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
viewBandwidth = bandWidth;
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
2020-07-19 21:26:37 +02:00
|
|
|
range = findBestRange(bandWidth, maxHSteps);
|
2021-04-13 04:54:47 +02:00
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
2020-08-10 02:30:25 +02:00
|
|
|
updateAllVFOs();
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
double WaterFall::getViewBandwidth() {
|
2020-07-19 15:59:44 +02:00
|
|
|
return viewBandwidth;
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterFall::setViewOffset(double offset) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-07-11 21:15:10 +02:00
|
|
|
if (offset == viewOffset) {
|
|
|
|
return;
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
if (offset - (viewBandwidth / 2.0) < -(wholeBandwidth / 2.0)) {
|
|
|
|
offset = (viewBandwidth / 2.0) - (wholeBandwidth / 2.0);
|
2020-07-19 15:59:44 +02:00
|
|
|
}
|
2020-10-15 16:09:01 +02:00
|
|
|
if (offset + (viewBandwidth / 2.0) > (wholeBandwidth / 2.0)) {
|
|
|
|
offset = (wholeBandwidth / 2.0) - (viewBandwidth / 2.0);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
|
|
|
viewOffset = offset;
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerFreq = (centerFreq + viewOffset) - (viewBandwidth / 2.0);
|
|
|
|
upperFreq = (centerFreq + viewOffset) + (viewBandwidth / 2.0);
|
2021-04-13 04:54:47 +02:00
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
2020-08-10 02:30:25 +02:00
|
|
|
updateAllVFOs();
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-07-19 15:59:44 +02:00
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
double WaterFall::getViewOffset() {
|
2020-07-19 15:59:44 +02:00
|
|
|
return viewOffset;
|
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
|
|
|
|
void WaterFall::setFFTMin(float min) {
|
|
|
|
fftMin = min;
|
2020-07-19 21:26:37 +02:00
|
|
|
vRange = findBestRange(fftMax - fftMin, maxVSteps);
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
float WaterFall::getFFTMin() {
|
|
|
|
return fftMin;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::setFFTMax(float max) {
|
|
|
|
fftMax = max;
|
2020-07-19 21:26:37 +02:00
|
|
|
vRange = findBestRange(fftMax - fftMin, maxVSteps);
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
float WaterFall::getFFTMax() {
|
|
|
|
return fftMax;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2021-04-13 04:54:47 +02:00
|
|
|
void WaterFall::setFullWaterfallUpdate(bool fullUpdate) {
|
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
|
|
|
_fullUpdate = fullUpdate;
|
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
void WaterFall::setWaterfallMin(float min) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-07-11 21:15:10 +02:00
|
|
|
if (min == waterfallMin) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
waterfallMin = min;
|
2021-04-13 04:54:47 +02:00
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
2020-07-11 21:15:10 +02:00
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
float WaterFall::getWaterfallMin() {
|
|
|
|
return waterfallMin;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::setWaterfallMax(float max) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-07-11 21:15:10 +02:00
|
|
|
if (max == waterfallMax) {
|
|
|
|
return;
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
2020-07-11 21:15:10 +02:00
|
|
|
waterfallMax = max;
|
2021-04-13 04:54:47 +02:00
|
|
|
if (_fullUpdate) { updateWaterfallFb(); };
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
|
|
|
|
2020-07-11 21:15:10 +02:00
|
|
|
float WaterFall::getWaterfallMax() {
|
|
|
|
return waterfallMax;
|
2020-06-10 04:13:56 +02:00
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
|
2021-05-04 02:52:59 +02:00
|
|
|
void WaterFall::updateAllVFOs(bool checkRedrawRequired) {
|
2020-08-10 02:30:25 +02:00
|
|
|
for (auto const& [name, vfo] : vfos) {
|
2021-05-04 02:52:59 +02:00
|
|
|
if (checkRedrawRequired && !vfo->redrawRequired) { continue; }
|
2020-08-10 02:30:25 +02:00
|
|
|
vfo->updateDrawingVars(viewBandwidth, dataWidth, viewOffset, widgetPos, fftHeight);
|
2021-02-10 21:35:56 +01:00
|
|
|
vfo->wfRectMin = ImVec2(vfo->rectMin.x, wfMin.y);
|
|
|
|
vfo->wfRectMax = ImVec2(vfo->rectMax.x, wfMax.y);
|
|
|
|
vfo->wfLineMin = ImVec2(vfo->lineMin.x, wfMin.y);
|
|
|
|
vfo->wfLineMax = ImVec2(vfo->lineMax.x, wfMax.y);
|
2021-04-17 22:37:50 +02:00
|
|
|
vfo->wfLbwSelMin = ImVec2(vfo->wfRectMin.x - 2, vfo->wfRectMin.y);
|
|
|
|
vfo->wfLbwSelMax = ImVec2(vfo->wfRectMin.x + 2, vfo->wfRectMax.y);
|
|
|
|
vfo->wfRbwSelMin = ImVec2(vfo->wfRectMax.x - 2, vfo->wfRectMin.y);
|
|
|
|
vfo->wfRbwSelMax = ImVec2(vfo->wfRectMax.x + 2, vfo->wfRectMax.y);
|
2021-07-04 16:41:46 +02:00
|
|
|
vfo->redrawRequired = false;
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
}
|
2020-12-12 05:34:58 +01:00
|
|
|
|
|
|
|
void WaterFall::setRawFFTSize(int size, bool lock) {
|
2020-12-31 14:26:12 +01:00
|
|
|
std::lock_guard<std::mutex> lck(buf_mtx);
|
2020-12-12 05:34:58 +01:00
|
|
|
rawFFTSize = size;
|
2021-07-28 22:53:38 +02:00
|
|
|
int wfSize = std::max<int>(1, waterfallHeight);
|
2020-12-12 05:34:58 +01:00
|
|
|
if (rawFFTs != NULL) {
|
2021-01-11 03:19:09 +01:00
|
|
|
rawFFTs = (float*)realloc(rawFFTs, rawFFTSize * wfSize * sizeof(float));
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
|
|
|
else {
|
2021-01-11 03:19:09 +01:00
|
|
|
rawFFTs = (float*)malloc(rawFFTSize * wfSize * sizeof(float));
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
2021-07-28 22:53:38 +02:00
|
|
|
fftLines = 0;
|
2020-12-12 05:34:58 +01:00
|
|
|
memset(rawFFTs, 0, rawFFTSize * waterfallHeight * sizeof(float));
|
2021-07-28 22:53:38 +02:00
|
|
|
updateWaterfallFb();
|
2020-12-12 05:34:58 +01:00
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
|
2021-04-14 01:45:21 +02:00
|
|
|
void WaterFall::setBandPlanPos(int pos) {
|
|
|
|
bandPlanPos = pos;
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterfallVFO::setOffset(double offset) {
|
2020-08-10 02:30:25 +02:00
|
|
|
generalOffset = offset;
|
|
|
|
if (reference == REF_CENTER) {
|
|
|
|
centerOffset = offset;
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerOffset = offset - (bandwidth / 2.0);
|
|
|
|
upperOffset = offset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
else if (reference == REF_LOWER) {
|
|
|
|
lowerOffset = offset;
|
2020-10-15 16:09:01 +02:00
|
|
|
centerOffset = offset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
upperOffset = offset + bandwidth;
|
|
|
|
}
|
|
|
|
else if (reference == REF_UPPER) {
|
|
|
|
upperOffset = offset;
|
2020-10-15 16:09:01 +02:00
|
|
|
centerOffset = offset - (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
lowerOffset = offset - bandwidth;
|
|
|
|
}
|
|
|
|
centerOffsetChanged = true;
|
|
|
|
upperOffsetChanged = true;
|
|
|
|
lowerOffsetChanged = true;
|
|
|
|
redrawRequired = true;
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterfallVFO::setCenterOffset(double offset) {
|
2020-08-10 02:30:25 +02:00
|
|
|
if (reference == REF_CENTER) {
|
|
|
|
generalOffset = offset;
|
|
|
|
}
|
|
|
|
else if (reference == REF_LOWER) {
|
2020-10-15 16:09:01 +02:00
|
|
|
generalOffset = offset - (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
else if (reference == REF_UPPER) {
|
2020-10-15 16:09:01 +02:00
|
|
|
generalOffset = offset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
centerOffset = offset;
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerOffset = offset - (bandwidth / 2.0);
|
|
|
|
upperOffset = offset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
centerOffsetChanged = true;
|
|
|
|
upperOffsetChanged = true;
|
|
|
|
lowerOffsetChanged = true;
|
|
|
|
redrawRequired = true;
|
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterfallVFO::setBandwidth(double bw) {
|
2020-08-10 02:30:25 +02:00
|
|
|
if (bandwidth == bw || bw < 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bandwidth = bw;
|
|
|
|
if (reference == REF_CENTER) {
|
2020-10-15 16:09:01 +02:00
|
|
|
lowerOffset = centerOffset - (bandwidth / 2.0);
|
|
|
|
upperOffset = centerOffset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
else if (reference == REF_LOWER) {
|
2020-10-15 16:09:01 +02:00
|
|
|
centerOffset = lowerOffset + (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
upperOffset = lowerOffset + bandwidth;
|
2020-08-21 15:34:50 +02:00
|
|
|
centerOffsetChanged = true;
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
else if (reference == REF_UPPER) {
|
2020-10-15 16:09:01 +02:00
|
|
|
centerOffset = upperOffset - (bandwidth / 2.0);
|
2020-08-10 02:30:25 +02:00
|
|
|
lowerOffset = upperOffset - bandwidth;
|
2020-08-21 15:34:50 +02:00
|
|
|
centerOffsetChanged = true;
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
bandwidthChanged = true;
|
2020-08-10 02:30:25 +02:00
|
|
|
redrawRequired = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaterfallVFO::setReference(int ref) {
|
|
|
|
if (reference == ref || ref < 0 || ref >= _REF_COUNT) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
reference = ref;
|
2020-08-12 16:43:44 +02:00
|
|
|
setOffset(generalOffset);
|
|
|
|
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
|
2020-10-15 16:09:01 +02:00
|
|
|
void WaterfallVFO::updateDrawingVars(double viewBandwidth, float dataWidth, double viewOffset, ImVec2 widgetPos, int fftHeight) {
|
|
|
|
double width = (bandwidth / viewBandwidth) * (double)dataWidth;
|
|
|
|
int center = roundf((((centerOffset - viewOffset) / (viewBandwidth / 2.0)) + 1.0) * ((double)dataWidth / 2.0));
|
|
|
|
int left = roundf((((lowerOffset - viewOffset) / (viewBandwidth / 2.0)) + 1.0) * ((double)dataWidth / 2.0));
|
|
|
|
int right = roundf((((upperOffset - viewOffset) / (viewBandwidth / 2.0)) + 1.0) * ((double)dataWidth / 2.0));
|
2020-08-10 02:30:25 +02:00
|
|
|
|
2021-07-04 16:41:46 +02:00
|
|
|
// Check weather the line is visible
|
2020-08-10 02:30:25 +02:00
|
|
|
if (left >= 0 && left < dataWidth && reference == REF_LOWER) {
|
|
|
|
lineVisible = true;
|
|
|
|
}
|
|
|
|
else if (center >= 0 && center < dataWidth && reference == REF_CENTER) {
|
|
|
|
lineVisible = true;
|
|
|
|
}
|
|
|
|
else if (right >= 0 && right < dataWidth && reference == REF_UPPER) {
|
|
|
|
lineVisible = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
lineVisible = false;
|
|
|
|
}
|
|
|
|
|
2021-07-04 16:41:46 +02:00
|
|
|
// Calculate the position of the line
|
|
|
|
if (reference == REF_LOWER) {
|
|
|
|
lineMin = ImVec2(widgetPos.x + 50 + left, widgetPos.y + 9);
|
|
|
|
lineMax = ImVec2(widgetPos.x + 50 + left, widgetPos.y + fftHeight + 9);
|
|
|
|
}
|
|
|
|
else if (reference == REF_CENTER) {
|
|
|
|
lineMin = ImVec2(widgetPos.x + 50 + center, widgetPos.y + 9);
|
|
|
|
lineMax = ImVec2(widgetPos.x + 50 + center, widgetPos.y + fftHeight + 9);
|
|
|
|
}
|
|
|
|
else if (reference == REF_UPPER) {
|
|
|
|
lineMin = ImVec2(widgetPos.x + 50 + right, widgetPos.y + 9);
|
|
|
|
lineMax = ImVec2(widgetPos.x + 50 + right, widgetPos.y + fftHeight + 9);
|
|
|
|
}
|
|
|
|
|
|
|
|
int _left = left;
|
|
|
|
int _right = right;
|
2020-08-10 02:30:25 +02:00
|
|
|
left = std::clamp<int>(left, 0, dataWidth - 1);
|
|
|
|
right = std::clamp<int>(right, 0, dataWidth - 1);
|
2021-07-10 21:15:20 +02:00
|
|
|
leftClamped = (left != _left);
|
|
|
|
rightClamped = (right != _right);
|
2020-08-10 02:30:25 +02:00
|
|
|
|
|
|
|
rectMin = ImVec2(widgetPos.x + 50 + left, widgetPos.y + 10);
|
|
|
|
rectMax = ImVec2(widgetPos.x + 51 + right, widgetPos.y + fftHeight + 10);
|
2021-04-17 03:38:48 +02:00
|
|
|
|
2021-04-17 22:37:50 +02:00
|
|
|
lbwSelMin = ImVec2(rectMin.x - 2, rectMin.y);
|
|
|
|
lbwSelMax = ImVec2(rectMin.x + 2, rectMax.y);
|
|
|
|
rbwSelMin = ImVec2(rectMax.x - 2, rectMin.y);
|
|
|
|
rbwSelMax = ImVec2(rectMax.x + 2, rectMax.y);
|
2020-08-10 02:30:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterfallVFO::draw(ImGuiWindow* window, bool selected) {
|
2021-05-04 20:41:23 +02:00
|
|
|
window->DrawList->AddRectFilled(rectMin, rectMax, color);
|
2020-08-10 02:30:25 +02:00
|
|
|
if (lineVisible) {
|
|
|
|
window->DrawList->AddLine(lineMin, lineMax, selected ? IM_COL32(255, 0, 0, 255) : IM_COL32(255, 255, 0, 255));
|
|
|
|
}
|
2021-04-17 22:37:50 +02:00
|
|
|
|
2021-07-12 16:53:59 +02:00
|
|
|
if (!gui::mainWindow.lockWaterfallControls && !gui::waterfall.inputHandled) {
|
2021-06-23 21:45:38 +02:00
|
|
|
ImVec2 mousePos = ImGui::GetMousePos();
|
2021-07-31 20:17:36 +02:00
|
|
|
if (rectMax.x - rectMin.x < 10) { return; }
|
2021-07-04 16:41:46 +02:00
|
|
|
if (reference != REF_LOWER && !bandwidthLocked && !leftClamped) {
|
2021-06-23 21:45:38 +02:00
|
|
|
if (IS_IN_AREA(mousePos, lbwSelMin, lbwSelMax)) { ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW); }
|
|
|
|
else if (IS_IN_AREA(mousePos, wfLbwSelMin, wfLbwSelMax)) { ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW); }
|
|
|
|
}
|
2021-07-04 16:41:46 +02:00
|
|
|
if (reference != REF_UPPER && !bandwidthLocked && !rightClamped) {
|
2021-06-23 21:45:38 +02:00
|
|
|
if (IS_IN_AREA(mousePos, rbwSelMin, rbwSelMax)) { ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW); }
|
|
|
|
else if (IS_IN_AREA(mousePos, wfRbwSelMin, wfRbwSelMax)) { ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeEW); }
|
|
|
|
}
|
2021-05-04 20:41:23 +02:00
|
|
|
}
|
2020-08-10 02:30:25 +02:00
|
|
|
};
|
2020-08-20 18:29:23 +02:00
|
|
|
|
|
|
|
void WaterFall::showWaterfall() {
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.lock();
|
2021-01-11 03:19:09 +01:00
|
|
|
waterfallVisible = true;
|
2020-08-20 18:29:23 +02:00
|
|
|
onResize();
|
2021-01-11 03:19:09 +01:00
|
|
|
memset(rawFFTs, 0, waterfallHeight * rawFFTSize * sizeof(float));
|
|
|
|
updateWaterfallFb();
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.unlock();
|
2020-08-20 18:29:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::hideWaterfall() {
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.lock();
|
2021-01-11 03:19:09 +01:00
|
|
|
waterfallVisible = false;
|
2020-08-20 18:29:23 +02:00
|
|
|
onResize();
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.unlock();
|
2020-08-20 18:29:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::setFFTHeight(int height) {
|
|
|
|
FFTAreaHeight = height;
|
2020-08-21 15:34:50 +02:00
|
|
|
newFFTAreaHeight = height;
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.lock();
|
2020-08-20 18:29:23 +02:00
|
|
|
onResize();
|
2020-12-14 19:33:30 +01:00
|
|
|
buf_mtx.unlock();
|
2020-08-20 18:29:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
int WaterFall::getFFTHeight() {
|
|
|
|
return FFTAreaHeight;
|
|
|
|
}
|
2020-09-25 14:25:36 +02:00
|
|
|
|
|
|
|
void WaterFall::showBandplan() {
|
|
|
|
bandplanVisible = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WaterFall::hideBandplan() {
|
|
|
|
bandplanVisible = false;
|
|
|
|
}
|
2020-10-20 14:59:42 +02:00
|
|
|
|
|
|
|
void WaterfallVFO::setSnapInterval(double interval) {
|
|
|
|
snapInterval = interval;
|
|
|
|
}
|
2020-06-10 04:13:56 +02:00
|
|
|
};
|
|
|
|
|