OpenCPN Partial API docs
Loading...
Searching...
No Matches
pluginmanager.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: PlugIn Manager Object
5 * Author: David Register
6 *
7 ***************************************************************************
8 * Copyright (C) 2010 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#include <algorithm>
26#include <archive.h>
27#include <cstdio>
28#include <cstdio>
29#include <errno.h>
30#include <fcntl.h>
31#include <fstream>
32#include <iostream>
33#include <iostream>
34#include <memory>
35#include <set>
36#include <sstream>
37#include <stdint.h>
38#include <string>
39#include <unordered_map>
40
41#ifdef __MINGW32__
42#undef IPV6STRICT // mingw FTBS fix: missing struct ip_mreq
43#include <windows.h>
44#endif
45
46#include <typeinfo>
47#if defined(__linux__) && !defined(__ANDROID__)
48#include <wordexp.h>
49#endif
50#include <wx/wx.h>
51#include <wx/dir.h>
52#include <wx/event.h>
53#include <wx/filename.h>
54#include <wx/aui/aui.h>
55#include <wx/platinfo.h>
56#include <wx/popupwin.h>
57#include <wx/progdlg.h>
58#include <wx/statline.h>
59#include <wx/tokenzr.h>
60#include <wx/tooltip.h>
61#include <wx/app.h>
62#include <wx/hashset.h>
63#include <wx/hashmap.h>
64#include <wx/jsonval.h>
65#include <wx/jsonreader.h>
66#include <wx/uri.h>
67#include <wx/zipstrm.h>
68#include <wx/zstream.h>
69#include <wx/tarstrm.h>
70#include <wx/textwrapper.h>
71#include <wx/app.h>
72
73#ifndef __WXMSW__
74#include <cxxabi.h>
75#endif // __WXMSW__
76
77#include <archive_entry.h>
78typedef __LA_INT64_T la_int64_t; // "older" libarchive versions support
79
80#ifdef USE_LIBELF
81#include <elf.h>
82#include <libelf.h>
83#include <gelf.h>
84#endif
85
86#include "config.h"
87
88#include "model/ais_target_data.h"
92#include "model/comm_drv_n2k.h"
95#include "model/comm_vars.h"
96#include "model/config_vars.h"
97#include "model/datetime.h"
98#include "model/downloader.h"
99#include "model/georef.h"
100#include "model/json_event.h"
101#include "model/logger.h"
102#include "model/multiplexer.h"
103#include "model/nav_object_database.h"
104#include "model/navutil_base.h"
105#include "model/ocpn_utils.h"
106#include "model/plugin_cache.h"
107#include "model/plugin_comm.h"
108#include "model/plugin_handler.h"
109#include "model/plugin_loader.h"
110#include "model/plugin_paths.h"
111#include "model/route.h"
112#include "model/routeman.h"
113#include "model/safe_mode.h"
114#include "model/select.h"
115#include "model/semantic_vers.h"
116#include "model/track.h"
117
118#include "ais.h"
119#include "canvasMenu.h"
120#include "cat_settings.h"
121#include "chartbase.h" // for ChartPlugInWrapper
122#include "chartdb.h"
123#include "chartdbs.h"
124#include "chcanv.h"
125#include "config.h"
126#include "download_mgr.h"
127#include "dychart.h"
128#include "FontMgr.h"
129#include "gshhs.h"
130#include "model/ais_decoder.h"
131#include "mygeom.h"
132#include "navutil.h"
133#include "observable_confvar.h"
134#include "observable_globvar.h"
135#include "ocpn_app.h"
136#include "OCPN_AUIManager.h"
137#include "ocpndc.h"
138#include "ocpn_frame.h"
139#include "ocpn_pixel.h"
140#include "OCPNPlatform.h"
141#include "OCPNRegion.h"
142#include "options.h"
143#include "piano.h"
144#include "pluginmanager.h"
145#include "routemanagerdialog.h"
146#include "routeman_gui.h"
147#include "s52plib.h"
148#include "s52utils.h"
149#include "SoundFactory.h"
150#include "styles.h"
151#include "svg_utils.h"
152#include "SystemCmdSound.h"
153#include "toolbar.h"
154#include "update_mgr.h"
155#include "waypointman_gui.h"
156
157#ifdef __ANDROID__
158#include <dlfcn.h>
159#include "androidUTIL.h"
160#endif
161
162#ifdef ocpnUSE_GL
163#include "glChartCanvas.h"
164#endif
165extern MyConfig* pConfig;
166extern OCPN_AUIManager* g_pauimgr;
167
168#if wxUSE_XLOCALE || !wxCHECK_VERSION(3, 0, 0)
169extern wxLocale* plocale_def_lang;
170#endif
171
172extern OCPNPlatform* g_Platform;
173extern ChartDB* ChartData;
174extern MyFrame* gFrame;
175extern ocpnStyle::StyleManager* g_StyleManager;
176extern options* g_pOptions;
177extern Multiplexer* g_pMUX;
178extern bool g_bShowChartBar;
179extern Routeman* g_pRouteMan;
180extern Select* pSelect;
181extern RouteManagerDialog* pRouteManagerDialog;
182extern RouteList* pRouteList;
183extern std::vector<Track*> g_TrackList;
184extern PlugInManager* g_pi_manager;
185extern s52plib* ps52plib;
186extern wxString ChartListFileName;
187extern options* g_options;
188extern ColorScheme global_color_scheme;
189extern wxArrayString g_locale_catalog_array;
190extern int g_GUIScaleFactor;
191extern int g_ChartScaleFactor;
192extern wxString g_locale;
193extern ocpnFloatingToolbarDialog* g_MainToolbar;
194
195extern int g_chart_zoom_modifier_raster;
196extern int g_chart_zoom_modifier_vector;
197extern double g_display_size_mm;
198extern bool g_bopengl;
199
200extern ChartGroupArray* g_pGroupArray;
201extern unsigned int g_canvasConfig;
202
203extern wxString g_CmdSoundString;
204
205extern unsigned int gs_plib_flags;
206extern ChartCanvas* g_focusCanvas;
207extern ChartCanvas* g_overlayCanvas;
208extern bool g_bquiting;
209
210WX_DEFINE_ARRAY_PTR(ChartCanvas*, arrayofCanvasPtr);
211extern arrayofCanvasPtr g_canvasArray;
212
213void NotifySetupOptionsPlugin(const PlugInData* pic);
214
215enum { CurlThreadId = wxID_HIGHEST + 1 };
216
217#include <wx/listimpl.cpp>
218WX_DEFINE_LIST(Plugin_WaypointList);
219WX_DEFINE_LIST(Plugin_HyperlinkList);
220
221wxDEFINE_EVENT(EVT_N0183_PLUGIN, ObservedEvt);
222wxDEFINE_EVENT(EVT_SIGNALK, ObservedEvt);
223
224static void SendAisJsonMessage(std::shared_ptr<const AisTargetData> pTarget) {
225 // Only send messages if someone is listening...
226 if (!GetJSONMessageTargetCount()) return;
227
228 // Do JSON message to all Plugin to inform of target
229 wxJSONValue jMsg;
230
231 wxLongLong t = ::wxGetLocalTimeMillis();
232
233 jMsg[wxS("Source")] = wxS("AisDecoder");
234 jMsg[wxT("Type")] = wxT("Information");
235 jMsg[wxT("Msg")] = wxS("AIS Target");
236 jMsg[wxT("MsgId")] = t.GetValue();
237 jMsg[wxS("lat")] = pTarget->Lat;
238 jMsg[wxS("lon")] = pTarget->Lon;
239 jMsg[wxS("sog")] = pTarget->SOG;
240 jMsg[wxS("cog")] = pTarget->COG;
241 jMsg[wxS("hdg")] = pTarget->HDG;
242 jMsg[wxS("mmsi")] = pTarget->MMSI;
243 jMsg[wxS("class")] = pTarget->Class;
244 jMsg[wxS("ownship")] = pTarget->b_OwnShip;
245 jMsg[wxS("active")] = pTarget->b_active;
246 jMsg[wxS("lost")] = pTarget->b_lost;
247 wxString l_ShipName = wxString::FromUTF8(pTarget->ShipName);
248 for (size_t i = 0; i < l_ShipName.Len(); i++) {
249 if (l_ShipName.GetChar(i) == '@') l_ShipName.SetChar(i, '\n');
250 }
251 jMsg[wxS("shipname")] = l_ShipName;
252 wxString l_CallSign = wxString::FromUTF8(pTarget->CallSign);
253 for (size_t i = 0; i < l_CallSign.Len(); i++) {
254 if (l_CallSign.GetChar(i) == '@') l_CallSign.SetChar(i, '\n');
255 }
256 jMsg[wxS("callsign")] = l_CallSign;
257 jMsg[wxS("removed")] = pTarget->b_removed;
258 SendJSONMessageToAllPlugins(wxT("AIS"), jMsg);
259}
260
261static int ComparePlugins(PlugInContainer** p1, PlugInContainer** p2) {
262 return (*p1)->Key().compare((*p2)->Key());
263}
264
265static SemanticVersion ParseVersion(const PluginMetadata& metadata) {
266 auto version = metadata.version;
267 // Handle tag versions like v1.2.3:
268 if (version[0] == 'v') version = version.substr(1);
269 return SemanticVersion::parse(version);
270}
271
272static bool IsUpdateAvailable(const PluginMetadata& metadata) {
273 auto imported_version = ParseVersion(metadata);
274 for (auto& md : PluginHandler::getInstance()->getAvailable()) {
275 if (md.name != metadata.name) continue;
276 if (md.is_imported) continue;
277 if (!PluginHandler::getInstance()->isCompatible(md)) continue;
278 if (ParseVersion(md) >= imported_version) return true;
279 }
280 return false;
281}
282
288public:
289 void message(const std::string& message) {
290 if (m_defer)
291 m_deferred_messages.push_back(message);
292 else
293 show_msg(message);
294 }
295
296 void show_deferred_messages() {
297 for (auto m : m_deferred_messages) show_msg(m);
298 m_defer = false;
299 }
300
301 BlacklistUI() : m_defer(true) {};
302
303private:
304 void show_msg(wxString msg) {
305 OCPNMessageBox(NULL, msg, wxString(_("OpenCPN Info")),
306 wxICON_INFORMATION | wxOK, 10); // 10 second timeout
307 }
308
309 bool m_defer; // defer dialogs until setup completed
310 std::vector<wxString> m_deferred_messages;
311};
312
313class PanelHardBreakWrapper : public wxTextWrapper {
314public:
315 PanelHardBreakWrapper(wxWindow* win, const wxString& text, int widthMax) {
316 m_lineCount = 0;
317 Wrap(win, text, widthMax);
318 }
319 wxString const& GetWrapped() const { return m_wrapped; }
320 int const GetLineCount() const { return m_lineCount; }
321 wxArrayString GetLineArray() { return m_array; }
322
323protected:
324 virtual void OnOutputLine(const wxString& line) {
325 m_wrapped += line;
326 m_array.Add(line);
327 }
328 virtual void OnNewLine() {
329 m_wrapped += '\n';
330 m_lineCount++;
331 }
332
333private:
334 wxString m_wrapped;
335 int m_lineCount;
336 wxArrayString m_array;
337};
338
340 template <typename T>
341 std::size_t operator()(T t) const {
342 return static_cast<std::size_t>(t);
343 }
344};
345
346wxString message_by_status(PluginStatus stat) {
347 switch (stat) {
348 case PluginStatus::System:
349 return _("Plugin is a standard system plugin");
350 case PluginStatus::Managed:
351 return _("Plugin is managed by OpenCPN");
352 case PluginStatus::Unmanaged:
353 return _("Plugin is not managed by OpenCPN");
354 case PluginStatus::Ghost:
355 return ("");
356 case PluginStatus::Unknown:
357 return _("Plugin status unknown");
358 case PluginStatus::LegacyUpdateAvailable:
359 return _("Update to managed Plugin is available");
360 case PluginStatus::ManagedInstallAvailable:
361 return _("New managed Plugin installation available");
362 case PluginStatus::ManagedInstalledUpdateAvailable:
363 return _("Update to installed Plugin is available");
364 case PluginStatus::ManagedInstalledCurrentVersion:
365 return _("Plugin is latest available");
366 case PluginStatus::Imported:
367 return _("Plugin is imported");
368 case PluginStatus::ManagedInstalledDowngradeAvailable:
369 return ("");
370 case PluginStatus::PendingListRemoval:
371 return ("");
372 default:
373 return ("");
374 }
375}
376
377static const std::unordered_map<PluginStatus, const char*, EnumClassHash>
378 icon_by_status(
379 {{PluginStatus::System, "emblem-default.svg"},
380 {PluginStatus::Managed, "emblem-default.svg"},
381 {PluginStatus::Unmanaged, "emblem-unmanaged.svg"},
382 {PluginStatus::Ghost, "ghost.svg"},
383 {PluginStatus::Unknown, "emblem-unmanaged.svg"},
384 {PluginStatus::LegacyUpdateAvailable, "emblem-legacy-update.svg"},
385 {PluginStatus::ManagedInstallAvailable, "emblem-download.svg"},
386 {PluginStatus::ManagedInstalledUpdateAvailable,
387 "emblem-legacy-update.svg"},
388 {PluginStatus::ManagedInstalledCurrentVersion, "emblem-default.svg"},
389 {PluginStatus::ManagedInstalledDowngradeAvailable,
390 "emblem-default.svg"},
391 {PluginStatus::PendingListRemoval, "emblem-default.svg"},
392 {PluginStatus::Imported, "emblem-default.svg"}});
393
394static const std::unordered_map<PluginStatus, const char*, EnumClassHash>
395 literalstatus_by_status(
396 {{PluginStatus::System, "System"},
397 {PluginStatus::Managed, "Managed"},
398 {PluginStatus::Unmanaged, "Unmanaged"},
399 {PluginStatus::Ghost, "Ghost"},
400 {PluginStatus::Unknown, "Unknown"},
401 {PluginStatus::LegacyUpdateAvailable, "LegacyUpdateAvailable"},
402 {PluginStatus::ManagedInstallAvailable, "ManagedInstallAvailable"},
403 {PluginStatus::ManagedInstalledUpdateAvailable,
404 "ManagedInstalledUpdateAvailable"},
405 {PluginStatus::ManagedInstalledCurrentVersion,
406 "ManagedInstalledCurrentVersion"},
407 {PluginStatus::ManagedInstalledDowngradeAvailable,
408 "ManagedInstalledDowngradeAvailable"},
409 {PluginStatus::PendingListRemoval, "PendingListRemoval"},
410 {PluginStatus::Imported, "Imported"}
411
412 });
413
418static std::vector<PluginMetadata> getCompatiblePlugins() {
420 struct metadata_compare {
421 bool operator()(const PluginMetadata& lhs,
422 const PluginMetadata& rhs) const {
423 return lhs.key() < rhs.key();
424 }
425 };
426
427 std::vector<PluginMetadata> returnArray;
428
429 std::set<PluginMetadata, metadata_compare> unique_plugins;
430 for (auto plugin : PluginHandler::getInstance()->getAvailable()) {
431 unique_plugins.insert(plugin);
432 }
433 for (auto plugin : unique_plugins) {
434 if (PluginHandler::isCompatible(plugin)) {
435 returnArray.push_back(plugin);
436 }
437 }
438 return returnArray;
439}
440
441static SemanticVersion metadata_version(const PluginMetadata pm) {
442 return SemanticVersion::parse(pm.name);
443}
444
445// Get installed version from manifest for given plugin. For
446// older plugins this contains more detailed info then the
447// plugin API. From API level 117 the API should contain the
448// same info.
449//
450// TODO: Get version from API for api level 117+
451SemanticVersion getInstalledVersion(const std::string name) {
452 std::string installed;
453 std::string path = PluginHandler::versionPath(name);
454 if (path == "" || !wxFileName::IsFileReadable(path)) {
455 return SemanticVersion(-1, -1);
456 }
457 std::ifstream stream;
458 stream.open(path, std::ifstream::in);
459 stream >> installed;
460 return SemanticVersion::parse(installed);
461}
462
467static std::vector<PluginMetadata> getUpdates(const char* name) {
468 auto updates = getCompatiblePlugins();
469 updates.erase(
470 std::remove_if(updates.begin(), updates.end(),
471 [&](const PluginMetadata m) { return m.name != name; }),
472 updates.end());
473
474 auto inst_vers = getInstalledVersion(name);
475 if (inst_vers.major == -1) {
476 return updates;
477 }
478
479 // Drop already installed plugin, it has its own update options.
480 updates.erase(std::remove_if(updates.begin(), updates.end(),
481 [&](const PluginMetadata m) {
482 return metadata_version(m) == inst_vers;
483 }),
484 updates.end());
485 return updates;
486}
487
489static void gui_uninstall(const PlugInData* pic, const char* plugin) {
490 g_Platform->ShowBusySpinner();
491 PluginLoader::getInstance()->DeactivatePlugIn(*pic);
492 PluginLoader::getInstance()->SetEnabled(pic->m_common_name, false);
493 PluginLoader::getInstance()->UpdatePlugIns();
494
495 wxLogMessage("Uninstalling %s", plugin);
496 PluginHandler::getInstance()->uninstall(plugin);
497 PluginLoader::getInstance()->UpdatePlugIns();
498 g_Platform->HideBusySpinner();
499}
500
501static bool LoadAllPlugIns(bool load_enabled, bool keep_orphans = false) {
502 g_Platform->ShowBusySpinner();
503 bool b =
504 PluginLoader::getInstance()->LoadAllPlugIns(load_enabled, keep_orphans);
505 g_Platform->HideBusySpinner();
506 return b;
507}
508
510static void UninstallPlugin(const std::string& name) {
511 auto handler = PluginHandler::getInstance();
512 auto loader = PluginLoader::getInstance();
513 auto finder = [name](const PluginMetadata pm) { return pm.name == name; };
514 const auto& installed = handler->getInstalled();
515 auto found = std::find_if(installed.begin(), installed.end(), finder);
516 if (found != installed.end()) {
517 for (size_t i = 0; i < loader->GetPlugInArray()->GetCount(); i++) {
518 auto const& item = loader->GetPlugInArray()->Item(i);
519 if (item->m_common_name.ToStdString() == name) {
520 DEBUG_LOG << "Unloading plugin: " << name;
521 loader->UnLoadPlugIn(i);
522 break;
523 }
524 }
525 handler->uninstall(found->name);
526 DEBUG_LOG << "Uninstalling: " << found->name;
527 }
528}
529
530static void run_update_dialog(PluginListPanel* parent, const PlugInData* pic,
531 bool uninstall, const char* name = 0,
532 bool b_forceEnable = false) {
533 wxString pluginName = pic->m_common_name;
534 const char* plugin = name == 0 ? pic->m_common_name.mb_str().data() : name;
535 auto updates = getUpdates(plugin);
536 auto parent_dlg = dynamic_cast<wxScrolledWindow*>(parent);
537 wxASSERT(parent_dlg != 0);
538 UpdateDialog dialog(parent_dlg, updates);
539 auto status = dialog.ShowModal();
540 status = dialog.GetReturnCode();
541 if (status != wxID_OK) {
542 return;
543 }
544
545 auto update = dialog.GetUpdate();
546 if (!g_pi_manager->CheckBlacklistedPlugin(update)) {
547 return;
548 }
549
550 wxLogMessage("Installing %s", update.name.c_str());
551
552 auto pluginHandler = PluginHandler::getInstance();
553 auto path = ocpn::lookup_tarball(update.tarball_url.c_str());
554 if (uninstall && path != "") {
555 gui_uninstall(pic, update.name.c_str());
556 }
557 bool cacheResult = pluginHandler->installPluginFromCache(update);
558
559 if (!cacheResult) {
560 g_Platform->ShowBusySpinner(); // Will be cancelled in downloader->run()
561 wxYield();
562
563 auto downloader = new GuiDownloader(parent_dlg, update);
564 std::string tempTarballPath = downloader->run(parent_dlg, uninstall);
565
566 if (!tempTarballPath.size()) // Error, dialog already presented
567 return;
568
569 // Provisional error check
570 bool bOK = true;
571 std::string manifestPath = PluginHandler::fileListPath(update.name);
572 if (!isRegularFile(manifestPath.c_str())) {
573 wxLogMessage("Installation of %s failed", update.name.c_str());
574 PluginHandler::cleanupFiles(manifestPath, update.name);
575 bOK = false;
576 }
577
578 // On successful installation, copy the temp tarball to the local cache
579 if (bOK) {
580 wxLogMessage("Installation of %s successful", update.name.c_str());
581 wxURI uri(wxString(update.tarball_url.c_str()));
582 wxFileName fn(uri.GetPath());
583 std::string basename = fn.GetFullName().ToStdString();
584
585 if (ocpn::store_tarball(tempTarballPath.c_str(), basename.c_str())) {
586 wxLogDebug("Copied %s to local cache at %s", tempTarballPath.c_str(),
587 basename);
588 remove(tempTarballPath.c_str());
589 }
590 }
591 }
592
593 // Check the library compatibility of the subject plugin
594 // Find the plugin library file, looking for "_pi.{dll/so/dylib file}
595#ifdef __WXMSW__
596 wxString pispec = _T("_pi.dll");
597#elif defined(__WXOSX__)
598 wxString pispec = _T("_pi.dylib");
599#else
600 wxString pispec = _T("_pi.so");
601#endif
602
603 std::string manifestPath = PluginHandler::fileListPath(update.name);
604 wxTextFile manifest_file(manifestPath);
605 wxString pluginFile;
606 if (manifest_file.Open()) {
607 wxString val;
608 for (wxString str = manifest_file.GetFirstLine(); !manifest_file.Eof();
609 str = manifest_file.GetNextLine()) {
610 if (str.Contains(pispec)) {
611 if (getenv("OCPN_KEEP_PLUGINS")) {
612 // Undocumented debug hook
613 continue;
614 }
615 auto loader = PluginLoader::getInstance();
616 if (!loader->CheckPluginCompatibility(str)) {
617 wxString msg =
618 _("The plugin is not compatible with this version of OpenCPN, "
619 "and will be uninstalled.");
620 OCPNMessageBox(NULL, msg, wxString(_("OpenCPN Info")),
621 wxICON_INFORMATION | wxOK, 10);
622
623 PluginHandler::cleanupFiles(manifestPath, update.name);
624 } else {
625 pluginFile = str;
626 }
627 break;
628 }
629 }
630 }
631
632 if (b_forceEnable && pluginFile.Length()) {
633 wxString config_section = (_T ( "/PlugIns/" ));
634 wxFileName fn(pluginFile);
635 config_section += fn.GetFullName();
636 pConfig->SetPath(config_section);
637 pConfig->Write(_T ( "bEnabled" ), true);
638 }
639
640 // This is installed from catalog, remove possible imported
641 // metadata leftovers
642 auto handler = PluginHandler::getInstance();
643 std::remove(handler->ImportedMetadataPath(update.name).c_str());
644
645 // Reload all plugins, which will bring in the action results.
646 LoadAllPlugIns(false);
647
648 parent->ReloadPluginPanels();
649}
650
651static PlugIn_ViewPort CreatePlugInViewport(const ViewPort& vp) {
652 // Create a PlugIn Viewport
653 ViewPort tvp = vp;
654 PlugIn_ViewPort pivp;
655
656 pivp.clat = tvp.clat; // center point
657 pivp.clon = tvp.clon;
659 pivp.skew = tvp.skew;
660 pivp.rotation = tvp.rotation;
661 pivp.chart_scale = tvp.chart_scale;
662 pivp.pix_width = tvp.pix_width;
663 pivp.pix_height = tvp.pix_height;
664 pivp.rv_rect = tvp.rv_rect;
665 pivp.b_quilt = tvp.b_quilt;
666 pivp.m_projection_type = tvp.m_projection_type;
667
668 pivp.lat_min = tvp.GetBBox().GetMinLat();
669 pivp.lat_max = tvp.GetBBox().GetMaxLat();
670 pivp.lon_min = tvp.GetBBox().GetMinLon();
671 pivp.lon_max = tvp.GetBBox().GetMaxLon();
672
673 pivp.bValid = tvp.IsValid(); // This VP is valid
674
675 return pivp;
676}
677
678static ViewPort CreateCompatibleViewport(const PlugIn_ViewPort& pivp) {
679 // Create a system ViewPort
680 ViewPort vp;
681
682 vp.clat = pivp.clat; // center point
683 vp.clon = pivp.clon;
685 vp.skew = pivp.skew;
686 vp.rotation = pivp.rotation;
687 vp.chart_scale = pivp.chart_scale;
688 vp.pix_width = pivp.pix_width;
689 vp.pix_height = pivp.pix_height;
690 vp.rv_rect = pivp.rv_rect;
691 vp.b_quilt = pivp.b_quilt;
692 vp.m_projection_type = pivp.m_projection_type;
693
694 if (gFrame->GetPrimaryCanvas())
695 vp.ref_scale = gFrame->GetPrimaryCanvas()->GetVP().ref_scale;
696 else
697 vp.ref_scale = vp.chart_scale;
698
699 vp.SetBoxes();
700 vp.Validate(); // This VP is valid
701
702 return vp;
703}
704
705class pluginUtilHandler : public wxEvtHandler {
706public:
709
710 void OnPluginUtilAction(wxCommandEvent& event);
711
712 DECLARE_EVENT_TABLE()
713};
714
715BEGIN_EVENT_TABLE(pluginUtilHandler, wxEvtHandler)
716EVT_BUTTON(ID_CMD_BUTTON_PERFORM_ACTION, pluginUtilHandler::OnPluginUtilAction)
717END_EVENT_TABLE()
718
720
721void pluginUtilHandler::OnPluginUtilAction(wxCommandEvent& event) {
722 auto panel = static_cast<PluginPanel*>(event.GetClientData());
723 PluginListPanel* plugin_list_panel =
724 dynamic_cast<PluginListPanel*>(panel->GetParent());
725 wxASSERT(plugin_list_panel != 0);
726
727 auto actionPIC = panel->GetPlugin();
728 wxString name = actionPIC->m_common_name;
729
730 // Perform the indicated action according to the verb...
731 switch (panel->GetAction()) {
732 case ActionVerb::UPGRADE_TO_MANAGED_VERSION: {
733 auto loader = PluginLoader::getInstance();
734
735 // capture the plugin name
736 std::string pluginName = actionPIC->m_managed_metadata.name;
737
738 wxLogMessage("Installing managed plugin: %s", pluginName.c_str());
739 auto downloader =
740 new GuiDownloader(plugin_list_panel, actionPIC->m_managed_metadata);
741 downloader->run(plugin_list_panel, false);
742
743 // Provisional error check
744 std::string manifestPath = PluginHandler::fileListPath(pluginName);
745 if (isRegularFile(manifestPath.c_str())) {
746 // dynamically deactivate the legacy plugin, making way for the upgrade.
747 for (unsigned i = 0; i < loader->GetPlugInArray()->GetCount(); i += 1) {
748 if (actionPIC->m_managed_metadata.name ==
749 loader->GetPlugInArray()->Item(i)->m_common_name.ToStdString()) {
750 loader->UnLoadPlugIn(i);
751 break;
752 }
753 }
754
755 // Reload all plugins, which will bring in the new, managed version.
756 LoadAllPlugIns(false);
757 } else {
758 PluginHandler::cleanupFiles(manifestPath,
759 actionPIC->m_managed_metadata.name);
760 }
761 plugin_list_panel->ReloadPluginPanels();
762 plugin_list_panel->SelectByName(name);
763
764 break;
765 }
766
767 case ActionVerb::UPDATE_IMPORTED_VERSION:
768 case ActionVerb::UPGRADE_INSTALLED_MANAGED_VERSION:
769 case ActionVerb::REINSTALL_MANAGED_VERSION:
770 case ActionVerb::DOWNGRADE_INSTALLED_MANAGED_VERSION: {
771 // Grab a copy of the managed metadata
772 auto metaSave = actionPIC->m_managed_metadata;
773 run_update_dialog(plugin_list_panel, actionPIC, true,
774 metaSave.name.c_str());
775 break;
776 }
777
778 case ActionVerb::INSTALL_MANAGED_VERSION: {
779 wxLogMessage("Installing new managed plugin.");
780 run_update_dialog(plugin_list_panel, actionPIC, false);
781 break;
782 }
783
784 case ActionVerb::UNINSTALL_MANAGED_VERSION: {
785 PluginLoader::getInstance()->DeactivatePlugIn(*actionPIC);
786
787 // Capture the confirmation dialog contents before the plugin goes away
788 wxString message;
789 message.Printf("%s %s\n", actionPIC->m_managed_metadata.name.c_str(),
790 actionPIC->m_managed_metadata.version.c_str());
791 message += _("successfully un-installed");
792
793 wxLogMessage("Uninstalling %s",
794 actionPIC->m_managed_metadata.name.c_str());
795
796 PluginHandler::getInstance()->uninstall(
797 actionPIC->m_managed_metadata.name);
798
799 // Reload all plugins, which will bring in the action results.
800 auto loader = PluginLoader::getInstance();
801 LoadAllPlugIns(false);
802 plugin_list_panel->ReloadPluginPanels();
803
804 OCPNMessageBox(gFrame, message, _("Un-Installation complete"),
805 wxICON_INFORMATION | wxOK);
806
807 break;
808 }
809
810 case ActionVerb::NOP:
811 default:
812 break;
813 }
814}
815
816//------------------------------------------------------------------------------
817// NMEA Event Implementation
818// PlugIn Messaging scheme Event
819//------------------------------------------------------------------------------
820
821const wxEventType wxEVT_OCPN_MSG = wxNewEventType();
822
823OCPN_MsgEvent::OCPN_MsgEvent(wxEventType commandType, int id)
824 : wxEvent(id, commandType) {}
825
826OCPN_MsgEvent::~OCPN_MsgEvent() {}
827
828wxEvent* OCPN_MsgEvent::Clone() const {
829 OCPN_MsgEvent* newevent = new OCPN_MsgEvent(*this);
830 newevent->m_MessageID =
831 this->m_MessageID
832 .c_str(); // this enforces a deep copy of the string data
833 newevent->m_MessageText = this->m_MessageText.c_str();
834 return newevent;
835}
836
837//------------------------------------------------------------------------------------------------
838//
839// The PlugInToolbarToolContainer Implementation
840//
841//------------------------------------------------------------------------------------------------
842PlugInToolbarToolContainer::PlugInToolbarToolContainer() {
843 bitmap_dusk = NULL;
844 bitmap_night = NULL;
845 bitmap_day = NULL;
846 bitmap_Rollover_day = NULL;
847 bitmap_Rollover_dusk = NULL;
848 bitmap_Rollover_night = NULL;
849}
850
851PlugInToolbarToolContainer::~PlugInToolbarToolContainer() {
852 delete bitmap_dusk;
853 delete bitmap_night;
854 delete bitmap_day;
855 delete bitmap_Rollover_day;
856 delete bitmap_Rollover_dusk;
857 delete bitmap_Rollover_night;
858}
859
860//-----------------------------------------------------------------------------------------------------
861//
862// The PlugIn Manager Implementation
863//
864//-----------------------------------------------------------------------------------------------------
865PlugInManager* s_ppim;
866
867BEGIN_EVENT_TABLE(PlugInManager, wxEvtHandler)
868#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
869EVT_CURL_END_PERFORM(CurlThreadId, PlugInManager::OnEndPerformCurlDownload)
870EVT_CURL_DOWNLOAD(CurlThreadId, PlugInManager::OnCurlDownload)
871#endif
872END_EVENT_TABLE()
873
874static void event_message_box(const wxString& msg) {
875 OCPNMessageBox(NULL, msg, wxString(_("OpenCPN Info")),
876 wxICON_INFORMATION | wxOK, 0); // no timeout
877}
878
879static void OnLoadPlugin(const PlugInContainer* pic) {
880 if (g_options) {
881 if ((pic->m_cap_flag & INSTALLS_TOOLBOX_PAGE)) {
882 if (!pic->m_toolbox_panel) NotifySetupOptionsPlugin(pic);
883 }
884 }
885}
886
887PlugInManager::PlugInManager(MyFrame* parent) {
888#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
889 m_pCurlThread = NULL;
890 m_pCurl = 0;
891#endif
892 pParent = parent;
893 s_ppim = this;
894
895 MyFrame* pFrame = GetParentFrame();
896 if (pFrame) {
897 m_plugin_menu_item_id_next = CanvasMenuHandler::GetNextContextMenuId();
898 m_plugin_tool_id_next = pFrame->GetNextToolbarToolId();
899 }
900
901#ifdef __ANDROID__
902 // Due to the oddball mixed static/dynamic linking model used in the Android
903 // architecture, all classes used in PlugIns must be present in the core,
904 // even if stubs.
905 //
906 // Here is where we do that....
907 if (pFrame) {
908 wxArrayString as;
909 as.Add(_T("Item0"));
910 wxRadioBox* box =
911 new wxRadioBox(pFrame, -1, _T(""), wxPoint(0, 0), wxSize(-1, -1), as);
912 delete box;
913 }
914
915#endif
916
917#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
918 wxCurlBase::Init();
919 m_last_online = false;
920 m_last_online_chk = -1;
921#endif
922
923 m_utilHandler = new pluginUtilHandler();
924 m_listPanel = NULL;
925 m_blacklist = blacklist_factory();
926 m_blacklist_ui = std::unique_ptr<BlacklistUI>(new BlacklistUI());
927
928 wxDEFINE_EVENT(EVT_JSON_TO_ALL_PLUGINS, ObservedEvt);
929 evt_json_to_all_plugins_listener.Listen(g_pRouteMan->json_msg, this,
930 EVT_JSON_TO_ALL_PLUGINS);
931 Bind(EVT_JSON_TO_ALL_PLUGINS, [&](ObservedEvt& ev) {
932 auto json = std::static_pointer_cast<const wxJSONValue>(ev.GetSharedPtr());
933 SendJSONMessageToAllPlugins(ev.GetString(), *json);
934 });
935
936 wxDEFINE_EVENT(EVT_LEGINFO_TO_ALL_PLUGINS, ObservedEvt);
937 evt_routeman_leginfo_listener.Listen(g_pRouteMan->json_leg_info, this,
938 EVT_LEGINFO_TO_ALL_PLUGINS);
939 Bind(EVT_LEGINFO_TO_ALL_PLUGINS, [&](ObservedEvt& ev) {
940 auto ptr = UnpackEvtPointer<ActiveLegDat>(ev);
941 SendActiveLegInfoToAllPlugIns(ptr.get());
942 });
943
944 HandlePluginLoaderEvents();
945 InitCommListeners();
946 auto msg_sent_action = [](ObservedEvt ev) {
947 SendNMEASentenceToAllPlugIns(ev.GetString());
948 };
949 m_on_msg_sent_listener.Init(g_pRouteMan->on_message_sent, msg_sent_action);
950}
951PlugInManager::~PlugInManager() {
952#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
953 wxCurlBase::Shutdown();
954#endif
955 delete m_utilHandler;
956}
957
958void PlugInManager::InitCommListeners(void) {
959 // Initialize the comm listener to support
960 // void SetNMEASentence(wxString &sentence);
961
962 auto& msgbus = NavMsgBus::GetInstance();
963
964 m_listener_N0183_all.Listen(Nmea0183Msg::MessageKey("ALL"), this,
965 EVT_N0183_PLUGIN);
966 Bind(EVT_N0183_PLUGIN, [&](ObservedEvt ev) {
967 auto ptr = ev.GetSharedPtr();
968 auto n0183_msg = std::static_pointer_cast<const Nmea0183Msg>(ptr);
969 HandleN0183(n0183_msg);
970 });
971
972 SignalkMsg sk_msg;
973 m_listener_SignalK.Listen(sk_msg, this, EVT_SIGNALK);
974
975 Bind(EVT_SIGNALK, [&](ObservedEvt ev) {
976 HandleSignalK(UnpackEvtPointer<SignalkMsg>(ev));
977 });
978}
979
980void PlugInManager::HandleN0183(std::shared_ptr<const Nmea0183Msg> n0183_msg) {
981 std::string s = n0183_msg->payload;
982 wxString sentence(s.c_str());
983
984 if (s[0] == '$') {
985 const auto& drivers = CommDriverRegistry::GetInstance().GetDrivers();
986 auto& target_driver = FindDriver(drivers, n0183_msg->source->iface);
987
988 bool bpass_input_filter = true;
989
990 // Get the params for the driver sending this message
991 ConnectionParams params;
992 auto drv_serial = dynamic_cast<CommDriverN0183Serial*>(target_driver.get());
993 if (drv_serial) {
994 params = drv_serial->GetParams();
995 } else {
996 auto drv_net = dynamic_cast<CommDriverN0183Net*>(target_driver.get());
997 if (drv_net) {
998 params = drv_net->GetParams();
999 }
1000 }
1001
1002 // Check to see if the message passes the source's input filter
1003 bpass_input_filter = params.SentencePassesFilter(sentence, FILTER_INPUT);
1004
1005 if (bpass_input_filter) SendNMEASentenceToAllPlugIns(sentence);
1006 } else if (s[0] == '!') {
1007 // printf("AIS to all: %s", s.c_str());
1008 SendAISSentenceToAllPlugIns(sentence);
1009 }
1010}
1011
1012void PlugInManager::HandleSignalK(std::shared_ptr<const SignalkMsg> sK_msg) {
1013 g_ownshipMMSI_SK = sK_msg->context_self;
1014
1015 wxJSONReader jsonReader;
1016 wxJSONValue root;
1017
1018 std::string msgTerminated = sK_msg->raw_message;
1019 ;
1020
1021 int errors = jsonReader.Parse(msgTerminated, &root);
1022 if (errors == 0) SendJSONMessageToAllPlugins(wxT("OCPN_CORE_SIGNALK"), root);
1023}
1024
1030wxDEFINE_EVENT(EVT_PLUGMGR_AIS_MSG, ObservedEvt);
1031wxDEFINE_EVENT(EVT_PLUGMGR_ROUTEMAN_MSG, ObservedEvt);
1032wxDEFINE_EVENT(EVT_BLACKLISTED_PLUGIN, wxCommandEvent);
1033wxDEFINE_EVENT(EVT_LOAD_DIRECTORY, wxCommandEvent);
1034wxDEFINE_EVENT(EVT_LOAD_PLUGIN, wxCommandEvent);
1035wxDEFINE_EVENT(EVT_PLUGIN_UNLOAD, wxCommandEvent);
1036wxDEFINE_EVENT(EVT_PLUGLIST_CHANGE, wxCommandEvent);
1037wxDEFINE_EVENT(EVT_UPDATE_CHART_TYPES, wxCommandEvent);
1038wxDEFINE_EVENT(EVT_PLUGIN_LOADALL_FINALIZE, wxCommandEvent);
1039
1040void PlugInManager::HandlePluginLoaderEvents() {
1041 auto loader = PluginLoader::getInstance();
1042
1043 evt_blacklisted_plugin_listener.Listen(loader->evt_blacklisted_plugin, this,
1044 EVT_BLACKLISTED_PLUGIN);
1045 Bind(EVT_BLACKLISTED_PLUGIN, [&](wxCommandEvent& ev) {
1046 m_blacklist_ui->message(ev.GetString().ToStdString());
1047 });
1048
1049 loader->SetOnDeactivateCb(
1050 [&](const PlugInContainer* pic) { OnPluginDeactivate(pic); });
1051 evt_pluglist_change_listener.Listen(loader->evt_pluglist_change, this,
1052 EVT_PLUGLIST_CHANGE);
1053 Bind(EVT_PLUGLIST_CHANGE, [&](wxCommandEvent&) {
1054 if (m_listPanel) m_listPanel->ReloadPluginPanels();
1055 if (g_options) g_options->itemBoxSizerPanelPlugins->Layout();
1056 });
1057
1058 evt_load_directory_listener.Listen(loader->evt_load_directory, this,
1059 EVT_LOAD_DIRECTORY);
1060 Bind(EVT_LOAD_DIRECTORY, [&](wxCommandEvent&) {
1061 pConfig->SetPath("/PlugIns/");
1062 SetPluginOrder(pConfig->Read("PluginOrder", wxEmptyString));
1063 });
1064
1065 evt_load_plugin_listener.Listen(loader->evt_load_plugin, this,
1066 EVT_LOAD_PLUGIN);
1067 Bind(EVT_LOAD_PLUGIN, [&](wxCommandEvent& ev) {
1068 auto pic = static_cast<const PlugInContainer*>(ev.GetClientData());
1069 OnLoadPlugin(pic);
1070 });
1071
1072 evt_update_chart_types_listener.Listen(loader->evt_update_chart_types, this,
1073 EVT_UPDATE_CHART_TYPES);
1074 Bind(EVT_UPDATE_CHART_TYPES,
1075 [&](wxCommandEvent& ev) { UpDateChartDataTypes(); });
1076
1077 evt_plugin_loadall_finalize_listener.Listen(
1078 loader->evt_plugin_loadall_finalize, this, EVT_PLUGIN_LOADALL_FINALIZE);
1079 Bind(EVT_PLUGIN_LOADALL_FINALIZE,
1080 [&](wxCommandEvent& ev) { FinalizePluginLoadall(); });
1081
1082 evt_ais_json_listener.Listen(g_pAIS->plugin_msg, this, EVT_PLUGMGR_AIS_MSG);
1083 evt_routeman_json_listener.Listen(g_pRouteMan->json_msg, this,
1084 EVT_PLUGMGR_ROUTEMAN_MSG);
1085 Bind(EVT_PLUGMGR_AIS_MSG, [&](ObservedEvt& ev) {
1086 auto pTarget = UnpackEvtPointer<AisTargetData>(ev);
1087 SendAisJsonMessage(pTarget);
1088 });
1089 Bind(EVT_PLUGMGR_ROUTEMAN_MSG, [&](ObservedEvt& ev) {
1090 auto msg = UnpackEvtPointer<wxJSONValue>(ev);
1091 SendJSONMessageToAllPlugins(ev.GetString(), *msg);
1092 });
1093}
1094
1099wxDEFINE_EVENT(EVT_DOWNLOAD_FAILED, wxCommandEvent);
1100wxDEFINE_EVENT(EVT_DOWNLOAD_OK, wxCommandEvent);
1101
1102void PlugInManager::HandlePluginHandlerEvents() {
1103 auto loader = PluginLoader::getInstance();
1104
1105 evt_download_failed_listener.Listen(loader->evt_update_chart_types, this,
1106 EVT_DOWNLOAD_FAILED);
1107 Bind(EVT_DOWNLOAD_FAILED, [&](wxCommandEvent& ev) {
1108 wxString message = _("Please check system log for more info.");
1109 OCPNMessageBox(gFrame, message, _("Installation error"),
1110 wxICON_ERROR | wxOK | wxCENTRE);
1111 });
1112
1113 evt_download_ok_listener.Listen(loader->evt_update_chart_types, this,
1114 EVT_DOWNLOAD_OK);
1115 Bind(EVT_DOWNLOAD_OK, [&](wxCommandEvent& ev) {
1116 wxString message(ev.GetString());
1117 message += _(" successfully installed from cache");
1118 OCPNMessageBox(gFrame, message, _("Installation complete"),
1119 wxICON_INFORMATION | wxOK | wxCENTRE);
1120 });
1121}
1122
1123bool PlugInManager::CallLateInit(void) {
1124 bool bret = true;
1125
1126 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1127 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1128 PlugInContainer* pic = (*plugin_array)[i];
1129
1130 switch (pic->m_api_version) {
1131 case 110:
1132 case 111:
1133 case 112:
1134 case 113:
1135 case 114:
1136 case 115:
1137 case 116:
1138 case 117:
1139 case 118:
1140 case 119:
1141 ProcessLateInit(pic);
1142 break;
1143 }
1144 }
1145
1146 return bret;
1147}
1148
1149void PlugInManager::ProcessLateInit(const PlugInContainer* pic) {
1150 if (pic->m_cap_flag & WANTS_LATE_INIT) {
1151 wxString msg("PlugInManager: Calling LateInit PlugIn: ");
1152 msg += pic->m_plugin_file;
1153 wxLogMessage(msg);
1154
1155 opencpn_plugin_110* ppi = dynamic_cast<opencpn_plugin_110*>(pic->m_pplugin);
1156 if (ppi) ppi->LateInit();
1157 }
1158}
1159
1160void PlugInManager::OnPluginDeactivate(const PlugInContainer* pic) {
1161 // Unload chart cache if this plugin is responsible for any charts
1162 if ((pic->m_cap_flag & INSTALLS_PLUGIN_CHART) ||
1164 ChartData->PurgeCachePlugins();
1165 gFrame->InvalidateAllQuilts();
1166 }
1167 // Deactivate (Remove) any ToolbarTools added by this PlugIn
1168 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
1169 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
1170
1171 if (pttc->m_pplugin == pic->m_pplugin) {
1172 m_PlugInToolbarTools.Remove(pttc);
1173 delete pttc;
1174 }
1175 }
1176
1177 // Deactivate (Remove) any ContextMenu items addded by this PlugIn
1178 for (unsigned int i = 0; i < m_PlugInMenuItems.GetCount(); i++) {
1179 PlugInMenuItemContainer* pimis = m_PlugInMenuItems[i];
1180 if (pimis->m_pplugin == pic->m_pplugin) {
1181 m_PlugInMenuItems.Remove(pimis);
1182 delete pimis;
1183 }
1184 }
1185}
1186
1187bool PlugInManager::IsAnyPlugInChartEnabled() {
1188 // Is there a PlugIn installed and active that implements PlugIn Chart
1189 // type(s)?
1190 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1191 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1192 PlugInContainer* pic = (*plugin_array)[i];
1193 if (pic->m_enabled && pic->m_init_state) {
1194 if ((pic->m_cap_flag & INSTALLS_PLUGIN_CHART) ||
1196 return true;
1197 }
1198 }
1199 return false;
1200}
1201
1202void PlugInManager::UpdateManagedPlugins() {
1203 PluginLoader::getInstance()->UpdateManagedPlugins(false);
1204 PluginLoader::getInstance()->SortPlugins(ComparePlugins);
1205
1206 if (m_listPanel) m_listPanel->ReloadPluginPanels();
1207 g_options->itemBoxSizerPanelPlugins->Layout();
1208}
1209
1210bool PlugInManager::UpDateChartDataTypes() {
1211 bool bret = false;
1212 if (NULL == ChartData) return bret;
1213
1214 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1215 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1216 PlugInContainer* pic = plugin_array->Item(i);
1217
1218 if (pic->m_init_state) {
1219 if ((pic->m_cap_flag & INSTALLS_PLUGIN_CHART) ||
1221 bret = true;
1222 }
1223 }
1224
1225 if (bret) ChartData->UpdateChartClassDescriptorArray();
1226
1227 return bret;
1228}
1229
1230void PlugInManager::FinalizePluginLoadall() {
1231 // FIXME
1232 // Maybe this does not need to be done for CLI instance?
1233 // Inform plugins of the current color scheme
1234 SetColorSchemeForAllPlugIns(global_color_scheme);
1235
1236 // Tell all the PlugIns about the current OCPN configuration
1237 SendBaseConfigToAllPlugIns();
1238 SendS52ConfigToAllPlugIns(true);
1239 SendSKConfigToAllPlugIns();
1240
1241 // Inform Plugins of OpenGL configuration, if enabled
1242#ifdef ocpnUSE_GL
1243 if (g_bopengl) {
1244 if (gFrame->GetPrimaryCanvas()->GetglCanvas())
1245 gFrame->GetPrimaryCanvas()->GetglCanvas()->SendJSONConfigMessage();
1246 }
1247#endif
1248
1249 // And then reload all catalogs.
1250 ReloadLocale();
1251}
1252
1253void PlugInManager::SetPluginOrder(wxString serialized_names) {
1254 m_plugin_order.Empty();
1255 wxStringTokenizer tokenizer(serialized_names, ";");
1256 while (tokenizer.HasMoreTokens()) {
1257 m_plugin_order.Add(tokenizer.GetNextToken());
1258 }
1259}
1260
1261wxString PlugInManager::GetPluginOrder() {
1262 wxString plugins = wxEmptyString;
1263 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1264 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1265 plugins.Append(plugin_array->Item(i)->m_common_name);
1266 if (i < plugin_array->GetCount() - 1) plugins.Append(';');
1267 }
1268 return plugins;
1269}
1270
1271bool PlugInManager::UpdateConfig() {
1272 // pConfig->SetPath( _T("/PlugIns/") );
1273 // pConfig->Write( _T("PluginOrder"), GetPluginOrder() );
1274
1275 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1276 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1277 PlugInContainer* pic = plugin_array->Item(i);
1278
1279 if (pic) {
1280 wxString config_section = (_T ( "/PlugIns/" ));
1281 config_section += pic->m_plugin_filename;
1282 pConfig->SetPath(config_section);
1283 pConfig->Write(_T ( "bEnabled" ), pic->m_enabled);
1284 }
1285 }
1286
1287 return true;
1288}
1289
1290void PlugInManager::ShowDeferredBlacklistMessages() {
1291 m_blacklist_ui->show_deferred_messages();
1292}
1293
1294bool PlugInManager::CheckBlacklistedPlugin(const PluginMetadata plugin) {
1295 auto v = SemanticVersion::parse(plugin.version);
1296 return CheckBlacklistedPlugin(wxString(plugin.name), v.major, v.minor);
1297}
1298
1299bool PlugInManager::CheckBlacklistedPlugin(opencpn_plugin* plugin) {
1300 int major = plugin->GetPlugInVersionMajor();
1301 int minor = plugin->GetPlugInVersionMinor();
1302
1303#ifdef __WXMSW__
1304 wxString name = wxString::FromAscii(typeid(*plugin).name());
1305 name.Replace("class ", wxEmptyString);
1306#else
1307 const std::type_info& ti = typeid(*plugin);
1308 int status;
1309 char* realname = abi::__cxa_demangle(ti.name(), 0, 0, &status);
1310 wxString name = wxString::FromAscii(realname);
1311 free(realname);
1312#endif // __WXMSW__
1313 return CheckBlacklistedPlugin(name, major, minor);
1314}
1315
1316bool PlugInManager::CheckBlacklistedPlugin(wxString name, int major,
1317 int minor) {
1318 auto block_status = m_blacklist->get_status(name.ToStdString(), major, minor);
1319 if (block_status == plug_status::unblocked) return true;
1320 plug_data data(name.ToStdString(), major, minor);
1321 auto msg = m_blacklist->get_message(block_status, data);
1322 m_blacklist_ui->message(msg);
1323 return false;
1324}
1325
1326bool PlugInManager::RenderAllCanvasOverlayPlugIns(ocpnDC& dc,
1327 const ViewPort& vp,
1328 int canvasIndex,
1329 int priority) {
1330 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1331 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1332 PlugInContainer* pic = plugin_array->Item(i);
1333 if (pic->m_enabled && pic->m_init_state) {
1335 PlugIn_ViewPort pivp = CreatePlugInViewport(vp);
1336
1337 wxDC* pdc = dc.GetDC();
1338 if (pdc) // not in OpenGL mode
1339 {
1340 switch (pic->m_api_version) {
1341 case 106: {
1342 if (priority > 0) break;
1343 opencpn_plugin_16* ppi =
1344 dynamic_cast<opencpn_plugin_16*>(pic->m_pplugin);
1345 if (ppi) ppi->RenderOverlay(*pdc, &pivp);
1346 break;
1347 }
1348 case 107: {
1349 if (priority > 0) break;
1350 opencpn_plugin_17* ppi =
1351 dynamic_cast<opencpn_plugin_17*>(pic->m_pplugin);
1352 if (ppi) ppi->RenderOverlay(*pdc, &pivp);
1353 break;
1354 }
1355 case 108:
1356 case 109:
1357 case 110:
1358 case 111:
1359 case 112:
1360 case 113:
1361 case 114:
1362 case 115: {
1363 if (priority > 0) break;
1364 opencpn_plugin_18* ppi =
1365 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1366 if (ppi) ppi->RenderOverlay(*pdc, &pivp);
1367 break;
1368 }
1369 case 116:
1370 case 117: {
1371 if (priority > 0) break;
1372 opencpn_plugin_18* ppi =
1373 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1374 if (ppi) {
1375 ppi->RenderOverlay(*pdc, &pivp);
1376 }
1377 opencpn_plugin_116* ppi116 =
1378 dynamic_cast<opencpn_plugin_116*>(pic->m_pplugin);
1379 if (ppi116)
1380 ppi116->RenderOverlayMultiCanvas(*pdc, &pivp, canvasIndex);
1381 break;
1382 }
1383 case 118:
1384 case 119: {
1385 if (priority <= 0) {
1386 opencpn_plugin_18* ppi =
1387 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1388 if (ppi) {
1389 ppi->RenderOverlay(*pdc, &pivp);
1390 }
1391 }
1392 opencpn_plugin_118* ppi118 =
1393 dynamic_cast<opencpn_plugin_118*>(pic->m_pplugin);
1394 if (ppi118)
1395 ppi118->RenderOverlayMultiCanvas(*pdc, &pivp, canvasIndex,
1396 priority);
1397 break;
1398 }
1399 default:
1400 break;
1401 }
1402 } else {
1403 // If in OpenGL mode, and the PlugIn has requested OpenGL render
1404 // callbacks, then there is no need to render by wxDC here.
1405 if (pic->m_cap_flag & WANTS_OPENGL_OVERLAY_CALLBACK) continue;
1406
1407 if ((m_cached_overlay_bm.GetWidth() != vp.pix_width) ||
1408 (m_cached_overlay_bm.GetHeight() != vp.pix_height))
1409 m_cached_overlay_bm.Create(vp.pix_width, vp.pix_height, -1);
1410
1411 wxMemoryDC mdc;
1412 mdc.SelectObject(m_cached_overlay_bm);
1413 mdc.SetBackground(*wxBLACK_BRUSH);
1414 mdc.Clear();
1415
1416 bool b_rendered = false;
1417
1418 switch (pic->m_api_version) {
1419 case 106: {
1420 if (priority > 0) break;
1421 opencpn_plugin_16* ppi =
1422 dynamic_cast<opencpn_plugin_16*>(pic->m_pplugin);
1423 if (ppi) b_rendered = ppi->RenderOverlay(mdc, &pivp);
1424 break;
1425 }
1426 case 107: {
1427 if (priority > 0) break;
1428 opencpn_plugin_17* ppi =
1429 dynamic_cast<opencpn_plugin_17*>(pic->m_pplugin);
1430 if (ppi) b_rendered = ppi->RenderOverlay(mdc, &pivp);
1431 break;
1432 }
1433 case 108:
1434 case 109:
1435 case 110:
1436 case 111:
1437 case 112:
1438 case 113:
1439 case 114:
1440 case 115: {
1441 if (priority > 0) break;
1442 opencpn_plugin_18* ppi =
1443 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1444 if (ppi) b_rendered = ppi->RenderOverlay(mdc, &pivp);
1445 break;
1446 }
1447 case 116:
1448 case 117: {
1449 if (priority > 0) break;
1450 opencpn_plugin_18* ppi =
1451 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1452 if (ppi) {
1453 b_rendered = ppi->RenderOverlay(mdc, &pivp);
1454 }
1455 opencpn_plugin_116* ppi116 =
1456 dynamic_cast<opencpn_plugin_116*>(pic->m_pplugin);
1457 if (ppi116)
1458 b_rendered = ppi116->RenderOverlayMultiCanvas(mdc, &pivp,
1459 g_canvasConfig);
1460 break;
1461 }
1462 case 118:
1463 case 119: {
1464 if (priority <= 0) {
1465 opencpn_plugin_18* ppi =
1466 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1467 if (ppi) {
1468 b_rendered = ppi->RenderOverlay(mdc, &pivp);
1469 }
1470 }
1471 opencpn_plugin_118* ppi118 =
1472 dynamic_cast<opencpn_plugin_118*>(pic->m_pplugin);
1473 if (ppi118)
1474 b_rendered = ppi118->RenderOverlayMultiCanvas(
1475 mdc, &pivp, g_canvasConfig, priority);
1476 break;
1477 }
1478 default: {
1479 b_rendered = pic->m_pplugin->RenderOverlay(&mdc, &pivp);
1480 break;
1481 }
1482 }
1483
1484 mdc.SelectObject(wxNullBitmap);
1485
1486 if (b_rendered) {
1487 wxMask* p_msk = new wxMask(m_cached_overlay_bm, wxColour(0, 0, 0));
1488 m_cached_overlay_bm.SetMask(p_msk);
1489
1490 dc.DrawBitmap(m_cached_overlay_bm, 0, 0, true);
1491 }
1492 }
1493 } else if (pic->m_cap_flag & WANTS_OPENGL_OVERLAY_CALLBACK) {
1494 }
1495 }
1496 }
1497
1498 return true;
1499}
1500
1501bool PlugInManager::RenderAllGLCanvasOverlayPlugIns(wxGLContext* pcontext,
1502 const ViewPort& vp,
1503 int canvasIndex,
1504 int priority) {
1505 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1506 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1507 PlugInContainer* pic = plugin_array->Item(i);
1508 if (pic->m_enabled && pic->m_init_state) {
1509 if (pic->m_cap_flag & WANTS_OPENGL_OVERLAY_CALLBACK) {
1510 PlugIn_ViewPort pivp = CreatePlugInViewport(vp);
1511
1512 switch (pic->m_api_version) {
1513 case 107: {
1514 if (priority > 0) break;
1515 opencpn_plugin_17* ppi =
1516 dynamic_cast<opencpn_plugin_17*>(pic->m_pplugin);
1517 if (ppi) ppi->RenderGLOverlay(pcontext, &pivp);
1518 break;
1519 }
1520
1521 case 108:
1522 case 109:
1523 case 110:
1524 case 111:
1525 case 112:
1526 case 113:
1527 case 114:
1528 case 115: {
1529 if (priority > 0) break;
1530 opencpn_plugin_18* ppi =
1531 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1532 if (ppi) ppi->RenderGLOverlay(pcontext, &pivp);
1533 break;
1534 }
1535 case 116:
1536 case 117: {
1537 if (priority > 0) break;
1538 opencpn_plugin_18* ppi =
1539 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1540 if (ppi) {
1541 ppi->RenderGLOverlay(pcontext, &pivp);
1542 }
1543 opencpn_plugin_116* ppi116 =
1544 dynamic_cast<opencpn_plugin_116*>(pic->m_pplugin);
1545 if (ppi116) {
1546 ppi116->RenderGLOverlayMultiCanvas(pcontext, &pivp, canvasIndex);
1547 }
1548 break;
1549 }
1550 case 118:
1551 case 119: {
1552 if (priority <= 0) {
1553 opencpn_plugin_18* ppi =
1554 dynamic_cast<opencpn_plugin_18*>(pic->m_pplugin);
1555 if (ppi) {
1556 ppi->RenderGLOverlay(pcontext, &pivp);
1557 }
1558 }
1559 opencpn_plugin_118* ppi118 =
1560 dynamic_cast<opencpn_plugin_118*>(pic->m_pplugin);
1561 if (ppi118) {
1562 ppi118->RenderGLOverlayMultiCanvas(pcontext, &pivp, canvasIndex,
1563 priority);
1564 }
1565 break;
1566 }
1567 default:
1568 break;
1569 }
1570 }
1571 }
1572 }
1573
1574 return true;
1575}
1576
1577void PlugInManager::SendViewPortToRequestingPlugIns(ViewPort& vp) {
1578 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1579 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1580 PlugInContainer* pic = plugin_array->Item(i);
1581 if (pic->m_enabled && pic->m_init_state) {
1583 PlugIn_ViewPort pivp = CreatePlugInViewport(vp);
1584 if (pic->m_pplugin) pic->m_pplugin->SetCurrentViewPort(pivp);
1585 }
1586 }
1587 }
1588}
1589
1590void NotifySetupOptionsPlugin(const PlugInData* pd) {
1591 PluginLoader::getInstance()->NotifySetupOptionsPlugin(pd);
1592}
1593
1594void PlugInManager::NotifySetupOptions() {
1595 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1596 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1597 PlugInContainer* pic = plugin_array->Item(i);
1598 NotifySetupOptionsPlugin(pic);
1599 }
1600}
1601
1602void PlugInManager::ClosePlugInPanel(const PlugInContainer* pic,
1603 int ok_apply_cancel) {
1604 if (pic->m_enabled && pic->m_init_state) {
1605 if ((pic->m_cap_flag & INSTALLS_TOOLBOX_PAGE) && pic->m_toolbox_panel) {
1606 pic->m_pplugin->OnCloseToolboxPanel(0, ok_apply_cancel);
1607 auto loader = PluginLoader::getInstance();
1608 loader->SetToolboxPanel(pic->m_common_name, false);
1609 loader->SetSetupOptions(pic->m_common_name, false);
1610 }
1611 }
1612}
1613
1614void PlugInManager::CloseAllPlugInPanels(int ok_apply_cancel) {
1615 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1616 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1617 PlugInContainer* pic = plugin_array->Item(i);
1618 if (pic) {
1619 ClosePlugInPanel(pic, ok_apply_cancel);
1620 }
1621 }
1622}
1623
1624int PlugInManager::AddCanvasContextMenuItem(wxMenuItem* pitem,
1625 opencpn_plugin* pplugin,
1626 const char* name) {
1628 pmic->pmenu_item = pitem;
1629 pmic->m_pplugin = pplugin;
1630 pmic->id = pitem->GetId() == wxID_SEPARATOR ? wxID_SEPARATOR
1631 : m_plugin_menu_item_id_next;
1632 pmic->b_viz = true;
1633 pmic->b_grey = false;
1634 pmic->m_in_menu = name;
1635
1636 m_PlugInMenuItems.Add(pmic);
1637
1638 m_plugin_menu_item_id_next++;
1639
1640 return pmic->id;
1641}
1642
1643void PlugInManager::RemoveCanvasContextMenuItem(int item, const char* name) {
1644 for (unsigned int i = 0; i < m_PlugInMenuItems.GetCount(); i++) {
1645 PlugInMenuItemContainer* pimis = m_PlugInMenuItems[i];
1646 {
1647 if (pimis->id == item && !strcmp(name, pimis->m_in_menu)) {
1648 m_PlugInMenuItems.Remove(pimis);
1649 delete pimis;
1650 break;
1651 }
1652 }
1653 }
1654}
1655
1656void PlugInManager::SetCanvasContextMenuItemViz(int item, bool viz,
1657 const char* name) {
1658 for (unsigned int i = 0; i < m_PlugInMenuItems.GetCount(); i++) {
1659 PlugInMenuItemContainer* pimis = m_PlugInMenuItems[i];
1660 {
1661 if (pimis->id == item && !strcmp(name, pimis->m_in_menu)) {
1662 pimis->b_viz = viz;
1663 break;
1664 }
1665 }
1666 }
1667}
1668
1669void PlugInManager::SetCanvasContextMenuItemGrey(int item, bool grey,
1670 const char* name) {
1671 for (unsigned int i = 0; i < m_PlugInMenuItems.GetCount(); i++) {
1672 PlugInMenuItemContainer* pimis = m_PlugInMenuItems[i];
1673 {
1674 if (pimis->id == item && !strcmp(name, pimis->m_in_menu)) {
1675 pimis->b_grey = grey;
1676 break;
1677 }
1678 }
1679 }
1680}
1681
1682void PlugInManager::SendResizeEventToAllPlugIns(int x, int y) {
1683 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1684 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1685 PlugInContainer* pic = plugin_array->Item(i);
1686 if (pic->m_enabled && pic->m_init_state)
1687 pic->m_pplugin->ProcessParentResize(x, y);
1688 }
1689}
1690
1691void PlugInManager::SetColorSchemeForAllPlugIns(ColorScheme cs) {
1692 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1693 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1694 PlugInContainer* pic = plugin_array->Item(i);
1695 if (pic->m_enabled && pic->m_init_state)
1696 pic->m_pplugin->SetColorScheme((PI_ColorScheme)cs);
1697 }
1698}
1699
1700void PlugInManager::PrepareAllPluginContextMenus() {
1701 int canvasIndex = gFrame->GetCanvasIndexUnderMouse();
1702 if (canvasIndex < 0) return;
1703
1704 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1705 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1706 PlugInContainer* pic = plugin_array->Item(i);
1707 if (pic->m_enabled && pic->m_init_state) {
1709 switch (pic->m_api_version) {
1710 case 116:
1711 case 117:
1712 case 118:
1713 case 119: {
1714 opencpn_plugin_116* ppi =
1715 dynamic_cast<opencpn_plugin_116*>(pic->m_pplugin);
1716 if (ppi) ppi->PrepareContextMenu(canvasIndex);
1717 break;
1718 }
1719 default:
1720 break;
1721 }
1722 }
1723 }
1724 }
1725}
1726
1727void PlugInManager::SendSKConfigToAllPlugIns() {
1728 // Send the current ownship MMSI, encoded as sK, to all PlugIns
1729 wxJSONValue v;
1730 v[_T("self")] = g_ownshipMMSI_SK;
1731 wxJSONWriter w;
1732 wxString out;
1733 w.Write(v, out);
1734 SendMessageToAllPlugins(wxString(_T("OCPN_CORE_SIGNALK")), out);
1735}
1736
1737void PlugInManager::SendBaseConfigToAllPlugIns() {
1738 // Send the current run-time configuration to all PlugIns
1739 wxJSONValue v;
1740 v[_T("OpenCPN Version Major")] = VERSION_MAJOR;
1741 v[_T("OpenCPN Version Minor")] = VERSION_MINOR;
1742 v[_T("OpenCPN Version Patch")] = VERSION_PATCH;
1743 v[_T("OpenCPN Version Date")] = VERSION_DATE;
1744 v[_T("OpenCPN Version Full")] = VERSION_FULL;
1745
1746 // Some useful display metrics
1747 if (g_MainToolbar) {
1748 v[_T("OpenCPN Toolbar Width")] = g_MainToolbar->GetToolbarRect().width;
1749 v[_T("OpenCPN Toolbar Height")] = g_MainToolbar->GetToolbarRect().height;
1750 v[_T("OpenCPN Toolbar PosnX")] = g_MainToolbar->GetToolbarRect().x;
1751 v[_T("OpenCPN Toolbar PosnY")] = g_MainToolbar->GetToolbarRect().y;
1752 }
1753
1754 // Some rendering parameters
1755 v[_T("OpenCPN Zoom Mod Vector")] = g_chart_zoom_modifier_vector;
1756 v[_T("OpenCPN Zoom Mod Other")] = g_chart_zoom_modifier_raster;
1757 v[_T("OpenCPN Scale Factor Exp")] =
1758 g_Platform->GetChartScaleFactorExp(g_ChartScaleFactor);
1759 v[_T("OpenCPN Display Width")] = (int)g_display_size_mm;
1760 v[_T("OpenCPN Content Scale Factor")] = OCPN_GetDisplayContentScaleFactor();
1761 v[_T("OpenCPN Display DIP Scale Factor")] = OCPN_GetWinDIPScaleFactor();
1762
1763 wxJSONWriter w;
1764 wxString out;
1765 w.Write(v, out);
1766 SendMessageToAllPlugins(wxString(_T("OpenCPN Config")), out);
1767}
1768
1769void PlugInManager::SendS52ConfigToAllPlugIns(bool bReconfig) {
1770 // Send the current run-time configuration to all PlugIns
1771 wxJSONValue v;
1772 v[_T("OpenCPN Version Major")] = VERSION_MAJOR;
1773 v[_T("OpenCPN Version Minor")] = VERSION_MINOR;
1774 v[_T("OpenCPN Version Patch")] = VERSION_PATCH;
1775 v[_T("OpenCPN Version Date")] = VERSION_DATE;
1776 v[_T("OpenCPN Version Full")] = VERSION_FULL;
1777
1778 // S52PLIB state
1779 if (ps52plib) {
1780 // v[_T("OpenCPN S52PLIB ShowText")] = ps52plib->GetShowS57Text();
1781 // v[_T("OpenCPN S52PLIB ShowSoundings")] =
1782 // ps52plib->GetShowSoundings(); v[_T("OpenCPN S52PLIB ShowLights")]
1783 // = !ps52plib->GetLightsOff();
1784 v[_T("OpenCPN S52PLIB ShowAnchorConditions")] = ps52plib->GetAnchorOn();
1785 v[_T("OpenCPN S52PLIB ShowQualityOfData")] = ps52plib->GetQualityOfData();
1786 // v[_T("OpenCPN S52PLIB DisplayCategory")] =
1787 // ps52plib->GetDisplayCategory();
1788
1789 // Global parameters
1790 v[_T("OpenCPN S52PLIB MetaDisplay")] = ps52plib->m_bShowMeta;
1791 v[_T("OpenCPN S52PLIB DeclutterText")] = ps52plib->m_bDeClutterText;
1792 v[_T("OpenCPN S52PLIB ShowNationalText")] = ps52plib->m_bShowNationalTexts;
1793 v[_T("OpenCPN S52PLIB ShowImportantTextOnly")] =
1794 ps52plib->m_bShowS57ImportantTextOnly;
1795 v[_T("OpenCPN S52PLIB UseSCAMIN")] = ps52plib->m_bUseSCAMIN;
1796 v[_T("OpenCPN S52PLIB UseSUPER_SCAMIN")] = ps52plib->m_bUseSUPER_SCAMIN;
1797 v[_T("OpenCPN S52PLIB SymbolStyle")] = ps52plib->m_nSymbolStyle;
1798 v[_T("OpenCPN S52PLIB BoundaryStyle")] = ps52plib->m_nBoundaryStyle;
1799 v[_T("OpenCPN S52PLIB ColorShades")] =
1800 S52_getMarinerParam(S52_MAR_TWO_SHADES);
1801 v[_T("OpenCPN S52PLIB Safety Depth")] =
1802 (double)S52_getMarinerParam(S52_MAR_SAFETY_DEPTH);
1803 v[_T("OpenCPN S52PLIB Shallow Contour")] =
1804 (double)S52_getMarinerParam(S52_MAR_SHALLOW_CONTOUR);
1805 v[_T("OpenCPN S52PLIB Deep Contour")] =
1806 (double)S52_getMarinerParam(S52_MAR_DEEP_CONTOUR);
1807 }
1808
1809 // Notify plugins that S52PLIB may have reconfigured global options
1810 v[_T("OpenCPN S52PLIB GlobalReconfig")] = bReconfig;
1811
1812 wxJSONWriter w;
1813 wxString out;
1814 w.Write(v, out);
1815 SendMessageToAllPlugins(wxString(_T("OpenCPN Config")), out);
1816}
1817
1818void PlugInManager::NotifyAuiPlugIns(void) {
1819 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
1820 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
1821 PlugInContainer* pic = plugin_array->Item(i);
1822 if (pic->m_enabled && pic->m_init_state &&
1824 pic->m_pplugin->UpdateAuiStatus();
1825 }
1826}
1827
1828int PlugInManager::AddToolbarTool(wxString label, wxBitmap* bitmap,
1829 wxBitmap* bmpRollover, wxItemKind kind,
1830 wxString shortHelp, wxString longHelp,
1831 wxObject* clientData, int position,
1832 int tool_sel, opencpn_plugin* pplugin) {
1834 pttc->label = label;
1835
1836 if (!bitmap->IsOk()) {
1837 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
1838 pttc->bitmap_day = new wxBitmap(style->GetIcon(_T("default_pi")));
1839 } else {
1840 // Force a non-reference copy of the bitmap from the PlugIn
1841 pttc->bitmap_day = new wxBitmap(*bitmap);
1842 pttc->bitmap_day->UnShare();
1843 }
1844
1845 if (!bmpRollover->IsOk()) {
1846 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
1847 pttc->bitmap_Rollover_day = new wxBitmap(style->GetIcon(_T("default_pi")));
1848 } else {
1849 // Force a non-reference copy of the bitmap from the PlugIn
1850 pttc->bitmap_Rollover_day = new wxBitmap(*bmpRollover);
1851 pttc->bitmap_Rollover_day->UnShare();
1852 }
1853
1854 pttc->bitmap_dusk = BuildDimmedToolBitmap(pttc->bitmap_day, 128);
1855 pttc->bitmap_night = BuildDimmedToolBitmap(pttc->bitmap_day, 32);
1856 pttc->bitmap_Rollover_dusk =
1857 BuildDimmedToolBitmap(pttc->bitmap_Rollover_day, 128);
1858 pttc->bitmap_Rollover_night =
1859 BuildDimmedToolBitmap(pttc->bitmap_Rollover_day, 32);
1860
1861 pttc->kind = kind;
1862 pttc->shortHelp = shortHelp;
1863 pttc->longHelp = longHelp;
1864 pttc->clientData = clientData;
1865 pttc->position = position;
1866 pttc->m_pplugin = pplugin;
1867 pttc->tool_sel = tool_sel;
1868 pttc->b_viz = true;
1869 pttc->b_toggle = false;
1870 pttc->id = m_plugin_tool_id_next;
1871
1872 m_PlugInToolbarTools.Add(pttc);
1873
1874 m_plugin_tool_id_next++;
1875
1876 return pttc->id;
1877}
1878
1879int PlugInManager::AddToolbarTool(wxString label, wxString SVGfile,
1880 wxString SVGRolloverfile,
1881 wxString SVGToggledfile, wxItemKind kind,
1882 wxString shortHelp, wxString longHelp,
1883 wxObject* clientData, int position,
1884 int tool_sel, opencpn_plugin* pplugin) {
1886 pttc->label = label;
1887
1888 pttc->pluginNormalIconSVG = SVGfile;
1889 pttc->pluginRolloverIconSVG = SVGRolloverfile;
1890 pttc->pluginToggledIconSVG = SVGToggledfile;
1891
1892 // Build a set of bitmaps based on the generic "puzzle piece" icon,
1893 // In case there is some problem with the SVG file(s) specified.
1894 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
1895 pttc->bitmap_day = new wxBitmap(style->GetIcon(_T("default_pi")));
1896 pttc->bitmap_dusk = BuildDimmedToolBitmap(pttc->bitmap_day, 128);
1897 pttc->bitmap_night = BuildDimmedToolBitmap(pttc->bitmap_day, 32);
1898 pttc->bitmap_Rollover_day = new wxBitmap(*pttc->bitmap_day);
1899 pttc->bitmap_Rollover_dusk =
1900 BuildDimmedToolBitmap(pttc->bitmap_Rollover_day, 128);
1901 pttc->bitmap_Rollover_night =
1902 BuildDimmedToolBitmap(pttc->bitmap_Rollover_day, 32);
1903
1904 pttc->kind = kind;
1905 pttc->shortHelp = shortHelp;
1906 pttc->longHelp = longHelp;
1907 pttc->clientData = clientData;
1908 pttc->position = position;
1909 pttc->m_pplugin = pplugin;
1910 pttc->tool_sel = tool_sel;
1911 pttc->b_viz = true;
1912 pttc->b_toggle = false;
1913 pttc->id = m_plugin_tool_id_next;
1914
1915 m_PlugInToolbarTools.Add(pttc);
1916
1917 m_plugin_tool_id_next++;
1918
1919 return pttc->id;
1920}
1921
1922void PlugInManager::RemoveToolbarTool(int tool_id) {
1923 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
1924 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
1925 {
1926 if (pttc->id == tool_id) {
1927 m_PlugInToolbarTools.Remove(pttc);
1928 delete pttc;
1929 break;
1930 }
1931 }
1932 }
1933 pParent->RequestNewToolbars();
1934}
1935
1936void PlugInManager::SetToolbarToolViz(int item, bool viz) {
1937 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
1938 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
1939 {
1940 if (pttc->id == item) {
1941 pttc->b_viz = viz;
1942 pParent->RequestNewToolbars(); // Apply the change
1943 break;
1944 }
1945 }
1946 }
1947}
1948
1949void PlugInManager::SetToolbarItemState(int item, bool toggle) {
1950 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
1951 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
1952 {
1953 if (pttc->id == item) {
1954 pttc->b_toggle = toggle;
1955 pParent->SetMasterToolbarItemState(item, toggle);
1956 break;
1957 }
1958 }
1959 }
1960}
1961
1962void PlugInManager::SetToolbarItemBitmaps(int item, wxBitmap* bitmap,
1963 wxBitmap* bmpRollover) {
1964 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
1965 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
1966 {
1967 if (pttc->id == item) {
1968 delete pttc->bitmap_day;
1969 delete pttc->bitmap_dusk;
1970 delete pttc->bitmap_night;
1971 delete pttc->bitmap_Rollover_day;
1972
1973 if (!bitmap->IsOk()) {
1974 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
1975 pttc->bitmap_day = new wxBitmap(style->GetIcon(_T("default_pi")));
1976 } else {
1977 // Force a non-reference copy of the bitmap from the PlugIn
1978 pttc->bitmap_day = new wxBitmap(*bitmap);
1979 pttc->bitmap_day->UnShare();
1980 }
1981
1982 if (!bmpRollover->IsOk()) {
1983 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
1984 pttc->bitmap_Rollover_day =
1985 new wxBitmap(style->GetIcon(_T("default_pi")));
1986 } else {
1987 // Force a non-reference copy of the bitmap from the PlugIn
1988 pttc->bitmap_Rollover_day = new wxBitmap(*bmpRollover);
1989 pttc->bitmap_Rollover_day->UnShare();
1990 }
1991
1992 pttc->bitmap_dusk = BuildDimmedToolBitmap(pttc->bitmap_day, 128);
1993 pttc->bitmap_night = BuildDimmedToolBitmap(pttc->bitmap_day, 32);
1994
1995 pParent->SetToolbarItemBitmaps(item, pttc->bitmap_day,
1996 pttc->bitmap_Rollover_day);
1997 break;
1998 }
1999 }
2000 }
2001}
2002
2003void PlugInManager::SetToolbarItemBitmaps(int item, wxString SVGfile,
2004 wxString SVGfileRollover,
2005 wxString SVGfileToggled) {
2006 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
2007 PlugInToolbarToolContainer* pttc = m_PlugInToolbarTools[i];
2008 {
2009 if (pttc->id == item) {
2010 pttc->pluginNormalIconSVG = SVGfile;
2011 pttc->pluginRolloverIconSVG = SVGfileRollover;
2012 pttc->pluginToggledIconSVG = SVGfileToggled;
2013 pParent->SetToolbarItemSVG(item, pttc->pluginNormalIconSVG,
2014 pttc->pluginRolloverIconSVG,
2015 pttc->pluginToggledIconSVG);
2016 break;
2017 }
2018 }
2019 }
2020}
2021
2022opencpn_plugin* PlugInManager::FindToolOwner(const int id) {
2023 for (unsigned int i = 0; i < m_PlugInToolbarTools.GetCount(); i++) {
2024 PlugInToolbarToolContainer* pc = m_PlugInToolbarTools[i];
2025 if (id == pc->id) return pc->m_pplugin;
2026 }
2027
2028 return NULL;
2029}
2030
2031wxString PlugInManager::GetToolOwnerCommonName(const int id) {
2032 opencpn_plugin* ppi = FindToolOwner(id);
2033 if (ppi) {
2034 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
2035 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
2036 PlugInContainer* pic = plugin_array->Item(i);
2037 if (pic && (pic->m_pplugin == ppi)) return pic->m_common_name;
2038 }
2039 }
2040
2041 return wxEmptyString;
2042}
2043
2044wxString PlugInManager::GetLastError() { return m_last_error_string; }
2045
2046wxBitmap* PlugInManager::BuildDimmedToolBitmap(wxBitmap* pbmp_normal,
2047 unsigned char dim_ratio) {
2048 wxImage img_dup = pbmp_normal->ConvertToImage();
2049
2050 if (!img_dup.IsOk()) return NULL;
2051
2052 if (dim_ratio < 200) {
2053 // Create a dimmed version of the image/bitmap
2054 int gimg_width = img_dup.GetWidth();
2055 int gimg_height = img_dup.GetHeight();
2056
2057 double factor = (double)(dim_ratio) / 256.0;
2058
2059 for (int iy = 0; iy < gimg_height; iy++) {
2060 for (int ix = 0; ix < gimg_width; ix++) {
2061 if (!img_dup.IsTransparent(ix, iy)) {
2062 wxImage::RGBValue rgb(img_dup.GetRed(ix, iy),
2063 img_dup.GetGreen(ix, iy),
2064 img_dup.GetBlue(ix, iy));
2065 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
2066 hsv.value = hsv.value * factor;
2067 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
2068 img_dup.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
2069 }
2070 }
2071 }
2072 }
2073 // Make a bitmap
2074 wxBitmap* ptoolBarBitmap;
2075
2076#ifdef __WXMSW__
2077 wxBitmap tbmp(img_dup.GetWidth(), img_dup.GetHeight(), -1);
2078 wxMemoryDC dwxdc;
2079 dwxdc.SelectObject(tbmp);
2080
2081 ptoolBarBitmap = new wxBitmap(img_dup, (wxDC&)dwxdc);
2082#else
2083 ptoolBarBitmap = new wxBitmap(img_dup);
2084#endif
2085
2086 // store it
2087 return ptoolBarBitmap;
2088}
2089
2090wxArrayString PlugInManager::GetPlugInChartClassNameArray(void) {
2091 wxArrayString array;
2092 auto plugin_array = PluginLoader::getInstance()->GetPlugInArray();
2093 for (unsigned int i = 0; i < plugin_array->GetCount(); i++) {
2094 PlugInContainer* pic = plugin_array->Item(i);
2095 if (pic && pic->m_enabled && pic->m_init_state &&
2098 wxArrayString carray = pic->m_pplugin->GetDynamicChartClassNameArray();
2099
2100 for (unsigned int j = 0; j < carray.GetCount(); j++) {
2101 array.Add(carray[j]);
2102 }
2103 }
2104 }
2105
2106 // Scrub the list for duplicates
2107 // Corrects a flaw in BSB4 and NVC PlugIns
2108 unsigned int j = 0;
2109 while (j < array.GetCount()) {
2110 wxString test = array[j];
2111 unsigned int k = j + 1;
2112 while (k < array.GetCount()) {
2113 if (test == array[k]) {
2114 array.RemoveAt(k);
2115 j = -1;
2116 break;
2117 } else
2118 k++;
2119 }
2120
2121 j++;
2122 }
2123
2124 return array;
2125}
2126
2127//-------------------------------------------------------------------------------
2128// PluginListPanel & PluginPanel Implementation
2129//-------------------------------------------------------------------------------
2130
2131#define DISABLED_SETTINGS_MSG \
2132 _("These settings might destabilize OpenCPN and are by default disabled." \
2133 " To despite the dangers enable them manually add a CatalogExpert=1" \
2134 " line in the [PlugIns] section in the configuration file.")
2135
2136/*
2137 * Panel with buttons to control plugin catalog management.
2138 */
2139CatalogMgrPanel::CatalogMgrPanel(wxWindow* parent)
2140 : wxPanel(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize),
2141 m_parent(parent) {
2142 wxBoxSizer* topSizer = new wxBoxSizer(wxVERTICAL);
2143 SetSizer(topSizer);
2144
2145 topSizer->Add(new wxStaticLine(this), 0, wxGROW | wxLEFT | wxRIGHT, 4);
2146
2147 wxStaticBox* itemStaticBoxSizer4Static =
2148 new wxStaticBox(this, wxID_ANY, _("Plugin Catalog"));
2149 wxStaticBoxSizer* itemStaticBoxSizer4 =
2150 new wxStaticBoxSizer(itemStaticBoxSizer4Static, wxVERTICAL);
2151 topSizer->Add(itemStaticBoxSizer4, 1, wxEXPAND | wxALL, 2);
2152
2153#ifndef __ANDROID__
2154 // First line
2155 m_catalogText = new wxStaticText(this, wxID_STATIC, _T(""));
2156 itemStaticBoxSizer4->Add(m_catalogText,
2157 wxSizerFlags().Border().Proportion(1));
2158 m_catalogText->SetLabel(GetCatalogText(false));
2159
2160 // Next line
2161 wxBoxSizer* rowSizer2 = new wxBoxSizer(wxHORIZONTAL);
2162 itemStaticBoxSizer4->Add(rowSizer2,
2163 wxSizerFlags().Expand().Border().Proportion(1));
2164
2165 m_updateButton = new wxButton(this, wxID_ANY, _("Update Plugin Catalog"),
2166 wxDefaultPosition, wxDefaultSize, 0);
2167 rowSizer2->Add(m_updateButton, 0, wxALIGN_LEFT);
2168 m_updateButton->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2169 &CatalogMgrPanel::OnUpdateButton, this);
2170 rowSizer2->AddSpacer(4 * GetCharWidth());
2171 m_tarballButton = new wxButton(this, wxID_ANY, _("Import plugin..."),
2172 wxDefaultPosition, wxDefaultSize, 0);
2173 rowSizer2->Add(m_tarballButton, 0, wxALIGN_LEFT | wxLEFT, 2 * GetCharWidth());
2174 m_tarballButton->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2175 &CatalogMgrPanel::OnTarballButton, this);
2176
2177 rowSizer2->AddSpacer(4 * GetCharWidth());
2178 m_adv_button = new wxButton(this, wxID_ANY, _("Settings..."),
2179 wxDefaultPosition, wxDefaultSize, 0);
2180 ConfigVar<bool> expert("/PlugIns", "CatalogExpert", pConfig);
2181 if (expert.Get(false)) {
2182 m_adv_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2183 &CatalogMgrPanel::OnPluginSettingsButton, this);
2184 } else {
2185 m_adv_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED, [&](wxCommandEvent&) {
2186 wxMessageBox(DISABLED_SETTINGS_MSG, _("Disabled"));
2187 });
2188 }
2189 rowSizer2->AddSpacer(4 * GetCharWidth());
2190 rowSizer2->Add(m_adv_button, 0, wxALIGN_LEFT);
2191
2192 SetUpdateButtonLabel();
2193
2194 // Next line
2195 wxBoxSizer* rowSizer3 = new wxBoxSizer(wxHORIZONTAL);
2196 itemStaticBoxSizer4->Add(rowSizer3, 0, wxEXPAND | wxALL, 4);
2197
2198 SetMinSize(wxSize(m_parent->GetClientSize().x - (4 * GetCharWidth()), -1));
2199 Fit();
2200
2201 GlobalVar<wxString> catalog(&g_catalog_channel);
2202 wxDEFINE_EVENT(EVT_CATALOG_CHANGE, wxCommandEvent);
2203 catalog_listener.Listen(catalog, this, EVT_CATALOG_CHANGE);
2204 Bind(EVT_CATALOG_CHANGE, [&](wxCommandEvent&) { SetUpdateButtonLabel(); });
2205
2206#else // __ANDROID__
2207 SetBackgroundColour(wxColour(0x7c, 0xb0, 0xe9)); // light blue
2208 ConfigVar<bool> expert("/PlugIns", "CatalogExpert", pConfig);
2209 if (!expert.Get(false)) {
2210 m_updateButton =
2211 new wxButton(this, wxID_ANY, _("Update Plugin Catalog: master"),
2212 wxDefaultPosition, wxDefaultSize, 0);
2213 itemStaticBoxSizer4->Add(m_updateButton, 0, wxALIGN_LEFT);
2214 m_updateButton->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2215 &CatalogMgrPanel::OnUpdateButton, this);
2216 SetUpdateButtonLabel();
2217 m_tarballButton = NULL;
2218 m_adv_button = NULL;
2219 } else {
2220 // First line
2221 m_catalogText = new wxStaticText(this, wxID_STATIC, GetCatalogText(false));
2222 itemStaticBoxSizer4->Add(m_catalogText,
2223 wxSizerFlags().Border(wxALL, 5).Proportion(1));
2224 // m_catalogText->SetLabel(GetCatalogText(false));
2225
2226 m_updateButton = new wxButton(
2227 this, wxID_ANY, "Update Plugin Catalog:master ",
2228 wxDefaultPosition, wxDefaultSize, 0);
2229 itemStaticBoxSizer4->Add(m_updateButton, 0, wxALIGN_LEFT | wxTOP, 5);
2230 m_updateButton->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2231 &CatalogMgrPanel::OnUpdateButton, this);
2232 SetUpdateButtonLabel();
2233
2234 // Next line
2235 m_adv_button = new wxButton(this, wxID_ANY, _("Settings..."),
2236 wxDefaultPosition, wxDefaultSize, 0);
2237 itemStaticBoxSizer4->Add(m_adv_button, 0, wxALIGN_LEFT | wxTOP,
2238 GetCharWidth());
2239 m_adv_button->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2240 &CatalogMgrPanel::OnPluginSettingsButton, this);
2241
2242 // Next line
2243 m_tarballButton = new wxButton(this, wxID_ANY, _("Import plugin..."),
2244 wxDefaultPosition, wxDefaultSize, 0);
2245 itemStaticBoxSizer4->Add(m_tarballButton, 0, wxALIGN_LEFT | wxALL,
2246 2 * GetCharWidth());
2247 m_tarballButton->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
2248 &CatalogMgrPanel::OnTarballButton, this);
2249 }
2250
2251#endif
2252}
2253
2254CatalogMgrPanel::~CatalogMgrPanel() {
2255 m_updateButton->Unbind(wxEVT_COMMAND_BUTTON_CLICKED,
2256 &CatalogMgrPanel::OnUpdateButton, this);
2257 if (m_tarballButton)
2258 m_tarballButton->Unbind(wxEVT_COMMAND_BUTTON_CLICKED,
2259 &CatalogMgrPanel::OnTarballButton, this);
2260}
2261
2262static const char* const DOWNLOAD_REPO_PROTO =
2263 "https://raw.githubusercontent.com/OpenCPN/plugins/@branch@/"
2264 "ocpn-plugins.xml";
2265
2266void CatalogMgrPanel::OnUpdateButton(wxCommandEvent& event) {
2267 // Craft the url
2268 std::string catalog(g_catalog_channel == "" ? "master" : g_catalog_channel);
2269 std::string url(g_catalog_custom_url);
2270 if (catalog != "custom") {
2271 url = std::string(DOWNLOAD_REPO_PROTO);
2272 ocpn::replace(url, "@branch@", catalog);
2273 }
2274 // Download to a temp file
2275 std::string filePath =
2276 wxFileName::CreateTempFileName("ocpn_dl").ToStdString();
2277
2278 auto catalogHdlr = CatalogHandler::getInstance();
2279
2280 g_Platform->ShowBusySpinner();
2281 auto status = catalogHdlr->DownloadCatalog(filePath, url);
2282 g_Platform->HideBusySpinner();
2283
2284 std::string message;
2285 if (status != CatalogHandler::ServerStatus::OK) {
2286 message = _("Cannot download data from url");
2287 OCPNMessageBox(this, message, _("OpenCPN Catalog update"),
2288 wxICON_ERROR | wxOK);
2289 return;
2290 }
2291
2292 // TODO Validate xml using xsd here....
2293#ifdef __ANDROID__
2294 if (!AndroidSecureCopyFile(wxString(filePath.c_str()),
2295 g_Platform->GetPrivateDataDir() +
2296 wxFileName::GetPathSeparator() +
2297 _T("ocpn-plugins.xml"))) {
2298 OCPNMessageBox(this, _("Unable to copy catalog file"),
2299 _("OpenCPN Catalog update"), wxICON_ERROR | wxOK);
2300 return;
2301 }
2302#else
2303 // Copy the downloaded file to proper local location
2304 if (!wxCopyFile(wxString(filePath.c_str()),
2305 g_Platform->GetPrivateDataDir() +
2306 wxFileName::GetPathSeparator() +
2307 _T("ocpn-plugins.xml"))) {
2308 OCPNMessageBox(this, _("Unable to copy catalog file"),
2309 _("OpenCPN Catalog update"), wxICON_ERROR | wxOK);
2310 return;
2311 }
2312#endif
2313
2314 // If this is the "master" catalog, also copy to plugin cache
2315 if (catalog == "master") {
2316 if (!ocpn::store_metadata(filePath.c_str())) {
2317 OCPNMessageBox(this, _("Unable to copy catalog file to cache"),
2318 _("OpenCPN Catalog update"), wxICON_ERROR | wxOK);
2319 return;
2320 }
2321 }
2322
2323 // Record in the config file the name of the catalog downloaded
2324 pConfig->SetPath(_T("/PlugIns/"));
2325 pConfig->Write("LatestCatalogDownloaded", catalog.c_str());
2326 pConfig->Flush();
2327
2328 // Reset the PluginHandler catalog file source.
2329 // This will case the Handler to find, load, and parse the just-downloaded
2330 // catalog as copied to g_Platform->GetPrivateDataDir()...
2331 auto pluginHandler = PluginHandler::getInstance();
2332 pluginHandler->setMetadata("");
2333
2334 // Also clear the cached values in the CatalogHandler, forcing
2335 // a reload and parse of the catalog.
2336 auto cataloghdlr = CatalogHandler::getInstance();
2337 cataloghdlr->ClearCatalogData();
2338
2339 // Reload all plugins, which will also update the status fields
2340 LoadAllPlugIns(false);
2341
2342 // Update this Panel, and the entire list.
2343#ifndef __ANDROID__
2344 m_catalogText->SetLabel(GetCatalogText(true));
2345#endif
2346 if (m_PluginListPanel) m_PluginListPanel->ReloadPluginPanels();
2347 OCPNMessageBox(this, _("Catalog update successful"),
2348 _("OpenCPN Catalog update"), wxICON_INFORMATION | wxOK);
2349}
2350
2351void CatalogMgrPanel::OnPluginSettingsButton(wxCommandEvent& event) {
2352 auto dialog = new CatalogSettingsDialog(this);
2353
2354#ifdef __ANDROID__
2355 androidDisableRotation();
2356#endif
2357
2358 dialog->ShowModal();
2359
2360#ifdef __ANDROID__
2361 androidEnableRotation();
2362#endif
2363}
2364
2365void CatalogMgrPanel::OnTarballButton(wxCommandEvent& event) {
2366 // Present a file selector dialog to get the file name..
2367 wxString path;
2368 int response = g_Platform->DoFileSelectorDialog(
2369 this, &path, _("Select tarball file"), GetImportInitDir(), "",
2370 "tar files (*.tar.gz)|*.tar.gz|All Files (*.*)|*.*");
2371
2372 if (response != wxID_OK) {
2373 return;
2374 }
2375 auto handler = PluginHandler::getInstance();
2376 PluginMetadata metadata;
2377 bool ok = handler->ExtractMetadata(path.ToStdString(), metadata);
2378 if (!ok) {
2379 OCPNMessageBox(
2380 this,
2381 _("Error extracting metadata from tarball (missing metadata.xml?)"),
2382 _("OpenCPN Plugin Import Error"));
2383 return;
2384 }
2385 if (!PluginHandler::isCompatible(metadata)) {
2386 OCPNMessageBox(this, _("Incompatible import plugin detected."),
2387 _("OpenCPN Plugin Import Error"));
2388 handler->uninstall(metadata.name);
2389 return;
2390 }
2391 UninstallPlugin(metadata.name);
2392 ok = handler->installPlugin(metadata, path.ToStdString());
2393 if (!ok) {
2394 OCPNMessageBox(this, _("Error extracting import plugin tarball."),
2395 _("OpenCPN Plugin Import Error"));
2396 return;
2397 }
2398 metadata.is_imported = true;
2399 auto metadata_path = PluginHandler::ImportedMetadataPath(metadata.name);
2400 std::ofstream file(metadata_path);
2401 file << metadata.to_string();
2402 if (!file.good()) {
2403 WARNING_LOG << "Error saving metadata file: " << metadata_path
2404 << " for imported plugin: " << metadata.name;
2405 }
2406 LoadAllPlugIns(false, true);
2407 PluginHandler::getInstance()->SetInstalledMetadata(metadata);
2408 m_PluginListPanel->ReloadPluginPanels();
2409 wxString ws(_("Plugin"));
2410 ws += metadata.name + _(" successfully imported");
2411 OCPNMessageBox(gFrame, ws, _("Installation complete"),
2412 wxICON_INFORMATION | wxOK | wxCENTRE);
2413}
2414
2415wxString CatalogMgrPanel::GetCatalogText(bool updated) {
2416 wxString catalog;
2417 catalog = updated ? _("Active Catalog") : _("Last Catalog");
2418 catalog += _T(": ");
2419
2420 // Check the config file to learn what was the last catalog downloaded.
2421 pConfig->SetPath(_T("/PlugIns/"));
2422 wxString latestCatalog =
2423 pConfig->Read(_T("LatestCatalogDownloaded"), _T("default"));
2424 catalog += latestCatalog;
2425
2426#ifndef __ANDROID__
2427 // Get the version from the currently active catalog, by which we mean
2428 // the latest catalog parsed.
2429 auto pluginHandler = PluginHandler::getInstance();
2430 std::string date = pluginHandler->GetCatalogData()->date;
2431
2432 catalog += wxString(" ") + _("Last change: ") + " " + date;
2433 if (!updated) catalog += _T(" : ") + _("Please Update Plugin Catalog.");
2434#endif
2435
2436 return catalog;
2437}
2438
2439void CatalogMgrPanel::SetUpdateButtonLabel() {
2440 wxString label = _("Update Plugin Catalog");
2441 label += _T(": ");
2442 label += g_catalog_channel == "" ? "master" : g_catalog_channel;
2443 m_updateButton->SetLabel(label);
2444 Layout();
2445}
2446
2447wxString CatalogMgrPanel::GetImportInitDir() {
2448 // Check the config file for the last Import path.
2449 pConfig->SetPath(_T("/PlugIns/"));
2450 wxString lastImportDir;
2451 lastImportDir = pConfig->Read(_T("LatestImportDir"),
2452 g_Platform->GetWritableDocumentsDir());
2453 if (wxDirExists(lastImportDir)) {
2454 return lastImportDir;
2455 }
2456 return (g_Platform->GetWritableDocumentsDir());
2457}
2458
2459BEGIN_EVENT_TABLE(PluginListPanel, wxScrolledWindow)
2460// EVT_BUTTON( ID_CMD_BUTTON_PERFORM_ACTION,
2461// PluginListPanel::OnPluginPanelAction )
2462END_EVENT_TABLE()
2463
2464PluginListPanel::PluginListPanel(wxWindow* parent, wxWindowID id,
2465 const wxPoint& pos, const wxSize& size)
2466 : wxScrolledWindow(parent, id, pos, size, wxTAB_TRAVERSAL | wxVSCROLL),
2467 m_PluginSelected(0) {
2468 m_is_loading.clear();
2469 SetSizer(new wxBoxSizer(wxVERTICAL));
2470 ReloadPluginPanels();
2471}
2472
2473void PluginListPanel::SelectByName(wxString& name) {
2474 for (auto it = GetChildren().GetFirst(); it; it = it->GetNext()) {
2475 auto pluginPanel = dynamic_cast<PluginPanel*>(it->GetData());
2476 if (pluginPanel) {
2477 if (pluginPanel->GetPluginPtr()->m_common_name.IsSameAs(name)) {
2478 pluginPanel->SetSelected(true);
2479 pluginPanel->Layout();
2480 SelectPlugin(pluginPanel);
2481 break;
2482 }
2483 }
2484 }
2485}
2486
2488std::vector<const PlugInData*> GetInstalled() {
2489 std::vector<const PlugInData*> result;
2490 auto loader = PluginLoader::getInstance();
2491 for (size_t i = 0; i < loader->GetPlugInArray()->GetCount(); i++) {
2492 auto const item = loader->GetPlugInArray()->Item(i);
2493 if (item->m_managed_metadata.name.empty()) {
2494 const auto name = item->m_common_name.ToStdString();
2495 item->m_managed_metadata = PluginLoader::MetadataByName(name);
2496 }
2497 PluginLoader::UpdatePlugin(item, item->m_managed_metadata);
2498 result.push_back(item);
2499 }
2500 auto compare = [](const PlugInData* lhs, const PlugInData* rhs) {
2501 std::string slhs, srhs;
2502 for (auto& cl : lhs->Key()) slhs += toupper(cl);
2503 for (auto& cr : rhs->Key()) srhs += toupper(cr);
2504 return slhs.compare(srhs) < 0;
2505 };
2506 std::sort(result.begin(), result.end(), compare);
2507 return result;
2508}
2509
2510/* Is plugin with given name present in loaded or safe list if installed? */
2511static bool IsPluginLoaded(const std::string& name) {
2512 if (safe_mode::get_mode()) {
2513 auto installed = PluginHandler::getInstance()->GetInstalldataPlugins();
2514 auto found =
2515 std::find(installed.begin(), installed.end(), ocpn::tolower(name));
2516 return found != installed.end();
2517 } else {
2518 auto loaded = PluginLoader::getInstance()->GetPlugInArray();
2519 for (size_t i = 0; i < loaded->GetCount(); i++) {
2520 if (loaded->Item(i)->m_common_name.ToStdString() == name) return true;
2521 }
2522 return false;
2523 }
2524}
2525
2527 if (m_is_loading.test_and_set()) {
2528 // recursive call...
2529 DEBUG_LOG << "LoadAllPlugins: recursive invocation";
2530 return;
2531 }
2532
2533 auto plugins = PluginLoader::getInstance()->GetPlugInArray();
2534 m_PluginItems.Clear();
2535
2536 wxWindowList kids = GetChildren();
2537 for (unsigned int i = 0; i < kids.GetCount(); i++) {
2538 wxWindowListNode* node = kids.Item(i);
2539 wxWindow* win = node->GetData();
2540 PluginPanel* pp = dynamic_cast<PluginPanel*>(win);
2541 if (pp) win->Destroy();
2542 }
2543 GetSizer()->Clear();
2544
2545 Hide();
2546 m_PluginSelected = 0;
2547
2548 if (safe_mode::get_mode()) {
2550 auto installed = PluginHandler::getInstance()->GetInstalldataPlugins();
2551 for (const auto& name : installed) AddPlugin(name);
2552 } else {
2553 /* The catalog entries. */
2554 auto available = getCompatiblePlugins();
2555
2556 /* Remove those which are loaded or in safe list. */
2557 auto predicate = [](const PluginMetadata& md) {
2558 return IsPluginLoaded(md.name);
2559 };
2560 auto end = std::remove_if(available.begin(), available.end(), predicate);
2561 available.erase(end, available.end());
2562
2563 // Sort on case-insensitive name
2564 struct CompSort {
2565 bool operator()(const PluginMetadata& lhs,
2566 const PluginMetadata rhs) const {
2567 std::string slhs, srhs;
2568 for (auto& cl : lhs.name) slhs += toupper(cl);
2569 for (auto& cr : rhs.name) srhs += toupper(cr);
2570 return slhs.compare(srhs) < 0;
2571 }
2572 } comp_sort;
2573
2574 std::set<PluginMetadata, CompSort> unique_sorted_entries(comp_sort);
2575 for (const auto& p : available) unique_sorted_entries.insert(p);
2576
2577 // Build the list of panels.
2578
2579 // Add Installed and active plugins
2580 for (const auto& p : GetInstalled())
2581 if (p->m_enabled) AddPlugin(*p);
2582
2583 // Add Installed and inactive plugins
2584 for (const auto& p : GetInstalled())
2585 if (!p->m_enabled) AddPlugin(*p);
2586
2587 // Add available plugins, sorted
2588 for (const auto& p : unique_sorted_entries) AddPlugin(PlugInData(p));
2589 }
2590
2591 Show();
2592 Layout();
2593 Refresh(true);
2594 Scroll(0, 0);
2595
2596 m_is_loading.clear();
2597}
2598
2599void PluginListPanel::AddPlugin(const std::string& name) {
2600 auto panel = new PluginPanel(this, name);
2601 DimeControl(panel);
2602 panel->SetSelected(false);
2603 GetSizer()->Add(panel, 0, wxEXPAND);
2604 m_PluginItems.Add(panel);
2605 m_pluginSpacer = g_Platform->GetDisplayDPmm() * 1.0;
2606 GetSizer()->AddSpacer(m_pluginSpacer);
2607}
2608
2609void PluginListPanel::AddPlugin(const PlugInData& pic) {
2610 auto pPluginPanel =
2611 new PluginPanel(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, pic);
2612 pPluginPanel->SetSelected(false);
2613 GetSizer()->Add(pPluginPanel, 0, wxEXPAND);
2614 m_PluginItems.Add(pPluginPanel);
2615
2616 m_pluginSpacer = g_Platform->GetDisplayDPmm() * 1.0;
2617 GetSizer()->AddSpacer(m_pluginSpacer);
2618
2619 // wxStaticLine* itemStaticLine = new wxStaticLine( m_panel, wxID_ANY,
2620 // wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL );
2621 // m_pitemBoxSizer01->Add( itemStaticLine, wxSizerFlags().Expand());
2622}
2623
2624// When a child Panel is selected, its size grows to include "Preferences"
2625// and Enable" buttons. As a consequence, the vertical size of the
2626// ListPanel grows as well.Calculate and add a spacer to bottom of
2627// ListPanel so that initial ListPanel minimum size calculations account
2628// for selected Panel size growth. Sadly, this does not work right on wxQt.
2629// So, just punt for now...
2630int PluginListPanel::ComputePluginSpace(ArrayOfPluginPanel plugins,
2631 wxBoxSizer* sizer) {
2632 int max_dy = 0;
2633 for (size_t i = 0; i < plugins.GetCount(); i++) {
2634 auto panel = plugins.Item(i);
2635 bool was_selected = panel->GetSelected();
2636 panel->SetSelected(false);
2637 sizer->Layout();
2638 wxSize unselected = panel->GetSize();
2639
2640 panel->SetSelected(true); // switch to selected, a bit bigger
2641 sizer->Layout();
2642 wxSize selected = panel->GetSize();
2643
2644 int dy = selected.GetHeight() - unselected.GetHeight();
2645 max_dy = wxMax(max_dy, dy);
2646 panel->SetSelected(was_selected);
2647 }
2648 return max_dy;
2649}
2650
2651PluginListPanel::~PluginListPanel() {}
2652
2653void PluginListPanel::UpdateSelections() {
2654 for (unsigned int i = 0; i < m_PluginItems.GetCount(); i++) {
2655 PluginPanel* pPluginPanel = m_PluginItems[i];
2656 if (pPluginPanel) {
2657 pPluginPanel->SetSelected(pPluginPanel->GetSelected());
2658 }
2659 }
2660}
2661
2662void PluginListPanel::SelectPlugin(PluginPanel* pi) {
2663 int xs, ys;
2664 GetViewStart(&xs, &ys);
2665 Scroll(0, 0);
2666
2667 if (m_PluginSelected) {
2668 m_PluginSelected->SetSelected(false);
2669 m_PluginSelected->Layout();
2670 }
2671
2672 if (pi == NULL) m_PluginSelected->SetSelected(false);
2673
2674 m_PluginSelected = pi;
2675
2676 GetSizer()->Layout();
2677 Refresh(false);
2678 wxSize size = GetBestVirtualSize();
2679 SetVirtualSize(size);
2680
2681 // Measure, and ensure that the selected item is fully visible in the
2682 // vertical scroll box.
2683 int htop = 0;
2684 for (unsigned int i = 0; i < m_PluginItems.GetCount(); i++) {
2685 PluginPanel* pPluginPanel = m_PluginItems[i];
2686 int yd = pPluginPanel->GetSize().y;
2687 htop += yd;
2688 htop += m_pluginSpacer;
2689 if (pPluginPanel == pi) {
2690 int piBottom = htop - (ys * g_options->GetScrollRate());
2691 if (piBottom > GetClientSize().y) {
2692 ys += (piBottom - GetClientSize().y) / g_options->GetScrollRate();
2693 }
2694 break;
2695 }
2696 }
2697
2698 Scroll(xs, ys);
2699}
2700
2701void PluginListPanel::MoveUp(PluginPanel* pi) {
2702 int pos = m_PluginItems.Index(pi);
2703 if (pos == 0) // The first one can't be moved further up
2704 return;
2705 m_PluginItems.RemoveAt(pos);
2706 // m_pitemBoxSizer01->Remove( pos * 2 + 1 );
2707 // m_pitemBoxSizer01->Remove( pos * 2 );
2708 m_PluginItems.Insert(pi, pos - 1);
2709 wxStaticLine* itemStaticLine = new wxStaticLine(
2710 this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL);
2711 // m_pitemBoxSizer01->Insert( (pos - 1) * 2, itemStaticLine, 0,
2712 // wxEXPAND|wxALL, 0 ); m_pitemBoxSizer01->Insert( (pos - 1) * 2, pi, 0,
2713 // wxEXPAND|wxALL, 0 );
2714
2715 m_PluginSelected = pi;
2716
2717 GetSizer()->Layout();
2718 m_parent->Layout();
2719 Refresh(true);
2720}
2721
2722void PluginListPanel::MoveDown(PluginPanel* pi) {
2723 int pos = m_PluginItems.Index(pi);
2724 if (pos == (int)m_PluginItems.Count() -
2725 1) // The last one can't be moved further down
2726 return;
2727 m_PluginItems.RemoveAt(pos);
2728 // m_pitemBoxSizer01->Remove( pos * 2 + 1 );
2729 // m_pitemBoxSizer01->Remove( pos * 2 );
2730 m_PluginItems.Insert(pi, pos + 1);
2731 wxStaticLine* itemStaticLine = new wxStaticLine(
2732 this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLI_HORIZONTAL);
2733 // m_pitemBoxSizer01->Insert( (pos + 1) * 2 - 1, itemStaticLine, 0,
2734 // wxEXPAND|wxALL, 0 ); m_pitemBoxSizer01->Insert( (pos + 1) * 2, pi, 0,
2735 // wxEXPAND|wxALL, 0 );
2736
2737 m_PluginSelected = pi;
2738
2739 GetSizer()->Layout();
2740 m_parent->Layout();
2741 Refresh(false);
2742}
2743
2744static bool canUninstall(std::string name) {
2745 PluginHandler* pluginHandler = PluginHandler::getInstance();
2746 // std::transform(name.begin(), name.end(), name.begin(), ::tolower);
2747
2748 for (auto plugin : pluginHandler->getInstalled()) {
2749 if (plugin.name == name) {
2750 if (safe_mode::get_mode())
2751 return true;
2752 else
2753 return !plugin.readonly;
2754 }
2755 }
2756 return false;
2757}
2758
2759PluginPanel::PluginPanel(wxPanel* parent, const std::string& name)
2760 : wxPanel(parent, wxID_ANY, wxDefaultPosition, wxDefaultSize,
2761 wxBORDER_NONE),
2762 m_is_safe_panel(true) {
2763 m_PluginListPanel = dynamic_cast<PluginListPanel*>(parent);
2764 wxASSERT(m_PluginListPanel != 0);
2765 wxBoxSizer* top_sizer = new wxBoxSizer(wxVERTICAL);
2766 SetSizer(top_sizer);
2767 wxBoxSizer* top_horizontal = new wxBoxSizer(wxHORIZONTAL);
2768 top_sizer->Add(top_horizontal, 0, wxEXPAND);
2769
2770 double iconSize = GetCharWidth() * 4;
2771 double dpi_mult = g_Platform->GetDisplayDIPMult(this);
2772 int icon_scale = iconSize * dpi_mult;
2773 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
2774 wxBitmap bitmap(style->GetIcon("default_pi", icon_scale, icon_scale));
2775 m_itemStaticBitmap = new wxStaticBitmap(this, wxID_ANY, bitmap);
2776 top_horizontal->Add(m_itemStaticBitmap, 0, wxEXPAND | wxALL, 10);
2777
2778 m_pName = new wxStaticText(this, wxID_ANY, name);
2779 top_horizontal->Add(m_pName, wxID_ANY, wxALIGN_CENTER_VERTICAL);
2780 m_pVersion = new wxStaticText(this, wxID_ANY, "");
2781 top_horizontal->Add(m_pVersion);
2782 m_pVersion->Hide();
2783
2784 m_pButtons = new wxBoxSizer(wxHORIZONTAL);
2785 top_horizontal->Add(m_pButtons);
2786 m_info_btn = new WebsiteButton(this, "https:\\opencpn.org");
2787 top_horizontal->Add(m_info_btn);
2788 m_pButtonUninstall = new wxButton(this, wxID_ANY, _("Uninstall"),
2789 wxDefaultPosition, wxDefaultSize, 0);
2790 top_horizontal->Add(m_pButtonUninstall, 0, wxALIGN_CENTER_VERTICAL | wxALL,
2791 2);
2792 auto uninstall = [&](wxCommandEvent ev) {
2793 auto n = m_pName->GetLabel().ToStdString();
2794 int result =
2795 OCPNMessageBox(gFrame, std::string(_("Uninstall plugin ")) + n + "?",
2796 _("Un-Installation"), wxICON_QUESTION | wxOK | wxCANCEL);
2797 if (result != wxID_OK) return;
2798 PluginHandler::getInstance()->ClearInstallData(n);
2799 m_PluginListPanel->ReloadPluginPanels();
2800 };
2801 m_pButtonUninstall->Bind(wxEVT_COMMAND_BUTTON_CLICKED, uninstall);
2802}
2803
2804BEGIN_EVENT_TABLE(PluginPanel, wxPanel)
2805EVT_PAINT(PluginPanel::OnPaint)
2806END_EVENT_TABLE()
2807
2808PluginPanel::PluginPanel(wxPanel* parent, wxWindowID id, const wxPoint& pos,
2809 const wxSize& size, const PlugInData plugin)
2810 : wxPanel(parent, id, pos, size, wxBORDER_NONE),
2811 m_plugin(plugin),
2812 m_is_safe_panel(false) {
2813 m_PluginListPanel = (PluginListPanel*)parent; //->GetParent();
2814 m_PluginListPanel = dynamic_cast<PluginListPanel*>(parent /*->GetParent()*/);
2815 wxASSERT(m_PluginListPanel != 0);
2816
2817 m_bSelected = false;
2818 m_penWidthUnselected = g_Platform->GetDisplayDPmm() * .25;
2819 m_penWidthSelected = g_Platform->GetDisplayDPmm() * .5;
2820
2821 wxBoxSizer* topSizer = new wxBoxSizer(wxVERTICAL);
2822 SetSizer(topSizer);
2823
2824 wxBoxSizer* itemBoxSizer01 = new wxBoxSizer(wxHORIZONTAL);
2825 topSizer->Add(itemBoxSizer01, 0, wxEXPAND);
2826 Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2827 Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2828
2829 double iconSize = GetCharWidth() * 4;
2830 double dpi_mult = g_Platform->GetDisplayDIPMult(this);
2831 int icon_scale = iconSize * dpi_mult;
2832
2833 wxImage plugin_icon;
2834 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
2835 if (m_plugin.m_bitmap.IsOk()) {
2836 plugin_icon = m_plugin.m_bitmap.ConvertToImage();
2837 }
2838 wxBitmap bitmap;
2839 if (m_plugin.m_status == PluginStatus::ManagedInstallAvailable) {
2840 wxFileName path(g_Platform->GetSharedDataDir(), "packageBox.svg");
2841 path.AppendDir("uidata");
2842 path.AppendDir("traditional"); // FIXME(leamas) cache it.
2843 bitmap = LoadSVG(path.GetFullPath(), icon_scale, icon_scale);
2844 } else if (plugin_icon.IsOk()) {
2845 int nowSize = plugin_icon.GetWidth();
2846 plugin_icon.Rescale(icon_scale, icon_scale, wxIMAGE_QUALITY_HIGH);
2847 bitmap = wxBitmap(plugin_icon);
2848 } else {
2849 bitmap = wxBitmap(style->GetIcon("default_pi", icon_scale, icon_scale));
2850 }
2851 m_itemStaticBitmap = new wxStaticBitmap(this, wxID_ANY, bitmap);
2852
2853 itemBoxSizer01->Add(m_itemStaticBitmap, 0, wxEXPAND | wxALL, 10);
2854 m_itemStaticBitmap->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected,
2855 this);
2856 m_itemStaticBitmap->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp,
2857 this);
2858
2859 wxBoxSizer* itemBoxSizer02 = new wxBoxSizer(wxVERTICAL);
2860 itemBoxSizer01->Add(itemBoxSizer02, 1, wxEXPAND | wxALL, 0);
2861
2862 // Calculate character width available
2863 int nChars = g_options->GetSize().x / GetCharWidth();
2864 bool bCompact = false;
2865 if (nChars < 60) // Arbitrary, detecting mobile devices in portrait mode.
2866 bCompact = true;
2867
2868 if (bCompact) {
2869 // Might need to shorten the Plugin name string
2870 wxString nameString = m_plugin.m_common_name;
2871 int maxWidth = g_Platform->getDisplaySize().x * 3 / 10;
2872 wxScreenDC dc;
2873 int nameWidth;
2874 dc.GetTextExtent(m_plugin.m_common_name, &nameWidth, NULL);
2875 if (nameWidth > maxWidth) {
2876 nameString = wxControl::Ellipsize(m_plugin.m_common_name, dc,
2877 wxELLIPSIZE_END, maxWidth);
2878 }
2879 m_pName = new wxStaticText(this, wxID_ANY, nameString);
2880 m_pName->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2881 m_pName->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2882 itemBoxSizer02->Add(m_pName, 0, /*wxEXPAND|*/ wxALL, 5);
2883
2884 wxFlexGridSizer* sl1 = new wxFlexGridSizer(2, 0, 0);
2885 sl1->AddGrowableCol(1);
2886 itemBoxSizer02->Add(sl1, 0, wxEXPAND);
2887
2888 m_pVersion = new wxStaticText(this, wxID_ANY, _T("X.YY.ZZ.AA"));
2889 sl1->Add(m_pVersion, 0, /*wxEXPAND|*/ wxALL, 5);
2890 if (m_plugin.m_status == PluginStatus::ManagedInstallAvailable) {
2891 m_pVersion->Hide();
2892 }
2893 m_pVersion->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2894 m_pVersion->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2895
2896 m_cbEnable = new wxCheckBox(this, wxID_ANY, _("Enabled"));
2897 sl1->Add(m_cbEnable, 1, wxALIGN_RIGHT | wxTOP, 5);
2898 m_cbEnable->Bind(wxEVT_CHECKBOX, &PluginPanel::OnPluginEnableToggle, this);
2899
2900 // Might need to shorten the Plugin description string
2901 wxString descriptionString = m_plugin.m_short_description;
2902 int maxDescriptionWidth = g_Platform->getDisplaySize().x - (iconSize * 4);
2903 int descriptionWidth;
2904 dc.GetTextExtent(m_plugin.m_short_description, &descriptionWidth, NULL);
2905 if (descriptionWidth > maxDescriptionWidth)
2906 descriptionString =
2907 wxControl::Ellipsize(m_plugin.m_short_description, dc,
2908 wxELLIPSIZE_END, maxDescriptionWidth);
2909
2910 // This invocation has the effect of setting the minimum width of the
2911 // descriptor field.
2912 m_pDescription =
2913 new wxStaticText(this, wxID_ANY, descriptionString, wxDefaultPosition,
2914 wxSize(maxDescriptionWidth, -1), wxST_NO_AUTORESIZE);
2915 itemBoxSizer02->Add(m_pDescription, 0, wxEXPAND | wxALL, 5);
2916 m_pDescription->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2917 m_pDescription->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2918
2919 } else {
2920 wxFlexGridSizer* itemBoxSizer03 = new wxFlexGridSizer(4, 0, 0);
2921 itemBoxSizer03->AddGrowableCol(2);
2922 itemBoxSizer02->Add(itemBoxSizer03, 0, wxEXPAND);
2923
2924 wxString nameString = m_plugin.m_common_name;
2925 m_pName = new wxStaticText(this, wxID_ANY, nameString);
2926 m_pName->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2927 m_pName->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2928
2929 // Avoid known bug in wxGTK3
2930#ifndef __WXGTK3__
2931 wxFont font = GetFont();
2932 font.SetWeight(wxFONTWEIGHT_BOLD);
2933 m_pName->SetFont(font);
2934#endif
2935
2936 itemBoxSizer03->Add(m_pName, 0, /*wxEXPAND|*/ wxALL, 10);
2937
2938 m_pVersion = new wxStaticText(this, wxID_ANY, _T("X.YY.ZZ.AA"));
2939 itemBoxSizer03->Add(m_pVersion, 0, /*wxEXPAND|*/ wxALL, 10);
2940 if (m_plugin.m_status == PluginStatus::ManagedInstallAvailable ||
2941 m_plugin.m_status == PluginStatus::System ||
2942 (m_plugin.m_status == PluginStatus::Unmanaged &&
2943 !m_plugin.m_managed_metadata.is_orphan)) {
2944 m_pVersion->Hide();
2945 }
2946 m_pVersion->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2947 m_pVersion->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2948
2949 m_cbEnable = new wxCheckBox(this, wxID_ANY, _("Enabled"));
2950 itemBoxSizer03->Add(m_cbEnable, 1, wxALIGN_RIGHT | wxTOP, 10);
2951 m_cbEnable->Bind(wxEVT_CHECKBOX, &PluginPanel::OnPluginEnableToggle, this);
2952
2953 itemBoxSizer03->Add(5 * GetCharWidth(), 1, 0, wxALIGN_RIGHT | wxTOP, 10);
2954
2955 m_pDescription = new wxStaticText(
2956 this, wxID_ANY, m_plugin.m_short_description, wxDefaultPosition,
2957 wxSize(-1, -1) /*, wxST_NO_AUTORESIZE*/);
2958 itemBoxSizer02->Add(m_pDescription, 1, wxEXPAND | wxALL, 5);
2959 m_pDescription->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
2960 m_pDescription->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp, this);
2961 }
2962
2963 if (!bCompact) {
2964 m_info_btn = new WebsiteButton(this, "https:\\opencpn.org");
2965 m_info_btn->Hide();
2966 itemBoxSizer02->Add(m_info_btn, 0);
2967
2968 m_pButtons = new wxBoxSizer(wxHORIZONTAL);
2969 itemBoxSizer02->Add(m_pButtons, 0, /*wxEXPAND|*/ wxALL, 0);
2970 m_pButtonPreferences = new wxButton(this, wxID_ANY, _("Preferences"),
2971 wxDefaultPosition, wxDefaultSize, 0);
2972 m_pButtons->Add(m_pButtonPreferences, 0, wxALIGN_LEFT | wxALL, 2);
2973
2974 m_pButtons->AddSpacer(3 * GetCharWidth());
2975
2976 m_pButtonAction =
2977 new wxButton(this, wxID_ANY, "Upgrade to Version XX.XX.XX",
2978 wxDefaultPosition, wxDefaultSize, 0);
2979 m_pButtons->Add(m_pButtonAction, 0, wxALIGN_LEFT | wxALL, 2);
2980
2981 m_pButtonUninstall = new wxButton(this, wxID_ANY, _("Uninstall"),
2982 wxDefaultPosition, wxDefaultSize, 0);
2983 m_pButtons->Add(m_pButtonUninstall, 0, wxALIGN_LEFT | wxALL, 2);
2984 } else {
2985 m_pButtons = new wxBoxSizer(wxVERTICAL);
2986 itemBoxSizer02->Add(m_pButtons, 0, /*wxEXPAND|*/ wxALL, 0);
2987
2988 wxBoxSizer* tline = new wxBoxSizer(wxHORIZONTAL);
2989 m_pButtons->Add(tline, 0, wxALL, 2);
2990
2991 m_pButtonPreferences = new wxButton(this, wxID_ANY, _("Preferences"),
2992 wxDefaultPosition, wxDefaultSize, 0);
2993 tline->Add(m_pButtonPreferences, 0, wxALIGN_LEFT | wxALL, 0);
2994
2995 tline->AddSpacer(3 * GetCharWidth());
2996
2997 m_info_btn = new WebsiteButton(this, "https:\\opencpn.org");
2998 m_info_btn->Hide();
2999 tline->Add(m_info_btn, 0);
3000
3001 m_pButtonAction =
3002 new wxButton(this, wxID_ANY, "Upgrade to Version XX.XX.XX",
3003 wxDefaultPosition, wxDefaultSize);
3004 m_pButtons->Add(m_pButtonAction, 0, wxALIGN_LEFT | wxALL, 2);
3005
3006 m_pButtonUninstall = new wxButton(this, wxID_ANY, _("Uninstall"),
3007 wxDefaultPosition, wxDefaultSize, 0);
3008 m_pButtons->Add(m_pButtonUninstall, 0, wxALIGN_LEFT | wxALL, 2);
3009 }
3010
3011 wxBitmap statusBitmap;
3012 const auto stat = m_plugin.m_status;
3013 auto icon_name = icon_by_status.at(stat);
3014 if (stat == PluginStatus::Imported &&
3015 IsUpdateAvailable(m_plugin.m_managed_metadata)) {
3016 icon_name =
3017 icon_by_status.at(PluginStatus::ManagedInstalledUpdateAvailable);
3018 }
3019
3020 wxFileName path(g_Platform->GetSharedDataDir(), icon_name);
3021 path.AppendDir("uidata");
3022 path.AppendDir("traditional");
3023 bool ok = false;
3024 int bmsize = GetCharWidth() * 3 * dpi_mult;
3025 if (path.IsFileReadable()) {
3026 statusBitmap = LoadSVG(path.GetFullPath(), bmsize, bmsize);
3027 ok = statusBitmap.IsOk();
3028 }
3029 if (!ok) {
3030 auto style = g_StyleManager->GetCurrentStyle();
3031 statusBitmap = wxBitmap(style->GetIcon(_T("default_pi"), bmsize, bmsize));
3032 wxLogMessage("Icon: %s not found.", path.GetFullPath());
3033 }
3034
3035 m_itemStatusIconBitmap = new wxStaticBitmap(this, wxID_ANY, statusBitmap);
3036 m_itemStatusIconBitmap->SetToolTip(message_by_status(stat));
3037 m_itemStatusIconBitmap->Bind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected,
3038 this);
3039 m_itemStatusIconBitmap->Bind(wxEVT_LEFT_UP, &PluginPanel::OnPluginSelectedUp,
3040 this);
3041
3042 itemBoxSizer01->Add(m_itemStatusIconBitmap, 0, wxEXPAND | wxALL, 20);
3043
3044 itemBoxSizer02->AddSpacer(GetCharWidth());
3045
3046 m_pButtonPreferences->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
3047 &PluginPanel::OnPluginPreferences, this);
3048 m_pButtonUninstall->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
3049 &PluginPanel::OnPluginUninstall, this);
3050 m_pButtonAction->Bind(wxEVT_COMMAND_BUTTON_CLICKED,
3051 &PluginPanel::OnPluginAction, this);
3052
3053 SetSelected(m_bSelected);
3054 SetAutoLayout(true);
3055 Fit();
3056}
3057
3058PluginPanel::PluginPanel(wxPanel* parent, wxWindowID id, const wxPoint& pos,
3059 const wxSize& size, PluginMetadata md)
3060 : PluginPanel(parent, id, pos, size, PlugInData(md)) {};
3061
3062PluginPanel::~PluginPanel() {
3063 Unbind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
3064 if (m_is_safe_panel) return;
3065 m_itemStaticBitmap->Unbind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected,
3066 this);
3067 m_pName->Unbind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
3068 m_pVersion->Unbind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
3069 m_pDescription->Unbind(wxEVT_LEFT_DOWN, &PluginPanel::OnPluginSelected, this);
3070 if (m_pButtonAction) {
3071 m_pButtonAction->Unbind(wxEVT_COMMAND_BUTTON_CLICKED,
3072 &PluginPanel::OnPluginAction, this);
3073 }
3074 m_pButtonPreferences->Unbind(wxEVT_COMMAND_BUTTON_CLICKED,
3075 &PluginPanel::OnPluginPreferences, this);
3076 m_cbEnable->Unbind(wxEVT_COMMAND_BUTTON_CLICKED,
3077 &PluginPanel::OnPluginEnableToggle, this);
3078}
3079
3080void PluginPanel::SetActionLabel(wxString& label) {
3081 m_pButtonAction->SetLabel(label);
3082 Refresh();
3083}
3084
3085static wxStopWatch swclick;
3086static int downx, downy;
3087
3088void PluginPanel::OnPluginSelected(wxMouseEvent& event) {
3089#ifdef __ANDROID__
3090 swclick.Start();
3091 event.GetPosition(&downx, &downy);
3092#else
3093 DoPluginSelect();
3094#endif
3095}
3096
3097void PluginPanel::OnPluginSelectedUp(wxMouseEvent& event) {
3098#ifdef __ANDROID__
3099 qDebug() << swclick.Time();
3100 if (swclick.Time() < 200) {
3101 int upx, upy;
3102 event.GetPosition(&upx, &upy);
3103 if ((fabs(upx - downx) < GetCharWidth()) &&
3104 (fabs(upy - downy) < GetCharWidth())) {
3105 DoPluginSelect();
3106 }
3107 }
3108 swclick.Start();
3109#endif
3110}
3111
3112void PluginPanel::DoPluginSelect() {
3113 if (m_plugin.m_status == PluginStatus::ManagedInstallAvailable) {
3114 // auto dialog = dynamic_cast<PluginListPanel*>(GetParent());
3115 // auto dialog = dynamic_cast<PluginListPanel*>(m_parent);
3116 // wxASSERT(dialog != 0);
3117
3118 // Install the new plugin, auto-enabling as a convenience measure.
3119 run_update_dialog(m_PluginListPanel, &m_plugin, false, 0, true);
3120 } else if (m_bSelected) {
3121 SetSelected(false);
3122 m_PluginListPanel->SelectPlugin(NULL);
3123 } else {
3124 SetSelected(true);
3125 m_PluginListPanel->SelectPlugin(this);
3126 }
3127}
3128
3133static PluginMetadata GetMetadataByName(const std::string& name) {
3134 auto plugins = PluginHandler::getInstance()->getInstalled();
3135 auto predicate = [name](const PluginMetadata& pm) { return pm.name == name; };
3136 auto found = std::find_if(plugins.begin(), plugins.end(), predicate);
3137 if (found == plugins.end()) {
3138 wxLogDebug("Cannot find metadata for %s", name.c_str());
3139 }
3140 return found != plugins.end() ? *found : PluginMetadata();
3141}
3142
3143void PluginPanel::SetSelected(bool selected) {
3144 m_bSelected = selected;
3145
3146 m_pVersion->SetLabel(
3147 PluginLoader::GetPluginVersion(m_plugin, GetMetadataByName));
3148 if (selected) {
3149 SetBackgroundColour(GetDialogColor(DLG_SELECTED_BACKGROUND));
3150 m_pButtons->Show(true);
3151 bool unInstallPossible = canUninstall(m_plugin.m_common_name.ToStdString());
3152
3153 // Directly mark Legacy and system plugins as "not uninstallable"
3154 if (m_plugin.m_status == PluginStatus::LegacyUpdateAvailable ||
3155 m_plugin.m_status == PluginStatus::Unmanaged ||
3156 m_plugin.m_status == PluginStatus::System)
3157 unInstallPossible = false;
3158
3159 // Orphan plugins can usually be uninstalled, at best effort.
3160 if (m_plugin.m_managed_metadata.is_orphan) unInstallPossible = true;
3161
3162 m_pButtonUninstall->Show(unInstallPossible);
3163
3164 if (m_plugin.m_managed_metadata.info_url.size()) {
3165 m_info_btn->SetURL(m_plugin.m_managed_metadata.info_url.c_str());
3166 m_info_btn->Show();
3167 }
3168
3169 m_cbEnable->Show(true);
3170
3171 // Configure the "Action" button
3172 wxString label;
3173 SemanticVersion newVersion;
3174 switch (m_plugin.m_status) {
3175 case PluginStatus::LegacyUpdateAvailable:
3176 label = _("Upgrade to Version ");
3177 label += wxString(m_plugin.m_managed_metadata.version.c_str());
3178 m_action = ActionVerb::UPGRADE_TO_MANAGED_VERSION;
3179 m_pButtonAction->Enable();
3180 break;
3181
3182 case PluginStatus::ManagedInstallAvailable:
3183 label = _("Install...");
3184 m_action = ActionVerb::INSTALL_MANAGED_VERSION;
3185 m_pButtonAction->Enable();
3186 break;
3187
3188 case PluginStatus::ManagedInstalledUpdateAvailable:
3189 label = _("Update to ");
3190 label += wxString(m_plugin.m_managed_metadata.version.c_str());
3191 m_action = ActionVerb::UPGRADE_INSTALLED_MANAGED_VERSION;
3192 m_pButtonAction->Enable();
3193 break;
3194
3195 case PluginStatus::ManagedInstalledCurrentVersion:
3196 label = _("Reinstall");
3197 m_action = ActionVerb::REINSTALL_MANAGED_VERSION;
3198 m_pButtonAction->Enable();
3199 break;
3200
3201 case PluginStatus::ManagedInstalledDowngradeAvailable:
3202 label = _("Downgrade");
3203 m_action = ActionVerb::DOWNGRADE_INSTALLED_MANAGED_VERSION;
3204 m_pButtonAction->Enable();
3205 break;
3206
3207 case PluginStatus::Imported:
3208 if (IsUpdateAvailable(m_plugin.m_managed_metadata)) {
3209 label = _("Update");
3210 m_action = ActionVerb::UPDATE_IMPORTED_VERSION;
3211 } else {
3212 m_pButtonAction->Hide();
3213 m_action = ActionVerb::NOP;
3214 }
3215 break;
3216
3217 case PluginStatus::Unmanaged:
3218 m_action = ActionVerb::NOP;
3219 m_pButtonAction->Hide();
3220 break;
3221
3222 case PluginStatus::System:
3223 m_action = ActionVerb::NOP;
3224 m_pButtonAction->Hide();
3225 break;
3226
3227 default:
3228 label = "TBD";
3229 m_action = ActionVerb::NOP;
3230 break;
3231 }
3232 SetActionLabel(label);
3233 Layout();
3234 } else {
3235 SetBackgroundColour(GetDialogColor(DLG_UNSELECTED_BACKGROUND));
3236 // m_pDescription->SetLabel( m_pPlugin->m_short_description );
3237#ifndef __WXQT__
3238 // m_pButtons->Show(false);
3239#else
3240 // m_pButtons->Show(true);
3241#endif
3242 //();
3243
3244 m_pButtons->Show(false);
3245 m_info_btn->Hide();
3246
3247 if (m_plugin.m_status == PluginStatus::ManagedInstallAvailable)
3248 m_cbEnable->Show(false);
3249
3250 Layout();
3251 }
3252
3253 // m_pButtons->Show(selected); // For most platforms, show buttons if
3254 // selected m_pButtonsUpDown->Show(selected);
3255#ifdef __ANDROID__
3256 // Some Android devices (e.g. Kyocera) have trouble with wxBitmapButton...
3257 // m_pButtonsUpDown->Show(false);
3258 // m_pButtons->Show(true); // Always enable buttons for Android
3259#endif
3260
3261 Layout();
3262
3263 if (selected) {
3264 SetBackgroundColour(GetDialogColor(DLG_SELECTED_BACKGROUND));
3265 } else {
3266 SetBackgroundColour(GetDialogColor(DLG_UNSELECTED_BACKGROUND));
3267 }
3268
3269 SetEnabled(m_plugin.m_enabled);
3270
3271#ifdef __ANDROID__
3272 // Android (wxQT) sizers have troubles...
3273 // So we set some layout factors to avoid re-sizing on select/deselect.
3274 // m_rgSizer->Show(true);
3275 // m_pButtons->Show(true);
3276 // m_pButtonAction->Hide();
3277 // m_pButtonUninstall->Hide();
3278
3279 Fit();
3280 // m_PluginListPanel->m_pitemBoxSizer01->Layout();
3281#endif
3282}
3283
3284void PluginPanel::OnPaint(wxPaintEvent& event) {
3285 wxPaintDC dc(this);
3286
3287 int penWidth = m_penWidthUnselected;
3288 wxColour color = GetDialogColor(DLG_UNSELECTED_BACKGROUND);
3289 wxColour border = GetDialogColor(DLG_UNSELECTED_ACCENT);
3290
3291 if (m_bSelected) {
3292 penWidth = m_penWidthSelected;
3293 color = GetDialogColor(DLG_SELECTED_BACKGROUND);
3294 border = GetDialogColor(DLG_SELECTED_ACCENT);
3295 }
3296
3297 wxBrush b(color, wxBRUSHSTYLE_SOLID);
3298 dc.SetBrush(b);
3299 dc.SetPen(wxPen(border, penWidth));
3300
3301 dc.DrawRoundedRectangle(5, 5, GetSize().x - 10, GetSize().y - 10, 5);
3302}
3303
3304void PluginPanel::OnPluginPreferences(wxCommandEvent& event) {
3305 if (m_plugin.m_enabled && m_plugin.m_init_state &&
3306 (m_plugin.m_cap_flag & WANTS_PREFERENCES)) {
3307#ifdef __ANDROID__
3308 androidDisableRotation();
3309 PluginLoader::getInstance()->ShowPreferencesDialog(m_plugin,
3310 GetGrandParent());
3311 // GrandParent will be the entire list panel, not the plugin panel.
3312 // Ensures better centering on small screens
3313#else
3314 PluginLoader::getInstance()->ShowPreferencesDialog(m_plugin, this);
3315#endif
3316 }
3317}
3318
3319void PluginPanel::OnPluginEnableToggle(wxCommandEvent& event) {
3320 SetEnabled(event.IsChecked());
3321 m_pVersion->SetLabel(
3322 PluginLoader::GetPluginVersion(m_plugin, GetMetadataByName));
3323 if (m_plugin.m_status == PluginStatus::System) {
3324 // Force pluginmanager to reload all panels. Not kosher --
3325 // the EventVar should really only be notified from within PluginLoader.
3326 PluginLoader::getInstance()->evt_pluglist_change.Notify();
3327 }
3328}
3329
3330void PluginPanel::OnPluginUninstall(wxCommandEvent& event) {
3331 m_action = ActionVerb::UNINSTALL_MANAGED_VERSION;
3332
3333 // Chain up to the utility event handler
3334 wxCommandEvent actionEvent(wxEVT_COMMAND_BUTTON_CLICKED);
3335 actionEvent.SetId(ID_CMD_BUTTON_PERFORM_ACTION);
3336 actionEvent.SetClientData(this);
3337 g_pi_manager->GetUtilHandler()->AddPendingEvent(actionEvent);
3338}
3339
3340void PluginPanel::OnPluginAction(wxCommandEvent& event) {
3341 // Chain up to the utility event handler
3342 wxCommandEvent actionEvent(wxEVT_COMMAND_BUTTON_CLICKED);
3343 actionEvent.SetId(ID_CMD_BUTTON_PERFORM_ACTION);
3344 actionEvent.SetClientData(this);
3345 g_pi_manager->GetUtilHandler()->AddPendingEvent(actionEvent);
3346
3347 return;
3348}
3349
3350static void SetWindowFontStyle(wxWindow* window, wxFontStyle style) {
3351 auto font = window->GetFont();
3352 font.SetStyle(style);
3353 window->SetFont(font);
3354}
3355
3356void PluginPanel::SetEnabled(bool enabled) {
3357 if (m_is_safe_panel) return;
3358 PluginLoader::getInstance()->SetEnabled(m_plugin.m_common_name, enabled);
3359 PluginLoader::getInstance()->UpdatePlugIns();
3360 NotifySetupOptionsPlugin(&m_plugin);
3361 if (!enabled && !m_bSelected) {
3362 SetWindowFontStyle(m_pName, wxFONTSTYLE_ITALIC);
3363 SetWindowFontStyle(m_pVersion, wxFONTSTYLE_ITALIC);
3364 SetWindowFontStyle(m_pDescription, wxFONTSTYLE_ITALIC);
3365#ifdef x__ANDROID__
3366 m_pName->Disable();
3367 m_pVersion->Disable();
3368 m_pDescription->Disable();
3369#endif
3370 } else {
3371 SetWindowFontStyle(m_pName, wxFONTSTYLE_NORMAL);
3372 SetWindowFontStyle(m_pVersion, wxFONTSTYLE_NORMAL);
3373 SetWindowFontStyle(m_pDescription, wxFONTSTYLE_NORMAL);
3374#ifdef x__ANDROID__
3375 m_pName->Enable();
3376 m_pVersion->Enable();
3377 m_pDescription->Enable();
3378#endif
3379 }
3380
3381#ifdef __ANDROID__
3382 m_pName->Enable(enabled || m_bSelected);
3383 m_pVersion->Enable(enabled || m_bSelected);
3384 m_pDescription->Enable(enabled || m_bSelected);
3385#endif
3386
3387 if (m_bSelected) {
3388 wxString description = m_plugin.m_long_description;
3389 if (description.IsEmpty())
3390 description = wxString(m_plugin.m_managed_metadata.description.c_str());
3391
3392 PanelHardBreakWrapper wrapper(this, description,
3393 g_options->GetSize().x * 7 / 10);
3394 m_pDescription->SetLabel(wrapper.GetWrapped());
3395 if (m_plugin.m_managed_metadata.info_url.size()) {
3396 m_info_btn->SetURL(m_plugin.m_managed_metadata.info_url.c_str());
3397 m_info_btn->Show();
3398 }
3399 } else {
3400 wxString description = m_plugin.m_short_description;
3401 if (description.IsEmpty())
3402 description = wxString(m_plugin.m_managed_metadata.summary.c_str());
3403 PanelHardBreakWrapper wrapper(this, description,
3404 g_options->GetSize().x * 7 / 10);
3405 m_pDescription->SetLabel(wrapper.GetWrapped());
3406 }
3407
3408 m_pButtonPreferences->Enable(enabled &&
3409 (m_plugin.m_cap_flag & WANTS_PREFERENCES));
3410 m_cbEnable->SetValue(enabled);
3411}
3412
3413void PluginPanel::OnPluginUp(wxCommandEvent& event) {
3414 m_PluginListPanel->MoveUp(this);
3415}
3416
3417void PluginPanel::OnPluginDown(wxCommandEvent& event) {
3418 m_PluginListPanel->MoveDown(this);
3419}
3420
3422WebsiteButton::WebsiteButton(wxWindow* parent, const char* url)
3423 : wxPanel(parent), m_url(url) {
3424 auto vbox = new wxBoxSizer(wxVERTICAL);
3425 auto button = new wxButton(this, wxID_ANY, _("Website"));
3426 button->Enable(strlen(url) > 0);
3427 vbox->Add(button);
3428 SetSizer(vbox);
3429 Bind(wxEVT_COMMAND_BUTTON_CLICKED,
3430 [=](wxCommandEvent&) { wxLaunchDefaultBrowser(m_url); });
3431}
3432
3433// ----------------------------------------------------------------------------
3434// PlugInChartBase Implmentation
3435// This class is the base class for Plug-able chart types
3436// ----------------------------------------------------------------------------
3437
3438PlugInChartBase::PlugInChartBase() { m_Chart_Error_Factor = 0.; }
3439
3440PlugInChartBase::~PlugInChartBase() {}
3441
3442wxString PlugInChartBase::GetFileSearchMask(void) { return _T(""); }
3443
3444int PlugInChartBase::Init(const wxString& name, int init_flags) { return 0; }
3445
3446// Accessors
3447
3448double PlugInChartBase::GetNormalScaleMin(double canvas_scale_factor,
3449 bool b_allow_overzoom) {
3450 return 1.0;
3451}
3452
3453double PlugInChartBase::GetNormalScaleMax(double canvas_scale_factor,
3454 int canvas_width) {
3455 return 2.0e7;
3456}
3457
3458bool PlugInChartBase::GetChartExtent(ExtentPI* pext) { return false; }
3459
3461 const wxRegion& Region) {
3462 return wxNullBitmap;
3463}
3464
3466 PlugIn_ViewPort& vp_proposed) {
3467 return false;
3468}
3469
3471 wxRegion* pValidRegion) {}
3472
3473void PlugInChartBase::SetColorScheme(int cs, bool bApplyImmediate) {}
3474
3475double PlugInChartBase::GetNearestPreferredScalePPM(double target_scale_ppm) {
3476 return 1.0;
3477}
3478
3479wxBitmap* PlugInChartBase::GetThumbnail(int tnx, int tny, int cs) {
3480 return NULL;
3481}
3482
3484 wxRect* pSourceRect) {}
3485
3487
3488bool PlugInChartBase::GetChartBits(wxRect& source, unsigned char* pPix,
3489 int sub_samp) {
3490 return false;
3491}
3492
3494
3496
3497void PlugInChartBase::latlong_to_chartpix(double lat, double lon, double& pixx,
3498 double& pixy) {}
3499
3500void PlugInChartBase::chartpix_to_latlong(double pixx, double pixy,
3501 double* plat, double* plon) {}
3502
3503// ----------------------------------------------------------------------------
3504// PlugInChartBaseGL Implementation
3505//
3506// ----------------------------------------------------------------------------
3507
3508PlugInChartBaseGL::PlugInChartBaseGL() {}
3509
3510PlugInChartBaseGL::~PlugInChartBaseGL() {}
3511
3513 const PlugIn_ViewPort& VPoint,
3514 const wxRegion& Region,
3515 bool b_use_stencil) {
3516 return 0;
3517}
3518
3520 float lat, float lon, float select_radius, PlugIn_ViewPort* VPoint) {
3521 return NULL;
3522}
3523
3524wxString PlugInChartBaseGL::CreateObjDescriptions(ListOfPI_S57Obj* obj_list) {
3525 return _T("");
3526}
3527
3529
3530int PlugInChartBaseGL::GetNoCOVRTablePoints(int iTable) { return 0; }
3531
3532int PlugInChartBaseGL::GetNoCOVRTablenPoints(int iTable) { return 0; }
3533
3534float* PlugInChartBaseGL::GetNoCOVRTableHead(int iTable) { return 0; }
3535
3536// ----------------------------------------------------------------------------
3537// PlugInChartBaseExtended Implementation
3538//
3539// ----------------------------------------------------------------------------
3540
3541PlugInChartBaseExtended::PlugInChartBaseExtended() {}
3542
3543PlugInChartBaseExtended::~PlugInChartBaseExtended() {}
3544
3546 const PlugIn_ViewPort& VPoint,
3547 const wxRegion& Region,
3548 bool b_use_stencil) {
3549 return 0;
3550}
3551
3553 const wxGLContext& glc, const PlugIn_ViewPort& VPoint,
3554 const wxRegion& Region, bool b_use_stencil) {
3555 return 0;
3556}
3557
3559 const wxGLContext& glc, const PlugIn_ViewPort& VPoint,
3560 const wxRegion& Region, bool b_use_stencil) {
3561 return 0;
3562}
3563
3565 const PlugIn_ViewPort& VPoint, const wxRegion& Region) {
3566 return wxNullBitmap;
3567}
3568
3570 wxMemoryDC& dc, const PlugIn_ViewPort& VPoint, const wxRegion& Region) {
3571 return false;
3572}
3573
3574ListOfPI_S57Obj* PlugInChartBaseExtended::GetObjRuleListAtLatLon(
3575 float lat, float lon, float select_radius, PlugIn_ViewPort* VPoint) {
3576 return NULL;
3577}
3578
3579wxString PlugInChartBaseExtended::CreateObjDescriptions(
3580 ListOfPI_S57Obj* obj_list) {
3581 return _T("");
3582}
3583
3584int PlugInChartBaseExtended::GetNoCOVREntries() { return 0; }
3585
3586int PlugInChartBaseExtended::GetNoCOVRTablePoints(int iTable) { return 0; }
3587
3588int PlugInChartBaseExtended::GetNoCOVRTablenPoints(int iTable) { return 0; }
3589
3590float* PlugInChartBaseExtended::GetNoCOVRTableHead(int iTable) { return 0; }
3591
3593
3594// ----------------------------------------------------------------------------
3595// PlugInChartBaseExtendedPlus2 Implementation
3596//
3597// ----------------------------------------------------------------------------
3598
3599PlugInChartBaseExtendedPlus2::PlugInChartBaseExtendedPlus2() {}
3600
3601PlugInChartBaseExtendedPlus2::~PlugInChartBaseExtendedPlus2() {}
3602
3603ListOfPI_S57Obj*
3604PlugInChartBaseExtendedPlus2::GetLightsObjRuleListVisibleAtLatLon(
3605 float lat, float lon, PlugIn_ViewPort* VPoint) {
3606 return NULL;
3607}
3608
3609// ----------------------------------------------------------------------------
3610// PlugInChartBaseGLPlus2 Implementation
3611//
3612// ----------------------------------------------------------------------------
3613
3614PlugInChartBaseGLPlus2::PlugInChartBaseGLPlus2() {}
3615
3616PlugInChartBaseGLPlus2::~PlugInChartBaseGLPlus2() {}
3617
3619 float lat, float lon, PlugIn_ViewPort* VPoint) {
3620 return NULL;
3621}
3622
3623// ----------------------------------------------------------------------------
3624// ChartPlugInWrapper Implementation
3625// This class is a wrapper/interface to PlugIn charts(PlugInChartBase)
3626// ----------------------------------------------------------------------------
3627
3628ChartPlugInWrapper::ChartPlugInWrapper() {}
3629
3630ChartPlugInWrapper::ChartPlugInWrapper(const wxString& chart_class) {
3631 m_ppo = ::wxCreateDynamicObject(chart_class);
3632 m_ppicb = dynamic_cast<PlugInChartBase*>(m_ppo);
3633}
3634
3635ChartPlugInWrapper::~ChartPlugInWrapper() {
3636 if (m_ppicb) delete m_ppicb;
3637}
3638
3639wxString ChartPlugInWrapper::GetFileSearchMask(void) {
3640 if (m_ppicb)
3641 return m_ppicb->GetFileSearchMask();
3642 else
3643 return _T("");
3644}
3645
3646InitReturn ChartPlugInWrapper::Init(const wxString& name,
3647 ChartInitFlag init_flags) {
3648 if (m_ppicb) {
3649 wxWindow* pa = wxWindow::FindFocus();
3650
3651 InitReturn ret_val = (InitReturn)m_ppicb->Init(name, (int)init_flags);
3652
3653 // Here we transcribe all the required wrapped member elements up into
3654 // the chartbase object which is the parent of this class
3655 if (ret_val == INIT_OK) {
3656 m_FullPath = m_ppicb->GetFullPath();
3657 m_ChartType = (ChartTypeEnum)m_ppicb->GetChartType();
3658 m_ChartFamily = (ChartFamilyEnum)m_ppicb->GetChartFamily();
3659 m_projection = (OcpnProjType)m_ppicb->GetChartProjection();
3660 m_EdDate = m_ppicb->GetEditionDate();
3661 m_Name = m_ppicb->GetName();
3662 m_ID = m_ppicb->GetID();
3663 m_DepthUnits = m_ppicb->GetDepthUnits();
3664 m_SoundingsDatum = m_ppicb->GetSoundingsDatum();
3665 m_datum_str = m_ppicb->GetDatumString();
3666 m_SE = m_ppicb->GetSE();
3667 m_EdDate = m_ppicb->GetEditionDate();
3668 m_ExtraInfo = m_ppicb->GetExtraInfo();
3669 Chart_Error_Factor = m_ppicb->GetChartErrorFactor();
3670 m_depth_unit_id = (ChartDepthUnitType)m_ppicb->GetDepthUnitId();
3671 m_Chart_Skew = m_ppicb->GetChartSkew();
3672 m_Chart_Scale = m_ppicb->GetNativeScale();
3673
3674 // We estimate ppm_avg as needed by raster texture cache logic...
3675 // This number works for average BSB charts, scanned with average
3676 // resolution
3677 m_ppm_avg = 10000. / m_ppicb->GetNativeScale(); // fallback value
3678
3679 // Calcuculate a "better" ppm from the chart geo extent and raster size.
3680 if ((fabs(m_Chart_Skew) < .01) &&
3681 (CHART_FAMILY_RASTER == m_ChartFamily)) {
3682 Extent extent;
3683 if (GetChartExtent(&extent)) {
3684 double lon_range = extent.ELON - extent.WLON;
3685 if ((lon_range > 0) &&
3686 (lon_range < 90.0)) // Be safe about IDL crossing and huge charts
3687 m_ppm_avg = GetSize_X() / (lon_range * 1852 * 60);
3688 }
3689 }
3690
3691 m_overlayENC = false;
3692 if (m_ChartFamily == (ChartFamilyEnum)PI_CHART_FAMILY_VECTOR) {
3693 wxCharBuffer buf = m_FullPath.ToUTF8();
3694 m_overlayENC = s57chart::IsCellOverlayType(buf.data());
3695 }
3696
3697 bReadyToRender = m_ppicb->IsReadyToRender();
3698 } else {
3699 // Mark the chart as unable to render
3700 m_ChartType = CHART_TYPE_UNKNOWN;
3701 m_ChartFamily = CHART_FAMILY_UNKNOWN;
3702 }
3703
3704 // PlugIn may invoke wxExecute(), which steals the keyboard focus
3705 // So take it back
3706 auto pc = dynamic_cast<ChartCanvas*>(pa);
3707 if (pc) pc->SetFocus();
3708
3709 return ret_val;
3710 } else
3711 return INIT_FAIL_REMOVE;
3712}
3713
3714// Accessors
3715int ChartPlugInWrapper::GetCOVREntries() {
3716 if (m_ppicb)
3717 return m_ppicb->GetCOVREntries();
3718 else
3719 return 0;
3720}
3721
3722int ChartPlugInWrapper::GetCOVRTablePoints(int iTable) {
3723 if (m_ppicb)
3724 return m_ppicb->GetCOVRTablePoints(iTable);
3725 else
3726 return 0;
3727}
3728
3729int ChartPlugInWrapper::GetCOVRTablenPoints(int iTable) {
3730 if (m_ppicb)
3731 return m_ppicb->GetCOVRTablenPoints(iTable);
3732 else
3733 return 0;
3734}
3735
3736float* ChartPlugInWrapper::GetCOVRTableHead(int iTable) {
3737 if (m_ppicb)
3738 return m_ppicb->GetCOVRTableHead(iTable);
3739 else
3740 return 0;
3741}
3742
3743// TODO
3744// PlugIn chart types do not properly support NoCovr Regions
3745// Proper fix is to update PlugIn Chart Type API
3746// Derive an extended PlugIn chart class from existing class,
3747// and use some kind of RTTI to figure out which class to call.
3748int ChartPlugInWrapper::GetNoCOVREntries() {
3749 if (m_ppicb) {
3750 PlugInChartBaseGL* ppicbgl = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3751 if (ppicbgl) {
3752 return ppicbgl->GetNoCOVREntries();
3753 }
3754 }
3755 return 0;
3756}
3757
3758int ChartPlugInWrapper::GetNoCOVRTablePoints(int iTable) {
3759 if (m_ppicb) {
3760 PlugInChartBaseGL* ppicbgl = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3761 if (ppicbgl) {
3762 return ppicbgl->GetNoCOVRTablePoints(iTable);
3763 }
3764 }
3765 return 0;
3766}
3767
3768int ChartPlugInWrapper::GetNoCOVRTablenPoints(int iTable) {
3769 if (m_ppicb) {
3770 PlugInChartBaseGL* ppicbgl = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3771 if (ppicbgl) {
3772 return ppicbgl->GetNoCOVRTablenPoints(iTable);
3773 }
3774 }
3775 return 0;
3776}
3777
3778float* ChartPlugInWrapper::GetNoCOVRTableHead(int iTable) {
3779 if (m_ppicb) {
3780 PlugInChartBaseGL* ppicbgl = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3781 if (ppicbgl) {
3782 return ppicbgl->GetNoCOVRTableHead(iTable);
3783 }
3784 }
3785 return 0;
3786}
3787
3788bool ChartPlugInWrapper::GetChartExtent(Extent* pext) {
3789 if (m_ppicb) {
3790 ExtentPI xpi;
3791 if (m_ppicb->GetChartExtent(&xpi)) {
3792 pext->NLAT = xpi.NLAT;
3793 pext->SLAT = xpi.SLAT;
3794 pext->ELON = xpi.ELON;
3795 pext->WLON = xpi.WLON;
3796
3797 return true;
3798 } else
3799 return false;
3800 } else
3801 return false;
3802}
3803
3804ThumbData* ChartPlugInWrapper::GetThumbData(int tnx, int tny, float lat,
3805 float lon) {
3806 if (m_ppicb) {
3807 // Create the bitmap if needed, doing a deep copy from the Bitmap owned
3808 // by the PlugIn Chart
3809 if (!pThumbData->pDIBThumb) {
3810 wxBitmap* pBMPOwnedByChart =
3811 m_ppicb->GetThumbnail(tnx, tny, m_global_color_scheme);
3812 if (pBMPOwnedByChart) {
3813 wxImage img = pBMPOwnedByChart->ConvertToImage();
3814 pThumbData->pDIBThumb = new wxBitmap(img);
3815 } else
3816 pThumbData->pDIBThumb = NULL;
3817 }
3818
3819 pThumbData->Thumb_Size_X = tnx;
3820 pThumbData->Thumb_Size_Y = tny;
3821
3822 /*
3823 // Plot the supplied Lat/Lon on the thumbnail
3824 int divx = m_ppicb->Size_X / tnx;
3825 int divy = m_ppicb->Size_Y / tny;
3826
3827 int div_factor = __min(divx, divy);
3828
3829 int pixx, pixy;
3830
3831
3832 // Using a temporary synthetic ViewPort and source rectangle,
3833 // calculate the ships position on the thumbnail
3834 ViewPort tvp;
3835 tvp.pix_width = tnx;
3836 tvp.pix_height = tny;
3837 tvp.view_scale_ppm = GetPPM() / div_factor;
3838 wxRect trex = Rsrc;
3839 Rsrc.x = 0;
3840 Rsrc.y = 0;
3841 latlong_to_pix_vp(lat, lon, pixx, pixy, tvp);
3842 Rsrc = trex;
3843
3844 pThumbData->ShipX = pixx;// / div_factor;
3845 pThumbData->ShipY = pixy;// / div_factor;
3846 */
3847 pThumbData->ShipX = 0;
3848 pThumbData->ShipY = 0;
3849
3850 return pThumbData;
3851 } else
3852 return NULL;
3853}
3854
3855ThumbData* ChartPlugInWrapper::GetThumbData() { return pThumbData; }
3856
3857bool ChartPlugInWrapper::UpdateThumbData(double lat, double lon) {
3858 return true;
3859}
3860
3861double ChartPlugInWrapper::GetNormalScaleMin(double canvas_scale_factor,
3862 bool b_allow_overzoom) {
3863 if (m_ppicb)
3864 return m_ppicb->GetNormalScaleMin(canvas_scale_factor, b_allow_overzoom);
3865 else
3866 return 1.0;
3867}
3868
3869double ChartPlugInWrapper::GetNormalScaleMax(double canvas_scale_factor,
3870 int canvas_width) {
3871 if (m_ppicb)
3872 return m_ppicb->GetNormalScaleMax(canvas_scale_factor, canvas_width);
3873 else
3874 return 2.0e7;
3875}
3876
3877/* RectRegion:
3878 * This is the Screen region desired to be updated. Will
3879 * be either 1 rectangle(full screen) or two rectangles (panning with FBO
3880 * accelerated pan logic)
3881 *
3882 * Region:
3883 * This is the LLRegion describing the quilt active region
3884 * for this chart.
3885 *
3886 * So, Actual rendering area onscreen should be clipped to the
3887 * intersection of the two regions.
3888 */
3889
3890// Render helpers
3891void RenderRotateToViewPort(const ViewPort& VPoint) {
3892#ifndef USE_ANDROID_GLES2
3893 float xt = VPoint.pix_width / 2.0, yt = VPoint.pix_height / 2.0;
3894 glTranslatef(xt, yt, 0);
3895 glRotatef(VPoint.rotation * 180. / PI, 0, 0, 1);
3896 glTranslatef(-xt, -yt, 0);
3897#endif
3898}
3899
3900void UndoRenderRotateToViewPort(const ViewPort& VPoint) {
3901#ifndef USE_ANDROID_GLES2
3902 float xt = VPoint.pix_width / 2.0, yt = VPoint.pix_height / 2.0;
3903 glTranslatef(xt, yt, 0);
3904 glRotatef(-VPoint.rotation * 180. / PI, 0, 0, 1);
3905 glTranslatef(-xt, -yt, 0);
3906#endif
3907}
3908
3909bool ChartPlugInWrapper::RenderRegionViewOnGL(const wxGLContext& glc,
3910 const ViewPort& VPoint,
3911 const OCPNRegion& RectRegion,
3912 const LLRegion& Region) {
3913#ifdef ocpnUSE_GL
3914 if (m_ppicb) {
3915 ViewPort vp = VPoint; // non-const copy
3916
3917 gs_plib_flags = 0; // reset the CAPs flag
3918 PlugInChartBaseGL* ppicb_gl = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3919 PlugInChartBaseExtended* ppicb_x =
3920 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
3921 if (!Region.Empty() && (ppicb_gl || ppicb_x)) {
3922 wxRegion* r = RectRegion.GetNew_wxRegion();
3923 for (OCPNRegionIterator upd(RectRegion); upd.HaveRects();
3924 upd.NextRect()) {
3925 LLRegion chart_region = vp.GetLLRegion(upd.GetRect());
3926 chart_region.Intersect(Region);
3927
3928 if (!chart_region.Empty()) {
3929 ViewPort cvp = glChartCanvas::ClippedViewport(VPoint, chart_region);
3930
3931 glChartCanvas::SetClipRect(cvp, upd.GetRect(), false);
3932
3933 // ps52plib->m_last_clip_rect = upd.GetRect();
3934
3935#ifndef USE_ANDROID_GLES2
3936// glPushMatrix(); // Adjust for rotation
3937#endif
3938 RenderRotateToViewPort(VPoint);
3939
3940 PlugIn_ViewPort pivp = CreatePlugInViewport(cvp);
3941 if (ppicb_x)
3942 ppicb_x->RenderRegionViewOnGL(glc, pivp, *r,
3943 glChartCanvas::s_b_useStencil);
3944 else if (ppicb_gl)
3945 ppicb_gl->RenderRegionViewOnGL(glc, pivp, *r,
3946 glChartCanvas::s_b_useStencil);
3947 UndoRenderRotateToViewPort(VPoint);
3948
3949#ifndef USE_ANDROID_GLES2
3950// glPopMatrix();
3951#endif
3952 glChartCanvas::DisableClipRegion();
3953
3954 }
3955 } // for
3956 delete r;
3957 }
3958 } else
3959 return false;
3960#endif
3961 return true;
3962}
3963
3964// int indexrr;
3965
3966bool ChartPlugInWrapper::RenderRegionViewOnGLNoText(
3967 const wxGLContext& glc, const ViewPort& VPoint,
3968 const OCPNRegion& RectRegion, const LLRegion& Region) {
3969#ifdef ocpnUSE_GL
3970 if (m_ppicb) {
3971 // printf("\nCPIW::RRVOGLNT %d %d \n", indexrr++, m_Chart_Scale);
3972
3973 gs_plib_flags = 0; // reset the CAPs flag
3974 PlugInChartBaseExtended* ppicb_x =
3975 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
3976 PlugInChartBaseGL* ppicb = dynamic_cast<PlugInChartBaseGL*>(m_ppicb);
3977 if (!Region.Empty() && ppicb_x) {
3978 // Start with a clean slate
3979 glChartCanvas::SetClipRect(VPoint, VPoint.rv_rect, false);
3980 glChartCanvas::DisableClipRegion();
3981
3982 // Apply rotation to this chart
3983 RenderRotateToViewPort(VPoint);
3984
3985 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
3986 wxRegion* r = RectRegion.GetNew_wxRegion();
3987
3988 ppicb_x->RenderRegionViewOnGLNoText(glc, pivp, *r,
3989 glChartCanvas::s_b_useStencil);
3990
3991 // Undo rotation
3992 UndoRenderRotateToViewPort(VPoint);
3993
3994 delete r;
3995 }
3996
3997 else if (!Region.Empty() &&
3998 ppicb) // Legacy Vector GL Plugin chart (e.g.S63)
3999 {
4000 ViewPort vp = VPoint; // non-const copy
4001 wxRegion* r = RectRegion.GetNew_wxRegion();
4002 for (OCPNRegionIterator upd(RectRegion); upd.HaveRects();
4003 upd.NextRect()) {
4004 LLRegion chart_region = vp.GetLLRegion(upd.GetRect());
4005 chart_region.Intersect(Region);
4006
4007 if (!chart_region.Empty()) {
4008 ViewPort cvp = glChartCanvas::ClippedViewport(VPoint, chart_region);
4009
4010 glChartCanvas::SetClipRect(cvp, upd.GetRect(), false);
4011
4012 RenderRotateToViewPort(VPoint);
4013
4014 PlugIn_ViewPort pivp = CreatePlugInViewport(cvp);
4015 ppicb->RenderRegionViewOnGL(glc, pivp, *r,
4016 glChartCanvas::s_b_useStencil);
4017
4018 // Undo rotation
4019 UndoRenderRotateToViewPort(VPoint);
4020
4021 glChartCanvas::DisableClipRegion();
4022
4023 }
4024 } // for
4025 delete r;
4026 }
4027
4028 } else
4029 return false;
4030#endif
4031 return true;
4032}
4033
4034bool ChartPlugInWrapper::RenderRegionViewOnGLTextOnly(
4035 const wxGLContext& glc, const ViewPort& VPoint, const OCPNRegion& Region) {
4036#ifdef ocpnUSE_GL
4037 if (m_ppicb) {
4038 gs_plib_flags = 0; // reset the CAPs flag
4039 PlugInChartBaseExtended* ppicb_x =
4040 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
4041 if (!Region.Empty() && ppicb_x) {
4042 wxRegion* r = Region.GetNew_wxRegion();
4043 for (OCPNRegionIterator upd(Region); upd.HaveRects(); upd.NextRect()) {
4044#ifndef USE_ANDROID_GLES2
4045// glPushMatrix(); // Adjust for rotation
4046#endif
4047 RenderRotateToViewPort(VPoint);
4048
4049 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4050 ppicb_x->RenderRegionViewOnGLTextOnly(glc, pivp, *r,
4051 glChartCanvas::s_b_useStencil);
4052 UndoRenderRotateToViewPort(VPoint);
4053
4054#ifndef USE_ANDROID_GLES2
4055// glPopMatrix();
4056#endif
4057
4058 } // for
4059 delete r;
4060 }
4061 } else
4062 return false;
4063#endif
4064 return true;
4065}
4066
4067bool ChartPlugInWrapper::RenderRegionViewOnDC(wxMemoryDC& dc,
4068 const ViewPort& VPoint,
4069 const OCPNRegion& Region) {
4070 if (m_ppicb) {
4071 gs_plib_flags = 0; // reset the CAPs flag
4072 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4073 if (Region.IsOk()) {
4074 wxRegion* r = Region.GetNew_wxRegion();
4075 if (!m_overlayENC)
4076 dc.SelectObject(m_ppicb->RenderRegionView(pivp, *r));
4077 else {
4078 wxBitmap& obmp = m_ppicb->RenderRegionView(pivp, *r);
4079
4080 // Create a mask to remove the NODTA areas from overlay cells.
4081 wxColour nodat = GetGlobalColor(_T ( "NODTA" ));
4082 wxColour nodat_sub = nodat;
4083
4084#ifdef ocpnUSE_ocpnBitmap
4085 nodat_sub = wxColour(nodat.Blue(), nodat.Green(), nodat.Red());
4086#endif
4087 m_pMask = new wxMask(obmp, nodat_sub);
4088 obmp.SetMask(m_pMask);
4089
4090 dc.SelectObject(obmp);
4091 }
4092
4093 delete r;
4094 return true;
4095 } else
4096 return false;
4097 } else
4098 return false;
4099}
4100
4101bool ChartPlugInWrapper::RenderRegionViewOnDCNoText(wxMemoryDC& dc,
4102 const ViewPort& VPoint,
4103 const OCPNRegion& Region) {
4104 if (m_ppicb) {
4105 gs_plib_flags = 0; // reset the CAPs flag
4106 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4107
4109 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
4110 PlugInChartBase* ppicb = dynamic_cast<PlugInChartBase*>(m_ppicb);
4111
4112 if (Region.IsOk() && (pCBx || ppicb)) {
4113 wxRegion* r = Region.GetNew_wxRegion();
4114
4115 if (pCBx)
4116 dc.SelectObject(pCBx->RenderRegionViewOnDCNoText(pivp, *r));
4117 else if (ppicb)
4118 dc.SelectObject(ppicb->RenderRegionView(pivp, *r));
4119
4120 delete r;
4121 return true;
4122 } else
4123 return false;
4124 } else
4125 return false;
4126}
4127
4128bool ChartPlugInWrapper::RenderRegionViewOnDCTextOnly(
4129 wxMemoryDC& dc, const ViewPort& VPoint, const OCPNRegion& Region) {
4130 if (m_ppicb) {
4131 bool ret_val = false;
4132 gs_plib_flags = 0; // reset the CAPs flag
4133 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4134 if (Region.IsOk()) {
4135 wxRegion* r = Region.GetNew_wxRegion();
4136
4138 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
4139 if (pCBx) ret_val = pCBx->RenderRegionViewOnDCTextOnly(dc, pivp, *r);
4140
4141 delete r;
4142 return ret_val;
4143 } else
4144 return false;
4145 } else
4146 return false;
4147}
4148
4149void ChartPlugInWrapper::ClearPLIBTextList() {
4150 if (m_ppicb) {
4152 dynamic_cast<PlugInChartBaseExtended*>(m_ppicb);
4153 if (pCBx) pCBx->ClearPLIBTextList();
4154 }
4155}
4156
4157bool ChartPlugInWrapper::AdjustVP(ViewPort& vp_last, ViewPort& vp_proposed) {
4158 if (m_ppicb) {
4159 PlugIn_ViewPort pivp_last = CreatePlugInViewport(vp_last);
4160 PlugIn_ViewPort pivp_proposed = CreatePlugInViewport(vp_proposed);
4161 return m_ppicb->AdjustVP(pivp_last, pivp_proposed);
4162 } else
4163 return false;
4164}
4165
4166void ChartPlugInWrapper::GetValidCanvasRegion(const ViewPort& VPoint,
4167 OCPNRegion* pValidRegion) {
4168 if (m_ppicb) {
4169 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4170 // currently convert using wxRegion,
4171 // this should be changed as wxRegion is proven unstable/buggy on various
4172 // platforms
4173 wxRegion region;
4174 m_ppicb->GetValidCanvasRegion(pivp, &region);
4175 *pValidRegion = OCPNRegion(region);
4176 }
4177
4178 return;
4179}
4180
4181void ChartPlugInWrapper::SetColorScheme(ColorScheme cs, bool bApplyImmediate) {
4182 if (m_ppicb) {
4183 m_ppicb->SetColorScheme(cs, bApplyImmediate);
4184 }
4185 m_global_color_scheme = cs;
4186 // Force a new thumbnail
4187 if (pThumbData) pThumbData->pDIBThumb = NULL;
4188}
4189
4191 double target_scale_ppm) {
4192 if (m_ppicb)
4193 return m_ppicb->GetNearestPreferredScalePPM(target_scale_ppm);
4194 else
4195 return 1.0;
4196}
4197
4198void ChartPlugInWrapper::ComputeSourceRectangle(const ViewPort& VPoint,
4199 wxRect* pSourceRect) {
4200 if (m_ppicb) {
4201 PlugIn_ViewPort pivp = CreatePlugInViewport(VPoint);
4202 m_ppicb->ComputeSourceRectangle(pivp, pSourceRect);
4203 }
4204}
4205
4206double ChartPlugInWrapper::GetRasterScaleFactor(const ViewPort& vp) {
4207 if (m_ppicb) {
4208 return (wxRound(100000 * GetPPM() / vp.view_scale_ppm)) / 100000.;
4209 } else
4210 return 1.0;
4211}
4212
4213bool ChartPlugInWrapper::GetChartBits(wxRect& source, unsigned char* pPix,
4214 int sub_samp) {
4215 wxCriticalSectionLocker locker(m_critSect);
4216
4217 if (m_ppicb)
4218
4219 return m_ppicb->GetChartBits(source, pPix, sub_samp);
4220 else
4221 return false;
4222}
4223
4224int ChartPlugInWrapper::GetSize_X() {
4225 if (m_ppicb)
4226 return m_ppicb->GetSize_X();
4227 else
4228 return 1;
4229}
4230
4231int ChartPlugInWrapper::GetSize_Y() {
4232 if (m_ppicb)
4233 return m_ppicb->GetSize_Y();
4234 else
4235 return 1;
4236}
4237
4238void ChartPlugInWrapper::latlong_to_chartpix(double lat, double lon,
4239 double& pixx, double& pixy) {
4240 if (m_ppicb) m_ppicb->latlong_to_chartpix(lat, lon, pixx, pixy);
4241}
4242
4243void ChartPlugInWrapper::chartpix_to_latlong(double pixx, double pixy,
4244 double* plat, double* plon) {
4245 if (m_ppicb) m_ppicb->chartpix_to_latlong(pixx, pixy, plat, plon);
4246}
4247
4248//----------------------------------------------------------------------------------------------------------
4249// The PlugIn CallBack API Implementation
4250// The definitions of this API are found in ocpn_plugin.h
4251//----------------------------------------------------------------------------------------------------------
4252
4253/* API 1.11 */
4254
4255/* API 1.11 adds some more common functions to avoid unnecessary code
4256 * duplication */
4257
4258wxString toSDMM_PlugIn(int NEflag, double a, bool hi_precision) {
4259 return toSDMM(NEflag, a, hi_precision);
4260}
4261
4262wxColour GetBaseGlobalColor(wxString colorName) {
4263 return GetGlobalColor(colorName);
4264}
4265
4266int OCPNMessageBox_PlugIn(wxWindow* parent, const wxString& message,
4267 const wxString& caption, int style, int x, int y) {
4268 return OCPNMessageBox(parent, message, caption, style, 100, x, y);
4269}
4270
4271wxString GetOCPN_ExePath(void) { return g_Platform->GetExePath(); }
4272
4273wxString* GetpPlugInLocation() { return g_Platform->GetPluginDirPtr(); }
4274
4275wxString GetWritableDocumentsDir(void) {
4276 return g_Platform->GetWritableDocumentsDir();
4277}
4278
4279wxString GetPlugInPath(opencpn_plugin* pplugin) {
4280 wxString ret_val;
4281 auto loader = PluginLoader::getInstance();
4282 for (unsigned int i = 0; i < loader->GetPlugInArray()->GetCount(); i++) {
4283 PlugInContainer* pic = loader->GetPlugInArray()->Item(i);
4284 if (pic->m_pplugin == pplugin) {
4285 ret_val = pic->m_plugin_file;
4286 break;
4287 }
4288 }
4289 return ret_val;
4290}
4291
4292// API 1.11 Access to Vector PlugIn charts
4293
4294ListOfPI_S57Obj* PlugInManager::GetPlugInObjRuleListAtLatLon(
4295 ChartPlugInWrapper* target, float zlat, float zlon, float SelectRadius,
4296 const ViewPort& vp) {
4297 ListOfPI_S57Obj* list = NULL;
4298 if (target) {
4299 PlugInChartBaseGL* picbgl =
4300 dynamic_cast<PlugInChartBaseGL*>(target->GetPlugInChart());
4301 if (picbgl) {
4302 PlugIn_ViewPort pi_vp = CreatePlugInViewport(vp);
4303 list = picbgl->GetObjRuleListAtLatLon(zlat, zlon, SelectRadius, &pi_vp);
4304
4305 return list;
4306 }
4308 dynamic_cast<PlugInChartBaseExtended*>(target->GetPlugInChart());
4309 if (picbx) {
4310 PlugIn_ViewPort pi_vp = CreatePlugInViewport(vp);
4311 list = picbx->GetObjRuleListAtLatLon(zlat, zlon, SelectRadius, &pi_vp);
4312
4313 return list;
4314 } else
4315 return list;
4316 } else
4317 return list;
4318}
4319
4320wxString PlugInManager::CreateObjDescriptions(ChartPlugInWrapper* target,
4321 ListOfPI_S57Obj* rule_list) {
4322 wxString ret_str;
4323 if (target) {
4324 PlugInChartBaseGL* picbgl =
4325 dynamic_cast<PlugInChartBaseGL*>(target->GetPlugInChart());
4326 if (picbgl) {
4327 ret_str = picbgl->CreateObjDescriptions(rule_list);
4328 } else {
4330 dynamic_cast<PlugInChartBaseExtended*>(target->GetPlugInChart());
4331 if (picbx) {
4332 ret_str = picbx->CreateObjDescriptions(rule_list);
4333 }
4334 }
4335 }
4336 return ret_str;
4337}
4338
4339// API 1.11 Access to S52 PLIB
4340wxString PI_GetPLIBColorScheme() {
4341 return _T(""); // ps52plib->GetPLIBColorScheme()
4342}
4343
4344int PI_GetPLIBDepthUnitInt() {
4345 if (ps52plib)
4346 return ps52plib->m_nDepthUnitDisplay;
4347 else
4348 return 0;
4349}
4350
4351int PI_GetPLIBSymbolStyle() {
4352 if (ps52plib)
4353 return ps52plib->m_nSymbolStyle;
4354 else
4355 return 0;
4356}
4357
4358int PI_GetPLIBBoundaryStyle() {
4359 if (ps52plib)
4360 return ps52plib->m_nBoundaryStyle;
4361 else
4362 return 0;
4363}
4364
4365bool PI_PLIBObjectRenderCheck(PI_S57Obj* pObj, PlugIn_ViewPort* vp) {
4366 if (ps52plib) {
4367 // Create and populate a compatible s57 Object
4368 S57Obj cobj;
4369 chart_context ctx;
4370 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4371
4372 ViewPort cvp = CreateCompatibleViewport(*vp);
4373
4374 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4375
4376 // Create and populate a minimally compatible object container
4377 ObjRazRules rzRules;
4378 rzRules.obj = &cobj;
4379 rzRules.LUP = pContext->LUP;
4380 rzRules.sm_transform_parms = 0;
4381 rzRules.child = NULL;
4382 rzRules.next = NULL;
4383
4384 if (pContext->LUP) {
4385 ps52plib->SetVPointCompat(
4386 cvp.pix_width, cvp.pix_height, cvp.view_scale_ppm, cvp.rotation,
4387 cvp.clat, cvp.clon, cvp.chart_scale, cvp.rv_rect, cvp.GetBBox(),
4388 cvp.ref_scale, GetOCPNCanvasWindow()->GetContentScaleFactor());
4389 ps52plib->PrepareForRender();
4390
4391 return ps52plib->ObjectRenderCheck(&rzRules);
4392 } else
4393 return false;
4394 } else
4395 return false;
4396}
4397
4398int PI_GetPLIBStateHash() {
4399 if (ps52plib)
4400 return ps52plib->GetStateHash();
4401 else
4402 return 0;
4403}
4404
4405void CreateCompatibleS57Object(PI_S57Obj* pObj, S57Obj* cobj,
4406 chart_context* pctx) {
4407 strncpy(cobj->FeatureName, pObj->FeatureName, 8);
4408 cobj->Primitive_type = (GeoPrim_t)pObj->Primitive_type;
4409 cobj->att_array = pObj->att_array;
4410 cobj->attVal = pObj->attVal;
4411 cobj->n_attr = pObj->n_attr;
4412
4413 cobj->x = pObj->x;
4414 cobj->y = pObj->y;
4415 cobj->z = pObj->z;
4416 cobj->npt = pObj->npt;
4417
4418 cobj->iOBJL = pObj->iOBJL;
4419 cobj->Index = pObj->Index;
4420
4421 cobj->geoPt = (pt*)pObj->geoPt;
4422 cobj->geoPtz = pObj->geoPtz;
4423 cobj->geoPtMulti = pObj->geoPtMulti;
4424
4425 cobj->m_lat = pObj->m_lat;
4426 cobj->m_lon = pObj->m_lon;
4427
4428 cobj->m_DisplayCat = (DisCat)pObj->m_DisplayCat;
4429 cobj->x_rate = pObj->x_rate;
4430 cobj->y_rate = pObj->y_rate;
4431 cobj->x_origin = pObj->x_origin;
4432 cobj->y_origin = pObj->y_origin;
4433
4434 cobj->Scamin = pObj->Scamin;
4435 cobj->nRef = pObj->nRef;
4436 cobj->bIsAton = pObj->bIsAton;
4437 cobj->bIsAssociable = pObj->bIsAssociable;
4438
4439 cobj->m_n_lsindex = pObj->m_n_lsindex;
4440 cobj->m_lsindex_array = pObj->m_lsindex_array;
4441 cobj->m_n_edge_max_points = pObj->m_n_edge_max_points;
4442
4443 if (gs_plib_flags & PLIB_CAPS_OBJSEGLIST) {
4444 cobj->m_ls_list_legacy =
4446 pObj->m_ls_list; // note the cast, assumes in-sync layout
4447 } else
4448 cobj->m_ls_list_legacy = 0;
4449 cobj->m_ls_list = 0;
4450
4451 if (gs_plib_flags & PLIB_CAPS_OBJCATMUTATE)
4452 cobj->m_bcategory_mutable = pObj->m_bcategory_mutable;
4453 else
4454 cobj->m_bcategory_mutable = true; // assume all objects are mutable
4455
4456 cobj->m_DPRI = -1; // default is unassigned, fixed at render time
4457 if (gs_plib_flags & PLIB_CAPS_OBJCATMUTATE) {
4458 if (pObj->m_DPRI == -1) {
4459 S52PLIB_Context* pCtx = (S52PLIB_Context*)pObj->S52_Context;
4460 if (pCtx->LUP) cobj->m_DPRI = pCtx->LUP->DPRI - '0';
4461 } else
4462 cobj->m_DPRI = pObj->m_DPRI;
4463 }
4464
4465 cobj->pPolyTessGeo = (PolyTessGeo*)pObj->pPolyTessGeo;
4466 cobj->m_chart_context = (chart_context*)pObj->m_chart_context;
4467
4468 if (pObj->auxParm3 != 1234) {
4469 pObj->auxParm3 = 1234;
4470 pObj->auxParm0 = -99;
4471 }
4472
4473 cobj->auxParm0 = pObj->auxParm0;
4474 cobj->auxParm1 = 0;
4475 cobj->auxParm2 = 0;
4476 cobj->auxParm3 = 0;
4477
4478 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4479
4480 if (pContext->bBBObj_valid)
4481 // this is ugly because plugins still use BoundingBox
4482 cobj->BBObj.Set(pContext->BBObj.GetMinY(), pContext->BBObj.GetMinX(),
4483 pContext->BBObj.GetMaxY(), pContext->BBObj.GetMaxX());
4484
4485 cobj->CSrules = pContext->CSrules;
4486 cobj->bCS_Added = pContext->bCS_Added;
4487
4488 cobj->FText = pContext->FText;
4489 cobj->bFText_Added = pContext->bFText_Added;
4490 cobj->rText = pContext->rText;
4491
4492 cobj->bIsClone = true; // Protect cloned object pointers in S57Obj dtor
4493
4494 if (pctx) {
4495 cobj->m_chart_context = pctx;
4496 chart_context* ppctx = (chart_context*)pObj->m_chart_context;
4497
4498 if (ppctx) {
4499 cobj->m_chart_context->m_pvc_hash = ppctx->m_pvc_hash;
4500 cobj->m_chart_context->m_pve_hash = ppctx->m_pve_hash;
4501 cobj->m_chart_context->ref_lat = ppctx->ref_lat;
4502 cobj->m_chart_context->ref_lon = ppctx->ref_lon;
4503 cobj->m_chart_context->pFloatingATONArray = ppctx->pFloatingATONArray;
4504 cobj->m_chart_context->pRigidATONArray = ppctx->pRigidATONArray;
4505 cobj->m_chart_context->safety_contour = ppctx->safety_contour;
4506 cobj->m_chart_context->vertex_buffer = ppctx->vertex_buffer;
4507 }
4508 cobj->m_chart_context->chart =
4509 0; // note bene, this is always NULL for a PlugIn chart
4510 cobj->m_chart_context->chart_type = S52_CHART_TYPE_PLUGIN;
4511 }
4512}
4513
4514bool PI_PLIBSetContext(PI_S57Obj* pObj) {
4515 S52PLIB_Context* ctx;
4516 if (!pObj->S52_Context) {
4517 ctx = new S52PLIB_Context;
4518 pObj->S52_Context = ctx;
4519 }
4520
4521 ctx = (S52PLIB_Context*)pObj->S52_Context;
4522
4523 S57Obj cobj;
4524 CreateCompatibleS57Object(pObj, &cobj, NULL);
4525
4526 LUPname LUP_Name = PAPER_CHART;
4527
4528 // Force a re-evaluation of CS rules
4529 ctx->CSrules = NULL;
4530 ctx->bCS_Added = false;
4531
4532 // Clear the rendered text cache
4533 if (ctx->bFText_Added) {
4534 ctx->bFText_Added = false;
4535 delete ctx->FText;
4536 ctx->FText = NULL;
4537 }
4538
4539 // Reset object selection box
4540 ctx->bBBObj_valid = true;
4541 ctx->BBObj.SetMin(pObj->lon_min, pObj->lat_min);
4542 ctx->BBObj.SetMax(pObj->lon_max, pObj->lat_max);
4543
4544 // This is where Simplified or Paper-Type point features are selected
4545 switch (cobj.Primitive_type) {
4546 case GEO_POINT:
4547 case GEO_META:
4548 case GEO_PRIM:
4549
4550 if (PAPER_CHART == ps52plib->m_nSymbolStyle)
4551 LUP_Name = PAPER_CHART;
4552 else
4553 LUP_Name = SIMPLIFIED;
4554
4555 break;
4556
4557 case GEO_LINE:
4558 LUP_Name = LINES;
4559 break;
4560
4561 case GEO_AREA:
4562 if (PLAIN_BOUNDARIES == ps52plib->m_nBoundaryStyle)
4563 LUP_Name = PLAIN_BOUNDARIES;
4564 else
4565 LUP_Name = SYMBOLIZED_BOUNDARIES;
4566
4567 break;
4568 }
4569
4570 LUPrec* lup = ps52plib->S52_LUPLookup(LUP_Name, cobj.FeatureName, &cobj);
4571 ctx->LUP = lup;
4572
4573 // Convert LUP to rules set
4574 ps52plib->_LUP2rules(lup, &cobj);
4575
4576 ctx->MPSRulesList = NULL;
4577
4578 return true;
4579}
4580
4581void PI_UpdateContext(PI_S57Obj* pObj) {
4582 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4583 if (pContext) {
4584 pContext->bBBObj_valid = true;
4585 pContext->BBObj.SetMin(pObj->lon_min, pObj->lat_min);
4586 pContext->BBObj.SetMax(pObj->lon_max, pObj->lat_max);
4587 }
4588}
4589
4590void UpdatePIObjectPlibContext(PI_S57Obj* pObj, S57Obj* cobj,
4591 ObjRazRules* rzRules) {
4592 // Update the PLIB context after the render operation
4593 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4594
4595 pContext->CSrules = cobj->CSrules;
4596 pContext->bCS_Added = cobj->bCS_Added;
4597
4598 pContext->FText = cobj->FText;
4599 pContext->bFText_Added = cobj->bFText_Added;
4600 pContext->rText = cobj->rText;
4601
4602 if (cobj->BBObj.GetValid()) {
4603 // ugly as plugins still use BoundingBox
4604 pContext->BBObj =
4605 BoundingBox(cobj->BBObj.GetMinLon(), cobj->BBObj.GetMinLat(),
4606 cobj->BBObj.GetMaxLon(), cobj->BBObj.GetMaxLat());
4607 pContext->bBBObj_valid = true;
4608 }
4609
4610 // Render operation may have promoted the object's display category
4611 // (e.g.WRECKS)
4612 pObj->m_DisplayCat = (PI_DisCat)cobj->m_DisplayCat;
4613
4614 if (gs_plib_flags & PLIB_CAPS_OBJCATMUTATE) pObj->m_DPRI = cobj->m_DPRI;
4615
4616 pContext->ChildRazRules = rzRules->child;
4617 pContext->MPSRulesList = rzRules->mps;
4618
4619 pObj->auxParm0 = cobj->auxParm0;
4620}
4621
4622bool PI_GetObjectRenderBox(PI_S57Obj* pObj, double* lat_min, double* lat_max,
4623 double* lon_min, double* lon_max) {
4624 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4625 if (pContext) {
4626 if (lat_min) *lat_min = pContext->BBObj.GetMinY();
4627 if (lat_max) *lat_max = pContext->BBObj.GetMaxY();
4628 if (lon_min) *lon_min = pContext->BBObj.GetMinX();
4629 if (lon_max) *lon_max = pContext->BBObj.GetMaxX();
4630 return pContext->bBBObj_valid;
4631 } else
4632 return false;
4633}
4634
4635PI_LUPname PI_GetObjectLUPName(PI_S57Obj* pObj) {
4636 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4637 if (pContext) {
4638 LUPrec* lup = pContext->LUP;
4639 if (lup) return (PI_LUPname)(lup->TNAM);
4640 }
4641 return (PI_LUPname)(-1);
4642}
4643
4644PI_DisPrio PI_GetObjectDisplayPriority(PI_S57Obj* pObj) {
4645 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4646 if (pContext) {
4647 LUPrec* lup = pContext->LUP;
4648 if (lup) return (PI_DisPrio)(lup->DPRI);
4649 }
4650
4651 return (PI_DisPrio)(-1);
4652}
4653
4654PI_DisCat PI_GetObjectDisplayCategory(PI_S57Obj* pObj) {
4655 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4656 if (pContext) {
4657 LUPrec* lup = pContext->LUP;
4658 if (lup) return (PI_DisCat)(lup->DISC);
4659 }
4660 return (PI_DisCat)(-1);
4661}
4662double PI_GetPLIBMarinerSafetyContour() {
4663 return S52_getMarinerParam(S52_MAR_SAFETY_CONTOUR);
4664}
4665
4666void PI_PLIBSetLineFeaturePriority(PI_S57Obj* pObj, int prio) {
4667 // Create and populate a compatible s57 Object
4668 S57Obj cobj;
4669 chart_context ctx;
4670 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4671
4672 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4673
4674 // Create and populate a minimally compatible object container
4675 ObjRazRules rzRules;
4676 rzRules.obj = &cobj;
4677 rzRules.LUP = pContext->LUP;
4678 rzRules.sm_transform_parms = 0;
4679 rzRules.child = NULL;
4680 rzRules.next = NULL;
4681 rzRules.mps = pContext->MPSRulesList;
4682
4683 if (pContext->LUP) {
4684 ps52plib->SetLineFeaturePriority(&rzRules, prio);
4685
4686 // Update the PLIB context after the render operation
4687 UpdatePIObjectPlibContext(pObj, &cobj, &rzRules);
4688 }
4689}
4690
4691void PI_PLIBPrepareForNewRender(void) {
4692 if (ps52plib) {
4693 ps52plib->PrepareForRender();
4694 ps52plib->ClearTextList();
4695
4696 if (gs_plib_flags & PLIB_CAPS_LINE_BUFFER)
4697 ps52plib->EnableGLLS(true); // Newer PlugIns can use GLLS
4698 else
4699 ps52plib->EnableGLLS(false); // Older cannot
4700 }
4701}
4702
4703void PI_PLIBSetRenderCaps(unsigned int flags) { gs_plib_flags = flags; }
4704
4705void PI_PLIBFreeContext(void* pContext) {
4706 S52PLIB_Context* pctx = (S52PLIB_Context*)pContext;
4707
4708 if (pctx->ChildRazRules) {
4709 ObjRazRules* ctop = pctx->ChildRazRules;
4710 while (ctop) {
4711 delete ctop->obj;
4712
4713 if (ps52plib) ps52plib->DestroyLUP(ctop->LUP);
4714
4715 ObjRazRules* cnxx = ctop->next;
4716 delete ctop;
4717 ctop = cnxx;
4718 }
4719 }
4720
4721 if (pctx->MPSRulesList) {
4722 if (ps52plib && pctx->MPSRulesList->cs_rules) {
4723 for (unsigned int i = 0; i < pctx->MPSRulesList->cs_rules->GetCount();
4724 i++) {
4725 Rules* top = pctx->MPSRulesList->cs_rules->Item(i);
4726 ps52plib->DestroyRulesChain(top);
4727 }
4728 delete pctx->MPSRulesList->cs_rules;
4729 }
4730 free(pctx->MPSRulesList);
4731 }
4732
4733 delete pctx->FText;
4734
4735 delete pctx;
4736}
4737
4738int PI_PLIBRenderObjectToDC(wxDC* pdc, PI_S57Obj* pObj, PlugIn_ViewPort* vp) {
4739 // Create and populate a compatible s57 Object
4740 S57Obj cobj;
4741 chart_context ctx;
4742 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4743
4744 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4745
4746 // Set up object SM rendering constants
4747 sm_parms transform;
4748 toSM(vp->clat, vp->clon, pObj->chart_ref_lat, pObj->chart_ref_lon,
4749 &transform.easting_vp_center, &transform.northing_vp_center);
4750
4751 // Create and populate a minimally compatible object container
4752 ObjRazRules rzRules;
4753 rzRules.obj = &cobj;
4754 rzRules.LUP = pContext->LUP;
4755 rzRules.sm_transform_parms = &transform;
4756 rzRules.child = pContext->ChildRazRules;
4757 rzRules.next = NULL;
4758 rzRules.mps = pContext->MPSRulesList;
4759
4760 if (pContext->LUP) {
4761 ViewPort cvp = CreateCompatibleViewport(*vp);
4762
4763 // Do the render
4764 // FIXME (plib)
4765 ps52plib->SetVPointCompat(cvp.pix_width, cvp.pix_height, cvp.view_scale_ppm,
4766 cvp.rotation, cvp.clat, cvp.clon, cvp.chart_scale,
4767 cvp.rv_rect, cvp.GetBBox(), cvp.ref_scale,
4768 GetOCPNCanvasWindow()->GetContentScaleFactor());
4769 ps52plib->PrepareForRender();
4770
4771 ps52plib->RenderObjectToDC(pdc, &rzRules);
4772
4773 // Update the PLIB context after the render operation
4774 UpdatePIObjectPlibContext(pObj, &cobj, &rzRules);
4775 }
4776
4777 return 1;
4778}
4779
4780int PI_PLIBRenderAreaToDC(wxDC* pdc, PI_S57Obj* pObj, PlugIn_ViewPort* vp,
4781 wxRect rect, unsigned char* pixbuf) {
4782 // Create a compatible render canvas
4783 render_canvas_parms pb_spec;
4784
4785 pb_spec.depth = BPP;
4786 pb_spec.pb_pitch = ((rect.width * pb_spec.depth / 8));
4787 pb_spec.lclip = rect.x;
4788 pb_spec.rclip = rect.x + rect.width - 1;
4789 pb_spec.pix_buff = pixbuf; // the passed buffer
4790 pb_spec.width = rect.width;
4791 pb_spec.height = rect.height;
4792 pb_spec.x = rect.x;
4793 pb_spec.y = rect.y;
4794#ifdef ocpnUSE_ocpnBitmap
4795 pb_spec.b_revrgb = true;
4796#else
4797 pb_spec.b_revrgb = false;
4798#endif
4799
4800 pb_spec.b_revrgb = false;
4801
4802 // Create and populate a compatible s57 Object
4803 S57Obj cobj;
4804 chart_context ctx;
4805 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4806
4807 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4808
4809 // Set up object SM rendering constants
4810 sm_parms transform;
4811 toSM(vp->clat, vp->clon, pObj->chart_ref_lat, pObj->chart_ref_lon,
4812 &transform.easting_vp_center, &transform.northing_vp_center);
4813
4814 // Create and populate a minimally compatible object container
4815 ObjRazRules rzRules;
4816 rzRules.obj = &cobj;
4817 rzRules.LUP = pContext->LUP;
4818 rzRules.sm_transform_parms = &transform;
4819 rzRules.child = pContext->ChildRazRules;
4820 rzRules.next = NULL;
4821 rzRules.mps = pContext->MPSRulesList;
4822
4823 ViewPort cvp = CreateCompatibleViewport(*vp);
4824
4825 // If the PlugIn does not support it nativiely, build a fully described
4826 // Geomoetry
4827 if (!(gs_plib_flags & PLIB_CAPS_SINGLEGEO_BUFFER)) {
4828 if (!pObj->geoPtMulti) { // do this only once
4829 PolyTessGeo* tess = (PolyTessGeo*)pObj->pPolyTessGeo;
4830
4831 if (!tess) return 1; // bail on empty data
4832
4833 PolyTriGroup* ptg = new PolyTriGroup;
4834 ptg->tri_prim_head =
4835 tess->Get_PolyTriGroup_head()->tri_prim_head; // tph;
4836 ptg->bsingle_alloc = false;
4837 ptg->data_type = DATA_TYPE_DOUBLE;
4838 tess->Set_PolyTriGroup_head(ptg);
4839
4840 double* pd = (double*)malloc(sizeof(double));
4841 pObj->geoPtMulti = pd; // Hack hack
4842 }
4843 }
4844
4845 if (pContext->LUP) {
4846 // Do the render
4847 // FIXME (plib)
4848 ps52plib->SetVPointCompat(cvp.pix_width, cvp.pix_height, cvp.view_scale_ppm,
4849 cvp.rotation, cvp.clat, cvp.clon, cvp.chart_scale,
4850 cvp.rv_rect, cvp.GetBBox(), cvp.ref_scale,
4851 GetOCPNCanvasWindow()->GetContentScaleFactor());
4852 ps52plib->PrepareForRender();
4853
4854 ps52plib->RenderAreaToDC(pdc, &rzRules, &pb_spec);
4855
4856 // Update the PLIB context after the render operation
4857 UpdatePIObjectPlibContext(pObj, &cobj, &rzRules);
4858 }
4859
4860 return 1;
4861}
4862
4863int PI_PLIBRenderAreaToGL(const wxGLContext& glcc, PI_S57Obj* pObj,
4864 PlugIn_ViewPort* vp, wxRect& render_rect) {
4865#ifdef ocpnUSE_GL
4866 // Create and populate a compatible s57 Object
4867 S57Obj cobj;
4868 chart_context ctx;
4869 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4870
4871 // chart_context *pct = (chart_context *)pObj->m_chart_context;
4872
4873 // If the PlugIn does not support it nativiely, build a fully described
4874 // Geomoetry
4875
4876 if (!(gs_plib_flags & PLIB_CAPS_SINGLEGEO_BUFFER)) {
4877 if (!pObj->geoPtMulti) { // only do this once
4878 PolyTessGeo* tess = (PolyTessGeo*)pObj->pPolyTessGeo;
4879
4880 if (!tess) return 1; // bail on empty data
4881
4882 PolyTriGroup* ptg =
4883 new PolyTriGroup; // this will leak a little, but is POD
4884 ptg->tri_prim_head = tess->Get_PolyTriGroup_head()->tri_prim_head;
4885 ptg->bsingle_alloc = false;
4886 ptg->data_type = DATA_TYPE_DOUBLE;
4887 tess->Set_PolyTriGroup_head(ptg);
4888
4889 // Mark this object using geoPtMulti
4890 // The malloc will get free'ed when the object is deleted.
4891 double* pd = (double*)malloc(sizeof(double));
4892 pObj->geoPtMulti = pd; // Hack hack
4893 }
4894 cobj.auxParm0 = -6; // signal that this object render cannot use VBO
4895 cobj.auxParm1 = -1; // signal that this object render cannot have single
4896 // buffer conversion done
4897 } else { // it is a newer PLugIn, so can do single buffer conversion and VBOs
4898 if (pObj->auxParm0 < 1)
4899 cobj.auxParm0 = -7; // signal that this object render can use a
4900 // persistent VBO for area triangle vertices
4901 }
4902
4903 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4904
4905 // Set up object SM rendering constants
4906 sm_parms transform;
4907 toSM(vp->clat, vp->clon, pObj->chart_ref_lat, pObj->chart_ref_lon,
4908 &transform.easting_vp_center, &transform.northing_vp_center);
4909
4910 // Create and populate a minimally compatible object container
4911 ObjRazRules rzRules;
4912 rzRules.obj = &cobj;
4913 rzRules.LUP = pContext->LUP;
4914 rzRules.sm_transform_parms = &transform;
4915 rzRules.child = pContext->ChildRazRules;
4916 rzRules.next = NULL;
4917 rzRules.mps = pContext->MPSRulesList;
4918
4919 if (pContext->LUP) {
4920 ViewPort cvp = CreateCompatibleViewport(*vp);
4921
4922 // Do the render
4923 // FIXME (plib)
4924 ps52plib->SetVPointCompat(cvp.pix_width, cvp.pix_height, cvp.view_scale_ppm,
4925 cvp.rotation, cvp.clat, cvp.clon, cvp.chart_scale,
4926 cvp.rv_rect, cvp.GetBBox(), cvp.ref_scale,
4927 GetOCPNCanvasWindow()->GetContentScaleFactor());
4928 ps52plib->PrepareForRender();
4929
4930 ps52plib->RenderAreaToGL(glcc, &rzRules);
4931
4932 // Update the PLIB context after the render operation
4933 UpdatePIObjectPlibContext(pObj, &cobj, &rzRules);
4934 }
4935
4936#endif
4937 return 1;
4938}
4939
4940int PI_PLIBRenderObjectToGL(const wxGLContext& glcc, PI_S57Obj* pObj,
4941 PlugIn_ViewPort* vp, wxRect& render_rect) {
4942 // Create and populate a compatible s57 Object
4943 S57Obj cobj;
4944 chart_context ctx;
4945 CreateCompatibleS57Object(pObj, &cobj, &ctx);
4946
4947 S52PLIB_Context* pContext = (S52PLIB_Context*)pObj->S52_Context;
4948
4949 // Set up object SM rendering constants
4950 sm_parms transform;
4951 toSM(vp->clat, vp->clon, pObj->chart_ref_lat, pObj->chart_ref_lon,
4952 &transform.easting_vp_center, &transform.northing_vp_center);
4953
4954 // Create and populate a minimally compatible object container
4955 ObjRazRules rzRules;
4956 rzRules.obj = &cobj;
4957 rzRules.LUP = pContext->LUP;
4958 rzRules.sm_transform_parms = &transform;
4959 rzRules.child = pContext->ChildRazRules;
4960 rzRules.next = NULL;
4961 rzRules.mps = pContext->MPSRulesList;
4962
4963 if (pContext->LUP) {
4964 ViewPort cvp = CreateCompatibleViewport(*vp);
4965
4966 // Do the render
4967 // FIXME (plib)
4968 ps52plib->SetVPointCompat(cvp.pix_width, cvp.pix_height, cvp.view_scale_ppm,
4969 cvp.rotation, cvp.clat, cvp.clon, cvp.chart_scale,
4970 cvp.rv_rect, cvp.GetBBox(), cvp.ref_scale,
4971 GetOCPNCanvasWindow()->GetContentScaleFactor());
4972 ps52plib->PrepareForRender();
4973
4974 ps52plib->RenderObjectToGL(glcc, &rzRules);
4975
4976 // Update the PLIB context after the render operation
4977 UpdatePIObjectPlibContext(pObj, &cobj, &rzRules);
4978 }
4979
4980 return 1;
4981}
4982
4983// http File Download Support
4984
4985// OCPN_downloadEvent Implementation
4986
4987OCPN_downloadEvent::OCPN_downloadEvent(wxEventType commandType, int id)
4988 : wxEvent(id, commandType) {
4989 m_stat = OCPN_DL_UNKNOWN;
4990 m_condition = OCPN_DL_EVENT_TYPE_UNKNOWN;
4991 m_b_complete = false;
4992 m_sofarBytes = 0;
4993}
4994
4995OCPN_downloadEvent::~OCPN_downloadEvent() {}
4996
4997wxEvent* OCPN_downloadEvent::Clone() const {
4998 OCPN_downloadEvent* newevent = new OCPN_downloadEvent(*this);
4999 newevent->m_stat = this->m_stat;
5000 newevent->m_condition = this->m_condition;
5001
5002 newevent->m_totalBytes = this->m_totalBytes;
5003 newevent->m_sofarBytes = this->m_sofarBytes;
5004 newevent->m_b_complete = this->m_b_complete;
5005
5006 return newevent;
5007}
5008
5009// const wxEventType wxEVT_DOWNLOAD_EVENT = wxNewEventType();
5010DECL_EXP wxEventType wxEVT_DOWNLOAD_EVENT = wxNewEventType();
5011
5012_OCPN_DLStatus g_download_status;
5013_OCPN_DLCondition g_download_condition;
5014
5015#define DL_EVENT_TIMER 4388
5016
5017class PI_DLEvtHandler : public wxEvtHandler {
5018public:
5021
5022 void onDLEvent(OCPN_downloadEvent& event);
5023 void setBackgroundMode(long ID, wxEvtHandler* handler);
5024 void clearBackgroundMode();
5025 void onTimerEvent(wxTimerEvent& event);
5026
5027 long m_id;
5028 wxTimer m_eventTimer;
5029 wxEvtHandler* m_download_evHandler;
5030
5031 long m_sofarBytes;
5032 long m_totalBytes;
5033};
5034
5035PI_DLEvtHandler::PI_DLEvtHandler() {
5036 g_download_status = OCPN_DL_UNKNOWN;
5037 g_download_condition = OCPN_DL_EVENT_TYPE_UNKNOWN;
5038
5039 m_download_evHandler = NULL;
5040 m_id = -1;
5041 m_sofarBytes = 0;
5042 m_totalBytes = 0;
5043}
5044
5045PI_DLEvtHandler::~PI_DLEvtHandler() {
5046 m_eventTimer.Stop();
5047 Disconnect(
5048 wxEVT_TIMER,
5049 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onTimerEvent);
5050}
5051
5052void PI_DLEvtHandler::onDLEvent(OCPN_downloadEvent& event) {
5053 // qDebug() << "Got Event " << (int)event.getDLEventStatus() <<
5054 // (int)event.getDLEventCondition();
5055
5056 g_download_status = event.getDLEventStatus();
5057 g_download_condition = event.getDLEventCondition();
5058
5059 // This is an END event, happening at the end of BACKGROUND file download
5060 if (m_download_evHandler &&
5061 (OCPN_DL_EVENT_TYPE_END == event.getDLEventCondition())) {
5062 OCPN_downloadEvent ev(wxEVT_DOWNLOAD_EVENT, 0);
5063 ev.setComplete(true);
5064 ev.setTransferred(m_sofarBytes);
5065 ev.setTotal(m_totalBytes);
5066
5067 ev.setDLEventStatus(event.getDLEventStatus());
5068 ev.setDLEventCondition(OCPN_DL_EVENT_TYPE_END);
5069
5070 m_download_evHandler->AddPendingEvent(ev);
5071 m_eventTimer.Stop();
5072#ifdef __ANDROID__
5073 finishAndroidFileDownload();
5074#endif
5075 }
5076
5077 event.Skip();
5078}
5079
5080void PI_DLEvtHandler::setBackgroundMode(long ID, wxEvtHandler* handler) {
5081 m_id = ID;
5082 m_download_evHandler = handler;
5083
5084 m_eventTimer.SetOwner(this, DL_EVENT_TIMER);
5085
5086 Connect(
5087 wxEVT_TIMER,
5088 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onTimerEvent);
5089 m_eventTimer.Start(1000, wxTIMER_CONTINUOUS);
5090}
5091
5092void PI_DLEvtHandler::clearBackgroundMode() {
5093 m_download_evHandler = NULL;
5094 m_eventTimer.Stop();
5095}
5096
5097void PI_DLEvtHandler::onTimerEvent(wxTimerEvent& event) {
5098#ifdef __ANDROID__
5099 // Query the download status, and post to the original requestor
5100 // This method only happens on Background file downloads
5101
5102 wxString sstat;
5103 int stat = queryAndroidFileDownload(m_id, &sstat);
5104
5105 OCPN_downloadEvent ev(wxEVT_DOWNLOAD_EVENT, 0);
5106 long sofarBytes = 0;
5107 long totalBytes = -1;
5108 long state = -1;
5109
5110 if (stat) { // some error
5111 qDebug() << "Error on queryAndroidFileDownload, ending download";
5112 ev.setComplete(true);
5113 ev.setTransferred(sofarBytes);
5114 ev.setTotal(totalBytes);
5115
5116 ev.setDLEventStatus(OCPN_DL_FAILED);
5117 ev.setDLEventCondition(OCPN_DL_EVENT_TYPE_END);
5118 } else {
5119 wxStringTokenizer tk(sstat, _T(";"));
5120 if (tk.HasMoreTokens()) {
5121 wxString token = tk.GetNextToken();
5122 token.ToLong(&state);
5123 token = tk.GetNextToken();
5124 token.ToLong(&sofarBytes);
5125 token = tk.GetNextToken();
5126 token.ToLong(&totalBytes);
5127 }
5128
5129 qDebug() << state << sofarBytes << totalBytes;
5130
5131 m_sofarBytes = sofarBytes;
5132 m_totalBytes = totalBytes;
5133
5134 ev.setTransferred(sofarBytes);
5135 ev.setTotal(totalBytes);
5136
5137 if (state == 16) { // error
5138 qDebug() << "Event OCPN_DL_FAILED/OCPN_DL_EVENT_TYPE_END";
5139 ev.setComplete(true);
5140 ev.setDLEventStatus(OCPN_DL_FAILED);
5141 ev.setDLEventCondition(OCPN_DL_EVENT_TYPE_END);
5142 } else if (state == 8) { // Completed OK
5143 qDebug() << "Event OCPN_DL_NO_ERROR/OCPN_DL_EVENT_TYPE_END";
5144 ev.setComplete(true);
5145 ev.setDLEventStatus(OCPN_DL_NO_ERROR);
5146 ev.setDLEventCondition(OCPN_DL_EVENT_TYPE_END);
5147 } else {
5148 ev.setComplete(false);
5149 ev.setDLEventStatus(OCPN_DL_UNKNOWN);
5150 ev.setDLEventCondition(OCPN_DL_EVENT_TYPE_PROGRESS);
5151 }
5152
5153 // 2;0;148686
5154 }
5155
5156 if (m_download_evHandler) {
5157 // qDebug() << "Sending event on timer...";
5158 m_download_evHandler->AddPendingEvent(ev);
5159 }
5160
5161 // Background download is all done.
5162 if (OCPN_DL_EVENT_TYPE_END == ev.getDLEventCondition()) {
5163 m_eventTimer.Stop();
5164 finishAndroidFileDownload();
5165 }
5166
5167#endif
5168}
5169
5170PI_DLEvtHandler* g_piEventHandler;
5171
5172// Blocking download of single file
5173_OCPN_DLStatus OCPN_downloadFile(const wxString& url,
5174 const wxString& outputFile,
5175 const wxString& title, const wxString& message,
5176 const wxBitmap& bitmap, wxWindow* parent,
5177 long style, int timeout_secs) {
5178#ifdef __ANDROID__
5179
5180 wxString msg = _T("Downloading file synchronously: ");
5181 msg += url;
5182 msg += _T(" to: ");
5183 msg += outputFile;
5184 wxLogMessage(msg);
5185
5186 // Validate the write location
5187 int vres = validateAndroidWriteLocation(outputFile);
5188 if (vres == 0) // Pending permission dialog
5189 return OCPN_DL_ABORTED;
5190
5191 // Create a single event handler to receive status events
5192 if (!g_piEventHandler) g_piEventHandler = new PI_DLEvtHandler;
5193
5194 // Reset global status indicators
5195 g_download_status = OCPN_DL_UNKNOWN;
5196 g_download_condition = OCPN_DL_EVENT_TYPE_UNKNOWN;
5197
5198 // Create a connection for the expected events from Android Activity
5199 g_piEventHandler->Connect(
5200 wxEVT_DOWNLOAD_EVENT,
5201 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onDLEvent);
5202
5203 long dl_ID = -1;
5204
5205 // Make sure the outputfile is a file URI
5206 wxString fURI = outputFile;
5207 if (!fURI.StartsWith(_T("file://"))) {
5208 fURI.Prepend(_T("file://"));
5209 }
5210
5211 int res = startAndroidFileDownload(url, fURI, g_piEventHandler, &dl_ID);
5212 // Started OK?
5213 if (res) {
5214 finishAndroidFileDownload();
5215 g_piEventHandler->Disconnect(
5216 wxEVT_DOWNLOAD_EVENT,
5217 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onDLEvent);
5218 // delete g_piEventHandler;
5219 return OCPN_DL_FAILED;
5220 }
5221
5222 wxDateTime dl_start_time = wxDateTime::Now();
5223
5224 // Spin, waiting for timeout or event from downstream, and checking status
5225 while (1) {
5226 wxTimeSpan dt = wxDateTime::Now() - dl_start_time;
5227 qDebug() << "Spin.." << dt.GetSeconds().GetLo();
5228
5229 if (dt.GetSeconds() > timeout_secs) {
5230 qDebug() << "USER_TIMOUT";
5231 finishAndroidFileDownload();
5232 g_piEventHandler->Disconnect(
5233 wxEVT_DOWNLOAD_EVENT,
5234 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onDLEvent);
5235 // delete g_piEventHandler;
5236 return (OCPN_DL_USER_TIMEOUT);
5237 }
5238
5239 if (g_download_condition != OCPN_DL_EVENT_TYPE_UNKNOWN) {
5240 if (OCPN_DL_EVENT_TYPE_END == g_download_condition) {
5241 _OCPN_DLStatus ss = g_download_status;
5242 finishAndroidFileDownload();
5243 g_piEventHandler->Disconnect(
5244 wxEVT_DOWNLOAD_EVENT,
5245 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::
5246 onDLEvent);
5247 // delete g_piEventHandler;
5248 qDebug() << "RETURN DL_END" << (int)ss;
5249 return ss; // The actual return code
5250 }
5251 }
5252
5253 wxString sstat;
5254 int stat = queryAndroidFileDownload(dl_ID, &sstat);
5255 if (stat) { // some error
5256 qDebug() << "Error on queryAndroidFileDownload";
5257 finishAndroidFileDownload();
5258 g_piEventHandler->Disconnect(
5259 wxEVT_DOWNLOAD_EVENT,
5260 (wxObjectEventFunction)(wxEventFunction)&PI_DLEvtHandler::onDLEvent);
5261 // delete g_piEventHandler;
5262
5263 return OCPN_DL_FAILED; // so abort
5264 }
5265
5266 wxSleep(1);
5267 wxSafeYield();
5268 }
5269
5270#elif defined(OCPN_USE_CURL)
5271 wxFileName tfn = wxFileName::CreateTempFileName(outputFile);
5272 wxFileOutputStream output(tfn.GetFullPath());
5273
5274 wxCurlDownloadDialog ddlg(url, &output, title, message + url, bitmap, parent,
5275 style);
5276 wxCurlDialogReturnFlag ret = ddlg.RunModal();
5277 output.Close();
5278
5280
5281 switch (ret) {
5282 case wxCDRF_SUCCESS: {
5283 if (wxCopyFile(tfn.GetFullPath(), outputFile))
5284 result = OCPN_DL_NO_ERROR;
5285 else
5286 result = OCPN_DL_FAILED;
5287 break;
5288 }
5289 case wxCDRF_FAILED: {
5290 result = OCPN_DL_FAILED;
5291 break;
5292 }
5293 case wxCDRF_USER_ABORTED: {
5294 result = OCPN_DL_ABORTED;
5295 break;
5296 }
5297 default:
5298 wxASSERT(false); // This should never happen because we handle all
5299 // possible cases of ret
5300 }
5301 if (wxFileExists(tfn.GetFullPath())) wxRemoveFile(tfn.GetFullPath());
5302 return result;
5303
5304#else
5305 return OCPN_DL_FAILED;
5306#endif
5307}
5308
5309wxString toUsrDateTimeFormat_Plugin(const wxDateTime date_time,
5311 return ocpn::toUsrDateTimeFormat(date_time, options);
5312}
5313
5314// Non-Blocking download of single file
5315_OCPN_DLStatus OCPN_downloadFileBackground(const wxString& url,
5316 const wxString& outputFile,
5317 wxEvtHandler* handler,
5318 long* handle) {
5319#ifdef __ANDROID__
5320 wxString msg = _T("Downloading file asynchronously: ");
5321 msg += url;
5322 msg += _T(" to: ");
5323 msg += outputFile;
5324 wxLogMessage(msg);
5325
5326 // Create a single event handler to receive status events
5327
5328 if (!g_piEventHandler) g_piEventHandler = new PI_DLEvtHandler;
5329
5330 long dl_ID = -1;
5331
5332 int res = startAndroidFileDownload(url, outputFile, NULL /*g_piEventHandler*/,
5333 &dl_ID);
5334 // Started OK?
5335 if (res) {
5336 finishAndroidFileDownload();
5337 return OCPN_DL_FAILED;
5338 }
5339
5340 // configure the local event handler for background transfer
5341 g_piEventHandler->setBackgroundMode(dl_ID, handler);
5342
5343 if (handle) *handle = dl_ID;
5344
5345 return OCPN_DL_STARTED;
5346
5347#elif defined(OCPN_USE_CURL)
5348 if (g_pi_manager->m_pCurlThread) // We allow just one download at a time. Do
5349 // we want more? Or at least return some
5350 // other status in this case?
5351 return OCPN_DL_FAILED;
5352 g_pi_manager->m_pCurlThread =
5353 new wxCurlDownloadThread(g_pi_manager, CurlThreadId);
5354 bool http = (url.StartsWith(wxS("http:")) || url.StartsWith(wxS("https:")));
5355 bool keep = false;
5356 if (http && g_pi_manager->m_pCurl &&
5357 dynamic_cast<wxCurlHTTP*>(g_pi_manager->m_pCurl.get())) {
5358 keep = true;
5359 }
5360 if (!keep) {
5361 g_pi_manager->m_pCurl = 0;
5362 }
5363
5364 bool failed = false;
5365 if (!g_pi_manager->HandleCurlThreadError(
5366 g_pi_manager->m_pCurlThread->SetURL(url, g_pi_manager->m_pCurl),
5367 g_pi_manager->m_pCurlThread, url))
5368 failed = true;
5369 if (!failed) {
5370 g_pi_manager->m_pCurl = g_pi_manager->m_pCurlThread->GetCurlSharedPtr();
5371 if (!g_pi_manager->HandleCurlThreadError(
5372 g_pi_manager->m_pCurlThread->SetOutputStream(
5373 new wxFileOutputStream(outputFile)),
5374 g_pi_manager->m_pCurlThread))
5375 failed = true;
5376 }
5377 if (!failed) {
5378 g_pi_manager->m_download_evHandler = handler;
5379 g_pi_manager->m_downloadHandle = handle;
5380
5381 wxCurlThreadError err = g_pi_manager->m_pCurlThread->Download();
5382 if (err != wxCTE_NO_ERROR) {
5383 g_pi_manager->HandleCurlThreadError(
5384 err, g_pi_manager->m_pCurlThread); // shows a message to the user
5385 g_pi_manager->m_pCurlThread->Abort();
5386 failed = true;
5387 }
5388 }
5389
5390 if (!failed) return OCPN_DL_STARTED;
5391
5392 if (g_pi_manager->m_pCurlThread) {
5393 if (g_pi_manager->m_pCurlThread->IsAlive())
5394 g_pi_manager->m_pCurlThread->Abort();
5395 if (g_pi_manager->m_pCurlThread->GetOutputStream())
5396 delete (g_pi_manager->m_pCurlThread->GetOutputStream());
5397 wxDELETE(g_pi_manager->m_pCurlThread);
5398 g_pi_manager->m_download_evHandler = NULL;
5399 g_pi_manager->m_downloadHandle = NULL;
5400 return OCPN_DL_STARTED;
5401 }
5402 g_pi_manager->m_pCurl = 0;
5403 return OCPN_DL_FAILED;
5404
5405#else
5406 return OCPN_DL_FAILED;
5407#endif
5408}
5409
5410void OCPN_cancelDownloadFileBackground(long handle) {
5411#ifdef OCPN_USE_CURL
5412
5413#ifdef __ANDROID__
5414 cancelAndroidFileDownload(handle);
5415 finishAndroidFileDownload();
5416 if (g_piEventHandler) g_piEventHandler->clearBackgroundMode();
5417#else
5418 if (g_pi_manager->m_pCurlThread) {
5419 g_pi_manager->m_pCurlThread->Abort();
5420 delete (g_pi_manager->m_pCurlThread->GetOutputStream());
5421 wxDELETE(g_pi_manager->m_pCurlThread);
5422 g_pi_manager->m_download_evHandler = NULL;
5423 g_pi_manager->m_downloadHandle = NULL;
5424 }
5425 g_pi_manager->m_pCurl = 0;
5426#endif
5427#endif
5428}
5429
5430_OCPN_DLStatus OCPN_postDataHttp(const wxString& url,
5431 const wxString& parameters, wxString& result,
5432 int timeout_secs) {
5433#ifdef __ANDROID__
5434 wxString lparms = parameters;
5435 wxString postResult = doAndroidPOST(url, lparms, timeout_secs * 1000);
5436 if (postResult.IsSameAs(_T("NOK"))) return OCPN_DL_FAILED;
5437
5438 result = postResult;
5439 return OCPN_DL_NO_ERROR;
5440
5441#elif defined(OCPN_USE_CURL)
5442 wxCurlHTTP post;
5443 post.SetOpt(CURLOPT_TIMEOUT, timeout_secs);
5444 size_t res = post.Post(parameters.ToAscii(), parameters.Len(), url);
5445
5446 if (res) {
5447 result = wxString(post.GetResponseBody().c_str(), wxConvUTF8);
5448 return OCPN_DL_NO_ERROR;
5449 } else
5450 result = wxEmptyString;
5451
5452 return OCPN_DL_FAILED;
5453
5454#else
5455 return OCPN_DL_FAILED;
5456#endif
5457}
5458
5459bool OCPN_isOnline() {
5460#ifdef __ANDROID__
5461 return androidCheckOnline();
5462#endif
5463
5464#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
5465 if (wxDateTime::GetTimeNow() >
5466 g_pi_manager->m_last_online_chk + ONLINE_CHECK_RETRY) {
5467 wxCurlHTTP get;
5468 get.Head(_T("http://yahoo.com/"));
5469 g_pi_manager->m_last_online = get.GetResponseCode() > 0;
5470
5471 g_pi_manager->m_last_online_chk = wxDateTime::GetTimeNow();
5472 }
5473 return g_pi_manager->m_last_online;
5474#else
5475 return false;
5476#endif
5477}
5478
5479#if !defined(__ANDROID__) && defined(OCPN_USE_CURL)
5480void PlugInManager::OnEndPerformCurlDownload(wxCurlEndPerformEvent& ev) {
5481 OCPN_downloadEvent event(wxEVT_DOWNLOAD_EVENT, 0);
5482 if (ev.IsSuccessful()) {
5483 event.setDLEventStatus(OCPN_DL_NO_ERROR);
5484 } else {
5485 g_pi_manager->m_pCurl = 0;
5486 event.setDLEventStatus(OCPN_DL_FAILED);
5487 }
5488 event.setDLEventCondition(OCPN_DL_EVENT_TYPE_END);
5489 event.setComplete(true);
5490
5491 if (m_download_evHandler) {
5492 m_download_evHandler->AddPendingEvent(event);
5493 m_download_evHandler = NULL;
5494 m_downloadHandle = NULL;
5495 }
5496
5497 if (m_pCurlThread) {
5498 m_pCurlThread->Wait();
5499 if (!m_pCurlThread->IsAborting()) {
5500 delete (m_pCurlThread->GetOutputStream());
5501 wxDELETE(m_pCurlThread);
5502 }
5503 }
5504}
5505
5506void PlugInManager::OnCurlDownload(wxCurlDownloadEvent& ev) {
5507 OCPN_downloadEvent event(wxEVT_DOWNLOAD_EVENT, 0);
5508 event.setDLEventStatus(OCPN_DL_UNKNOWN);
5509 event.setDLEventCondition(OCPN_DL_EVENT_TYPE_PROGRESS);
5510 event.setTotal(ev.GetTotalBytes());
5511 event.setTransferred(ev.GetDownloadedBytes());
5512 event.setComplete(false);
5513
5514 if (m_download_evHandler) {
5515 m_download_evHandler->AddPendingEvent(event);
5516 }
5517}
5518
5519bool PlugInManager::HandleCurlThreadError(wxCurlThreadError err,
5520 wxCurlBaseThread* p,
5521 const wxString& url) {
5522 switch (err) {
5523 case wxCTE_NO_ERROR:
5524 return true; // ignore this
5525
5526 case wxCTE_NO_RESOURCE:
5527 wxLogError(
5528 wxS("Insufficient resources for correct execution of the program."));
5529 break;
5530
5531 case wxCTE_ALREADY_RUNNING:
5532 wxFAIL; // should never happen!
5533 break;
5534
5535 case wxCTE_INVALID_PROTOCOL:
5536 wxLogError(wxS("The URL '%s' uses an unsupported protocol."),
5537 url.c_str());
5538 break;
5539
5540 case wxCTE_NO_VALID_STREAM:
5541 wxFAIL; // should never happen - the user streams should always be valid!
5542 break;
5543
5544 case wxCTE_ABORTED:
5545 return true; // ignore this
5546
5547 case wxCTE_CURL_ERROR: {
5548 wxString ws = wxS("unknown");
5549 if (p->GetCurlSession())
5550 ws =
5551 wxString(p->GetCurlSession()->GetErrorString().c_str(), wxConvUTF8);
5552 wxLogError(wxS("Network error: %s"), ws.c_str());
5553 } break;
5554 }
5555
5556 // stop the thread
5557 if (p->IsAlive()) p->Abort();
5558
5559 // this is an unrecoverable error:
5560 return false;
5561}
5562#endif
Plugin catalog management: Build the runtime catalog, handling downloads as required.
double GetDisplayDIPMult(wxWindow *win)
Get the display scaling factor for DPI-aware rendering.
wxString & GetPrivateDataDir()
Return dir path for opencpn.log, etc., respecting -c cli option.
EventVar plugin_msg
A JSON message should be sent.
Handle messages for blacklisted plugins.
Modal dialog for plugin catalog settings.
ChartCanvas - Main chart display and interaction component.
Definition chcanv.h:148
Manages the chart database and provides access to chart data.
Definition chartdb.h:95
Wrapper class for plugin-based charts.
Definition chartimg.h:392
virtual double GetNearestPreferredScalePPM(double target_scale_ppm)
Find the nearest preferred viewport scale (in pixels/meter) for this chart.
const std::vector< DriverPtr > & GetDrivers() const
Wrapper for configuration variables which lives in a wxBaseConfig object.
const void Notify()
Notify all listeners, no data supplied.
Wrapper for global variable, supports notification events when value changes.
Downloader with progress and final message dialogs.
Main application frame.
Definition ocpn_frame.h:135
static std::string MessageKey(const char *type="ALL")
Return key which should be used to listen to given message type.
Provides platform-specific support utilities for OpenCPN.
wxSize getDisplaySize()
Get the display size in logical pixels.
An iterator class for OCPNRegion.
Definition OCPNRegion.h:156
A wrapper class for wxRegion with additional functionality.
Definition OCPNRegion.h:56
void Init(const KeyProvider &kp, std::function< void(ObservedEvt &ev)> action)
Initiate an object yet not listening.
Definition observable.h:255
void Listen(const std::string &key, wxEvtHandler *listener, wxEventType evt)
Set object to send wxEventType ev to listener on changes in key.
Custom event class for OpenCPN's notification system.
std::shared_ptr< const void > GetSharedPtr() const
Gets the event's payload data.
Class representing an S-57 chart object.
double lon_max
Bounding box maximum longitude.
double lon_min
Bounding box minimum longitude.
double lat_max
Bounding box maximum latitude.
int auxParm0
Auxiliary parameter 0.
void * pPolyTessGeo
Tesselated polygon geometry.
double chart_ref_lon
Chart reference longitude.
double * geoPtMulti
Lat/lon pairs for decomposed points.
void * S52_Context
S52 presentation data.
double chart_ref_lat
Chart reference latitude.
char FeatureName[8]
S-57 feature type code (e.g., "DEPARE")
double lat_min
Bounding box minimum latitude.
int Primitive_type
Geometry type (point, line, area)
Represents a line segment element for efficient vector chart rendering.
Extended chart base class with separated text rendering.
virtual wxBitmap & RenderRegionViewOnDCNoText(const PlugIn_ViewPort &VPoint, const wxRegion &Region)
Standard DC rendering without text.
virtual bool RenderRegionViewOnDCTextOnly(wxMemoryDC &dc, const PlugIn_ViewPort &VPoint, const wxRegion &Region)
Standard DC text-only rendering.
virtual void ClearPLIBTextList()
Clears any cached text elements.
virtual int RenderRegionViewOnGLNoText(const wxGLContext &glc, const PlugIn_ViewPort &VPoint, const wxRegion &Region, bool b_use_stencil)
OpenGL rendering without text.
virtual int RenderRegionViewOnGLTextOnly(const wxGLContext &glc, const PlugIn_ViewPort &VPoint, const wxRegion &Region, bool b_use_stencil)
OpenGL text-only rendering.
virtual int RenderRegionViewOnGL(const wxGLContext &glc, const PlugIn_ViewPort &VPoint, const wxRegion &Region, bool b_use_stencil)
OpenGL rendering with combined text and graphics.
virtual ListOfPI_S57Obj * GetLightsObjRuleListVisibleAtLatLon(float lat, float lon, PlugIn_ViewPort *VPoint)
Gets list of visible light objects at specified position.
OpenGL-optimized chart base class for plugin chart types.
virtual int GetNoCOVRTablenPoints(int iTable)
Alternative to GetNoCOVRTablePoints().
virtual int GetNoCOVREntries()
Gets number of no-coverage areas in chart.
virtual int GetNoCOVRTablePoints(int iTable)
Gets number of points in no-coverage area boundary.
virtual wxString CreateObjDescriptions(ListOfPI_S57Obj *obj_list)
Creates description text for chart objects.
virtual float * GetNoCOVRTableHead(int iTable)
Gets coordinate data for no-coverage area boundary.
virtual int RenderRegionViewOnGL(const wxGLContext &glc, const PlugIn_ViewPort &VPoint, const wxRegion &Region, bool b_use_stencil)
Renders chart content using OpenGL.
virtual ListOfPI_S57Obj * GetObjRuleListAtLatLon(float lat, float lon, float select_radius, PlugIn_ViewPort *VPoint)
Gets chart objects near specified position.
Base class for implementing custom chart types in OpenCPN plugins.
virtual void SetColorScheme(int cs, bool bApplyImmediate)
Sets the color scheme for chart display.
virtual bool GetChartExtent(ExtentPI *pext)
Gets the geographic boundaries of the chart.
virtual void chartpix_to_latlong(double pixx, double pixy, double *plat, double *plon)
Converts chart pixel coordinates to geographic coordinates.
virtual double GetNearestPreferredScalePPM(double target_scale_ppm)
Returns the nearest preferred scale value for this chart.
virtual double GetNormalScaleMin(double canvas_scale_factor, bool b_allow_overzoom)
Returns the minimum recommended scale for this chart.
virtual wxBitmap & RenderRegionView(const PlugIn_ViewPort &VPoint, const wxRegion &Region)
Renders a region of the chart for display.
virtual ChartFamilyEnumPI GetChartFamily()
Returns the chart family classification.
virtual wxString GetID()
Returns a unique identifier for the chart.
virtual wxString GetDatumString()
Returns the horizontal geodetic datum of the chart.
virtual wxString GetName()
Returns the chart's name or title.
virtual double GetRasterScaleFactor()
Returns the scale factor for raster chart rendering.
virtual wxDateTime GetEditionDate(void)
Returns the edition date of the chart.
virtual int GetCOVRTablenPoints(int iTable)
Alternative method to get the number of points in a coverage table entry.
virtual bool GetChartBits(wxRect &source, unsigned char *pPix, int sub_samp)
Gets pixel data for a portion of a raster chart.
virtual int GetCOVREntries()
Returns the number of coverage table entries for this chart.
virtual double GetNormalScaleMax(double canvas_scale_factor, int canvas_width)
Returns the maximum recommended scale for this chart.
virtual wxString GetSoundingsDatum()
Returns the vertical datum used for soundings in the chart.
virtual wxString GetExtraInfo()
Returns additional information about the chart.
virtual int GetSize_X()
Gets the width of the chart in pixels.
virtual wxString GetFullPath() const
Returns the full file path of the chart.
virtual wxString GetSE()
Returns the chart's source edition identifier.
virtual bool IsReadyToRender()
Indicates whether the chart is ready for rendering.
virtual int GetNativeScale()
Returns the native scale of the chart.
virtual double GetChartSkew()
Returns the skew/rotation angle of the chart.
virtual wxString GetDepthUnits()
Returns the depth units used in the chart.
virtual int Init(const wxString &full_path, int init_flags)
Initializes a chart instance from a file.
virtual wxString GetFileSearchMask(void)
Returns file pattern(s) for chart files this plugin can handle.
virtual int GetSize_Y()
Gets the height of the chart in pixels.
virtual void latlong_to_chartpix(double lat, double lon, double &pixx, double &pixy)
Converts geographic coordinates to chart pixel coordinates.
virtual void ComputeSourceRectangle(const PlugIn_ViewPort &vp, wxRect *pSourceRect)
Computes the source rectangle for the chart based on a given viewport.
virtual float * GetCOVRTableHead(int iTable)
Returns a pointer to the coverage table data for a specific entry.
virtual double GetChartErrorFactor()
Returns the error factor for the chart.
virtual OcpnProjTypePI GetChartProjection()
Returns the projection type used by the chart.
virtual wxBitmap * GetThumbnail(int tnx, int tny, int cs)
Generates a thumbnail image of the chart.
virtual ChartTypeEnumPI GetChartType()
Returns the chart type identifier.
virtual void GetValidCanvasRegion(const PlugIn_ViewPort &VPoint, wxRegion *pValidRegion)
Determines the valid display area for this chart.
virtual bool AdjustVP(PlugIn_ViewPort &vp_last, PlugIn_ViewPort &vp_proposed)
Adjusts viewport parameters for chart-specific requirements.
virtual int GetCOVRTablePoints(int iTable)
Returns the number of points in a specific coverage table entry.
virtual ChartDepthUnitTypePI GetDepthUnitId()
Returns the depth unit type identifier.
Data for a loaded plugin, including dl-loaded library.
Basic data for a loaded plugin, trivially copyable.
wxString m_plugin_filename
The short file path.
wxString m_plugin_file
The full file path.
int m_cap_flag
PlugIn Capabilities descriptor.
wxString m_common_name
A common name string for the plugin.
void HandleN0183(std::shared_ptr< const Nmea0183Msg > n0183_msg)
Process incoming NMEA 0183 messages from the message bus.
void HandleSignalK(std::shared_ptr< const SignalkMsg > sK_msg)
Process incoming SignalK messages from the message bus.
Contains view parameters and status information for a chart display viewport.
double view_scale_ppm
Display scale in pixels per meter.
wxRect rv_rect
Rectangle defining the rendered view area.
int pix_width
Viewport width in pixels.
double lon_max
Maximum longitude of the viewport.
double clon
Center longitude of the viewport in decimal degrees.
double lat_max
Maximum latitude of the viewport.
int pix_height
Viewport height in pixels.
double clat
Center latitude of the viewport in decimal degrees.
double skew
Display skew angle in radians.
double rotation
Display rotation angle in radians.
bool bValid
True if this viewport is valid and can be used for rendering.
double lon_min
Minimum longitude of the viewport.
double lat_min
Minimum latitude of the viewport.
int m_projection_type
Chart projection type (PROJECTION_MERCATOR, etc.)
bool b_quilt
True if the viewport is in quilt mode (showing multiple charts)
float chart_scale
Conventional chart displayed scale (e.g., 1:50000)
const std::vector< PluginMetadata > getInstalled()
Return list of all installed and loaded plugins.
static std::string ImportedMetadataPath(std::string name)
Return path to imported metadata for given plugin.
std::vector< std::string > GetInstalldataPlugins()
Return list of installed plugins lower case names, not necessarily loaded.
static std::string fileListPath(std::string name)
Return path to installation manifest for given plugin.
static std::string versionPath(std::string name)
Return path to file containing version for given plugin.
bool uninstall(const std::string plugin)
Uninstall an installed and loaded plugin.
bool ClearInstallData(const std::string plugin_name)
Remove installation data for not loaded plugin.
void SetInstalledMetadata(const PluginMetadata &pm)
Set metadata for an installed plugin.
static bool isCompatible(const PluginMetadata &metadata, const char *os=PKG_TARGET, const char *os_version=PKG_TARGET_VERSION)
Return true if given plugin is loadable on given os/version.
void ReloadPluginPanels()
Complete reload from plugins array.
bool LoadAllPlugIns(bool enabled_plugins, bool keep_orphans=false)
Update catalog with imported metadata and load all plugin library files.
static std::string GetPluginVersion(const PlugInData pd, std::function< const PluginMetadata(const std::string &)> get_metadata)
Return version string for a plugin, possibly with an "Imported" suffix.
void SortPlugins(int(*cmp_func)(PlugInContainer **, PlugInContainer **))
Sort GetPluginArray().
static void UpdatePlugin(PlugInContainer *plugin, const PluginMetadata &md)
Update PlugInContainer status using data from PluginMetadata and manifest.
void SetEnabled(const wxString &common_name, bool enabled)
Update enabled/disabled state for plugin with given name.
static PluginMetadata MetadataByName(const std::string &name)
Find metadata for given plugin.
void ShowPreferencesDialog(const PlugInData &pd, wxWindow *parent)
Display the preferences dialog for a plugin.
PluginPanel(wxPanel *parent, wxWindowID id, const wxPoint &pos, const wxSize &size, const PlugInData plugin)
An entry in the list of plugins presented by Options | Plugins.
EventVar json_msg
Notified with message targeting all plugins.
Definition routeman.h:189
EventVar json_leg_info
Notified with a shared_ptr<ActiveLegDat>, leg info to all plugins.
Definition routeman.h:192
EventVar on_message_sent
Notified when a message available as GetString() is sent to garmin.
Definition routeman.h:195
A parsed SignalK message over ipv4.
Modal dialog, displays available updates (possibly just one) and lets user select and eventually conf...
Definition update_mgr.h:41
ViewPort - Core geographic projection and coordinate transformation engine.
Definition viewport.h:81
double view_scale_ppm
Requested view scale in physical pixels per meter (ppm), before applying projections.
Definition viewport.h:229
double ref_scale
The nominal scale of the "reference chart" for this view.
Definition viewport.h:246
int pix_height
Height of the viewport in physical pixels.
Definition viewport.h:258
void SetBoxes(void)
Computes the bounding box coordinates for the current viewport.
Definition viewport.cpp:823
double rotation
Rotation angle of the viewport in radians.
Definition viewport.h:239
int pix_width
Width of the viewport in physical pixels.
Definition viewport.h:256
double skew
Angular distortion (shear transform) applied to the viewport in radians.
Definition viewport.h:237
double clon
Center longitude of the viewport in degrees.
Definition viewport.h:224
double clat
Center latitude of the viewport in degrees.
Definition viewport.h:222
double chart_scale
Chart scale denominator (e.g., 50000 for a 1:50000 scale).
Definition viewport.h:244
Invokes client browser on plugin info_url when clicked.
WebsiteButton(wxWindow *parent, const char *url)
Invokes client browser on plugin info_url when clicked.
Device context class that can use either wxDC or OpenGL for drawing.
Definition ocpndc.h:64
Floating toolbar dialog for OpenCPN.
Definition toolbar.h:386
virtual void PrepareContextMenu(int canvasIndex)
Prepares plugin context menu items.
virtual bool RenderOverlayMultiCanvas(wxDC &dc, PlugIn_ViewPort *vp, int canvasIndex)
Renders plugin overlay graphics with canvas selection.
virtual bool RenderGLOverlayMultiCanvas(wxGLContext *pcontext, PlugIn_ViewPort *vp, int canvasIndex)
Renders plugin overlay graphics in OpenGL mode with canvas selection.
virtual bool RenderGLOverlayMultiCanvas(wxGLContext *pcontext, PlugIn_ViewPort *vp, int canvasIndex, int priority)
Render plugin overlay over chart canvas in OpenGL mode.
virtual bool RenderOverlayMultiCanvas(wxDC &dc, PlugIn_ViewPort *vp, int canvas_ix, int priority)
Render plugin overlay over chart canvas in non-OpenGL mode.
virtual bool RenderOverlay(wxDC &dc, PlugIn_ViewPort *vp)
Renders plugin overlay graphics in standard (non-OpenGL) mode.
virtual bool RenderGLOverlay(wxGLContext *pcontext, PlugIn_ViewPort *vp)
Renders plugin overlay graphics in OpenGL mode.
Base class for OpenCPN plugins.
virtual void OnCloseToolboxPanel(int page_sel, int ok_apply_cancel)
Handles preference page closure.
virtual void SetCurrentViewPort(PlugIn_ViewPort &vp)
Notifies plugin of viewport changes.
virtual void UpdateAuiStatus(void)
Updates AUI manager status.
virtual void SetColorScheme(PI_ColorScheme cs)
Updates plugin color scheme.
virtual wxArrayString GetDynamicChartClassNameArray(void)
Returns array of dynamically loaded chart class names.
virtual int GetPlugInVersionMajor()
Returns the major version number of the plugin itself.
virtual bool RenderOverlay(wxMemoryDC *pmdc, PlugIn_ViewPort *vp)
Render plugin overlay graphics using standard device context.
virtual void ProcessParentResize(int x, int y)
Handles parent window resize events.
virtual int GetPlugInVersionMinor()
Returns the minor version number of the plugin itself.
The JSON parser.
Definition jsonreader.h:50
int Parse(const wxString &doc, wxJSONValue *val)
Parse the JSON document.
The JSON value class implementation.
Definition jsonval.h:84
The JSON document writer.
Definition jsonwriter.h:50
void Write(const wxJSONValue &value, wxString &str)
Write the JSONvalue object to a JSON text.
NMEA0183 over IP driver.
NMEA0183 serial driver.
DriverPtr & FindDriver(const std::vector< DriverPtr > &drivers, const std::string &iface, const NavAddr::Bus _bus)
Search list of drivers for a driver with given interface string.
Driver registration container, a singleton.
Raw messages layer, supports sending and recieving navmsg messages.
Enhanced logging interface on top of wx/log.h.
bool replace(std::string &str, const std::string &from, const std::string &to)
Perform in place substitution in str, replacing "from" with "to".
std::string lookup_tarball(const char *uri)
Get path to tarball in cache for given filename.
std::string tolower(const std::string &input)
Return copy of s with all characters converted to lower case.
bool store_metadata(const char *path)
Store metadata in metadata cache, return success/fail.
bool store_tarball(const char *path, const char *basename)
Store a tarball in tarball cache, return success/fail.
enum _PI_DisCat PI_DisCat
Display categories for S52 chart features.
#define INSTALLS_PLUGIN_CHART
Plugin provides new chart type for standard (non-GL) view.
enum _PI_DisPrio PI_DisPrio
Display priority levels for S52 chart objects.
#define PLIB_CAPS_OBJCATMUTATE
Support for object category mutation Allows dynamic changes to object display categories.
#define INSTALLS_CONTEXTMENU_ITEMS
Plugin will add items to chart context menu.
#define WANTS_LATE_INIT
Delay full plugin initialization until system is ready.
_OCPN_DLCondition
Event types for HTTP file download operations.
@ OCPN_DL_EVENT_TYPE_UNKNOWN
Unknown event type.
@ OCPN_DL_EVENT_TYPE_PROGRESS
Download progress update.
@ OCPN_DL_EVENT_TYPE_END
Download has completed.
PI_ColorScheme
Enumeration of color schemes.
#define PLIB_CAPS_SINGLEGEO_BUFFER
Support for single geometry buffers Allows combining multiple geometries into a single buffer.
#define PLIB_CAPS_LINE_BUFFER
Support for line vertex buffers Enables batched line rendering using vertex buffers.
_OCPN_DLStatus
Status codes for HTTP file download operations.
@ OCPN_DL_NO_ERROR
Download completed successfully.
@ OCPN_DL_USER_TIMEOUT
Download timed out waiting for user action.
@ OCPN_DL_STARTED
Download has begun but not yet complete.
@ OCPN_DL_FAILED
Download failed (general error)
@ OCPN_DL_UNKNOWN
Unknown or uninitialized status.
@ OCPN_DL_ABORTED
Download was cancelled by user.
@ PI_CHART_FAMILY_VECTOR
Vector chart formats (S-57, CM93, etc.)
#define INSTALLS_TOOLBOX_PAGE
Plugin will add pages to the toolbox/settings dialog.
enum _PI_LUPname PI_LUPname
Name identifiers for S57 lookup table sets.
#define USES_AUI_MANAGER
Plugin uses wxAuiManager for window management.
#define PLIB_CAPS_OBJSEGLIST
Support for object segment lists Enables breaking complex geometries into optimized segments.
#define INSTALLS_PLUGIN_CHART_GL
Plugin provides new chart type for OpenGL view.
#define WANTS_ONPAINT_VIEWPORT
Receive callbacks during chart viewport painting.
#define WANTS_PREFERENCES
Plugin will add page(s) to global preferences dialog.
#define WANTS_OVERLAY_CALLBACK
Receive callbacks to render custom overlay graphics on the chart.
Definition ocpn_plugin.h:86
wxWindow * GetOCPNCanvasWindow()
Gets OpenCPN's main canvas window.
double OCPN_GetDisplayContentScaleFactor()
Gets content scaling factor for current display.
double OCPN_GetWinDIPScaleFactor()
Gets Windows-specific DPI scaling factor.
Miscellaneous utilities, many of which string related.
void SendNMEASentenceToAllPlugIns(const wxString &sentence)
Distribute a NMEA 0183 sentence to all plugins that have registered interest by setting the WANTS_NME...
Tools to send data to plugins.
Configuration options for date and time formatting.
Plugin metadata, reflects the xml format directly.
std::string to_string()
Return printable XML representation.
Versions uses a modified semantic versioning scheme: major.minor.revision.post-tag+build.
static SemanticVersion parse(std::string s)
Parse a version string, sets major == -1 on errors.
Geographic extent structure defining a bounding box.
double NLAT
Northern latitude boundary in decimal degrees.
double WLON
Western longitude boundary in decimal degrees.
double SLAT
Southern latitude boundary in decimal degrees.
double ELON
Eastern longitude boundary in decimal degrees.