OpenCPN Partial API docs
Loading...
Searching...
No Matches
priority_gui.cpp
1/******************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose:
5 * Author: David Register
6 *
7 ***************************************************************************
8 * Copyright (C) 2022 by David S. Register *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 ***************************************************************************
25 *
26 *
27 */
28
29#include <wx/wxprec.h>
30
31#ifndef WX_PRECOMP
32#include <wx/wx.h>
33#endif // precompiled headers
34
35#include <wx/app.h>
36#include <wx/tokenzr.h>
37
38#ifdef __OCPN__ANDROID__
39#include "androidUTIL.h"
40#include "qdebug.h"
41#include <QtWidgets/QScroller>
42#endif
43
44#include "priority_gui.h"
45#include "ocpn_app.h"
46#include "model/comm_bridge.h"
47#include "ocpn_frame.h"
48#include "ocpn_plugin.h"
49
50extern MyFrame* gFrame;
51
52class PriorityEntry : public wxTreeItemData {
53public:
54 PriorityEntry(int category, int index) {
55 m_category = category, m_index = index;
56 }
57 ~PriorityEntry() {};
58
59 int m_category, m_index;
60};
61
62PriorityDlg::PriorityDlg(wxWindow* parent)
63 : wxDialog(parent, wxID_ANY, _("Adjust Comm Priorities"), wxDefaultPosition,
64 wxSize(480, 420), wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER) {
65 m_selIndex = 0;
66 m_selmap_index = 0;
67
68 wxBoxSizer* mainSizer = new wxBoxSizer(wxVERTICAL);
69 SetSizer(mainSizer);
70
71 wxBoxSizer* secondSizer = new wxBoxSizer(wxHORIZONTAL);
72 mainSizer->Add(secondSizer, 1, wxEXPAND, 5);
73
74 wxStaticBox* pclbBox = new wxStaticBox(this, wxID_ANY, _("Priority List"));
75 wxStaticBoxSizer* stcSizer = new wxStaticBoxSizer(pclbBox, wxVERTICAL);
76 secondSizer->Add(stcSizer, 1, wxALL | wxEXPAND, 5);
77
78 m_prioTree = new wxTreeCtrl(this, wxID_ANY, wxDefaultPosition, wxDefaultSize);
79 stcSizer->Add(m_prioTree, 1, wxALL | wxEXPAND, 5);
80 wxFont* pF = OCPNGetFont(_("Dialog"));
81 m_pF = pF;
82 m_prioTree->SetFont(*pF);
83#ifdef __ANDROID__
84 m_prioTree->GetHandle()->setStyleSheet(
85 getWideScrollBarsStyleSheet() /*getScrollBarsStyleSheet()*/);
86 QScroller::ungrabGesture(m_prioTree->GetHandle());
87#endif
88
89 wxBoxSizer* btnEntrySizer = new wxBoxSizer(wxVERTICAL);
90 secondSizer->Add(btnEntrySizer, 0, wxALL | wxEXPAND, 5);
91 btnMoveUp = new wxButton(this, wxID_ANY, _("Move Up"));
92 btnMoveDown = new wxButton(this, wxID_ANY, _("Move Down"));
93 btnMoveUp->Disable();
94 btnMoveDown->Disable();
95
96 btnEntrySizer->Add(btnMoveUp, 0, wxALL, 5);
97 btnEntrySizer->Add(btnMoveDown, 0, wxALL, 5);
98
99 btnEntrySizer->AddSpacer(15);
100
101 btnRefresh = new wxButton(this, wxID_ANY, _("Refresh"));
102 btnClear = new wxButton(this, wxID_ANY, _("Clear All"));
103
104 btnEntrySizer->Add(btnRefresh, 0, wxALL, 5);
105 btnEntrySizer->Add(btnClear, 0, wxALL, 5);
106
107 wxStdDialogButtonSizer* btnSizer = new wxStdDialogButtonSizer();
108 wxButton* btnOK = new wxButton(this, wxID_OK);
109 wxButton* btnCancel = new wxButton(this, wxID_CANCEL, _("Cancel"));
110 btnSizer->AddButton(btnOK);
111 btnSizer->AddButton(btnCancel);
112 btnSizer->Realize();
113 mainSizer->Add(btnSizer, 0, wxALL | wxEXPAND, 5);
114
115 // Connect Events
116 btnMoveUp->Connect(wxEVT_COMMAND_BUTTON_CLICKED,
117 wxCommandEventHandler(PriorityDlg::OnMoveUpClick), NULL,
118 this);
119 btnMoveDown->Connect(wxEVT_COMMAND_BUTTON_CLICKED,
120 wxCommandEventHandler(PriorityDlg::OnMoveDownClick),
121 NULL, this);
122
123 btnRefresh->Connect(wxEVT_COMMAND_BUTTON_CLICKED,
124 wxCommandEventHandler(PriorityDlg::OnRefreshClick), NULL,
125 this);
126
127 btnClear->Connect(wxEVT_COMMAND_BUTTON_CLICKED,
128 wxCommandEventHandler(PriorityDlg::OnClearClick), NULL,
129 this);
130
131 m_prioTree->Connect(wxEVT_TREE_SEL_CHANGED,
132 wxCommandEventHandler(PriorityDlg::OnItemSelected), NULL,
133 this);
134
135 // Get the current status
136 MyApp& app = wxGetApp();
137 m_map = app.m_comm_bridge.GetPriorityMaps();
138
139 Populate();
140
141 int n_lines = wxMax(m_prioTree->GetCount(), 15);
142
143 wxScreenDC dc;
144 int char_width, char_height;
145 dc.GetTextExtent("W", &char_width, &char_height, NULL, NULL, m_pF);
146
147 int stcw = wxMax(m_maxStringLength * 15 / 10, 15 * char_width);
148 wxSize min_size = wxSize(
149 stcw, wxMin(gFrame->GetSize().y * 2 / 4, n_lines * GetCharHeight()));
150
151 stcSizer->SetMinSize(min_size);
152
153 SetMaxSize(gFrame->GetSize());
154
155 Layout();
156 Fit();
157 Centre();
158
159#ifdef __ANDROID__
160 androidDisableRotation();
161#endif
162}
163
164PriorityDlg::~PriorityDlg() {
165#ifdef __ANDROID__
166 androidEnableRotation();
167#endif
168}
169
170void PriorityDlg::AddLeaves(const std::vector<std::string>& map_list,
171 size_t map_index, std::string map_name,
172 wxTreeItemId leaf_parent) {
173 if (map_list.size() < (size_t)map_index) return;
174
175 // Get the current Priority container for this branch
176 MyApp& app = wxGetApp();
177 PriorityContainer pc = app.m_comm_bridge.GetPriorityContainer(map_name);
178
179 wxString priority_string(map_list[map_index].c_str());
180 wxStringTokenizer tk(priority_string, "|");
181 size_t index = 0;
182 while (tk.HasMoreTokens()) {
183 wxString item_string = tk.GetNextToken();
184
185 wxScreenDC dc;
186 int char_width, char_height;
187 dc.GetTextExtent(item_string, &char_width, &char_height, NULL, NULL, m_pF);
188
189 // Record the maximum display string length, for use in dialog sizing.
190 if (char_width > m_maxStringLength) {
191 m_maxStringLength = char_width;
192 m_max_string = item_string;
193 }
194
195 PriorityEntry* pe = new PriorityEntry(map_index, index);
196 wxTreeItemId id_tk =
197 m_prioTree->AppendItem(leaf_parent, item_string, -1, -1, pe);
198
199 // Set bold text on item currently active (usually 0)
200 if ((size_t)(pc.active_priority) == index) m_prioTree->SetItemBold(id_tk);
201
202 index++;
203 }
204}
205
206void PriorityDlg::Populate() {
207 m_prioTree->Unselect();
208 m_prioTree->DeleteAllItems();
209 m_maxStringLength = 15; // default width calculation
210
211 // wxTreeItemId* rootData = new wxDirItemData(_T("Dummy"), _T("Dummy"),
212 // TRUE);
213 wxTreeItemId m_rootId = m_prioTree->AddRoot(_("Priorities"), -1, -1, NULL);
214 m_prioTree->SetItemHasChildren(m_rootId);
215
216 wxTreeItemId id_position =
217 m_prioTree->AppendItem(m_rootId, _("Position"), -1, -1, NULL);
218 m_prioTree->SetItemHasChildren(id_position);
219 AddLeaves(m_map, 0, "position", id_position);
220
221 wxTreeItemId id_velocity =
222 m_prioTree->AppendItem(m_rootId, _("Speed/Course"), -1, -1, NULL);
223 m_prioTree->SetItemHasChildren(id_velocity);
224 AddLeaves(m_map, 1, "velocity", id_velocity);
225
226 wxTreeItemId id_heading =
227 m_prioTree->AppendItem(m_rootId, _("Heading"), -1, -1, NULL);
228 m_prioTree->SetItemHasChildren(id_heading);
229 AddLeaves(m_map, 2, "heading", id_heading);
230
231 wxTreeItemId id_magvar =
232 m_prioTree->AppendItem(m_rootId, _("Mag Variation"), -1, -1, NULL);
233 m_prioTree->SetItemHasChildren(id_magvar);
234 AddLeaves(m_map, 3, "variation", id_magvar);
235
236 wxTreeItemId id_sats =
237 m_prioTree->AppendItem(m_rootId, _("Satellites"), -1, -1, NULL);
238 m_prioTree->SetItemHasChildren(id_sats);
239 AddLeaves(m_map, 4, "satellites", id_sats);
240
241 m_prioTree->ExpandAll();
242
243 // Restore selection
244 wxTreeItemId rootID = m_prioTree->GetRootItem();
245 wxTreeItemIdValue cookie;
246 int i = m_selmap_index;
247 wxTreeItemId cid = m_prioTree->GetFirstChild(rootID, cookie);
248
249 while ((i > 0) && cid.IsOk()) {
250 cid = m_prioTree->GetNextChild(rootID, cookie);
251 i--;
252 }
253
254 wxTreeItemId ccid = m_prioTree->GetFirstChild(cid, cookie);
255
256 int j = m_selIndex;
257 while ((j > 0) && ccid.IsOk()) {
258 ccid = m_prioTree->GetNextChild(cid, cookie);
259 j--;
260 }
261
262 if (ccid.IsOk()) m_prioTree->SelectItem(ccid);
263}
264
265void PriorityDlg::OnItemSelected(wxCommandEvent& event) {
266 btnMoveUp->Disable();
267 btnMoveDown->Disable();
268
269 wxTreeItemId id = m_prioTree->GetSelection();
270 PriorityEntry* pe = (PriorityEntry*)m_prioTree->GetItemData(id);
271 if (!pe) return;
272
273 m_selIndex = pe->m_index;
274 m_selmap_index = pe->m_category;
275
276 if (pe->m_index > 0) {
277 btnMoveUp->Enable();
278 }
279
280 wxTreeItemId id_parent = m_prioTree->GetItemParent(id);
281
282 // count siblings
283 int n_sibs = 0;
284 wxTreeItemIdValue cookie;
285 wxTreeItemId ch = m_prioTree->GetFirstChild(id_parent, cookie);
286 while (ch.IsOk()) {
287 n_sibs++;
288 ch = m_prioTree->GetNextChild(id_parent, cookie);
289 }
290
291 if (pe->m_index < n_sibs - 1) btnMoveDown->Enable();
292}
293
294void PriorityDlg::OnMoveUpClick(wxCommandEvent& event) {
295 ProcessMove(m_prioTree->GetSelection(), -1);
296}
297
298void PriorityDlg::OnMoveDownClick(wxCommandEvent& event) {
299 ProcessMove(m_prioTree->GetSelection(), 1);
300}
301
302void PriorityDlg::ProcessMove(wxTreeItemId id, int dir) {
303 // Get the extra data
304 PriorityEntry* pe = (PriorityEntry*)m_prioTree->GetItemData(id);
305 if (!pe) return;
306 if (pe->m_category > 4) return;
307
308 // Get the selected category string from the map
309 wxString priority_string = wxString(m_map[pe->m_category].c_str());
310
311 // Build an array
312 wxString prio_array[16]; // enough, plus
313
314 wxStringTokenizer tk(priority_string, "|");
315 int index = 0;
316 while (tk.HasMoreTokens() && index < 16) {
317 prio_array[index] = tk.GetNextToken();
318 index++;
319 }
320 int max_index = index;
321
322 // perform the action
323 if (dir == -1) { // Move UP
324 if (pe->m_index > 0) {
325 // swap entries in array
326 wxString s_above = prio_array[pe->m_index - 1];
327 wxString s_move = prio_array[pe->m_index];
328 prio_array[pe->m_index - 1] = s_move;
329 prio_array[pe->m_index] = s_above;
330 m_selIndex--;
331 }
332 } else { // Move DOWN
333 if (pe->m_index < max_index) {
334 // swap entries in array
335 wxString s_below = prio_array[pe->m_index + 1];
336 wxString s_move = prio_array[pe->m_index];
337 prio_array[pe->m_index + 1] = s_move;
338 prio_array[pe->m_index] = s_below;
339 m_selIndex++;
340 }
341 }
342
343 // create the new string
344 wxString prio_mod;
345 for (int i = 0; i < 16; i++) {
346 if (prio_array[i].Length()) {
347 prio_mod += prio_array[i];
348 prio_mod += wxString("|");
349 }
350 }
351
352 // update the string in the map
353 std::string s_upd(prio_mod.c_str());
354 m_map[pe->m_category] = s_upd;
355
356 // Auto-adjust Sat and COG/SOG priorities if POS has been moved up/down
357 if (pe->m_category == 0) {
358 AdjustSatPriority();
359 AdjustCOGSOGPriority();
360 }
361
362 // Update the priority mechanism
363 MyApp& app = wxGetApp();
364 app.m_comm_bridge.UpdateAndApplyMaps(m_map);
365
366 // And reload the tree GUI
367 m_map = app.m_comm_bridge.GetPriorityMaps();
368 Populate();
369}
370
371void PriorityDlg::OnRefreshClick(wxCommandEvent& event) {
372 // Reload the tree GUI
373 MyApp& app = wxGetApp();
374 m_map = app.m_comm_bridge.GetPriorityMaps();
375 Populate();
376}
377
378void PriorityDlg::OnClearClick(wxCommandEvent& event) {
379 m_map[0].clear();
380 m_map[1].clear();
381 m_map[2].clear();
382 m_map[3].clear();
383 m_map[4].clear();
384
385 m_selmap_index = m_selIndex = 0;
386
387 // Update the priority mechanism
388 MyApp& app = wxGetApp();
389 app.m_comm_bridge.UpdateAndApplyMaps(m_map);
390
391 // And reload the tree GUI
392 m_map = app.m_comm_bridge.GetPriorityMaps();
393 Populate();
394}
395
396void PriorityDlg::AdjustSatPriority() {
397 // Get an array of available sat sources
398 std::string sat_prio = m_map[4];
399 wxArrayString sat_sources;
400 wxString sat_priority_string(sat_prio.c_str());
401 wxStringTokenizer tks(sat_priority_string, "|");
402 while (tks.HasMoreTokens()) {
403 wxString item_string = tks.GetNextToken();
404 sat_sources.Add(item_string);
405 }
406
407 // Step thru the POS priority map
408 std::string pos_prio = m_map[0];
409 wxString pos_priority_string(pos_prio.c_str());
410 wxStringTokenizer tk(pos_priority_string, "|");
411 wxArrayString new_sat_prio;
412 while (tk.HasMoreTokens()) {
413 wxString item_string = tk.GetNextToken();
414 wxString pos_channel = item_string.BeforeFirst(';');
415
416 // search the sat sources array for a match
417 // if found, add to proposed new priority array
418 for (size_t i = 0; i < sat_sources.GetCount(); i++) {
419 if (pos_channel.IsSameAs(sat_sources[i].BeforeFirst(';'))) {
420 new_sat_prio.Add(sat_sources[i]);
421 // Mark this source as "used"
422 sat_sources[i] = "USED";
423 break;
424 } else { // no match, what to do? //FIXME (dave)
425 int yyp = 4;
426 }
427 }
428 }
429 // Create a new sat priority string from new_sat_prio array
430 wxString proposed_sat_prio;
431 for (size_t i = 0; i < new_sat_prio.GetCount(); i++) {
432 proposed_sat_prio += new_sat_prio[i];
433 proposed_sat_prio += wxString("|");
434 }
435
436 // Update the maps with the new sat priority string
437 m_map[4] = proposed_sat_prio.ToStdString();
438}
439
440void PriorityDlg::AdjustCOGSOGPriority() {
441 // Get an array of available COG/SOG sources
442 std::string cogsog_prio = m_map[1];
443 wxArrayString cogsog_sources;
444 wxString cogsog_priority_string(cogsog_prio.c_str());
445 wxStringTokenizer tks(cogsog_priority_string, "|");
446 while (tks.HasMoreTokens()) {
447 wxString item_string = tks.GetNextToken();
448 cogsog_sources.Add(item_string);
449 }
450
451 // Step thru the POS priority map
452 std::string pos_prio = m_map[0];
453 wxString pos_priority_string(pos_prio.c_str());
454 wxStringTokenizer tk(pos_priority_string, "|");
455 wxArrayString new_cogsog_prio;
456 while (tk.HasMoreTokens()) {
457 wxString item_string = tk.GetNextToken();
458 wxString pos_channel = item_string.BeforeFirst(';');
459
460 // search the cogsog sources array for a match
461 // if found, add to proposed new priority array
462 for (size_t i = 0; i < cogsog_sources.GetCount(); i++) {
463 if (pos_channel.IsSameAs(cogsog_sources[i].BeforeFirst(';'))) {
464 new_cogsog_prio.Add(cogsog_sources[i]);
465 // Mark this source as "used"
466 cogsog_sources[i] = "USED";
467 break;
468 } else { // no match, what to do? //FIXME (dave)
469 int yyp = 4;
470 }
471 }
472 }
473 // Create a new cog/sog priority string from new_cogsog_prio array
474 wxString proposed_cogsog_prio;
475 for (size_t i = 0; i < new_cogsog_prio.GetCount(); i++) {
476 proposed_cogsog_prio += new_cogsog_prio[i];
477 proposed_cogsog_prio += wxString("|");
478 }
479
480 // Update the maps with the new cog/sog priority string
481 m_map[1] = proposed_cogsog_prio.ToStdString();
482}
Main application frame.
Definition ocpn_frame.h:135
PlugIn Object Definition/API.
wxFont * OCPNGetFont(wxString TextElement, int default_size)
Gets a font for UI elements.