OpenCPN Partial API docs
Loading...
Searching...
No Matches
ocpn_app.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: OpenCPN Main wxWidgets Program
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 "config.h"
26
27#ifdef __MINGW32__
28#undef IPV6STRICT // mingw FTBS fix: missing struct ip_mreq
29#include <windows.h>
30#endif
31
32#include <wx/wxprec.h>
33
34#ifndef WX_PRECOMP
35#include <wx/wx.h>
36#endif // precompiled headers
37#ifdef __WXMSW__
38// #include "c:\\Program Files\\visual leak detector\\include\\vld.h"
39#endif
40
41#include <algorithm>
42#include <limits.h>
43#include <memory>
44#include <thread>
45
46#ifdef __WXMSW__
47#include <math.h>
48#include <psapi.h>
49#include <stdlib.h>
50#include <time.h>
51#endif
52
53#ifndef __WXMSW__
54#include <setjmp.h>
55#include <signal.h>
56#endif
57
58#ifdef OCPN_HAVE_X11
59#include <X11/Xatom.h>
60#include <X11/Xlib.h>
61#endif
62
63#if (defined(OCPN_GHC_FILESYSTEM) || \
64 (defined(__clang_major__) && (__clang_major__ < 15)))
65// MacOS 1.13
66#include <ghc/filesystem.hpp>
67namespace fs = ghc::filesystem;
68#else
69#include <filesystem>
70#include <utility>
71namespace fs = std::filesystem;
72#endif
73
74using namespace std::literals::chrono_literals;
75
76#include <wx/apptrait.h>
77#include <wx/arrimpl.cpp>
78#include <wx/artprov.h>
79#include <wx/aui/aui.h>
80#include <wx/clrpicker.h>
81#include <wx/cmdline.h>
82#include <wx/dialog.h>
83#include <wx/dialog.h>
84#include <wx/dir.h>
85#include <wx/display.h>
86#include <wx/dynlib.h>
87#include <wx/image.h>
88#include <wx/intl.h>
89#include <wx/ipc.h>
90#include <wx/jsonreader.h>
91#include <wx/listctrl.h>
92#include <wx/power.h>
93#include <wx/printdlg.h>
94#include <wx/print.h>
95#include <wx/progdlg.h>
96#include <wx/settings.h>
97#include <wx/stdpaths.h>
98#include <wx/tokenzr.h>
99
100#include "model/ais_decoder.h"
101#include "model/ais_state_vars.h"
102#include "model/certificates.h"
103#include "model/cmdline.h"
104#include "model/comm_bridge.h"
105#include "model/comm_n0183_output.h"
106#include "model/comm_vars.h"
107#include "model/config_vars.h"
108#include "model/instance_check.h"
109#include "model/local_api.h"
110#include "model/logger.h"
111#include "model/mdns_query.h"
112#include "model/mdns_service.h"
113#include "model/multiplexer.h"
114#include "model/nav_object_database.h"
115#include "model/navutil_base.h"
116#include "model/own_ship.h"
117#include "model/plugin_handler.h"
118#include "model/route.h"
119#include "model/routeman.h"
120#include "model/select.h"
121#include "model/track.h"
122
123#include "AboutFrameImpl.h"
124#include "about.h"
125#include "ais_info_gui.h"
126#include "AISTargetAlertDialog.h"
127#include "AISTargetListDialog.h"
128#include "AISTargetQueryDialog.h"
129#include "CanvasConfig.h"
130#include "chartdb.h"
131#include "chcanv.h"
132#include "cm93.h"
133#include "concanv.h"
134#include "config.h"
135#include "ConfigMgr.h"
136#include "DetailSlider.h"
137#include "dychart.h"
138#include "FontMgr.h"
139#include "gdal/cpl_csv.h"
140#include "glTexCache.h"
141#include "GoToPositionDialog.h"
142#include "Layer.h"
143#include "MarkInfo.h"
144#include "navutil.h"
145#include "NMEALogWindow.h"
146#include "observable.h"
147#include "ocpn_app.h"
148#include "OCPN_AUIManager.h"
149#include "ocpn_frame.h"
150#include "OCPNPlatform.h"
151#include "options.h"
152#include "rest_server_gui.h"
153#include "route_ctx_factory.h"
154#include "routemanagerdialog.h"
155#include "routeman_gui.h"
156#include "RoutePropDlgImpl.h"
157#include "s52plib.h"
158#include "s57chart.h"
159#include "S57QueryDialog.h"
160#include "safe_mode_gui.h"
161#include "SoundFactory.h"
162#include "styles.h"
163#include "tcmgr.h"
164#include "thumbwin.h"
165#include "TrackPropDlg.h"
166#include "udev_rule_mgr.h"
167
168#ifdef ocpnUSE_GL
169#include "glChartCanvas.h"
170#endif
171
172#ifdef __WXOSX__
173#include "model/macutils.h"
174#endif
175
176#ifdef __WXMSW__
177#include "model/garmin_protocol_mgr.h" // Used for port probing on Windows
178void RedirectIOToConsole();
179#endif
180
181#if defined(__WXMSW__) && defined(__MSVC__LEAK)
182#include "Stackwalker.h"
183#endif
184
185#ifdef LINUX_CRASHRPT
186#include "crashprint.h"
187#endif
188
189#ifdef __ANDROID__
190#include "androidUTIL.h"
191#else
192#include "serial/serial.h"
193#endif
194#include "wiz_ui.h"
195
196const char *const kUsage =
197 R"(Usage:
198 opencpn -h | --help
199 opencpn [-p] [-f] [-G] [-g] [-P] [-l <str>] [-u <num>] [-U] [-s] [GPX file ...]
200 opencpn --remote [-R] | -q] | -e] |-o <str>]
201
202Options for starting opencpn
203
204 -c, --configdir=<dirpath> Use alternative configuration directory.
205 -p, --portable Run in portable mode.
206 -f, --fullscreen Switch to full screen mode on start.
207 -G, --no_opengl Disable OpenGL video acceleration. This setting will
208 be remembered.
209 -g, --rebuild_gl_raster_cache Rebuild OpenGL raster cache on start.
210 -D, --rebuild_chart_db Rescan chart directories and rebuild the chart database
211 -P, --parse_all_enc Convert all S-57 charts to OpenCPN's internal format on start.
212 -l, --loglevel=<str> Amount of logging: error, warning, message, info, debug or trace
213 -u, --unit_test_1=<num> Display a slideshow of <num> charts and then exit.
214 Zero or negative <num> specifies no limit.
215 -U, --unit_test_2
216 -s, --safe_mode Run without plugins, opengl and other "dangerous" stuff
217 -W, --config_wizard Start with initial configuration wizard
218
219Options manipulating already started opencpn
220 -r, --remote Execute commands on already running instance
221 -R, --raise Make running OpenCPN visible if hidden
222 -q, --quit Terminate already running opencpn
223 -e, --get_rest_endpoint Print rest server endpoint and exit.
224 -o, --open=<GPX file> Open file in running opencpn
225
226Arguments:
227 GPX file GPX-formatted file with waypoints or routes.
228)";
229
230// comm event definitions
231wxDEFINE_EVENT(EVT_N2K_129029, wxCommandEvent);
232wxDEFINE_EVENT(EVT_N2K_129026, wxCommandEvent);
233
234wxDEFINE_EVENT(EVT_N0183_RMC, wxCommandEvent);
235wxDEFINE_EVENT(EVT_N0183_HDT, wxCommandEvent);
236wxDEFINE_EVENT(EVT_N0183_HDG, wxCommandEvent);
237wxDEFINE_EVENT(EVT_N0183_HDM, wxCommandEvent);
238wxDEFINE_EVENT(EVT_N0183_VTG, wxCommandEvent);
239wxDEFINE_EVENT(EVT_N0183_GSV, wxCommandEvent);
240wxDEFINE_EVENT(EVT_N0183_GGA, wxCommandEvent);
241wxDEFINE_EVENT(EVT_N0183_GLL, wxCommandEvent);
242wxDEFINE_EVENT(EVT_N0183_AIVDO, wxCommandEvent);
243
244//------------------------------------------------------------------------------
245// Fwd Declarations
246//------------------------------------------------------------------------------
247
248//------------------------------------------------------------------------------
249// Static variable definition
250//------------------------------------------------------------------------------
251
252WX_DEFINE_OBJARRAY(ArrayOfCDI);
253
254OCPNPlatform *g_Platform;
255
256bool g_bFirstRun;
257bool g_bUpgradeInProcess;
258
259bool g_bPauseTest;
260
261// Files specified on the command line, if any.
262
263MyFrame *gFrame;
264
265ConsoleCanvas *console;
266
267MyConfig *pConfig;
268ChartDB *ChartData;
269int g_restore_stackindex;
270int g_restore_dbindex;
271double g_ChartNotRenderScaleFactor;
272
273LayerList *pLayerList;
274
275Select *pSelectTC;
276
277MarkInfoDlg *g_pMarkInfoDialog;
278RoutePropDlgImpl *pRoutePropDialog;
279TrackPropDlg *pTrackPropDialog;
280RouteManagerDialog *pRouteManagerDialog;
281GoToPositionDialog *pGoToPositionDialog;
282
283double vLat, vLon;
284
285int g_nbrightness = 100;
286
287bool bDBUpdateInProgress;
288
289ThumbWin *pthumbwin;
290TCMgr *ptcmgr;
291
292bool g_bshowToolbar = true;
293bool g_bexpert = true;
294bool g_bBasicMenus = false;
295
296bool bDrawCurrentValues;
297
298wxString ChartListFileName;
299wxString gWorldMapLocation, gDefaultWorldMapLocation;
300wxString gWorldShapefileLocation;
301wxString *pInit_Chart_Dir;
302wxString g_csv_locn;
303wxString g_SENCPrefix;
304wxString g_UserPresLibData;
305wxString g_VisibleLayers;
306wxString g_InvisibleLayers;
307wxString g_VisiNameinLayers;
308wxString g_InVisiNameinLayers;
309
310bool g_bcompression_wait;
311bool g_FlushNavobjChanges;
312int g_FlushNavobjChangesTimeout;
313
314wxString g_uploadConnection;
315
316int user_user_id;
317int file_user_id;
318
319int quitflag;
320int g_tick = 0;
321int g_mem_total, g_mem_initial;
322
323bool s_bSetSystemTime;
324
325static unsigned int malloc_max;
326
327wxDateTime g_start_time;
328wxDateTime g_loglast_time;
329static OcpnSound *_bells_sounds[] = {SoundFactory(), SoundFactory()};
330std::vector<OcpnSound *> bells_sound(_bells_sounds, _bells_sounds + 2);
331
332OcpnSound *g_anchorwatch_sound = SoundFactory();
333
334double AnchorPointMinDist;
335bool AnchorAlertOn1, AnchorAlertOn2;
336bool g_bCruising;
337
338ChartDummy *pDummyChart;
339
340ocpnStyle::StyleManager *g_StyleManager;
341
342bool g_bShowOutlines;
343bool g_bShowDepthUnits;
344bool g_bDisplayGrid; // Flag indicating weather the lat/lon grid should be
345 // displayed
346bool g_bShowChartBar;
347bool g_bShowActiveRouteHighway;
348int g_nAWDefault;
349int g_nAWMax;
350bool g_bPlayShipsBells;
351bool g_bFullscreenToolbar;
352bool g_bShowLayers;
353bool g_bTransparentToolbar;
354bool g_bTransparentToolbarInOpenGLOK;
355int g_nAutoHideToolbar;
356bool g_bAutoHideToolbar;
357
358bool g_bPermanentMOBIcon;
359bool g_bTempShowMenuBar;
360
361int g_iNavAidRadarRingsNumberVisible;
362bool g_bNavAidRadarRingsShown;
363float g_fNavAidRadarRingsStep;
364int g_pNavAidRadarRingsStepUnits;
365bool g_bWayPointPreventDragging;
366bool g_bConfirmObjectDelete;
367wxColour g_colourOwnshipRangeRingsColour;
368int g_maxzoomin;
369
370// Set default color scheme
371ColorScheme global_color_scheme = GLOBAL_COLOR_SCHEME_DAY;
372
373wxArrayPtrVoid *UserColourHashTableArray;
374wxColorHashMap *pcurrent_user_color_hash;
375
376bool bVelocityValid;
377
378int gHDx_Watchdog;
379
380bool g_bDebugCM93;
381bool g_bDebugS57;
382
383int g_ChartUpdatePeriod;
384int g_SkewCompUpdatePeriod;
385
386int g_lastClientRectx;
387int g_lastClientRecty;
388int g_lastClientRectw;
389int g_lastClientRecth;
393double g_display_size_mm;
394std::vector<size_t> g_config_display_size_mm;
395bool g_config_display_size_manual;
396
397int g_GUIScaleFactor;
398int g_ChartScaleFactor;
399float g_MarkScaleFactorExp;
400int g_last_ChartScaleFactor;
401int g_ShipScaleFactor;
402float g_ShipScaleFactorExp;
403int g_ENCSoundingScaleFactor;
404int g_ENCTextScaleFactor;
405
406bool g_bShowTide;
407bool g_bShowCurrent;
408
409s52plib *ps52plib;
410s57RegistrarMgr *m_pRegistrarMan;
411
412CM93OffsetDialog *g_pCM93OffsetDialog;
413
414#ifdef __WXOSX__
415#include "model/macutils.h"
416#endif
417
418// begin rms
419#ifdef __WXOSX__
420#ifdef __WXMSW__
421#ifdef USE_GLU_TESS
422#ifdef USE_GLU_DLL
423// end rms
424extern bool s_glu_dll_ready;
425extern HINSTANCE s_hGLU_DLL; // Handle to DLL
426#endif
427#endif
428#endif
429#endif
430
431double g_ownship_predictor_minutes;
432double g_ownship_HDTpredictor_miles;
433int g_cog_predictor_style;
434wxString g_cog_predictor_color;
435int g_cog_predictor_endmarker;
436int g_ownship_HDTpredictor_style;
437wxString g_ownship_HDTpredictor_color;
438int g_ownship_HDTpredictor_endmarker;
439int g_ownship_HDTpredictor_width;
440
441bool g_own_ship_sog_cog_calc;
442int g_own_ship_sog_cog_calc_damp_sec;
443
444AisInfoGui *g_pAISGUI;
445
446AISTargetQueryDialog *g_pais_query_dialog_active;
447int g_iSoundDeviceIndex;
448
449int g_S57_dialog_sx, g_S57_dialog_sy;
450
451int g_nframewin_x;
452int g_nframewin_y;
453int g_nframewin_posx;
454int g_nframewin_posy;
455bool g_bframemax;
456
457bool g_bAutoAnchorMark;
458
459int gpIDXn;
460long gStart_LMT_Offset;
461
462wxArrayString *pMessageOnceArray;
463
464bool g_bUseGLL = true;
465
466int g_nCacheLimit;
467int g_memCacheLimit;
468bool g_bGDAL_Debug;
469
470bool g_bCourseUp;
471int g_COGAvgSec = 15; // COG average period (sec.) for Course Up Mode
472double g_COGAvg;
473bool g_bLookAhead;
474bool g_bskew_comp;
475bool g_bopengl;
476bool g_bSoftwareGL;
495bool g_bsmoothpanzoom;
496// toggle for smooth position jumping
497bool g_bSmoothRecenter = true;
498bool g_fog_overzoom;
499double g_overzoom_emphasis_base;
500bool g_oz_vector_scale;
501double g_plus_minus_zoom_factor;
502bool g_bChartBarEx;
503
504bool g_b_legacy_input_filter_behaviour; // Support original input filter
505 // process or new process
506
507PlugInManager *g_pi_manager;
508
509bool g_bDebugGPSD;
510
511bool g_bFullScreenQuilt = true;
512bool g_bQuiltEnable;
513bool g_bQuiltStart;
514
515ChartGroupArray *g_pGroupArray;
516
517S57QueryDialog *g_pObjectQueryDialog;
518
519std::vector<std::string> TideCurrentDataSet;
520wxString g_TCData_Dir;
521
522options *g_options;
523bool g_bDeferredInitDone;
524int options_lastPage = 0;
525int options_subpage = 0;
526
527wxPoint options_lastWindowPos(0, 0);
528wxSize options_lastWindowSize(0, 0);
529
530bool g_bSleep;
531bool g_bsimplifiedScalebar;
532
533int osMajor, osMinor;
534
535bool GetMemoryStatus(int *mem_total, int *mem_used);
536bool g_bHasHwClock;
537
538int g_nAIS_activity_timer;
539
540bool g_bEnableZoomToCursor;
541
542bool g_bTrackActive;
543bool g_bDeferredStartTrack;
544bool g_bHighliteTracks;
545wxColour g_colourTrackLineColour;
546wxString g_default_wp_icon;
547
548ActiveTrack *g_pActiveTrack;
549double g_TrackIntervalSeconds;
550
551int g_cm93_zoom_factor;
552PopUpDSlide *pPopupDetailSlider;
553bool g_bShowDetailSlider;
554int g_detailslider_dialog_x, g_detailslider_dialog_y;
555
556bool g_bUseGreenShip;
557
558wxString g_AW1GUID;
559wxString g_AW2GUID;
560
561bool g_b_overzoom_x = true; // Allow high overzoom
562
563int g_OwnShipIconType;
564double g_n_ownship_length_meters;
565double g_n_ownship_beam_meters;
566double g_n_gps_antenna_offset_y;
567double g_n_gps_antenna_offset_x;
568int g_n_ownship_min_mm;
569
570int g_NeedDBUpdate; // 0 - No update needed, 1 - Update needed because there is
571 // no chart database, inform user, 2 - Start update right
572 // away
573bool g_bPreserveScaleOnX;
574
575AboutFrameImpl *g_pAboutDlg;
576about *g_pAboutDlgLegacy;
577
578#if wxUSE_XLOCALE || !wxCHECK_VERSION(3, 0, 0)
579wxLocale *plocale_def_lang = 0;
580#endif
581
590wxString g_locale;
591wxString g_localeOverride;
592bool g_b_assume_azerty;
593
594int g_click_stop;
595
596int g_MemFootMB;
597
598bool g_bShowStatusBar;
599
600bool g_bquiting;
601int g_BSBImgDebug;
602
603AISTargetListDialog *g_pAISTargetList;
604int g_AisTargetList_count;
605bool g_bAisTargetList_autosort;
606
607bool g_bFullscreen;
608
609OCPN_AUIManager *g_pauimgr;
610wxAuiDefaultDockArt *g_pauidockart;
611
612wxString g_toolbarConfig = _T("XXXXXXXXXXXXXXXXXXXXXXXXXXXXX");
613
614ocpnFloatingToolbarDialog *g_MainToolbar;
615int g_maintoolbar_x;
616int g_maintoolbar_y;
617long g_maintoolbar_orient;
618float g_toolbar_scalefactor;
619
620float g_compass_scalefactor;
621bool g_bShowMenuBar;
622bool g_bShowCompassWin;
623
624bool g_benable_rotate;
625
626int g_GPU_MemSize;
627
628wxString g_uiStyle;
629
630// Values returned from WMM_PI for variation computation request.
631// Initialize to invalid so we don't use it if WMM hasn't updated yet
632double gQueryVar = 361.0;
633
634char bells_sound_file_name[2][12] = {"1bells.wav", "2bells.wav"};
635
636int portaudio_initialized;
637
638char nmea_tick_chars[] = {'|', '/', '-', '\\', '|', '/', '-', '\\'};
639
640int g_sticky_chart;
641int g_sticky_projection;
642
643int n_NavMessageShown;
644wxString g_config_version_string;
645
646wxString g_CmdSoundString;
647
660bool g_bresponsive;
662bool g_bRollover;
663
664bool b_inCompressAllCharts;
665bool g_bGLexpert;
666bool g_bUIexpert;
667
668int g_chart_zoom_modifier_raster;
669int g_chart_zoom_modifier_vector;
670
671bool g_bAdvanceRouteWaypointOnArrivalOnly;
672
673bool g_bSpaceDropMark;
674
675wxArrayString g_locale_catalog_array;
676bool b_reloadForPlugins;
677bool g_btrackContinuous;
678
679unsigned int g_canvasConfig;
680bool g_useMUI;
681bool g_bmasterToolbarFull = true;
682
683int g_AndroidVersionCode;
684
685int g_memUsed;
686SENCThreadManager *g_SencThreadManager;
687
688WX_DEFINE_ARRAY_PTR(ChartCanvas *, arrayofCanvasPtr);
689
690arrayofCanvasPtr g_canvasArray;
691wxString g_lastAppliedTemplateGUID;
692
693ChartCanvas *g_focusCanvas;
694ChartCanvas *g_overlayCanvas;
695
696bool b_inCloseWindow;
697bool g_disable_main_toolbar;
698bool g_btenhertz;
699
700#ifdef LINUX_CRASHRPT
701wxCrashPrint g_crashprint;
702#endif
703
704#ifndef __WXMSW__
705sigjmp_buf env; // the context saved by sigsetjmp();
706#endif
707
708// {2C9C45C2-8E7D-4C08-A12D-816BBAE722C0}
709#ifdef __WXMSW__
710DEFINE_GUID(GARMIN_DETECT_GUID, 0x2c9c45c2L, 0x8e7d, 0x4c08, 0xa1, 0x2d, 0x81,
711 0x6b, 0xba, 0xe7, 0x22, 0xc0);
712#endif
713
714#ifdef __VISUALC__
715#include <wx/msw/msvcrt.h>
716#endif
717
718#if !defined(NAN)
719static const long long lNaN = 0xfff8000000000000;
720#define NAN (*(double *)&lNaN)
721#endif
722
723// Some static helpers
724void appendOSDirSlash(wxString *pString);
725
726void InitializeUserColors(void);
727void DeInitializeUserColors(void);
728void SetSystemColors(ColorScheme cs);
729
730static bool LoadAllPlugIns(bool load_enabled) {
731 g_Platform->ShowBusySpinner();
732 bool b = PluginLoader::getInstance()->LoadAllPlugIns(load_enabled);
733 g_Platform->HideBusySpinner();
734 return b;
735}
736
737//------------------------------------------------------------------------------
738// PNG Icon resources
739//------------------------------------------------------------------------------
740
741#if defined(__WXGTK__) || defined(__WXQT__)
742#include "bitmaps/opencpn.xpm"
743#endif
744
745wxString newPrivateFileName(wxString, const char *name,
746 [[maybe_unused]] const char *windowsName) {
747 wxString fname = wxString::FromUTF8(name);
748 wxString filePathAndName;
749
750 filePathAndName = g_Platform->GetPrivateDataDir();
751 if (filePathAndName.Last() != wxFileName::GetPathSeparator())
752 filePathAndName.Append(wxFileName::GetPathSeparator());
753
754#ifdef __WXMSW__
755 wxString fwname = wxString::FromUTF8(windowsName);
756 filePathAndName.Append(fwname);
757#else
758 filePathAndName.Append(fname);
759#endif
760
761 return filePathAndName;
762}
763
764// `Main program` equivalent, creating windows and returning main app frame
765//------------------------------------------------------------------------------
766// MyApp
767//------------------------------------------------------------------------------
768IMPLEMENT_APP(MyApp)
769
770BEGIN_EVENT_TABLE(MyApp, wxApp)
771EVT_ACTIVATE_APP(MyApp::OnActivateApp)
772END_EVENT_TABLE()
773
774static void ActivateRoute(const std::string &guid) {
775 Route *route = g_pRouteMan->FindRouteByGUID(guid);
776 if (!route) {
777 wxLogMessage("Cannot activate guid: no such route");
778 return;
779 }
780 if (g_pRouteMan->GetpActiveRoute()) g_pRouteMan->DeactivateRoute();
781 // If this is an auto-created MOB route, always select the second point
782 // (the MOB)
783 // as the destination.
784 RoutePoint *point;
785 if (wxNOT_FOUND == route->m_RouteNameString.Find("MOB")) {
786 point = g_pRouteMan->FindBestActivatePoint(route, gLat, gLon, gCog, gSog);
787 } else {
788 point = route->GetPoint(2);
789 }
790 g_pRouteMan->ActivateRoute(route, point);
791 if (g_pRouteMan) g_pRouteMan->on_routes_update.Notify();
792 route->m_bRtIsSelected = false;
793}
794
795static void ReverseRoute(const std::string &guid) {
796 Route *route = g_pRouteMan->FindRouteByGUID(guid);
797 if (!route) {
798 wxLogMessage("Cannot activate guid: no such route");
799 return;
800 }
801 route->Reverse();
802 if (g_pRouteMan) g_pRouteMan->on_routes_update.Notify();
803}
804
805void MyApp::InitRestListeners() {
806 auto activate_route = [&](wxCommandEvent ev) {
807 auto guid = ev.GetString().ToStdString();
808 ActivateRoute(guid);
809 };
810 rest_activate_listener.Init(m_rest_server.activate_route, activate_route);
811 auto reverse_route = [&](wxCommandEvent ev) {
812 auto guid = ev.GetString().ToStdString();
813 ReverseRoute(guid);
814 };
815 rest_reverse_listener.Init(m_rest_server.reverse_route, reverse_route);
816}
817
818bool MyApp::OpenFile(const std::string &path) {
819 NavObjectCollection1 nav_objects;
820 auto result = nav_objects.load_file(path.c_str());
821 if (!result) {
822 std::string s(_("Cannot load route or waypoint file: "));
823 s += std::string("\"") + path + "\"";
824 wxMessageBox(s, "OpenCPN", wxICON_WARNING | wxOK);
825 return false;
826 }
827
828 int wpt_dups;
829 // Import with full vizibility of names and objects
830 nav_objects.LoadAllGPXObjects(!nav_objects.IsOpenCPN(), wpt_dups, true);
831
832 if (pRouteManagerDialog && pRouteManagerDialog->IsShown())
833 pRouteManagerDialog->UpdateLists();
834 LLBBox box = nav_objects.GetBBox();
835 if (box.GetValid()) {
836 gFrame->CenterView(gFrame->GetPrimaryCanvas(), box);
837 }
838 return true;
839}
840
841#ifndef __ANDROID__
842void MyApp::OnInitCmdLine(wxCmdLineParser &parser) {
843 // Add OpenCPN specific command line options. Help message
844 // is hardcoded in kUsage;
845 parser.AddSwitch("h", "help", "", wxCMD_LINE_OPTION_HELP);
846 parser.AddSwitch("p", "portable");
847 parser.AddOption("c", "configdir", "", wxCMD_LINE_VAL_STRING,
848 wxCMD_LINE_PARAM_OPTIONAL);
849 parser.AddSwitch("f", "fullscreen");
850 parser.AddSwitch("G", "no_opengl");
851 parser.AddSwitch("W", "config_wizard");
852 parser.AddSwitch("g", "rebuild_gl_raster_cache");
853 parser.AddSwitch("D", "rebuild_chart_db");
854 parser.AddSwitch("P", "parse_all_enc");
855 parser.AddOption("l", "loglevel");
856 parser.AddOption("u", "unit_test_1", "", wxCMD_LINE_VAL_NUMBER);
857 parser.AddSwitch("U", "unit_test_2");
858 parser.AddParam("import GPX files", wxCMD_LINE_VAL_STRING,
859 wxCMD_LINE_PARAM_OPTIONAL | wxCMD_LINE_PARAM_MULTIPLE);
860 parser.AddSwitch("s", "safe_mode");
861 parser.AddSwitch("r", "remote");
862 parser.AddSwitch("R", "raise");
863 parser.AddSwitch("q", "quit");
864 parser.AddSwitch("e", "get_rest_endpoint");
865 parser.AddOption("o", "open", "", wxCMD_LINE_VAL_STRING,
866 wxCMD_LINE_PARAM_OPTIONAL | wxCMD_LINE_PARAM_MULTIPLE);
867}
868#endif // __ANDROID__
869
871#ifdef __ANDROID__
872static void ParseLoglevel(wxCmdLineParser &parser) {
873 wxLog::SetLogLevel(wxLOG_Message);
874}
875#else
876static void ParseLoglevel(wxCmdLineParser &parser) {
877 const char *strLevel = std::getenv("OPENCPN_LOGLEVEL");
878 strLevel = strLevel ? strLevel : "info";
879 wxString wxLevel;
880 if (parser.Found("l", &wxLevel)) {
881 strLevel = wxLevel.c_str();
882 }
883 wxLogLevel level = OcpnLog::str2level(strLevel);
884 if (level == OcpnLog::LOG_BADLEVEL) {
885 fprintf(stderr, "Bad loglevel %s, using \"info\"", strLevel);
886 level = wxLOG_Info;
887 }
888 wxLog::SetLogLevel(level);
889}
890#endif // __ANDROID__
891
892#ifndef __ANDROID__
893bool MyApp::OnCmdLineHelp(wxCmdLineParser &parser) {
894 std::cout << kUsage;
895 return false;
896}
897#endif
898
899#ifndef __ANDROID__
900bool MyApp::OnCmdLineParsed(wxCmdLineParser &parser) {
901 long number;
902 wxString repo;
903 wxString plugin;
904
905 g_unit_test_2 = parser.Found("unit_test_2");
906 g_bportable = parser.Found("p");
907 g_start_fullscreen = parser.Found("fullscreen");
908 g_bdisable_opengl = parser.Found("no_opengl");
909 g_rebuild_gl_cache = parser.Found("rebuild_gl_raster_cache");
910 g_NeedDBUpdate = parser.Found("rebuild_chart_db") ? 2 : 0;
911 g_parse_all_enc = parser.Found("parse_all_enc");
912 g_config_wizard = parser.Found("config_wizard");
913 if (parser.Found("unit_test_1", &number)) {
914 g_unit_test_1 = static_cast<int>(number);
915 if (g_unit_test_1 == 0) g_unit_test_1 = -1;
916 }
917 safe_mode::set_mode(parser.Found("safe_mode"));
918 ParseLoglevel(parser);
919 wxString wxstr;
920 if (parser.Found("configdir", &wxstr)) {
921 g_configdir = wxstr.ToStdString();
922 fs::path path(g_configdir);
923 if (!fs::exists(path) || !fs::is_directory(path)) {
924 std::cerr << g_configdir << " is not an existing directory.\n";
925 return false;
926 }
927 }
928
929 bool has_start_options = false;
930 static const std::vector<std::string> kStartOptions = {
931 "unit_test_2",
932 "p",
933 "fullscreen",
934 "no_opengl",
935 "rebuild_gl_raster_cache",
936 "rebuild_chart_db",
937 "parse_all_enc",
938 "unit_test_1",
939 "safe_mode",
940 "loglevel"};
941 for (const auto &opt : kStartOptions) {
942 if (parser.Found(opt)) has_start_options = true;
943 }
944 if (has_start_options && parser.Found("remote")) {
945 std::cerr << "this option is not compatible with --remote\n";
946 return false;
947 }
948
949 bool has_remote_options = false;
950 static const std::vector<std::string> kRemoteOptions = {
951 "raise", "quit", "open", "get_rest_endpoint"};
952 for (const auto &opt : kRemoteOptions) {
953 if (parser.Found(opt)) has_remote_options = true;
954 }
955 if (has_remote_options && !parser.Found("remote")) {
956 std::cerr << "This option requires --remote\n";
957 return false;
958 }
959
960 for (size_t paramNr = 0; paramNr < parser.GetParamCount(); ++paramNr)
961 g_params.push_back(parser.GetParam(paramNr).ToStdString());
962
963 wxString optarg;
964 if (!parser.Found("remote"))
965 m_parsed_cmdline = ParsedCmdline();
966 else if (parser.Found("raise"))
967 m_parsed_cmdline = ParsedCmdline(CmdlineAction::Raise);
968 else if (parser.Found("quit"))
969 m_parsed_cmdline = ParsedCmdline(CmdlineAction::Quit);
970 else if (parser.Found("get_rest_endpoint"))
971 m_parsed_cmdline = ParsedCmdline(CmdlineAction::GetRestEndpoint);
972 else if (parser.Found("open", &optarg))
973 m_parsed_cmdline = ParsedCmdline(CmdlineAction::Open, optarg.ToStdString());
974 else if (parser.GetParamCount() == 1)
975 m_parsed_cmdline =
976 ParsedCmdline(CmdlineAction::Open, parser.GetParam(0).ToStdString());
977 else if (!has_start_options && !has_remote_options) {
978 // Neither arguments nor options
979 m_parsed_cmdline = ParsedCmdline(CmdlineAction::Raise);
980 }
981 return true;
982}
983#endif // __ANDROID__
984
985#ifdef __WXMSW__
986// Handle any exception not handled by CrashRpt
987// Most probable: Malloc/new failure
988
989bool MyApp::OnExceptionInMainLoop() {
990 wxLogWarning(_T("Caught MainLoopException, continuing..."));
991 return true;
992}
993#endif
994
995void MyApp::OnActivateApp(wxActivateEvent &event) { return; }
996
997static wxStopWatch init_sw;
998
999int MyApp::OnRun() {
1000 if (m_exitcode != -2) return m_exitcode;
1001 return wxAppConsole::OnRun();
1002}
1003
1004MyApp::MyApp()
1005 : m_checker(InstanceCheck::GetInstance()),
1006 m_rest_server(PINCreateDialog::GetDlgCtx(), RouteCtxFactory(),
1007 g_bportable),
1008 m_usb_watcher(UsbWatchDaemon::GetInstance()),
1009 m_exitcode(-2) {
1010#ifdef __linux__
1011 // Handle e. g., wayland default display -- see #1166.
1012 if (wxGetEnv("WAYLAND_DISPLAY", NULL)) {
1013 setenv("GDK_BACKEND", "x11", 1);
1014 }
1015 setenv(
1016 "mesa_glthread", "false",
1017 1); // Explicitly disable glthread. This may have some impact on OpenGL
1018 // performance, but we know it is problematic for us. See #2889
1019#endif // __linux__
1020}
1021
1022bool MyApp::OnInit() {
1023 if (!wxApp::OnInit()) return false;
1024#ifdef __ANDROID__
1025 androidEnableBackButton(false);
1026 androidEnableOptionItems(false);
1027#endif
1028
1029 GpxDocument::SeedRandom();
1030
1031#if defined(__WXGTK__) && defined(ocpnUSE_GLES) && defined(__ARM_ARCH)
1032 // There is a race condition between cairo which is used for text rendering
1033 // by gtk and EGL which without the below code causes a bus error and the
1034 // program aborts before startup
1035 // this hack forces cairo to load right now by rendering some text
1036
1037 wxBitmap bmp(10, 10, -1);
1038 wxMemoryDC dc;
1039 dc.SelectObject(bmp);
1040 dc.DrawText(_T("X"), 0, 0);
1041#endif
1042
1043 // Instantiate the global OCPNPlatform class
1044 g_Platform = new OCPNPlatform;
1045 g_BasePlatform = g_Platform;
1046#ifndef __ANDROID__
1047 // We allow only one instance unless the portable option is used
1048 if (!g_bportable && wxDirExists(g_Platform->GetPrivateDataDir())) {
1049 m_checker.WaitUntilValid();
1050 if (m_checker.IsMainInstance()) {
1051 // Server is created on first call to GetInstance()
1052 if (m_parsed_cmdline.action == CmdlineAction::Skip) {
1053 // Server starts running when referenced.
1054 [[maybe_unused]] auto &server = LocalServerApi::GetInstance();
1055 } else {
1056 std::cerr << "No remote opencpn found. Giving up.\n";
1057 m_exitcode = 1;
1058 return true;
1059 }
1060 } else {
1061 std::unique_ptr<LocalClientApi> client;
1062 try {
1063 client = LocalClientApi::GetClient();
1064 } catch (LocalApiException &ie) {
1065 WARNING_LOG << "Ipc client exception: " << ie.str();
1066 // If we get here it means that the instance_chk found another
1067 // running instance. But that instance is for some reason not
1068 // reachable. The safe thing to do is delete the lockfile and exit.
1069 // Next start will proceed normally. This may leave a zombie OpenCPN,
1070 // but at least O starts.
1071 m_checker.CleanUp();
1072 wxMessageBox(_("Sorry, an existing instance of OpenCPN may be too busy "
1073 "to respond.\nPlease retry."),
1074 "OpenCPN", wxICON_INFORMATION | wxOK);
1075 m_exitcode = 2;
1076 return true; // main program quiet exit.
1077 }
1078 if (client) {
1079 auto result = client->HandleCmdline(m_parsed_cmdline.action,
1080 m_parsed_cmdline.arg);
1081 if (result.first) {
1082 m_exitcode = 0;
1083 } else {
1084 wxLogDebug("Error running remote command: %s", result.second.c_str());
1085 m_exitcode = 1;
1086 }
1087 return true;
1088 }
1089 }
1090 }
1091#endif // __ANDROID__
1092
1093 if (getenv("OPENCPN_FATAL_ERROR") != 0) {
1094 wxLogFatalError(getenv("OPENCPN_FATAL_ERROR"));
1095 }
1096
1097#ifndef __ANDROID__
1098 // Check if last run failed, set up safe_mode.
1099 if (!safe_mode::get_mode()) {
1101 }
1102#endif
1103
1104 // Perform first stage initialization
1105 OCPNPlatform::Initialize_1();
1106
1107 // Set the name of the app as displayed to the user.
1108 // This is necessary at least on OS X, for the capitalisation to be correct in
1109 // the system menus.
1110 MyApp::SetAppDisplayName("OpenCPN");
1111
1112 // Seed the random number generator
1113 wxDateTime x = wxDateTime::UNow();
1114 long seed = x.GetMillisecond();
1115 seed *= x.GetTicks();
1116 srand(seed);
1117
1118 // Fulup: force floating point to use dot as separation.
1119 // This needs to be set early to catch numerics in config file.
1120 setlocale(LC_NUMERIC, "C");
1121
1122 g_start_time = wxDateTime::Now();
1123
1124 g_loglast_time = g_start_time;
1125 g_loglast_time.MakeGMT();
1126 g_loglast_time.Subtract(
1127 wxTimeSpan(0, 29, 0, 0)); // give 1 minute for GPS to get a fix
1128
1129 AnchorPointMinDist = 5.0;
1130
1131 // Init the private memory manager
1132 malloc_max = 0;
1133
1134 // Record initial memory status
1135 GetMemoryStatus(&g_mem_total, &g_mem_initial);
1136
1137 // Set up default FONT encoding, which should have been done by wxWidgets some
1138 // time before this......
1139 wxFont temp_font(10, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL,
1140 wxFONTWEIGHT_NORMAL, FALSE, wxString(_T("")),
1141 wxFONTENCODING_SYSTEM);
1142 temp_font.SetDefaultEncoding(wxFONTENCODING_SYSTEM);
1143
1144 // Establish Log File location
1145 if (!g_Platform->InitializeLogFile()) {
1146 return false;
1147 };
1148
1149#ifdef __WXMSW__
1150
1151 // Un-comment the following to establish a separate console window as a
1152 // target for printf() in Windows RedirectIOToConsole();
1153
1154#endif
1155
1156 // Send init message
1157 wxLogMessage(_T("\n\n________\n"));
1158
1159 wxDateTime now = wxDateTime::Now();
1160 LOG_INFO("------- OpenCPN version %s restarted at %s -------\n", VERSION_FULL,
1161 now.FormatISODate().mb_str().data());
1162 wxLogLevel level = wxLog::GetLogLevel();
1163 LOG_INFO("Using loglevel %s", OcpnLog::level2str(level).c_str());
1164
1165 wxString wxver(wxVERSION_STRING);
1166 wxver.Prepend(_T("wxWidgets version: "));
1167
1168 wxPlatformInfo platforminfo = wxPlatformInfo::Get();
1169
1170 wxString os_name;
1171#ifndef __ANDROID__
1172 os_name = platforminfo.GetOperatingSystemIdName();
1173#else
1174 os_name = platforminfo.GetOperatingSystemFamilyName();
1175#endif
1176
1177 wxString platform = os_name + _T(" ") + platforminfo.GetArchName() + _T(" ") +
1178 platforminfo.GetPortIdName();
1179
1180 wxLogMessage(wxver + _T(" ") + platform);
1181
1182 ::wxGetOsVersion(&osMajor, &osMinor);
1183 wxString osVersionMsg;
1184 osVersionMsg.Printf(_T("OS Version reports as: %d.%d"), osMajor, osMinor);
1185 wxLogMessage(osVersionMsg);
1186
1187 wxLogMessage(_T("MemoryStatus: mem_total: %d mb, mem_initial: %d mb"),
1188 g_mem_total / 1024, g_mem_initial / 1024);
1189
1190 OCPN_OSDetail *detail = g_Platform->GetOSDetail();
1191 wxString msgplat;
1192 wxString like0;
1193 if (!detail->osd_names_like.empty())
1194 like0 = detail->osd_names_like[0].c_str();
1195 msgplat.Printf("OCPN_OSDetail: %s ; %s ; %s ; %s ; %s",
1196 detail->osd_arch.c_str(), detail->osd_name.c_str(),
1197 detail->osd_version.c_str(), detail->osd_ID.c_str(),
1198 like0.mb_str());
1199 wxLogMessage(msgplat);
1200
1201 wxString imsg = _T("SData_Locn is ");
1202 imsg += g_Platform->GetSharedDataDir();
1203 wxLogMessage(imsg);
1204
1205 // Initialize embedded PNG icon graphics
1206 ::wxInitAllImageHandlers();
1207
1208#ifdef __WXQT__
1209 // Now we can configure the Qt StyleSheets, if present
1210 prepareAndroidStyleSheets();
1211#endif
1212
1213 // Create some static strings
1214 pInit_Chart_Dir = new wxString();
1215
1216 // Establish an empty ChartCroupArray
1217 g_pGroupArray = new ChartGroupArray;
1218
1219 imsg = _T("PrivateDataDir is ");
1220 imsg += g_Platform->GetPrivateDataDir();
1221 wxLogMessage(imsg);
1222
1223 // Create an array string to hold repeating messages, so they don't
1224 // overwhelm the log
1225 pMessageOnceArray = new wxArrayString;
1226
1227 // Init the Route Manager
1228
1229 g_pRouteMan =
1230 new Routeman(RoutePropDlg::GetDlgCtx(), RoutemanGui::GetDlgCtx(),
1231 NMEALogWindow::GetInstance());
1232
1233 // Init the Selectable Route Items List
1234 pSelect = new Select();
1235 pSelect->SetSelectPixelRadius(12);
1236
1237 // Init the Selectable Tide/Current Items List
1238 pSelectTC = new Select();
1239 // Increase the select radius for tide/current stations
1240 pSelectTC->SetSelectPixelRadius(25);
1241
1242 // Init the Selectable AIS Target List
1243 pSelectAIS = new Select();
1244 pSelectAIS->SetSelectPixelRadius(12);
1245
1246 // Initially AIS display is always on
1247 g_bShowAIS = true;
1248 g_pais_query_dialog_active = NULL;
1249
1250 // Who am I?
1251 g_hostname = ::wxGetHostName();
1252 if (g_hostname.IsEmpty()) g_hostname = wxGetUserName();
1253#ifdef __ANDROID__
1254 androidGetDeviceInfo();
1255 g_hostname = wxString("Android-") + g_android_Device_Model;
1256 g_hostname.Replace(" ", "-", true);
1257#endif
1258
1259 // A Portabel need a unique mDNS data hostname to share routes.
1260 if (g_bportable) {
1261 wxString p("Portable-");
1262 g_hostname = p + g_hostname;
1263 }
1264
1265 // Initialize some lists
1266 // Layers
1267 pLayerList = new LayerList;
1268 // Routes
1269 pRouteList = new RouteList;
1270
1271 // (Optionally) Capture the user and file(effective) ids
1272 // Some build environments may need root privileges for hardware
1273 // port I/O, as in the NMEA data input class. Set that up here.
1274
1275#ifndef __WXMSW__
1276#ifdef PROBE_PORTS__WITH_HELPER
1277 user_user_id = getuid();
1278 file_user_id = geteuid();
1279#endif
1280#endif
1281
1282 bool b_initial_load = false;
1283
1284 wxFileName config_test_file_name(g_Platform->GetConfigFileName());
1285 if (config_test_file_name.FileExists())
1286 wxLogMessage(_T("Using existing Config_File: ") +
1287 g_Platform->GetConfigFileName());
1288 else {
1289 {
1290 wxLogMessage(_T("Creating new Config_File: ") +
1291 g_Platform->GetConfigFileName());
1292
1293 b_initial_load = true;
1294
1295 if (true !=
1296 config_test_file_name.DirExists(config_test_file_name.GetPath()))
1297 if (!config_test_file_name.Mkdir(config_test_file_name.GetPath()))
1298 wxLogMessage(_T("Cannot create config file directory for ") +
1299 g_Platform->GetConfigFileName());
1300 }
1301 }
1302
1303 // Open/Create the Config Object
1304 pConfig = g_Platform->GetConfigObject();
1305 InitBaseConfig(pConfig);
1306 pConfig->LoadMyConfig();
1307
1308 // Override for some safe and nice default values if the config file was
1309 // created from scratch
1310 if (b_initial_load) g_Platform->SetDefaultOptions();
1311
1312 g_Platform->applyExpertMode(g_bUIexpert);
1313
1314 // Now initialize UI Style.
1315 g_StyleManager = new ocpnStyle::StyleManager();
1316
1317 // if(g_useMUI)
1318 // g_uiStyle = _T("MUI_flat");
1319
1320 g_StyleManager->SetStyle(_T("MUI_flat"));
1321 if (!g_StyleManager->IsOK()) {
1322 wxString msg = _("Failed to initialize the user interface. ");
1323 msg << _("OpenCPN cannot start. ");
1324 msg << _("The necessary configuration files were not found. ");
1325 msg << _("See the log file at ") << g_Platform->GetLogFileName()
1326 << _(" for details.") << _T("\n\n");
1327 msg << g_Platform->GetSharedDataDir();
1328
1329 wxMessageDialog w(NULL, msg, _("Failed to initialize the user interface. "),
1330 wxCANCEL | wxICON_ERROR);
1331 w.ShowModal();
1332 exit(EXIT_FAILURE);
1333 }
1334
1335 if (g_useMUI) {
1336 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
1337 if (style) style->chartStatusWindowTransparent = true;
1338 }
1339
1340 // Init the WayPoint Manager
1341 pWayPointMan = NULL;
1342
1343 g_display_size_mm = wxMax(50, g_Platform->GetDisplaySizeMM());
1344 wxString msg;
1345 msg.Printf(_T("Detected display size (horizontal): %d mm"),
1346 (int)g_display_size_mm);
1347 wxLogMessage(msg);
1348
1349 // User override....
1350 if (g_config_display_size_manual &&
1351 g_config_display_size_mm.size() > g_current_monitor &&
1352 g_config_display_size_mm[g_current_monitor] > 0) {
1353 g_display_size_mm = g_config_display_size_mm[g_current_monitor];
1354 wxString msg;
1355 msg.Printf(_T("Display size (horizontal) config override: %d mm"),
1356 (int)g_display_size_mm);
1357 wxLogMessage(msg);
1358 g_Platform->SetDisplaySizeMM(g_current_monitor, g_display_size_mm);
1359 }
1360
1361 g_display_size_mm = wxMax(50, g_display_size_mm);
1362
1363 if (g_btouch) {
1364 int SelectPixelRadius = 50;
1365
1366 pSelect->SetSelectPixelRadius(SelectPixelRadius);
1367 pSelectTC->SetSelectPixelRadius(wxMax(25, SelectPixelRadius));
1368 pSelectAIS->SetSelectPixelRadius(SelectPixelRadius);
1369 }
1370
1371 // Is this the first run after a clean installation?
1372 if (!n_NavMessageShown) {
1373 g_bFirstRun = true;
1374 }
1375
1376 // Now we can set the locale
1377 // using wxWidgets/gettext methodology....
1378
1379#if wxUSE_XLOCALE || !wxCHECK_VERSION(3, 0, 0)
1380
1381 // Where are the opencpn.mo files?
1382 g_Platform->SetLocaleSearchPrefixes();
1383
1384 wxString def_lang_canonical = g_Platform->GetDefaultSystemLocale();
1385
1386 imsg = _T("System default Language: ") + def_lang_canonical;
1387 wxLogMessage(imsg);
1388
1389 wxString cflmsg = _T("Config file language: ") + g_locale;
1390 wxLogMessage(cflmsg);
1391
1392 if (g_locale.IsEmpty()) {
1393 g_locale = def_lang_canonical;
1394 cflmsg =
1395 _T("Config file language empty, using system default: ") + g_locale;
1396 wxLogMessage(cflmsg);
1397 }
1398
1399 // Make any adjustments necessary
1400 g_locale = g_Platform->GetAdjustedAppLocale();
1401 cflmsg = _T("Adjusted App language: ") + g_locale;
1402 wxLogMessage(cflmsg);
1403
1404 // Set the desired locale
1405 g_Platform->ChangeLocale(g_locale, plocale_def_lang, &plocale_def_lang);
1406
1407 imsg = _T("Opencpn language set to: ");
1408 imsg += g_locale;
1409 wxLogMessage(imsg);
1410
1411 // French language locale is assumed to include the AZERTY keyboard
1412 // This applies to either the system language, or to OpenCPN language
1413 // selection
1414 if (g_locale == _T("fr_FR")) g_b_assume_azerty = true;
1415#else
1416 wxLogMessage(_T("wxLocale support not available"));
1417#endif
1418
1419#ifndef __ANDROID__
1420 // Now that locale is established, possibly run the startup wizard.
1421 if (g_config_wizard || b_initial_load) {
1422 FirstUseWizImpl wiz(gFrame, pConfig);
1423 auto res = wiz.Run();
1424 if (res) {
1425 g_NeedDBUpdate = 2;
1426 }
1427 }
1428#endif
1429
1430 // Instantiate and initialize the Config Manager
1431 ConfigMgr::Get();
1432
1433 // Is this an upgrade?
1434 wxString vs = wxString("Version ") + VERSION_FULL + " Build " + VERSION_DATE;
1435 g_bUpgradeInProcess = (vs != g_config_version_string);
1436
1437 g_Platform->SetUpgradeOptions(vs, g_config_version_string);
1438
1439 // log deferred log restart message, if it exists.
1440 if (!g_Platform->GetLargeLogMessage().IsEmpty()) {
1441 wxLogMessage(g_Platform->GetLargeLogMessage());
1442 }
1443
1444 // Validate OpenGL functionality, if selected
1445#ifndef ocpnUSE_GL
1446 g_bdisable_opengl = true;
1447 ;
1448#endif
1449
1450 if (g_bdisable_opengl) g_bopengl = false;
1451
1452#if defined(__linux__) && !defined(__ANDROID__)
1453 if (g_bSoftwareGL) {
1454 setenv("LIBGL_ALWAYS_SOFTWARE", "1", 1);
1455 }
1456#endif
1457
1458 // FIXMW (dave) move to frame
1459 // g_bTransparentToolbarInOpenGLOK = isTransparentToolbarInOpenGLOK();
1460
1461 // On Windows platforms, establish a default cache managment policy
1462 // as allowing OpenCPN a percentage of available physical memory,
1463 // not to exceed 1 GB
1464 // Note that this logic implies that Windows platforms always use
1465 // the memCacheLimit policy, and never use the fallback nCacheLimit policy
1466#ifdef __WXMSW__
1467 if (0 == g_memCacheLimit) g_memCacheLimit = (int)(g_mem_total * 0.5);
1468 g_memCacheLimit =
1469 wxMin(g_memCacheLimit, 1024 * 1024); // math in kBytes, Max is 1 GB
1470#else
1471 // All other platforms will use the nCacheLimit policy
1472 // sinc on linux it is impossible to accurately measure the application memory
1473 // footprint without expensive methods such as malloc/free tracking, and such
1474
1475 g_memCacheLimit = 0;
1476 if (0 == g_nCacheLimit) // allow config file override
1477 g_nCacheLimit = CACHE_N_LIMIT_DEFAULT;
1478#endif
1479
1480 // Establish location and name of chart database
1481 ChartListFileName = newPrivateFileName(g_Platform->GetPrivateDataDir(),
1482 "chartlist.dat", "CHRTLIST.DAT");
1483
1484 // Establish location and name of AIS MMSI -> Target Name mapping
1485 AISTargetNameFileName = newPrivateFileName(g_Platform->GetPrivateDataDir(),
1486 "mmsitoname.csv", "MMSINAME.CSV");
1487
1488 // Establish guessed location of chart tree
1489 if (pInit_Chart_Dir->IsEmpty()) {
1490 wxStandardPaths &std_path = g_Platform->GetStdPaths();
1491
1492 if (!g_bportable)
1493#ifndef __ANDROID__
1494 pInit_Chart_Dir->Append(std_path.GetDocumentsDir());
1495#else
1496 pInit_Chart_Dir->Append(androidGetExtStorageDir());
1497#endif
1498 }
1499
1500 InitRestListeners();
1501
1502 // Establish the GSHHS Dataset location
1503 gDefaultWorldMapLocation = "gshhs";
1504 gDefaultWorldMapLocation.Prepend(g_Platform->GetSharedDataDir());
1505 gDefaultWorldMapLocation.Append(wxFileName::GetPathSeparator());
1506 if (gWorldMapLocation == wxEmptyString) {
1507 gWorldMapLocation = gDefaultWorldMapLocation;
1508 }
1509
1510 // Check the global Tide/Current data source array
1511 // If empty, preset default (US + ROW) data sources
1512 wxString default_tcdata0 =
1513 (g_Platform->GetSharedDataDir() + _T("tcdata") +
1514 wxFileName::GetPathSeparator() + _T("harmonics-dwf-20210110-free.tcd"));
1515 wxString default_tcdata1 =
1516 (g_Platform->GetSharedDataDir() + _T("tcdata") +
1517 wxFileName::GetPathSeparator() + _T("HARMONICS_NO_US.IDX"));
1518
1519 if (TideCurrentDataSet.empty()) {
1520 TideCurrentDataSet.push_back(
1521 g_Platform->NormalizePath(default_tcdata0).ToStdString());
1522 TideCurrentDataSet.push_back(
1523 g_Platform->NormalizePath(default_tcdata1).ToStdString());
1524 }
1525
1526 // Check the global AIS alarm sound file
1527 // If empty, preset default
1528 if (g_sAIS_Alert_Sound_File.IsEmpty()) {
1529 wxString default_sound =
1530 (g_Platform->GetSharedDataDir() + _T("sounds") +
1531 wxFileName::GetPathSeparator() + _T("2bells.wav"));
1532 g_sAIS_Alert_Sound_File = g_Platform->NormalizePath(default_sound);
1533 }
1534
1535 gpIDXn = 0;
1536
1537 g_Platform->Initialize_2();
1538
1539 // Set up the frame initial visual parameters
1540 // Default size, resized later
1541 wxSize new_frame_size(-1, -1);
1542 int cx, cy, cw, ch;
1543 ::wxClientDisplayRect(&cx, &cy, &cw, &ch);
1544
1545 InitializeUserColors();
1546
1547 auto style = g_StyleManager->GetCurrentStyle();
1548 auto bitmap = new wxBitmap(style->GetIcon("default_pi", 32, 32));
1549 if (bitmap->IsOk())
1550 PluginLoader::getInstance()->SetPluginDefaultIcon(bitmap);
1551 else
1552 wxLogWarning("Cannot initiate plugin default jigsaw icon.");
1553
1554 if ((g_nframewin_x > 100) && (g_nframewin_y > 100) && (g_nframewin_x <= cw) &&
1555 (g_nframewin_y <= ch))
1556 new_frame_size.Set(g_nframewin_x, g_nframewin_y);
1557 else
1558 new_frame_size.Set(cw * 7 / 10, ch * 7 / 10);
1559
1560 // Try to detect any change in physical screen configuration
1561 // This can happen when drivers are changed, for instance....
1562 // and can confuse the WUI layout perspective stored in the config file.
1563 // If detected, force a nominal window size and position....
1564 if ((g_lastClientRectx != cx) || (g_lastClientRecty != cy) ||
1565 (g_lastClientRectw != cw) || (g_lastClientRecth != ch)) {
1566 new_frame_size.Set(cw * 7 / 10, ch * 7 / 10);
1567 g_bframemax = false;
1568 }
1569
1570 g_lastClientRectx = cx;
1571 g_lastClientRecty = cy;
1572 g_lastClientRectw = cw;
1573 g_lastClientRecth = ch;
1574
1575 // Validate config file position
1576 wxPoint position(0, 0);
1577 wxSize dsize = wxGetDisplaySize();
1578
1579#ifdef __WXMAC__
1580 g_nframewin_posy = wxMax(g_nframewin_posy, 22);
1581#endif
1582
1583 if ((g_nframewin_posx < dsize.x) && (g_nframewin_posy < dsize.y))
1584 position = wxPoint(g_nframewin_posx, g_nframewin_posy);
1585
1586#ifdef __WXMSW__
1587 // Support MultiMonitor setups which can allow negative window positions.
1588 RECT frame_rect;
1589 frame_rect.left = position.x;
1590 frame_rect.top = position.y;
1591 frame_rect.right = position.x + new_frame_size.x;
1592 frame_rect.bottom = position.y + new_frame_size.y;
1593
1594 // If the requested frame window does not intersect any installed monitor,
1595 // then default to simple primary monitor positioning.
1596 if (NULL == MonitorFromRect(&frame_rect, MONITOR_DEFAULTTONULL))
1597 position = wxPoint(10, 10);
1598#endif
1599
1600#ifdef __WXOSX__
1601 // Support MultiMonitor setups which can allow negative window positions.
1602 const wxPoint ptScreen(position.x, position.y);
1603 const int displayIndex = wxDisplay::GetFromPoint(ptScreen);
1604
1605 if (displayIndex == wxNOT_FOUND) position = wxPoint(10, 30);
1606#endif
1607
1608 g_nframewin_posx = position.x;
1609 g_nframewin_posy = position.y;
1610
1611#ifdef __ANDROID__
1612 wxSize asz = getAndroidDisplayDimensions();
1613 ch = asz.y;
1614 cw = asz.x;
1615 // qDebug() << cw << ch;
1616
1617 if ((cw > 200) && (ch > 200))
1618 new_frame_size.Set(cw, ch);
1619 else
1620 new_frame_size.Set(800, 400);
1621#endif
1622
1623 // For Windows and GTK, provide the expected application Minimize/Close bar
1624 long app_style = wxDEFAULT_FRAME_STYLE;
1625 app_style |= wxWANTS_CHARS;
1626
1627 // Create the main frame window
1628
1629 // Strip the commit SHA number from the string to be shown in frame title.
1630 wxString short_version_name = wxString(PACKAGE_VERSION).BeforeFirst('+');
1631 wxString myframe_window_title =
1632 wxString(wxT("OpenCPN ") + short_version_name);
1633
1634 if (g_bportable) {
1635 myframe_window_title += _(" -- [Portable(-p) executing from ");
1636 myframe_window_title += g_Platform->GetHomeDir();
1637 myframe_window_title += _T("]");
1638 }
1639
1640 wxString fmsg;
1641 fmsg.Printf(_T("Creating MyFrame...size(%d, %d) position(%d, %d)"),
1642 new_frame_size.x, new_frame_size.y, position.x, position.y);
1643 wxLogMessage(fmsg);
1644
1645 gFrame = new MyFrame(NULL, myframe_window_title, position, new_frame_size,
1646 app_style); // Gunther
1647
1648 // Do those platform specific initialization things that need gFrame
1649 g_Platform->Initialize_3();
1650
1651 // Initialize the Plugin Manager
1652 g_pi_manager = new PlugInManager(gFrame);
1653
1654 // g_pauimgr = new wxAuiManager;
1655 g_pauimgr = new OCPN_AUIManager;
1656 g_pauidockart = new wxAuiDefaultDockArt;
1657 g_pauimgr->SetArtProvider(g_pauidockart);
1658 g_pauimgr->SetDockSizeConstraint(.9, .9);
1659
1660 // g_pauimgr->SetFlags(g_pauimgr->GetFlags() | wxAUI_MGR_LIVE_RESIZE);
1661
1662 // tell wxAuiManager to manage the frame
1663 g_pauimgr->SetManagedWindow(gFrame);
1664
1665 gFrame->CreateCanvasLayout();
1666
1667 // gFrame->RequestNewMasterToolbar( true );
1668
1669 gFrame->SetChartUpdatePeriod(); // Reasonable default
1670
1671 gFrame->Enable();
1672
1673 gFrame->GetPrimaryCanvas()->SetFocus();
1674
1675 pthumbwin = new ThumbWin(gFrame->GetPrimaryCanvas());
1676
1677 gFrame->ApplyGlobalSettings(false); // done once on init with resize
1678
1679 gFrame->SetAllToolbarScale();
1680
1681 // Show the frame
1682 gFrame->Show(TRUE);
1683 Yield(); // required for Gnome 45
1684
1685 gFrame->SetAndApplyColorScheme(global_color_scheme);
1686
1687 if (g_bframemax) gFrame->Maximize(true);
1688
1689#ifdef __ANDROID__
1690 if (g_bresponsive && (gFrame->GetPrimaryCanvas()->GetPixPerMM() > 4.0))
1691 gFrame->Maximize(true);
1692#endif
1693
1694 // Yield to pick up the OnSize() calls that result from Maximize()
1695 Yield();
1696
1697 // Build the initial chart dir array
1698 ArrayOfCDI ChartDirArray;
1699 pConfig->LoadChartDirArray(ChartDirArray);
1700
1701 // Windows installer may have left hints regarding the initial chart dir
1702 // selection
1703#ifdef __WXMSW__
1704 if (g_bFirstRun && (ChartDirArray.GetCount() == 0)) {
1705 int ndirs = 0;
1706
1707 wxRegKey RegKey(wxString(_T("HKEY_LOCAL_MACHINE\\SOFTWARE\\OpenCPN")));
1708 if (RegKey.Exists()) {
1709 wxLogMessage(
1710 _("Retrieving initial Chart Directory set from Windows Registry"));
1711 wxString dirs;
1712 RegKey.QueryValue(wxString(_T("ChartDirs")), dirs);
1713
1714 wxStringTokenizer tkz(dirs, _T(";"));
1715 while (tkz.HasMoreTokens()) {
1716 wxString token = tkz.GetNextToken();
1717
1718 ChartDirInfo cdi;
1719 cdi.fullpath = token.Trim();
1720 cdi.magic_number = _T("");
1721
1722 ChartDirArray.Add(cdi);
1723 ndirs++;
1724 }
1725 }
1726
1727 if (g_bportable) {
1728 ChartDirInfo cdi;
1729 cdi.fullpath = _T("charts");
1730 cdi.fullpath.Prepend(g_Platform->GetSharedDataDir());
1731 cdi.magic_number = _T("");
1732 ChartDirArray.Add(cdi);
1733 ndirs++;
1734 }
1735
1736 if (ndirs) pConfig->UpdateChartDirs(ChartDirArray);
1737 }
1738#endif
1739
1740 // If the ChartDirArray is empty at this point, any existing chart database
1741 // file must be declared invalid, So it is best to simply delete it if
1742 // present.
1743 // TODO There is a possibility of recreating the dir list from the
1744 // database itself......
1745
1746 if (!ChartDirArray.GetCount())
1747 if (::wxFileExists(ChartListFileName)) ::wxRemoveFile(ChartListFileName);
1748
1749 // Try to load the current chart list Data file
1750 ChartData = new ChartDB();
1751 if (g_NeedDBUpdate == 0 &&
1752 !ChartData->LoadBinary(ChartListFileName, ChartDirArray)) {
1753 g_NeedDBUpdate = 1;
1754 }
1755
1756 // Verify any saved chart database startup index
1757 if (g_restore_dbindex >= 0) {
1758 if (ChartData->GetChartTableEntries() == 0)
1759 g_restore_dbindex = -1;
1760
1761 else if (g_restore_dbindex > (ChartData->GetChartTableEntries() - 1))
1762 g_restore_dbindex = 0;
1763 }
1764
1765 // Apply the inital Group Array structure to the chart database
1766 ChartData->ApplyGroupArray(g_pGroupArray);
1767
1768 // All set to go.....
1769
1770 // Process command line option to rebuild cache
1771#ifdef ocpnUSE_GL
1772 extern ocpnGLOptions g_GLOptions;
1773
1774 if (g_rebuild_gl_cache && g_bopengl && g_GLOptions.m_bTextureCompression &&
1775 g_GLOptions.m_bTextureCompressionCaching) {
1776 gFrame->ReloadAllVP(); // Get a nice chart background loaded
1777
1778 // Turn off the toolbar as a clear signal that the system is busy right
1779 // now.
1780 // Note: I commented this out because the toolbar never comes back for me
1781 // and is unusable until I restart opencpn without generating the cache
1782 // if( g_MainToolbar )
1783 // g_MainToolbar->Hide();
1784
1785 if (g_glTextureManager) g_glTextureManager->BuildCompressedCache();
1786 }
1787#endif
1788
1789 // FIXME (dave)
1790 // move method to frame
1791 // if (g_parse_all_enc) ParseAllENC(gFrame);
1792
1793 // establish GPS timeout value as multiple of frame timer
1794 // This will override any nonsense or unset value from the config file
1795 if ((gps_watchdog_timeout_ticks > 60) || (gps_watchdog_timeout_ticks <= 0))
1796 gps_watchdog_timeout_ticks = (GPS_TIMEOUT_SECONDS * 1000) / TIMER_GFRAME_1;
1797
1798 wxString dogmsg;
1799 dogmsg.Printf(_T("GPS Watchdog Timeout is: %d sec."),
1800 gps_watchdog_timeout_ticks);
1801 wxLogMessage(dogmsg);
1802
1803 sat_watchdog_timeout_ticks = gps_watchdog_timeout_ticks;
1804
1805 g_priSats = 99;
1806
1807 // Most likely installations have no ownship heading information
1808 g_bVAR_Rx = false;
1809
1810 // Start up a new track if enabled in config file
1811 if (g_bTrackCarryOver) g_bDeferredStartTrack = true;
1812
1813 pAnchorWatchPoint1 = NULL;
1814 pAnchorWatchPoint2 = NULL;
1815
1816 Yield();
1817
1818 gFrame->DoChartUpdate();
1819
1820 FontMgr::Get()
1821 .ScrubList(); // Clean the font list, removing nonsensical entries
1822
1823 gFrame->ReloadAllVP(); // once more, and good to go
1824
1825 gFrame->Refresh(false);
1826 gFrame->Raise();
1827
1828 gFrame->GetPrimaryCanvas()->Enable();
1829 gFrame->GetPrimaryCanvas()->SetFocus();
1830
1831 // This little hack fixes a problem seen with some UniChrome OpenGL drivers
1832 // We need a deferred resize to get glDrawPixels() to work right.
1833 // So we set a trigger to generate a resize after 5 seconds....
1834 // See the "UniChrome" hack elsewhere
1835#ifdef ocpnUSE_GL
1836 if (!g_bdisable_opengl) {
1837 glChartCanvas *pgl =
1838 (glChartCanvas *)gFrame->GetPrimaryCanvas()->GetglCanvas();
1839 if (pgl &&
1840 (pgl->GetRendererString().Find(_T("UniChrome")) != wxNOT_FOUND)) {
1841 gFrame->m_defer_size = gFrame->GetSize();
1842 gFrame->SetSize(gFrame->m_defer_size.x - 10, gFrame->m_defer_size.y);
1843 g_pauimgr->Update();
1844 gFrame->m_bdefer_resize = true;
1845 }
1846 }
1847#endif
1848
1849 // Horrible Hack (tm): Make sure the RoutePoint destructor can invoke
1850 // glDeleteTextures. Truly awful.
1851#ifdef ocpnUSE_GL
1852 if (g_bopengl)
1853 RoutePoint::delete_gl_textures = [](unsigned n, const unsigned *texts) {
1854 glDeleteTextures(n, texts);
1855 };
1856#else
1857 RoutePoint::delete_gl_textures = [](unsigned n, const unsigned *texts) {};
1858#endif
1859
1860 if (g_start_fullscreen) gFrame->ToggleFullScreen();
1861
1862#ifdef __ANDROID__
1863 // We need a resize to pick up height adjustment after building android
1864 // ActionBar
1865 gFrame->SetSize(getAndroidDisplayDimensions());
1866 androidSetFollowTool(gFrame->GetPrimaryCanvas()->m_bFollow ? 1 : 0, true);
1867#endif
1868
1869 gFrame->Raise();
1870 gFrame->GetPrimaryCanvas()->Enable();
1871 gFrame->GetPrimaryCanvas()->SetFocus();
1872
1873 // Setup Tides/Currents to settings present at last shutdown
1874 // TODO
1875 // gFrame->ShowTides( g_bShowTide );
1876 // gFrame->ShowCurrents( g_bShowCurrent );
1877
1878 // Start up the ticker....
1879 gFrame->FrameTimer1.Start(TIMER_GFRAME_1, wxTIMER_CONTINUOUS);
1880
1881 // Start up the ViewPort Rotation angle Averaging Timer....
1882 gFrame->FrameCOGTimer.Start(2000, wxTIMER_CONTINUOUS);
1883
1884 // Start up the Ten Hz timer....
1885 gFrame->FrameTenHzTimer.Start(100, wxTIMER_CONTINUOUS);
1886
1887 // wxLogMessage( wxString::Format(_T("OpenCPN Initialized in %ld ms."),
1888 // init_sw.Time() ) );
1889
1890 OCPNPlatform::Initialize_4();
1891
1892#ifdef __ANDROID__
1893 androidHideBusyIcon();
1894#endif
1895 wxLogMessage(
1896 wxString::Format(_("OpenCPN Initialized in %ld ms."), init_sw.Time()));
1897
1898 wxMilliSleep(500);
1899
1900#ifdef __ANDROID__
1901 // We defer the startup message to here to allow the app frame to be
1902 // contructed, thus avoiding a dialog with NULL parent which might not work
1903 // on some devices.
1904 if (!n_NavMessageShown || (vs != g_config_version_string) ||
1905 (g_AndroidVersionCode != androidGetVersionCode())) {
1906 // qDebug() << "Showing NavWarning";
1907 wxMilliSleep(500);
1908
1909 if (!ShowNavWarning()) {
1910 qDebug() << "Closing due to NavWarning Cancel";
1911 gFrame->Close();
1912 androidTerminate();
1913 return true;
1914 }
1915
1916 n_NavMessageShown = 1;
1917 }
1918
1919 // Finished with upgrade checking, so persist the currect Version Code
1920 g_AndroidVersionCode = androidGetVersionCode();
1921 qDebug() << "Persisting Version Code: " << g_AndroidVersionCode;
1922#else
1923 // Send the Welcome/warning message if it has never been sent before,
1924 // or if the version string has changed at all
1925 // We defer until here to allow for localization of the message
1926 if (!n_NavMessageShown || (vs != g_config_version_string)) {
1927 if (!ShowNavWarning()) return false;
1928 n_NavMessageShown = 1;
1929 pConfig->Flush();
1930 }
1931#endif
1932
1933 // As an a.e. Raspberry does not have a hardwareclock we will have some
1934 // problems with date/time setting
1935 g_bHasHwClock = true; // by default most computers do have a hwClock
1936#if defined(__UNIX__) && !defined(__ANDROID__)
1937 struct stat buffer;
1938 g_bHasHwClock =
1939 ((stat("/dev/rtc", &buffer) == 0) || (stat("/dev/rtc0", &buffer) == 0) ||
1940 (stat("/dev/misc/rtc", &buffer) == 0));
1941#endif
1942
1943 g_config_version_string = vs;
1944
1945 // The user accepted the "not for navigation" nag, so persist it here...
1946 pConfig->UpdateSettings();
1947
1948 // Start delayed initialization chain after some milliseconds
1949 gFrame->InitTimer.Start(5, wxTIMER_CONTINUOUS);
1950
1951 g_pauimgr->Update();
1952
1953 for (auto *cp : TheConnectionParams()) {
1954 if (cp->bEnabled) {
1955 if (cp->GetDSPort().Contains("Serial")) {
1956 std::string port(cp->Port.ToStdString());
1957 CheckSerialAccess(gFrame, port);
1958 }
1959 }
1960 }
1961 CheckDongleAccess(gFrame);
1962
1963 // Initialize the CommBridge
1964 m_comm_bridge.Initialize();
1965
1966 std::vector<std::string> ipv4_addrs = get_local_ipv4_addresses();
1967
1968 // If network connection is available, start the server and mDNS client
1969 if (ipv4_addrs.size()) {
1970 std::string ipAddr = ipv4_addrs[0];
1971
1972 wxString data_dir = g_Platform->GetPrivateDataDir();
1973 if (data_dir.Last() != wxFileName::GetPathSeparator())
1974 data_dir.Append(wxFileName::GetPathSeparator());
1975
1976 make_certificate(ipAddr, data_dir.ToStdString());
1977
1978 m_rest_server.StartServer(fs::path(data_dir.ToStdString()));
1979 StartMDNSService(g_hostname.ToStdString(), "opencpn-object-control-service",
1980 8000);
1981 }
1982 return TRUE;
1983}
1984
1985int MyApp::OnExit() {
1986 wxLogMessage(_T("opencpn::MyApp starting exit."));
1987 m_checker.OnExit();
1988 m_usb_watcher.Stop();
1989 // Send current nav status data to log file // pjotrc 2010.02.09
1990
1991 wxDateTime lognow = wxDateTime::Now();
1992 lognow.MakeGMT();
1993 wxString day = lognow.FormatISODate();
1994 wxString utc = lognow.FormatISOTime();
1995 wxString navmsg = _T("LOGBOOK: ");
1996 navmsg += day;
1997 navmsg += _T(" ");
1998 navmsg += utc;
1999 navmsg += _T(" UTC ");
2000
2001 if (bGPSValid) {
2002 wxString data;
2003 data.Printf(_T("OFF: Lat %10.5f Lon %10.5f "), gLat, gLon);
2004 navmsg += data;
2005
2006 wxString cog;
2007 if (std::isnan(gCog))
2008 cog.Printf(_T("COG ----- "));
2009 else
2010 cog.Printf(_T("COG %10.5f "), gCog);
2011
2012 wxString sog;
2013 if (std::isnan(gSog))
2014 sog.Printf(_T("SOG ----- "));
2015 else
2016 sog.Printf(_T("SOG %6.2f ") + getUsrSpeedUnit(), toUsrSpeed(gSog));
2017
2018 navmsg += cog;
2019 navmsg += sog;
2020
2021 } else {
2022 wxString data;
2023 data.Printf(_T("OFF: Lat %10.5f Lon %10.5f"), gLat, gLon);
2024 navmsg += data;
2025 }
2026 wxLogMessage(navmsg);
2027 g_loglast_time = lognow;
2028
2029 if (ptcmgr) delete ptcmgr;
2030
2031 for (Track *track : g_TrackList) {
2032 delete track;
2033 }
2034 g_TrackList.clear();
2035
2036 delete pConfig;
2037 delete pSelect;
2038 delete pSelectTC;
2039 delete pSelectAIS;
2040
2041 delete ps52plib;
2042 delete g_SencThreadManager;
2043
2044 if (g_pGroupArray) {
2045 for (unsigned int igroup = 0; igroup < g_pGroupArray->GetCount();
2046 igroup++) {
2047 delete g_pGroupArray->Item(igroup);
2048 }
2049
2050 g_pGroupArray->Clear();
2051 delete g_pGroupArray;
2052 }
2053
2054 wxLogMessage(_T("opencpn::MyApp exiting cleanly...\n"));
2055 wxLog::FlushActive();
2056
2057 g_Platform->CloseLogFile();
2058
2059 delete pInit_Chart_Dir;
2060
2061 for (Track *track : g_TrackList) {
2062 delete track;
2063 }
2064 g_TrackList.clear();
2065
2066 delete g_pRouteMan;
2067 delete pWayPointMan;
2068
2069 delete pMessageOnceArray;
2070
2071 DeInitializeUserColors();
2072
2073 delete pLayerList;
2074
2075 delete m_pRegistrarMan;
2076 CSVDeaccess(NULL);
2077
2078 delete g_StyleManager;
2079
2080#ifdef __WXMSW__
2081#ifdef USE_GLU_TESS
2082#ifdef USE_GLU_DLL
2083 if (s_glu_dll_ready) {
2084 FreeLibrary(s_hGLU_DLL);
2085 } // free the glu32.dll
2086#endif
2087#endif
2088#endif
2089
2090 // Restore any changed system colors
2091
2092#ifdef __WXMSW__
2093 void RestoreSystemColors(void);
2094 RestoreSystemColors();
2095#endif
2096
2097#ifdef __MSVC__LEAK
2098 DeInitAllocCheck();
2099#endif
2100
2101#if wxUSE_XLOCALE || !wxCHECK_VERSION(3, 0, 0)
2102 if (plocale_def_lang) delete plocale_def_lang;
2103#endif
2104
2105 FontMgr::Shutdown();
2106
2107 g_Platform->OnExit_2();
2109 delete g_Platform;
2110
2111 return TRUE;
2112}
2113
2114#ifdef LINUX_CRASHRPT
2115void MyApp::OnFatalException() { g_crashprint.Report(); }
2116#endif
2117
2118//----------------------------------------------------------------------------------------------------------
2119// Application-wide CPL Error handler
2120//----------------------------------------------------------------------------------------------------------
2121void MyCPLErrorHandler(CPLErr eErrClass, int nError, const char *pszErrorMsg)
2122
2123{
2124 char msg[256];
2125
2126 if (eErrClass == CE_Debug)
2127 snprintf(msg, 255, "CPL: %s", pszErrorMsg);
2128 else if (eErrClass == CE_Warning)
2129 snprintf(msg, 255, "CPL Warning %d: %s", nError, pszErrorMsg);
2130 else
2131 snprintf(msg, 255, "CPL ERROR %d: %s", nError, pszErrorMsg);
2132
2133 wxString str(msg, wxConvUTF8);
2134 wxLogMessage(str);
2135}
Global state for AIS decoder.
Dialog for displaying a list of AIS targets.
Dialog for querying detailed information about an AIS target.
Implements the AboutFrame class with additional functionality.
wxString & GetPrivateDataDir()
Return dir path for opencpn.log, etc., respecting -c cli option.
EventVar reverse_route
Notified with a string GUID when user wants to reverse a route.
EventVar activate_route
Notified with a string GUID when user wants to activate a route.
Represents an active track that is currently being recorded.
Definition track.h:194
Handles the AIS information GUI and sound alerts.
Dialog for managing CM93 chart offsets.
Definition cm93.h:547
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
bool LoadBinary(const wxString &filename, ArrayOfCDI &dir_array_check)
Load the chart database from a binary file.
Definition chartdb.cpp:233
Primary navigation console display for route and vessel tracking.
Definition concanv.h:127
const void Notify()
Notify all listeners, no data supplied.
void ScrubList()
Cleans up stale font entries after a locale change.
Definition FontMgr.cpp:565
Common interface for all instance checkers.
virtual bool IsMainInstance()=0
Return true if this process is the primary opencpn instance.
virtual void CleanUp()
Remove all persistent instance state, including possible lock file and defunct opencpn processes.
virtual void OnExit()
Do whatever needed before wxWidget's checks triggers.
virtual void WaitUntilValid()
Wait until this object can be used for example for Dbus connection.
static LocalServerApi & GetInstance()
Dialog for displaying and editing waypoint properties.
Definition MarkInfo.h:212
Main application frame.
Definition ocpn_frame.h:135
Provides platform-specific support utilities for OpenCPN.
void SetDisplaySizeMM(size_t monitor, double size)
Set the width of the monitor in millimeters.
double GetDisplaySizeMM()
Get the width of the screen in millimeters.
void Init(const KeyProvider &kp, std::function< void(ObservedEvt &ev)> action)
Initiate an object yet not listening.
Definition observable.h:255
bool LoadAllPlugIns(bool enabled_plugins, bool keep_orphans=false)
Update catalog with imported metadata and load all plugin library files.
A popup frame containing a detail slider for chart display.
bool StartServer(const fs::path &certificate_location) override
Start the server thread.
Represents a waypoint or mark within the navigation system.
Definition route_point.h:68
static std::function< void(unsigned, const unsigned *)> delete_gl_textures
Horrible Hack (tm).
Definition route_point.h:50
Represents a navigational route in the navigation system.
Definition route.h:96
EventVar on_routes_update
Notified when list of routes is updated (no data in event)
Definition routeman.h:198
Dialog for displaying query results of S57 objects.
Manager for S57 chart SENC creation threads.
Definition tcmgr.h:86
Window for displaying chart thumbnails.
Definition thumbwin.h:56
Class TrackPropDlg.
Represents a track, which is a series of connected track points.
Definition track.h:79
Listen to hardware events and notifies SystemEvents when new devices are plugged in.
Represents the About dialog for OpenCPN.
Definition about.h:52
OpenGL chart rendering canvas.
Floating toolbar dialog for OpenCPN.
Definition toolbar.h:386
Handles crash reporting in wxWidgets applications.
Definition crashprint.h:35
Global variables reflecting command line options and arguments.
The local API has a server side handling commands and a client part issuing commands.
Enhanced logging interface on top of wx/log.h.
mDNS lookup wrappers.
Start/stop mdns service routines.
void check_last_start()
Check if the last start failed, possibly invoke user dialog and set safe mode state.
void clear_check()
Mark last run as successful.
Definition safe_mode.cpp:39
bool CheckDongleAccess(wxWindow *parent)
Runs checks and if required dialogs to make dongle accessible.
bool CheckSerialAccess(wxWindow *parent, const std::string device)
Run checks and possible dialogs to ensure device is accessible.
Access checks for comm devices and dongle.