Fixed vfo selection issue

This commit is contained in:
Ryzerth 2021-05-18 02:26:55 +02:00
parent c59b83e564
commit 55017f876d
5 changed files with 259 additions and 648 deletions

View File

@ -24,7 +24,8 @@ option(OPT_BUILD_FALCON9_DECODER "Build the falcon9 live decoder (Dependencies:
option(OPT_BUILD_METEOR_DEMODULATOR "Build the meteor demodulator module (no dependencies required)" ON) option(OPT_BUILD_METEOR_DEMODULATOR "Build the meteor demodulator module (no dependencies required)" ON)
option(OPT_BUILD_WEATHER_SAT_DECODER "Build the HRPT decoder module (no dependencies required)" ON) option(OPT_BUILD_WEATHER_SAT_DECODER "Build the HRPT decoder module (no dependencies required)" ON)
option(OPT_BUILD_DISCORD_PRESENCE "Build the Discord Rich Presence module" ON) option(OPT_BUILD_DISCORD_PRESENCE "Build the Discord Rich Presence module" ON)
option(OPT_BUILD_FREQUENCY_MANAGER "Build the Frequency Manager module" OFF) option(OPT_BUILD_FREQUENCY_MANAGER "Build the Frequency Manager module" ON)
# Core of SDR++ # Core of SDR++
add_subdirectory("core") add_subdirectory("core")

View File

@ -241,6 +241,18 @@ namespace ImGui {
int mouseWheel = ImGui::GetIO().MouseWheel; int mouseWheel = ImGui::GetIO().MouseWheel;
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;
}
}
// Deselect everything if the mouse is released // Deselect everything if the mouse is released
if (!ImGui::IsMouseDown(ImGuiMouseButton_Left)) { if (!ImGui::IsMouseDown(ImGuiMouseButton_Left)) {
freqScaleSelect = false; freqScaleSelect = false;
@ -274,17 +286,12 @@ namespace ImGui {
} }
// Next, check if a VFO was selected // Next, check if a VFO was selected
if (!targetFound) { if (!targetFound && hoveredVFOName != "") {
for (auto const& [name, _vfo] : vfos) { selectedVFO = hoveredVFOName;
// If another VFO is selected, select it and cancel out
if (IS_IN_AREA(mousePos, _vfo->rectMin, _vfo->rectMax) || IS_IN_AREA(mousePos, _vfo->wfRectMin, _vfo->wfRectMax)) {
selectedVFO = name;
selectedVFOChanged = true; selectedVFOChanged = true;
targetFound = true; targetFound = true;
return; return;
} }
}
}
// Now, check frequency scale // Now, check frequency scale
if (!targetFound && mouseInFreq) { if (!targetFound && mouseInFreq) {
@ -393,7 +400,7 @@ namespace ImGui {
} }
// Finally, if nothing else was selected, just move the VFO // Finally, if nothing else was selected, just move the VFO
if (ImGui::IsMouseDown(ImGuiMouseButton_Left) && (mouseInFFT|mouseInWaterfall)) { if (ImGui::IsMouseDown(ImGuiMouseButton_Left) && (mouseInFFT|mouseInWaterfall) && (mouseMoved || hoveredVFOName == "")) {
if (selVfo != NULL) { if (selVfo != NULL) {
int refCenter = mousePos.x - (widgetPos.x + 50); int refCenter = mousePos.x - (widgetPos.x + 50);
if (refCenter >= 0 && refCenter < dataWidth) { if (refCenter >= 0 && refCenter < dataWidth) {
@ -404,7 +411,7 @@ namespace ImGui {
} }
} }
} }
else { else if (!ImGui::IsMouseDown(ImGuiMouseButton_Left)) {
// Check if a VFO is hovered. If yes, show tooltip // Check if a VFO is hovered. If yes, show tooltip
for (auto const& [name, _vfo] : vfos) { for (auto const& [name, _vfo] : vfos) {
if (ImGui::IsMouseHoveringRect(_vfo->rectMin, _vfo->rectMax) || ImGui::IsMouseHoveringRect(_vfo->wfRectMin, _vfo->wfRectMax)) { if (ImGui::IsMouseHoveringRect(_vfo->rectMin, _vfo->rectMax) || ImGui::IsMouseHoveringRect(_vfo->wfRectMin, _vfo->wfRectMax)) {

View File

@ -249,5 +249,7 @@ namespace ImGui {
bool vfoBorderSelect = false; bool vfoBorderSelect = false;
WaterfallVFO* relatedVfo = NULL; WaterfallVFO* relatedVfo = NULL;
ImVec2 mouseDownPos; ImVec2 mouseDownPos;
ImVec2 lastMousePos;
}; };
}; };

View File

@ -9,12 +9,12 @@ else ()
set(CMAKE_CXX_FLAGS "-O3 -std=c++17") set(CMAKE_CXX_FLAGS "-O3 -std=c++17")
endif () endif ()
include_directories("src/" "../radio/src")
file(GLOB SRC "src/*.cpp") file(GLOB SRC "src/*.cpp")
include_directories("src/" "../radio/src" "../core/src" "../core/src/gui" "../core/src/imgui")
add_library(frequency_manager SHARED ${SRC}) add_library(frequency_manager SHARED ${SRC})
target_link_libraries(frequency_manager PUBLIC sdrpp_core) target_link_libraries(frequency_manager PRIVATE sdrpp_core)
set_target_properties(frequency_manager PROPERTIES PREFIX "") set_target_properties(frequency_manager PROPERTIES PREFIX "")
# Install directives # Install directives

View File

@ -1,59 +1,39 @@
#include <core.h> #include <imgui.h>
#include <spdlog/spdlog.h>
#include <module.h>
#include <gui/gui.h> #include <gui/gui.h>
#include <gui/style.h> #include <gui/style.h>
#include <imgui.h> #include <core.h>
#include <iomanip> #include <thread>
#include <module.h>
#include <options.h>
#include <radio_interface.h> #include <radio_interface.h>
#include <signal_path/signal_path.h> #include <signal_path/signal_path.h>
#include <spdlog/spdlog.h> #include <vector>
#include <sstream>
#ifdef _WIN32
#define strcpy strcpy_s
#endif
SDRPP_MOD_INFO { SDRPP_MOD_INFO {
/* Name: */ "frequency_manager", /* Name: */ "frequency_manager",
/* Description: */ "Basic frequency manager module for SDR++", /* Description: */ "Frequency manager module for SDR++",
/* Author: */ "zimm", /* Author: */ "Ryzerth;zimm",
/* Version: */ 0, 0, 1, /* Version: */ 0, 3, 0,
/* Max instances */ 1}; /* Max instances */ -1
};
static ConfigManager config; struct FrequencyBookmark {
double frequency;
double bandwidth;
int mode;
bool selected;
};
class FreqManModule : public ModuleManager::Instance { class FrequencyManagerModule : public ModuleManager::Instance {
public: public:
FreqManModule(std::string name) { FrequencyManagerModule(std::string name) {
this->name = name; this->name = name;
CheckConfigFileIntegrity(); gui::menu.registerEntry(name, menuHandler, this, NULL);
config.aquire();
if (config.conf.size() > 0) {
for (const auto &item : config.conf.items()) {
bookmarks.push_back((std::string)item.key());
if (config.conf[item.key()].contains("category"))
if (!VectorArrayContainsString(categories, (std::string)config.conf[item.key()]["category"]))
categories.push_back((std::string)config.conf[item.key()]["category"]);
} }
std::sort(categories.begin(), categories.end()); ~FrequencyManagerModule() {
selected_category = categories[0]; gui::menu.removeEntry(name);
for (const auto &i : bookmarks) {
if (config.conf[i].contains("category") && ((std::string)config.conf[i]["category"] == selected_category))
displayed_bookmarks.push_back(i);
}
}
config.release();
std::sort(displayed_bookmarks.begin(), displayed_bookmarks.end());
// add collapsable header
gui::menu.registerEntry(name, menuHandler, this);
}
~FreqManModule() {
} }
void enable() { void enable() {
@ -69,564 +49,185 @@ class FreqManModule : public ModuleManager::Instance {
} }
private: private:
static void CheckConfigFileIntegrity() { static std::string freqToStr(double freq) {
config.aquire(); char str[128];
bool modified = false; if (freq >= 1000000.0) {
if (config.conf.size() > 0) { sprintf(str, "%.06lf", freq / 1000000.0);
for (const auto &item : config.conf.items()) { int len = strlen(str) - 1;
if ((!config.conf[item.key()].contains("category") || !config.conf[item.key()]["category"].is_string()) || while ((str[len] == '0' || str[len] == '.') && len > 0) { len--; }
(!config.conf[item.key()].contains("frequency") || !config.conf[item.key()]["frequency"].is_number()) || return std::string(str).substr(0, len + 1) + "MHz";
(!config.conf[item.key()].contains("mode") || !config.conf[item.key()]["mode"].is_string()) ||
(!config.conf[item.key()].contains("bandwidth") || !config.conf[item.key()]["bandwidth"].is_number())) {
spdlog::warn("Frequency Manager: Invalid entry found in config file, removing '{0}'.", (std::string)item.key());
config.conf.erase(item.key());
modified = true;
} }
else if (freq >= 1000.0) {
sprintf(str, "%.06lf", freq / 1000.0);
int len = strlen(str) - 1;
while ((str[len] == '0' || str[len] == '.') && len > 0) { len--; }
return std::string(str).substr(0, len + 1) + "KHz";
} }
else {
sprintf(str, "%.06lf", freq);
int len = strlen(str) - 1;
while ((str[len] == '0' || str[len] == '.') && len > 0) { len--; }
return std::string(str).substr(0, len + 1) + "Hz";
} }
config.release(modified);
} }
static std::string GetActiveVFOName() { static void applyBookmark(FrequencyBookmark bm, std::string vfoName) {
return gui::waterfall.selectedVFO; if (vfoName == "") {
// TODO: Replace with proper tune call
gui::waterfall.setCenterFrequency(bm.frequency);
gui::waterfall.centerFreqMoved = true;
}
else {
}
} }
static void GetActiveVFOFrequency(double *frequency) { bool bookmarkEditDialog(FrequencyBookmark& bm) {
*frequency = (double)gui::freqSelect.frequency; bool open = true;
std::string id = "Edit##freq_manager_edit_popup_" + name;
ImGui::OpenPopup(id.c_str());
FrequencyBookmark tmp = bm;
if (ImGui::BeginPopup(id.c_str(), ImGuiWindowFlags_NoResize)) {
if (ImGui::Button("Apply")) {
bm = tmp;
open = false;
} }
ImGui::SameLine();
static std::string GetActiveVFOFrequencyShort(double freq) { if (ImGui::Button("Cancel")) {
std::string ret_val = ""; open = false;
std::stringstream stream;
if (freq < 1000000) {
stream << std::fixed << std::setprecision(3) << freq / 1000.0;
ret_val = stream.str();
ret_val += " kHz";
} else if (freq >= 1000000000) {
stream << std::fixed << std::setprecision(3) << freq / 1000000000.0;
ret_val = stream.str();
ret_val += " GHz";
} else {
stream << std::fixed << std::setprecision(3) << freq / 1000000.0;
ret_val = stream.str();
ret_val += " MHz";
} }
ImGui::EndPopup();
return ret_val;
} }
return open;
static void GetActiveVFOBandwidth(int *bw) {
if (GetActiveVFOName() != "")
*bw = (int)sigpath::vfoManager.getBandwidth(GetActiveVFOName());
else
*bw = 0;
}
static std::string GetActiveVFODemod() {
std::string demod;
std::string vfo = GetActiveVFOName();
if (core::modComManager.interfaceExists(vfo)) {
int modeNum;
core::modComManager.callInterface(vfo, RADIO_IFACE_CMD_GET_MODE, NULL, &modeNum);
switch (modeNum) {
case RADIO_IFACE_MODE_NFM:
demod = "NFM";
break;
case RADIO_IFACE_MODE_WFM:
demod = "WFM";
break;
case RADIO_IFACE_MODE_AM:
demod = "AM";
break;
case RADIO_IFACE_MODE_DSB:
demod = "DSB";
break;
case RADIO_IFACE_MODE_LSB:
demod = "LSB";
break;
case RADIO_IFACE_MODE_USB:
demod = "USB";
break;
case RADIO_IFACE_MODE_CW:
demod = "CW";
break;
case RADIO_IFACE_MODE_RAW:
demod = "RAW";
break;
default:
demod = "N/A";
break;
}
}
return demod;
}
static void SaveBookmark(std::string description, std::string category, double frequency, int bandwidth, std::string demod) {
config.aquire();
config.conf[description] = json({});
config.conf[description]["category"] = category;
config.conf[description]["frequency"] = frequency;
config.conf[description]["bandwidth"] = bandwidth;
config.conf[description]["mode"] = demod;
config.release(true);
}
static void EditBookmark(std::string description, std::string category, double frequency, int bandwidth) {
config.aquire();
config.conf[description]["category"] = category;
config.conf[description]["frequency"] = frequency;
config.conf[description]["bandwidth"] = bandwidth;
config.release(true);
}
static void DeleteBookmark(std::string description) {
config.aquire();
if (config.conf.contains(description))
config.conf.erase(description);
config.release(true);
}
static bool VectorArrayContainsString(std::vector<std::string> &v, std::string item) {
auto it = std::find(v.begin(), v.end(), item);
if (it != v.end())
return true;
else
return false;
}
static void VectorArrayDeleteElement(std::vector<std::string> &v, std::string item) {
auto itr = std::find(v.begin(), v.end(), item);
if (itr != v.end())
v.erase(itr);
} }
static void menuHandler(void* ctx) { static void menuHandler(void* ctx) {
FreqManModule *_this = (FreqManModule *)ctx; FrequencyManagerModule* _this = (FrequencyManagerModule*)ctx;
float menuWidth = ImGui::GetContentRegionAvailWidth(); float menuWidth = ImGui::GetContentRegionAvailWidth();
ImGuiStyle &style = ImGui::GetStyle();
ImGui::SetNextItemWidth(menuWidth);
static int vfoBW; // TODO: Replace with something that won't iterate every frame
static double vfoFreq; std::vector<std::string> selectedNames;
static char description[128] = ""; for (auto& [name, bm] : _this->bookmarks) { if (bm.selected) { selectedNames.push_back(name); } }
static char category[128] = "";
std::string vfoName = GetActiveVFOName();
std::string vfoDemod = GetActiveVFODemod();
static std::string original_description = "";
static std::string original_category = "";
// categories dropdown //Draw buttons on top of the list
static int current_category_index; ImGui::BeginTable(("freq_manager_btn_table"+_this->name).c_str(), 3);
std::string categories_dropdown_list = "";
for (int i = 0; i < _this->categories.size(); i++) {
categories_dropdown_list += _this->categories[i];
categories_dropdown_list += '\0';
}
ImGui::AlignTextToFramePadding();
ImGui::Text("Category: ");
ImGui::SameLine();
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
bool disable_ctl = false;
if (_this->selected_category.size() == 0) {
style::beginDisabled();
disable_ctl = true;
}
if (ImGui::Combo("##_sdrpp_freq_mgr_category", &current_category_index, categories_dropdown_list.c_str())) {
_this->selected_category = _this->categories[current_category_index];
_this->displayed_bookmarks.clear();
_this->selected_bookmarks.clear();
for (const auto &i : _this->bookmarks) {
if (config.conf[i].contains("category") && ((std::string)config.conf[i]["category"] == _this->selected_category))
_this->displayed_bookmarks.push_back(i);
}
}
if (disable_ctl) {
style::endDisabled();
disable_ctl = false;
}
// VFO dropdown
const char *vfos[] = {"Selected"};
static int current_vfo = 0;
ImGui::AlignTextToFramePadding();
style::beginDisabled();
ImGui::Text("VFO: ");
ImGui::SameLine(ImGui::CalcTextSize("Category: ").x + style.ItemSpacing.x);
ImGui::SetNextItemWidth(menuWidth - ImGui::GetCursorPosX());
ImGui::Combo("##_sdrpp_freq_mgr_vfo", &current_vfo, vfos, IM_ARRAYSIZE(vfos));
style::endDisabled();
// controls buttons
bool unused_open = true;
ImVec2 button_sz((menuWidth / 3 - style.ItemSpacing.x / 3 - 2), 24);
ImGui::AlignTextToFramePadding();
// add new bookmark button
if (vfoName.size() == 0) {
style::beginDisabled();
disable_ctl = true;
}
if (ImGui::Button("Add", button_sz)) {
ImGui::OpenPopup("Add new bookmark");
GetActiveVFOBandwidth(&vfoBW);
GetActiveVFOFrequency(&vfoFreq);
strcpy(description, (GetActiveVFOFrequencyShort(vfoFreq) + " " + vfoDemod).c_str());
}
if (disable_ctl) {
style::endDisabled();
disable_ctl = false;
}
ImGui::SetNextWindowPos(ImVec2(2, 300));
if (ImGui::BeginPopupModal("Add new bookmark", &unused_open, ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoSavedSettings)) {
ImGui::SetWindowSize("Add new bookmark", ImVec2(330, 210));
ImGui::Text("Category: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
if (ImGui::InputTextWithHint("##_sdrpp_freq_mgr_add_history", "General", category, IM_ARRAYSIZE(category), ImGuiInputTextFlags_CallbackHistory, HistoryHandling::CategoryHistoryCallback)) {
}
ImGui::SameLine();
ImGui::TextDisabled(" ?");
if (ImGui::IsItemHovered())
ImGui::SetTooltip("If a category is not provided, \"General\" will be used.\nStart typing and use UP/DOWN keys to cycle through history.");
if (strlen(description) == 0 || VectorArrayContainsString(_this->bookmarks, (std::string)description)) {
ImGui::PushStyleColor(ImGuiCol_Text, (ImVec4)ImColor::HSV(254.0f, 255.0f, 210.0f));
disable_ctl = true;
}
ImGui::Text("Description: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
if (disable_ctl) {
ImGui::PopStyleColor();
disable_ctl = false;
}
if (ImGui::InputText("##_sdrpp_freq_mgr_add_descr", description, IM_ARRAYSIZE(description), ImGuiInputTextFlags_AutoSelectAll)) {
}
if (strlen(description) == 0) {
ImGui::SameLine();
ImGui::TextDisabled(" ?");
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Description cannot be empty.");
}
if (VectorArrayContainsString(_this->bookmarks, (std::string)description)) {
ImGui::SameLine();
ImGui::TextDisabled(" ?");
if (ImGui::IsItemHovered())
ImGui::SetTooltip("A bookmark by that name already exists.\nDescription must be unique.");
}
ImGui::Text("Frequency: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
ImGui::InputDouble("##_sdrpp_freq_mgr_add_freq", &vfoFreq, 1.0, 100.0, "%.0f");
ImGui::Text("Bandwidth: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
ImGui::InputInt("##_sdrpp_freq_mgr_add_bw", &vfoBW);
ImGui::Text("Demodulator: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::Text(vfoDemod.c_str());
//ImGui::NewLine();
ImGui::TextDisabled("Active VFO: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::TextDisabled(vfoName.c_str());
ImGui::Separator();
ImGui::SetCursorPosX(ImGui::GetWindowContentRegionWidth() / 2 - 94);
if (ImGui::Button("Add", ImVec2(90, 24))) {
if (strlen(description) == 0) {
spdlog::error("Frequency Manager: Description cannot be empty.");
} else if (VectorArrayContainsString(_this->bookmarks, (std::string)description)) {
spdlog::error("Frequency Manager: A bookmark by that name already exists. Description must be unique.");
} else {
if (strlen(category) == 0)
strcpy(category, "General");
SaveBookmark(description, category, vfoFreq, vfoBW, vfoDemod);
if (!VectorArrayContainsString(_this->categories, std::string(category)))
_this->categories.push_back(category);
std::sort(_this->categories.begin(), _this->categories.end());
_this->bookmarks.push_back(description);
if (_this->selected_category.size() == 0)
_this->selected_category = category;
if (strncmp(_this->selected_category.c_str(), category, 1) > 0)
current_category_index += 1;
config.aquire();
if (config.conf[description].contains("category") && ((std::string)config.conf[description]["category"] == _this->selected_category))
_this->displayed_bookmarks.push_back(description);
config.release();
spdlog::info("Frequency Manager: Saving bookmark '{0}' set at {1} Hz.", description, std::to_string((unsigned int)vfoFreq));
strcpy(category, "");
strcpy(description, "");
ImGui::CloseCurrentPopup();
}
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(90, 24)))
ImGui::CloseCurrentPopup();
ImGui::EndPopup();
}
ImGui::SameLine();
// edit bookmark button
if (_this->selected_bookmarks.size() != 1) {
style::beginDisabled();
disable_ctl = true;
}
static char mode[4] = "";
if (ImGui::Button("Edit", button_sz)) {
ImGui::OpenPopup("Edit bookmark");
config.aquire();
vfoFreq = config.conf[_this->selected_bookmarks[0]]["frequency"];
vfoBW = config.conf[_this->selected_bookmarks[0]]["bandwidth"];
strcpy(mode, std::string(config.conf[_this->selected_bookmarks[0]]["mode"]).c_str());
strcpy(description, _this->selected_bookmarks[0].c_str());
strcpy(category, std::string(config.conf[_this->selected_bookmarks[0]]["category"]).c_str());
config.release();
original_description = description;
original_category = category;
}
if (disable_ctl) {
style::endDisabled();
disable_ctl = false;
}
ImGui::SetNextWindowPos(ImVec2(2, 300));
if (ImGui::BeginPopupModal("Edit bookmark", &unused_open, ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoSavedSettings)) {
ImGui::SetWindowSize("Edit bookmark", ImVec2(330, 210));
ImGui::Text("Category: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
ImGui::InputTextWithHint("##_sdrpp_freq_mgr_edit_category", "General", category, IM_ARRAYSIZE(category), ImGuiInputTextFlags_CallbackHistory, HistoryHandling::CategoryHistoryCallback);
ImGui::SameLine();
ImGui::TextDisabled(" ?");
if (ImGui::IsItemHovered())
ImGui::SetTooltip("If a category is not provided, \"General\" will be used.\nStart typing and use UP/DOWN keys to cycle through history.");
if (strlen(description) == 0) {
ImGui::PushStyleColor(ImGuiCol_Text, (ImVec4)ImColor::HSV(254.0f, 255.0f, 210.0f));
disable_ctl = true;
}
ImGui::Text("Description: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
if (disable_ctl) {
ImGui::PopStyleColor();
disable_ctl = false;
}
ImGui::InputText("##_sdrpp_freq_mgr_edit_descr", description, IM_ARRAYSIZE(description));
if (strlen(description) == 0) {
ImGui::SameLine();
ImGui::TextDisabled(" ?");
if (ImGui::IsItemHovered())
ImGui::SetTooltip("Description cannot be empty.");
}
ImGui::Text("Frequency: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
ImGui::InputDouble("##_sdrpp_freq_mgr_edit_freq", &vfoFreq, 1.0, 100.0, "%.0f");
ImGui::Text("Bandwidth: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::SetNextItemWidth(200.0);
ImGui::InputInt("##_sdrpp_freq_mgr_edit_bw", &vfoBW);
ImGui::Text("Demodulator: ");
ImGui::SameLine();
ImGui::SetCursorPosX(100.0);
ImGui::Text(mode);
ImGui::NewLine();
ImGui::Separator();
ImGui::SetCursorPosX(ImGui::GetWindowContentRegionWidth() / 2 - 94);
if (ImGui::Button("Edit", ImVec2(90, 24))) {
if (strlen(description) == 0) {
spdlog::error("Frequency Manager: Description cannot be empty.");
} else {
if (strlen(category) == 0)
strcpy(category, "General");
if (strcmp(original_description.c_str(), description) == 0) {
EditBookmark(original_description, category, vfoFreq, vfoBW);
} else {
DeleteBookmark(original_description);
SaveBookmark(description, category, vfoFreq, vfoBW, mode);
VectorArrayDeleteElement(_this->displayed_bookmarks, original_description);
_this->displayed_bookmarks.push_back(description);
VectorArrayDeleteElement(_this->bookmarks, original_description);
if (!VectorArrayContainsString(_this->bookmarks, std::string(description)))
_this->bookmarks.push_back(description);
}
if (strcmp(original_category.c_str(), category) != 0) {
if (!VectorArrayContainsString(_this->categories, std::string(category))) {
_this->categories.push_back(category);
std::sort(_this->categories.begin(), _this->categories.end());
if (strncmp(_this->selected_category.c_str(), category, 1) > 0)
current_category_index += 1;
}
VectorArrayDeleteElement(_this->displayed_bookmarks, std::string(description));
}
std::sort(_this->displayed_bookmarks.begin(), _this->displayed_bookmarks.end());
strcpy(category, "");
strcpy(description, "");
_this->selected_bookmarks.clear();
ImGui::CloseCurrentPopup();
}
}
ImGui::SameLine();
if (ImGui::Button("Cancel", ImVec2(90, 24)))
ImGui::CloseCurrentPopup();
ImGui::EndPopup();
}
ImGui::SameLine();
// delete bookmark button
if (_this->selected_bookmarks.size() == 0) {
style::beginDisabled();
disable_ctl = true;
}
if (ImGui::Button("Delete", button_sz)) {
if (_this->selected_bookmarks.size() > 0)
for (const auto &i : _this->selected_bookmarks) {
spdlog::info("Frequency Manager: Deleting saved bookmark '{0}'.", i);
auto itr = std::find(_this->bookmarks.begin(), _this->bookmarks.end(), i);
if (itr != _this->bookmarks.end())
_this->bookmarks.erase(itr);
itr = std::find(_this->displayed_bookmarks.begin(), _this->displayed_bookmarks.end(), i);
if (itr != _this->displayed_bookmarks.end())
_this->displayed_bookmarks.erase(itr);
DeleteBookmark(i);
}
_this->selected_bookmarks.clear();
}
if (disable_ctl) {
style::endDisabled();
disable_ctl = false;
}
// saved frequencies table
static ImGuiTableFlags table_flags = ImGuiTableFlags_Borders | ImGuiTableFlags_Resizable | ImGuiTableFlags_NoSavedSettings | ImGuiTableFlags_ScrollY | ImGuiTableFlags_RowBg;
ImGui::BeginTable("bookmarks_table", 2, table_flags, ImVec2(0.0f, 300));
ImGui::TableSetupScrollFreeze(0, 1);
ImGui::TableSetupColumn("Description", ImGuiTableColumnFlags_WidthStretch, 67);
ImGui::TableSetupColumn("Frequency", ImGuiTableColumnFlags_WidthStretch, 33);
ImGui::TableHeadersRow();
static bool tmpzzz;
if (_this->displayed_bookmarks.size() > 0) {
for (int i = 0; i < _this->displayed_bookmarks.size(); i++) {
ImGui::TableNextRow(); ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0); ImGui::TableSetColumnIndex(0);
const bool selected = VectorArrayContainsString(_this->selected_bookmarks, _this->displayed_bookmarks[i]); if (ImGui::Button(("Add##_freq_mgr_add_" + _this->name).c_str(), ImVec2(ImGui::GetContentRegionAvailWidth(), 0))) {
if (ImGui::Selectable(_this->displayed_bookmarks[i].c_str(), selected, ImGuiSelectableFlags_AllowDoubleClick | ImGuiSelectableFlags_SpanAllColumns | ImGuiSelectableFlags_AllowItemOverlap)) { // If there's no VFO selected, just save the center freq
if (ImGui::GetIO().KeyCtrl) { FrequencyBookmark bm;
if (selected) if (gui::waterfall.selectedVFO == "") {
_this->selected_bookmarks.erase(std::remove(_this->selected_bookmarks.begin(), _this->selected_bookmarks.end(), _this->displayed_bookmarks[i]), _this->selected_bookmarks.end()); bm.frequency = gui::waterfall.getCenterFrequency();
else bm.bandwidth = 0;
_this->selected_bookmarks.push_back(_this->displayed_bookmarks[i]); bm.mode = -1;
} else { }
_this->selected_bookmarks.clear(); else {
_this->selected_bookmarks.push_back(_this->displayed_bookmarks[i]); bm.frequency = gui::waterfall.getCenterFrequency() + sigpath::vfoManager.getOffset(gui::waterfall.selectedVFO);
bm.bandwidth = sigpath::vfoManager.getBandwidth(gui::waterfall.selectedVFO);
bm.mode = -1;
if (core::modComManager.getModuleName(gui::waterfall.selectedVFO) == "radio") {
int mode;
core::modComManager.callInterface(gui::waterfall.selectedVFO, RADIO_IFACE_CMD_GET_MODE, NULL, &mode);
bm.mode = mode;
}
} }
if (ImGui::IsMouseDoubleClicked(ImGuiMouseButton_Left)) { bm.selected = false;
// TODO: set frequency, bw, demod, fft marker
spdlog::info("Frequency Manager: applying bookmark {0} to VFO {1}", _this->displayed_bookmarks[i], vfoName); char name[1024];
spdlog::warn("Functionality not yet implemented."); sprintf(name, "Test Bookmark (%d)", _this->testN);
} _this->bookmarks[name] = bm;
_this->testN++;
_this->editOpen = true;
_this->editedBookmarkName = name;
} }
ImGui::TableSetColumnIndex(1); ImGui::TableSetColumnIndex(1);
std::string fr_sh = "err"; if (ImGui::Button(("Remove##_freq_mgr_rem_" + _this->name).c_str(), ImVec2(ImGui::GetContentRegionAvailWidth(), 0))) {
config.aquire(); for (auto& name : selectedNames) { _this->bookmarks.erase(name); }
if (config.conf[_this->displayed_bookmarks[i]].contains("frequency") && config.conf[_this->displayed_bookmarks[i]]["frequency"].is_number())
fr_sh = GetActiveVFOFrequencyShort((double)config.conf[_this->displayed_bookmarks[i]]["frequency"]);
config.release();
ImGui::Text(fr_sh.c_str());
} }
ImGui::TableSetColumnIndex(2);
if (selectedNames.size() != 1) { style::beginDisabled(); }
if (ImGui::Button(("Edit##_freq_mgr_edt_" + _this->name).c_str(), ImVec2(ImGui::GetContentRegionAvailWidth(), 0))) {
_this->editOpen = true;
_this->editedBookmarkName = selectedNames[0];
}
if (selectedNames.size() != 1) { style::endDisabled(); }
ImGui::EndTable();
// Bookmark list
ImGui::BeginTable(("freq_manager_bkm_table"+_this->name).c_str(), 2, ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg, ImVec2(0, 300));
ImGui::TableSetupColumn("Name", ImGuiTableColumnFlags_WidthStretch);
ImGui::TableSetupColumn("Bookmark", ImGuiTableColumnFlags_WidthStretch);
ImGui::TableHeadersRow();
for (auto& [name, bm] : _this->bookmarks) {
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImVec2 min = ImGui::GetCursorPos();
ImGui::Selectable((name + "##_freq_mgr_bkm_name_" + _this->name).c_str(), &bm.selected, ImGuiSelectableFlags_SpanAllColumns | ImGuiSelectableFlags_SelectOnClick);
if (ImGui::TableGetHoveredColumn() >= 0 && ImGui::IsItemHovered() && ImGui::IsMouseDoubleClicked(ImGuiMouseButton_Left)) {
applyBookmark(bm, gui::waterfall.selectedVFO);
}
ImGui::TableSetColumnIndex(1);
ImGui::Text(freqToStr(bm.frequency).c_str());
ImVec2 max = ImGui::GetCursorPos();
} }
ImGui::EndTable(); ImGui::EndTable();
// import / export if (selectedNames.size() != 1) { style::beginDisabled(); }
ImGui::Separator(); if (ImGui::Button(("Apply##_freq_mgr_apply_" + _this->name).c_str(), ImVec2(menuWidth, 0))) {
ImGui::AlignTextToFramePadding(); FrequencyBookmark& bm = _this->bookmarks[selectedNames[0]];
button_sz.x = menuWidth / 2 - style.ItemSpacing.x / 2; applyBookmark(bm, gui::waterfall.selectedVFO);
style::beginDisabled(); bm.selected = false;
ImGui::Button("Import Bookmarks", button_sz); }
ImGui::SameLine(); if (selectedNames.size() != 1) { style::endDisabled(); }
ImGui::Button("Export Selected", button_sz);
style::endDisabled(); if (_this->editOpen) {
FrequencyBookmark& bm = _this->bookmarks[_this->editedBookmarkName];
_this->editOpen = _this->bookmarkEditDialog(bm);
}
if (_this->addOpen) {
FrequencyBookmark& bm = _this->bookmarks[_this->editedBookmarkName];
_this->addOpen = _this->bookmarkEditDialog(bm);
}
} }
std::string name; std::string name;
bool enabled = true; bool enabled = true;
bool editOpen = false;
bool addOpen = false;
std::vector<std::string> categories; std::map<std::string, FrequencyBookmark> bookmarks;
std::string selected_category;
std::vector<std::string> bookmarks; std::string editedBookmarkName = "";
std::vector<std::string> displayed_bookmarks;
std::vector<std::string> selected_bookmarks; int testN = 0;
struct HistoryHandling {
static int CategoryHistoryCallback(ImGuiInputTextCallbackData *data) {
return 0;
}
};
}; };
MOD_EXPORT void _INIT_() { MOD_EXPORT void _INIT_() {
json def = json({}); // Nothing here
config.setPath(options::opts.root + "/frequency_manager_config.json");
config.load(def);
config.enableAutoSave();
} }
MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) { MOD_EXPORT ModuleManager::Instance* _CREATE_INSTANCE_(std::string name) {
return new FreqManModule(name); return new FrequencyManagerModule(name);
} }
MOD_EXPORT void _DELETE_INSTANCE_(void* instance) { MOD_EXPORT void _DELETE_INSTANCE_(void* instance) {
config.disableAutoSave(); delete (FrequencyManagerModule*)instance;
config.save();
delete (FreqManModule *)instance;
} }
MOD_EXPORT void _END_() { MOD_EXPORT void _END_() {
// Nothing here
} }