You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
dvmhost/src/tged/TGListWnd.h

570 lines
18 KiB

// SPDX-License-Identifier: GPL-2.0-only
/*
* Digital Voice Modem - Talkgroup Editor
* GPLv2 Open Source. Use is subject to license terms.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* Copyright (C) 2024 Bryan Biedenkapp, N2PLL
*
*/
/**
* @file TGListWnd.h
* @ingroup tged
*/
#if !defined(__TG_LIST_WND_H__)
#define __TG_LIST_WND_H__
#include "common/Log.h"
#include "FDblDialog.h"
#include "TGEdMainWnd.h"
#include "TGEditWnd.h"
#include "PeerEntryWnd.h"
#include <final/final.h>
using namespace finalcut;
struct PrivateFListViewScrollToY { typedef void(FListView::*type)(int); };
template class HackTheGibson<PrivateFListViewScrollToY, &FListView::scrollToY>;
struct PrivateFListViewIteratorFirst { typedef FListViewIterator FListView::*type; };
template class HackTheGibson<PrivateFListViewIteratorFirst, &FListView::first_visible_line>;
struct PrivateFListViewVBarPtr { typedef FScrollbarPtr FListView::*type; };
template class HackTheGibson<PrivateFListViewVBarPtr, &FListView::vbar>;
// ---------------------------------------------------------------------------
// Constants
// ---------------------------------------------------------------------------
#define TG_LIST_WIDTH 74
#define TG_LIST_HEIGHT 15
// ---------------------------------------------------------------------------
// Class Declaration
// ---------------------------------------------------------------------------
/**
* @brief This class implements the talkgroup list window.
* @ingroup tged
*/
class HOST_SW_API TGListWnd final : public FDblDialog {
public:
/**
* @brief Initializes a new instance of the TGListWnd class.
* @param widget
*/
explicit TGListWnd(FWidget* widget = nullptr) : FDblDialog{widget}
{
/* stub */
}
/**
* @brief Copy constructor.
*/
TGListWnd(const TGListWnd&) = delete;
/**
* @brief Move constructor.
*/
TGListWnd(TGListWnd&&) noexcept = delete;
/**
* @brief Finalizes an instance of the TGListWnd class.
*/
~TGListWnd() noexcept override = default;
/**
* @brief Disable copy assignment operator (=).
*/
auto operator= (const TGListWnd&) -> TGListWnd& = delete;
/**
* @brief Disable move assignment operator (=).
*/
auto operator= (TGListWnd&&) noexcept -> TGListWnd& = delete;
/**
* @brief Disable set X coordinate.
*/
void setX(int, bool = true) override { }
/**
* @brief Disable set Y coordinate.
*/
void setY(int, bool = true) override { }
/**
* @brief Disable set position.
*/
void setPos(const FPoint&, bool = true) override { }
/**
* @brief Populates the talkgroup listview.
*/
void loadListView()
{
m_selected = TalkgroupRuleGroupVoice();
m_selectedTgId = 0U;
auto entry = g_tidLookups->groupVoice()[0U];
m_selected = entry;
// bryanb: HACK -- use HackTheGibson to access the private current listview iterator to get the scroll position
/*
* This uses the RTTI hack to access private members on FListView; and this code *could* break as a consequence.
*/
int firstScrollLinePos = 0;
if (m_listView.getCount() > 0) {
firstScrollLinePos = (m_listView.*RTTIResult<PrivateFListViewIteratorFirst>::ptr).getPosition();
}
m_listView.clear();
for (auto entry : g_tidLookups->groupVoice()) {
// pad TGs properly
std::ostringstream oss;
oss << std::setw(5) << std::setfill('0') << entry.source().tgId();
// build list view entry
const std::array<std::string, 10U> columns = {
entry.name(), entry.nameAlias(), oss.str(), std::to_string(entry.source().tgSlot()),
(entry.config().active()) ? "X" : "",
(entry.config().affiliated()) ? "X" : "",
std::to_string(entry.config().inclusionSize()),
std::to_string(entry.config().exclusionSize()),
std::to_string(entry.config().alwaysSendSize()),
std::to_string(entry.config().permittedRIDsSize())
};
const finalcut::FStringList line(columns.cbegin(), columns.cend());
m_listView.insert(line);
}
// bryanb: HACK -- use HackTheGibson to access the private set scroll Y to set the scroll position
/*
* This uses the RTTI hack to access private members on FListView; and this code *could* break as a consequence.
*/
if ((size_t)firstScrollLinePos > m_listView.getCount())
firstScrollLinePos = 0;
if (firstScrollLinePos > 0 && m_listView.getCount() > 0) {
(m_listView.*RTTIResult<PrivateFListViewScrollToY>::ptr)(firstScrollLinePos);
(m_listView.*RTTIResult<PrivateFListViewVBarPtr>::ptr)->setValue(firstScrollLinePos);
}
// generate dialog title
uint32_t len = g_tidLookups->groupVoice().size();
std::stringstream ss;
ss << "Talkgroup List (" << len << " TGs)";
FDialog::setText(ss.str());
setFocusWidget(&m_listView);
redraw();
}
/**
* @brief
*/
void addPeerToInclusion()
{
PeerEntryWnd wnd{"Add Peer to Inclusions", this};
wnd.show();
uint32_t peerId = wnd.peerId;
if (peerId > 0U) {
auto groupVoice = g_tidLookups->groupVoice();
for (auto rule : groupVoice) {
uint32_t tgId = rule.source().tgId();
uint8_t tgSlot = rule.source().tgSlot();
auto config = rule.config();
if (!config.active()) {
continue; // don't update rules that aren't active
}
std::vector<uint32_t> inclusions = config.inclusion();
auto it = std::find_if(inclusions.begin(), inclusions.end(), [&](uint32_t x) { return x == wnd.peerId; });
if (it == inclusions.end()) {
LogMessage(LOG_HOST, "Updating TG %s (%u) adding inclusion peer %u", rule.name().c_str(), rule.source().tgId(), peerId);
inclusions.push_back(peerId);
}
config.inclusion(inclusions);
rule.config(config);
g_tidLookups->eraseEntry(tgId, tgSlot);
g_tidLookups->addEntry(rule);
}
}
loadListView();
}
/**
* @brief
*/
void removePeerFromInclusion()
{
PeerEntryWnd wnd{"Remove Peer from Inclusions", this};
wnd.show();
uint32_t peerId = wnd.peerId;
if (peerId > 0U) {
auto groupVoice = g_tidLookups->groupVoice();
for (auto rule : groupVoice) {
uint32_t tgId = rule.source().tgId();
uint8_t tgSlot = rule.source().tgSlot();
auto config = rule.config();
if (!config.active()) {
continue; // don't update rules that aren't active
}
std::vector<uint32_t> inclusions = config.inclusion();
auto it = std::find_if(inclusions.begin(), inclusions.end(), [&](uint32_t x) { return x == wnd.peerId; });
if (it != inclusions.end()) {
LogMessage(LOG_HOST, "Updating TG %s (%u) removing inclusion peer %u", rule.name().c_str(), rule.source().tgId(), peerId);
inclusions.erase(it);
}
config.inclusion(inclusions);
rule.config(config);
g_tidLookups->eraseEntry(tgId, tgSlot);
g_tidLookups->addEntry(rule);
}
}
loadListView();
}
/**
* @brief
*/
void addPeerToAlways()
{
PeerEntryWnd wnd{"Add Peer to Always", this};
wnd.show();
uint32_t peerId = wnd.peerId;
if (peerId > 0U) {
auto groupVoice = g_tidLookups->groupVoice();
for (auto rule : groupVoice) {
uint32_t tgId = rule.source().tgId();
uint8_t tgSlot = rule.source().tgSlot();
auto config = rule.config();
if (!config.active()) {
continue; // don't update rules that aren't active
}
if (!config.affiliated()) {
continue; // don't add peers to always lists for those not marked affiliated
}
std::vector<uint32_t> alwaysSend = config.alwaysSend();
auto it = std::find_if(alwaysSend.begin(), alwaysSend.end(), [&](uint32_t x) { return x == wnd.peerId; });
if (it == alwaysSend.end()) {
LogMessage(LOG_HOST, "Updating TG %s (%u) adding always peer %u", rule.name().c_str(), rule.source().tgId(), peerId);
alwaysSend.push_back(peerId);
}
config.alwaysSend(alwaysSend);
rule.config(config);
g_tidLookups->eraseEntry(tgId, tgSlot);
g_tidLookups->addEntry(rule);
}
}
loadListView();
}
/**
* @brief
*/
void removePeerFromAlways()
{
PeerEntryWnd wnd{"Remove Peer from Always", this};
wnd.show();
uint32_t peerId = wnd.peerId;
if (peerId > 0U) {
auto groupVoice = g_tidLookups->groupVoice();
for (auto rule : groupVoice) {
uint32_t tgId = rule.source().tgId();
uint8_t tgSlot = rule.source().tgSlot();
auto config = rule.config();
if (!config.active()) {
continue; // don't update rules that aren't active
}
if (!config.affiliated()) {
continue; // don't remove peers from always lists for those not marked affiliated
}
std::vector<uint32_t> alwaysSend = config.alwaysSend();
auto it = std::find_if(alwaysSend.begin(), alwaysSend.end(), [&](uint32_t x) { return x == wnd.peerId; });
if (it != alwaysSend.end()) {
LogMessage(LOG_HOST, "Updating TG %s (%u) removing always peer %u", rule.name().c_str(), rule.source().tgId(), peerId);
alwaysSend.erase(it);
}
config.alwaysSend(alwaysSend);
rule.config(config);
g_tidLookups->eraseEntry(tgId, tgSlot);
g_tidLookups->addEntry(rule);
}
}
loadListView();
}
private:
lookups::TalkgroupRuleGroupVoice m_selected;
uint32_t m_selectedTgId;
FListView m_listView{this};
FButton m_addTG{"&Add", this};
FButton m_editTG{"&Edit", this};
FLabel m_fileName{"/path/to/file.yml", this};
FButton m_deleteTG{"&Delete", this};
/**
* @brief Initializes the window layout.
*/
void initLayout() override
{
FDialog::setMinimumSize(FSize{TG_LIST_WIDTH, TG_LIST_HEIGHT});
FDialog::setResizeable(false);
FDialog::setMinimizable(false);
FDialog::setTitlebarButtonVisibility(false);
FDialog::setModal(false);
FDialog::setText("Talkgroup List");
initControls();
loadListView();
FDialog::initLayout();
}
/**
* @brief Initializes window controls.
*/
void initControls()
{
m_addTG.setGeometry(FPoint(2, int(getHeight() - 4)), FSize(9, 1));
m_addTG.setBackgroundColor(FColor::DarkGreen);
m_addTG.setFocusBackgroundColor(FColor::DarkGreen);
m_addTG.addCallback("clicked", [&]() { addEntry(); });
m_editTG.setGeometry(FPoint(13, int(getHeight() - 4)), FSize(10, 1));
m_editTG.setDisable();
m_editTG.addCallback("clicked", [&]() { editEntry(); });
m_fileName.setGeometry(FPoint(27, int(getHeight() - 4)), FSize(42, 1));
m_fileName.setText(g_iniFile);
m_deleteTG.setGeometry(FPoint(int(getWidth()) - 13, int(getHeight() - 4)), FSize(10, 1));
m_deleteTG.setDisable();
m_deleteTG.addCallback("clicked", [&]() { deleteEntry(); });
m_listView.setGeometry(FPoint{1, 1}, FSize{getWidth() - 1, getHeight() - 5});
// configure list view columns
m_listView.addColumn("Name", 25);
m_listView.addColumn("Alias", 40);
m_listView.addColumn("TGID", 9);
m_listView.addColumn("Slot", 4);
m_listView.addColumn("Active", 5);
m_listView.addColumn("Affiliated", 5);
m_listView.addColumn("Inclusions", 5);
m_listView.addColumn("Exclusions", 5);
m_listView.addColumn("Always", 5);
m_listView.addColumn("Permitted RIDs", 5);
// set right alignment for TGID
m_listView.setColumnAlignment(3, finalcut::Align::Right);
m_listView.setColumnAlignment(4, finalcut::Align::Right);
m_listView.setColumnAlignment(5, finalcut::Align::Center);
m_listView.setColumnAlignment(6, finalcut::Align::Center);
m_listView.setColumnAlignment(7, finalcut::Align::Right);
m_listView.setColumnAlignment(8, finalcut::Align::Right);
m_listView.setColumnAlignment(9, finalcut::Align::Right);
m_listView.setColumnAlignment(10, finalcut::Align::Right);
// set type of sorting
m_listView.setColumnSortType(1, finalcut::SortType::Name);
m_listView.setColumnSortType(2, finalcut::SortType::Name);
m_listView.setColumnSortType(3, finalcut::SortType::Name);
// sort by TGID
m_listView.setColumnSort(3, finalcut::SortOrder::Ascending);
m_listView.addCallback("clicked", [&]() { editEntry(); });
m_listView.addCallback("row-changed", [&]() {
FListViewItem* curItem = m_listView.getCurrentItem();
if (curItem != nullptr) {
FString strTgid = curItem->getText(3);
uint32_t tgid = ::atoi(strTgid.c_str());
if (tgid != m_selectedTgId) {
auto entry = g_tidLookups->find(tgid);
if (!entry.isInvalid()) {
m_selected = entry;
/*
if (m_selectedTgId != tgid)
LogMessage(LOG_HOST, "Selected TG %s (%u) for editing", m_selected.name().c_str(), m_selected.source().tgId());
*/
m_selectedTgId = tgid;
m_editTG.setEnable();
m_deleteTG.setEnable();
m_deleteTG.setBackgroundColor(FColor::DarkRed);
m_deleteTG.setFocusBackgroundColor(FColor::DarkRed);
} else {
m_editTG.setDisable();
m_deleteTG.setDisable();
m_deleteTG.resetColors();
}
redraw();
}
}
});
setFocusWidget(&m_listView);
redraw();
}
/**
* @brief
*/
void addEntry()
{
this->lowerWindow();
this->deactivateWindow();
TGEditWnd wnd{TalkgroupRuleGroupVoice(), this};
wnd.show();
this->raiseWindow();
this->activateWindow();
loadListView();
}
/**
* @brief
*/
void editEntry()
{
if (m_selected.isInvalid())
return;
this->lowerWindow();
this->deactivateWindow();
TGEditWnd wnd{m_selected, this};
wnd.show();
this->raiseWindow();
this->activateWindow();
loadListView();
}
/**
* @brief
*/
void deleteEntry()
{
if (m_selected.isInvalid())
return;
LogMessage(LOG_HOST, "Deleting TG %s (%u)", m_selected.name().c_str(), m_selected.source().tgId());
g_tidLookups->eraseEntry(m_selected.source().tgId(), m_selected.source().tgSlot());
// bryanb: HACK -- use HackTheGibson to access the private current listview iterator to get the scroll position
/*
* This uses the RTTI hack to access private members on FListView; and this code *could* break as a consequence.
*/
int firstScrollLinePos = 0;
if (m_listView.getCount() > 0) {
firstScrollLinePos = (m_listView.*RTTIResult<PrivateFListViewIteratorFirst>::ptr).getPosition();
}
if ((size_t)firstScrollLinePos > m_listView.getCount())
firstScrollLinePos = 0;
if (firstScrollLinePos > 0 && m_listView.getCount() > 0) {
--firstScrollLinePos;
(m_listView.*RTTIResult<PrivateFListViewScrollToY>::ptr)(firstScrollLinePos);
(m_listView.*RTTIResult<PrivateFListViewVBarPtr>::ptr)->setValue(firstScrollLinePos);
}
loadListView();
}
/**
* @brief
*/
void drawBorder() override
{
if (!hasBorder())
return;
setColor();
FRect box{{1, 2}, getSize()};
box.scaleBy(0, -1);
FRect rect = box;
if (rect.x1_ref() > rect.x2_ref())
std::swap(rect.x1_ref(), rect.x2_ref());
if (rect.y1_ref() > rect.y2_ref())
std::swap(rect.y1_ref(), rect.y2_ref());
rect.x1_ref() = std::max(rect.x1_ref(), 1);
rect.y1_ref() = std::max(rect.y1_ref(), 1);
rect.x2_ref() = std::min(rect.x2_ref(), rect.x1_ref() + int(getWidth()) - 1);
rect.y2_ref() = std::min(rect.y2_ref(), rect.y1_ref() + int(getHeight()) - 1);
if (box.getWidth() < 3)
return;
// Use box-drawing characters to draw a border
constexpr std::array<wchar_t, 8> box_char
{{
static_cast<wchar_t>(0x2554), // ╔
static_cast<wchar_t>(0x2550), // ═
static_cast<wchar_t>(0x2557), // ╗
static_cast<wchar_t>(0x2551), // ║
static_cast<wchar_t>(0x2551), // ║
static_cast<wchar_t>(0x255A), // ╚
static_cast<wchar_t>(0x2550), // ═
static_cast<wchar_t>(0x255D) // ╝
}};
drawGenericBox(this, box, box_char);
}
/*
** Event Handlers
*/
/**
* @brief Event that occurs on keyboard key press.
* @param e Keyboard Event.
*/
void onKeyPress(finalcut::FKeyEvent* e) override
{
const auto key = e->key();
if (key == FKey::Insert) {
addEntry();
} else if (key == FKey::Enter || key == FKey::Return) {
editEntry();
}
}
};
#endif // __TG_LIST_WND_H__

Powered by TurnKey Linux.