OpenCPN Partial API docs
Loading...
Searching...
No Matches
connections_dlg.cpp
1#include <array>
2#include <algorithm>
3#include <sstream>
4#include <string>
5#include <vector>
6
7#include <wx/arrstr.h>
8#include <wx/bitmap.h>
9#include <wx/grid.h>
10#include <wx/panel.h>
11#include <wx/sizer.h>
12#include <wx/spinctrl.h>
13#include <wx/textctrl.h>
14#include <wx/window.h>
15
16#include "model/base_platform.h"
17#include "model/comm_drv_factory.h"
19#include "model/comm_util.h"
20#include "model/config_vars.h"
21#include "model/conn_params.h"
22#include "model/conn_states.h"
23
24#include "connections_dlg.h"
25
26#include "color_handler.h"
27#include "connection_edit.h"
28#include "conn_params_panel.h"
29#include "gui_lib.h"
30#include "navutil.h"
31#include "priority_gui.h"
32#include "std_filesystem.h"
33#include "svg_utils.h"
34#include "OCPNPlatform.h"
35
36#ifdef __ANDROID__
37#include "androidUTIL.h"
38#endif
39
40extern OCPNPlatform* g_Platform;
41
42static const auto kUtfArrowDown = wxString::FromUTF8(u8"\u25bc");
43static const auto kUtfArrowRight = wxString::FromUTF8(u8"\u25ba");
44static const auto kUtfCheckmark = wxString::FromUTF8(u8"\u2713");
45static const auto kUtfCircle = wxString::FromUTF8(u8"\u3007");
46static const auto kUtfCrossmark = wxString::FromUTF8(u8"\u274c");
47static const auto kUtfDegrees = wxString::FromUTF8(u8"\u00B0");
48static const auto kUtfExclamationMark = wxString::FromUTF8("!");
49static const auto kUtfFilledCircle = wxString::FromUTF8(u8"\u2b24");
50static const auto kUtfFisheye = wxString::FromUTF8(u8"\u25c9");
51static const auto kUtfGear = wxString::FromUTF8(u8"\u2699");
52static const auto kUtfMultiplyX = wxString::FromUTF8(u8"\u2715");
53static const auto kUtfTrashbin = wxString::FromUTF8(u8"\U0001f5d1");
54
55static inline bool IsWindows() {
56 return wxPlatformInfo::Get().GetOperatingSystemId() & wxOS_WINDOWS;
57}
58
60class StdIcons {
61private:
62 const double m_size;
63 const fs::path m_svg_dir;
64
66 double GetSize(wxWindow* parent) {
67 double size = parent->GetCharHeight() * (IsWindows() ? 1.3 : 1.0);
68#if wxCHECK_VERSION(3, 1, 2)
69 // Apply scale factor, mostly for Windows. Other platforms
70 // does this in the toolkits, ToDIP() is aware of this.
71 size *= static_cast<double>(parent->ToDIP(100)) / 100.;
72#endif
73 // Force minimum physical size for touch screens
74 if (g_btouch) {
75 double pixel_per_mm =
76 wxGetDisplaySize().x / g_Platform->GetDisplaySizeMM();
77 size = std::max(size, 7.0 * pixel_per_mm);
78 }
79 return size;
80 }
81
82 wxBitmap LoadIcon(const std::string filename) {
83 fs::path path = m_svg_dir / filename;
84 return LoadSVG(path.string(), m_size, m_size);
85 }
86
87public:
88 StdIcons(wxWindow* parent)
89 : m_size(GetSize(parent)),
90 m_svg_dir(fs::path(g_Platform->GetSharedDataDir().ToStdString()) /
91 "uidata" / "MUI_flat"),
92 trashbin(LoadIcon("trash_bin.svg")),
93 settings(LoadIcon("setting_gear.svg")),
94 filled_circle(LoadIcon("circle-on.svg")),
95 open_circle(LoadIcon("circle-off.svg")),
96 exclaim_mark(LoadIcon("exclaim_mark.svg")),
97 x_mult(LoadIcon("X_mult.svg")),
98 check_mark(LoadIcon("check_mark.svg")) {}
99
100 const wxBitmap trashbin;
101 const wxBitmap settings;
102 const wxBitmap filled_circle;
103 const wxBitmap open_circle;
104 const wxBitmap exclaim_mark;
105 const wxBitmap x_mult;
106 const wxBitmap check_mark;
107};
108
109// Custom renderer class for rendering bitmap in a grid cell
110class BitmapCellRenderer : public wxGridCellRenderer {
111public:
112 BitmapCellRenderer(const wxBitmap& bitmap)
113 : status(ConnState::Disabled), m_bitmap(bitmap) {}
114
115 // Update the bitmap dynamically
116 void SetBitmap(const wxBitmap& bitmap) { m_bitmap = bitmap; }
117
118 void Draw(wxGrid& grid, wxGridCellAttr& attr, wxDC& dc, const wxRect& rect,
119 int row, int col, bool isSelected) {
120 if (IsWindows()) {
121 dc.SetBrush(wxBrush(GetGlobalColor("DILG1")));
122 dc.DrawRectangle(rect);
123 }
124 // Draw the bitmap centered in the cell
125 dc.DrawBitmap(m_bitmap, rect.x + (rect.width - m_bitmap.GetWidth()) / 2,
126 rect.y + (rect.height - m_bitmap.GetHeight()) / 2, true);
127 }
128
129 wxSize GetBestSize(wxGrid& grid, wxGridCellAttr& attr, wxDC& dc, int row,
130 int col) {
131 // Return the size of the bitmap as the best size for the cell
132 return wxSize(m_bitmap.GetWidth(), m_bitmap.GetHeight());
133 }
134
135 BitmapCellRenderer* Clone() const { return new BitmapCellRenderer(m_bitmap); }
136 ConnState status;
137
138private:
139 wxBitmap m_bitmap;
140};
141
144public:
145 ConnCompare(int col) : m_col(col) {}
146
147 bool operator()(ConnectionParams* p1, ConnectionParams* p2) {
148 switch (m_col) {
149 case 0:
150 return int(p1->bEnabled) > int(p2->bEnabled);
151 case 1:
152 return p1->GetCommProtocol() < p2->GetCommProtocol();
153 case 2:
154 return p1->GetIOTypeValueStr() < p2->GetIOTypeValueStr();
155 case 3:
156 return p1->GetStrippedDSPort() < p2->GetStrippedDSPort();
157 default:
158 return false;
159 }
160 }
161
162private:
163 const int m_col;
164};
165
170public:
172 virtual void Apply() = 0;
173
177 virtual void Cancel() = 0;
178};
179
181class AddConnectionButton : public wxButton {
182public:
183 AddConnectionButton(wxWindow* parent, EventVar& evt_add_connection)
184 : wxButton(parent, wxID_ANY, _("Add new connection...")),
185 m_evt_add_connection(evt_add_connection) {
186 Bind(wxEVT_COMMAND_BUTTON_CLICKED,
187 [&](wxCommandEvent& ev) { OnAddConnection(); });
188 }
189
190private:
191 void OnAddConnection() {
192 ConnectionEditDialog dialog(this);
193 dialog.SetPropsLabel(_("Configure new connection"));
194 dialog.SetDefaultConnectionParams();
195 wxWindow* options = wxWindow::FindWindowByName("Options");
196 assert(options && "Null Options window!");
197 dialog.SetSize(wxSize(options->GetSize().x, options->GetSize().y * 8 / 10));
198 auto rv = dialog.ShowModal();
199 if (rv == wxID_OK) {
200 ConnectionParams* cp = dialog.GetParamsFromControls();
201 if (cp) {
202 cp->b_IsSetup = false; // Trigger new stream
203 TheConnectionParams().push_back(cp);
204 }
205 UpdateDatastreams();
206 m_evt_add_connection.Notify();
207 }
208#ifdef __ANDROID__
209 androidEnableRotation();
210#endif
211 }
212
213 EventVar& m_evt_add_connection;
214};
215
217class ScrolledWindow : public wxScrolledWindow {
218public:
219 ScrolledWindow(wxWindow* parent)
220 : wxScrolledWindow(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize,
221 wxVSCROLL) {}
222
224 void AddClient(wxWindow* client, wxSize max_size, wxSize min_size) {
225 auto vbox = new wxBoxSizer(wxVERTICAL);
226 vbox->Add(client, wxSizerFlags().Border());
227 SetSizer(vbox);
228 SetMinClientSize(min_size);
229 SetMaxSize(max_size);
230 vbox->Layout();
231 SetScrollRate(0, 10);
232 }
233};
234
236class Connections : public wxGrid {
237public:
238 Connections(wxWindow* parent,
239 const std::vector<ConnectionParams*>& connections,
240 EventVar& on_conn_update)
241 : wxGrid(parent, wxID_ANY),
242 m_connections(connections),
243 m_on_conn_delete(on_conn_update),
244 m_last_tooltip_cell(100),
245 m_icons(parent) {
246 SetTable(new wxGridStringTable(), false);
247 GetTable()->AppendCols(8);
248 HideCol(7);
249 static const std::array<wxString, 7> headers = {
250 "", _("Protocol"), _("In/Out"), _("Data port"), _("Status"), "", ""};
251 for (auto hdr = headers.begin(); hdr != headers.end(); hdr++)
252 SetColLabelValue(static_cast<int>(hdr - headers.begin()), *hdr);
253 if (IsWindows()) {
254 SetLabelBackgroundColour(GetGlobalColor("DILG1"));
255 SetLabelTextColour(GetGlobalColor("DILG3"));
256 }
257 HideRowLabels();
258 SetColAttributes(parent);
259
260 ReloadGrid(connections);
261 DisableDragColSize();
262 DisableDragRowSize();
263 wxWindow* options = wxWindow::FindWindowByName("Options");
264 assert(options && "Null Options window!");
265 SetSize(wxSize(options->GetSize().x, options->GetSize().y * 8 / 10));
266 wxWindow::Show(GetNumberRows() > 0);
267
268 GetGridWindow()->Bind(wxEVT_MOTION, [&](wxMouseEvent& ev) {
269 OnMouseMove(ev);
270 ev.Skip();
271 });
272
273 GetGridWindow()->Bind(wxEVT_MOUSEWHEEL, [&](wxMouseEvent& ev) {
274 OnWheel(ev);
275 ev.Skip();
276 });
277
278 Bind(wxEVT_GRID_LABEL_LEFT_CLICK,
279 [&](wxGridEvent& ev) { HandleSort(ev.GetCol()); });
280 Bind(wxEVT_GRID_CELL_LEFT_CLICK,
281 [&](wxGridEvent& ev) { OnClickCell(ev.GetRow(), ev.GetCol()); });
282 Bind(wxEVT_PAINT, [&](wxPaintEvent& ev) {
283 SetColAttributes(static_cast<wxWindow*>(ev.GetEventObject()));
284 ev.Skip();
285 });
286 conn_change_lstnr.Init(
287 m_conn_states.evt_conn_status_change,
288 [&](ObservedEvt&) { OnConnectionChange(m_connections); });
289 }
290
291 void OnWheel(wxMouseEvent& ev) {
292 auto p = GetParent();
293 auto psw = static_cast<ScrolledWindow*>(p);
294 int dir = ev.GetWheelRotation();
295 int xpos, ypos;
296 psw->GetViewStart(&xpos, &ypos);
297 int xsu, ysu;
298 psw->GetScrollPixelsPerUnit(&xsu, &ysu);
299 // Not sure where the factor "4" comes from...
300 psw->Scroll(-1, ypos - (dir / ysu) / 4);
301 }
302
304 void ReloadGrid(const std::vector<ConnectionParams*>& connections) {
305 ClearGrid();
306 m_renderer_status_vector.clear();
307
308 for (auto it = connections.begin(); it != connections.end(); it++) {
309 auto row = static_cast<int>(it - connections.begin());
310 EnsureRows(row);
311 SetCellValue(row, 0, (*it)->bEnabled ? "1" : "");
312 if ((*it)->bEnabled)
313 m_tooltips[row][0] = _("Enabled, click to disable");
314 else
315 m_tooltips[row][0] = _("Disabled, click to enable");
316 std::string protocol = NavAddr::BusToString((*it)->GetCommProtocol());
317 SetCellValue(row, 1, protocol);
318 SetCellValue(row, 2, (*it)->GetIOTypeValueStr());
319 SetCellValue(row, 3, (*it)->GetStrippedDSPort());
320 m_tooltips[row][3] = (*it)->UserComment;
321 SetCellRenderer(row, 5, new BitmapCellRenderer(m_icons.settings));
322 m_tooltips[row][5] = _("Edit connection");
323 SetCellRenderer(row, 6, new BitmapCellRenderer(m_icons.trashbin));
324 m_tooltips[row][6] = _("Delete connection");
325 SetCellValue(row, 7, (*it)->GetKey());
326
327 auto stat_renderer = new BitmapCellRenderer(m_icons.filled_circle);
328 stat_renderer->status = ConnState::Disabled;
329 m_renderer_status_vector.push_back(stat_renderer);
330 SetCellRenderer(row, 4, stat_renderer);
331 }
332 OnConnectionChange(m_connections);
333 AutoSize();
334 }
335
336 wxSize GetGridMaxSize() const {
337 return wxSize(GetCharWidth() * 120,
338 std::min(GetNumberRows() + 3, 10) * 2 * GetCharHeight());
339 }
340
341 wxSize GetGridMinSize() const {
342 return wxSize(GetCharWidth() * 80,
343 std::min(GetNumberRows() + 3, 6) * 2 * GetCharHeight());
344 }
345
348 public:
349 ConnStateCompare(Connections* connections) : m_conns(connections) {}
350 bool operator()(ConnectionParams* p1, ConnectionParams* p2) {
351 int row1 = m_conns->FindConnectionIndex(p1);
352 int row2 = m_conns->FindConnectionIndex(p2);
353 if (row1 == -1 && row2 == -1) return false;
354 if (row1 == -1) return false;
355 if (row2 == -1) return true;
356 int v1 = static_cast<int>(m_conns->GetCellValue(row1, 4)[0]);
357 int v2 = static_cast<int>(m_conns->GetCellValue(row2, 4)[0]);
358 return v1 < v2;
359 }
360 Connections* m_conns;
361 };
362
363private:
368 ConnectionParams* FindRowConnection(int row) {
369 auto found = find_if(m_connections.begin(), m_connections.end(),
370 [&](ConnectionParams* p) {
371 return GetCellValue(row, 7) == p->GetKey();
372 });
373 return found != m_connections.end() ? *found : nullptr;
374 }
375
380 int FindConnectionIndex(ConnectionParams* cp) {
381 using namespace std;
382 auto key = cp->GetKey();
383 auto found =
384 find_if(m_connections.begin(), m_connections.end(),
385 [key](ConnectionParams* cp) { return cp->GetKey() == key; });
386 if (found == m_connections.end()) return -1;
387 return static_cast<int>(found - m_connections.begin());
388 }
389
394 void EnsureRows(size_t rows) {
395 while (m_tooltips.size() <= rows)
396 m_tooltips.push_back(std::vector<std::string>(7));
397 while (GetNumberRows() <= static_cast<int>(rows)) AppendRows(1, false);
398 }
399
401 void SetColAttributes(wxWindow* parent) {
402 if (IsWindows()) {
403 // Set all cells to global color scheme
404 SetDefaultCellBackgroundColour(GetGlobalColor("DILG1"));
405 }
406 auto enable_attr = new wxGridCellAttr();
407 enable_attr->SetAlignment(wxALIGN_CENTRE, wxALIGN_CENTRE);
408 enable_attr->SetRenderer(new wxGridCellBoolRenderer());
409 enable_attr->SetEditor(new wxGridCellBoolEditor());
410 if (IsWindows()) enable_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
411 SetColAttr(0, enable_attr);
412
413 auto protocol_attr = new wxGridCellAttr();
414 protocol_attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTRE);
415 protocol_attr->SetReadOnly(true);
416 if (IsWindows())
417 protocol_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
418 SetColAttr(1, protocol_attr);
419
420 auto in_out_attr = new wxGridCellAttr();
421 in_out_attr->SetAlignment(wxALIGN_CENTRE, wxALIGN_CENTRE);
422 in_out_attr->SetReadOnly(true);
423 if (IsWindows()) in_out_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
424 SetColAttr(2, in_out_attr);
425
426 auto port_attr = new wxGridCellAttr();
427 port_attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTRE);
428 port_attr->SetReadOnly(true);
429 if (IsWindows()) port_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
430 SetColAttr(3, port_attr);
431
432 auto status_attr = new wxGridCellAttr();
433 status_attr->SetAlignment(wxALIGN_CENTRE, wxALIGN_CENTRE);
434 status_attr->SetReadOnly(true);
435 status_attr->SetFont(parent->GetFont().Scale(1.3));
436 if (IsWindows()) status_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
437 SetColAttr(4, status_attr);
438
439 auto edit_attr = new wxGridCellAttr();
440 edit_attr->SetAlignment(wxALIGN_CENTRE, wxALIGN_CENTRE);
441 edit_attr->SetFont(parent->GetFont().Scale(1.3));
442 edit_attr->SetReadOnly(true);
443 if (IsWindows()) edit_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
444 SetColAttr(5, edit_attr);
445
446 auto delete_attr = new wxGridCellAttr();
447 delete_attr->SetAlignment(wxALIGN_LEFT, wxALIGN_CENTRE);
448 delete_attr->SetFont(parent->GetFont().Scale(1.3));
449 delete_attr->SetTextColour(*wxRED);
450 delete_attr->SetReadOnly(true);
451 if (IsWindows()) delete_attr->SetBackgroundColour(GetGlobalColor("DILG1"));
452 SetColAttr(6, delete_attr);
453 }
454
456 void OnClickCell(int row, int col) {
457 if (col == 0)
458 HandleEnable(row);
459 else if (col == 5) {
460 HandleEdit(row);
461 } else if (col == 6) {
462 HandleDelete(row);
463 }
464 }
466 void OnMouseMove(wxMouseEvent& ev) {
467 wxPoint pt = ev.GetPosition();
468 int row = YToRow(pt.y);
469 int col = XToCol(pt.x);
470 if (col < 0 || col >= 7 || row < 0 || row >= GetNumberRows()) return;
471 if (row * 7 + col == m_last_tooltip_cell) return;
472 m_last_tooltip_cell = row * 7 + col;
473 GetGridWindow()->SetToolTip(m_tooltips[row][col]);
474 }
475
477 void OnConnectionChange(const std::vector<ConnectionParams*>& connections) {
478 bool refresh_needed = false;
479 for (auto it = connections.begin(); it != connections.end(); it++) {
480 ConnState state = m_conn_states.GetDriverState(
481 (*it)->GetCommProtocol(), (*it)->GetStrippedDSPort());
482 if (!(*it)->bEnabled) state = ConnState::Disabled;
483 auto row = static_cast<int>(it - connections.begin());
484 EnsureRows(row);
485 if (m_renderer_status_vector.size() < (size_t)(row + 1)) continue;
486 switch (state) {
487 case ConnState::Disabled:
488 if (m_renderer_status_vector[row]->status != ConnState::Disabled) {
489 m_renderer_status_vector[row]->SetBitmap(m_icons.filled_circle);
490 m_renderer_status_vector[row]->status = ConnState::Disabled;
491 refresh_needed = true;
492 }
493 m_tooltips[row][4] = _("Disabled");
494 break;
495 case ConnState::NoStats:
496 if (m_renderer_status_vector[row]->status != ConnState::NoStats) {
497 m_renderer_status_vector[row]->SetBitmap(m_icons.open_circle);
498 m_renderer_status_vector[row]->status = ConnState::NoStats;
499 refresh_needed = true;
500 }
501 m_tooltips[row][4] = _("No driver statistics available");
502 break;
503 case ConnState::NoData:
504 if (m_renderer_status_vector[row]->status != ConnState::NoData) {
505 m_renderer_status_vector[row]->SetBitmap(m_icons.exclaim_mark);
506 m_renderer_status_vector[row]->status = ConnState::NoData;
507 refresh_needed = true;
508 }
509 m_tooltips[row][4] = _("No data flowing through connection");
510 break;
511 case ConnState::Unavailable:
512 if (m_renderer_status_vector[row]->status != ConnState::Unavailable) {
513 m_renderer_status_vector[row]->SetBitmap(m_icons.x_mult);
514 m_renderer_status_vector[row]->status = ConnState::Unavailable;
515 refresh_needed = true;
516 }
517 m_tooltips[row][4] = _("The device is unavailable");
518 break;
519 case ConnState::Ok:
520 if (m_renderer_status_vector[row]->status != ConnState::Ok) {
521 m_renderer_status_vector[row]->SetBitmap(m_icons.check_mark);
522 m_renderer_status_vector[row]->status = ConnState::Ok;
523 refresh_needed = true;
524 }
525 m_tooltips[row][4] = _("Data is flowing");
526 break;
527 }
528 }
529 if (refresh_needed) ForceRefresh();
530 }
531
532 void SetSortingColumn(int col) {
533 if (GetSortingColumn() != wxNOT_FOUND) {
534 int old_col = GetSortingColumn();
535 auto label = GetColLabelValue(old_col);
536 if (label[0] == kUtfArrowDown[0])
537 SetColLabelValue(old_col, label.substr(2));
538 }
539 auto label = GetColLabelValue(col);
540 if (label[0] != kUtfArrowDown[0])
541 SetColLabelValue(col, kUtfArrowDown + " " + label);
542 wxGrid::SetSortingColumn(col);
543 Fit();
544 }
545
547 void HandleSort(int col) {
548 if (col > 4) return;
549 auto& params = TheConnectionParams();
550 if (col < 4)
551 std::sort(params.begin(), params.end(), ConnCompare(col));
552 else // col == 4
553 std::sort(params.begin(), params.end(), ConnStateCompare(this));
554 ReloadGrid(TheConnectionParams());
555 SetSortingColumn(col);
556 }
557
559 void HandleEnable(int row) {
560 ConnectionParams* cp = FindRowConnection(row);
561 if (!cp) return;
562 cp->bEnabled = !cp->bEnabled;
563 cp->b_IsSetup = FALSE; // trigger a rebuild/takedown of the connection
564 SetCellValue(row, 0, cp->bEnabled ? "1" : "");
565 if (cp->bEnabled)
566 m_tooltips[row][0] = _("Enabled, click to disable");
567 else
568 m_tooltips[row][0] = _("Disabled, click to enable");
569 DriverStats stats;
570 stats.driver_iface = cp->GetStrippedDSPort();
571 stats.driver_bus = cp->GetCommProtocol();
572 m_conn_states.HandleDriverStats(stats);
573 StopAndRemoveCommDriver(cp->GetStrippedDSPort(), cp->GetCommProtocol());
574 if (cp->bEnabled) MakeCommDriver(cp);
575 cp->b_IsSetup = true;
576 if (!cp->bEnabled) SetCellValue(row, 4, kUtfFilledCircle);
577 }
578
580 void HandleEdit(int row) {
581 ConnectionParams* cp = FindRowConnection(row);
582 if (cp) {
583 ConnectionEditDialog dialog(this);
584 DimeControl(&dialog);
585 dialog.SetPropsLabel(_("Edit Selected Connection"));
586 dialog.PreloadControls(cp);
587 wxWindow* options = wxWindow::FindWindowByName("Options");
588 assert(options && "Null Options window!");
589 dialog.SetSize(
590 wxSize(options->GetSize().x, options->GetSize().y * 8 / 10));
591 Show(GetNumberRows() > 0);
592
593 auto rv = dialog.ShowModal();
594 if (rv == wxID_OK) {
595 ConnectionParams* cp_edited = dialog.GetParamsFromControls();
596 delete cp->m_optionsPanel;
597 StopAndRemoveCommDriver(cp->GetStrippedDSPort(), cp->GetCommProtocol());
598 int index = FindConnectionIndex(cp);
599 assert(index != -1 && "Cannot look up connection index");
600 TheConnectionParams()[index] = cp_edited;
601 cp_edited->b_IsSetup = false; // Trigger new stream
602 ReloadGrid(m_connections);
603 UpdateDatastreams();
604 }
605 }
606 }
607
609 void HandleDelete(int row) {
610 ConnectionParams* cp = FindRowConnection(row);
611 auto found = std::find(m_connections.begin(), m_connections.end(), cp);
612 if (found != m_connections.end()) {
613 std::stringstream ss;
614 ss << _("Ok to delete connection on ") << (*found)->GetStrippedDSPort();
615 int rcode = OCPNMessageBox(this, ss.str(), _("Delete connection?"),
616 wxOK | wxCANCEL);
617 if (rcode != wxID_OK && rcode != wxID_YES) return;
618 delete (*found)->m_optionsPanel;
619 StopAndRemoveCommDriver((*found)->GetStrippedDSPort(),
620 (*found)->GetCommProtocol());
621 TheConnectionParams().erase(found);
622 if (GetNumberRows() > static_cast<int>(m_connections.size()))
623 DeleteRows(GetNumberRows() - 1);
624 m_on_conn_delete.Notify();
625 }
626 }
627
628 ObsListener conn_change_lstnr;
629 std::vector<std::vector<std::string>> m_tooltips;
630 ConnStates m_conn_states;
631 const std::vector<ConnectionParams*>& m_connections;
632 EventVar& m_on_conn_delete;
633 int m_last_tooltip_cell;
634 StdIcons m_icons;
635 std::vector<BitmapCellRenderer*> m_renderer_status_vector;
636};
637
639class GeneralPanel : public wxPanel {
640public:
641 explicit GeneralPanel(wxWindow* parent) : wxPanel(parent, wxID_ANY) {
642 auto sizer = new wxStaticBoxSizer(wxVERTICAL, this, _("General"));
643 SetSizer(sizer);
644 auto flags = wxSizerFlags().Border();
645 auto hbox = new wxBoxSizer(wxHORIZONTAL);
646 hbox->Add(new wxStaticText(this, wxID_ANY, _("Upload format:")), flags);
647 hbox->Add(new UploadOptionsChoice(this), flags);
648 sizer->Add(hbox, flags);
649 }
650
651private:
653 class UploadOptionsChoice : public wxChoice, public ApplyCancel {
654 public:
655 explicit UploadOptionsChoice(wxWindow* parent) : wxChoice() {
656 wxArrayString wx_choices;
657 for (auto& c : choices) wx_choices.Add(c);
658 Create(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize, wx_choices);
659 Cancel();
660 }
661
662 void Cancel() {
663 if (g_bGarminHostUpload)
664 SetSelection(1);
665 else if (g_GPS_Ident == "FurunoGP3X")
666 SetSelection(2);
667 else
668 SetSelection(0);
669 }
670
671 void Apply() {
672 switch (GetSelection()) {
673 case 0:
674 g_bGarminHostUpload = false;
675 g_GPS_Ident = "Generic";
676 break;
677 case 1:
678 g_bGarminHostUpload = true;
679 g_GPS_Ident = "Generic";
680 break;
681 case 2:
682 g_bGarminHostUpload = false;
683 g_GPS_Ident = "FurunoGP3X";
684 break;
685 }
686 }
687
688 const std::array<wxString, 3> choices = {
689 _("Generic"), _("Use Garmin GRMN (Host) mode for uploads"),
690 _("Format uploads for Furuno GP4X")};
691 };
692};
693
695class ShowAdvanced : public wxStaticText {
696public:
697 ShowAdvanced(wxWindow* parent, std::function<void(bool)> on_toggle)
698 : wxStaticText(parent, wxID_ANY, ""),
699 m_on_toggle(on_toggle),
700 m_show_advanced(true) {
701 Toggle();
702 Bind(wxEVT_LEFT_DOWN, [&](wxMouseEvent& ev) { Toggle(); });
703 }
704
705private:
706 wxString m_label = _("Advanced ");
707 std::function<void(bool)> m_on_toggle;
708 bool m_show_advanced;
709
710 void Toggle() {
711 m_show_advanced = !m_show_advanced;
712 m_label = _("Advanced ");
713 m_label += (m_show_advanced ? kUtfArrowDown : kUtfArrowRight);
714 SetLabelText(m_label);
715 m_on_toggle(m_show_advanced);
716 }
717};
718
720class AdvancedPanel : public wxPanel {
721public:
722 explicit AdvancedPanel(wxWindow* parent) : wxPanel(parent, wxID_ANY) {
723 auto sizer = new wxStaticBoxSizer(wxVERTICAL, this, "");
724 sizer->Add(new BearingsCheckbox(this), wxSizerFlags().Expand());
725 sizer->Add(new NmeaFilterRow(this), wxSizerFlags().Expand());
726 sizer->Add(new TalkerIdRow(this), wxSizerFlags().Expand());
727 sizer->Add(new NetmaskRow(this), wxSizerFlags().Expand());
728 SetSizer(sizer);
729 }
730
731private:
733 class BearingsCheckbox : public wxCheckBox, public ApplyCancel {
734 public:
735 BearingsCheckbox(wxWindow* parent)
736 : wxCheckBox(parent, wxID_ANY,
737 _("Use magnetic bearing in output sentence APB")) {
738 SetValue(g_bMagneticAPB);
739 wxCheckBox::SetValue(g_bMagneticAPB);
740 }
741
742 void Apply() override { g_bMagneticAPB = GetValue(); }
743 void Cancel() override { SetValue(g_bMagneticAPB); }
744 };
745
747 class NmeaFilterRow : public wxPanel, public ApplyCancel {
748 wxCheckBox* checkbox;
749 wxTextCtrl* filter_period;
750
751 public:
752 NmeaFilterRow(wxWindow* parent) : wxPanel(parent) {
753 auto hbox = new wxBoxSizer(wxHORIZONTAL);
754 checkbox = new wxCheckBox(
755 this, wxID_ANY,
756 _("Filter NMEA course and speed data. Filter period: "));
757 checkbox->SetValue(g_bfilter_cogsog);
758 hbox->Add(checkbox, wxSizerFlags().Align(wxALIGN_CENTRE));
759 filter_period =
760 new wxTextCtrl(this, wxID_ANY, "", wxDefaultPosition,
761 wxSize(50, 3 * wxWindow::GetCharWidth()), 0);
762 filter_period->SetValue(std::to_string(g_COGFilterSec));
763 hbox->Add(filter_period, wxSizerFlags().Border());
764 SetSizer(hbox);
765 Cancel();
766 }
767
768 void Apply() override {
769 std::stringstream ss;
770 ss << filter_period->GetValue();
771 ss >> g_COGFilterSec;
772 g_bfilter_cogsog = checkbox->GetValue();
773 }
774
775 void Cancel() override {
776 std::stringstream ss;
777 ss << g_COGFilterSec;
778 filter_period->SetValue(ss.str());
779 checkbox->SetValue(g_bfilter_cogsog);
780 }
781 };
782
784 class TalkerIdRow : public wxPanel, public ApplyCancel {
785 wxTextCtrl* text_ctrl;
786
787 public:
788 TalkerIdRow(wxWindow* parent) : wxPanel(parent) {
789 auto hbox = new wxBoxSizer(wxHORIZONTAL);
790 hbox->Add(new wxStaticText(this, wxID_ANY, _("NMEA 0183 Talker Id: ")),
791 wxSizerFlags().Align(wxALIGN_CENTRE_VERTICAL).Border());
792 text_ctrl = new wxTextCtrl(this, wxID_ANY, "", wxDefaultPosition,
793 wxSize(50, 3 * wxWindow::GetCharWidth()));
794 text_ctrl->SetValue(g_TalkerIdText);
795 hbox->Add(text_ctrl, wxSizerFlags().Border());
796 SetSizer(hbox);
797 Cancel();
798 }
799
800 void Apply() override { g_TalkerIdText = text_ctrl->GetValue(); }
801 void Cancel() override { text_ctrl->SetValue(g_TalkerIdText); }
802 };
803
805 class NetmaskRow : public wxPanel, public ApplyCancel {
806 public:
807 NetmaskRow(wxWindow* parent)
808 : wxPanel(parent),
809 m_spin_ctrl(new wxSpinCtrl(this, wxID_ANY)),
810 m_text(new wxStaticText(this, wxID_ANY, "")) {
811 m_spin_ctrl->SetRange(8, 32);
812 auto hbox = new wxBoxSizer(wxHORIZONTAL);
813 auto flags = wxSizerFlags().Align(wxALIGN_CENTRE_VERTICAL).Border();
814 hbox->Add(new wxStaticText(this, wxID_ANY, _("Netmask: ")), flags);
815 hbox->Add(m_text, flags);
816 hbox->Add(new wxStaticText(this, wxID_ANY, _("length (bits): ")), flags);
817 hbox->Add(m_spin_ctrl, flags);
818 SetSizer(hbox);
819 Cancel();
820
821 Bind(wxEVT_SPINCTRL, [&](wxSpinEvent& ev) {
822 m_text->SetLabel(BitsToDottedMask(m_spin_ctrl->GetValue()));
823 Layout();
824 });
825 }
826
827 void Apply() override { g_netmask_bits = m_spin_ctrl->GetValue(); }
828
829 void Cancel() override {
830 m_spin_ctrl->SetValue(g_netmask_bits);
831 m_text->SetLabel(BitsToDottedMask(m_spin_ctrl->GetValue()));
832 Layout();
833 }
834
835 private:
836 wxSpinCtrl* m_spin_ctrl;
837 wxStaticText* m_text;
838
839 std::string BitsToDottedMask(unsigned bits) {
840 uint32_t mask = 0xffffffff << (32 - bits);
841 std::stringstream ss;
842 ss << ((mask & 0xff000000) >> 24) << ".";
843 ss << ((mask & 0x00ff0000) >> 16) << ".";
844 ss << ((mask & 0x0000ff00) >> 8) << ".";
845 ss << (mask & 0x000000ff);
846 return ss.str();
847 }
848 };
849};
850
852class PrioritiesBtn : public wxButton {
853public:
854 PrioritiesBtn(wxWindow* parent)
855 : wxButton(parent, wxID_ANY, _("Adjust Nav data priorities...")) {
856 Bind(wxEVT_COMMAND_BUTTON_CLICKED, [&](wxCommandEvent&) {
857 PriorityDlg dialog(this);
858 dialog.ShowModal();
859 });
860 }
861};
862
865 wxWindow* parent, const std::vector<ConnectionParams*>& connections)
866 : wxPanel(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize,
867 wxTAB_TRAVERSAL, "ConnectionsDlg"),
868 m_connections(connections) {
869 auto vbox = new wxBoxSizer(wxVERTICAL);
870 auto scrolled_window = new ScrolledWindow(this);
871 auto conn_grid =
872 new Connections(scrolled_window, m_connections, m_evt_add_connection);
873 scrolled_window->AddClient(conn_grid, conn_grid->GetGridMaxSize(),
874 conn_grid->GetGridMinSize());
875 vbox->Add(scrolled_window, wxSizerFlags(5).Expand().Border());
876 vbox->Add(new AddConnectionButton(this, m_evt_add_connection),
877 wxSizerFlags().Border());
878 vbox->Add(new PrioritiesBtn(this), wxSizerFlags().Border());
879 vbox->Add(0, wxWindow::GetCharHeight()); // Expanding spacer
880 auto panel_flags = wxSizerFlags().Border(wxLEFT | wxDOWN | wxRIGHT).Expand();
881 vbox->Add(new GeneralPanel(this), panel_flags);
882
883 auto advanced_panel = new AdvancedPanel(this);
884 auto on_toggle = [&, advanced_panel, vbox](bool show) {
885 advanced_panel->Show(show);
886 vbox->SetSizeHints(this);
887 vbox->Fit(this);
888 };
889 vbox->Add(new ShowAdvanced(this, on_toggle), panel_flags);
890 vbox->Add(advanced_panel, panel_flags.ReserveSpaceEvenIfHidden());
891
892 SetSizer(vbox);
893 SetAutoLayout(true);
894 wxWindow::Fit();
895
896 auto on_evt_update_connections = [&, conn_grid,
897 scrolled_window](ObservedEvt&) {
898 conn_grid->ReloadGrid(TheConnectionParams());
899 conn_grid->Show(conn_grid->GetNumberRows() > 0);
900 scrolled_window->SetMinClientSize(conn_grid->GetGridMinSize());
901 scrolled_window->SetMaxSize(conn_grid->GetGridMaxSize());
902 Layout();
903 };
904 m_add_connection_lstnr.Init(m_evt_add_connection, on_evt_update_connections);
905};
906
907void ConnectionsDlg::OnResize() {
908 Layout();
909 Refresh();
910 Update();
911}
912
913void ConnectionsDlg::DoApply(wxWindow* root) {
914 for (wxWindow* child : root->GetChildren()) {
915 auto widget = dynamic_cast<ApplyCancel*>(child);
916 if (widget) widget->Apply();
917 DoApply(child);
918 }
919}
920
921void ConnectionsDlg::DoCancel(wxWindow* root) {
922 for (wxWindow* child : root->GetChildren()) {
923 auto widget = dynamic_cast<ApplyCancel*>(child);
924 if (widget) widget->Cancel();
925 DoCancel(child);
926 }
927}
928
929void ConnectionsDlg::ApplySettings() { DoApply(this); }
930
931void ConnectionsDlg::CancelSettings() { DoCancel(this); }
The "Add new connection" button.
Indeed: The "Advanced" panel.
Interface implemented by widgets supporting Apply and Cancel.
virtual void Apply()=0
Make values set by user actually being used.
virtual void Cancel()=0
Restore values modified by user to their pristine state, often in a global.
std::sort support: Compare two ConnectionParams w r t given column
Filter reading driver driver_stats status reports from CommDriverRegistry transforming these to a str...
Definition conn_states.h:65
EventVar evt_conn_status_change
Notified with a shared_ptr<ConnData> when the state is changed.
Definition conn_states.h:71
Dialog for editing connection parameters.
std::string GetKey() const
Return string unique for each instance.
void ApplySettings()
Traverse root's children and invoke Apply if they implement ApplyCancel.
ConnectionsDlg(wxWindow *parent, const std::vector< ConnectionParams * > &connections)
Main window: connections grid, "Add new connection", general options.
void CancelSettings()
Traverse root's children and invoke Cancel if they implement ApplyCancel.
std::sort support: Compare two ConnectionParams w r t state.
Grid with existing connections: type, port, status, etc.
void ReloadGrid(const std::vector< ConnectionParams * > &connections)
Reload grid using data from given list of connections.
Generic event handling between MVC Model and Controller based on a shared EventVar variable.
const void Notify()
Notify all listeners, no data supplied.
Indeed: the General panel.
Provides platform-specific support utilities for OpenCPN.
double GetDisplaySizeMM()
Get the width of the screen in millimeters.
Define an action to be performed when a KeyProvider is notified.
Definition observable.h:228
void Init(const KeyProvider &kp, std::function< void(ObservedEvt &ev)> action)
Initiate an object yet not listening.
Definition observable.h:255
Custom event class for OpenCPN's notification system.
Button invokes "Adjust communication priorities" GUI.
Scrollable window wrapping the client i.
void AddClient(wxWindow *client, wxSize max_size, wxSize min_size)
Set contents and size limits for scrollable area.
The "Show advanced" text + right/down triangle and handler.
Standard icons bitmaps: settings gear, trash bin, etc.
Driver registration container, a singleton.
Runtime connection/driver state definitions.
Dialog and support code for editing a connection.
General purpose GUI support.
Driver statistics report.