OpenCPN Partial API docs
Loading...
Searching...
No Matches
chcanv.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: Chart Canvas
5 * Author: David Register
6 *
7 ***************************************************************************
8 * Copyright (C) 2018 by David S. Register *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25
26// For compilers that support precompilation, includes "wx.h".
27#include <wx/wxprec.h>
28
29#ifndef WX_PRECOMP
30#include <wx/wx.h>
31#endif // precompiled headers
32#include <wx/image.h>
33#include <wx/graphics.h>
34#include <wx/clipbrd.h>
35#include <wx/aui/aui.h>
36
37#include "config.h"
38
39#include <wx/listimpl.cpp>
40
41#include "model/ais_decoder.h"
43#include "model/ais_target_data.h"
44#include "model/cmdline.h"
45#include "model/conn_params.h"
46#include "model/cutil.h"
47#include "model/geodesic.h"
48#include "model/gui.h"
49#include "model/idents.h"
50#include "model/multiplexer.h"
51#include "model/nav_object_database.h"
52#include "model/navutil_base.h"
53#include "model/own_ship.h"
54#include "model/plugin_comm.h"
55#include "model/route.h"
56#include "model/routeman.h"
57#include "model/select.h"
58#include "model/select_item.h"
59#include "model/track.h"
60#include "model/wx28compat.h"
61
62#include "ais.h"
63#include "AISTargetAlertDialog.h"
64#include "CanvasConfig.h"
65#include "canvasMenu.h"
66#include "CanvasOptions.h"
67#include "chartdb.h"
68#include "chartimg.h"
69#include "chcanv.h"
70#include "ChInfoWin.h"
71#include "cm93.h" // for chart outline draw
72#include "compass.h"
73#include "concanv.h"
74#include "displays.h"
75#include "hotkeys_dlg.h"
76#include "FontMgr.h"
77#include "glTextureDescriptor.h"
78#include "gshhs.h"
79#include "iENCToolbar.h"
80#include "kml.h"
81#include "line_clip.h"
82#include "MarkInfo.h"
83#include "mbtiles.h"
84#include "MUIBar.h"
85#include "navutil.h"
86#include "NMEALogWindow.h"
87#include "OCPN_AUIManager.h"
88#include "ocpndc.h"
89#include "ocpn_frame.h"
90#include "ocpn_pixel.h"
91#include "OCPNRegion.h"
92#include "options.h"
93#include "piano.h"
94#include "pluginmanager.h"
95#include "Quilt.h"
96#include "route_gui.h"
97#include "routemanagerdialog.h"
98#include "route_point_gui.h"
99#include "route_validator.h"
100#include "RoutePropDlgImpl.h"
101#include "s52plib.h"
102#include "s52utils.h"
103#include "s57chart.h" // for ArrayOfS57Obj
104#include "SendToGpsDlg.h"
105#include "shapefile_basemap.h"
106#include "styles.h"
107#include "SystemCmdSound.h"
108#include "tcmgr.h"
109#include "TCWin.h"
110#include "thumbwin.h"
111#include "tide_time.h"
112#include "timers.h"
113#include "toolbar.h"
114#include "track_gui.h"
115#include "TrackPropDlg.h"
116#include "undo.h"
117
118#include "s57_ocpn_utils.h"
119
120#ifdef __ANDROID__
121#include "androidUTIL.h"
122#endif
123
124#ifdef ocpnUSE_GL
125#include "glChartCanvas.h"
126#endif
127
128#ifdef __VISUALC__
129#include <wx/msw/msvcrt.h>
130#endif
131
132#ifndef __WXMSW__
133#include <signal.h>
134#include <setjmp.h>
135
136#endif
137
138extern float g_ShipScaleFactorExp;
139extern double g_mouse_zoom_sensitivity;
140
141#include <vector>
142
143#ifdef __WXMSW__
144#define printf printf2
145
146int __cdecl printf2(const char *format, ...) {
147 char str[1024];
148
149 va_list argptr;
150 va_start(argptr, format);
151 int ret = vsnprintf(str, sizeof(str), format, argptr);
152 va_end(argptr);
153 OutputDebugStringA(str);
154 return ret;
155}
156#endif
157
158#if defined(__MSVC__) && (_MSC_VER < 1700)
159#define trunc(d) ((d > 0) ? floor(d) : ceil(d))
160#endif
161
162// Define to enable the invocation of a temporary menubar by pressing the Alt
163// key. Not implemented for Windows XP, as it interferes with Alt-Tab
164// processing.
165#define OCPN_ALT_MENUBAR 1
166
167// Profiling support
168// #include "/usr/include/valgrind/callgrind.h"
169
170// ----------------------------------------------------------------------------
171// Useful Prototypes
172// ----------------------------------------------------------------------------
173extern bool G_FloatPtInPolygon(MyFlPoint *rgpts, int wnumpts, float x, float y);
174extern void catch_signals(int signo);
175
176extern void AlphaBlending(ocpnDC &dc, int x, int y, int size_x, int size_y,
177 float radius, wxColour color,
178 unsigned char transparency);
179
180extern double g_ChartNotRenderScaleFactor;
181extern ChartDB *ChartData;
182extern bool bDBUpdateInProgress;
183extern ColorScheme global_color_scheme;
184extern int g_nbrightness;
185
186extern ConsoleCanvas *console;
187extern OCPNPlatform *g_Platform;
188
189extern RouteList *pRouteList;
190extern std::vector<Track *> g_TrackList;
191extern MyConfig *pConfig;
192extern Routeman *g_pRouteMan;
193extern ThumbWin *pthumbwin;
194extern TCMgr *ptcmgr;
195extern Select *pSelectTC;
196extern MarkInfoDlg *g_pMarkInfoDialog;
197extern RoutePropDlgImpl *pRoutePropDialog;
198extern TrackPropDlg *pTrackPropDialog;
199extern ActiveTrack *g_pActiveTrack;
200
201extern double AnchorPointMinDist;
202extern bool AnchorAlertOn1;
203extern bool AnchorAlertOn2;
204extern int g_nAWMax;
205
206extern RouteManagerDialog *pRouteManagerDialog;
207extern GoToPositionDialog *pGoToPositionDialog;
208extern wxString GetLayerName(int id);
209extern wxString g_uploadConnection;
210extern bool g_bsimplifiedScalebar;
211
212extern bool bDrawCurrentValues;
213
214extern s52plib *ps52plib;
215
216extern bool g_bTempShowMenuBar;
217extern bool g_bShowMenuBar;
218extern bool g_bShowCompassWin;
219
220extern MyFrame *gFrame;
221extern options *g_options;
222
223extern int g_iNavAidRadarRingsNumberVisible;
224extern bool g_bNavAidRadarRingsShown;
225extern float g_fNavAidRadarRingsStep;
226extern int g_pNavAidRadarRingsStepUnits;
227extern bool g_bWayPointPreventDragging;
228extern bool g_bEnableZoomToCursor;
229extern bool g_bShowChartBar;
230extern int g_ENCSoundingScaleFactor;
231extern int g_ENCTextScaleFactor;
232extern int g_maxzoomin;
233
234bool g_bShowShipToActive;
235int g_shipToActiveStyle;
236int g_shipToActiveColor;
237
238extern AISTargetQueryDialog *g_pais_query_dialog_active;
239
240extern int g_S57_dialog_sx, g_S57_dialog_sy;
241
242extern PopUpDSlide *pPopupDetailSlider;
243extern int g_detailslider_dialog_x, g_detailslider_dialog_y;
244
245extern bool g_b_overzoom_x; // Allow high overzoom
246extern double g_plus_minus_zoom_factor;
247
248extern int g_OwnShipIconType;
249extern double g_n_ownship_length_meters;
250extern double g_n_ownship_beam_meters;
251extern double g_n_gps_antenna_offset_y;
252extern double g_n_gps_antenna_offset_x;
253extern int g_n_ownship_min_mm;
254
255extern double g_COGAvg; // only needed for debug....
256
257extern int g_click_stop;
258
259extern double g_ownship_predictor_minutes;
260extern int g_cog_predictor_style;
261extern wxString g_cog_predictor_color;
262extern int g_cog_predictor_endmarker;
263extern int g_ownship_HDTpredictor_style;
264extern wxString g_ownship_HDTpredictor_color;
265extern int g_ownship_HDTpredictor_endmarker;
266extern int g_ownship_HDTpredictor_width;
267extern double g_ownship_HDTpredictor_miles;
268
269extern bool g_bquiting;
270extern AISTargetListDialog *g_pAISTargetList;
271
272extern PlugInManager *g_pi_manager;
273
274extern OCPN_AUIManager *g_pauimgr;
275
276extern bool g_bopengl;
277
278extern bool g_bFullScreenQuilt;
279
280extern bool g_bsmoothpanzoom;
281extern bool g_bSmoothRecenter;
282
283bool g_bDebugOGL;
284
285extern bool g_b_assume_azerty;
286
287extern ChartGroupArray *g_pGroupArray;
288
289extern S57QueryDialog *g_pObjectQueryDialog;
290extern ocpnStyle::StyleManager *g_StyleManager;
291
292extern OcpnSound *g_anchorwatch_sound;
293
294extern bool g_bresponsive;
295extern int g_chart_zoom_modifier_raster;
296extern int g_chart_zoom_modifier_vector;
297extern int g_ChartScaleFactor;
298
299#ifdef ocpnUSE_GL
300#endif
301
302extern double g_gl_ms_per_frame;
303extern bool g_benable_rotate;
304extern bool g_bRollover;
305
306extern bool g_bSpaceDropMark;
307extern bool g_bAutoHideToolbar;
308extern int g_nAutoHideToolbar;
309extern bool g_bDeferredInitDone;
310
311extern wxString g_CmdSoundString;
312ShapeBaseChartSet gShapeBasemap;
313
314// TODO why are these static?
315
325static int mouse_x;
335static int mouse_y;
336static bool mouse_leftisdown;
337
338bool g_brouteCreating;
339
340bool g_bShowTrackPointTime;
341
342int r_gamma_mult;
343int g_gamma_mult;
344int b_gamma_mult;
345int gamma_state;
346bool g_brightness_init;
347int last_brightness;
348
349int g_cog_predictor_width;
350extern double g_display_size_mm;
351
352extern ocpnFloatingToolbarDialog *g_MainToolbar;
353extern iENCToolbar *g_iENCToolbar;
354extern wxColour g_colourOwnshipRangeRingsColour;
355
356// LIVE ETA OPTION
357bool g_bShowLiveETA;
358extern double g_defaultBoatSpeed;
359double g_defaultBoatSpeedUserUnit;
360
361extern int g_nAIS_activity_timer;
362extern bool g_bskew_comp;
363extern float g_compass_scalefactor;
364extern int g_COGAvgSec; // COG average period (sec.) for Course Up Mode
365extern bool g_btenhertz;
366
367wxGLContext *g_pGLcontext; // shared common context
368
369extern bool g_useMUI;
370extern unsigned int g_canvasConfig;
371
372extern ChartCanvas *g_focusCanvas;
373extern ChartCanvas *g_overlayCanvas;
374
375extern float g_toolbar_scalefactor;
376extern SENCThreadManager *g_SencThreadManager;
377
378wxString g_ObjQFileExt;
379
380// "Curtain" mode parameters
381wxDialog *g_pcurtain;
382
383extern int g_GUIScaleFactor;
384// Win DPI scale factor
385double g_scaler;
386wxString g_lastS52PLIBPluginMessage;
387extern bool g_bChartBarEx;
388bool g_PrintingInProgress;
389
390#define MIN_BRIGHT 10
391#define MAX_BRIGHT 100
392
393//------------------------------------------------------------------------------
394// ChartCanvas Implementation
395//------------------------------------------------------------------------------
396BEGIN_EVENT_TABLE(ChartCanvas, wxWindow)
397EVT_PAINT(ChartCanvas::OnPaint)
398EVT_ACTIVATE(ChartCanvas::OnActivate)
399EVT_SIZE(ChartCanvas::OnSize)
400#ifndef HAVE_WX_GESTURE_EVENTS
401EVT_MOUSE_EVENTS(ChartCanvas::MouseEvent)
402#endif
403EVT_TIMER(DBLCLICK_TIMER, ChartCanvas::MouseTimedEvent)
404EVT_TIMER(PAN_TIMER, ChartCanvas::PanTimerEvent)
405EVT_TIMER(MOVEMENT_TIMER, ChartCanvas::MovementTimerEvent)
406EVT_TIMER(MOVEMENT_STOP_TIMER, ChartCanvas::MovementStopTimerEvent)
407EVT_TIMER(CURTRACK_TIMER, ChartCanvas::OnCursorTrackTimerEvent)
408EVT_TIMER(ROT_TIMER, ChartCanvas::RotateTimerEvent)
409EVT_TIMER(ROPOPUP_TIMER, ChartCanvas::OnRolloverPopupTimerEvent)
410EVT_TIMER(ROUTEFINISH_TIMER, ChartCanvas::OnRouteFinishTimerEvent)
411EVT_KEY_DOWN(ChartCanvas::OnKeyDown)
412EVT_KEY_UP(ChartCanvas::OnKeyUp)
413EVT_CHAR(ChartCanvas::OnKeyChar)
414EVT_MOUSE_CAPTURE_LOST(ChartCanvas::LostMouseCapture)
415EVT_KILL_FOCUS(ChartCanvas::OnKillFocus)
416EVT_SET_FOCUS(ChartCanvas::OnSetFocus)
417EVT_MENU(-1, ChartCanvas::OnToolLeftClick)
418EVT_TIMER(DEFERRED_FOCUS_TIMER, ChartCanvas::OnDeferredFocusTimerEvent)
419EVT_TIMER(MOVEMENT_VP_TIMER, ChartCanvas::MovementVPTimerEvent)
420EVT_TIMER(DRAG_INERTIA_TIMER, ChartCanvas::OnChartDragInertiaTimer)
421EVT_TIMER(JUMP_EASE_TIMER, ChartCanvas::OnJumpEaseTimer)
422
423END_EVENT_TABLE()
424
425// Define a constructor for my canvas
426ChartCanvas::ChartCanvas(wxFrame *frame, int canvasIndex)
427 : wxWindow(frame, wxID_ANY, wxPoint(20, 20), wxSize(5, 5), wxNO_BORDER) {
428 parent_frame = (MyFrame *)frame; // save a pointer to parent
429 m_canvasIndex = canvasIndex;
430
431 pscratch_bm = NULL;
432
433 SetBackgroundColour(wxColour(0, 0, 0));
434 SetBackgroundStyle(wxBG_STYLE_CUSTOM); // on WXMSW, this prevents flashing on
435 // color scheme change
436
437 m_groupIndex = 0;
438 m_bDrawingRoute = false;
439 m_bRouteEditing = false;
440 m_bMarkEditing = false;
441 m_bRoutePoinDragging = false;
442 m_bIsInRadius = false;
443 m_bMayToggleMenuBar = true;
444
445 m_bFollow = false;
446 m_bShowNavobjects = true;
447 m_bTCupdate = false;
448 m_bAppendingRoute = false; // was true in MSW, why??
449 pThumbDIBShow = NULL;
450 m_bShowCurrent = false;
451 m_bShowTide = false;
452 bShowingCurrent = false;
453 pCwin = NULL;
454 warp_flag = false;
455 m_bzooming = false;
456 m_b_paint_enable = true;
457 m_routeState = 0;
458
459 pss_overlay_bmp = NULL;
460 pss_overlay_mask = NULL;
461 m_bChartDragging = false;
462 m_bMeasure_Active = false;
463 m_bMeasure_DistCircle = false;
464 m_pMeasureRoute = NULL;
465 m_pTrackRolloverWin = NULL;
466 m_pRouteRolloverWin = NULL;
467 m_pAISRolloverWin = NULL;
468 m_bedge_pan = false;
469 m_disable_edge_pan = false;
470 m_dragoffsetSet = false;
471 m_bautofind = false;
472 m_bFirstAuto = true;
473 m_groupIndex = 0;
474 m_singleChart = NULL;
475 m_upMode = NORTH_UP_MODE;
476 m_bShowAIS = true;
477 m_bShowAISScaled = false;
478 m_timed_move_vp_active = false;
479
480 m_vLat = 0.;
481 m_vLon = 0.;
482
483 m_pCIWin = NULL;
484
485 m_pSelectedRoute = NULL;
486 m_pSelectedTrack = NULL;
487 m_pRoutePointEditTarget = NULL;
488 m_pFoundPoint = NULL;
489 m_pMouseRoute = NULL;
490 m_prev_pMousePoint = NULL;
491 m_pEditRouteArray = NULL;
492 m_pFoundRoutePoint = NULL;
493 m_FinishRouteOnKillFocus = true;
494
495 m_pRolloverRouteSeg = NULL;
496 m_pRolloverTrackSeg = NULL;
497 m_bsectors_shown = false;
498
499 m_bbrightdir = false;
500 r_gamma_mult = 1;
501 g_gamma_mult = 1;
502 b_gamma_mult = 1;
503
504 m_pos_image_user_day = NULL;
505 m_pos_image_user_dusk = NULL;
506 m_pos_image_user_night = NULL;
507 m_pos_image_user_grey_day = NULL;
508 m_pos_image_user_grey_dusk = NULL;
509 m_pos_image_user_grey_night = NULL;
510
511 m_zoom_factor = 1;
512 m_rotation_speed = 0;
513 m_mustmove = 0;
514
515 m_OSoffsetx = 0.;
516 m_OSoffsety = 0.;
517
518 m_pos_image_user_yellow_day = NULL;
519 m_pos_image_user_yellow_dusk = NULL;
520 m_pos_image_user_yellow_night = NULL;
521
522 SetOwnShipState(SHIP_INVALID);
523
524 undo = new Undo(this);
525
526 VPoint.Invalidate();
527
528 m_glcc = NULL;
529
530 m_focus_indicator_pix = 1;
531
532 m_pCurrentStack = NULL;
533 m_bpersistent_quilt = false;
534 m_piano_ctx_menu = NULL;
535 m_Compass = NULL;
536
537 g_ChartNotRenderScaleFactor = 2.0;
538 m_bShowScaleInStatusBar = true;
539
540 m_muiBar = NULL;
541 m_bShowScaleInStatusBar = false;
542 m_show_focus_bar = true;
543
544 m_bShowOutlines = false;
545 m_bDisplayGrid = false;
546 m_bShowDepthUnits = true;
547 m_encDisplayCategory = (int)STANDARD;
548
549 m_encShowLights = true;
550 m_encShowAnchor = true;
551 m_encShowDataQual = false;
552 m_bShowGPS = true;
553 m_pQuilt = new Quilt(this);
554 SetQuiltMode(true);
555 SetAlertString(_T(""));
556 m_sector_glat = 0;
557 m_sector_glon = 0;
558 g_PrintingInProgress = false;
559
560#ifdef HAVE_WX_GESTURE_EVENTS
561 m_oldVPSScale = -1.0;
562 m_popupWanted = false;
563 m_leftdown = false;
564#endif /* HAVE_WX_GESTURE_EVENTS */
565
566 SetupGlCanvas();
567
568 singleClickEventIsValid = false;
569
570 // Build the cursors
571
572 pCursorLeft = NULL;
573 pCursorRight = NULL;
574 pCursorUp = NULL;
575 pCursorDown = NULL;
576 pCursorArrow = NULL;
577 pCursorPencil = NULL;
578 pCursorCross = NULL;
579
580 RebuildCursors();
581
582 SetCursor(*pCursorArrow);
583
584 pPanTimer = new wxTimer(this, m_MouseDragging);
585 pPanTimer->Stop();
586
587 pMovementTimer = new wxTimer(this, MOVEMENT_TIMER);
588 pMovementTimer->Stop();
589
590 pMovementStopTimer = new wxTimer(this, MOVEMENT_STOP_TIMER);
591 pMovementStopTimer->Stop();
592
593 pRotDefTimer = new wxTimer(this, ROT_TIMER);
594 pRotDefTimer->Stop();
595
596 m_DoubleClickTimer = new wxTimer(this, DBLCLICK_TIMER);
597 m_DoubleClickTimer->Stop();
598
599 m_VPMovementTimer.SetOwner(this, MOVEMENT_VP_TIMER);
600 m_chart_drag_inertia_timer.SetOwner(this, DRAG_INERTIA_TIMER);
601 m_chart_drag_inertia_active = false;
602
603 m_easeTimer.SetOwner(this, JUMP_EASE_TIMER);
604 m_animationActive = false;
605
606 m_panx = m_pany = 0;
607 m_panspeed = 0;
608 m_panx_target_final = m_pany_target_final = 0;
609 m_panx_target_now = m_pany_target_now = 0;
610
611 pCurTrackTimer = new wxTimer(this, CURTRACK_TIMER);
612 pCurTrackTimer->Stop();
613 m_curtrack_timer_msec = 10;
614
615 m_wheelzoom_stop_oneshot = 0;
616 m_last_wheel_dir = 0;
617
618 m_RolloverPopupTimer.SetOwner(this, ROPOPUP_TIMER);
619
620 m_deferredFocusTimer.SetOwner(this, DEFERRED_FOCUS_TIMER);
621
622 m_rollover_popup_timer_msec = 20;
623
624 m_routeFinishTimer.SetOwner(this, ROUTEFINISH_TIMER);
625
626 m_b_rot_hidef = true;
627
628 proute_bm = NULL;
629 m_prot_bm = NULL;
630
631 m_upMode = NORTH_UP_MODE;
632 m_bLookAhead = false;
633
634 // Set some benign initial values
635
636 m_cs = GLOBAL_COLOR_SCHEME_DAY;
637 VPoint.clat = 0;
638 VPoint.clon = 0;
639 VPoint.view_scale_ppm = 1;
640 VPoint.Invalidate();
641 m_nMeasureState = 0;
642
643 m_canvas_scale_factor = 1.;
644
645 m_canvas_width = 1000;
646
647 m_overzoomTextWidth = 0;
648 m_overzoomTextHeight = 0;
649
650 // Create the default world chart
651 pWorldBackgroundChart = new GSHHSChart;
652 gShapeBasemap.Reset();
653
654 // Create the default depth unit emboss maps
655 m_pEM_Feet = NULL;
656 m_pEM_Meters = NULL;
657 m_pEM_Fathoms = NULL;
658
659 CreateDepthUnitEmbossMaps(GLOBAL_COLOR_SCHEME_DAY);
660
661 m_pEM_OverZoom = NULL;
662 SetOverzoomFont();
663 CreateOZEmbossMapData(GLOBAL_COLOR_SCHEME_DAY);
664
665 // Build icons for tide/current points
666 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
667 m_bmTideDay = style->GetIconScaled(_T("tidesml"),
668 1. / g_Platform->GetDisplayDIPMult(this));
669
670 // Dusk
671 m_bmTideDusk = CreateDimBitmap(m_bmTideDay, .50);
672
673 // Night
674 m_bmTideNight = CreateDimBitmap(m_bmTideDay, .20);
675
676 // Build Dusk/Night ownship icons
677 double factor_dusk = 0.5;
678 double factor_night = 0.25;
679
680 // Red
681 m_os_image_red_day = style->GetIcon(_T("ship-red")).ConvertToImage();
682
683 int rimg_width = m_os_image_red_day.GetWidth();
684 int rimg_height = m_os_image_red_day.GetHeight();
685
686 m_os_image_red_dusk = m_os_image_red_day.Copy();
687 m_os_image_red_night = m_os_image_red_day.Copy();
688
689 for (int iy = 0; iy < rimg_height; iy++) {
690 for (int ix = 0; ix < rimg_width; ix++) {
691 if (!m_os_image_red_day.IsTransparent(ix, iy)) {
692 wxImage::RGBValue rgb(m_os_image_red_day.GetRed(ix, iy),
693 m_os_image_red_day.GetGreen(ix, iy),
694 m_os_image_red_day.GetBlue(ix, iy));
695 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
696 hsv.value = hsv.value * factor_dusk;
697 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
698 m_os_image_red_dusk.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
699
700 hsv = wxImage::RGBtoHSV(rgb);
701 hsv.value = hsv.value * factor_night;
702 nrgb = wxImage::HSVtoRGB(hsv);
703 m_os_image_red_night.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
704 }
705 }
706 }
707
708 // Grey
709 m_os_image_grey_day =
710 style->GetIcon(_T("ship-red")).ConvertToImage().ConvertToGreyscale();
711
712 int gimg_width = m_os_image_grey_day.GetWidth();
713 int gimg_height = m_os_image_grey_day.GetHeight();
714
715 m_os_image_grey_dusk = m_os_image_grey_day.Copy();
716 m_os_image_grey_night = m_os_image_grey_day.Copy();
717
718 for (int iy = 0; iy < gimg_height; iy++) {
719 for (int ix = 0; ix < gimg_width; ix++) {
720 if (!m_os_image_grey_day.IsTransparent(ix, iy)) {
721 wxImage::RGBValue rgb(m_os_image_grey_day.GetRed(ix, iy),
722 m_os_image_grey_day.GetGreen(ix, iy),
723 m_os_image_grey_day.GetBlue(ix, iy));
724 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
725 hsv.value = hsv.value * factor_dusk;
726 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
727 m_os_image_grey_dusk.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
728
729 hsv = wxImage::RGBtoHSV(rgb);
730 hsv.value = hsv.value * factor_night;
731 nrgb = wxImage::HSVtoRGB(hsv);
732 m_os_image_grey_night.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
733 }
734 }
735 }
736
737 // Yellow
738 m_os_image_yellow_day = m_os_image_red_day.Copy();
739
740 gimg_width = m_os_image_yellow_day.GetWidth();
741 gimg_height = m_os_image_yellow_day.GetHeight();
742
743 m_os_image_yellow_dusk = m_os_image_red_day.Copy();
744 m_os_image_yellow_night = m_os_image_red_day.Copy();
745
746 for (int iy = 0; iy < gimg_height; iy++) {
747 for (int ix = 0; ix < gimg_width; ix++) {
748 if (!m_os_image_yellow_day.IsTransparent(ix, iy)) {
749 wxImage::RGBValue rgb(m_os_image_yellow_day.GetRed(ix, iy),
750 m_os_image_yellow_day.GetGreen(ix, iy),
751 m_os_image_yellow_day.GetBlue(ix, iy));
752 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
753 hsv.hue += 60. / 360.; // shift to yellow
754 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
755 m_os_image_yellow_day.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
756
757 hsv = wxImage::RGBtoHSV(rgb);
758 hsv.value = hsv.value * factor_dusk;
759 hsv.hue += 60. / 360.; // shift to yellow
760 nrgb = wxImage::HSVtoRGB(hsv);
761 m_os_image_yellow_dusk.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
762
763 hsv = wxImage::RGBtoHSV(rgb);
764 hsv.hue += 60. / 360.; // shift to yellow
765 hsv.value = hsv.value * factor_night;
766 nrgb = wxImage::HSVtoRGB(hsv);
767 m_os_image_yellow_night.SetRGB(ix, iy, nrgb.red, nrgb.green, nrgb.blue);
768 }
769 }
770 }
771
772 // Set initial pointers to ownship images
773 m_pos_image_red = &m_os_image_red_day;
774 m_pos_image_yellow = &m_os_image_yellow_day;
775 m_pos_image_grey = &m_os_image_grey_day;
776
777 SetUserOwnship();
778
779 m_pBrightPopup = NULL;
780
781#ifdef ocpnUSE_GL
782 if (!g_bdisable_opengl) m_pQuilt->EnableHighDefinitionZoom(true);
783#endif
784
785 int gridFontSize = 8;
786#if defined(__WXOSX__) || defined(__WXGTK3__)
787 // Support scaled HDPI displays.
788 gridFontSize *= GetContentScaleFactor();
789#endif
790
791 m_pgridFont = FontMgr::Get().FindOrCreateFont(
792 gridFontSize, wxFONTFAMILY_SWISS, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL,
793 FALSE, wxString(_T ( "Arial" )));
794
795 m_Piano = new Piano(this);
796
797 m_bShowCompassWin = true;
798
799 m_Compass = new ocpnCompass(this);
800 m_Compass->SetScaleFactor(g_compass_scalefactor);
801 m_Compass->Show(m_bShowCompassWin && g_bShowCompassWin);
802
803 m_pianoFrozen = false;
804
805 SetMinSize(wxSize(200, 200));
806
807 m_displayScale = 1.0;
808#if defined(__WXOSX__) || defined(__WXGTK3__)
809 // Support scaled HDPI displays.
810 m_displayScale = GetContentScaleFactor();
811#endif
812 VPoint.SetPixelScale(m_displayScale);
813
814#ifdef HAVE_WX_GESTURE_EVENTS
815 // if (!m_glcc)
816 {
817 if (!EnableTouchEvents(wxTOUCH_ZOOM_GESTURE | wxTOUCH_PRESS_GESTURES)) {
818 wxLogError("Failed to enable touch events");
819 }
820
821 // Bind(wxEVT_GESTURE_ZOOM, &ChartCanvas::OnZoom, this);
822
823 Bind(wxEVT_LONG_PRESS, &ChartCanvas::OnLongPress, this);
824 Bind(wxEVT_PRESS_AND_TAP, &ChartCanvas::OnPressAndTap, this);
825
826 Bind(wxEVT_RIGHT_UP, &ChartCanvas::OnRightUp, this);
827 Bind(wxEVT_RIGHT_DOWN, &ChartCanvas::OnRightDown, this);
828
829 Bind(wxEVT_LEFT_UP, &ChartCanvas::OnLeftUp, this);
830 Bind(wxEVT_LEFT_DOWN, &ChartCanvas::OnLeftDown, this);
831
832 Bind(wxEVT_MOUSEWHEEL, &ChartCanvas::OnWheel, this);
833 Bind(wxEVT_MOTION, &ChartCanvas::OnMotion, this);
834 }
835#endif
836}
837
838ChartCanvas::~ChartCanvas() {
839 delete pThumbDIBShow;
840
841 // Delete Cursors
842 delete pCursorLeft;
843 delete pCursorRight;
844 delete pCursorUp;
845 delete pCursorDown;
846 delete pCursorArrow;
847 delete pCursorPencil;
848 delete pCursorCross;
849
850 delete pPanTimer;
851 delete pMovementTimer;
852 delete pMovementStopTimer;
853 delete pCurTrackTimer;
854 delete pRotDefTimer;
855 delete m_DoubleClickTimer;
856
857 delete m_pTrackRolloverWin;
858 delete m_pRouteRolloverWin;
859 delete m_pAISRolloverWin;
860 delete m_pBrightPopup;
861
862 delete m_pCIWin;
863
864 delete pscratch_bm;
865
866 m_dc_route.SelectObject(wxNullBitmap);
867 delete proute_bm;
868
869 delete pWorldBackgroundChart;
870 delete pss_overlay_bmp;
871
872 delete m_pEM_Feet;
873 delete m_pEM_Meters;
874 delete m_pEM_Fathoms;
875
876 delete m_pEM_OverZoom;
877 // delete m_pEM_CM93Offset;
878
879 delete m_prot_bm;
880
881 delete m_pos_image_user_day;
882 delete m_pos_image_user_dusk;
883 delete m_pos_image_user_night;
884 delete m_pos_image_user_grey_day;
885 delete m_pos_image_user_grey_dusk;
886 delete m_pos_image_user_grey_night;
887 delete m_pos_image_user_yellow_day;
888 delete m_pos_image_user_yellow_dusk;
889 delete m_pos_image_user_yellow_night;
890
891 delete undo;
892#ifdef ocpnUSE_GL
893 if (!g_bdisable_opengl) {
894 delete m_glcc;
895
896#if wxCHECK_VERSION(2, 9, 0)
897 if (IsPrimaryCanvas() && g_bopengl) delete g_pGLcontext;
898#endif
899 }
900#endif
901
902 // Delete the MUI bar, but make sure there is no pointer to it during destroy.
903 // wx tries to deliver events to this canvas during destroy.
904 MUIBar *muiBar = m_muiBar;
905 m_muiBar = 0;
906 delete muiBar;
907 delete m_pQuilt;
908 delete m_pCurrentStack;
909 delete m_Compass;
910 delete m_Piano;
911}
912
913void ChartCanvas::RebuildCursors() {
914 delete pCursorLeft;
915 delete pCursorRight;
916 delete pCursorUp;
917 delete pCursorDown;
918 delete pCursorArrow;
919 delete pCursorPencil;
920 delete pCursorCross;
921
922 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
923 double cursorScale = exp(g_GUIScaleFactor * (0.693 / 5.0));
924
925 double pencilScale = 1.0 / g_Platform->GetDisplayDIPMult(gFrame);
926
927 wxImage ICursorLeft = style->GetIcon(_T("left")).ConvertToImage();
928 wxImage ICursorRight = style->GetIcon(_T("right")).ConvertToImage();
929 wxImage ICursorUp = style->GetIcon(_T("up")).ConvertToImage();
930 wxImage ICursorDown = style->GetIcon(_T("down")).ConvertToImage();
931 wxImage ICursorPencil =
932 style->GetIconScaled(_T("pencil"), pencilScale).ConvertToImage();
933 wxImage ICursorCross = style->GetIcon(_T("cross")).ConvertToImage();
934
935#if !defined(__WXMSW__) && !defined(__WXQT__)
936 ICursorLeft.ConvertAlphaToMask(128);
937 ICursorRight.ConvertAlphaToMask(128);
938 ICursorUp.ConvertAlphaToMask(128);
939 ICursorDown.ConvertAlphaToMask(128);
940 ICursorPencil.ConvertAlphaToMask(10);
941 ICursorCross.ConvertAlphaToMask(10);
942#endif
943
944 if (ICursorLeft.Ok()) {
945 ICursorLeft.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 0);
946 ICursorLeft.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 15);
947 pCursorLeft = new wxCursor(ICursorLeft);
948 } else
949 pCursorLeft = new wxCursor(wxCURSOR_ARROW);
950
951 if (ICursorRight.Ok()) {
952 ICursorRight.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 31);
953 ICursorRight.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 15);
954 pCursorRight = new wxCursor(ICursorRight);
955 } else
956 pCursorRight = new wxCursor(wxCURSOR_ARROW);
957
958 if (ICursorUp.Ok()) {
959 ICursorUp.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 15);
960 ICursorUp.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 0);
961 pCursorUp = new wxCursor(ICursorUp);
962 } else
963 pCursorUp = new wxCursor(wxCURSOR_ARROW);
964
965 if (ICursorDown.Ok()) {
966 ICursorDown.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 15);
967 ICursorDown.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 31);
968 pCursorDown = new wxCursor(ICursorDown);
969 } else
970 pCursorDown = new wxCursor(wxCURSOR_ARROW);
971
972 if (ICursorPencil.Ok()) {
973 ICursorPencil.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 0 * pencilScale);
974 ICursorPencil.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 16 * pencilScale);
975 pCursorPencil = new wxCursor(ICursorPencil);
976 } else
977 pCursorPencil = new wxCursor(wxCURSOR_ARROW);
978
979 if (ICursorCross.Ok()) {
980 ICursorCross.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_X, 13);
981 ICursorCross.SetOption(wxIMAGE_OPTION_CUR_HOTSPOT_Y, 12);
982 pCursorCross = new wxCursor(ICursorCross);
983 } else
984 pCursorCross = new wxCursor(wxCURSOR_ARROW);
985
986 pCursorArrow = new wxCursor(wxCURSOR_ARROW);
987 pPlugIn_Cursor = NULL;
988}
989
990void ChartCanvas::CanvasApplyLocale() {
991 CreateDepthUnitEmbossMaps(m_cs);
992 CreateOZEmbossMapData(m_cs);
993}
994
995void ChartCanvas::SetupGlCanvas() {
996#ifndef __ANDROID__
997#ifdef ocpnUSE_GL
998 if (!g_bdisable_opengl) {
999 if (g_bopengl) {
1000 wxLogMessage(_T("Creating glChartCanvas"));
1001 m_glcc = new glChartCanvas(this);
1002
1003 // We use one context for all GL windows, so that textures etc will be
1004 // automatically shared
1005 if (IsPrimaryCanvas()) {
1006 // qDebug() << "Creating Primary Context";
1007
1008 // wxGLContextAttrs ctxAttr;
1009 // ctxAttr.PlatformDefaults().CoreProfile().OGLVersion(3,
1010 // 2).EndList(); wxGLContext *pctx = new wxGLContext(m_glcc,
1011 // NULL, &ctxAttr);
1012 wxGLContext *pctx = new wxGLContext(m_glcc);
1013 m_glcc->SetContext(pctx);
1014 g_pGLcontext = pctx; // Save a copy of the common context
1015 } else {
1016#ifdef __WXOSX__
1017 m_glcc->SetContext(new wxGLContext(m_glcc, g_pGLcontext));
1018#else
1019 m_glcc->SetContext(g_pGLcontext); // If not primary canvas, use the
1020 // saved common context
1021#endif
1022 }
1023 }
1024 }
1025#endif
1026#endif
1027
1028#ifdef __ANDROID__ // ocpnUSE_GL
1029 if (!g_bdisable_opengl) {
1030 if (g_bopengl) {
1031 // qDebug() << "SetupGlCanvas";
1032 wxLogMessage(_T("Creating glChartCanvas"));
1033
1034 // We use one context for all GL windows, so that textures etc will be
1035 // automatically shared
1036 if (IsPrimaryCanvas()) {
1037 qDebug() << "Creating Primary glChartCanvas";
1038
1039 // wxGLContextAttrs ctxAttr;
1040 // ctxAttr.PlatformDefaults().CoreProfile().OGLVersion(3,
1041 // 2).EndList(); wxGLContext *pctx = new wxGLContext(m_glcc,
1042 // NULL, &ctxAttr);
1043 m_glcc = new glChartCanvas(this);
1044
1045 wxGLContext *pctx = new wxGLContext(m_glcc);
1046 m_glcc->SetContext(pctx);
1047 g_pGLcontext = pctx; // Save a copy of the common context
1048 m_glcc->m_pParentCanvas = this;
1049 // m_glcc->Reparent(this);
1050 } else {
1051 qDebug() << "Creating Secondary glChartCanvas";
1052 // QGLContext *pctx =
1053 // gFrame->GetPrimaryCanvas()->GetglCanvas()->GetQGLContext(); qDebug()
1054 // << "pctx: " << pctx;
1055
1056 m_glcc = new glChartCanvas(
1057 gFrame, gFrame->GetPrimaryCanvas()->GetglCanvas()); // Shared
1058 // m_glcc = new glChartCanvas(this, pctx); //Shared
1059 // m_glcc = new glChartCanvas(this, wxPoint(900, 0));
1060 wxGLContext *pwxctx = new wxGLContext(m_glcc);
1061 m_glcc->SetContext(pwxctx);
1062 m_glcc->m_pParentCanvas = this;
1063 // m_glcc->Reparent(this);
1064 }
1065 }
1066 }
1067#endif
1068}
1069
1070void ChartCanvas::OnKillFocus(wxFocusEvent &WXUNUSED(event)) {
1071 RefreshRect(wxRect(0, 0, GetClientSize().x, m_focus_indicator_pix), false);
1072
1073 // On Android, we get a KillFocus on just about every keystroke.
1074 // Why?
1075#ifdef __ANDROID__
1076 return;
1077#endif
1078
1079 // Special logic:
1080 // On OSX in GL mode, each mouse click causes a kill and immediate regain of
1081 // canvas focus. Why??? Who knows... So, we provide for this case by
1082 // starting a timer if required to actually Finish() a route on a legitimate
1083 // focus change, but not if the focus is quickly regained ( <20 msec.) on
1084 // this canvas.
1085#ifdef __WXOSX__
1086 if (m_routeState && m_FinishRouteOnKillFocus)
1087 m_routeFinishTimer.Start(20, wxTIMER_ONE_SHOT);
1088#else
1089 if (m_routeState && m_FinishRouteOnKillFocus) FinishRoute();
1090#endif
1091}
1092
1093void ChartCanvas::OnSetFocus(wxFocusEvent &WXUNUSED(event)) {
1094 m_routeFinishTimer.Stop();
1095
1096 // Try to keep the global top-line menubar selections up to date with the
1097 // current "focus" canvas
1098 gFrame->UpdateGlobalMenuItems(this);
1099
1100 RefreshRect(wxRect(0, 0, GetClientSize().x, m_focus_indicator_pix), false);
1101}
1102
1103void ChartCanvas::OnRouteFinishTimerEvent(wxTimerEvent &event) {
1104 if (m_routeState && m_FinishRouteOnKillFocus) FinishRoute();
1105}
1106
1107#ifdef HAVE_WX_GESTURE_EVENTS
1108void ChartCanvas::OnLongPress(wxLongPressEvent &event) {
1109 /* we defer the popup menu call upon the leftup event
1110 else the menu disappears immediately,
1111 (see
1112 http://wxwidgets.10942.n7.nabble.com/Popupmenu-disappears-immediately-if-called-from-QueueEvent-td92572.html)
1113 */
1114 m_popupWanted = true;
1115}
1116
1117void ChartCanvas::OnPressAndTap(wxPressAndTapEvent &event) {
1118 // not implemented yet
1119}
1120
1121void ChartCanvas::OnRightUp(wxMouseEvent &event) { MouseEvent(event); }
1122
1123void ChartCanvas::OnRightDown(wxMouseEvent &event) { MouseEvent(event); }
1124
1125void ChartCanvas::OnLeftUp(wxMouseEvent &event) {
1126 wxPoint pos = event.GetPosition();
1127
1128 m_leftdown = false;
1129
1130 if (!m_popupWanted) {
1131 wxMouseEvent ev(wxEVT_LEFT_UP);
1132 ev.m_x = pos.x;
1133 ev.m_y = pos.y;
1134 MouseEvent(ev);
1135 return;
1136 }
1137
1138 m_popupWanted = false;
1139
1140 wxMouseEvent ev(wxEVT_RIGHT_DOWN);
1141 ev.m_x = pos.x;
1142 ev.m_y = pos.y;
1143
1144 MouseEvent(ev);
1145}
1146
1147void ChartCanvas::OnLeftDown(wxMouseEvent &event) {
1148 m_leftdown = true;
1149
1150 wxPoint pos = event.GetPosition();
1151 MouseEvent(event);
1152}
1153
1154void ChartCanvas::OnMotion(wxMouseEvent &event) {
1155 /* This is a workaround, to the fact that on touchscreen, OnMotion comes with
1156 dragging, upon simple click, and without the OnLeftDown event before Thus,
1157 this consists in skiping it, and setting the leftdown bit according to a
1158 status that we trust */
1159 event.m_leftDown = m_leftdown;
1160 MouseEvent(event);
1161}
1162
1163void ChartCanvas::OnZoom(wxZoomGestureEvent &event) {
1164 /* there are spurious end zoom events upon right-click */
1165 if (event.IsGestureEnd()) return;
1166
1167 double factor = event.GetZoomFactor();
1168
1169 if (event.IsGestureStart() || m_oldVPSScale < 0) {
1170 m_oldVPSScale = GetVPScale();
1171 }
1172
1173 double current_vps = GetVPScale();
1174 double wanted_factor = m_oldVPSScale / current_vps * factor;
1175
1176 ZoomCanvas(wanted_factor, true, false);
1177
1178 // Allow combined zoom/pan operation
1179 if (event.IsGestureStart()) {
1180 m_zoomStartPoint = event.GetPosition();
1181 } else {
1182 wxPoint delta = event.GetPosition() - m_zoomStartPoint;
1183 PanCanvas(-delta.x, -delta.y);
1184 m_zoomStartPoint = event.GetPosition();
1185 }
1186}
1187
1188void ChartCanvas::OnWheel(wxMouseEvent &event) { MouseEvent(event); }
1189
1190void ChartCanvas::OnDoubleLeftClick(wxMouseEvent &event) {
1191 DoRotateCanvas(0.0);
1192}
1193#endif /* HAVE_WX_GESTURE_EVENTS */
1194
1195void ChartCanvas::ApplyCanvasConfig(canvasConfig *pcc) {
1196 SetViewPoint(pcc->iLat, pcc->iLon, pcc->iScale, 0., pcc->iRotation);
1197 m_vLat = pcc->iLat;
1198 m_vLon = pcc->iLon;
1199
1200 m_restore_dbindex = pcc->DBindex;
1201 m_bFollow = pcc->bFollow;
1202 if (pcc->GroupID < 0) pcc->GroupID = 0;
1203
1204 if (pcc->GroupID > (int)g_pGroupArray->GetCount())
1205 m_groupIndex = 0;
1206 else
1207 m_groupIndex = pcc->GroupID;
1208
1209 if (pcc->bQuilt != GetQuiltMode()) ToggleCanvasQuiltMode();
1210
1211 ShowTides(pcc->bShowTides);
1212 ShowCurrents(pcc->bShowCurrents);
1213
1214 SetShowDepthUnits(pcc->bShowDepthUnits);
1215 SetShowGrid(pcc->bShowGrid);
1216 SetShowOutlines(pcc->bShowOutlines);
1217
1218 SetShowAIS(pcc->bShowAIS);
1219 SetAttenAIS(pcc->bAttenAIS);
1220
1221 // ENC options
1222 SetShowENCText(pcc->bShowENCText);
1223 m_encDisplayCategory = pcc->nENCDisplayCategory;
1224 m_encShowDepth = pcc->bShowENCDepths;
1225 m_encShowLightDesc = pcc->bShowENCLightDescriptions;
1226 m_encShowBuoyLabels = pcc->bShowENCBuoyLabels;
1227 m_encShowLights = pcc->bShowENCLights;
1228 m_bShowVisibleSectors = pcc->bShowENCVisibleSectorLights;
1229 m_encShowAnchor = pcc->bShowENCAnchorInfo;
1230 m_encShowDataQual = pcc->bShowENCDataQuality;
1231
1232 bool courseUp = pcc->bCourseUp;
1233 bool headUp = pcc->bHeadUp;
1234 m_upMode = NORTH_UP_MODE;
1235 if (courseUp)
1236 m_upMode = COURSE_UP_MODE;
1237 else if (headUp)
1238 m_upMode = HEAD_UP_MODE;
1239
1240 m_bLookAhead = pcc->bLookahead;
1241
1242 m_singleChart = NULL;
1243}
1244
1245void ChartCanvas::ApplyGlobalSettings() {
1246 // GPS compas window
1247 if (m_Compass) {
1248 m_Compass->Show(m_bShowCompassWin && g_bShowCompassWin);
1249 if (m_bShowCompassWin && g_bShowCompassWin) m_Compass->UpdateStatus();
1250 }
1251}
1252
1253void ChartCanvas::CheckGroupValid(bool showMessage, bool switchGroup0) {
1254 bool groupOK = CheckGroup(m_groupIndex);
1255
1256 if (!groupOK) {
1257 SetGroupIndex(m_groupIndex, true);
1258 }
1259}
1260
1261void ChartCanvas::SetShowGPS(bool bshow) {
1262 if (m_bShowGPS != bshow) {
1263 delete m_Compass;
1264 m_Compass = new ocpnCompass(this, bshow);
1265 m_Compass->SetScaleFactor(g_compass_scalefactor);
1266 m_Compass->Show(m_bShowCompassWin && g_bShowCompassWin);
1267 }
1268 m_bShowGPS = bshow;
1269}
1270
1271void ChartCanvas::SetShowGPSCompassWindow(bool bshow) {
1272 m_bShowCompassWin = bshow;
1273 if (m_Compass) {
1274 m_Compass->Show(m_bShowCompassWin && g_bShowCompassWin);
1275 if (m_bShowCompassWin && g_bShowCompassWin) m_Compass->UpdateStatus();
1276 }
1277}
1278
1279int ChartCanvas::GetPianoHeight() {
1280 int height = 0;
1281 if (g_bShowChartBar && GetPiano()) height = m_Piano->GetHeight();
1282
1283 return height;
1284}
1285
1286void ChartCanvas::ConfigureChartBar() {
1287 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
1288
1289 m_Piano->SetVizIcon(new wxBitmap(style->GetIcon(_T("viz"))));
1290 m_Piano->SetInVizIcon(new wxBitmap(style->GetIcon(_T("redX"))));
1291
1292 if (GetQuiltMode()) {
1293 m_Piano->SetRoundedRectangles(true);
1294 }
1295 m_Piano->SetTMercIcon(new wxBitmap(style->GetIcon(_T("tmercprj"))));
1296 m_Piano->SetPolyIcon(new wxBitmap(style->GetIcon(_T("polyprj"))));
1297 m_Piano->SetSkewIcon(new wxBitmap(style->GetIcon(_T("skewprj"))));
1298}
1299
1300void ChartCanvas::ShowTides(bool bShow) {
1301 gFrame->LoadHarmonics();
1302
1303 if (ptcmgr->IsReady()) {
1304 SetbShowTide(bShow);
1305
1306 parent_frame->SetMenubarItemState(ID_MENU_SHOW_TIDES, bShow);
1307 } else {
1308 wxLogMessage(_T("Chart1::Event...TCMgr Not Available"));
1309 SetbShowTide(false);
1310 parent_frame->SetMenubarItemState(ID_MENU_SHOW_TIDES, false);
1311 }
1312
1313 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
1314 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
1315
1316 // TODO
1317 // if( GetbShowTide() ) {
1318 // FrameTCTimer.Start( TIMER_TC_VALUE_SECONDS * 1000,
1319 // wxTIMER_CONTINUOUS ); SetbTCUpdate( true ); // force immediate
1320 // update
1321 // } else
1322 // FrameTCTimer.Stop();
1323}
1324
1325void ChartCanvas::ShowCurrents(bool bShow) {
1326 gFrame->LoadHarmonics();
1327
1328 if (ptcmgr->IsReady()) {
1329 SetbShowCurrent(bShow);
1330 parent_frame->SetMenubarItemState(ID_MENU_SHOW_CURRENTS, bShow);
1331 } else {
1332 wxLogMessage(_T("Chart1::Event...TCMgr Not Available"));
1333 SetbShowCurrent(false);
1334 parent_frame->SetMenubarItemState(ID_MENU_SHOW_CURRENTS, false);
1335 }
1336
1337 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
1338 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
1339
1340 // TODO
1341 // if( GetbShowCurrent() ) {
1342 // FrameTCTimer.Start( TIMER_TC_VALUE_SECONDS * 1000,
1343 // wxTIMER_CONTINUOUS ); SetbTCUpdate( true ); // force immediate
1344 // update
1345 // } else
1346 // FrameTCTimer.Stop();
1347}
1348
1349// TODO
1350extern bool g_bPreserveScaleOnX;
1351extern ChartDummy *pDummyChart;
1352extern int g_sticky_chart;
1353
1354void ChartCanvas::canvasRefreshGroupIndex(void) { SetGroupIndex(m_groupIndex); }
1355
1356void ChartCanvas::SetGroupIndex(int index, bool autoSwitch) {
1357 SetAlertString(_T(""));
1358
1359 int new_index = index;
1360 if (index > (int)g_pGroupArray->GetCount()) new_index = 0;
1361
1362 bool bgroup_override = false;
1363 int old_group_index = new_index;
1364
1365 if (!CheckGroup(new_index)) {
1366 new_index = 0;
1367 bgroup_override = true;
1368 }
1369
1370 if (!autoSwitch && (index <= (int)g_pGroupArray->GetCount()))
1371 new_index = index;
1372
1373 // Get the currently displayed chart native scale, and the current ViewPort
1374 int current_chart_native_scale = GetCanvasChartNativeScale();
1375 ViewPort vp = GetVP();
1376
1377 m_groupIndex = new_index;
1378
1379 // Are there ENCs in this group
1380 if (ChartData) m_bENCGroup = ChartData->IsENCInGroup(m_groupIndex);
1381
1382 // Update the MUIBar for ENC availability
1383 if (m_muiBar) m_muiBar->SetCanvasENCAvailable(m_bENCGroup);
1384
1385 // Allow the chart database to pre-calculate the MBTile inclusion test
1386 // boolean...
1387 ChartData->CheckExclusiveTileGroup(m_canvasIndex);
1388
1389 // Invalidate the "sticky" chart on group change, since it might not be in
1390 // the new group
1391 g_sticky_chart = -1;
1392
1393 // We need a chartstack and quilt to figure out which chart to open in the
1394 // new group
1395 UpdateCanvasOnGroupChange();
1396
1397 int dbi_now = -1;
1398 if (GetQuiltMode()) dbi_now = GetQuiltReferenceChartIndex();
1399
1400 int dbi_hint = FindClosestCanvasChartdbIndex(current_chart_native_scale);
1401
1402 // If a new reference chart is indicated, set a good scale for it.
1403 if ((dbi_now != dbi_hint) || !GetQuiltMode()) {
1404 double best_scale = GetBestStartScale(dbi_hint, vp);
1405 SetVPScale(best_scale);
1406 }
1407
1408 if (GetQuiltMode()) dbi_hint = GetQuiltReferenceChartIndex();
1409
1410 // Refresh the canvas, selecting the "best" chart,
1411 // applying the prior ViewPort exactly
1412 canvasChartsRefresh(dbi_hint);
1413
1414 UpdateCanvasControlBar();
1415
1416 if (!autoSwitch && bgroup_override) {
1417 // show a short timed message box
1418 wxString msg(_("Group \""));
1419
1420 ChartGroup *pGroup = g_pGroupArray->Item(new_index - 1);
1421 msg += pGroup->m_group_name;
1422
1423 msg += _("\" is empty.");
1424
1425 OCPNMessageBox(this, msg, _("OpenCPN Group Notice"), wxICON_INFORMATION, 2);
1426
1427 return;
1428 }
1429
1430 // Message box is deferred so that canvas refresh occurs properly before
1431 // dialog
1432 if (bgroup_override) {
1433 wxString msg(_("Group \""));
1434
1435 ChartGroup *pGroup = g_pGroupArray->Item(old_group_index - 1);
1436 msg += pGroup->m_group_name;
1437
1438 msg += _("\" is empty, switching to \"All Active Charts\" group.");
1439
1440 OCPNMessageBox(this, msg, _("OpenCPN Group Notice"), wxOK, 5);
1441 }
1442}
1443
1444bool ChartCanvas::CheckGroup(int igroup) {
1445 if (!ChartData) return true; // Not known yet...
1446
1447 if (igroup == 0) return true; // "all charts" is always OK
1448
1449 if (igroup < 0) // negative group is an error
1450 return false;
1451
1452 ChartGroup *pGroup = g_pGroupArray->Item(igroup - 1);
1453
1454 if (pGroup->m_element_array.empty()) // truly empty group prompts a warning,
1455 // and auto-shift to group 0
1456 return false;
1457
1458 for (const auto &elem : pGroup->m_element_array) {
1459 for (unsigned int ic = 0;
1460 ic < (unsigned int)ChartData->GetChartTableEntries(); ic++) {
1461 ChartTableEntry *pcte = ChartData->GetpChartTableEntry(ic);
1462 wxString chart_full_path(pcte->GetpFullPath(), wxConvUTF8);
1463
1464 if (chart_full_path.StartsWith(elem.m_element_name)) return true;
1465 }
1466 }
1467
1468 // If necessary, check for GSHHS
1469 for (const auto &elem : pGroup->m_element_array) {
1470 const wxString &element_root = elem.m_element_name;
1471 wxString test_string = _T("GSHH");
1472 if (element_root.Upper().Contains(test_string)) return true;
1473 }
1474
1475 return false;
1476}
1477
1478void ChartCanvas::canvasChartsRefresh(int dbi_hint) {
1479 if (!ChartData) return;
1480
1481 AbstractPlatform::ShowBusySpinner();
1482
1483 double old_scale = GetVPScale();
1484 InvalidateQuilt();
1485 SetQuiltRefChart(-1);
1486
1487 m_singleChart = NULL;
1488
1489 // delete m_pCurrentStack;
1490 // m_pCurrentStack = NULL;
1491
1492 // Build a new ChartStack
1493 if (!m_pCurrentStack) {
1494 m_pCurrentStack = new ChartStack;
1495 ChartData->BuildChartStack(m_pCurrentStack, m_vLat, m_vLon, m_groupIndex);
1496 }
1497
1498 if (-1 != dbi_hint) {
1499 if (GetQuiltMode()) {
1500 GetpCurrentStack()->SetCurrentEntryFromdbIndex(dbi_hint);
1501 SetQuiltRefChart(dbi_hint);
1502 } else {
1503 // Open the saved chart
1504 ChartBase *pTentative_Chart;
1505 pTentative_Chart = ChartData->OpenChartFromDB(dbi_hint, FULL_INIT);
1506
1507 if (pTentative_Chart) {
1508 /* m_singleChart is always NULL here, (set above) should this go before
1509 * that? */
1510 if (m_singleChart) m_singleChart->Deactivate();
1511
1512 m_singleChart = pTentative_Chart;
1513 m_singleChart->Activate();
1514
1515 GetpCurrentStack()->CurrentStackEntry = ChartData->GetStackEntry(
1516 GetpCurrentStack(), m_singleChart->GetFullPath());
1517 }
1518 }
1519
1520 // refresh_Piano();
1521 } else {
1522 // Select reference chart from the stack, as though clicked by user
1523 // Make it the smallest scale chart on the stack
1524 GetpCurrentStack()->CurrentStackEntry = GetpCurrentStack()->nEntry - 1;
1525 int selected_index = GetpCurrentStack()->GetCurrentEntrydbIndex();
1526 SetQuiltRefChart(selected_index);
1527 }
1528
1529 // Validate the correct single chart, or set the quilt mode as appropriate
1530 SetupCanvasQuiltMode();
1531 if (!GetQuiltMode() && m_singleChart == 0) {
1532 // use a dummy like in DoChartUpdate
1533 if (NULL == pDummyChart) pDummyChart = new ChartDummy;
1534 m_singleChart = pDummyChart;
1535 SetVPScale(old_scale);
1536 }
1537
1538 ReloadVP();
1539
1540 UpdateCanvasControlBar();
1541 UpdateGPSCompassStatusBox(true);
1542
1543 SetCursor(wxCURSOR_ARROW);
1544
1545 AbstractPlatform::HideBusySpinner();
1546}
1547
1548bool ChartCanvas::DoCanvasUpdate(void) {
1549 double tLat, tLon; // Chart Stack location
1550 double vpLat, vpLon; // ViewPort location
1551 bool blong_jump = false;
1552 meters_to_shift = 0;
1553 dir_to_shift = 0;
1554
1555 bool bNewChart = false;
1556 bool bNewView = false;
1557 bool bCanvasChartAutoOpen = true; // debugging
1558
1559 bool bNewPiano = false;
1560 bool bOpenSpecified;
1561 ChartStack LastStack;
1562 ChartBase *pLast_Ch;
1563
1564 ChartStack WorkStack;
1565
1566 if (bDBUpdateInProgress) return false;
1567 if (!ChartData) return false;
1568
1569 if (ChartData->IsBusy()) return false;
1570
1571 // Startup case:
1572 // Quilting is enabled, but the last chart seen was not quiltable
1573 // In this case, drop to single chart mode, set persistence flag,
1574 // And open the specified chart
1575 // TODO implement this
1576 // if( m_bFirstAuto && ( g_restore_dbindex >= 0 ) ) {
1577 // if( GetQuiltMode() ) {
1578 // if( !IsChartQuiltableRef( g_restore_dbindex ) ) {
1579 // gFrame->ToggleQuiltMode();
1580 // m_bpersistent_quilt = true;
1581 // m_singleChart = NULL;
1582 // }
1583 // }
1584 // }
1585
1586 // If in auto-follow mode, use the current glat,glon to build chart
1587 // stack. Otherwise, use vLat, vLon gotten from click on chart canvas, or
1588 // other means
1589
1590 if (m_bFollow) {
1591 tLat = gLat;
1592 tLon = gLon;
1593
1594 // Set the ViewPort center based on the OWNSHIP offset
1595 double dx = m_OSoffsetx;
1596 double dy = m_OSoffsety;
1597 double d_east = dx / GetVP().view_scale_ppm;
1598 double d_north = dy / GetVP().view_scale_ppm;
1599
1600 if (GetUpMode() == NORTH_UP_MODE) {
1601 fromSM(d_east, d_north, gLat, gLon, &vpLat, &vpLon);
1602 } else {
1603 double offset_angle = atan2(d_north, d_east);
1604 double offset_distance = sqrt((d_north * d_north) + (d_east * d_east));
1605 double chart_angle = GetVPRotation();
1606 double target_angle = chart_angle + offset_angle;
1607 double d_east_mod = offset_distance * cos(target_angle);
1608 double d_north_mod = offset_distance * sin(target_angle);
1609 fromSM(d_east_mod, d_north_mod, gLat, gLon, &vpLat, &vpLon);
1610 }
1611
1612 extern double gCog_gt;
1613
1614 // on lookahead mode, adjust the vp center point
1615 if (m_bLookAhead && bGPSValid && !m_MouseDragging) {
1616 double cog_to_use = gCog;
1617 if (g_btenhertz &&
1618 (fabs(gCog - gCog_gt) > 20)) { // big COG change in process
1619 cog_to_use = gCog_gt;
1620 blong_jump = true;
1621 }
1622 if (!g_btenhertz) cog_to_use = g_COGAvg;
1623
1624 double angle = cog_to_use + (GetVPRotation() * 180. / PI);
1625
1626 double pixel_deltay = (cos(angle * PI / 180.)) * GetCanvasHeight() / 4;
1627 double pixel_deltax = (sin(angle * PI / 180.)) * GetCanvasWidth() / 4;
1628
1629 double pixel_delta_tent =
1630 sqrt((pixel_deltay * pixel_deltay) + (pixel_deltax * pixel_deltax));
1631
1632 double pixel_delta = 0;
1633
1634 // The idea here is to cancel the effect of LookAhead for slow gSog, to
1635 // avoid jumping of the vp center point during slow maneuvering, or at
1636 // anchor....
1637 if (!std::isnan(gSog)) {
1638 if (gSog < 2.0)
1639 pixel_delta = 0.;
1640 else
1641 pixel_delta = pixel_delta_tent;
1642 }
1643
1644 meters_to_shift = 0;
1645 dir_to_shift = 0;
1646 if (!std::isnan(gCog)) {
1647 meters_to_shift = cos(gLat * PI / 180.) * pixel_delta / GetVPScale();
1648 dir_to_shift = cog_to_use;
1649 ll_gc_ll(gLat, gLon, dir_to_shift, meters_to_shift / 1852., &vpLat,
1650 &vpLon);
1651 } else {
1652 vpLat = gLat;
1653 vpLon = gLon;
1654 }
1655 } else if (m_bLookAhead && (!bGPSValid || m_MouseDragging)) {
1656 m_OSoffsetx = 0; // center ownship on loss of GPS
1657 m_OSoffsety = 0;
1658 vpLat = gLat;
1659 vpLon = gLon;
1660 }
1661
1662 } else {
1663 tLat = m_vLat;
1664 tLon = m_vLon;
1665 vpLat = m_vLat;
1666 vpLon = m_vLon;
1667 }
1668
1669 if (GetQuiltMode()) {
1670 int current_db_index = -1;
1671 if (m_pCurrentStack)
1672 current_db_index =
1673 m_pCurrentStack
1674 ->GetCurrentEntrydbIndex(); // capture the currently selected Ref
1675 // chart dbIndex
1676 else
1677 m_pCurrentStack = new ChartStack;
1678
1679 // This logic added to enable opening a chart when there is no
1680 // previous chart indication, either from inital startup, or from adding
1681 // new chart directory
1682 if (m_bautofind && (-1 == GetQuiltReferenceChartIndex()) &&
1683 m_pCurrentStack) {
1684 if (m_pCurrentStack->nEntry) {
1685 int new_dbIndex = m_pCurrentStack->GetDBIndex(m_pCurrentStack->nEntry -
1686 1); // smallest scale
1687 SelectQuiltRefdbChart(new_dbIndex, true);
1688 m_bautofind = false;
1689 }
1690 }
1691
1692 ChartData->BuildChartStack(m_pCurrentStack, tLat, tLon, m_groupIndex);
1693 m_pCurrentStack->SetCurrentEntryFromdbIndex(current_db_index);
1694
1695 if (m_bFirstAuto) {
1696 // Allow the chart database to pre-calculate the MBTile inclusion test
1697 // boolean...
1698 ChartData->CheckExclusiveTileGroup(m_canvasIndex);
1699
1700 // Calculate DPI compensation scale, i.e., the ratio of logical pixels to
1701 // physical pixels. On standard DPI displays where logical = physical
1702 // pixels, this ratio would be 1.0. On Retina displays where physical = 2x
1703 // logical pixels, this ratio would be 0.5.
1704 double proposed_scale_onscreen =
1705 GetCanvasScaleFactor() / GetVPScale(); // as set from config load
1706
1707 int initial_db_index = m_restore_dbindex;
1708 if (initial_db_index < 0) {
1709 if (m_pCurrentStack->nEntry) {
1710 initial_db_index =
1711 m_pCurrentStack->GetDBIndex(m_pCurrentStack->nEntry - 1);
1712 } else
1713 m_bautofind = true; // initial_db_index = 0;
1714 }
1715
1716 if (m_pCurrentStack->nEntry) {
1717 int initial_type = ChartData->GetDBChartType(initial_db_index);
1718
1719 // Check to see if the target new chart is quiltable as a reference
1720 // chart
1721
1722 if (!IsChartQuiltableRef(initial_db_index)) {
1723 // If it is not quiltable, then walk the stack up looking for a
1724 // satisfactory chart i.e. one that is quiltable and of the same type
1725 // XXX if there's none?
1726 int stack_index = 0;
1727
1728 if (stack_index >= 0) {
1729 while ((stack_index < m_pCurrentStack->nEntry - 1)) {
1730 int test_db_index = m_pCurrentStack->GetDBIndex(stack_index);
1731 if (IsChartQuiltableRef(test_db_index) &&
1732 (initial_type ==
1733 ChartData->GetDBChartType(initial_db_index))) {
1734 initial_db_index = test_db_index;
1735 break;
1736 }
1737 stack_index++;
1738 }
1739 }
1740 }
1741
1742 ChartBase *pc = ChartData->OpenChartFromDB(initial_db_index, FULL_INIT);
1743 if (pc) {
1744 SetQuiltRefChart(initial_db_index);
1745 m_pCurrentStack->SetCurrentEntryFromdbIndex(initial_db_index);
1746 }
1747 }
1748 // TODO: GetCanvasScaleFactor() / proposed_scale_onscreen simplifies to
1749 // just GetVPScale(), so I'm not sure why it's necessary to define the
1750 // proposed_scale_onscreen variable.
1751 bNewView |= SetViewPoint(vpLat, vpLon,
1752 GetCanvasScaleFactor() / proposed_scale_onscreen,
1753 0, GetVPRotation());
1754 }
1755 // Measure rough jump distance if in bfollow mode
1756 // No good reason to do smooth pan for
1757 // jump distance more than one screen width at scale.
1758 bool super_jump = false;
1759 if (m_bFollow) {
1760 double pixlt = fabs(vpLat - m_vLat) * 1852 * 60 * GetVPScale();
1761 double pixlg = fabs(vpLon - m_vLon) * 1852 * 60 * GetVPScale();
1762 if (wxMax(pixlt, pixlg) > GetCanvasWidth()) super_jump = true;
1763 }
1764 if (m_bFollow && g_btenhertz && !super_jump && !m_bLookAhead) {
1765 int nstep = 5;
1766 if (blong_jump) nstep = 20;
1767 StartTimedMovementVP(vpLat, vpLon, nstep);
1768 } else {
1769 bNewView |= SetViewPoint(vpLat, vpLon, GetVPScale(), 0, GetVPRotation());
1770 }
1771
1772 goto update_finish;
1773 }
1774
1775 // Single Chart Mode from here....
1776 pLast_Ch = m_singleChart;
1777 ChartTypeEnum new_open_type;
1778 ChartFamilyEnum new_open_family;
1779 if (pLast_Ch) {
1780 new_open_type = pLast_Ch->GetChartType();
1781 new_open_family = pLast_Ch->GetChartFamily();
1782 } else {
1783 new_open_type = CHART_TYPE_KAP;
1784 new_open_family = CHART_FAMILY_RASTER;
1785 }
1786
1787 bOpenSpecified = m_bFirstAuto;
1788
1789 // Make sure the target stack is valid
1790 if (NULL == m_pCurrentStack) m_pCurrentStack = new ChartStack;
1791
1792 // Build a chart stack based on tLat, tLon
1793 if (0 == ChartData->BuildChartStack(&WorkStack, tLat, tLon, g_sticky_chart,
1794 m_groupIndex)) { // Bogus Lat, Lon?
1795 if (NULL == pDummyChart) {
1796 pDummyChart = new ChartDummy;
1797 bNewChart = true;
1798 }
1799
1800 if (m_singleChart)
1801 if (m_singleChart->GetChartType() != CHART_TYPE_DUMMY) bNewChart = true;
1802
1803 m_singleChart = pDummyChart;
1804
1805 // If the current viewpoint is invalid, set the default scale to
1806 // something reasonable.
1807 double set_scale = GetVPScale();
1808 if (!GetVP().IsValid()) set_scale = 1. / 20000.;
1809
1810 bNewView |= SetViewPoint(tLat, tLon, set_scale, 0, GetVPRotation());
1811
1812 // If the chart stack has just changed, there is new status
1813 if (WorkStack.nEntry && m_pCurrentStack->nEntry) {
1814 if (!ChartData->EqualStacks(&WorkStack, m_pCurrentStack)) {
1815 bNewPiano = true;
1816 bNewChart = true;
1817 }
1818 }
1819
1820 // Copy the new (by definition empty) stack into the target stack
1821 ChartData->CopyStack(m_pCurrentStack, &WorkStack);
1822
1823 goto update_finish;
1824 }
1825
1826 // Check to see if Chart Stack has changed
1827 if (!ChartData->EqualStacks(&WorkStack, m_pCurrentStack)) {
1828 // New chart stack, so...
1829 bNewPiano = true;
1830
1831 // Save a copy of the current stack
1832 ChartData->CopyStack(&LastStack, m_pCurrentStack);
1833
1834 // Copy the new stack into the target stack
1835 ChartData->CopyStack(m_pCurrentStack, &WorkStack);
1836
1837 // Is Current Chart in new stack?
1838
1839 int tEntry = -1;
1840 if (NULL != m_singleChart) // this handles startup case
1841 tEntry = ChartData->GetStackEntry(m_pCurrentStack,
1842 m_singleChart->GetFullPath());
1843
1844 if (tEntry != -1) { // m_singleChart is in the new stack
1845 m_pCurrentStack->CurrentStackEntry = tEntry;
1846 bNewChart = false;
1847 }
1848
1849 else // m_singleChart is NOT in new stack
1850 { // So, need to open a new chart
1851 // Find the largest scale raster chart that opens OK
1852
1853 ChartBase *pProposed = NULL;
1854
1855 if (bCanvasChartAutoOpen) {
1856 bool search_direction =
1857 false; // default is to search from lowest to highest
1858 int start_index = 0;
1859
1860 // A special case: If panning at high scale, open largest scale
1861 // chart first
1862 if ((LastStack.CurrentStackEntry == LastStack.nEntry - 1) ||
1863 (LastStack.nEntry == 0)) {
1864 search_direction = true;
1865 start_index = m_pCurrentStack->nEntry - 1;
1866 }
1867
1868 // Another special case, open specified index on program start
1869 if (bOpenSpecified) {
1870 search_direction = false;
1871 start_index = 0;
1872 if ((start_index < 0) | (start_index >= m_pCurrentStack->nEntry))
1873 start_index = 0;
1874
1875 new_open_type = CHART_TYPE_DONTCARE;
1876 }
1877
1878 pProposed = ChartData->OpenStackChartConditional(
1879 m_pCurrentStack, start_index, search_direction, new_open_type,
1880 new_open_family);
1881
1882 // Try to open other types/families of chart in some priority
1883 if (NULL == pProposed)
1884 pProposed = ChartData->OpenStackChartConditional(
1885 m_pCurrentStack, start_index, search_direction,
1886 CHART_TYPE_CM93COMP, CHART_FAMILY_VECTOR);
1887
1888 if (NULL == pProposed)
1889 pProposed = ChartData->OpenStackChartConditional(
1890 m_pCurrentStack, start_index, search_direction,
1891 CHART_TYPE_CM93COMP, CHART_FAMILY_RASTER);
1892
1893 bNewChart = true;
1894
1895 } // bCanvasChartAutoOpen
1896
1897 else
1898 pProposed = NULL;
1899
1900 // If no go, then
1901 // Open a Dummy Chart
1902 if (NULL == pProposed) {
1903 if (NULL == pDummyChart) {
1904 pDummyChart = new ChartDummy;
1905 bNewChart = true;
1906 }
1907
1908 if (pLast_Ch)
1909 if (pLast_Ch->GetChartType() != CHART_TYPE_DUMMY) bNewChart = true;
1910
1911 pProposed = pDummyChart;
1912 }
1913
1914 // Arriving here, pProposed points to an opened chart, or NULL.
1915 if (m_singleChart) m_singleChart->Deactivate();
1916 m_singleChart = pProposed;
1917
1918 if (m_singleChart) {
1919 m_singleChart->Activate();
1920 m_pCurrentStack->CurrentStackEntry = ChartData->GetStackEntry(
1921 m_pCurrentStack, m_singleChart->GetFullPath());
1922 }
1923 } // need new chart
1924
1925 // Arriving here, m_singleChart is opened and OK, or NULL
1926 if (NULL != m_singleChart) {
1927 // Setup the view using the current scale
1928 double set_scale = GetVPScale();
1929
1930 // If the current viewpoint is invalid, set the default scale to
1931 // something reasonable.
1932 if (!GetVP().IsValid())
1933 set_scale = 1. / 20000.;
1934 else { // otherwise, match scale if elected.
1935 double proposed_scale_onscreen;
1936
1937 if (m_bFollow) { // autoset the scale only if in autofollow
1938 double new_scale_ppm =
1939 m_singleChart->GetNearestPreferredScalePPM(GetVPScale());
1940 proposed_scale_onscreen = GetCanvasScaleFactor() / new_scale_ppm;
1941 } else
1942 proposed_scale_onscreen = GetCanvasScaleFactor() / set_scale;
1943
1944 // This logic will bring a new chart onscreen at roughly twice the true
1945 // paper scale equivalent. Note that first chart opened on application
1946 // startup (bOpenSpecified = true) will open at the config saved scale
1947 if (bNewChart && !g_bPreserveScaleOnX && !bOpenSpecified) {
1948 proposed_scale_onscreen = m_singleChart->GetNativeScale() / 2;
1949 double equivalent_vp_scale =
1950 GetCanvasScaleFactor() / proposed_scale_onscreen;
1951 double new_scale_ppm =
1952 m_singleChart->GetNearestPreferredScalePPM(equivalent_vp_scale);
1953 proposed_scale_onscreen = GetCanvasScaleFactor() / new_scale_ppm;
1954 }
1955
1956 if (m_bFollow) { // bounds-check the scale only if in autofollow
1957 proposed_scale_onscreen =
1958 wxMin(proposed_scale_onscreen,
1959 m_singleChart->GetNormalScaleMax(GetCanvasScaleFactor(),
1960 GetCanvasWidth()));
1961 proposed_scale_onscreen =
1962 wxMax(proposed_scale_onscreen,
1963 m_singleChart->GetNormalScaleMin(GetCanvasScaleFactor(),
1964 g_b_overzoom_x));
1965 }
1966
1967 set_scale = GetCanvasScaleFactor() / proposed_scale_onscreen;
1968 }
1969
1970 bNewView |= SetViewPoint(vpLat, vpLon, set_scale,
1971 m_singleChart->GetChartSkew() * PI / 180.,
1972 GetVPRotation());
1973 }
1974 } // new stack
1975
1976 else // No change in Chart Stack
1977 {
1978 if ((m_bFollow) && m_singleChart)
1979 bNewView |= SetViewPoint(vpLat, vpLon, GetVPScale(),
1980 m_singleChart->GetChartSkew() * PI / 180.,
1981 GetVPRotation());
1982 }
1983
1984update_finish:
1985
1986 // TODO
1987 // if( bNewPiano ) UpdateControlBar();
1988
1989 m_bFirstAuto = false; // Auto open on program start
1990
1991 // If we need a Refresh(), do it here...
1992 // But don't duplicate a Refresh() done by SetViewPoint()
1993 if (bNewChart && !bNewView) Refresh(false);
1994
1995#ifdef ocpnUSE_GL
1996 // If a new chart, need to invalidate gl viewport for refresh
1997 // so the fbo gets flushed
1998 if (m_glcc && g_bopengl && bNewChart) GetglCanvas()->Invalidate();
1999#endif
2000
2001 return bNewChart | bNewView;
2002}
2003
2004void ChartCanvas::SelectQuiltRefdbChart(int db_index, bool b_autoscale) {
2005 if (m_pCurrentStack) m_pCurrentStack->SetCurrentEntryFromdbIndex(db_index);
2006
2007 SetQuiltRefChart(db_index);
2008 if (ChartData) {
2009 ChartBase *pc = ChartData->OpenChartFromDB(db_index, FULL_INIT);
2010 if (pc) {
2011 if (b_autoscale) {
2012 double best_scale_ppm = GetBestVPScale(pc);
2013 SetVPScale(best_scale_ppm);
2014 }
2015 } else
2016 SetQuiltRefChart(-1);
2017 } else
2018 SetQuiltRefChart(-1);
2019}
2020
2021void ChartCanvas::SelectQuiltRefChart(int selected_index) {
2022 std::vector<int> piano_chart_index_array =
2023 GetQuiltExtendedStackdbIndexArray();
2024 int current_db_index = piano_chart_index_array[selected_index];
2025
2026 SelectQuiltRefdbChart(current_db_index);
2027}
2028
2029double ChartCanvas::GetBestVPScale(ChartBase *pchart) {
2030 if (pchart) {
2031 double proposed_scale_onscreen = GetCanvasScaleFactor() / GetVPScale();
2032
2033 if ((g_bPreserveScaleOnX) ||
2034 (CHART_TYPE_CM93COMP == pchart->GetChartType())) {
2035 double new_scale_ppm = GetVPScale();
2036 proposed_scale_onscreen = GetCanvasScaleFactor() / new_scale_ppm;
2037 } else {
2038 // This logic will bring the new chart onscreen at roughly twice the true
2039 // paper scale equivalent.
2040 proposed_scale_onscreen = pchart->GetNativeScale() / 2;
2041 double equivalent_vp_scale =
2042 GetCanvasScaleFactor() / proposed_scale_onscreen;
2043 double new_scale_ppm =
2044 pchart->GetNearestPreferredScalePPM(equivalent_vp_scale);
2045 proposed_scale_onscreen = GetCanvasScaleFactor() / new_scale_ppm;
2046 }
2047
2048 // Do not allow excessive underzoom, even if the g_bPreserveScaleOnX flag is
2049 // set. Otherwise, we get severe performance problems on all platforms
2050
2051 double max_underzoom_multiplier = 2.0;
2052 if (GetVP().b_quilt) {
2053 double scale_max = m_pQuilt->GetNomScaleMin(pchart->GetNativeScale(),
2054 pchart->GetChartType(),
2055 pchart->GetChartFamily());
2056 max_underzoom_multiplier = scale_max / pchart->GetNativeScale();
2057 }
2058
2059 proposed_scale_onscreen = wxMin(
2060 proposed_scale_onscreen,
2061 pchart->GetNormalScaleMax(GetCanvasScaleFactor(), GetCanvasWidth()) *
2062 max_underzoom_multiplier);
2063
2064 // And, do not allow excessive overzoom either
2065 proposed_scale_onscreen =
2066 wxMax(proposed_scale_onscreen,
2067 pchart->GetNormalScaleMin(GetCanvasScaleFactor(), false));
2068
2069 return GetCanvasScaleFactor() / proposed_scale_onscreen;
2070 } else
2071 return 1.0;
2072}
2073
2074void ChartCanvas::SetupCanvasQuiltMode(void) {
2075 if (GetQuiltMode()) // going to quilt mode
2076 {
2077 ChartData->LockCache();
2078
2079 m_Piano->SetNoshowIndexArray(m_quilt_noshow_index_array);
2080
2081 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
2082
2083 m_Piano->SetVizIcon(new wxBitmap(style->GetIcon(_T("viz"))));
2084 m_Piano->SetInVizIcon(new wxBitmap(style->GetIcon(_T("redX"))));
2085 m_Piano->SetTMercIcon(new wxBitmap(style->GetIcon(_T("tmercprj"))));
2086 m_Piano->SetSkewIcon(new wxBitmap(style->GetIcon(_T("skewprj"))));
2087
2088 m_Piano->SetRoundedRectangles(true);
2089
2090 // Select the proper Ref chart
2091 int target_new_dbindex = -1;
2092 if (m_pCurrentStack) {
2093 target_new_dbindex =
2094 GetQuiltReferenceChartIndex(); // m_pCurrentStack->GetCurrentEntrydbIndex();
2095
2096 if (-1 != target_new_dbindex) {
2097 if (!IsChartQuiltableRef(target_new_dbindex)) {
2098 int proj = ChartData->GetDBChartProj(target_new_dbindex);
2099 int type = ChartData->GetDBChartType(target_new_dbindex);
2100
2101 // walk the stack up looking for a satisfactory chart
2102 int stack_index = m_pCurrentStack->CurrentStackEntry;
2103
2104 while ((stack_index < m_pCurrentStack->nEntry - 1) &&
2105 (stack_index >= 0)) {
2106 int proj_tent = ChartData->GetDBChartProj(
2107 m_pCurrentStack->GetDBIndex(stack_index));
2108 int type_tent = ChartData->GetDBChartType(
2109 m_pCurrentStack->GetDBIndex(stack_index));
2110
2111 if (IsChartQuiltableRef(m_pCurrentStack->GetDBIndex(stack_index))) {
2112 if ((proj == proj_tent) && (type_tent == type)) {
2113 target_new_dbindex = m_pCurrentStack->GetDBIndex(stack_index);
2114 break;
2115 }
2116 }
2117 stack_index++;
2118 }
2119 }
2120 }
2121 }
2122
2123 if (IsChartQuiltableRef(target_new_dbindex))
2124 SelectQuiltRefdbChart(target_new_dbindex,
2125 false); // Try not to allow a scale change
2126 else
2127 SelectQuiltRefdbChart(-1, false);
2128
2129 m_singleChart = NULL; // Bye....
2130
2131 // Re-qualify the quilt reference chart selection
2132 AdjustQuiltRefChart();
2133
2134 // Restore projection type saved on last quilt mode toggle
2135 // TODO
2136 // if(g_sticky_projection != -1)
2137 // GetVP().SetProjectionType(g_sticky_projection);
2138 // else
2139 // GetVP().SetProjectionType(PROJECTION_MERCATOR);
2140 GetVP().SetProjectionType(PROJECTION_UNKNOWN);
2141
2142 } else // going to SC Mode
2143 {
2144 std::vector<int> empty_array;
2145 m_Piano->SetActiveKeyArray(empty_array);
2146 m_Piano->SetNoshowIndexArray(empty_array);
2147 m_Piano->SetEclipsedIndexArray(empty_array);
2148
2149 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
2150 m_Piano->SetVizIcon(new wxBitmap(style->GetIcon(_T("viz"))));
2151 m_Piano->SetInVizIcon(new wxBitmap(style->GetIcon(_T("redX"))));
2152 m_Piano->SetTMercIcon(new wxBitmap(style->GetIcon(_T("tmercprj"))));
2153 m_Piano->SetSkewIcon(new wxBitmap(style->GetIcon(_T("skewprj"))));
2154
2155 m_Piano->SetRoundedRectangles(false);
2156 // TODO Make this a member g_sticky_projection = GetVP().m_projection_type;
2157 }
2158
2159 // When shifting from quilt to single chart mode, select the "best" single
2160 // chart to show
2161 if (!GetQuiltMode()) {
2162 if (ChartData && ChartData->IsValid()) {
2163 UnlockQuilt();
2164
2165 double tLat, tLon;
2166 if (m_bFollow == true) {
2167 tLat = gLat;
2168 tLon = gLon;
2169 } else {
2170 tLat = m_vLat;
2171 tLon = m_vLon;
2172 }
2173
2174 if (!m_singleChart) {
2175 // Build a temporary chart stack based on tLat, tLon
2176 ChartStack TempStack;
2177 ChartData->BuildChartStack(&TempStack, tLat, tLon, g_sticky_chart,
2178 m_groupIndex);
2179
2180 // Iterate over the quilt charts actually shown, looking for the
2181 // largest scale chart that will be in the new chartstack.... This
2182 // will (almost?) always be the reference chart....
2183
2184 ChartBase *Candidate_Chart = NULL;
2185 int cur_max_scale = (int)1e8;
2186
2187 ChartBase *pChart = GetFirstQuiltChart();
2188 while (pChart) {
2189 // Is this pChart in new stack?
2190 int tEntry =
2191 ChartData->GetStackEntry(&TempStack, pChart->GetFullPath());
2192 if (tEntry != -1) {
2193 if (pChart->GetNativeScale() < cur_max_scale) {
2194 Candidate_Chart = pChart;
2195 cur_max_scale = pChart->GetNativeScale();
2196 }
2197 }
2198 pChart = GetNextQuiltChart();
2199 }
2200
2201 m_singleChart = Candidate_Chart;
2202
2203 // If the quilt is empty, there is no "best" chart.
2204 // So, open the smallest scale chart in the current stack
2205 if (NULL == m_singleChart) {
2206 m_singleChart = ChartData->OpenStackChartConditional(
2207 &TempStack, TempStack.nEntry - 1, true, CHART_TYPE_DONTCARE,
2208 CHART_FAMILY_DONTCARE);
2209 }
2210 }
2211
2212 // Invalidate all the charts in the quilt,
2213 // as any cached data may be region based and not have fullscreen coverage
2214 InvalidateAllQuiltPatchs();
2215
2216 if (m_singleChart) {
2217 int dbi = ChartData->FinddbIndex(m_singleChart->GetFullPath());
2218 std::vector<int> one_array;
2219 one_array.push_back(dbi);
2220 m_Piano->SetActiveKeyArray(one_array);
2221 }
2222
2223 if (m_singleChart) {
2224 GetVP().SetProjectionType(m_singleChart->GetChartProjectionType());
2225 }
2226 }
2227 // Invalidate the current stack so that it will be rebuilt on next tick
2228 if (m_pCurrentStack) m_pCurrentStack->b_valid = false;
2229 }
2230}
2231
2232bool ChartCanvas::IsTempMenuBarEnabled() {
2233#ifdef __WXMSW__
2234 int major;
2235 wxGetOsVersion(&major);
2236 return (major >
2237 5); // For Windows, function is only available on Vista and above
2238#else
2239 return true;
2240#endif
2241}
2242
2243double ChartCanvas::GetCanvasRangeMeters() {
2244 int width, height;
2245 GetSize(&width, &height);
2246 int minDimension = wxMin(width, height);
2247
2248 double range = (minDimension / GetVP().view_scale_ppm) / 2;
2249 range *= cos(GetVP().clat * PI / 180.);
2250 return range;
2251}
2252
2253void ChartCanvas::SetCanvasRangeMeters(double range) {
2254 int width, height;
2255 GetSize(&width, &height);
2256 int minDimension = wxMin(width, height);
2257
2258 double scale_ppm = minDimension / (range / cos(GetVP().clat * PI / 180.));
2259 SetVPScale(scale_ppm / 2);
2260}
2261
2262bool ChartCanvas::SetUserOwnship() {
2263 // Look for user defined ownship image
2264 // This may be found in the shared data location along with other user
2265 // defined icons. and will be called "ownship.xpm" or "ownship.png"
2266 if (pWayPointMan && pWayPointMan->DoesIconExist(_T("ownship"))) {
2267 double factor_dusk = 0.5;
2268 double factor_night = 0.25;
2269
2270 wxBitmap *pbmp = pWayPointMan->GetIconBitmap(_T("ownship"));
2271 m_pos_image_user_day = new wxImage;
2272 *m_pos_image_user_day = pbmp->ConvertToImage();
2273 if (!m_pos_image_user_day->HasAlpha()) m_pos_image_user_day->InitAlpha();
2274
2275 int gimg_width = m_pos_image_user_day->GetWidth();
2276 int gimg_height = m_pos_image_user_day->GetHeight();
2277
2278 // Make dusk and night images
2279 m_pos_image_user_dusk = new wxImage;
2280 m_pos_image_user_night = new wxImage;
2281
2282 *m_pos_image_user_dusk = m_pos_image_user_day->Copy();
2283 *m_pos_image_user_night = m_pos_image_user_day->Copy();
2284
2285 for (int iy = 0; iy < gimg_height; iy++) {
2286 for (int ix = 0; ix < gimg_width; ix++) {
2287 if (!m_pos_image_user_day->IsTransparent(ix, iy)) {
2288 wxImage::RGBValue rgb(m_pos_image_user_day->GetRed(ix, iy),
2289 m_pos_image_user_day->GetGreen(ix, iy),
2290 m_pos_image_user_day->GetBlue(ix, iy));
2291 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
2292 hsv.value = hsv.value * factor_dusk;
2293 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
2294 m_pos_image_user_dusk->SetRGB(ix, iy, nrgb.red, nrgb.green,
2295 nrgb.blue);
2296
2297 hsv = wxImage::RGBtoHSV(rgb);
2298 hsv.value = hsv.value * factor_night;
2299 nrgb = wxImage::HSVtoRGB(hsv);
2300 m_pos_image_user_night->SetRGB(ix, iy, nrgb.red, nrgb.green,
2301 nrgb.blue);
2302 }
2303 }
2304 }
2305
2306 // Make some alternate greyed out day/dusk/night images
2307 m_pos_image_user_grey_day = new wxImage;
2308 *m_pos_image_user_grey_day = m_pos_image_user_day->ConvertToGreyscale();
2309
2310 m_pos_image_user_grey_dusk = new wxImage;
2311 m_pos_image_user_grey_night = new wxImage;
2312
2313 *m_pos_image_user_grey_dusk = m_pos_image_user_grey_day->Copy();
2314 *m_pos_image_user_grey_night = m_pos_image_user_grey_day->Copy();
2315
2316 for (int iy = 0; iy < gimg_height; iy++) {
2317 for (int ix = 0; ix < gimg_width; ix++) {
2318 if (!m_pos_image_user_grey_day->IsTransparent(ix, iy)) {
2319 wxImage::RGBValue rgb(m_pos_image_user_grey_day->GetRed(ix, iy),
2320 m_pos_image_user_grey_day->GetGreen(ix, iy),
2321 m_pos_image_user_grey_day->GetBlue(ix, iy));
2322 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
2323 hsv.value = hsv.value * factor_dusk;
2324 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
2325 m_pos_image_user_grey_dusk->SetRGB(ix, iy, nrgb.red, nrgb.green,
2326 nrgb.blue);
2327
2328 hsv = wxImage::RGBtoHSV(rgb);
2329 hsv.value = hsv.value * factor_night;
2330 nrgb = wxImage::HSVtoRGB(hsv);
2331 m_pos_image_user_grey_night->SetRGB(ix, iy, nrgb.red, nrgb.green,
2332 nrgb.blue);
2333 }
2334 }
2335 }
2336
2337 // Make a yellow image for rendering under low accuracy chart conditions
2338 m_pos_image_user_yellow_day = new wxImage;
2339 m_pos_image_user_yellow_dusk = new wxImage;
2340 m_pos_image_user_yellow_night = new wxImage;
2341
2342 *m_pos_image_user_yellow_day = m_pos_image_user_grey_day->Copy();
2343 *m_pos_image_user_yellow_dusk = m_pos_image_user_grey_day->Copy();
2344 *m_pos_image_user_yellow_night = m_pos_image_user_grey_day->Copy();
2345
2346 for (int iy = 0; iy < gimg_height; iy++) {
2347 for (int ix = 0; ix < gimg_width; ix++) {
2348 if (!m_pos_image_user_grey_day->IsTransparent(ix, iy)) {
2349 wxImage::RGBValue rgb(m_pos_image_user_grey_day->GetRed(ix, iy),
2350 m_pos_image_user_grey_day->GetGreen(ix, iy),
2351 m_pos_image_user_grey_day->GetBlue(ix, iy));
2352
2353 // Simply remove all "blue" from the greyscaled image...
2354 // so, what is not black becomes yellow.
2355 wxImage::HSVValue hsv = wxImage::RGBtoHSV(rgb);
2356 wxImage::RGBValue nrgb = wxImage::HSVtoRGB(hsv);
2357 m_pos_image_user_yellow_day->SetRGB(ix, iy, nrgb.red, nrgb.green, 0);
2358
2359 hsv = wxImage::RGBtoHSV(rgb);
2360 hsv.value = hsv.value * factor_dusk;
2361 nrgb = wxImage::HSVtoRGB(hsv);
2362 m_pos_image_user_yellow_dusk->SetRGB(ix, iy, nrgb.red, nrgb.green, 0);
2363
2364 hsv = wxImage::RGBtoHSV(rgb);
2365 hsv.value = hsv.value * factor_night;
2366 nrgb = wxImage::HSVtoRGB(hsv);
2367 m_pos_image_user_yellow_night->SetRGB(ix, iy, nrgb.red, nrgb.green,
2368 0);
2369 }
2370 }
2371 }
2372
2373 return true;
2374 } else
2375 return false;
2376}
2377
2379 m_display_size_mm = size;
2380
2381 // int sx, sy;
2382 // wxDisplaySize( &sx, &sy );
2383
2384 // Calculate logical pixels per mm for later reference.
2385 wxSize sd = g_Platform->getDisplaySize();
2386 double horizontal = sd.x;
2387 // Set DPI (Win) scale factor
2388 g_scaler = g_Platform->GetDisplayDIPMult(this);
2389
2390 m_pix_per_mm = (horizontal) / ((double)m_display_size_mm);
2391 m_canvas_scale_factor = (horizontal) / (m_display_size_mm / 1000.);
2392
2393 if (ps52plib) {
2394 ps52plib->SetDisplayWidth(g_monitor_info[g_current_monitor].width);
2395 ps52plib->SetPPMM(m_pix_per_mm);
2396 }
2397
2398 wxString msg;
2399 msg.Printf(
2400 _T("Metrics: m_display_size_mm: %g g_Platform->getDisplaySize(): ")
2401 _T("%d:%d "),
2402 m_display_size_mm, sd.x, sd.y);
2403 wxLogMessage(msg);
2404
2405 int ssx, ssy;
2406 ssx = g_monitor_info[g_current_monitor].width;
2407 ssy = g_monitor_info[g_current_monitor].height;
2408 msg.Printf(_T("monitor size: %d %d"), ssx, ssy);
2409 wxLogMessage(msg);
2410
2411 m_focus_indicator_pix = /*std::round*/ wxRound(1 * GetPixPerMM());
2412}
2413#if 0
2414void ChartCanvas::OnEvtCompressProgress( OCPN_CompressProgressEvent & event )
2415{
2416 wxString msg(event.m_string.c_str(), wxConvUTF8);
2417 // if cpus are removed between runs
2418 if(pprog_threads > 0 && compress_msg_array.GetCount() > (unsigned int)pprog_threads) {
2419 compress_msg_array.RemoveAt(pprog_threads, compress_msg_array.GetCount() - pprog_threads);
2420 }
2421
2422 if(compress_msg_array.GetCount() > (unsigned int)event.thread )
2423 {
2424 compress_msg_array.RemoveAt(event.thread);
2425 compress_msg_array.Insert( msg, event.thread);
2426 }
2427 else
2428 compress_msg_array.Add(msg);
2429
2430
2431 wxString combined_msg;
2432 for(unsigned int i=0 ; i < compress_msg_array.GetCount() ; i++) {
2433 combined_msg += compress_msg_array[i];
2434 combined_msg += _T("\n");
2435 }
2436
2437 bool skip = false;
2438 pprog->Update(pprog_count, combined_msg, &skip );
2439 pprog->SetSize(pprog_size);
2440 if(skip)
2441 b_skipout = skip;
2442}
2443#endif
2444void ChartCanvas::InvalidateGL() {
2445 if (!m_glcc) return;
2446#ifdef ocpnUSE_GL
2447 if (g_bopengl) m_glcc->Invalidate();
2448#endif
2449 if (m_Compass) m_Compass->UpdateStatus(true);
2450}
2451
2452int ChartCanvas::GetCanvasChartNativeScale() {
2453 int ret = 1;
2454 if (!VPoint.b_quilt) {
2455 if (m_singleChart) ret = m_singleChart->GetNativeScale();
2456 } else
2457 ret = (int)m_pQuilt->GetRefNativeScale();
2458
2459 return ret;
2460}
2461
2462ChartBase *ChartCanvas::GetChartAtCursor() {
2463 ChartBase *target_chart;
2464 if (m_singleChart && (m_singleChart->GetChartFamily() == CHART_FAMILY_VECTOR))
2465 target_chart = m_singleChart;
2466 else if (VPoint.b_quilt)
2467 target_chart = m_pQuilt->GetChartAtPix(VPoint, wxPoint(mouse_x, mouse_y));
2468 else
2469 target_chart = NULL;
2470 return target_chart;
2471}
2472
2473ChartBase *ChartCanvas::GetOverlayChartAtCursor() {
2474 ChartBase *target_chart;
2475 if (VPoint.b_quilt)
2476 target_chart =
2477 m_pQuilt->GetOverlayChartAtPix(VPoint, wxPoint(mouse_x, mouse_y));
2478 else
2479 target_chart = NULL;
2480 return target_chart;
2481}
2482
2483int ChartCanvas::FindClosestCanvasChartdbIndex(int scale) {
2484 int new_dbIndex = -1;
2485 if (!VPoint.b_quilt) {
2486 if (m_pCurrentStack) {
2487 for (int i = 0; i < m_pCurrentStack->nEntry; i++) {
2488 int sc = ChartData->GetStackChartScale(m_pCurrentStack, i, NULL, 0);
2489 if (sc >= scale) {
2490 new_dbIndex = m_pCurrentStack->GetDBIndex(i);
2491 break;
2492 }
2493 }
2494 }
2495 } else {
2496 // Using the current quilt, select a useable reference chart
2497 // Said chart will be in the extended (possibly full-screen) stack,
2498 // And will have a scale equal to or just greater than the stipulated
2499 // value
2500 unsigned int im = m_pQuilt->GetExtendedStackIndexArray().size();
2501 if (im > 0) {
2502 for (unsigned int is = 0; is < im; is++) {
2503 const ChartTableEntry &m = ChartData->GetChartTableEntry(
2504 m_pQuilt->GetExtendedStackIndexArray()[is]);
2505 if ((m.Scale_ge(
2506 scale)) /* && (m_reference_family == m.GetChartFamily())*/) {
2507 new_dbIndex = m_pQuilt->GetExtendedStackIndexArray()[is];
2508 break;
2509 }
2510 }
2511 }
2512 }
2513
2514 return new_dbIndex;
2515}
2516
2517void ChartCanvas::EnablePaint(bool b_enable) {
2518 m_b_paint_enable = b_enable;
2519#ifdef ocpnUSE_GL
2520 if (m_glcc) m_glcc->EnablePaint(b_enable);
2521#endif
2522}
2523
2524bool ChartCanvas::IsQuiltDelta() { return m_pQuilt->IsQuiltDelta(VPoint); }
2525
2526void ChartCanvas::UnlockQuilt() { m_pQuilt->UnlockQuilt(); }
2527
2528std::vector<int> ChartCanvas::GetQuiltIndexArray(void) {
2529 return m_pQuilt->GetQuiltIndexArray();
2530 ;
2531}
2532
2533void ChartCanvas::SetQuiltMode(bool b_quilt) {
2534 VPoint.b_quilt = b_quilt;
2535 VPoint.b_FullScreenQuilt = g_bFullScreenQuilt;
2536}
2537
2538bool ChartCanvas::GetQuiltMode(void) { return VPoint.b_quilt; }
2539
2540int ChartCanvas::GetQuiltReferenceChartIndex(void) {
2541 return m_pQuilt->GetRefChartdbIndex();
2542}
2543
2544void ChartCanvas::InvalidateAllQuiltPatchs(void) {
2545 m_pQuilt->InvalidateAllQuiltPatchs();
2546}
2547
2548ChartBase *ChartCanvas::GetLargestScaleQuiltChart() {
2549 return m_pQuilt->GetLargestScaleChart();
2550}
2551
2552ChartBase *ChartCanvas::GetFirstQuiltChart() {
2553 return m_pQuilt->GetFirstChart();
2554}
2555
2556ChartBase *ChartCanvas::GetNextQuiltChart() { return m_pQuilt->GetNextChart(); }
2557
2558int ChartCanvas::GetQuiltChartCount() { return m_pQuilt->GetnCharts(); }
2559
2560void ChartCanvas::SetQuiltChartHiLiteIndex(int dbIndex) {
2561 m_pQuilt->SetHiliteIndex(dbIndex);
2562}
2563
2564void ChartCanvas::SetQuiltChartHiLiteIndexArray(std::vector<int> hilite_array) {
2565 m_pQuilt->SetHiliteIndexArray(hilite_array);
2566}
2567
2568void ChartCanvas::ClearQuiltChartHiLiteIndexArray() {
2569 m_pQuilt->ClearHiliteIndexArray();
2570}
2571
2572std::vector<int> ChartCanvas::GetQuiltCandidatedbIndexArray(bool flag1,
2573 bool flag2) {
2574 return m_pQuilt->GetCandidatedbIndexArray(flag1, flag2);
2575}
2576
2577int ChartCanvas::GetQuiltRefChartdbIndex(void) {
2578 return m_pQuilt->GetRefChartdbIndex();
2579}
2580
2581std::vector<int> &ChartCanvas::GetQuiltExtendedStackdbIndexArray() {
2582 return m_pQuilt->GetExtendedStackIndexArray();
2583}
2584
2585std::vector<int> &ChartCanvas::GetQuiltFullScreendbIndexArray() {
2586 return m_pQuilt->GetFullscreenIndexArray();
2587}
2588
2589std::vector<int> ChartCanvas::GetQuiltEclipsedStackdbIndexArray() {
2590 return m_pQuilt->GetEclipsedStackIndexArray();
2591}
2592
2593void ChartCanvas::InvalidateQuilt(void) { return m_pQuilt->Invalidate(); }
2594
2595double ChartCanvas::GetQuiltMaxErrorFactor() {
2596 return m_pQuilt->GetMaxErrorFactor();
2597}
2598
2599bool ChartCanvas::IsChartQuiltableRef(int db_index) {
2600 return m_pQuilt->IsChartQuiltableRef(db_index);
2601}
2602
2603bool ChartCanvas::IsChartLargeEnoughToRender(ChartBase *chart, ViewPort &vp) {
2604 double chartMaxScale =
2605 chart->GetNormalScaleMax(GetCanvasScaleFactor(), GetCanvasWidth());
2606 return (chartMaxScale * g_ChartNotRenderScaleFactor > vp.chart_scale);
2607}
2608
2609void ChartCanvas::StartMeasureRoute() {
2610 if (!m_routeState) { // no measure tool if currently creating route
2611 if (m_bMeasure_Active) {
2612 g_pRouteMan->DeleteRoute(m_pMeasureRoute,
2613 NavObjectChanges::getInstance());
2614 m_pMeasureRoute = NULL;
2615 }
2616
2617 m_bMeasure_Active = true;
2618 m_nMeasureState = 1;
2619 m_bDrawingRoute = false;
2620
2621 SetCursor(*pCursorPencil);
2622 Refresh();
2623 }
2624}
2625
2626void ChartCanvas::CancelMeasureRoute() {
2627 m_bMeasure_Active = false;
2628 m_nMeasureState = 0;
2629 m_bDrawingRoute = false;
2630
2631 g_pRouteMan->DeleteRoute(m_pMeasureRoute, NavObjectChanges::getInstance());
2632 m_pMeasureRoute = NULL;
2633
2634 SetCursor(*pCursorArrow);
2635}
2636
2637ViewPort &ChartCanvas::GetVP() { return VPoint; }
2638
2639void ChartCanvas::SetVP(ViewPort &vp) {
2640 VPoint = vp;
2641 VPoint.SetPixelScale(m_displayScale);
2642}
2643
2644// void ChartCanvas::SetFocus()
2645// {
2646// printf("set %d\n", m_canvasIndex);
2647// //wxWindow:SetFocus();
2648// }
2649
2650void ChartCanvas::TriggerDeferredFocus() {
2651 // #if defined(__WXGTK__) || defined(__WXOSX__)
2652
2653 m_deferredFocusTimer.Start(20, true);
2654
2655#if defined(__WXGTK__) || defined(__WXOSX__)
2656 gFrame->Raise();
2657#endif
2658
2659 // gFrame->Raise();
2660 // #else
2661 // SetFocus();
2662 // Refresh(true);
2663 // #endif
2664}
2665
2666void ChartCanvas::OnDeferredFocusTimerEvent(wxTimerEvent &event) {
2667 SetFocus();
2668 Refresh(true);
2669}
2670
2671void ChartCanvas::OnKeyChar(wxKeyEvent &event) {
2672 if (SendKeyEventToPlugins(event))
2673 return; // PlugIn did something, and does not want the canvas to do
2674 // anything else
2675
2676 int key_char = event.GetKeyCode();
2677 switch (key_char) {
2678 case '?':
2679 HotkeysDlg(wxWindow::FindWindowByName("MainWindow")).ShowModal();
2680 break;
2681 case '+':
2682 ZoomCanvas(g_plus_minus_zoom_factor, false);
2683 break;
2684 case '-':
2685 ZoomCanvas(1.0 / g_plus_minus_zoom_factor, false);
2686 break;
2687 default:
2688 break;
2689 }
2690 if (g_benable_rotate) {
2691 switch (key_char) {
2692 case ']':
2693 RotateCanvas(1);
2694 Refresh();
2695 break;
2696
2697 case '[':
2698 RotateCanvas(-1);
2699 Refresh();
2700 break;
2701
2702 case '\\':
2703 DoRotateCanvas(0);
2704 break;
2705 }
2706 }
2707
2708 event.Skip();
2709}
2710
2711void ChartCanvas::OnKeyDown(wxKeyEvent &event) {
2712 if (SendKeyEventToPlugins(event))
2713 return; // PlugIn did something, and does not want the canvas to do
2714 // anything else
2715
2716 bool b_handled = false;
2717
2718 m_modkeys = event.GetModifiers();
2719
2720 int panspeed = m_modkeys == wxMOD_ALT ? 1 : 100;
2721
2722#ifdef OCPN_ALT_MENUBAR
2723#ifndef __WXOSX__
2724 // If the permanent menubar is disabled, we show it temporarily when Alt is
2725 // pressed or when Alt + a letter is presssed (for the top-menu-level
2726 // hotkeys). The toggling normally takes place in OnKeyUp, but here we handle
2727 // some special cases.
2728 if (IsTempMenuBarEnabled() && event.AltDown() && !g_bShowMenuBar) {
2729 // If Alt + a letter is pressed, and the menubar is hidden, show it now
2730 if (event.GetKeyCode() >= 'A' && event.GetKeyCode() <= 'Z') {
2731 if (!g_bTempShowMenuBar) {
2732 g_bTempShowMenuBar = true;
2733 parent_frame->ApplyGlobalSettings(false);
2734 }
2735 m_bMayToggleMenuBar = false; // don't hide it again when we release Alt
2736 event.Skip();
2737 return;
2738 }
2739 // If another key is pressed while Alt is down, do NOT toggle the menus when
2740 // Alt is released
2741 if (event.GetKeyCode() != WXK_ALT) {
2742 m_bMayToggleMenuBar = false;
2743 }
2744 }
2745#endif
2746#endif
2747
2748 // HOTKEYS
2749 switch (event.GetKeyCode()) {
2750 case WXK_TAB:
2751 // parent_frame->SwitchKBFocus( this );
2752 break;
2753
2754 case WXK_MENU:
2755 int x, y;
2756 event.GetPosition(&x, &y);
2757 m_FinishRouteOnKillFocus = false;
2758 CallPopupMenu(x, y);
2759 m_FinishRouteOnKillFocus = true;
2760 break;
2761
2762 case WXK_ALT:
2763 m_modkeys |= wxMOD_ALT;
2764 break;
2765
2766 case WXK_CONTROL:
2767 m_modkeys |= wxMOD_CONTROL;
2768 break;
2769
2770#ifdef __WXOSX__
2771 // On macOS Cmd generates WXK_CONTROL and Ctrl generates WXK_RAW_CONTROL
2772 case WXK_RAW_CONTROL:
2773 m_modkeys |= wxMOD_RAW_CONTROL;
2774 break;
2775#endif
2776
2777 case WXK_LEFT:
2778 if (m_modkeys == wxMOD_CONTROL)
2779 parent_frame->DoStackDown(this);
2780 else if (g_bsmoothpanzoom) {
2781 StartTimedMovement();
2782 m_panx = -1;
2783 } else {
2784 PanCanvas(-panspeed, 0);
2785 }
2786 b_handled = true;
2787 break;
2788
2789 case WXK_UP:
2790 if (g_bsmoothpanzoom) {
2791 StartTimedMovement();
2792 m_pany = -1;
2793 } else
2794 PanCanvas(0, -panspeed);
2795 b_handled = true;
2796 break;
2797
2798 case WXK_RIGHT:
2799 if (m_modkeys == wxMOD_CONTROL)
2800 parent_frame->DoStackUp(this);
2801 else if (g_bsmoothpanzoom) {
2802 StartTimedMovement();
2803 m_panx = 1;
2804 } else
2805 PanCanvas(panspeed, 0);
2806 b_handled = true;
2807
2808 break;
2809
2810 case WXK_DOWN:
2811 if (g_bsmoothpanzoom) {
2812 StartTimedMovement();
2813 m_pany = 1;
2814 } else
2815 PanCanvas(0, panspeed);
2816 b_handled = true;
2817 break;
2818
2819 case WXK_F2:
2820 TogglebFollow();
2821 break;
2822
2823 case WXK_F3: {
2824 SetShowENCText(!GetShowENCText());
2825 Refresh(true);
2826 InvalidateGL();
2827 break;
2828 }
2829 case WXK_F4:
2830 if (!m_bMeasure_Active) {
2831 if (event.ShiftDown())
2832 m_bMeasure_DistCircle = true;
2833 else
2834 m_bMeasure_DistCircle = false;
2835
2836 StartMeasureRoute();
2837 } else {
2838 CancelMeasureRoute();
2839
2840 SetCursor(*pCursorArrow);
2841
2842 // SurfaceToolbar();
2843 InvalidateGL();
2844 Refresh(false);
2845 }
2846
2847 break;
2848
2849 case WXK_F5:
2850 parent_frame->ToggleColorScheme();
2851 gFrame->Raise();
2852 TriggerDeferredFocus();
2853 break;
2854
2855 case WXK_F6: {
2856 int mod = m_modkeys & wxMOD_SHIFT;
2857 if (mod != m_brightmod) {
2858 m_brightmod = mod;
2859 m_bbrightdir = !m_bbrightdir;
2860 }
2861
2862 if (!m_bbrightdir) {
2863 g_nbrightness -= 10;
2864 if (g_nbrightness <= MIN_BRIGHT) {
2865 g_nbrightness = MIN_BRIGHT;
2866 m_bbrightdir = true;
2867 }
2868 } else {
2869 g_nbrightness += 10;
2870 if (g_nbrightness >= MAX_BRIGHT) {
2871 g_nbrightness = MAX_BRIGHT;
2872 m_bbrightdir = false;
2873 }
2874 }
2875
2876 SetScreenBrightness(g_nbrightness);
2877 ShowBrightnessLevelTimedPopup(g_nbrightness / 10, 1, 10);
2878
2879 SetFocus(); // just in case the external program steals it....
2880 gFrame->Raise(); // And reactivate the application main
2881
2882 break;
2883 }
2884
2885 case WXK_F7:
2886 parent_frame->DoStackDown(this);
2887 break;
2888
2889 case WXK_F8:
2890 parent_frame->DoStackUp(this);
2891 break;
2892
2893#ifndef __WXOSX__
2894 case WXK_F9: {
2895 double t0 = wxGetLocalTimeMillis().ToDouble();
2896 pConfig->Flush();
2897 double t1 = wxGetLocalTimeMillis().ToDouble() - t0;
2898
2899 ToggleCanvasQuiltMode();
2900 break;
2901 }
2902#endif
2903
2904 case WXK_F11:
2905 parent_frame->ToggleFullScreen();
2906 b_handled = true;
2907 break;
2908
2909 case WXK_F12: {
2910 if (m_modkeys == wxMOD_ALT) {
2911 // m_nMeasureState = *(volatile int *)(0); // generate a fault for
2912 // testing
2913 bool b = GetEnableTenHertzUpdate();
2915 UpdateGPSCompassStatusBox(true);
2916 } else
2917 ToggleChartOutlines();
2918 break;
2919 }
2920
2921 case WXK_PAUSE: // Drop MOB
2922 parent_frame->ActivateMOB();
2923 break;
2924
2925 // NUMERIC PAD
2926 case WXK_NUMPAD_ADD: // '+' on NUM PAD
2927 case WXK_PAGEUP: {
2928 ZoomCanvas(g_plus_minus_zoom_factor, false);
2929 break;
2930 }
2931 case WXK_NUMPAD_SUBTRACT: // '-' on NUM PAD
2932 case WXK_PAGEDOWN: {
2933 ZoomCanvas(1.0 / g_plus_minus_zoom_factor, false);
2934 break;
2935 }
2936 case WXK_DELETE:
2937 case WXK_BACK:
2938 if (m_bMeasure_Active) {
2939 if (m_nMeasureState > 2) {
2940 m_pMeasureRoute->DeletePoint(m_pMeasureRoute->GetLastPoint());
2941 m_pMeasureRoute->m_lastMousePointIndex =
2942 m_pMeasureRoute->GetnPoints();
2943 m_nMeasureState--;
2944 gFrame->RefreshAllCanvas();
2945 } else {
2946 CancelMeasureRoute();
2947 StartMeasureRoute();
2948 }
2949 }
2950 break;
2951 default:
2952 break;
2953 }
2954
2955 if (event.GetKeyCode() < 128) // ascii
2956 {
2957 int key_char = event.GetKeyCode();
2958
2959 // Handle both QWERTY and AZERTY keyboard separately for a few control
2960 // codes
2961 if (!g_b_assume_azerty) {
2962#ifdef __WXMAC__
2963 if (g_benable_rotate) {
2964 switch (key_char) {
2965 // On other platforms these are handled in OnKeyChar, which
2966 // (apparently) works better in some locales. On OS X it is better
2967 // to handle them here, since pressing Alt (which should change the
2968 // rotation speed) changes the key char and so prevents the keys
2969 // from working.
2970 case ']':
2971 RotateCanvas(1);
2972 b_handled = true;
2973 break;
2974
2975 case '[':
2976 RotateCanvas(-1);
2977 b_handled = true;
2978 break;
2979
2980 case '\\':
2981 DoRotateCanvas(0);
2982 b_handled = true;
2983 break;
2984 }
2985 }
2986#endif
2987 } else { // AZERTY
2988 switch (key_char) {
2989 case 43:
2990 ZoomCanvas(g_plus_minus_zoom_factor, false);
2991 break;
2992
2993 case 54: // '-' alpha/num pad
2994 // case 56: // '_' alpha/num pad
2995 ZoomCanvas(1.0 / g_plus_minus_zoom_factor, false);
2996 break;
2997 }
2998 }
2999
3000#ifdef __WXOSX__
3001 // Ctrl+Cmd+F toggles fullscreen on macOS
3002 if (key_char == 'F' && m_modkeys & wxMOD_CONTROL &&
3003 m_modkeys & wxMOD_RAW_CONTROL) {
3004 parent_frame->ToggleFullScreen();
3005 return;
3006 }
3007#endif
3008
3009 if (event.ControlDown()) key_char -= 64;
3010
3011 if (key_char >= '0' && key_char <= '9')
3012 SetGroupIndex(key_char - '0');
3013 else
3014
3015 switch (key_char) {
3016 case 'A':
3017 SetShowENCAnchor(!GetShowENCAnchor());
3018 ReloadVP();
3019
3020 break;
3021
3022 case 'C':
3023 parent_frame->ToggleColorScheme();
3024 break;
3025
3026 case 'D': {
3027 int x, y;
3028 event.GetPosition(&x, &y);
3029 ChartTypeEnum ChartType = CHART_TYPE_UNKNOWN;
3030 ChartFamilyEnum ChartFam = CHART_FAMILY_UNKNOWN;
3031 // First find out what kind of chart is being used
3032 if (!pPopupDetailSlider) {
3033 if (VPoint.b_quilt) {
3034 if (m_pQuilt) {
3035 if (m_pQuilt->GetChartAtPix(
3036 VPoint,
3037 wxPoint(
3038 x, y))) // = null if no chart loaded for this point
3039 {
3040 ChartType = m_pQuilt->GetChartAtPix(VPoint, wxPoint(x, y))
3041 ->GetChartType();
3042 ChartFam = m_pQuilt->GetChartAtPix(VPoint, wxPoint(x, y))
3043 ->GetChartFamily();
3044 }
3045 }
3046 } else {
3047 if (m_singleChart) {
3048 ChartType = m_singleChart->GetChartType();
3049 ChartFam = m_singleChart->GetChartFamily();
3050 }
3051 }
3052 // If a charttype is found show the popupslider
3053 if ((ChartType != CHART_TYPE_UNKNOWN) ||
3054 (ChartFam != CHART_FAMILY_UNKNOWN)) {
3055 pPopupDetailSlider = new PopUpDSlide(
3056 this, -1, ChartType, ChartFam,
3057 wxPoint(g_detailslider_dialog_x, g_detailslider_dialog_y),
3058 wxDefaultSize, wxSIMPLE_BORDER, _T(""));
3059 if (pPopupDetailSlider) pPopupDetailSlider->Show();
3060 }
3061 } else //( !pPopupDetailSlider ) close popupslider
3062 {
3063 if (pPopupDetailSlider) pPopupDetailSlider->Close();
3064 pPopupDetailSlider = NULL;
3065 }
3066 break;
3067 }
3068
3069 case 'E':
3070 if (!wxWindow::FindWindowByName("NmeaDebugWindow")) {
3071 auto top_window = wxWindow::FindWindowByName(kTopLevelWindowName);
3072 NMEALogWindow::GetInstance().Create(top_window, 35);
3073 }
3074 wxWindow::FindWindowByName("NmeaDebugWindow")->Show();
3075 break;
3076
3077 case 'L':
3078 SetShowENCLights(!GetShowENCLights());
3079 ReloadVP();
3080
3081 break;
3082
3083 case 'M':
3084 if (event.ShiftDown())
3085 m_bMeasure_DistCircle = true;
3086 else
3087 m_bMeasure_DistCircle = false;
3088
3089 StartMeasureRoute();
3090 break;
3091
3092 case 'N':
3093 if (g_bInlandEcdis && ps52plib) {
3094 SetENCDisplayCategory((_DisCat)STANDARD);
3095 }
3096 break;
3097
3098 case 'O':
3099 ToggleChartOutlines();
3100 break;
3101
3102 case 'Q':
3103 ToggleCanvasQuiltMode();
3104 break;
3105
3106 case 'P':
3107 parent_frame->ToggleTestPause();
3108 break;
3109 case 'R':
3110 g_bNavAidRadarRingsShown = !g_bNavAidRadarRingsShown;
3111 if (g_bNavAidRadarRingsShown && g_iNavAidRadarRingsNumberVisible == 0)
3112 g_iNavAidRadarRingsNumberVisible = 1;
3113 else if (!g_bNavAidRadarRingsShown &&
3114 g_iNavAidRadarRingsNumberVisible == 1)
3115 g_iNavAidRadarRingsNumberVisible = 0;
3116 break;
3117 case 'S':
3118 SetShowENCDepth(!m_encShowDepth);
3119 ReloadVP();
3120 break;
3121
3122 case 'T':
3123 SetShowENCText(!GetShowENCText());
3124 ReloadVP();
3125 break;
3126
3127 case 'U':
3128 SetShowENCDataQual(!GetShowENCDataQual());
3129 ReloadVP();
3130 break;
3131
3132 case 'V':
3133 m_bShowNavobjects = !m_bShowNavobjects;
3134 Refresh(true);
3135 break;
3136
3137 case 'W': // W Toggle CPA alarm
3138 ToggleCPAWarn();
3139
3140 break;
3141
3142 case 1: // Ctrl A
3143 TogglebFollow();
3144
3145 break;
3146
3147 case 2: // Ctrl B
3148 if (g_bShowMenuBar == false) parent_frame->ToggleChartBar(this);
3149 break;
3150
3151 case 13: // Ctrl M // Drop Marker at cursor
3152 {
3153 if (event.ControlDown()) gFrame->DropMarker(false);
3154 break;
3155 }
3156
3157 case 14: // Ctrl N - Activate next waypoint in a route
3158 {
3159 if (Route *r = g_pRouteMan->GetpActiveRoute()) {
3160 int indexActive = r->GetIndexOf(r->m_pRouteActivePoint);
3161 if ((indexActive + 1) <= r->GetnPoints()) {
3162 g_pRouteMan->ActivateNextPoint(r, true);
3163 InvalidateGL();
3164 Refresh(false);
3165 }
3166 }
3167 break;
3168 }
3169
3170 case 15: // Ctrl O - Drop Marker at boat's position
3171 {
3172 if (!g_bShowMenuBar) gFrame->DropMarker(true);
3173 break;
3174 }
3175
3176 case 32: // Special needs use space bar
3177 {
3178 if (g_bSpaceDropMark) gFrame->DropMarker(true);
3179 break;
3180 }
3181
3182 case -32: // Ctrl Space // Drop MOB
3183 {
3184 if (m_modkeys == wxMOD_CONTROL) parent_frame->ActivateMOB();
3185
3186 break;
3187 }
3188
3189 case -20: // Ctrl ,
3190 {
3191 parent_frame->DoSettings();
3192 break;
3193 }
3194 case 17: // Ctrl Q
3195 parent_frame->Close();
3196 return;
3197
3198 case 18: // Ctrl R
3199 StartRoute();
3200 return;
3201
3202 case 20: // Ctrl T
3203 if (NULL == pGoToPositionDialog) // There is one global instance of
3204 // the Go To Position Dialog
3205 pGoToPositionDialog = new GoToPositionDialog(this);
3206 pGoToPositionDialog->SetCanvas(this);
3207 pGoToPositionDialog->Show();
3208 break;
3209
3210 case 25: // Ctrl Y
3211 if (undo->AnythingToRedo()) {
3212 undo->RedoNextAction();
3213 InvalidateGL();
3214 Refresh(false);
3215 }
3216 break;
3217
3218 case 26:
3219 if (event.ShiftDown()) { // Shift-Ctrl-Z
3220 if (undo->AnythingToRedo()) {
3221 undo->RedoNextAction();
3222 InvalidateGL();
3223 Refresh(false);
3224 }
3225 } else { // Ctrl Z
3226 if (undo->AnythingToUndo()) {
3227 undo->UndoLastAction();
3228 InvalidateGL();
3229 Refresh(false);
3230 }
3231 }
3232 break;
3233
3234 case 27:
3235 // Generic break
3236 if (m_bMeasure_Active) {
3237 CancelMeasureRoute();
3238
3239 SetCursor(*pCursorArrow);
3240
3241 // SurfaceToolbar();
3242 gFrame->RefreshAllCanvas();
3243 }
3244
3245 if (m_routeState) // creating route?
3246 {
3247 FinishRoute();
3248 // SurfaceToolbar();
3249 InvalidateGL();
3250 Refresh(false);
3251 }
3252
3253 break;
3254
3255 case 7: // Ctrl G
3256 switch (gamma_state) {
3257 case (0):
3258 r_gamma_mult = 0;
3259 g_gamma_mult = 1;
3260 b_gamma_mult = 0;
3261 gamma_state = 1;
3262 break;
3263 case (1):
3264 r_gamma_mult = 1;
3265 g_gamma_mult = 0;
3266 b_gamma_mult = 0;
3267 gamma_state = 2;
3268 break;
3269 case (2):
3270 r_gamma_mult = 1;
3271 g_gamma_mult = 1;
3272 b_gamma_mult = 1;
3273 gamma_state = 0;
3274 break;
3275 }
3276 SetScreenBrightness(g_nbrightness);
3277
3278 break;
3279
3280 case 9: // Ctrl I
3281 if (event.ControlDown()) {
3282 m_bShowCompassWin = !m_bShowCompassWin;
3283 SetShowGPSCompassWindow(m_bShowCompassWin);
3284 Refresh(false);
3285 }
3286 break;
3287
3288 default:
3289 break;
3290
3291 } // switch
3292 }
3293
3294 // Allow OnKeyChar to catch the key events too.
3295 if (!b_handled) {
3296 event.Skip();
3297 }
3298}
3299
3300void ChartCanvas::OnKeyUp(wxKeyEvent &event) {
3301 if (SendKeyEventToPlugins(event))
3302 return; // PlugIn did something, and does not want the canvas to do
3303 // anything else
3304
3305 switch (event.GetKeyCode()) {
3306 case WXK_TAB:
3307 parent_frame->SwitchKBFocus(this);
3308 break;
3309
3310 case WXK_LEFT:
3311 case WXK_RIGHT:
3312 m_panx = 0;
3313 if (!m_pany) m_panspeed = 0;
3314 break;
3315
3316 case WXK_UP:
3317 case WXK_DOWN:
3318 m_pany = 0;
3319 if (!m_panx) m_panspeed = 0;
3320 break;
3321
3322 case WXK_NUMPAD_ADD: // '+' on NUM PAD
3323 case WXK_NUMPAD_SUBTRACT: // '-' on NUM PAD
3324 case WXK_PAGEUP:
3325 case WXK_PAGEDOWN:
3326 if (m_mustmove) DoMovement(m_mustmove);
3327
3328 m_zoom_factor = 1;
3329 break;
3330
3331 case WXK_ALT:
3332 m_modkeys &= ~wxMOD_ALT;
3333#ifdef OCPN_ALT_MENUBAR
3334#ifndef __WXOSX__
3335 // If the permanent menu bar is disabled, and we are not in the middle of
3336 // another key combo, then show the menu bar temporarily when Alt is
3337 // released (or hide it if already visible).
3338 if (IsTempMenuBarEnabled() && !g_bShowMenuBar && m_bMayToggleMenuBar) {
3339 g_bTempShowMenuBar = !g_bTempShowMenuBar;
3340 parent_frame->ApplyGlobalSettings(false);
3341 }
3342 m_bMayToggleMenuBar = true;
3343#endif
3344#endif
3345 break;
3346
3347 case WXK_CONTROL:
3348 m_modkeys &= ~wxMOD_CONTROL;
3349 break;
3350 }
3351
3352 if (event.GetKeyCode() < 128) // ascii
3353 {
3354 int key_char = event.GetKeyCode();
3355
3356 // Handle both QWERTY and AZERTY keyboard separately for a few control
3357 // codes
3358 if (!g_b_assume_azerty) {
3359 switch (key_char) {
3360 case '+':
3361 case '=':
3362 case '-':
3363 case '_':
3364 case 54:
3365 case 56: // '_' alpha/num pad
3366 DoMovement(m_mustmove);
3367
3368 // m_zoom_factor = 1;
3369 break;
3370 case '[':
3371 case ']':
3372 DoMovement(m_mustmove);
3373 m_rotation_speed = 0;
3374 break;
3375 }
3376 } else {
3377 switch (key_char) {
3378 case 43:
3379 case 54: // '-' alpha/num pad
3380 case 56: // '_' alpha/num pad
3381 DoMovement(m_mustmove);
3382
3383 m_zoom_factor = 1;
3384 break;
3385 }
3386 }
3387 }
3388 event.Skip();
3389}
3390
3391void ChartCanvas::ToggleChartOutlines(void) {
3392 m_bShowOutlines = !m_bShowOutlines;
3393
3394 Refresh(false);
3395
3396#ifdef ocpnUSE_GL // opengl renders chart outlines as part of the chart this
3397 // needs a full refresh
3398 if (g_bopengl) InvalidateGL();
3399#endif
3400}
3401
3402void ChartCanvas::ToggleLookahead() {
3403 m_bLookAhead = !m_bLookAhead;
3404 m_OSoffsetx = 0; // center ownship
3405 m_OSoffsety = 0;
3406}
3407
3408void ChartCanvas::SetUpMode(int mode) {
3409 m_upMode = mode;
3410
3411 if (mode != NORTH_UP_MODE) {
3412 // Stuff the COGAvg table in case COGUp is selected
3413 double stuff = 0;
3414 if (!std::isnan(gCog)) stuff = gCog;
3415
3416 if (g_COGAvgSec > 0) {
3417 for (int i = 0; i < g_COGAvgSec; i++) gFrame->COGTable[i] = stuff;
3418 }
3419 g_COGAvg = stuff;
3420 gFrame->FrameCOGTimer.Start(100, wxTIMER_CONTINUOUS);
3421 } else {
3422 if (!g_bskew_comp && (fabs(GetVPSkew()) > 0.0001))
3423 SetVPRotation(GetVPSkew());
3424 else
3425 SetVPRotation(0); /* reset to north up */
3426 }
3427
3428 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
3429 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
3430
3431 UpdateGPSCompassStatusBox(true);
3432 gFrame->DoChartUpdate();
3433}
3434
3435bool ChartCanvas::DoCanvasCOGSet(void) {
3436 if (GetUpMode() == NORTH_UP_MODE) return false;
3437 double cog_use = g_COGAvg;
3438 if (g_btenhertz) cog_use = gCog;
3439
3440 double rotation = 0;
3441 if ((GetUpMode() == HEAD_UP_MODE) && !std::isnan(gHdt)) {
3442 rotation = -gHdt * PI / 180.;
3443 } else if ((GetUpMode() == COURSE_UP_MODE) && !std::isnan(cog_use))
3444 rotation = -cog_use * PI / 180.;
3445
3446 SetVPRotation(rotation);
3447 return true;
3448}
3449
3450double easeOutCubic(double t) {
3451 // Starts quickly and slows down toward the end
3452 return 1.0 - pow(1.0 - t, 3.0);
3453}
3454
3455void ChartCanvas::StartChartDragInertia() {
3456 //
3457 // printf("\nStart ChartDragInertia\n");
3458 m_bChartDragging = false;
3459
3460 // Set some parameters
3461 m_chart_drag_inertia_time = 750; // msec
3462 m_chart_drag_inertia_start_time = wxGetLocalTimeMillis();
3463 m_last_elapsed = 0;
3464
3465 // Calculate ending drag velocity
3466 size_t n_vel = 10;
3467 n_vel = wxMin(n_vel, m_drag_vec_t.size());
3468 int xacc = 0;
3469 int yacc = 0;
3470 double tacc = 0;
3471 size_t length = m_drag_vec_t.size();
3472 for (size_t i = 0; i < n_vel; i++) {
3473 xacc += m_drag_vec_x.at(length - 1 - i);
3474 yacc += m_drag_vec_y.at(length - 1 - i);
3475 tacc += m_drag_vec_t.at(length - 1 - i);
3476 // printf("%d %g\n", xacc, tacc);
3477 }
3478 m_chart_drag_velocity_x = xacc / tacc;
3479 m_chart_drag_velocity_y = yacc / tacc;
3480
3481 m_chart_drag_inertia_active = true;
3482
3483 // First callback as fast as possible.
3484 m_chart_drag_inertia_timer.Start(1, wxTIMER_ONE_SHOT);
3485
3486 // printf(" Drag parms %d %d %g\n", m_chart_drag_total_x,
3487 // m_chart_drag_total_y, m_chart_drag_total_time);
3488}
3489
3490void ChartCanvas::OnChartDragInertiaTimer(wxTimerEvent &event) {
3491 if (!m_chart_drag_inertia_active) return;
3492
3493 // Calculate time fraction from 0..1
3494 wxLongLong now = wxGetLocalTimeMillis();
3495 double elapsed = (now - m_chart_drag_inertia_start_time).ToDouble();
3496 double t = elapsed / m_chart_drag_inertia_time.ToDouble();
3497 if (t > 1.0) t = 1.0;
3498 double e = 1.0 - easeOutCubic(t); // 0..1
3499
3500 double dx =
3501 m_chart_drag_velocity_x * ((elapsed - m_last_elapsed) / 1000.) * e;
3502 double dy =
3503 m_chart_drag_velocity_y * ((elapsed - m_last_elapsed) / 1000.) * e;
3504
3505 // double distance = pow((pow(dx, 2) + pow(dy, 2)), 0.5);
3506 // printf(" %5g %5g %5g %5g pix/sec\n", elapsed,
3507 // elapsed - m_last_elapsed, distance, distance * 1000 / elapsed);
3508
3509 m_last_elapsed = elapsed;
3510
3511 // Ensure that target destination lies on whole-pixel boundary
3512 // This allows the render engine to use a faster FBO copy method for drawing
3513 double destination_x = (GetCanvasWidth() / 2) + wxRound(dx);
3514 double destination_y = (GetCanvasHeight() / 2) + wxRound(dy);
3515 double inertia_lat, inertia_lon;
3516 GetCanvasPixPoint(destination_x, destination_y, inertia_lat, inertia_lon);
3517 SetViewPoint(inertia_lat, inertia_lon); // about 1 msec
3518
3519 Refresh(false);
3520
3521 // Stop condition
3522 if ((t >= 1) || (fabs(dx) < 1) || (fabs(dy) < 1)) {
3523 m_chart_drag_inertia_timer.Stop();
3524 m_chart_drag_inertia_active = false;
3525
3526 // Disable chart pan movement logic
3527 m_target_lat = GetVP().clat;
3528 m_target_lon = GetVP().clon;
3529 m_pan_drag.x = m_pan_drag.y = 0;
3530 m_panx = m_pany = 0;
3531
3532 } else {
3533 int target_redraw_interval = 40; // msec
3534 m_chart_drag_inertia_timer.Start(target_redraw_interval, wxTIMER_ONE_SHOT);
3535 }
3536}
3537
3538void ChartCanvas::StopMovement() {
3539 m_panx = m_pany = 0;
3540 m_panspeed = 0;
3541 m_zoom_factor = 1;
3542 m_rotation_speed = 0;
3543 m_mustmove = 0;
3544#if 0
3545#if !defined(__WXGTK__) && !defined(__WXQT__)
3546 SetFocus();
3547 gFrame->Raise();
3548#endif
3549#endif
3550}
3551
3552/* instead of integrating in timer callbacks
3553 (which do not always get called fast enough)
3554 we can perform the integration of movement
3555 at each render frame based on the time change */
3556bool ChartCanvas::StartTimedMovement(bool stoptimer) {
3557 // Start/restart the stop movement timer
3558 if (stoptimer) pMovementStopTimer->Start(800, wxTIMER_ONE_SHOT);
3559
3560 if (!pMovementTimer->IsRunning()) {
3561 // printf("timer not running, starting\n");
3562 pMovementTimer->Start(1, wxTIMER_ONE_SHOT);
3563 }
3564
3565 if (m_panx || m_pany || m_zoom_factor != 1 || m_rotation_speed) {
3566 // already moving, gets called again because of key-repeat event
3567 return false;
3568 }
3569
3570 m_last_movement_time = wxDateTime::UNow();
3571
3572 return true;
3573}
3574void ChartCanvas::StartTimedMovementVP(double target_lat, double target_lon,
3575 int nstep) {
3576 // Save the target
3577 m_target_lat = target_lat;
3578 m_target_lon = target_lon;
3579
3580 // Save the start point
3581 m_start_lat = GetVP().clat;
3582 m_start_lon = GetVP().clon;
3583
3584 m_VPMovementTimer.Start(1, true); // oneshot
3585 m_timed_move_vp_active = true;
3586 m_stvpc = 0;
3587 m_timedVP_step = nstep;
3588}
3589
3590void ChartCanvas::DoTimedMovementVP() {
3591 if (!m_timed_move_vp_active) return; // not active
3592 if (m_stvpc++ > m_timedVP_step * 2) { // Backstop
3593 StopMovement();
3594 return;
3595 }
3596 // Stop condition
3597 double one_pix = (1. / (1852 * 60)) / GetVP().view_scale_ppm;
3598 double d2 =
3599 pow(m_run_lat - m_target_lat, 2) + pow(m_run_lon - m_target_lon, 2);
3600 d2 = pow(d2, 0.5);
3601
3602 if (d2 < one_pix) {
3603 SetViewPoint(m_target_lat, m_target_lon); // Embeds a refresh
3604 StopMovementVP();
3605 return;
3606 }
3607
3608 // if ((fabs(m_run_lat - m_target_lat) < one_pix) &&
3609 // (fabs(m_run_lon - m_target_lon) < one_pix)) {
3610 // StopMovementVP();
3611 // return;
3612 // }
3613
3614 double new_lat = GetVP().clat + (m_target_lat - m_start_lat) / m_timedVP_step;
3615 double new_lon = GetVP().clon + (m_target_lon - m_start_lon) / m_timedVP_step;
3616
3617 m_run_lat = new_lat;
3618 m_run_lon = new_lon;
3619
3620 // printf(" Timed\n");
3621 SetViewPoint(new_lat, new_lon); // Embeds a refresh
3622}
3623
3624void ChartCanvas::StopMovementVP() { m_timed_move_vp_active = false; }
3625
3626void ChartCanvas::MovementVPTimerEvent(wxTimerEvent &) { DoTimedMovementVP(); }
3627
3628void ChartCanvas::StartTimedMovementTarget() {}
3629
3630void ChartCanvas::DoTimedMovementTarget() {}
3631
3632void ChartCanvas::StopMovementTarget() {}
3633
3634void ChartCanvas::DoTimedMovement() {
3635 if (m_pan_drag == wxPoint(0, 0) && !m_panx && !m_pany && m_zoom_factor == 1 &&
3636 !m_rotation_speed)
3637 return; /* not moving */
3638
3639 wxDateTime now = wxDateTime::UNow();
3640 long dt = 0;
3641 if (m_last_movement_time.IsValid())
3642 dt = (now - m_last_movement_time).GetMilliseconds().ToLong();
3643
3644 m_last_movement_time = now;
3645
3646 if (dt > 500) /* if we are running very slow, don't integrate too fast */
3647 dt = 500;
3648
3649 DoMovement(dt);
3650}
3651
3652void ChartCanvas::DoMovement(long dt) {
3653 /* if we get here quickly assume 1ms so that some movement occurs */
3654 if (dt == 0) dt = 1;
3655
3656 m_mustmove -= dt;
3657 if (m_mustmove < 0) m_mustmove = 0;
3658
3659 if (m_pan_drag.x || m_pan_drag.y) {
3660 PanCanvas(m_pan_drag.x, m_pan_drag.y);
3661 m_pan_drag.x = m_pan_drag.y = 0;
3662 }
3663
3664 if (m_panx || m_pany) {
3665 const double slowpan = .1, maxpan = 2;
3666 if (m_modkeys == wxMOD_ALT)
3667 m_panspeed = slowpan;
3668 else {
3669 m_panspeed += (double)dt / 500; /* apply acceleration */
3670 m_panspeed = wxMin(maxpan, m_panspeed);
3671 }
3672 PanCanvas(m_panspeed * m_panx * dt, m_panspeed * m_pany * dt);
3673 }
3674
3675 if (m_zoom_factor != 1) {
3676 double alpha = 400, beta = 1.5;
3677 double zoom_factor = (exp(dt / alpha) - 1) / beta + 1;
3678
3679 if (m_modkeys == wxMOD_ALT) zoom_factor = pow(zoom_factor, .15);
3680
3681 if (m_zoom_factor < 1) zoom_factor = 1 / zoom_factor;
3682
3683 // Try to hit the zoom target exactly.
3684 // if(m_wheelzoom_stop_oneshot > 0)
3685 {
3686 if (zoom_factor > 1) {
3687 if (VPoint.chart_scale / zoom_factor <= m_zoom_target)
3688 zoom_factor = VPoint.chart_scale / m_zoom_target;
3689 }
3690
3691 else if (zoom_factor < 1) {
3692 if (VPoint.chart_scale / zoom_factor >= m_zoom_target)
3693 zoom_factor = VPoint.chart_scale / m_zoom_target;
3694 }
3695 }
3696
3697 if (fabs(zoom_factor - 1) > 1e-4)
3698 DoZoomCanvas(zoom_factor, m_bzooming_to_cursor);
3699
3700 if (m_wheelzoom_stop_oneshot > 0) {
3701 if (m_wheelstopwatch.Time() > m_wheelzoom_stop_oneshot) {
3702 m_wheelzoom_stop_oneshot = 0;
3703 StopMovement();
3704 }
3705
3706 // Don't overshoot the zoom target.
3707 if (zoom_factor > 1) {
3708 if (VPoint.chart_scale <= m_zoom_target) {
3709 m_wheelzoom_stop_oneshot = 0;
3710 StopMovement();
3711 }
3712 } else if (zoom_factor < 1) {
3713 if (VPoint.chart_scale >= m_zoom_target) {
3714 m_wheelzoom_stop_oneshot = 0;
3715 StopMovement();
3716 }
3717 }
3718 }
3719 }
3720
3721 if (m_rotation_speed) { /* in degrees per second */
3722 double speed = m_rotation_speed;
3723 if (m_modkeys == wxMOD_ALT) speed /= 10;
3724 DoRotateCanvas(VPoint.rotation + speed * PI / 180 * dt / 1000.0);
3725 }
3726}
3727
3728void ChartCanvas::SetColorScheme(ColorScheme cs) {
3729 SetAlertString(_T(""));
3730
3731 // Setup ownship image pointers
3732 switch (cs) {
3733 case GLOBAL_COLOR_SCHEME_DAY:
3734 m_pos_image_red = &m_os_image_red_day;
3735 m_pos_image_grey = &m_os_image_grey_day;
3736 m_pos_image_yellow = &m_os_image_yellow_day;
3737 m_pos_image_user = m_pos_image_user_day;
3738 m_pos_image_user_grey = m_pos_image_user_grey_day;
3739 m_pos_image_user_yellow = m_pos_image_user_yellow_day;
3740 m_cTideBitmap = m_bmTideDay;
3741 m_cCurrentBitmap = m_bmCurrentDay;
3742
3743 break;
3744 case GLOBAL_COLOR_SCHEME_DUSK:
3745 m_pos_image_red = &m_os_image_red_dusk;
3746 m_pos_image_grey = &m_os_image_grey_dusk;
3747 m_pos_image_yellow = &m_os_image_yellow_dusk;
3748 m_pos_image_user = m_pos_image_user_dusk;
3749 m_pos_image_user_grey = m_pos_image_user_grey_dusk;
3750 m_pos_image_user_yellow = m_pos_image_user_yellow_dusk;
3751 m_cTideBitmap = m_bmTideDusk;
3752 m_cCurrentBitmap = m_bmCurrentDusk;
3753 break;
3754 case GLOBAL_COLOR_SCHEME_NIGHT:
3755 m_pos_image_red = &m_os_image_red_night;
3756 m_pos_image_grey = &m_os_image_grey_night;
3757 m_pos_image_yellow = &m_os_image_yellow_night;
3758 m_pos_image_user = m_pos_image_user_night;
3759 m_pos_image_user_grey = m_pos_image_user_grey_night;
3760 m_pos_image_user_yellow = m_pos_image_user_yellow_night;
3761 m_cTideBitmap = m_bmTideNight;
3762 m_cCurrentBitmap = m_bmCurrentNight;
3763 break;
3764 default:
3765 m_pos_image_red = &m_os_image_red_day;
3766 m_pos_image_grey = &m_os_image_grey_day;
3767 m_pos_image_yellow = &m_os_image_yellow_day;
3768 m_pos_image_user = m_pos_image_user_day;
3769 m_pos_image_user_grey = m_pos_image_user_grey_day;
3770 m_pos_image_user_yellow = m_pos_image_user_yellow_day;
3771 m_cTideBitmap = m_bmTideDay;
3772 m_cCurrentBitmap = m_bmCurrentDay;
3773 break;
3774 }
3775
3776 CreateDepthUnitEmbossMaps(cs);
3777 CreateOZEmbossMapData(cs);
3778
3779 // Set up fog effect base color
3780 m_fog_color = wxColor(
3781 170, 195, 240); // this is gshhs (backgound world chart) ocean color
3782 float dim = 1.0;
3783 switch (cs) {
3784 case GLOBAL_COLOR_SCHEME_DUSK:
3785 dim = 0.5;
3786 break;
3787 case GLOBAL_COLOR_SCHEME_NIGHT:
3788 dim = 0.25;
3789 break;
3790 default:
3791 break;
3792 }
3793 m_fog_color.Set(m_fog_color.Red() * dim, m_fog_color.Green() * dim,
3794 m_fog_color.Blue() * dim);
3795
3796 // Really dark
3797#if 0
3798 if( cs == GLOBAL_COLOR_SCHEME_DUSK || cs == GLOBAL_COLOR_SCHEME_NIGHT ) {
3799 SetBackgroundColour( wxColour(0,0,0) );
3800
3801 SetWindowStyleFlag( (GetWindowStyleFlag() & ~wxSIMPLE_BORDER) | wxNO_BORDER);
3802 }
3803 else{
3804 SetWindowStyleFlag( (GetWindowStyleFlag() & ~wxNO_BORDER) | wxSIMPLE_BORDER);
3805#ifndef __WXMAC__
3806 SetBackgroundColour( wxNullColour );
3807#endif
3808 }
3809#endif
3810
3811 // UpdateToolbarColorScheme(cs);
3812
3813 m_Piano->SetColorScheme(cs);
3814
3815 m_Compass->SetColorScheme(cs);
3816
3817 if (m_muiBar) m_muiBar->SetColorScheme(cs);
3818
3819 if (pWorldBackgroundChart) pWorldBackgroundChart->SetColorScheme(cs);
3820#ifdef ocpnUSE_GL
3821 if (g_bopengl && m_glcc) {
3822 m_glcc->SetColorScheme(cs);
3823 g_glTextureManager->ClearAllRasterTextures();
3824 // m_glcc->FlushFBO();
3825 }
3826#endif
3827 SetbTCUpdate(true); // force re-render of tide/current locators
3828 m_brepaint_piano = true;
3829
3830 ReloadVP();
3831
3832 m_cs = cs;
3833}
3834
3835wxBitmap ChartCanvas::CreateDimBitmap(wxBitmap &Bitmap, double factor) {
3836 wxImage img = Bitmap.ConvertToImage();
3837 int sx = img.GetWidth();
3838 int sy = img.GetHeight();
3839
3840 wxImage new_img(img);
3841
3842 for (int i = 0; i < sx; i++) {
3843 for (int j = 0; j < sy; j++) {
3844 if (!img.IsTransparent(i, j)) {
3845 new_img.SetRGB(i, j, (unsigned char)(img.GetRed(i, j) * factor),
3846 (unsigned char)(img.GetGreen(i, j) * factor),
3847 (unsigned char)(img.GetBlue(i, j) * factor));
3848 }
3849 }
3850 }
3851
3852 wxBitmap ret = wxBitmap(new_img);
3853
3854 return ret;
3855}
3856
3857void ChartCanvas::ShowBrightnessLevelTimedPopup(int brightness, int min,
3858 int max) {
3859 wxFont *pfont = FontMgr::Get().FindOrCreateFont(
3860 40, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_BOLD);
3861
3862 if (!m_pBrightPopup) {
3863 // Calculate size
3864 int x, y;
3865 GetTextExtent(_T("MAX"), &x, &y, NULL, NULL, pfont);
3866
3867 m_pBrightPopup = new TimedPopupWin(this, 3);
3868
3869 m_pBrightPopup->SetSize(x, y);
3870 m_pBrightPopup->Move(120, 120);
3871 }
3872
3873 int bmpsx = m_pBrightPopup->GetSize().x;
3874 int bmpsy = m_pBrightPopup->GetSize().y;
3875
3876 wxBitmap bmp(bmpsx, bmpsx);
3877 wxMemoryDC mdc(bmp);
3878
3879 mdc.SetTextForeground(GetGlobalColor(_T("GREEN4")));
3880 mdc.SetBackground(wxBrush(GetGlobalColor(_T("UINFD"))));
3881 mdc.SetPen(wxPen(wxColour(0, 0, 0)));
3882 mdc.SetBrush(wxBrush(GetGlobalColor(_T("UINFD"))));
3883 mdc.Clear();
3884
3885 mdc.DrawRectangle(0, 0, bmpsx, bmpsy);
3886
3887 mdc.SetFont(*pfont);
3888 wxString val;
3889
3890 if (brightness == max)
3891 val = _T("MAX");
3892 else if (brightness == min)
3893 val = _T("MIN");
3894 else
3895 val.Printf(_T("%3d"), brightness);
3896
3897 mdc.DrawText(val, 0, 0);
3898
3899 mdc.SelectObject(wxNullBitmap);
3900
3901 m_pBrightPopup->SetBitmap(bmp);
3902 m_pBrightPopup->Show();
3903 m_pBrightPopup->Refresh();
3904}
3905
3906void ChartCanvas::RotateTimerEvent(wxTimerEvent &event) {
3907 m_b_rot_hidef = true;
3908 ReloadVP();
3909}
3910
3911void ChartCanvas::OnRolloverPopupTimerEvent(wxTimerEvent &event) {
3912 if (!g_bRollover) return;
3913
3914 bool b_need_refresh = false;
3915
3916 wxSize win_size = GetSize() * m_displayScale;
3917 if (console && console->IsShown()) win_size.x -= console->GetSize().x;
3918
3919 // Handle the AIS Rollover Window first
3920 bool showAISRollover = false;
3921 if (g_pAIS && g_pAIS->GetNumTargets() && m_bShowAIS) {
3922 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
3923 SelectItem *pFind = pSelectAIS->FindSelection(
3924 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_AISTARGET);
3925 if (pFind) {
3926 int FoundAIS_MMSI = (wxIntPtr)pFind->m_pData1;
3927 auto ptarget = g_pAIS->Get_Target_Data_From_MMSI(FoundAIS_MMSI);
3928
3929 if (ptarget) {
3930 showAISRollover = true;
3931
3932 if (NULL == m_pAISRolloverWin) {
3933 m_pAISRolloverWin = new RolloverWin(this);
3934 m_pAISRolloverWin->IsActive(false);
3935 b_need_refresh = true;
3936 } else if (m_pAISRolloverWin->IsActive() && m_AISRollover_MMSI &&
3937 m_AISRollover_MMSI != FoundAIS_MMSI) {
3938 // Sometimes the mouse moves fast enough to get over a new AIS
3939 // target before the one-shot has fired to remove the old target.
3940 // Result: wrong target data is shown.
3941 // Detect this case,close the existing rollover ASAP, and restart
3942 // the timer.
3943 m_RolloverPopupTimer.Start(50, wxTIMER_ONE_SHOT);
3944 m_pAISRolloverWin->IsActive(false);
3945 m_AISRollover_MMSI = 0;
3946 Refresh();
3947 return;
3948 }
3949
3950 m_AISRollover_MMSI = FoundAIS_MMSI;
3951
3952 if (!m_pAISRolloverWin->IsActive()) {
3953 wxString s = ptarget->GetRolloverString();
3954 m_pAISRolloverWin->SetString(s);
3955
3956 m_pAISRolloverWin->SetBestPosition(mouse_x, mouse_y, 16, 16,
3957 AIS_ROLLOVER, win_size);
3958 m_pAISRolloverWin->SetBitmap(AIS_ROLLOVER);
3959 m_pAISRolloverWin->IsActive(true);
3960 b_need_refresh = true;
3961 }
3962 }
3963 } else {
3964 m_AISRollover_MMSI = 0;
3965 showAISRollover = false;
3966 }
3967 }
3968
3969 // Maybe turn the rollover off
3970 if (m_pAISRolloverWin && m_pAISRolloverWin->IsActive() && !showAISRollover) {
3971 m_pAISRolloverWin->IsActive(false);
3972 m_AISRollover_MMSI = 0;
3973 b_need_refresh = true;
3974 }
3975
3976 // Now the Route info rollover
3977 // Show the route segment info
3978 bool showRouteRollover = false;
3979
3980 if (NULL == m_pRolloverRouteSeg) {
3981 // Get a list of all selectable sgements, and search for the first
3982 // visible segment as the rollover target.
3983
3984 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
3985 SelectableItemList SelList = pSelect->FindSelectionList(
3986 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_ROUTESEGMENT);
3987 wxSelectableItemListNode *node = SelList.GetFirst();
3988 while (node) {
3989 SelectItem *pFindSel = node->GetData();
3990
3991 Route *pr = (Route *)pFindSel->m_pData3; // candidate
3992
3993 if (pr && pr->IsVisible()) {
3994 m_pRolloverRouteSeg = pFindSel;
3995 showRouteRollover = true;
3996
3997 if (NULL == m_pRouteRolloverWin) {
3998 m_pRouteRolloverWin = new RolloverWin(this, 10);
3999 m_pRouteRolloverWin->IsActive(false);
4000 }
4001
4002 if (!m_pRouteRolloverWin->IsActive()) {
4003 wxString s;
4004 RoutePoint *segShow_point_a =
4005 (RoutePoint *)m_pRolloverRouteSeg->m_pData1;
4006 RoutePoint *segShow_point_b =
4007 (RoutePoint *)m_pRolloverRouteSeg->m_pData2;
4008
4009 double brg, dist;
4010 DistanceBearingMercator(
4011 segShow_point_b->m_lat, segShow_point_b->m_lon,
4012 segShow_point_a->m_lat, segShow_point_a->m_lon, &brg, &dist);
4013
4014 if (!pr->m_bIsInLayer)
4015 s.Append(_("Route") + _T(": "));
4016 else
4017 s.Append(_("Layer Route: "));
4018
4019 if (pr->m_RouteNameString.IsEmpty())
4020 s.Append(_("(unnamed)"));
4021 else
4022 s.Append(pr->m_RouteNameString);
4023
4024 s << _T("\n") << _("Total Length: ")
4025 << FormatDistanceAdaptive(pr->m_route_length) << _T("\n")
4026 << _("Leg: from ") << segShow_point_a->GetName() << _(" to ")
4027 << segShow_point_b->GetName() << _T("\n");
4028
4029 if (g_bShowTrue)
4030 s << wxString::Format(wxString("%03d%c(T) ", wxConvUTF8),
4031 (int)floor(brg + 0.5), 0x00B0);
4032 if (g_bShowMag) {
4033 double latAverage =
4034 (segShow_point_b->m_lat + segShow_point_a->m_lat) / 2;
4035 double lonAverage =
4036 (segShow_point_b->m_lon + segShow_point_a->m_lon) / 2;
4037 double varBrg = gFrame->GetMag(brg, latAverage, lonAverage);
4038
4039 s << wxString::Format(wxString("%03d%c(M) ", wxConvUTF8),
4040 (int)floor(varBrg + 0.5), 0x00B0);
4041 }
4042
4043 s << FormatDistanceAdaptive(dist);
4044
4045 // Compute and display cumulative distance from route start point to
4046 // current leg end point and RNG,TTG,ETA from ship to current leg end
4047 // point for active route
4048 double shiptoEndLeg = 0.;
4049 bool validActive = false;
4050 if (pr->IsActive() &&
4051 pr->pRoutePointList->GetFirst()->GetData()->m_bIsActive)
4052 validActive = true;
4053
4054 if (segShow_point_a != pr->pRoutePointList->GetFirst()->GetData()) {
4055 wxRoutePointListNode *node =
4056 (pr->pRoutePointList)->GetFirst()->GetNext();
4057 RoutePoint *prp;
4058 float dist_to_endleg = 0;
4059 wxString t;
4060
4061 while (node) {
4062 prp = node->GetData();
4063 if (validActive)
4064 shiptoEndLeg += prp->m_seg_len;
4065 else if (prp->m_bIsActive)
4066 validActive = true;
4067 dist_to_endleg += prp->m_seg_len;
4068 if (prp->IsSame(segShow_point_a)) break;
4069 node = node->GetNext();
4070 }
4071 s << _T(" (+") << FormatDistanceAdaptive(dist_to_endleg) << _T(")");
4072 }
4073 // write from ship to end selected leg point data if the route is
4074 // active
4075 if (validActive) {
4076 s << _T("\n") << _("From Ship To") << _T(" ")
4077 << segShow_point_b->GetName() << _T("\n");
4078 shiptoEndLeg +=
4079 g_pRouteMan
4080 ->GetCurrentRngToActivePoint(); // add distance from ship
4081 // to active point
4082 shiptoEndLeg +=
4083 segShow_point_b
4084 ->m_seg_len; // add the lenght of the selected leg
4085 s << FormatDistanceAdaptive(shiptoEndLeg);
4086 // ensure sog/cog are valid and vmg is positive to keep data
4087 // coherent
4088 double vmg = 0.;
4089 if (!std::isnan(gCog) && !std::isnan(gSog))
4090 vmg = gSog *
4091 cos((g_pRouteMan->GetCurrentBrgToActivePoint() - gCog) *
4092 PI / 180.);
4093 if (vmg > 0.) {
4094 float ttg_sec = (shiptoEndLeg / gSog) * 3600.;
4095 wxTimeSpan ttg_span = wxTimeSpan::Seconds((long)ttg_sec);
4096 s << _T(" - ")
4097 << wxString(ttg_sec > SECONDS_PER_DAY
4098 ? ttg_span.Format(_("%Dd %H:%M"))
4099 : ttg_span.Format(_("%H:%M")));
4100 wxDateTime dtnow, eta;
4101 eta = dtnow.SetToCurrent().Add(ttg_span);
4102 s << _T(" - ") << eta.Format(_T("%b")).Mid(0, 4)
4103 << eta.Format(_T(" %d %H:%M"));
4104 } else
4105 s << _T(" ---- ----");
4106 }
4107 m_pRouteRolloverWin->SetString(s);
4108
4109 m_pRouteRolloverWin->SetBestPosition(mouse_x, mouse_y, 16, 16,
4110 LEG_ROLLOVER, win_size);
4111 m_pRouteRolloverWin->SetBitmap(LEG_ROLLOVER);
4112 m_pRouteRolloverWin->IsActive(true);
4113 b_need_refresh = true;
4114 showRouteRollover = true;
4115 break;
4116 }
4117 } else
4118 node = node->GetNext();
4119 }
4120 } else {
4121 // Is the cursor still in select radius, and not timed out?
4122 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
4123 if (!pSelect->IsSelectableSegmentSelected(ctx, m_cursor_lat, m_cursor_lon,
4124 m_pRolloverRouteSeg))
4125 showRouteRollover = false;
4126 else if (m_pRouteRolloverWin && !m_pRouteRolloverWin->IsActive())
4127 showRouteRollover = false;
4128 else
4129 showRouteRollover = true;
4130 }
4131
4132 // If currently creating a route, do not show this rollover window
4133 if (m_routeState) showRouteRollover = false;
4134
4135 // Similar for AIS target rollover window
4136 if (m_pAISRolloverWin && m_pAISRolloverWin->IsActive())
4137 showRouteRollover = false;
4138
4139 if (m_pRouteRolloverWin /*&& m_pRouteRolloverWin->IsActive()*/ &&
4140 !showRouteRollover) {
4141 m_pRouteRolloverWin->IsActive(false);
4142 m_pRolloverRouteSeg = NULL;
4143 m_pRouteRolloverWin->Destroy();
4144 m_pRouteRolloverWin = NULL;
4145 b_need_refresh = true;
4146 } else if (m_pRouteRolloverWin && showRouteRollover) {
4147 m_pRouteRolloverWin->IsActive(true);
4148 b_need_refresh = true;
4149 }
4150
4151 // Now the Track info rollover
4152 // Show the track segment info
4153 bool showTrackRollover = false;
4154
4155 if (NULL == m_pRolloverTrackSeg) {
4156 // Get a list of all selectable sgements, and search for the first
4157 // visible segment as the rollover target.
4158
4159 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
4160 SelectableItemList SelList = pSelect->FindSelectionList(
4161 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_TRACKSEGMENT);
4162 wxSelectableItemListNode *node = SelList.GetFirst();
4163 while (node) {
4164 SelectItem *pFindSel = node->GetData();
4165
4166 Track *pt = (Track *)pFindSel->m_pData3; // candidate
4167
4168 if (pt && pt->IsVisible()) {
4169 m_pRolloverTrackSeg = pFindSel;
4170 showTrackRollover = true;
4171
4172 if (NULL == m_pTrackRolloverWin) {
4173 m_pTrackRolloverWin = new RolloverWin(this, 10);
4174 m_pTrackRolloverWin->IsActive(false);
4175 }
4176
4177 if (!m_pTrackRolloverWin->IsActive()) {
4178 wxString s;
4179 TrackPoint *segShow_point_a =
4180 (TrackPoint *)m_pRolloverTrackSeg->m_pData1;
4181 TrackPoint *segShow_point_b =
4182 (TrackPoint *)m_pRolloverTrackSeg->m_pData2;
4183
4184 double brg, dist;
4185 DistanceBearingMercator(
4186 segShow_point_b->m_lat, segShow_point_b->m_lon,
4187 segShow_point_a->m_lat, segShow_point_a->m_lon, &brg, &dist);
4188
4189 if (!pt->m_bIsInLayer)
4190 s.Append(_("Track") + _T(": "));
4191 else
4192 s.Append(_("Layer Track: "));
4193
4194 if (pt->GetName().IsEmpty())
4195 s.Append(_("(unnamed)"));
4196 else
4197 s.Append(pt->GetName());
4198 double tlenght = pt->Length();
4199 s << _T("\n") << _("Total Track: ")
4200 << FormatDistanceAdaptive(tlenght);
4201 if (pt->GetLastPoint()->GetTimeString() &&
4202 pt->GetPoint(0)->GetTimeString()) {
4203 wxDateTime lastPointTime = pt->GetLastPoint()->GetCreateTime();
4204 wxDateTime zeroPointTime = pt->GetPoint(0)->GetCreateTime();
4205 if (lastPointTime.IsValid() && zeroPointTime.IsValid()) {
4206 wxTimeSpan ttime = lastPointTime - zeroPointTime;
4207 double htime = ttime.GetSeconds().ToDouble() / 3600.;
4208 s << wxString::Format(_T(" %.1f "), (float)(tlenght / htime))
4209 << getUsrSpeedUnit();
4210 s << wxString(htime > 24. ? ttime.Format(_T(" %Dd %H:%M"))
4211 : ttime.Format(_T(" %H:%M")));
4212 }
4213 }
4214
4215 if (g_bShowTrackPointTime && strlen(segShow_point_b->GetTimeString()))
4216 s << _T("\n") << _("Segment Created: ")
4217 << segShow_point_b->GetTimeString();
4218
4219 s << _T("\n");
4220 if (g_bShowTrue)
4221 s << wxString::Format(wxString("%03d%c ", wxConvUTF8), (int)brg,
4222 0x00B0);
4223
4224 if (g_bShowMag) {
4225 double latAverage =
4226 (segShow_point_b->m_lat + segShow_point_a->m_lat) / 2;
4227 double lonAverage =
4228 (segShow_point_b->m_lon + segShow_point_a->m_lon) / 2;
4229 double varBrg = gFrame->GetMag(brg, latAverage, lonAverage);
4230
4231 s << wxString::Format(wxString("%03d%c ", wxConvUTF8), (int)varBrg,
4232 0x00B0);
4233 }
4234
4235 s << FormatDistanceAdaptive(dist);
4236
4237 if (segShow_point_a->GetTimeString() &&
4238 segShow_point_b->GetTimeString()) {
4239 wxDateTime apoint = segShow_point_a->GetCreateTime();
4240 wxDateTime bpoint = segShow_point_b->GetCreateTime();
4241 if (apoint.IsValid() && bpoint.IsValid()) {
4242 double segmentSpeed = toUsrSpeed(
4243 dist / ((bpoint - apoint).GetSeconds().ToDouble() / 3600.));
4244 s << wxString::Format(_T(" %.1f "), (float)segmentSpeed)
4245 << getUsrSpeedUnit();
4246 }
4247 }
4248
4249 m_pTrackRolloverWin->SetString(s);
4250
4251 m_pTrackRolloverWin->SetBestPosition(mouse_x, mouse_y, 16, 16,
4252 LEG_ROLLOVER, win_size);
4253 m_pTrackRolloverWin->SetBitmap(LEG_ROLLOVER);
4254 m_pTrackRolloverWin->IsActive(true);
4255 b_need_refresh = true;
4256 showTrackRollover = true;
4257 break;
4258 }
4259 } else
4260 node = node->GetNext();
4261 }
4262 } else {
4263 // Is the cursor still in select radius, and not timed out?
4264 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
4265 if (!pSelect->IsSelectableSegmentSelected(ctx, m_cursor_lat, m_cursor_lon,
4266 m_pRolloverTrackSeg))
4267 showTrackRollover = false;
4268 else if (m_pTrackRolloverWin && !m_pTrackRolloverWin->IsActive())
4269 showTrackRollover = false;
4270 else
4271 showTrackRollover = true;
4272 }
4273
4274 // Similar for AIS target rollover window
4275 if (m_pAISRolloverWin && m_pAISRolloverWin->IsActive())
4276 showTrackRollover = false;
4277
4278 // Similar for route rollover window
4279 if (m_pRouteRolloverWin && m_pRouteRolloverWin->IsActive())
4280 showTrackRollover = false;
4281
4282 // TODO We onlt show tracks on primary canvas....
4283 // if(!IsPrimaryCanvas())
4284 // showTrackRollover = false;
4285
4286 if (m_pTrackRolloverWin /*&& m_pTrackRolloverWin->IsActive()*/ &&
4287 !showTrackRollover) {
4288 m_pTrackRolloverWin->IsActive(false);
4289 m_pRolloverTrackSeg = NULL;
4290 m_pTrackRolloverWin->Destroy();
4291 m_pTrackRolloverWin = NULL;
4292 b_need_refresh = true;
4293 } else if (m_pTrackRolloverWin && showTrackRollover) {
4294 m_pTrackRolloverWin->IsActive(true);
4295 b_need_refresh = true;
4296 }
4297
4298 if (b_need_refresh) Refresh();
4299}
4300
4301void ChartCanvas::OnCursorTrackTimerEvent(wxTimerEvent &event) {
4302 if ((GetShowENCLights() || m_bsectors_shown) &&
4303 s57_CheckExtendedLightSectors(this, mouse_x, mouse_y, VPoint,
4304 extendedSectorLegs)) {
4305 if (!m_bsectors_shown) {
4306 ReloadVP(false);
4307 m_bsectors_shown = true;
4308 }
4309 } else {
4310 if (m_bsectors_shown) {
4311 ReloadVP(false);
4312 m_bsectors_shown = false;
4313 }
4314 }
4315
4316// This is here because GTK status window update is expensive..
4317// cairo using pango rebuilds the font every time so is very
4318// inefficient
4319// Anyway, only update the status bar when this timer expires
4320#if defined(__WXGTK__) || defined(__WXQT__)
4321 {
4322 // Check the absolute range of the cursor position
4323 // There could be a window wherein the chart geoereferencing is not
4324 // valid....
4325 double cursor_lat, cursor_lon;
4326 GetCanvasPixPoint(mouse_x, mouse_y, cursor_lat, cursor_lon);
4327
4328 if ((fabs(cursor_lat) < 90.) && (fabs(cursor_lon) < 360.)) {
4329 while (cursor_lon < -180.) cursor_lon += 360.;
4330
4331 while (cursor_lon > 180.) cursor_lon -= 360.;
4332
4333 SetCursorStatus(cursor_lat, cursor_lon);
4334 }
4335 }
4336#endif
4337}
4338
4339void ChartCanvas::SetCursorStatus(double cursor_lat, double cursor_lon) {
4340 if (!parent_frame->m_pStatusBar) return;
4341
4342 wxString s1;
4343 s1 += _T(" ");
4344 s1 += toSDMM(1, cursor_lat);
4345 s1 += _T(" ");
4346 s1 += toSDMM(2, cursor_lon);
4347
4348 if (STAT_FIELD_CURSOR_LL >= 0)
4349 parent_frame->SetStatusText(s1, STAT_FIELD_CURSOR_LL);
4350
4351 if (STAT_FIELD_CURSOR_BRGRNG < 0) return;
4352
4353 double brg, dist;
4354 wxString sm;
4355 wxString st;
4356 DistanceBearingMercator(cursor_lat, cursor_lon, gLat, gLon, &brg, &dist);
4357 if (g_bShowMag) sm.Printf("%03d%c(M) ", (int)toMagnetic(brg), 0x00B0);
4358 if (g_bShowTrue) st.Printf("%03d%c(T) ", (int)brg, 0x00B0);
4359
4360 wxString s = st + sm;
4361 s << FormatDistanceAdaptive(dist);
4362
4363 // CUSTOMIZATION - LIVE ETA OPTION
4364 // -------------------------------------------------------
4365 // Calculate an "live" ETA based on route starting from the current
4366 // position of the boat and goes to the cursor of the mouse.
4367 // In any case, an standard ETA will be calculated with a default speed
4368 // of the boat to give an estimation of the route (in particular if GPS
4369 // is off).
4370
4371 // Display only if option "live ETA" is selected in Settings > Display >
4372 // General.
4373 if (g_bShowLiveETA) {
4374 float realTimeETA;
4375 float boatSpeed;
4376 float boatSpeedDefault = g_defaultBoatSpeed;
4377
4378 // Calculate Estimate Time to Arrival (ETA) in minutes
4379 // Check before is value not closed to zero (it will make an very big
4380 // number...)
4381 if (!std::isnan(gSog)) {
4382 boatSpeed = gSog;
4383 if (boatSpeed < 0.5) {
4384 realTimeETA = 0;
4385 } else {
4386 realTimeETA = dist / boatSpeed * 60;
4387 }
4388 } else {
4389 realTimeETA = 0;
4390 }
4391
4392 // Add space after distance display
4393 s << " ";
4394 // Display ETA
4395 s << minutesToHoursDays(realTimeETA);
4396
4397 // In any case, display also an ETA with default speed at 6knts
4398
4399 s << " [@";
4400 s << wxString::Format(_T("%d"), (int)toUsrSpeed(boatSpeedDefault, -1));
4401 s << wxString::Format(_T("%s"), getUsrSpeedUnit(-1));
4402 s << " ";
4403 s << minutesToHoursDays(dist / boatSpeedDefault * 60);
4404 s << "]";
4405 }
4406 // END OF - LIVE ETA OPTION
4407
4408 parent_frame->SetStatusText(s, STAT_FIELD_CURSOR_BRGRNG);
4409}
4410
4411// CUSTOMIZATION - FORMAT MINUTES
4412// -------------------------------------------------------
4413// New function to format minutes into a more readable format:
4414// * Hours + minutes, or
4415// * Days + hours.
4416wxString minutesToHoursDays(float timeInMinutes) {
4417 wxString s;
4418
4419 if (timeInMinutes == 0) {
4420 s << "--min";
4421 }
4422
4423 // Less than 60min, keep time in minutes
4424 else if (timeInMinutes < 60 && timeInMinutes != 0) {
4425 s << wxString::Format(_T("%d"), (int)timeInMinutes);
4426 s << "min";
4427 }
4428
4429 // Between 1h and less than 24h, display time in hours, minutes
4430 else if (timeInMinutes >= 60 && timeInMinutes < 24 * 60) {
4431 int hours;
4432 int min;
4433 hours = (int)timeInMinutes / 60;
4434 min = (int)timeInMinutes % 60;
4435
4436 if (min == 0) {
4437 s << wxString::Format(_T("%d"), hours);
4438 s << "h";
4439 } else {
4440 s << wxString::Format(_T("%d"), hours);
4441 s << "h";
4442 s << wxString::Format(_T("%d"), min);
4443 s << "min";
4444 }
4445
4446 }
4447
4448 // More than 24h, display time in days, hours
4449 else if (timeInMinutes > 24 * 60) {
4450 int days;
4451 int hours;
4452 days = (int)(timeInMinutes / 60) / 24;
4453 hours = (int)(timeInMinutes / 60) % 24;
4454
4455 if (hours == 0) {
4456 s << wxString::Format(_T("%d"), days);
4457 s << "d";
4458 } else {
4459 s << wxString::Format(_T("%d"), days);
4460 s << "d";
4461 s << wxString::Format(_T("%d"), hours);
4462 s << "h";
4463 }
4464 }
4465
4466 return s;
4467}
4468
4469// END OF CUSTOMIZATION - FORMAT MINUTES
4470// Thanks open source code ;-)
4471// -------------------------------------------------------
4472
4473void ChartCanvas::GetCursorLatLon(double *lat, double *lon) {
4474 double clat, clon;
4475 GetCanvasPixPoint(mouse_x, mouse_y, clat, clon);
4476 *lat = clat;
4477 *lon = clon;
4478}
4479
4480void ChartCanvas::GetDoubleCanvasPointPix(double rlat, double rlon,
4481 wxPoint2DDouble *r) {
4482 return GetDoubleCanvasPointPixVP(GetVP(), rlat, rlon, r);
4483}
4484
4486 double rlon, wxPoint2DDouble *r) {
4487 // If the Current Chart is a raster chart, and the
4488 // requested lat/long is within the boundaries of the chart,
4489 // and the VP is not rotated,
4490 // then use the embedded BSB chart georeferencing algorithm
4491 // for greater accuracy
4492 // Additionally, use chart embedded georef if the projection is TMERC
4493 // i.e. NOT MERCATOR and NOT POLYCONIC
4494
4495 // If for some reason the chart rejects the request by returning an error,
4496 // then fall back to Viewport Projection estimate from canvas parameters
4497 if (!g_bopengl && m_singleChart &&
4498 (m_singleChart->GetChartFamily() == CHART_FAMILY_RASTER) &&
4499 (((fabs(vp.rotation) < .0001) && (fabs(vp.skew) < .0001)) ||
4500 ((m_singleChart->GetChartProjectionType() != PROJECTION_MERCATOR) &&
4501 (m_singleChart->GetChartProjectionType() !=
4502 PROJECTION_TRANSVERSE_MERCATOR) &&
4503 (m_singleChart->GetChartProjectionType() != PROJECTION_POLYCONIC))) &&
4504 (m_singleChart->GetChartProjectionType() == vp.m_projection_type) &&
4505 (m_singleChart->GetChartType() != CHART_TYPE_PLUGIN)) {
4506 ChartBaseBSB *Cur_BSB_Ch = dynamic_cast<ChartBaseBSB *>(m_singleChart);
4507 // bool bInside = G_FloatPtInPolygon ( ( MyFlPoint *
4508 // ) Cur_BSB_Ch->GetCOVRTableHead ( 0 ),
4509 // Cur_BSB_Ch->GetCOVRTablenPoints
4510 // ( 0 ), rlon,
4511 // rlat );
4512 // bInside = true;
4513 // if ( bInside )
4514 if (Cur_BSB_Ch) {
4515 // This is a Raster chart....
4516 // If the VP is changing, the raster chart parameters may not yet be
4517 // setup So do that before accessing the chart's embedded
4518 // georeferencing
4519 Cur_BSB_Ch->SetVPRasterParms(vp);
4520 double rpixxd, rpixyd;
4521 if (0 == Cur_BSB_Ch->latlong_to_pix_vp(rlat, rlon, rpixxd, rpixyd, vp)) {
4522 r->m_x = rpixxd;
4523 r->m_y = rpixyd;
4524 return;
4525 }
4526 }
4527 }
4528
4529 // if needed, use the VPoint scaling estimator,
4530 *r = vp.GetDoublePixFromLL(rlat, rlon);
4531}
4532
4533// This routine might be deleted and all of the rendering improved
4534// to have floating point accuracy
4535bool ChartCanvas::GetCanvasPointPix(double rlat, double rlon, wxPoint *r) {
4536 return GetCanvasPointPixVP(GetVP(), rlat, rlon, r);
4537}
4538
4539bool ChartCanvas::GetCanvasPointPixVP(ViewPort &vp, double rlat, double rlon,
4540 wxPoint *r) {
4541 wxPoint2DDouble p;
4542 GetDoubleCanvasPointPixVP(vp, rlat, rlon, &p);
4543
4544 // some projections give nan values when invisible values (other side of
4545 // world) are requested we should stop using integer coordinates or return
4546 // false here (and test it everywhere)
4547 if (std::isnan(p.m_x)) {
4548 *r = wxPoint(INVALID_COORD, INVALID_COORD);
4549 return false;
4550 }
4551
4552 if ((abs(p.m_x) < 1e6) && (abs(p.m_y) < 1e6))
4553 *r = wxPoint(wxRound(p.m_x), wxRound(p.m_y));
4554 else
4555 *r = wxPoint(INVALID_COORD, INVALID_COORD);
4556
4557 return true;
4558}
4559
4560void ChartCanvas::GetCanvasPixPoint(double x, double y, double &lat,
4561 double &lon) {
4562 // If the Current Chart is a raster chart, and the
4563 // requested x,y is within the boundaries of the chart,
4564 // and the VP is not rotated,
4565 // then use the embedded BSB chart georeferencing algorithm
4566 // for greater accuracy
4567 // Additionally, use chart embedded georef if the projection is TMERC
4568 // i.e. NOT MERCATOR and NOT POLYCONIC
4569
4570 // If for some reason the chart rejects the request by returning an error,
4571 // then fall back to Viewport Projection estimate from canvas parameters
4572 bool bUseVP = true;
4573
4574 if (!g_bopengl && m_singleChart &&
4575 (m_singleChart->GetChartFamily() == CHART_FAMILY_RASTER) &&
4576 (((fabs(GetVP().rotation) < .0001) && (fabs(GetVP().skew) < .0001)) ||
4577 ((m_singleChart->GetChartProjectionType() != PROJECTION_MERCATOR) &&
4578 (m_singleChart->GetChartProjectionType() !=
4579 PROJECTION_TRANSVERSE_MERCATOR) &&
4580 (m_singleChart->GetChartProjectionType() != PROJECTION_POLYCONIC))) &&
4581 (m_singleChart->GetChartProjectionType() == GetVP().m_projection_type) &&
4582 (m_singleChart->GetChartType() != CHART_TYPE_PLUGIN)) {
4583 ChartBaseBSB *Cur_BSB_Ch = dynamic_cast<ChartBaseBSB *>(m_singleChart);
4584
4585 // TODO maybe need iterative process to validate bInside
4586 // first pass is mercator, then check chart boundaries
4587
4588 if (Cur_BSB_Ch) {
4589 // This is a Raster chart....
4590 // If the VP is changing, the raster chart parameters may not yet be
4591 // setup So do that before accessing the chart's embedded
4592 // georeferencing
4593 Cur_BSB_Ch->SetVPRasterParms(GetVP());
4594
4595 double slat, slon;
4596 if (0 == Cur_BSB_Ch->vp_pix_to_latlong(GetVP(), x, y, &slat, &slon)) {
4597 lat = slat;
4598
4599 if (slon < -180.)
4600 slon += 360.;
4601 else if (slon > 180.)
4602 slon -= 360.;
4603
4604 lon = slon;
4605 bUseVP = false;
4606 }
4607 }
4608 }
4609
4610 // if needed, use the VPoint scaling estimator
4611 if (bUseVP) {
4612 GetVP().GetLLFromPix(wxPoint2DDouble(x, y), &lat, &lon);
4613 }
4614}
4615
4617 DoZoomCanvas(factor, false);
4618 extendedSectorLegs.clear();
4619}
4620
4621void ChartCanvas::ZoomCanvas(double factor, bool can_zoom_to_cursor,
4622 bool stoptimer) {
4623 m_bzooming_to_cursor = can_zoom_to_cursor && g_bEnableZoomToCursor;
4624
4625 if (g_bsmoothpanzoom) {
4626 if (StartTimedMovement(stoptimer)) {
4627 m_mustmove += 150; /* for quick presses register as 200 ms duration */
4628 m_zoom_factor = factor;
4629 }
4630
4631 m_zoom_target = VPoint.chart_scale / factor;
4632 } else {
4633 if (m_modkeys == wxMOD_ALT) factor = pow(factor, .15);
4634
4635 DoZoomCanvas(factor, can_zoom_to_cursor);
4636 }
4637
4638 extendedSectorLegs.clear();
4639}
4640
4641void ChartCanvas::DoZoomCanvas(double factor, bool can_zoom_to_cursor) {
4642 // possible on startup
4643 if (!ChartData) return;
4644 if (!m_pCurrentStack) return;
4645
4646 /* TODO: queue the quilted loading code to a background thread
4647 so yield is never called from here, and also rendering is not delayed */
4648
4649 // Cannot allow Yield() re-entrancy here
4650 if (m_bzooming) return;
4651 m_bzooming = true;
4652
4653 double old_ppm = GetVP().view_scale_ppm;
4654
4655 // Capture current cursor position for zoom to cursor
4656 double zlat = m_cursor_lat;
4657 double zlon = m_cursor_lon;
4658
4659 double proposed_scale_onscreen =
4660 GetVP().chart_scale /
4661 factor; // GetCanvasScaleFactor() / ( GetVPScale() * factor );
4662 bool b_do_zoom = false;
4663
4664 if (factor > 1) {
4665 b_do_zoom = true;
4666
4667 // double zoom_factor = factor;
4668
4669 ChartBase *pc = NULL;
4670
4671 if (!VPoint.b_quilt) {
4672 pc = m_singleChart;
4673 } else {
4674 int new_db_index = m_pQuilt->AdjustRefOnZoomIn(proposed_scale_onscreen);
4675 if (new_db_index >= 0)
4676 pc = ChartData->OpenChartFromDB(new_db_index, FULL_INIT);
4677 else { // for whatever reason, no reference chart is known
4678 // Choose the smallest scale chart on the current stack
4679 // and then adjust for scale range
4680 int current_ref_stack_index = -1;
4681 if (m_pCurrentStack->nEntry) {
4682 int trial_index =
4683 m_pCurrentStack->GetDBIndex(m_pCurrentStack->nEntry - 1);
4684 m_pQuilt->SetReferenceChart(trial_index);
4685 new_db_index = m_pQuilt->AdjustRefOnZoomIn(proposed_scale_onscreen);
4686 if (new_db_index >= 0)
4687 pc = ChartData->OpenChartFromDB(new_db_index, FULL_INIT);
4688 }
4689 }
4690
4691 if (m_pCurrentStack)
4692 m_pCurrentStack->SetCurrentEntryFromdbIndex(
4693 new_db_index); // highlite the correct bar entry
4694 }
4695
4696 if (pc) {
4697 // double target_scale_ppm = GetVPScale() * zoom_factor;
4698 // proposed_scale_onscreen = GetCanvasScaleFactor() /
4699 // target_scale_ppm;
4700
4701 // Query the chart to determine the appropriate zoom range
4702 double min_allowed_scale =
4703 g_maxzoomin; // Roughly, latitude dependent for mercator charts
4704
4705 if (proposed_scale_onscreen < min_allowed_scale) {
4706 if (min_allowed_scale == GetCanvasScaleFactor() / (GetVPScale())) {
4707 m_zoom_factor = 1; /* stop zooming */
4708 b_do_zoom = false;
4709 } else
4710 proposed_scale_onscreen = min_allowed_scale;
4711 }
4712
4713 } else {
4714 proposed_scale_onscreen = wxMax(proposed_scale_onscreen, g_maxzoomin);
4715 }
4716
4717 } else if (factor < 1) {
4718 b_do_zoom = true;
4719
4720 ChartBase *pc = NULL;
4721
4722 bool b_smallest = false;
4723
4724 if (!VPoint.b_quilt) { // not quilted
4725 pc = m_singleChart;
4726
4727 if (pc) {
4728 // If m_singleChart is not on the screen, unbound the zoomout
4729 LLBBox viewbox = VPoint.GetBBox();
4730 // BoundingBox chart_box;
4731 int current_index = ChartData->FinddbIndex(pc->GetFullPath());
4732 double max_allowed_scale;
4733
4734 max_allowed_scale = GetCanvasScaleFactor() / m_absolute_min_scale_ppm;
4735
4736 // We can allow essentially unbounded zoomout in single chart mode
4737 // if( ChartData->GetDBBoundingBox( current_index,
4738 // &chart_box ) &&
4739 // !viewbox.IntersectOut( chart_box ) )
4740 // // Clamp the minimum scale zoom-out to the value
4741 // specified by the chart max_allowed_scale =
4742 // wxMin(max_allowed_scale, 4.0 *
4743 // pc->GetNormalScaleMax(
4744 // GetCanvasScaleFactor(),
4745 // GetCanvasWidth() ) );
4746 if (proposed_scale_onscreen > max_allowed_scale) {
4747 m_zoom_factor = 1; /* stop zooming */
4748 proposed_scale_onscreen = max_allowed_scale;
4749 }
4750 }
4751
4752 } else {
4753 int new_db_index = m_pQuilt->AdjustRefOnZoomOut(proposed_scale_onscreen);
4754 if (new_db_index >= 0)
4755 pc = ChartData->OpenChartFromDB(new_db_index, FULL_INIT);
4756
4757 if (m_pCurrentStack)
4758 m_pCurrentStack->SetCurrentEntryFromdbIndex(
4759 new_db_index); // highlite the correct bar entry
4760
4761 b_smallest = m_pQuilt->IsChartSmallestScale(new_db_index);
4762
4763 if (b_smallest || (0 == m_pQuilt->GetExtendedStackCount()))
4764 proposed_scale_onscreen =
4765 wxMin(proposed_scale_onscreen,
4766 GetCanvasScaleFactor() / m_absolute_min_scale_ppm);
4767 }
4768
4769 // set a minimum scale
4770 if ((GetCanvasScaleFactor() / proposed_scale_onscreen) <
4771 m_absolute_min_scale_ppm)
4772 b_do_zoom = false;
4773 }
4774
4775 double new_scale =
4776 GetVPScale() * (GetVP().chart_scale / proposed_scale_onscreen);
4777
4778 if (b_do_zoom) {
4779 // Disable ZTC if lookahead is ON, and currently b_follow is active
4780 bool b_allow_ztc = true;
4781 if (m_bFollow && m_bLookAhead) b_allow_ztc = false;
4782 if (can_zoom_to_cursor && g_bEnableZoomToCursor && b_allow_ztc) {
4783 if (m_bLookAhead) {
4784 double brg, distance;
4785 ll_gc_ll_reverse(gLat, gLon, GetVP().clat, GetVP().clon, &brg,
4786 &distance);
4787 dir_to_shift = brg;
4788 meters_to_shift = distance * 1852;
4789 }
4790 // Arrange to combine the zoom and pan into one operation for smoother
4791 // appearance
4792 SetVPScale(new_scale, false); // adjust, but deferred refresh
4793 wxPoint r;
4794 GetCanvasPointPix(zlat, zlon, &r);
4795 // this will emit the Refresh()
4796 PanCanvas(r.x - mouse_x, r.y - mouse_y);
4797 } else {
4798 SetVPScale(new_scale);
4799 if (m_bFollow) DoCanvasUpdate();
4800 }
4801 }
4802
4803 m_bzooming = false;
4804}
4805int rot;
4806void ChartCanvas::RotateCanvas(double dir) {
4807 // SetUpMode(NORTH_UP_MODE);
4808
4809 if (g_bsmoothpanzoom) {
4810 if (StartTimedMovement()) {
4811 m_mustmove += 150; /* for quick presses register as 200 ms duration */
4812 m_rotation_speed = dir * 60;
4813 }
4814 } else {
4815 double speed = dir * 10;
4816 if (m_modkeys == wxMOD_ALT) speed /= 20;
4817 DoRotateCanvas(VPoint.rotation + PI / 180 * speed);
4818 }
4819}
4820
4821void ChartCanvas::DoRotateCanvas(double rotation) {
4822 while (rotation < 0) rotation += 2 * PI;
4823 while (rotation > 2 * PI) rotation -= 2 * PI;
4824
4825 if (rotation == VPoint.rotation || std::isnan(rotation)) return;
4826
4827 SetVPRotation(rotation);
4828 parent_frame->UpdateRotationState(VPoint.rotation);
4829}
4830
4831void ChartCanvas::DoTiltCanvas(double tilt) {
4832 while (tilt < 0) tilt = 0;
4833 while (tilt > .95) tilt = .95;
4834
4835 if (tilt == VPoint.tilt || std::isnan(tilt)) return;
4836
4837 VPoint.tilt = tilt;
4838 Refresh(false);
4839}
4840
4841void ChartCanvas::TogglebFollow(void) {
4842 if (!m_bFollow)
4843 SetbFollow();
4844 else
4845 ClearbFollow();
4846}
4847
4848void ChartCanvas::ClearbFollow(void) {
4849 m_bFollow = false; // update the follow flag
4850
4851 parent_frame->SetMenubarItemState(ID_MENU_NAV_FOLLOW, false);
4852
4853 UpdateFollowButtonState();
4854
4855 DoCanvasUpdate();
4856 ReloadVP();
4857 parent_frame->SetChartUpdatePeriod();
4858}
4859
4860void ChartCanvas::SetbFollow(void) {
4861 // Is the OWNSHIP on-screen?
4862 // If not, then reset the OWNSHIP offset to 0 (center screen)
4863 if ((fabs(m_OSoffsetx) > VPoint.pix_width / 2) ||
4864 (fabs(m_OSoffsety) > VPoint.pix_height / 2)) {
4865 m_OSoffsetx = 0;
4866 m_OSoffsety = 0;
4867 }
4868
4869 // Apply the present b_follow offset values to ship position
4870 wxPoint2DDouble p;
4871 GetDoubleCanvasPointPix(gLat, gLon, &p);
4872 p.m_x += m_OSoffsetx;
4873 p.m_y -= m_OSoffsety;
4874
4875 // compute the target center screen lat/lon
4876 double dlat, dlon;
4877 GetCanvasPixPoint(p.m_x, p.m_y, dlat, dlon);
4878
4879 JumpToPosition(dlat, dlon, GetVPScale());
4880 m_bFollow = true;
4881
4882 parent_frame->SetMenubarItemState(ID_MENU_NAV_FOLLOW, true);
4883 UpdateFollowButtonState();
4884
4885 if (!g_bSmoothRecenter) {
4886 DoCanvasUpdate();
4887 ReloadVP();
4888 }
4889 parent_frame->SetChartUpdatePeriod();
4890}
4891
4892void ChartCanvas::UpdateFollowButtonState(void) {
4893 if (m_muiBar) {
4894 if (!m_bFollow)
4895 m_muiBar->SetFollowButtonState(0);
4896 else {
4897 if (m_bLookAhead)
4898 m_muiBar->SetFollowButtonState(2);
4899 else
4900 m_muiBar->SetFollowButtonState(1);
4901 }
4902 }
4903
4904#ifdef __ANDROID__
4905 if (!m_bFollow)
4906 androidSetFollowTool(0);
4907 else {
4908 if (m_bLookAhead)
4909 androidSetFollowTool(2);
4910 else
4911 androidSetFollowTool(1);
4912 }
4913#endif
4914}
4915
4916void ChartCanvas::JumpToPosition(double lat, double lon, double scale_ppm) {
4917 if (g_bSmoothRecenter && !m_routeState) {
4918 if (StartSmoothJump(lat, lon, scale_ppm))
4919 return;
4920 else {
4921 // move closer to the target destination, and try again
4922 double gcDist, gcBearingEnd;
4923 Geodesic::GreatCircleDistBear(m_vLon, m_vLat, lon, lat, &gcDist, NULL,
4924 &gcBearingEnd);
4925 gcBearingEnd += 180;
4926 double lat_offset = cos(gcBearingEnd * PI / 180.) * 0.5 *
4927 GetCanvasWidth() / GetVPScale(); // meters
4928 double lon_offset =
4929 sin(gcBearingEnd * PI / 180.) * 0.5 * GetCanvasWidth() / GetVPScale();
4930 double new_lat = lat + (lat_offset / (1852 * 60));
4931 double new_lon = lon + (lon_offset / (1852 * 60));
4932 SetViewPoint(new_lat, new_lon);
4933 ReloadVP();
4934 StartSmoothJump(lat, lon, scale_ppm);
4935 return;
4936 }
4937 }
4938
4939 if (lon > 180.0) lon -= 360.0;
4940 m_vLat = lat;
4941 m_vLon = lon;
4942 StopMovement();
4943 m_bFollow = false;
4944
4945 if (!GetQuiltMode()) {
4946 double skew = 0;
4947 if (m_singleChart) skew = m_singleChart->GetChartSkew() * PI / 180.;
4948 SetViewPoint(lat, lon, scale_ppm, skew, GetVPRotation());
4949 } else {
4950 if (scale_ppm != GetVPScale()) {
4951 // XXX should be done in SetViewPoint
4952 VPoint.chart_scale = m_canvas_scale_factor / (scale_ppm);
4953 AdjustQuiltRefChart();
4954 }
4955 SetViewPoint(lat, lon, scale_ppm, 0, GetVPRotation());
4956 }
4957
4958 ReloadVP();
4959
4960 UpdateFollowButtonState();
4961
4962 // TODO
4963 // if( g_pi_manager ) {
4964 // g_pi_manager->SendViewPortToRequestingPlugIns( cc1->GetVP() );
4965 // }
4966}
4967
4968bool ChartCanvas::StartSmoothJump(double lat, double lon, double scale_ppm) {
4969 // Check distance to jump, in pixels at current chart scale
4970 // Modify smooth jump dynamics if jump distance is greater than 0.5x screen
4971 // width.
4972 double gcDist;
4973 Geodesic::GreatCircleDistBear(m_vLon, m_vLat, lon, lat, &gcDist, NULL, NULL);
4974 double distance_pixels = gcDist * GetVPScale();
4975 if (distance_pixels > 0.5 * GetCanvasWidth()) {
4976 // Jump is too far, try again
4977 return false;
4978 }
4979
4980 // Save where we're coming from
4981 m_startLat = m_vLat;
4982 m_startLon = m_vLon;
4983 m_startScale = GetVPScale(); // or VPoint.view_scale_ppm
4984
4985 // Save where we want to end up
4986 m_endLat = lat;
4987 m_endLon = (lon > 180.0) ? (lon - 360.0) : lon;
4988 m_endScale = scale_ppm;
4989
4990 // Setup timing
4991 m_animationDuration = 600; // ms
4992 m_animationStart = wxGetLocalTimeMillis();
4993
4994 // Stop any previous movement, ensure no conflicts
4995 StopMovement();
4996 m_bFollow = false;
4997
4998 // Start the timer with ~60 FPS (16 ms). Tweak as needed.
4999 m_easeTimer.Start(16, wxTIMER_CONTINUOUS);
5000 m_animationActive = true;
5001
5002 return true;
5003}
5004
5005void ChartCanvas::OnJumpEaseTimer(wxTimerEvent &event) {
5006 // Calculate time fraction from 0..1
5007 wxLongLong now = wxGetLocalTimeMillis();
5008 double elapsed = (now - m_animationStart).ToDouble();
5009 double t = elapsed / m_animationDuration.ToDouble();
5010 if (t > 1.0) t = 1.0;
5011
5012 // Ease function for smoother movement
5013 double e = easeOutCubic(t);
5014
5015 // Interpolate lat/lon/scale
5016 double curLat = m_startLat + (m_endLat - m_startLat) * e;
5017 double curLon = m_startLon + (m_endLon - m_startLon) * e;
5018 double curScale = m_startScale + (m_endScale - m_startScale) * e;
5019
5020 // Update viewpoint
5021 // (Essentially the same code used in JumpToPosition, but skipping the "snap"
5022 // portion)
5023 SetViewPoint(curLat, curLon, curScale, 0.0, GetVPRotation());
5024 ReloadVP();
5025
5026 // If we reached the end, stop the timer and finalize
5027 if (t >= 1.0) {
5028 m_easeTimer.Stop();
5029 m_animationActive = false;
5030 UpdateFollowButtonState();
5031 DoCanvasUpdate();
5032 ReloadVP();
5033 }
5034}
5035
5036bool ChartCanvas::PanCanvas(double dx, double dy) {
5037 if (!ChartData) return false;
5038
5039 if (g_btouch) {
5040 // Stop bfollow state, without a refresh
5041 m_bFollow = false; // update the follow flag
5042 parent_frame->SetMenubarItemState(ID_MENU_NAV_FOLLOW, false);
5043 UpdateFollowButtonState();
5044 // Clear the bfollow offset
5045 m_OSoffsetx = 0;
5046 m_OSoffsety = 0;
5047 }
5048
5049 extendedSectorLegs.clear();
5050
5051 // double clat = VPoint.clat, clon = VPoint.clon;
5052 double dlat, dlon;
5053 wxPoint2DDouble p(VPoint.pix_width / 2.0, VPoint.pix_height / 2.0);
5054
5055 int iters = 0;
5056 for (;;) {
5057 GetCanvasPixPoint(p.m_x + trunc(dx), p.m_y + trunc(dy), dlat, dlon);
5058
5059 if (iters++ > 5) return false;
5060 if (!std::isnan(dlat)) break;
5061
5062 dx *= .5, dy *= .5;
5063 if (fabs(dx) < 1 && fabs(dy) < 1) return false;
5064 }
5065
5066 // avoid overshooting the poles
5067 if (dlat > 90)
5068 dlat = 90;
5069 else if (dlat < -90)
5070 dlat = -90;
5071
5072 if (dlon > 360.) dlon -= 360.;
5073 if (dlon < -360.) dlon += 360.;
5074
5075 // This should not really be necessary, but round-trip georef on some
5076 // charts is not perfect, So we can get creep on repeated unidimensional
5077 // pans, and corrupt chart cacheing.......
5078
5079 // But this only works on north-up projections
5080 // TODO: can we remove this now?
5081 // if( ( ( fabs( GetVP().skew ) < .001 ) ) && ( fabs( GetVP().rotation ) <
5082 // .001 ) ) {
5083 //
5084 // if( dx == 0 ) dlon = clon;
5085 // if( dy == 0 ) dlat = clat;
5086 // }
5087
5088 int cur_ref_dbIndex = m_pQuilt->GetRefChartdbIndex();
5089
5090 SetViewPoint(dlat, dlon, VPoint.view_scale_ppm, VPoint.skew, VPoint.rotation);
5091
5092 if (VPoint.b_quilt) {
5093 int new_ref_dbIndex = m_pQuilt->GetRefChartdbIndex();
5094 if ((new_ref_dbIndex != cur_ref_dbIndex) && (new_ref_dbIndex != -1)) {
5095 // Tweak the scale slightly for a new ref chart
5096 ChartBase *pc = ChartData->OpenChartFromDB(new_ref_dbIndex, FULL_INIT);
5097 if (pc) {
5098 double tweak_scale_ppm =
5099 pc->GetNearestPreferredScalePPM(VPoint.view_scale_ppm);
5100 SetVPScale(tweak_scale_ppm);
5101 }
5102 }
5103
5104 if (new_ref_dbIndex == -1) {
5105 // for whatever reason, no reference chart is known
5106 // Probably panned out of the coverage region
5107 // If any charts are anywhere on-screen, choose the smallest
5108 // scale chart on the screen to be a new reference chart.
5109 int trial_index = -1;
5110 if (m_pCurrentStack->nEntry) {
5111 int trial_index =
5112 m_pCurrentStack->GetDBIndex(m_pCurrentStack->nEntry - 1);
5113 }
5114
5115 if (trial_index < 0) {
5116 auto full_screen_array = GetQuiltFullScreendbIndexArray();
5117 if (full_screen_array.size())
5118 trial_index = full_screen_array[full_screen_array.size() - 1];
5119 }
5120
5121 if (trial_index >= 0) {
5122 m_pQuilt->SetReferenceChart(trial_index);
5123 SetViewPoint(dlat, dlon, VPoint.view_scale_ppm, VPoint.skew,
5124 VPoint.rotation);
5125 ReloadVP();
5126 }
5127 }
5128 }
5129
5130 // Turn off bFollow only if the ownship has left the screen
5131 if (m_bFollow) {
5132 double offx, offy;
5133 toSM(dlat, dlon, gLat, gLon, &offx, &offy);
5134
5135 double offset_angle = atan2(offy, offx);
5136 double offset_distance = sqrt((offy * offy) + (offx * offx));
5137 double chart_angle = GetVPRotation();
5138 double target_angle = chart_angle - offset_angle;
5139 double d_east_mod = offset_distance * cos(target_angle);
5140 double d_north_mod = offset_distance * sin(target_angle);
5141
5142 m_OSoffsetx = d_east_mod * VPoint.view_scale_ppm;
5143 m_OSoffsety = -d_north_mod * VPoint.view_scale_ppm;
5144
5145 if (m_bFollow && ((fabs(m_OSoffsetx) > VPoint.pix_width / 2) ||
5146 (fabs(m_OSoffsety) > VPoint.pix_height / 2))) {
5147 m_bFollow = false; // update the follow flag
5148 UpdateFollowButtonState();
5149 }
5150 }
5151
5152 Refresh(false);
5153
5154 pCurTrackTimer->Start(m_curtrack_timer_msec, wxTIMER_ONE_SHOT);
5155
5156 return true;
5157}
5158
5159void ChartCanvas::ReloadVP(bool b_adjust) {
5160 if (g_brightness_init) SetScreenBrightness(g_nbrightness);
5161
5162 LoadVP(VPoint, b_adjust);
5163}
5164
5165void ChartCanvas::LoadVP(ViewPort &vp, bool b_adjust) {
5166#ifdef ocpnUSE_GL
5167 if (g_bopengl && m_glcc) {
5168 m_glcc->Invalidate();
5169 if (m_glcc->GetSize() != GetSize()) {
5170 m_glcc->SetSize(GetSize());
5171 }
5172 } else
5173#endif
5174 {
5175 m_cache_vp.Invalidate();
5176 m_bm_cache_vp.Invalidate();
5177 }
5178
5179 VPoint.Invalidate();
5180
5181 if (m_pQuilt) m_pQuilt->Invalidate();
5182
5183 // Make sure that the Selected Group is sensible...
5184 // if( m_groupIndex > (int) g_pGroupArray->GetCount() )
5185 // m_groupIndex = 0;
5186 // if( !CheckGroup( m_groupIndex ) )
5187 // m_groupIndex = 0;
5188
5189 SetViewPoint(vp.clat, vp.clon, vp.view_scale_ppm, vp.skew, vp.rotation,
5190 vp.m_projection_type, b_adjust);
5191}
5192
5193void ChartCanvas::SetQuiltRefChart(int dbIndex) {
5194 m_pQuilt->SetReferenceChart(dbIndex);
5195 VPoint.Invalidate();
5196 m_pQuilt->Invalidate();
5197}
5198
5199double ChartCanvas::GetBestStartScale(int dbi_hint, const ViewPort &vp) {
5200 if (m_pQuilt)
5201 return m_pQuilt->GetBestStartScale(dbi_hint, vp);
5202 else
5203 return vp.view_scale_ppm;
5204}
5205
5206// Verify and adjust the current reference chart,
5207// so that it will not lead to excessive overzoom or underzoom onscreen
5208int ChartCanvas::AdjustQuiltRefChart() {
5209 int ret = -1;
5210 if (m_pQuilt) {
5211 wxASSERT(ChartData);
5212 ChartBase *pc =
5213 ChartData->OpenChartFromDB(m_pQuilt->GetRefChartdbIndex(), FULL_INIT);
5214 if (pc) {
5215 double min_ref_scale =
5216 pc->GetNormalScaleMin(m_canvas_scale_factor, false);
5217 double max_ref_scale =
5218 pc->GetNormalScaleMax(m_canvas_scale_factor, m_canvas_width);
5219
5220 if (VPoint.chart_scale < min_ref_scale) {
5221 ret = m_pQuilt->AdjustRefOnZoomIn(VPoint.chart_scale);
5222 } else if (VPoint.chart_scale > max_ref_scale) {
5223 ret = m_pQuilt->AdjustRefOnZoomOut(VPoint.chart_scale);
5224 } else {
5225 bool brender_ok = IsChartLargeEnoughToRender(pc, VPoint);
5226
5227 int ref_family = pc->GetChartFamily();
5228
5229 if (!brender_ok) {
5230 unsigned int target_stack_index = 0;
5231 int target_stack_index_check =
5232 m_pQuilt->GetExtendedStackIndexArray()
5233 [m_pQuilt->GetRefChartdbIndex()]; // Lookup
5234
5235 if (wxNOT_FOUND != target_stack_index_check)
5236 target_stack_index = target_stack_index_check;
5237
5238 int extended_array_count =
5239 m_pQuilt->GetExtendedStackIndexArray().size();
5240 while ((!brender_ok) &&
5241 ((int)target_stack_index < (extended_array_count - 1))) {
5242 target_stack_index++;
5243 int test_db_index =
5244 m_pQuilt->GetExtendedStackIndexArray()[target_stack_index];
5245
5246 if ((ref_family == ChartData->GetDBChartFamily(test_db_index)) &&
5247 IsChartQuiltableRef(test_db_index)) {
5248 // open the target, and check the min_scale
5249 ChartBase *ptest_chart =
5250 ChartData->OpenChartFromDB(test_db_index, FULL_INIT);
5251 if (ptest_chart) {
5252 brender_ok = IsChartLargeEnoughToRender(ptest_chart, VPoint);
5253 }
5254 }
5255 }
5256
5257 if (brender_ok) { // found a better reference chart
5258 int new_db_index =
5259 m_pQuilt->GetExtendedStackIndexArray()[target_stack_index];
5260 if ((ref_family == ChartData->GetDBChartFamily(new_db_index)) &&
5261 IsChartQuiltableRef(new_db_index)) {
5262 m_pQuilt->SetReferenceChart(new_db_index);
5263 ret = new_db_index;
5264 } else
5265 ret = m_pQuilt->GetRefChartdbIndex();
5266 } else
5267 ret = m_pQuilt->GetRefChartdbIndex();
5268
5269 } else
5270 ret = m_pQuilt->GetRefChartdbIndex();
5271 }
5272 } else
5273 ret = -1;
5274 }
5275
5276 return ret;
5277}
5278
5279void ChartCanvas::UpdateCanvasOnGroupChange(void) {
5280 delete m_pCurrentStack;
5281 m_pCurrentStack = new ChartStack;
5282 wxASSERT(ChartData);
5283 ChartData->BuildChartStack(m_pCurrentStack, VPoint.clat, VPoint.clon,
5284 m_groupIndex);
5285
5286 if (m_pQuilt) {
5287 m_pQuilt->Compose(VPoint);
5288 SetFocus();
5289 }
5290}
5291
5292bool ChartCanvas::SetViewPointByCorners(double latSW, double lonSW,
5293 double latNE, double lonNE) {
5294 // Center Point
5295 double latc = (latSW + latNE) / 2.0;
5296 double lonc = (lonSW + lonNE) / 2.0;
5297
5298 // Get scale in ppm (latitude)
5299 double ne_easting, ne_northing;
5300 toSM(latNE, lonNE, latc, lonc, &ne_easting, &ne_northing);
5301
5302 double sw_easting, sw_northing;
5303 toSM(latSW, lonSW, latc, lonc, &sw_easting, &sw_northing);
5304
5305 double scale_ppm = VPoint.pix_height / fabs(ne_northing - sw_northing);
5306
5307 return SetViewPoint(latc, lonc, scale_ppm, VPoint.skew, VPoint.rotation);
5308}
5309
5310bool ChartCanvas::SetVPScale(double scale, bool refresh) {
5311 return SetViewPoint(VPoint.clat, VPoint.clon, scale, VPoint.skew,
5312 VPoint.rotation, VPoint.m_projection_type, true, refresh);
5313}
5314
5315bool ChartCanvas::SetVPProjection(int projection) {
5316 if (!g_bopengl) // alternative projections require opengl
5317 return false;
5318
5319 // the view scale varies depending on geographic location and projection
5320 // rescale to keep the relative scale on the screen the same
5321 double prev_true_scale_ppm = m_true_scale_ppm;
5322 return SetViewPoint(VPoint.clat, VPoint.clon, VPoint.view_scale_ppm,
5323 VPoint.skew, VPoint.rotation, projection) &&
5324 SetVPScale(wxMax(
5325 VPoint.view_scale_ppm * prev_true_scale_ppm / m_true_scale_ppm,
5326 m_absolute_min_scale_ppm));
5327}
5328
5329bool ChartCanvas::SetViewPoint(double lat, double lon) {
5330 return SetViewPoint(lat, lon, VPoint.view_scale_ppm, VPoint.skew,
5331 VPoint.rotation);
5332}
5333
5334bool ChartCanvas::SetVPRotation(double angle) {
5335 return SetViewPoint(VPoint.clat, VPoint.clon, VPoint.view_scale_ppm,
5336 VPoint.skew, angle);
5337}
5338bool ChartCanvas::SetViewPoint(double lat, double lon, double scale_ppm,
5339 double skew, double rotation, int projection,
5340 bool b_adjust, bool b_refresh) {
5341 bool b_ret = false;
5342 if (skew > PI) /* so our difference tests work, put in range of +-Pi */
5343 skew -= 2 * PI;
5344 // Any sensible change?
5345 if (VPoint.IsValid()) {
5346 if ((fabs(VPoint.view_scale_ppm - scale_ppm) / scale_ppm < 1e-5) &&
5347 (fabs(VPoint.skew - skew) < 1e-9) &&
5348 (fabs(VPoint.rotation - rotation) < 1e-9) &&
5349 (fabs(VPoint.clat - lat) < 1e-9) && (fabs(VPoint.clon - lon) < 1e-9) &&
5350 (VPoint.m_projection_type == projection ||
5351 projection == PROJECTION_UNKNOWN))
5352 return false;
5353 }
5354 if (VPoint.m_projection_type != projection)
5355 VPoint.InvalidateTransformCache(); // invalidate
5356
5357 // Take a local copy of the last viewport
5358 ViewPort last_vp = VPoint;
5359
5360 VPoint.skew = skew;
5361 VPoint.clat = lat;
5362 VPoint.clon = lon;
5363 VPoint.rotation = rotation;
5364 VPoint.view_scale_ppm = scale_ppm;
5365 if (projection != PROJECTION_UNKNOWN)
5366 VPoint.SetProjectionType(projection);
5367 else if (VPoint.m_projection_type == PROJECTION_UNKNOWN)
5368 VPoint.SetProjectionType(PROJECTION_MERCATOR);
5369
5370 // don't allow latitude above 88 for mercator (90 is infinity)
5371 if (VPoint.m_projection_type == PROJECTION_MERCATOR ||
5372 VPoint.m_projection_type == PROJECTION_TRANSVERSE_MERCATOR) {
5373 if (VPoint.clat > 89.5)
5374 VPoint.clat = 89.5;
5375 else if (VPoint.clat < -89.5)
5376 VPoint.clat = -89.5;
5377 }
5378
5379 // don't zoom out too far for transverse mercator polyconic until we resolve
5380 // issues
5381 if (VPoint.m_projection_type == PROJECTION_POLYCONIC ||
5382 VPoint.m_projection_type == PROJECTION_TRANSVERSE_MERCATOR)
5383 VPoint.view_scale_ppm = wxMax(VPoint.view_scale_ppm, 2e-4);
5384
5385 // SetVPRotation(rotation);
5386
5387 if (!g_bopengl) // tilt is not possible without opengl
5388 VPoint.tilt = 0;
5389
5390 if ((VPoint.pix_width <= 0) ||
5391 (VPoint.pix_height <= 0)) // Canvas parameters not yet set
5392 return false;
5393
5394 bool bwasValid = VPoint.IsValid();
5395 VPoint.Validate(); // Mark this ViewPoint as OK
5396
5397 // Has the Viewport scale changed? If so, invalidate the vp
5398 if (last_vp.view_scale_ppm != scale_ppm) {
5399 m_cache_vp.Invalidate();
5400 InvalidateGL();
5401 }
5402
5403 // A preliminary value, may be tweaked below
5404 VPoint.chart_scale = m_canvas_scale_factor / (scale_ppm);
5405
5406 // recompute cursor position
5407 // and send to interested plugins if the mouse is actually in this window
5408 int mouseX = mouse_x;
5409 int mouseY = mouse_y;
5410 if ((mouseX > 0) && (mouseX < VPoint.pix_width) && (mouseY > 0) &&
5411 (mouseY < VPoint.pix_height)) {
5412 double lat, lon;
5413 GetCanvasPixPoint(mouseX, mouseY, lat, lon);
5414 m_cursor_lat = lat;
5415 m_cursor_lon = lon;
5416 SendCursorLatLonToAllPlugIns(lat, lon);
5417 }
5418
5419 if (!VPoint.b_quilt && m_singleChart) {
5420 VPoint.SetBoxes();
5421
5422 // Allow the chart to adjust the new ViewPort for performance optimization
5423 // This will normally be only a fractional (i.e.sub-pixel) adjustment...
5424 if (b_adjust) m_singleChart->AdjustVP(last_vp, VPoint);
5425
5426 // If there is a sensible change in the chart render, refresh the whole
5427 // screen
5428 if ((!m_cache_vp.IsValid()) ||
5429 (m_cache_vp.view_scale_ppm != VPoint.view_scale_ppm)) {
5430 Refresh(false);
5431 b_ret = true;
5432 } else {
5433 wxPoint cp_last, cp_this;
5434 GetCanvasPointPix(m_cache_vp.clat, m_cache_vp.clon, &cp_last);
5435 GetCanvasPointPix(VPoint.clat, VPoint.clon, &cp_this);
5436
5437 if (cp_last != cp_this) {
5438 Refresh(false);
5439 b_ret = true;
5440 }
5441 }
5442 // Create the stack
5443 if (m_pCurrentStack) {
5444 assert(ChartData != 0);
5445 int current_db_index;
5446 current_db_index =
5447 m_pCurrentStack->GetCurrentEntrydbIndex(); // capture the current
5448
5449 ChartData->BuildChartStack(m_pCurrentStack, lat, lon, current_db_index,
5450 m_groupIndex);
5451 m_pCurrentStack->SetCurrentEntryFromdbIndex(current_db_index);
5452 }
5453
5454 if (!g_bopengl) VPoint.b_MercatorProjectionOverride = false;
5455 }
5456
5457 // Handle the quilted case
5458 if (VPoint.b_quilt) {
5459 if (last_vp.view_scale_ppm != scale_ppm)
5460 m_pQuilt->InvalidateAllQuiltPatchs();
5461
5462 // Create the quilt
5463 if (ChartData /*&& ChartData->IsValid()*/) {
5464 if (!m_pCurrentStack) return false;
5465
5466 int current_db_index;
5467 current_db_index =
5468 m_pCurrentStack->GetCurrentEntrydbIndex(); // capture the current
5469
5470 ChartData->BuildChartStack(m_pCurrentStack, lat, lon, m_groupIndex);
5471 m_pCurrentStack->SetCurrentEntryFromdbIndex(current_db_index);
5472
5473 // Check to see if the current quilt reference chart is in the new stack
5474 int current_ref_stack_index = -1;
5475 for (int i = 0; i < m_pCurrentStack->nEntry; i++) {
5476 if (m_pQuilt->GetRefChartdbIndex() == m_pCurrentStack->GetDBIndex(i))
5477 current_ref_stack_index = i;
5478 }
5479
5480 if (g_bFullScreenQuilt) {
5481 current_ref_stack_index = m_pQuilt->GetRefChartdbIndex();
5482 }
5483
5484 // We might need a new Reference Chart
5485 bool b_needNewRef = false;
5486
5487 // If the new stack does not contain the current ref chart....
5488 if ((-1 == current_ref_stack_index) &&
5489 (m_pQuilt->GetRefChartdbIndex() >= 0))
5490 b_needNewRef = true;
5491
5492 // Would the current Ref Chart be excessively underzoomed?
5493 // We need to check this here to be sure, since we cannot know where the
5494 // reference chart was assigned. For instance, the reference chart may
5495 // have been selected from the config file, or from a long jump with a
5496 // chart family switch implicit. Anyway, we check to be sure....
5497 bool renderable = true;
5498 ChartBase *referenceChart =
5499 ChartData->OpenChartFromDB(m_pQuilt->GetRefChartdbIndex(), FULL_INIT);
5500 if (referenceChart) {
5501 double chartMaxScale = referenceChart->GetNormalScaleMax(
5502 GetCanvasScaleFactor(), GetCanvasWidth());
5503 renderable = chartMaxScale * 64 >= VPoint.chart_scale;
5504 }
5505 if (!renderable) b_needNewRef = true;
5506
5507 // Need new refchart?
5508 if (b_needNewRef) {
5509 const ChartTableEntry &cte_ref =
5510 ChartData->GetChartTableEntry(m_pQuilt->GetRefChartdbIndex());
5511 int target_scale = cte_ref.GetScale();
5512 int target_type = cte_ref.GetChartType();
5513 int candidate_stack_index;
5514
5515 // reset the ref chart in a way that does not lead to excessive
5516 // underzoom, for performance reasons Try to find a chart that is the
5517 // same type, and has a scale of just smaller than the current ref
5518 // chart
5519
5520 candidate_stack_index = 0;
5521 while (candidate_stack_index <= m_pCurrentStack->nEntry - 1) {
5522 const ChartTableEntry &cte_candidate = ChartData->GetChartTableEntry(
5523 m_pCurrentStack->GetDBIndex(candidate_stack_index));
5524 int candidate_scale = cte_candidate.GetScale();
5525 int candidate_type = cte_candidate.GetChartType();
5526
5527 if ((candidate_scale >= target_scale) &&
5528 (candidate_type == target_type)) {
5529 bool renderable = true;
5530 ChartBase *tentative_referenceChart = ChartData->OpenChartFromDB(
5531 m_pCurrentStack->GetDBIndex(candidate_stack_index), FULL_INIT);
5532 if (tentative_referenceChart) {
5533 double chartMaxScale =
5534 tentative_referenceChart->GetNormalScaleMax(
5535 GetCanvasScaleFactor(), GetCanvasWidth());
5536 renderable = chartMaxScale * 1.5 > VPoint.chart_scale;
5537 }
5538
5539 if (renderable) break;
5540 }
5541
5542 candidate_stack_index++;
5543 }
5544
5545 // If that did not work, look for a chart of just larger scale and
5546 // same type
5547 if (candidate_stack_index >= m_pCurrentStack->nEntry) {
5548 candidate_stack_index = m_pCurrentStack->nEntry - 1;
5549 while (candidate_stack_index >= 0) {
5550 int idx = m_pCurrentStack->GetDBIndex(candidate_stack_index);
5551 if (idx >= 0) {
5552 const ChartTableEntry &cte_candidate =
5553 ChartData->GetChartTableEntry(idx);
5554 int candidate_scale = cte_candidate.GetScale();
5555 int candidate_type = cte_candidate.GetChartType();
5556
5557 if ((candidate_scale <= target_scale) &&
5558 (candidate_type == target_type))
5559 break;
5560 }
5561 candidate_stack_index--;
5562 }
5563 }
5564
5565 // and if that did not work, chose stack entry 0
5566 if ((candidate_stack_index >= m_pCurrentStack->nEntry) ||
5567 (candidate_stack_index < 0))
5568 candidate_stack_index = 0;
5569
5570 int new_ref_index = m_pCurrentStack->GetDBIndex(candidate_stack_index);
5571
5572 m_pQuilt->SetReferenceChart(new_ref_index); // maybe???
5573 }
5574
5575 if (!g_bopengl) {
5576 // Preset the VPoint projection type to match what the quilt projection
5577 // type will be
5578 int ref_db_index = m_pQuilt->GetRefChartdbIndex(), proj;
5579
5580 // Always keep the default Mercator projection if the reference chart is
5581 // not in the PatchList or the scale is too small for it to render.
5582
5583 bool renderable = true;
5584 ChartBase *referenceChart =
5585 ChartData->OpenChartFromDB(ref_db_index, FULL_INIT);
5586 if (referenceChart) {
5587 double chartMaxScale = referenceChart->GetNormalScaleMax(
5588 GetCanvasScaleFactor(), GetCanvasWidth());
5589 renderable = chartMaxScale * 1.5 > VPoint.chart_scale;
5590 proj = ChartData->GetDBChartProj(ref_db_index);
5591 } else
5592 proj = PROJECTION_MERCATOR;
5593
5594 VPoint.b_MercatorProjectionOverride =
5595 (m_pQuilt->GetnCharts() == 0 || !renderable);
5596
5597 if (VPoint.b_MercatorProjectionOverride) proj = PROJECTION_MERCATOR;
5598
5599 VPoint.SetProjectionType(proj);
5600 }
5601
5602 VPoint.SetBoxes();
5603
5604 // If this quilt will be a perceptible delta from the existing quilt,
5605 // then refresh the entire screen
5606 if (m_pQuilt->IsQuiltDelta(VPoint)) {
5607 // Allow the quilt to adjust the new ViewPort for performance
5608 // optimization This will normally be only a fractional (i.e.
5609 // sub-pixel) adjustment...
5610 if (b_adjust) {
5611 m_pQuilt->AdjustQuiltVP(last_vp, VPoint);
5612 }
5613
5614 // ChartData->ClearCacheInUseFlags();
5615 // unsigned long hash1 = m_pQuilt->GetXStackHash();
5616
5617 // wxStopWatch sw;
5618
5619#ifdef __ANDROID__
5620 // This is an optimization for panning on touch screen systems.
5621 // The quilt composition is deferred until the OnPaint() message gets
5622 // finally removed and processed from the message queue.
5623 // Takes advantage of the fact that touch-screen pan gestures are
5624 // usually short in distance,
5625 // so not requiring a full quilt rebuild until the pan gesture is
5626 // complete.
5627 if ((last_vp.view_scale_ppm != scale_ppm) || !bwasValid) {
5628 // qDebug() << "Force compose";
5629 m_pQuilt->Compose(VPoint);
5630 } else {
5631 m_pQuilt->Invalidate();
5632 }
5633#else
5634 m_pQuilt->Compose(VPoint);
5635#endif
5636
5637 // printf("comp time %ld\n", sw.Time());
5638
5639 // If the extended chart stack has changed, invalidate any cached
5640 // render bitmap
5641 // if(m_pQuilt->GetXStackHash() != hash1) {
5642 // m_bm_cache_vp.Invalidate();
5643 // InvalidateGL();
5644 // }
5645
5646 ChartData->PurgeCacheUnusedCharts(0.7);
5647
5648 if (b_refresh) Refresh(false);
5649
5650 b_ret = true;
5651 }
5652 }
5653
5654 VPoint.skew = 0.; // Quilting supports 0 Skew
5655 } else if (!g_bopengl) {
5656 OcpnProjType projection = PROJECTION_UNKNOWN;
5657 if (m_singleChart) // viewport projection must match chart projection
5658 // without opengl
5659 projection = m_singleChart->GetChartProjectionType();
5660 if (projection == PROJECTION_UNKNOWN) projection = PROJECTION_MERCATOR;
5661 VPoint.SetProjectionType(projection);
5662 }
5663
5664 // Has the Viewport projection changed? If so, invalidate the vp
5665 if (last_vp.m_projection_type != VPoint.m_projection_type) {
5666 m_cache_vp.Invalidate();
5667 InvalidateGL();
5668 }
5669
5670 UpdateCanvasControlBar(); // Refresh the Piano
5671
5672 VPoint.chart_scale = 1.0; // fallback default value
5673
5674 /*if (!VPoint.GetBBox().GetValid())*/ VPoint.SetBoxes();
5675
5676 if (VPoint.GetBBox().GetValid()) {
5677 // Update the viewpoint reference scale
5678 if (m_singleChart)
5679 VPoint.ref_scale = m_singleChart->GetNativeScale();
5680 else {
5681#ifdef __ANDROID__
5682 // This is an optimization for panning on touch screen systems.
5683 // See above.
5684 // Quilt might not be fully composed at this point, so for cm93
5685 // the reference scale may not be known.
5686 // In this case, do not update the VP ref_scale.
5687 if ((last_vp.view_scale_ppm != scale_ppm) || !bwasValid) {
5688 VPoint.ref_scale = m_pQuilt->GetRefNativeScale();
5689 }
5690#else
5691 VPoint.ref_scale = m_pQuilt->GetRefNativeScale();
5692#endif
5693 }
5694
5695 // Calculate the on-screen displayed actual scale
5696 // by a simple traverse northward from the center point
5697 // of roughly one eighth of the canvas height
5698 wxPoint2DDouble r, r1; // Screen coordinates in physical pixels.
5699
5700 double delta_check =
5701 (VPoint.pix_height / VPoint.view_scale_ppm) / (1852. * 60);
5702 delta_check /= 8.;
5703
5704 double check_point = wxMin(89., VPoint.clat);
5705
5706 while ((delta_check + check_point) > 90.) delta_check /= 2.;
5707
5708 double rhumbDist;
5709 DistanceBearingMercator(check_point, VPoint.clon, check_point + delta_check,
5710 VPoint.clon, 0, &rhumbDist);
5711
5712 GetDoubleCanvasPointPix(check_point, VPoint.clon, &r1);
5713 GetDoubleCanvasPointPix(check_point + delta_check, VPoint.clon, &r);
5714 // Calculate the distance between r1 and r in physical pixels.
5715 double delta_p = sqrt(((r1.m_y - r.m_y) * (r1.m_y - r.m_y)) +
5716 ((r1.m_x - r.m_x) * (r1.m_x - r.m_x)));
5717
5718 m_true_scale_ppm = delta_p / (rhumbDist * 1852);
5719
5720 // A fall back in case of very high zoom-out, giving delta_y == 0
5721 // which can probably only happen with vector charts
5722 if (0.0 == m_true_scale_ppm) m_true_scale_ppm = scale_ppm;
5723
5724 // Another fallback, for highly zoomed out charts
5725 // This adjustment makes the displayed TrueScale correspond to the
5726 // same algorithm used to calculate the chart zoom-out limit for
5727 // ChartDummy.
5728 if (scale_ppm < 1e-4) m_true_scale_ppm = scale_ppm;
5729
5730 if (m_true_scale_ppm)
5731 VPoint.chart_scale = m_canvas_scale_factor / (m_true_scale_ppm);
5732 else
5733 VPoint.chart_scale = 1.0;
5734
5735 // Create a nice renderable string
5736 double round_factor = 1000.;
5737 if (VPoint.chart_scale <= 1000.)
5738 round_factor = 10.;
5739 else if (VPoint.chart_scale <= 10000.)
5740 round_factor = 100.;
5741 else if (VPoint.chart_scale <= 100000.)
5742 round_factor = 1000.;
5743
5744 // Fixme: Workaround the wrongly calculated scale on Retina displays (#3117)
5745 double retina_coef = 1;
5746#ifdef ocpnUSE_GL
5747#ifdef __WXOSX__
5748 if (g_bopengl) {
5749 retina_coef = GetContentScaleFactor();
5750 }
5751#endif
5752#endif
5753
5754 // The chart scale denominator (e.g., 50000 if the scale is 1:50000),
5755 // rounded to the nearest 10, 100 or 1000.
5756 //
5757 // @todo: on MacOS there is 2x ratio between VPoint.chart_scale and
5758 // true_scale_display. That does not make sense. The chart scale should be
5759 // the same as the true scale within the limits of the rounding factor.
5760 double true_scale_display =
5761 wxRound(VPoint.chart_scale / round_factor) * round_factor * retina_coef;
5762 wxString text;
5763
5764 m_displayed_scale_factor = VPoint.ref_scale / VPoint.chart_scale;
5765
5766 if (m_displayed_scale_factor > 10.0)
5767 text.Printf(_T("%s %4.0f (%1.0fx)"), _("Scale"), true_scale_display,
5768 m_displayed_scale_factor);
5769 else if (m_displayed_scale_factor > 1.0)
5770 text.Printf(_T("%s %4.0f (%1.1fx)"), _("Scale"), true_scale_display,
5771 m_displayed_scale_factor);
5772 else if (m_displayed_scale_factor > 0.1) {
5773 double sfr = wxRound(m_displayed_scale_factor * 10.) / 10.;
5774 text.Printf(_T("%s %4.0f (%1.2fx)"), _("Scale"), true_scale_display, sfr);
5775 } else if (m_displayed_scale_factor > 0.01) {
5776 double sfr = wxRound(m_displayed_scale_factor * 100.) / 100.;
5777 text.Printf(_T("%s %4.0f (%1.2fx)"), _("Scale"), true_scale_display, sfr);
5778 } else {
5779 text.Printf(
5780 _T("%s %4.0f (---)"), _("Scale"),
5781 true_scale_display); // Generally, no chart, so no chart scale factor
5782 }
5783
5784 m_scaleValue = true_scale_display;
5785 m_scaleText = text;
5786 if (m_muiBar) m_muiBar->UpdateDynamicValues();
5787
5788 if (m_bShowScaleInStatusBar && parent_frame->GetStatusBar() &&
5789 (parent_frame->GetStatusBar()->GetFieldsCount() > STAT_FIELD_SCALE)) {
5790 // Check to see if the text will fit in the StatusBar field...
5791 bool b_noshow = false;
5792 {
5793 int w = 0;
5794 int h;
5795 wxClientDC dc(parent_frame->GetStatusBar());
5796 if (dc.IsOk()) {
5797 wxFont *templateFont = FontMgr::Get().GetFont(_("StatusBar"), 0);
5798 dc.SetFont(*templateFont);
5799 dc.GetTextExtent(text, &w, &h);
5800
5801 // If text is too long for the allocated field, try to reduce the text
5802 // string a bit.
5803 wxRect rect;
5804 parent_frame->GetStatusBar()->GetFieldRect(STAT_FIELD_SCALE, rect);
5805 if (w && w > rect.width) {
5806 text.Printf(_T("%s (%1.1fx)"), _("Scale"),
5807 m_displayed_scale_factor);
5808 }
5809
5810 // Test again...if too big still, then give it up.
5811 dc.GetTextExtent(text, &w, &h);
5812
5813 if (w && w > rect.width) {
5814 b_noshow = true;
5815 }
5816 }
5817 }
5818
5819 if (!b_noshow) parent_frame->SetStatusText(text, STAT_FIELD_SCALE);
5820 }
5821 }
5822
5823 // Maintain member vLat/vLon
5824 m_vLat = VPoint.clat;
5825 m_vLon = VPoint.clon;
5826
5827 return b_ret;
5828}
5829
5830// Static Icon definitions for some symbols requiring
5831// scaling/rotation/translation Very specific wxDC draw commands are
5832// necessary to properly render these icons...See the code in
5833// ShipDraw()
5834
5835// This icon was adapted and scaled from the S52 Presentation Library
5836// version 3_03.
5837// Symbol VECGND02
5838
5839static int s_png_pred_icon[] = {-10, -10, -10, 10, 10, 10, 10, -10};
5840
5841// This ownship icon was adapted and scaled from the S52 Presentation
5842// Library version 3_03 Symbol OWNSHP05
5843static int s_ownship_icon[] = {5, -42, 11, -28, 11, 42, -11, 42, -11, -28,
5844 -5, -42, -11, 0, 11, 0, 0, 42, 0, -42};
5845
5846wxColour ChartCanvas::PredColor() {
5847 // RAdjust predictor color change on LOW_ACCURACY ship state in interests of
5848 // visibility.
5849 if (SHIP_NORMAL == m_ownship_state)
5850 return GetGlobalColor(_T ( "URED" ));
5851
5852 else if (SHIP_LOWACCURACY == m_ownship_state)
5853 return GetGlobalColor(_T ( "YELO1" ));
5854
5855 return GetGlobalColor(_T ( "NODTA" ));
5856}
5857
5858wxColour ChartCanvas::ShipColor() {
5859 // Establish ship color
5860 // It changes color based on GPS and Chart accuracy/availability
5861
5862 if (SHIP_NORMAL != m_ownship_state) return GetGlobalColor(_T ( "GREY1" ));
5863
5864 if (SHIP_LOWACCURACY == m_ownship_state)
5865 return GetGlobalColor(_T ( "YELO1" ));
5866
5867 return GetGlobalColor(_T ( "URED" )); // default is OK
5868}
5869
5870void ChartCanvas::ShipDrawLargeScale(ocpnDC &dc,
5871 wxPoint2DDouble lShipMidPoint) {
5872 dc.SetPen(wxPen(PredColor(), 2));
5873
5874 if (SHIP_NORMAL == m_ownship_state)
5875 dc.SetBrush(wxBrush(ShipColor(), wxBRUSHSTYLE_TRANSPARENT));
5876 else
5877 dc.SetBrush(wxBrush(GetGlobalColor(_T ( "YELO1" ))));
5878
5879 dc.DrawEllipse(lShipMidPoint.m_x - 10, lShipMidPoint.m_y - 10, 20, 20);
5880 dc.DrawEllipse(lShipMidPoint.m_x - 6, lShipMidPoint.m_y - 6, 12, 12);
5881
5882 dc.DrawLine(lShipMidPoint.m_x - 12, lShipMidPoint.m_y, lShipMidPoint.m_x + 12,
5883 lShipMidPoint.m_y);
5884 dc.DrawLine(lShipMidPoint.m_x, lShipMidPoint.m_y - 12, lShipMidPoint.m_x,
5885 lShipMidPoint.m_y + 12);
5886}
5887
5888void ChartCanvas::ShipIndicatorsDraw(ocpnDC &dc, int img_height,
5889 wxPoint GPSOffsetPixels,
5890 wxPoint2DDouble lGPSPoint) {
5891 if (m_animationActive) return;
5892 // Develop a uniform length for course predictor line dash length, based on
5893 // physical display size Use this reference length to size all other graphics
5894 // elements
5895 float ref_dim = m_display_size_mm / 24;
5896 ref_dim = wxMin(ref_dim, 12);
5897 ref_dim = wxMax(ref_dim, 6);
5898
5899 wxColour cPred;
5900 cPred.Set(g_cog_predictor_color);
5901 if (cPred == wxNullColour) cPred = PredColor();
5902
5903 // Establish some graphic element line widths dependent on the platform
5904 // display resolution
5905 // double nominal_line_width_pix = wxMax(1.0,
5906 // floor(g_Platform->GetDisplayDPmm() / 2)); //0.5 mm nominal, but
5907 // not less than 1 pixel
5908 double nominal_line_width_pix = wxMax(
5909 1.0,
5910 floor(m_pix_per_mm / 2)); // 0.5 mm nominal, but not less than 1 pixel
5911
5912 // If the calculated value is greater than the config file spec value, then
5913 // use it.
5914 if (nominal_line_width_pix > g_cog_predictor_width)
5915 g_cog_predictor_width = nominal_line_width_pix;
5916
5917 // Calculate ownship Position Predictor
5918 wxPoint lPredPoint, lHeadPoint;
5919
5920 float pCog = std::isnan(gCog) ? 0 : gCog;
5921 float pSog = std::isnan(gSog) ? 0 : gSog;
5922
5923 double pred_lat, pred_lon;
5924 ll_gc_ll(gLat, gLon, pCog, pSog * g_ownship_predictor_minutes / 60.,
5925 &pred_lat, &pred_lon);
5926 GetCanvasPointPix(pred_lat, pred_lon, &lPredPoint);
5927
5928 // test to catch the case where COG/HDG line crosses the screen
5929 LLBBox box;
5930
5931 // Should we draw the Head vector?
5932 // Compare the points lHeadPoint and lPredPoint
5933 // If they differ by more than n pixels, and the head vector is valid, then
5934 // render the head vector
5935
5936 float ndelta_pix = 10.;
5937 double hdg_pred_lat, hdg_pred_lon;
5938 bool b_render_hdt = false;
5939 if (!std::isnan(gHdt)) {
5940 // Calculate ownship Heading pointer as a predictor
5941 ll_gc_ll(gLat, gLon, gHdt, g_ownship_HDTpredictor_miles, &hdg_pred_lat,
5942 &hdg_pred_lon);
5943 GetCanvasPointPix(hdg_pred_lat, hdg_pred_lon, &lHeadPoint);
5944 float dist = sqrtf(powf((float)(lHeadPoint.x - lPredPoint.x), 2) +
5945 powf((float)(lHeadPoint.y - lPredPoint.y), 2));
5946 if (dist > ndelta_pix /*&& !std::isnan(gSog)*/) {
5947 box.SetFromSegment(gLat, gLon, hdg_pred_lat, hdg_pred_lon);
5948 if (!GetVP().GetBBox().IntersectOut(box)) b_render_hdt = true;
5949 }
5950 }
5951
5952 // draw course over ground if they are longer than the ship
5953 wxPoint lShipMidPoint;
5954 lShipMidPoint.x = lGPSPoint.m_x + GPSOffsetPixels.x;
5955 lShipMidPoint.y = lGPSPoint.m_y + GPSOffsetPixels.y;
5956 float lpp = sqrtf(powf((float)(lPredPoint.x - lShipMidPoint.x), 2) +
5957 powf((float)(lPredPoint.y - lShipMidPoint.y), 2));
5958
5959 if (lpp >= img_height / 2) {
5960 box.SetFromSegment(gLat, gLon, pred_lat, pred_lon);
5961 if (!GetVP().GetBBox().IntersectOut(box) && !std::isnan(gCog) &&
5962 !std::isnan(gSog)) {
5963 // COG Predictor
5964 float dash_length = ref_dim;
5965 wxDash dash_long[2];
5966 dash_long[0] =
5967 (int)(floor(g_Platform->GetDisplayDPmm() * dash_length) /
5968 g_cog_predictor_width); // Long dash , in mm <---------+
5969 dash_long[1] = dash_long[0] / 2.0; // Short gap
5970
5971 // On ultra-hi-res displays, do not allow the dashes to be greater than
5972 // 250, since it is defined as (char)
5973 if (dash_length > 250.) {
5974 dash_long[0] = 250. / g_cog_predictor_width;
5975 dash_long[1] = dash_long[0] / 2;
5976 }
5977
5978 wxPen ppPen2(cPred, g_cog_predictor_width,
5979 (wxPenStyle)g_cog_predictor_style);
5980 if (g_cog_predictor_style == (wxPenStyle)wxUSER_DASH)
5981 ppPen2.SetDashes(2, dash_long);
5982 dc.SetPen(ppPen2);
5983 dc.StrokeLine(
5984 lGPSPoint.m_x + GPSOffsetPixels.x, lGPSPoint.m_y + GPSOffsetPixels.y,
5985 lPredPoint.x + GPSOffsetPixels.x, lPredPoint.y + GPSOffsetPixels.y);
5986
5987 if (g_cog_predictor_width > 1) {
5988 float line_width = g_cog_predictor_width / 3.;
5989
5990 wxDash dash_long3[2];
5991 dash_long3[0] = g_cog_predictor_width / line_width * dash_long[0];
5992 dash_long3[1] = g_cog_predictor_width / line_width * dash_long[1];
5993
5994 wxPen ppPen3(GetGlobalColor(_T ( "UBLCK" )), wxMax(1, line_width),
5995 (wxPenStyle)g_cog_predictor_style);
5996 if (g_cog_predictor_style == (wxPenStyle)wxUSER_DASH)
5997 ppPen3.SetDashes(2, dash_long3);
5998 dc.SetPen(ppPen3);
5999 dc.StrokeLine(lGPSPoint.m_x + GPSOffsetPixels.x,
6000 lGPSPoint.m_y + GPSOffsetPixels.y,
6001 lPredPoint.x + GPSOffsetPixels.x,
6002 lPredPoint.y + GPSOffsetPixels.y);
6003 }
6004
6005 if (g_cog_predictor_endmarker) {
6006 // Prepare COG predictor endpoint icon
6007 double png_pred_icon_scale_factor = .4;
6008 if (g_ShipScaleFactorExp > 1.0)
6009 png_pred_icon_scale_factor *= (log(g_ShipScaleFactorExp) + 1.0) * 1.1;
6010 if (g_scaler) png_pred_icon_scale_factor *= 1.0 / g_scaler;
6011
6012 wxPoint icon[4];
6013
6014 float cog_rad = atan2f((float)(lPredPoint.y - lShipMidPoint.y),
6015 (float)(lPredPoint.x - lShipMidPoint.x));
6016 cog_rad += (float)PI;
6017
6018 for (int i = 0; i < 4; i++) {
6019 int j = i * 2;
6020 double pxa = (double)(s_png_pred_icon[j]);
6021 double pya = (double)(s_png_pred_icon[j + 1]);
6022
6023 pya *= png_pred_icon_scale_factor;
6024 pxa *= png_pred_icon_scale_factor;
6025
6026 double px = (pxa * sin(cog_rad)) + (pya * cos(cog_rad));
6027 double py = (pya * sin(cog_rad)) - (pxa * cos(cog_rad));
6028
6029 icon[i].x = (int)wxRound(px) + lPredPoint.x + GPSOffsetPixels.x;
6030 icon[i].y = (int)wxRound(py) + lPredPoint.y + GPSOffsetPixels.y;
6031 }
6032
6033 // Render COG endpoint icon
6034 wxPen ppPen1(GetGlobalColor(_T ( "UBLCK" )), g_cog_predictor_width / 2,
6035 wxPENSTYLE_SOLID);
6036 dc.SetPen(ppPen1);
6037 dc.SetBrush(wxBrush(cPred));
6038
6039 dc.StrokePolygon(4, icon);
6040 }
6041 }
6042 }
6043
6044 // HDT Predictor
6045 if (b_render_hdt) {
6046 float hdt_dash_length = ref_dim * 0.4;
6047
6048 cPred.Set(g_ownship_HDTpredictor_color);
6049 if (cPred == wxNullColour) cPred = PredColor();
6050 float hdt_width =
6051 (g_ownship_HDTpredictor_width > 0 ? g_ownship_HDTpredictor_width
6052 : g_cog_predictor_width * 0.8);
6053 wxDash dash_short[2];
6054 dash_short[0] =
6055 (int)(floor(g_Platform->GetDisplayDPmm() * hdt_dash_length) /
6056 hdt_width); // Short dash , in mm <---------+
6057 dash_short[1] =
6058 (int)(floor(g_Platform->GetDisplayDPmm() * hdt_dash_length * 0.9) /
6059 hdt_width); // Short gap |
6060
6061 wxPen ppPen2(cPred, hdt_width, (wxPenStyle)g_ownship_HDTpredictor_style);
6062 if (g_ownship_HDTpredictor_style == (wxPenStyle)wxUSER_DASH)
6063 ppPen2.SetDashes(2, dash_short);
6064
6065 dc.SetPen(ppPen2);
6066 dc.StrokeLine(
6067 lGPSPoint.m_x + GPSOffsetPixels.x, lGPSPoint.m_y + GPSOffsetPixels.y,
6068 lHeadPoint.x + GPSOffsetPixels.x, lHeadPoint.y + GPSOffsetPixels.y);
6069
6070 wxPen ppPen1(cPred, g_cog_predictor_width / 3, wxPENSTYLE_SOLID);
6071 dc.SetPen(ppPen1);
6072 dc.SetBrush(wxBrush(GetGlobalColor(_T ( "GREY2" ))));
6073
6074 if (g_ownship_HDTpredictor_endmarker) {
6075 double nominal_circle_size_pixels = wxMax(
6076 4.0, floor(m_pix_per_mm * (ref_dim / 5.0))); // not less than 4 pixel
6077
6078 // Scale the circle to ChartScaleFactor, slightly softened....
6079 if (g_ShipScaleFactorExp > 1.0)
6080 nominal_circle_size_pixels *= (log(g_ShipScaleFactorExp) + 1.0) * 1.1;
6081
6082 dc.StrokeCircle(lHeadPoint.x + GPSOffsetPixels.x,
6083 lHeadPoint.y + GPSOffsetPixels.y,
6084 nominal_circle_size_pixels / 2);
6085 }
6086 }
6087
6088 // Draw radar rings if activated
6089 if (g_bNavAidRadarRingsShown && g_iNavAidRadarRingsNumberVisible > 0) {
6090 double factor = 1.00;
6091 if (g_pNavAidRadarRingsStepUnits == 1) // kilometers
6092 factor = 1 / 1.852;
6093 else if (g_pNavAidRadarRingsStepUnits == 2) { // minutes (time)
6094 if (std::isnan(gSog))
6095 factor = 0.0;
6096 else
6097 factor = gSog / 60;
6098 }
6099 factor *= g_fNavAidRadarRingsStep;
6100
6101 double tlat, tlon;
6102 wxPoint r;
6103 ll_gc_ll(gLat, gLon, 0, factor, &tlat, &tlon);
6104 GetCanvasPointPix(tlat, tlon, &r);
6105
6106 double lpp = sqrt(pow((double)(lGPSPoint.m_x - r.x), 2) +
6107 pow((double)(lGPSPoint.m_y - r.y), 2));
6108 int pix_radius = (int)lpp;
6109
6110 extern wxColor GetDimColor(wxColor c);
6111 wxColor rangeringcolour = GetDimColor(g_colourOwnshipRangeRingsColour);
6112
6113 wxPen ppPen1(rangeringcolour, g_cog_predictor_width);
6114
6115 dc.SetPen(ppPen1);
6116 dc.SetBrush(wxBrush(rangeringcolour, wxBRUSHSTYLE_TRANSPARENT));
6117
6118 for (int i = 1; i <= g_iNavAidRadarRingsNumberVisible; i++)
6119 dc.StrokeCircle(lGPSPoint.m_x, lGPSPoint.m_y, i * pix_radius);
6120 }
6121}
6122
6123void ChartCanvas::ComputeShipScaleFactor(
6124 float icon_hdt, int ownShipWidth, int ownShipLength,
6125 wxPoint2DDouble &lShipMidPoint, wxPoint &GPSOffsetPixels,
6126 wxPoint2DDouble lGPSPoint, float &scale_factor_x, float &scale_factor_y) {
6127 float screenResolution = m_pix_per_mm;
6128
6129 // Calculate the true ship length in exact pixels
6130 double ship_bow_lat, ship_bow_lon;
6131 ll_gc_ll(gLat, gLon, icon_hdt, g_n_ownship_length_meters / 1852.,
6132 &ship_bow_lat, &ship_bow_lon);
6133 wxPoint lShipBowPoint;
6134 wxPoint2DDouble b_point =
6135 GetVP().GetDoublePixFromLL(ship_bow_lat, ship_bow_lon);
6136 wxPoint2DDouble a_point = GetVP().GetDoublePixFromLL(gLat, gLon);
6137
6138 float shipLength_px = sqrtf(powf((float)(b_point.m_x - a_point.m_x), 2) +
6139 powf((float)(b_point.m_y - a_point.m_y), 2));
6140
6141 // And in mm
6142 float shipLength_mm = shipLength_px / screenResolution;
6143
6144 // Set minimum ownship drawing size
6145 float ownship_min_mm = g_n_ownship_min_mm;
6146 ownship_min_mm = wxMax(ownship_min_mm, 1.0);
6147
6148 // Calculate Nautical Miles distance from midships to gps antenna
6149 float hdt_ant = icon_hdt + 180.;
6150 float dy = (g_n_ownship_length_meters / 2 - g_n_gps_antenna_offset_y) / 1852.;
6151 float dx = g_n_gps_antenna_offset_x / 1852.;
6152 if (g_n_gps_antenna_offset_y > g_n_ownship_length_meters / 2) // reverse?
6153 {
6154 hdt_ant = icon_hdt;
6155 dy = -dy;
6156 }
6157
6158 // If the drawn ship size is going to be clamped, adjust the gps antenna
6159 // offsets
6160 if (shipLength_mm < ownship_min_mm) {
6161 dy /= shipLength_mm / ownship_min_mm;
6162 dx /= shipLength_mm / ownship_min_mm;
6163 }
6164
6165 double ship_mid_lat, ship_mid_lon, ship_mid_lat1, ship_mid_lon1;
6166
6167 ll_gc_ll(gLat, gLon, hdt_ant, dy, &ship_mid_lat, &ship_mid_lon);
6168 ll_gc_ll(ship_mid_lat, ship_mid_lon, icon_hdt - 90., dx, &ship_mid_lat1,
6169 &ship_mid_lon1);
6170
6171 GetDoubleCanvasPointPixVP(GetVP(), ship_mid_lat1, ship_mid_lon1,
6172 &lShipMidPoint);
6173
6174 GPSOffsetPixels.x = lShipMidPoint.m_x - lGPSPoint.m_x;
6175 GPSOffsetPixels.y = lShipMidPoint.m_y - lGPSPoint.m_y;
6176
6177 float scale_factor = shipLength_px / ownShipLength;
6178
6179 // Calculate a scale factor that would produce a reasonably sized icon
6180 float scale_factor_min = ownship_min_mm / (ownShipLength / screenResolution);
6181
6182 // And choose the correct one
6183 scale_factor = wxMax(scale_factor, scale_factor_min);
6184
6185 scale_factor_y = scale_factor;
6186 scale_factor_x = scale_factor_y * ((float)ownShipLength / ownShipWidth) /
6187 ((float)g_n_ownship_length_meters / g_n_ownship_beam_meters);
6188}
6189
6190void ChartCanvas::ShipDraw(ocpnDC &dc) {
6191 if (!GetVP().IsValid()) return;
6192
6193 wxPoint GPSOffsetPixels(0, 0);
6194 wxPoint2DDouble lGPSPoint, lShipMidPoint;
6195
6196 // COG/SOG may be undefined in NMEA data stream
6197 float pCog = std::isnan(gCog) ? 0 : gCog;
6198 float pSog = std::isnan(gSog) ? 0 : gSog;
6199
6200 GetDoubleCanvasPointPixVP(GetVP(), gLat, gLon, &lGPSPoint);
6201
6202 lShipMidPoint = lGPSPoint;
6203
6204 // Draw the icon rotated to the COG
6205 // or to the Hdt if available
6206 float icon_hdt = pCog;
6207 if (!std::isnan(gHdt)) icon_hdt = gHdt;
6208
6209 // COG may be undefined in NMEA data stream
6210 if (std::isnan(icon_hdt)) icon_hdt = 0.0;
6211
6212 // Calculate the ownship drawing angle icon_rad using an assumed 10 minute
6213 // predictor
6214 double osd_head_lat, osd_head_lon;
6215 wxPoint osd_head_point;
6216
6217 ll_gc_ll(gLat, gLon, icon_hdt, pSog * 10. / 60., &osd_head_lat,
6218 &osd_head_lon);
6219
6220 GetCanvasPointPix(osd_head_lat, osd_head_lon, &osd_head_point);
6221
6222 float icon_rad = atan2f((float)(osd_head_point.y - lShipMidPoint.m_y),
6223 (float)(osd_head_point.x - lShipMidPoint.m_x));
6224 icon_rad += (float)PI;
6225
6226 if (pSog < 0.2) icon_rad = ((icon_hdt + 90.) * PI / 180) + GetVP().rotation;
6227
6228 // Another draw test ,based on pixels, assuming the ship icon is a fixed
6229 // nominal size and is just barely outside the viewport ....
6230 BoundingBox bb_screen(0, 0, GetVP().pix_width, GetVP().pix_height);
6231
6232 // TODO: fix to include actual size of boat that will be rendered
6233 int img_height = 0;
6234 if (bb_screen.PointInBox(lShipMidPoint, 20)) {
6235 if (GetVP().chart_scale >
6236 300000) // According to S52, this should be 50,000
6237 {
6238 ShipDrawLargeScale(dc, lShipMidPoint);
6239 img_height = 20;
6240 } else {
6241 wxImage pos_image;
6242
6243 // Substitute user ownship image if found
6244 if (m_pos_image_user)
6245 pos_image = m_pos_image_user->Copy();
6246 else if (SHIP_NORMAL == m_ownship_state)
6247 pos_image = m_pos_image_red->Copy();
6248 if (SHIP_LOWACCURACY == m_ownship_state)
6249 pos_image = m_pos_image_yellow->Copy();
6250 else if (SHIP_NORMAL != m_ownship_state)
6251 pos_image = m_pos_image_grey->Copy();
6252
6253 // Substitute user ownship image if found
6254 if (m_pos_image_user) {
6255 pos_image = m_pos_image_user->Copy();
6256
6257 if (SHIP_LOWACCURACY == m_ownship_state)
6258 pos_image = m_pos_image_user_yellow->Copy();
6259 else if (SHIP_NORMAL != m_ownship_state)
6260 pos_image = m_pos_image_user_grey->Copy();
6261 }
6262
6263 img_height = pos_image.GetHeight();
6264
6265 if (g_n_ownship_beam_meters > 0.0 && g_n_ownship_length_meters > 0.0 &&
6266 g_OwnShipIconType > 0) // use large ship
6267 {
6268 int ownShipWidth = 22; // Default values from s_ownship_icon
6269 int ownShipLength = 84;
6270 if (g_OwnShipIconType == 1) {
6271 ownShipWidth = pos_image.GetWidth();
6272 ownShipLength = pos_image.GetHeight();
6273 }
6274
6275 float scale_factor_x, scale_factor_y;
6276 ComputeShipScaleFactor(icon_hdt, ownShipWidth, ownShipLength,
6277 lShipMidPoint, GPSOffsetPixels, lGPSPoint,
6278 scale_factor_x, scale_factor_y);
6279
6280 if (g_OwnShipIconType == 1) { // Scaled bitmap
6281 pos_image.Rescale(ownShipWidth * scale_factor_x,
6282 ownShipLength * scale_factor_y,
6283 wxIMAGE_QUALITY_HIGH);
6284 wxPoint rot_ctr(pos_image.GetWidth() / 2, pos_image.GetHeight() / 2);
6285 wxImage rot_image =
6286 pos_image.Rotate(-(icon_rad - (PI / 2.)), rot_ctr, true);
6287
6288 // Simple sharpening algorithm.....
6289 for (int ip = 0; ip < rot_image.GetWidth(); ip++)
6290 for (int jp = 0; jp < rot_image.GetHeight(); jp++)
6291 if (rot_image.GetAlpha(ip, jp) > 64)
6292 rot_image.SetAlpha(ip, jp, 255);
6293
6294 wxBitmap os_bm(rot_image);
6295
6296 int w = os_bm.GetWidth();
6297 int h = os_bm.GetHeight();
6298 img_height = h;
6299
6300 dc.DrawBitmap(os_bm, lShipMidPoint.m_x - w / 2,
6301 lShipMidPoint.m_y - h / 2, true);
6302
6303 // Maintain dirty box,, missing in __WXMSW__ library
6304 dc.CalcBoundingBox(lShipMidPoint.m_x - w / 2,
6305 lShipMidPoint.m_y - h / 2);
6306 dc.CalcBoundingBox(lShipMidPoint.m_x - w / 2 + w,
6307 lShipMidPoint.m_y - h / 2 + h);
6308 }
6309
6310 else if (g_OwnShipIconType == 2) { // Scaled Vector
6311 wxPoint ownship_icon[10];
6312
6313 for (int i = 0; i < 10; i++) {
6314 int j = i * 2;
6315 float pxa = (float)(s_ownship_icon[j]);
6316 float pya = (float)(s_ownship_icon[j + 1]);
6317 pya *= scale_factor_y;
6318 pxa *= scale_factor_x;
6319
6320 float px = (pxa * sinf(icon_rad)) + (pya * cosf(icon_rad));
6321 float py = (pya * sinf(icon_rad)) - (pxa * cosf(icon_rad));
6322
6323 ownship_icon[i].x = (int)(px) + lShipMidPoint.m_x;
6324 ownship_icon[i].y = (int)(py) + lShipMidPoint.m_y;
6325 }
6326
6327 wxPen ppPen1(GetGlobalColor(_T ( "UBLCK" )), 1, wxPENSTYLE_SOLID);
6328 dc.SetPen(ppPen1);
6329 dc.SetBrush(wxBrush(ShipColor()));
6330
6331 dc.StrokePolygon(6, &ownship_icon[0], 0, 0);
6332
6333 // draw reference point (midships) cross
6334 dc.StrokeLine(ownship_icon[6].x, ownship_icon[6].y, ownship_icon[7].x,
6335 ownship_icon[7].y);
6336 dc.StrokeLine(ownship_icon[8].x, ownship_icon[8].y, ownship_icon[9].x,
6337 ownship_icon[9].y);
6338 }
6339
6340 img_height = ownShipLength * scale_factor_y;
6341
6342 // Reference point, where the GPS antenna is
6343 int circle_rad = 3;
6344 if (m_pos_image_user) circle_rad = 1;
6345
6346 dc.SetPen(wxPen(GetGlobalColor(_T ( "UBLCK" )), 1));
6347 dc.SetBrush(wxBrush(GetGlobalColor(_T ( "UIBCK" ))));
6348 dc.StrokeCircle(lGPSPoint.m_x, lGPSPoint.m_y, circle_rad);
6349 } else { // Fixed bitmap icon.
6350 /* non opengl, or suboptimal opengl via ocpndc: */
6351 wxPoint rot_ctr(pos_image.GetWidth() / 2, pos_image.GetHeight() / 2);
6352 wxImage rot_image =
6353 pos_image.Rotate(-(icon_rad - (PI / 2.)), rot_ctr, true);
6354
6355 // Simple sharpening algorithm.....
6356 for (int ip = 0; ip < rot_image.GetWidth(); ip++)
6357 for (int jp = 0; jp < rot_image.GetHeight(); jp++)
6358 if (rot_image.GetAlpha(ip, jp) > 64)
6359 rot_image.SetAlpha(ip, jp, 255);
6360
6361 wxBitmap os_bm(rot_image);
6362
6363 if (g_ShipScaleFactorExp > 1) {
6364 wxImage scaled_image = os_bm.ConvertToImage();
6365 double factor = (log(g_ShipScaleFactorExp) + 1.0) *
6366 1.0; // soften the scale factor a bit
6367 os_bm = wxBitmap(scaled_image.Scale(scaled_image.GetWidth() * factor,
6368 scaled_image.GetHeight() * factor,
6369 wxIMAGE_QUALITY_HIGH));
6370 }
6371 int w = os_bm.GetWidth();
6372 int h = os_bm.GetHeight();
6373 img_height = h;
6374
6375 dc.DrawBitmap(os_bm, lShipMidPoint.m_x - w / 2,
6376 lShipMidPoint.m_y - h / 2, true);
6377
6378 // Reference point, where the GPS antenna is
6379 int circle_rad = 3;
6380 if (m_pos_image_user) circle_rad = 1;
6381
6382 dc.SetPen(wxPen(GetGlobalColor(_T ( "UBLCK" )), 1));
6383 dc.SetBrush(wxBrush(GetGlobalColor(_T ( "UIBCK" ))));
6384 dc.StrokeCircle(lShipMidPoint.m_x, lShipMidPoint.m_y, circle_rad);
6385
6386 // Maintain dirty box,, missing in __WXMSW__ library
6387 dc.CalcBoundingBox(lShipMidPoint.m_x - w / 2,
6388 lShipMidPoint.m_y - h / 2);
6389 dc.CalcBoundingBox(lShipMidPoint.m_x - w / 2 + w,
6390 lShipMidPoint.m_y - h / 2 + h);
6391 }
6392 } // ownship draw
6393 }
6394
6395 ShipIndicatorsDraw(dc, img_height, GPSOffsetPixels, lGPSPoint);
6396}
6397
6398/* @ChartCanvas::CalcGridSpacing
6399 **
6400 ** Calculate the major and minor spacing between the lat/lon grid
6401 **
6402 ** @param [r] WindowDegrees [float] displayed number of lat or lan in the
6403 *window
6404 ** @param [w] MajorSpacing [float &] Major distance between grid lines
6405 ** @param [w] MinorSpacing [float &] Minor distance between grid lines
6406 ** @return [void]
6407 */
6408void CalcGridSpacing(float view_scale_ppm, float &MajorSpacing,
6409 float &MinorSpacing) {
6410 // table for calculating the distance between the grids
6411 // [0] view_scale ppm
6412 // [1] spacing between major grid lines in degrees
6413 // [2] spacing between minor grid lines in degrees
6414 const float lltab[][3] = {{0.0f, 90.0f, 30.0f},
6415 {.000001f, 45.0f, 15.0f},
6416 {.0002f, 30.0f, 10.0f},
6417 {.0003f, 10.0f, 2.0f},
6418 {.0008f, 5.0f, 1.0f},
6419 {.001f, 2.0f, 30.0f / 60.0f},
6420 {.003f, 1.0f, 20.0f / 60.0f},
6421 {.006f, 0.5f, 10.0f / 60.0f},
6422 {.03f, 15.0f / 60.0f, 5.0f / 60.0f},
6423 {.01f, 10.0f / 60.0f, 2.0f / 60.0f},
6424 {.06f, 5.0f / 60.0f, 1.0f / 60.0f},
6425 {.1f, 2.0f / 60.0f, 1.0f / 60.0f},
6426 {.4f, 1.0f / 60.0f, 0.5f / 60.0f},
6427 {.6f, 0.5f / 60.0f, 0.1f / 60.0f},
6428 {1.0f, 0.2f / 60.0f, 0.1f / 60.0f},
6429 {1e10f, 0.1f / 60.0f, 0.05f / 60.0f}};
6430
6431 unsigned int tabi;
6432 for (tabi = 0; tabi < ((sizeof lltab) / (sizeof *lltab)) - 1; tabi++)
6433 if (view_scale_ppm < lltab[tabi][0]) break;
6434 MajorSpacing = lltab[tabi][1]; // major latitude distance
6435 MinorSpacing = lltab[tabi][2]; // minor latitude distance
6436 return;
6437}
6438/* @ChartCanvas::CalcGridText *************************************
6439 **
6440 ** Calculates text to display at the major grid lines
6441 **
6442 ** @param [r] latlon [float] latitude or longitude of grid line
6443 ** @param [r] spacing [float] distance between two major grid lines
6444 ** @param [r] bPostfix [bool] true for latitudes, false for longitudes
6445 **
6446 ** @return
6447 */
6448
6449wxString CalcGridText(float latlon, float spacing, bool bPostfix) {
6450 int deg = (int)fabs(latlon); // degrees
6451 float min = fabs((fabs(latlon) - deg) * 60.0); // Minutes
6452 char postfix;
6453
6454 // calculate postfix letter (NSEW)
6455 if (latlon > 0.0) {
6456 if (bPostfix) {
6457 postfix = 'N';
6458 } else {
6459 postfix = 'E';
6460 }
6461 } else if (latlon < 0.0) {
6462 if (bPostfix) {
6463 postfix = 'S';
6464 } else {
6465 postfix = 'W';
6466 }
6467 } else {
6468 postfix = ' '; // no postfix for equator and greenwich
6469 }
6470 // calculate text, display minutes only if spacing is smaller than one degree
6471
6472 wxString ret;
6473 if (spacing >= 1.0) {
6474 ret.Printf(_T("%3d%c %c"), deg, 0x00b0, postfix);
6475 } else if (spacing >= (1.0 / 60.0)) {
6476 ret.Printf(_T("%3d%c%02.0f %c"), deg, 0x00b0, min, postfix);
6477 } else {
6478 ret.Printf(_T("%3d%c%02.2f %c"), deg, 0x00b0, min, postfix);
6479 }
6480
6481 return ret;
6482}
6483
6484/* @ChartCanvas::GridDraw *****************************************
6485 **
6486 ** Draws major and minor Lat/Lon Grid on the chart
6487 ** - distance between Grid-lm ines are calculated automatic
6488 ** - major grid lines will be across the whole chart window
6489 ** - minor grid lines will be 10 pixel at each edge of the chart window.
6490 **
6491 ** @param [w] dc [wxDC&] the wx drawing context
6492 **
6493 ** @return [void]
6494 ************************************************************************/
6495void ChartCanvas::GridDraw(ocpnDC &dc) {
6496 if (!(m_bDisplayGrid && (fabs(GetVP().rotation) < 1e-5))) return;
6497
6498 double nlat, elon, slat, wlon;
6499 float lat, lon;
6500 float dlon;
6501 float gridlatMajor, gridlatMinor, gridlonMajor, gridlonMinor;
6502 wxCoord w, h;
6503 wxPen GridPen(GetGlobalColor(_T ( "SNDG1" )), 1, wxPENSTYLE_SOLID);
6504 dc.SetPen(GridPen);
6505 dc.SetFont(*m_pgridFont);
6506 dc.SetTextForeground(GetGlobalColor(_T ( "SNDG1" )));
6507
6508 w = m_canvas_width;
6509 h = m_canvas_height;
6510
6511 GetCanvasPixPoint(0, 0, nlat,
6512 wlon); // get lat/lon of upper left point of the window
6513 GetCanvasPixPoint(w, h, slat,
6514 elon); // get lat/lon of lower right point of the window
6515 dlon =
6516 elon -
6517 wlon; // calculate how many degrees of longitude are shown in the window
6518 if (dlon < 0.0) // concider datum border at 180 degrees longitude
6519 {
6520 dlon = dlon + 360.0;
6521 }
6522 // calculate distance between latitude grid lines
6523 CalcGridSpacing(GetVP().view_scale_ppm, gridlatMajor, gridlatMinor);
6524
6525 // calculate position of first major latitude grid line
6526 lat = ceil(slat / gridlatMajor) * gridlatMajor;
6527
6528 // Draw Major latitude grid lines and text
6529 while (lat < nlat) {
6530 wxPoint r;
6531 wxString st =
6532 CalcGridText(lat, gridlatMajor, true); // get text for grid line
6533 GetCanvasPointPix(lat, (elon + wlon) / 2, &r);
6534 dc.DrawLine(0, r.y, w, r.y, false); // draw grid line
6535 dc.DrawText(st, 0, r.y); // draw text
6536 lat = lat + gridlatMajor;
6537
6538 if (fabs(lat - wxRound(lat)) < 1e-5) lat = wxRound(lat);
6539 }
6540
6541 // calculate position of first minor latitude grid line
6542 lat = ceil(slat / gridlatMinor) * gridlatMinor;
6543
6544 // Draw minor latitude grid lines
6545 while (lat < nlat) {
6546 wxPoint r;
6547 GetCanvasPointPix(lat, (elon + wlon) / 2, &r);
6548 dc.DrawLine(0, r.y, 10, r.y, false);
6549 dc.DrawLine(w - 10, r.y, w, r.y, false);
6550 lat = lat + gridlatMinor;
6551 }
6552
6553 // calculate distance between grid lines
6554 CalcGridSpacing(GetVP().view_scale_ppm, gridlonMajor, gridlonMinor);
6555
6556 // calculate position of first major latitude grid line
6557 lon = ceil(wlon / gridlonMajor) * gridlonMajor;
6558
6559 // draw major longitude grid lines
6560 for (int i = 0, itermax = (int)(dlon / gridlonMajor); i <= itermax; i++) {
6561 wxPoint r;
6562 wxString st = CalcGridText(lon, gridlonMajor, false);
6563 GetCanvasPointPix((nlat + slat) / 2, lon, &r);
6564 dc.DrawLine(r.x, 0, r.x, h, false);
6565 dc.DrawText(st, r.x, 0);
6566 lon = lon + gridlonMajor;
6567 if (lon > 180.0) {
6568 lon = lon - 360.0;
6569 }
6570
6571 if (fabs(lon - wxRound(lon)) < 1e-5) lon = wxRound(lon);
6572 }
6573
6574 // calculate position of first minor longitude grid line
6575 lon = ceil(wlon / gridlonMinor) * gridlonMinor;
6576 // draw minor longitude grid lines
6577 for (int i = 0, itermax = (int)(dlon / gridlonMinor); i <= itermax; i++) {
6578 wxPoint r;
6579 GetCanvasPointPix((nlat + slat) / 2, lon, &r);
6580 dc.DrawLine(r.x, 0, r.x, 10, false);
6581 dc.DrawLine(r.x, h - 10, r.x, h, false);
6582 lon = lon + gridlonMinor;
6583 if (lon > 180.0) {
6584 lon = lon - 360.0;
6585 }
6586 }
6587}
6588
6589void ChartCanvas::ScaleBarDraw(ocpnDC &dc) {
6590 if (0 ) {
6591 double blat, blon, tlat, tlon;
6592 wxPoint r;
6593
6594 int x_origin = m_bDisplayGrid ? 60 : 20;
6595 int y_origin = m_canvas_height - 50;
6596
6597 float dist;
6598 int count;
6599 wxPen pen1, pen2;
6600
6601 if (GetVP().chart_scale > 80000) // Draw 10 mile scale as SCALEB11
6602 {
6603 dist = 10.0;
6604 count = 5;
6605 pen1 = wxPen(GetGlobalColor(_T ( "SNDG2" )), 3, wxPENSTYLE_SOLID);
6606 pen2 = wxPen(GetGlobalColor(_T ( "SNDG1" )), 3, wxPENSTYLE_SOLID);
6607 } else // Draw 1 mile scale as SCALEB10
6608 {
6609 dist = 1.0;
6610 count = 10;
6611 pen1 = wxPen(GetGlobalColor(_T ( "SCLBR" )), 3, wxPENSTYLE_SOLID);
6612 pen2 = wxPen(GetGlobalColor(_T ( "CHGRD" )), 3, wxPENSTYLE_SOLID);
6613 }
6614
6615 GetCanvasPixPoint(x_origin, y_origin, blat, blon);
6616 double rotation = -VPoint.rotation;
6617
6618 ll_gc_ll(blat, blon, rotation * 180 / PI, dist, &tlat, &tlon);
6619 GetCanvasPointPix(tlat, tlon, &r);
6620 int l1 = (y_origin - r.y) / count;
6621
6622 for (int i = 0; i < count; i++) {
6623 int y = l1 * i;
6624 if (i & 1)
6625 dc.SetPen(pen1);
6626 else
6627 dc.SetPen(pen2);
6628
6629 dc.DrawLine(x_origin, y_origin - y, x_origin, y_origin - (y + l1));
6630 }
6631 } else {
6632 double blat, blon, tlat, tlon;
6633
6634 int x_origin = 5.0 * GetPixPerMM();
6635 int chartbar_height = GetChartbarHeight();
6636 // ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
6637 // if (style->chartStatusWindowTransparent)
6638 // chartbar_height = 0;
6639 int y_origin = m_canvas_height - chartbar_height - 5;
6640#ifdef __WXOSX__
6641 if (!g_bopengl)
6642 y_origin =
6643 m_canvas_height / GetContentScaleFactor() - chartbar_height - 5;
6644#endif
6645
6646 GetCanvasPixPoint(x_origin, y_origin, blat, blon);
6647 GetCanvasPixPoint(x_origin + m_canvas_width, y_origin, tlat, tlon);
6648
6649 double d;
6650 ll_gc_ll_reverse(blat, blon, tlat, tlon, 0, &d);
6651 d /= 2;
6652
6653 int unit = g_iDistanceFormat;
6654 if (d < .5 &&
6655 (unit == DISTANCE_KM || unit == DISTANCE_MI || unit == DISTANCE_NMI))
6656 unit = (unit == DISTANCE_MI) ? DISTANCE_FT : DISTANCE_M;
6657
6658 // nice number
6659 float dist = toUsrDistance(d, unit), logdist = log(dist) / log(10.F);
6660 float places = floor(logdist), rem = logdist - places;
6661 dist = pow(10, places);
6662
6663 if (rem < .2)
6664 dist /= 5;
6665 else if (rem < .5)
6666 dist /= 2;
6667
6668 wxString s = wxString::Format(_T("%g "), dist) + getUsrDistanceUnit(unit);
6669 wxPen pen1 = wxPen(GetGlobalColor(_T ( "UBLCK" )), 3, wxPENSTYLE_SOLID);
6670 double rotation = -VPoint.rotation;
6671
6672 ll_gc_ll(blat, blon, rotation * 180 / PI + 90, fromUsrDistance(dist, unit),
6673 &tlat, &tlon);
6674 wxPoint r;
6675 GetCanvasPointPix(tlat, tlon, &r);
6676 int l1 = r.x - x_origin;
6677
6678 m_scaleBarRect = wxRect(x_origin, y_origin - 12, l1,
6679 12); // Store this for later reference
6680
6681 dc.SetPen(pen1);
6682
6683 dc.DrawLine(x_origin, y_origin, x_origin, y_origin - 12);
6684 dc.DrawLine(x_origin, y_origin, x_origin + l1, y_origin);
6685 dc.DrawLine(x_origin + l1, y_origin, x_origin + l1, y_origin - 12);
6686
6687 dc.SetFont(*m_pgridFont);
6688 dc.SetTextForeground(GetGlobalColor(_T ( "UBLCK" )));
6689 int w, h;
6690 dc.GetTextExtent(s, &w, &h);
6691 dc.DrawText(s, x_origin + l1 / 2 - w / 2, y_origin - h - 1);
6692 }
6693}
6694
6695void ChartCanvas::JaggyCircle(ocpnDC &dc, wxPen pen, int x, int y, int radius) {
6696 // Constants?
6697 double da_min = 2.;
6698 double da_max = 6.;
6699 double ra_min = 0.;
6700 double ra_max = 40.;
6701
6702 wxPen pen_save = dc.GetPen();
6703
6704 wxDateTime now = wxDateTime::Now();
6705
6706 dc.SetPen(pen);
6707
6708 int x0, y0, x1, y1;
6709
6710 x0 = x1 = x + radius; // Start point
6711 y0 = y1 = y;
6712 double angle = 0.;
6713 int i = 0;
6714
6715 while (angle < 360.) {
6716 double da = da_min + (((double)rand() / RAND_MAX) * (da_max - da_min));
6717 angle += da;
6718
6719 if (angle > 360.) angle = 360.;
6720
6721 double ra = ra_min + (((double)rand() / RAND_MAX) * (ra_max - ra_min));
6722
6723 double r;
6724 if (i & 1)
6725 r = radius + ra;
6726 else
6727 r = radius - ra;
6728
6729 x1 = (int)(x + cos(angle * PI / 180.) * r);
6730 y1 = (int)(y + sin(angle * PI / 180.) * r);
6731
6732 dc.DrawLine(x0, y0, x1, y1);
6733
6734 x0 = x1;
6735 y0 = y1;
6736
6737 i++;
6738 }
6739
6740 dc.DrawLine(x + radius, y, x1, y1); // closure
6741
6742 dc.SetPen(pen_save);
6743}
6744
6745static bool bAnchorSoundPlaying = false;
6746
6747static void onAnchorSoundFinished(void *ptr) {
6748 g_anchorwatch_sound->UnLoad();
6749 bAnchorSoundPlaying = false;
6750}
6751
6752void ChartCanvas::AlertDraw(ocpnDC &dc) {
6753 // Visual and audio alert for anchorwatch goes here
6754 bool play_sound = false;
6755 if (pAnchorWatchPoint1 && AnchorAlertOn1) {
6756 if (AnchorAlertOn1) {
6757 wxPoint TargetPoint;
6758 GetCanvasPointPix(pAnchorWatchPoint1->m_lat, pAnchorWatchPoint1->m_lon,
6759 &TargetPoint);
6760 JaggyCircle(dc, wxPen(GetGlobalColor(_T("URED")), 2), TargetPoint.x,
6761 TargetPoint.y, 100);
6762 play_sound = true;
6763 }
6764 } else
6765 AnchorAlertOn1 = false;
6766
6767 if (pAnchorWatchPoint2 && AnchorAlertOn2) {
6768 if (AnchorAlertOn2) {
6769 wxPoint TargetPoint;
6770 GetCanvasPointPix(pAnchorWatchPoint2->m_lat, pAnchorWatchPoint2->m_lon,
6771 &TargetPoint);
6772 JaggyCircle(dc, wxPen(GetGlobalColor(_T("URED")), 2), TargetPoint.x,
6773 TargetPoint.y, 100);
6774 play_sound = true;
6775 }
6776 } else
6777 AnchorAlertOn2 = false;
6778
6779 if (play_sound) {
6780 if (!bAnchorSoundPlaying) {
6781 auto cmd_sound = dynamic_cast<SystemCmdSound *>(g_anchorwatch_sound);
6782 if (cmd_sound) cmd_sound->SetCmd(g_CmdSoundString.mb_str(wxConvUTF8));
6783 g_anchorwatch_sound->Load(g_anchorwatch_sound_file);
6784 if (g_anchorwatch_sound->IsOk()) {
6785 bAnchorSoundPlaying = true;
6786 g_anchorwatch_sound->SetFinishedCallback(onAnchorSoundFinished, NULL);
6787 g_anchorwatch_sound->Play();
6788 }
6789 }
6790 }
6791}
6792
6793void ChartCanvas::UpdateShips() {
6794 // Get the rectangle in the current dc which bounds the "ownship" symbol
6795
6796 wxClientDC dc(this);
6797 if (!dc.IsOk()) return;
6798
6799 wxBitmap test_bitmap(dc.GetSize().x, dc.GetSize().y);
6800 if (!test_bitmap.IsOk()) return;
6801
6802 wxMemoryDC temp_dc(test_bitmap);
6803
6804 temp_dc.ResetBoundingBox();
6805 temp_dc.DestroyClippingRegion();
6806 temp_dc.SetClippingRegion(0, 0, dc.GetSize().x, dc.GetSize().y);
6807
6808 // Draw the ownship on the temp_dc
6809 ocpnDC ocpndc = ocpnDC(temp_dc);
6810 ShipDraw(ocpndc);
6811
6812 if (g_pActiveTrack && g_pActiveTrack->IsRunning()) {
6813 TrackPoint *p = g_pActiveTrack->GetLastPoint();
6814 if (p) {
6815 wxPoint px;
6816 GetCanvasPointPix(p->m_lat, p->m_lon, &px);
6817 ocpndc.CalcBoundingBox(px.x, px.y);
6818 }
6819 }
6820
6821 ship_draw_rect =
6822 wxRect(temp_dc.MinX(), temp_dc.MinY(), temp_dc.MaxX() - temp_dc.MinX(),
6823 temp_dc.MaxY() - temp_dc.MinY());
6824
6825 wxRect own_ship_update_rect = ship_draw_rect;
6826
6827 if (!own_ship_update_rect.IsEmpty()) {
6828 // The required invalidate rectangle is the union of the last drawn
6829 // rectangle and this drawn rectangle
6830 own_ship_update_rect.Union(ship_draw_last_rect);
6831 own_ship_update_rect.Inflate(2);
6832 }
6833
6834 if (!own_ship_update_rect.IsEmpty()) RefreshRect(own_ship_update_rect, false);
6835
6836 ship_draw_last_rect = ship_draw_rect;
6837
6838 temp_dc.SelectObject(wxNullBitmap);
6839}
6840
6841void ChartCanvas::UpdateAlerts() {
6842 // Get the rectangle in the current dc which bounds the detected Alert
6843 // targets
6844
6845 // Use this dc
6846 wxClientDC dc(this);
6847
6848 // Get dc boundary
6849 int sx, sy;
6850 dc.GetSize(&sx, &sy);
6851
6852 // Need a bitmap
6853 wxBitmap test_bitmap(sx, sy, -1);
6854
6855 // Create a memory DC
6856 wxMemoryDC temp_dc;
6857 temp_dc.SelectObject(test_bitmap);
6858
6859 temp_dc.ResetBoundingBox();
6860 temp_dc.DestroyClippingRegion();
6861 temp_dc.SetClippingRegion(wxRect(0, 0, sx, sy));
6862
6863 // Draw the Alert Targets on the temp_dc
6864 ocpnDC ocpndc = ocpnDC(temp_dc);
6865 AlertDraw(ocpndc);
6866
6867 // Retrieve the drawing extents
6868 wxRect alert_rect(temp_dc.MinX(), temp_dc.MinY(),
6869 temp_dc.MaxX() - temp_dc.MinX(),
6870 temp_dc.MaxY() - temp_dc.MinY());
6871
6872 if (!alert_rect.IsEmpty())
6873 alert_rect.Inflate(2); // clear all drawing artifacts
6874
6875 if (!alert_rect.IsEmpty() || !alert_draw_rect.IsEmpty()) {
6876 // The required invalidate rectangle is the union of the last drawn
6877 // rectangle and this drawn rectangle
6878 wxRect alert_update_rect = alert_draw_rect;
6879 alert_update_rect.Union(alert_rect);
6880
6881 // Invalidate the rectangular region
6882 RefreshRect(alert_update_rect, false);
6883 }
6884
6885 // Save this rectangle for next time
6886 alert_draw_rect = alert_rect;
6887
6888 temp_dc.SelectObject(wxNullBitmap); // clean up
6889}
6890
6891void ChartCanvas::UpdateAIS() {
6892 if (!g_pAIS) return;
6893
6894 // Get the rectangle in the current dc which bounds the detected AIS targets
6895
6896 // Use this dc
6897 wxClientDC dc(this);
6898
6899 // Get dc boundary
6900 int sx, sy;
6901 dc.GetSize(&sx, &sy);
6902
6903 wxRect ais_rect;
6904
6905 // How many targets are there?
6906
6907 // If more than "some number", it will be cheaper to refresh the entire
6908 // screen than to build update rectangles for each target.
6909 if (g_pAIS->GetTargetList().size() > 10) {
6910 ais_rect = wxRect(0, 0, sx, sy); // full screen
6911 } else {
6912 // Need a bitmap
6913 wxBitmap test_bitmap(sx, sy, -1);
6914
6915 // Create a memory DC
6916 wxMemoryDC temp_dc;
6917 temp_dc.SelectObject(test_bitmap);
6918
6919 temp_dc.ResetBoundingBox();
6920 temp_dc.DestroyClippingRegion();
6921 temp_dc.SetClippingRegion(wxRect(0, 0, sx, sy));
6922
6923 // Draw the AIS Targets on the temp_dc
6924 ocpnDC ocpndc = ocpnDC(temp_dc);
6925 AISDraw(ocpndc, GetVP(), this);
6926 AISDrawAreaNotices(ocpndc, GetVP(), this);
6927
6928 // Retrieve the drawing extents
6929 ais_rect =
6930 wxRect(temp_dc.MinX(), temp_dc.MinY(), temp_dc.MaxX() - temp_dc.MinX(),
6931 temp_dc.MaxY() - temp_dc.MinY());
6932
6933 if (!ais_rect.IsEmpty())
6934 ais_rect.Inflate(2); // clear all drawing artifacts
6935
6936 temp_dc.SelectObject(wxNullBitmap); // clean up
6937 }
6938
6939 if (!ais_rect.IsEmpty() || !ais_draw_rect.IsEmpty()) {
6940 // The required invalidate rectangle is the union of the last drawn
6941 // rectangle and this drawn rectangle
6942 wxRect ais_update_rect = ais_draw_rect;
6943 ais_update_rect.Union(ais_rect);
6944
6945 // Invalidate the rectangular region
6946 RefreshRect(ais_update_rect, false);
6947 }
6948
6949 // Save this rectangle for next time
6950 ais_draw_rect = ais_rect;
6951}
6952
6953void ChartCanvas::ToggleCPAWarn() {
6954 if (!g_AisFirstTimeUse) g_bCPAWarn = !g_bCPAWarn;
6955 wxString mess;
6956 if (g_bCPAWarn) {
6957 g_bTCPA_Max = true;
6958 mess = _("ON");
6959 } else {
6960 g_bTCPA_Max = false;
6961 mess = _("OFF");
6962 }
6963 // Print to status bar if available.
6964 if (STAT_FIELD_SCALE >= 4 && parent_frame->GetStatusBar()) {
6965 parent_frame->SetStatusText(_("CPA alarm ") + mess, STAT_FIELD_SCALE);
6966 } else {
6967 if (!g_AisFirstTimeUse) {
6968 OCPNMessageBox(this,
6969 _("CPA Alarm is switched") + _T(" ") + mess.MakeLower(),
6970 _("CPA") + _T(" ") + mess, 4, 4);
6971 }
6972 }
6973}
6974
6975void ChartCanvas::OnActivate(wxActivateEvent &event) { ReloadVP(); }
6976
6977void ChartCanvas::OnSize(wxSizeEvent &event) {
6978 if ((event.m_size.GetWidth() < 1) || (event.m_size.GetHeight() < 1)) return;
6979 // GetClientSize returns the size of the canvas area in logical pixels.
6980 GetClientSize(&m_canvas_width, &m_canvas_height);
6981
6982#ifdef __WXOSX__
6983 // Support scaled HDPI displays.
6984 m_displayScale = GetContentScaleFactor();
6985#endif
6986
6987 // Convert to physical pixels.
6988 m_canvas_width *= m_displayScale;
6989 m_canvas_height *= m_displayScale;
6990
6991 // Resize the current viewport
6992 VPoint.pix_width = m_canvas_width;
6993 VPoint.pix_height = m_canvas_height;
6994 VPoint.SetPixelScale(m_displayScale);
6995
6996 // Get some canvas metrics
6997
6998 // Rescale to current value, in order to rebuild VPoint data
6999 // structures for new canvas size
7001
7002 m_absolute_min_scale_ppm =
7003 m_canvas_width /
7004 (1.2 * WGS84_semimajor_axis_meters * PI); // something like 180 degrees
7005
7006 // Inform the parent Frame that I am being resized...
7007 gFrame->ProcessCanvasResize();
7008
7009 // if MUIBar is active, size the bar
7010 // if(g_useMUI && !m_muiBar){ // rebuild if
7011 // necessary
7012 // m_muiBar = new MUIBar(this, wxHORIZONTAL);
7013 // m_muiBarHOSize = m_muiBar->GetSize();
7014 // }
7015
7016 if (m_muiBar) {
7017 SetMUIBarPosition();
7018 UpdateFollowButtonState();
7019 m_muiBar->SetCanvasENCAvailable(m_bENCGroup);
7020 }
7021
7022 // Set up the scroll margins
7023 xr_margin = m_canvas_width * 95 / 100;
7024 xl_margin = m_canvas_width * 5 / 100;
7025 yt_margin = m_canvas_height * 5 / 100;
7026 yb_margin = m_canvas_height * 95 / 100;
7027
7028 if (m_pQuilt)
7029 m_pQuilt->SetQuiltParameters(m_canvas_scale_factor, m_canvas_width);
7030
7031 // Resize the scratch BM
7032 delete pscratch_bm;
7033 pscratch_bm = new wxBitmap(VPoint.pix_width, VPoint.pix_height, -1);
7034 m_brepaint_piano = true;
7035
7036 // Resize the Route Calculation BM
7037 m_dc_route.SelectObject(wxNullBitmap);
7038 delete proute_bm;
7039 proute_bm = new wxBitmap(VPoint.pix_width, VPoint.pix_height, -1);
7040 m_dc_route.SelectObject(*proute_bm);
7041
7042 // Resize the saved Bitmap
7043 m_cached_chart_bm.Create(VPoint.pix_width, VPoint.pix_height, -1);
7044
7045 // Resize the working Bitmap
7046 m_working_bm.Create(VPoint.pix_width, VPoint.pix_height, -1);
7047
7048 // Rescale again, to capture all the changes for new canvas size
7050
7051#ifdef ocpnUSE_GL
7052 if (/*g_bopengl &&*/ m_glcc) {
7053 // FIXME (dave) This can go away?
7054 m_glcc->OnSize(event);
7055 }
7056#endif
7057
7058 FormatPianoKeys();
7059 // Invalidate the whole window
7060 ReloadVP();
7061}
7062
7063void ChartCanvas::ProcessNewGUIScale() {
7064 // m_muiBar->Hide();
7065 delete m_muiBar;
7066 m_muiBar = 0;
7067
7068 CreateMUIBar();
7069}
7070
7071void ChartCanvas::CreateMUIBar() {
7072 if (g_useMUI && !m_muiBar) { // rebuild if necessary
7073
7074 // We need to update the m_bENCGroup flag, at least for the initial creation
7075 // of a MUIBar
7076 if (ChartData) m_bENCGroup = ChartData->IsENCInGroup(m_groupIndex);
7077
7078 m_muiBar = new MUIBar(this, wxHORIZONTAL, g_toolbar_scalefactor);
7079 m_muiBar->SetColorScheme(m_cs);
7080 m_muiBarHOSize = m_muiBar->m_size;
7081 }
7082
7083 if (m_muiBar) {
7084 SetMUIBarPosition();
7085 UpdateFollowButtonState();
7086 m_muiBar->UpdateDynamicValues();
7087 m_muiBar->SetCanvasENCAvailable(m_bENCGroup);
7088 }
7089}
7090
7091void ChartCanvas::SetMUIBarPosition() {
7092 // if MUIBar is active, size the bar
7093 if (m_muiBar) {
7094 // We estimate the piano width based on the canvas width
7095 int pianoWidth = GetClientSize().x * 0.6f;
7096 // If the piano already exists, we can use its exact width
7097 // if(m_Piano)
7098 // pianoWidth = m_Piano->GetWidth();
7099
7100 if ((m_muiBar->GetOrientation() == wxHORIZONTAL)) {
7101 if (m_muiBarHOSize.x > (GetClientSize().x - pianoWidth)) {
7102 delete m_muiBar;
7103 m_muiBar = new MUIBar(this, wxVERTICAL, g_toolbar_scalefactor);
7104 m_muiBar->SetColorScheme(m_cs);
7105 }
7106 }
7107
7108 if ((m_muiBar->GetOrientation() == wxVERTICAL)) {
7109 if (m_muiBarHOSize.x < (GetClientSize().x - pianoWidth)) {
7110 delete m_muiBar;
7111 m_muiBar = new MUIBar(this, wxHORIZONTAL, g_toolbar_scalefactor);
7112 m_muiBar->SetColorScheme(m_cs);
7113 }
7114 }
7115
7116 m_muiBar->SetBestPosition();
7117 }
7118}
7119
7120void ChartCanvas::DestroyMuiBar() {
7121 if (m_muiBar) {
7122 delete m_muiBar;
7123 m_muiBar = NULL;
7124 }
7125}
7126
7127void ChartCanvas::ShowCompositeInfoWindow(
7128 int x, int n_charts, int scale, const std::vector<int> &index_vector) {
7129 if (n_charts > 0) {
7130 if (NULL == m_pCIWin) {
7131 m_pCIWin = new ChInfoWin(this);
7132 m_pCIWin->Hide();
7133 }
7134
7135 if (!m_pCIWin->IsShown() || (m_pCIWin->chart_scale != scale)) {
7136 wxString s;
7137
7138 s = _("Composite of ");
7139
7140 wxString s1;
7141 s1.Printf("%d ", n_charts);
7142 if (n_charts > 1)
7143 s1 += _("charts");
7144 else
7145 s1 += _("chart");
7146 s += s1;
7147 s += '\n';
7148
7149 s1.Printf(_("Chart scale"));
7150 s1 += ": ";
7151 wxString s2;
7152 s2.Printf("1:%d\n", scale);
7153 s += s1;
7154 s += s2;
7155
7156 s1 = _("Zoom in for more information");
7157 s += s1;
7158 s += '\n';
7159
7160 int char_width = s1.Length();
7161 int char_height = 3;
7162
7163 if (g_bChartBarEx) {
7164 s += '\n';
7165 int j = 0;
7166 for (int i : index_vector) {
7167 const ChartTableEntry &cte = ChartData->GetChartTableEntry(i);
7168 wxString path = cte.GetFullSystemPath();
7169 s += path;
7170 s += '\n';
7171 char_height++;
7172 char_width = wxMax(char_width, path.Length());
7173 if (j++ >= 9) break;
7174 }
7175 if (j >= 9) {
7176 s += " .\n .\n .\n";
7177 char_height += 3;
7178 }
7179 s += '\n';
7180 char_height += 1;
7181
7182 char_width += 4; // Fluff
7183 }
7184
7185 m_pCIWin->SetString(s);
7186
7187 m_pCIWin->FitToChars(char_width, char_height);
7188
7189 wxPoint p;
7190 p.x = x / GetContentScaleFactor();
7191 if ((p.x + m_pCIWin->GetWinSize().x) >
7192 (m_canvas_width / GetContentScaleFactor()))
7193 p.x = ((m_canvas_width / GetContentScaleFactor()) -
7194 m_pCIWin->GetWinSize().x) /
7195 2; // centered
7196
7197 p.y = (m_canvas_height - m_Piano->GetHeight()) / GetContentScaleFactor() -
7198 4 - m_pCIWin->GetWinSize().y;
7199
7200 m_pCIWin->dbIndex = 0;
7201 m_pCIWin->chart_scale = 0;
7202 m_pCIWin->SetPosition(p);
7203 m_pCIWin->SetBitmap();
7204 m_pCIWin->Refresh();
7205 m_pCIWin->Show();
7206 }
7207 } else {
7208 HideChartInfoWindow();
7209 }
7210}
7211
7212void ChartCanvas::ShowChartInfoWindow(int x, int dbIndex) {
7213 if (dbIndex >= 0) {
7214 if (NULL == m_pCIWin) {
7215 m_pCIWin = new ChInfoWin(this);
7216 m_pCIWin->Hide();
7217 }
7218
7219 if (!m_pCIWin->IsShown() || (m_pCIWin->dbIndex != dbIndex)) {
7220 wxString s;
7221 ChartBase *pc = NULL;
7222
7223 // TOCTOU race but worst case will reload chart.
7224 // need to lock it or the background spooler may evict charts in
7225 // OpenChartFromDBAndLock
7226 if ((ChartData->IsChartInCache(dbIndex)) && ChartData->IsValid())
7227 pc = ChartData->OpenChartFromDBAndLock(
7228 dbIndex, FULL_INIT); // this must come from cache
7229
7230 int char_width, char_height;
7231 s = ChartData->GetFullChartInfo(pc, dbIndex, &char_width, &char_height);
7232 if (pc) ChartData->UnLockCacheChart(dbIndex);
7233
7234 m_pCIWin->SetString(s);
7235 m_pCIWin->FitToChars(char_width, char_height);
7236
7237 wxPoint p;
7238 p.x = x / GetContentScaleFactor();
7239 if ((p.x + m_pCIWin->GetWinSize().x) >
7240 (m_canvas_width / GetContentScaleFactor()))
7241 p.x = ((m_canvas_width / GetContentScaleFactor()) -
7242 m_pCIWin->GetWinSize().x) /
7243 2; // centered
7244
7245 p.y = (m_canvas_height - m_Piano->GetHeight()) / GetContentScaleFactor() -
7246 4 - m_pCIWin->GetWinSize().y;
7247
7248 m_pCIWin->dbIndex = dbIndex;
7249 m_pCIWin->SetPosition(p);
7250 m_pCIWin->SetBitmap();
7251 m_pCIWin->Refresh();
7252 m_pCIWin->Show();
7253 }
7254 } else {
7255 HideChartInfoWindow();
7256 }
7257}
7258
7259void ChartCanvas::HideChartInfoWindow(void) {
7260 if (m_pCIWin /*&& m_pCIWin->IsShown()*/) {
7261 m_pCIWin->Hide();
7262 m_pCIWin->Destroy();
7263 m_pCIWin = NULL;
7264
7265#ifdef __ANDROID__
7266 androidForceFullRepaint();
7267#endif
7268 }
7269}
7270
7271void ChartCanvas::PanTimerEvent(wxTimerEvent &event) {
7272 wxMouseEvent ev(wxEVT_MOTION);
7273 ev.m_x = mouse_x;
7274 ev.m_y = mouse_y;
7275 ev.m_leftDown = mouse_leftisdown;
7276
7277 wxEvtHandler *evthp = GetEventHandler();
7278
7279 ::wxPostEvent(evthp, ev);
7280}
7281
7282void ChartCanvas::MovementTimerEvent(wxTimerEvent &) {
7283 if ((m_panx_target_final - m_panx_target_now) ||
7284 (m_pany_target_final - m_pany_target_now)) {
7285 DoTimedMovementTarget();
7286 } else
7287 DoTimedMovement();
7288}
7289
7290void ChartCanvas::MovementStopTimerEvent(wxTimerEvent &) { StopMovement(); }
7291
7292bool ChartCanvas::CheckEdgePan(int x, int y, bool bdragging, int margin,
7293 int delta) {
7294 if (m_disable_edge_pan) return false;
7295
7296 bool bft = false;
7297 int pan_margin = m_canvas_width * margin / 100;
7298 int pan_timer_set = 200;
7299 double pan_delta = GetVP().pix_width * delta / 100;
7300 int pan_x = 0;
7301 int pan_y = 0;
7302
7303 if (x > m_canvas_width - pan_margin) {
7304 bft = true;
7305 pan_x = pan_delta;
7306 }
7307
7308 else if (x < pan_margin) {
7309 bft = true;
7310 pan_x = -pan_delta;
7311 }
7312
7313 if (y < pan_margin) {
7314 bft = true;
7315 pan_y = -pan_delta;
7316 }
7317
7318 else if (y > m_canvas_height - pan_margin) {
7319 bft = true;
7320 pan_y = pan_delta;
7321 }
7322
7323 // Of course, if dragging, and the mouse left button is not down, we must
7324 // stop the event injection
7325 if (bdragging) {
7326 if (!g_btouch) {
7327 wxMouseState state = ::wxGetMouseState();
7328#if wxCHECK_VERSION(3, 0, 0)
7329 if (!state.LeftIsDown())
7330#else
7331 if (!state.LeftDown())
7332#endif
7333 bft = false;
7334 }
7335 }
7336 if ((bft) && !pPanTimer->IsRunning()) {
7337 PanCanvas(pan_x, pan_y);
7338 pPanTimer->Start(pan_timer_set, wxTIMER_ONE_SHOT);
7339 return true;
7340 }
7341
7342 // This mouse event must not be due to pan timer event injector
7343 // Mouse is out of the pan zone, so prevent any orphan event injection
7344 if ((!bft) && pPanTimer->IsRunning()) {
7345 pPanTimer->Stop();
7346 }
7347
7348 return (false);
7349}
7350
7351// Look for waypoints at the current position.
7352// Used to determine what a mouse event should act on.
7353
7354void ChartCanvas::FindRoutePointsAtCursor(float selectRadius,
7355 bool setBeingEdited) {
7356 m_lastRoutePointEditTarget = m_pRoutePointEditTarget; // save a copy
7357 m_pRoutePointEditTarget = NULL;
7358 m_pFoundPoint = NULL;
7359
7360 SelectItem *pFind = NULL;
7361 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7362 SelectableItemList SelList = pSelect->FindSelectionList(
7363 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_ROUTEPOINT);
7364 wxSelectableItemListNode *node = SelList.GetFirst();
7365 while (node) {
7366 pFind = node->GetData();
7367
7368 RoutePoint *frp = (RoutePoint *)pFind->m_pData1;
7369
7370 // Get an array of all routes using this point
7371 m_pEditRouteArray = g_pRouteMan->GetRouteArrayContaining(frp);
7372
7373 // Use route array to determine actual visibility for the point
7374 bool brp_viz = false;
7375 if (m_pEditRouteArray) {
7376 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount(); ir++) {
7377 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
7378 if (pr->IsVisible()) {
7379 brp_viz = true;
7380 break;
7381 }
7382 }
7383 } else
7384 brp_viz = frp->IsVisible(); // isolated point
7385
7386 if (brp_viz) {
7387 // Use route array to rubberband all affected routes
7388 if (m_pEditRouteArray) // Editing Waypoint as part of route
7389 {
7390 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount(); ir++) {
7391 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
7392 pr->m_bIsBeingEdited = setBeingEdited;
7393 }
7394 m_bRouteEditing = setBeingEdited;
7395 } else // editing Mark
7396 {
7397 frp->m_bRPIsBeingEdited = setBeingEdited;
7398 m_bMarkEditing = setBeingEdited;
7399 }
7400
7401 m_pRoutePointEditTarget = frp;
7402 m_pFoundPoint = pFind;
7403 break; // out of the while(node)
7404 }
7405
7406 node = node->GetNext();
7407 } // while (node)
7408}
7409
7410void ChartCanvas::MouseTimedEvent(wxTimerEvent &event) {
7411 if (singleClickEventIsValid) MouseEvent(singleClickEvent);
7412 singleClickEventIsValid = false;
7413 m_DoubleClickTimer->Stop();
7414}
7415
7416bool leftIsDown;
7417
7418bool ChartCanvas::MouseEventOverlayWindows(wxMouseEvent &event) {
7419 if (!m_bChartDragging && !m_bDrawingRoute) {
7420 /*
7421 * The m_Compass->GetRect() coordinates are in physical pixels, whereas the
7422 * mouse event coordinates are in logical pixels.
7423 */
7424 wxRect logicalRect = m_Compass->GetLogicalRect();
7425 bool isInCompass = m_Compass && m_Compass->IsShown() &&
7426 logicalRect.Contains(event.GetPosition());
7427 if (isInCompass) {
7428 if (m_Compass->MouseEvent(event)) {
7429 cursor_region = CENTER;
7430 if (!g_btouch) SetCanvasCursor(event);
7431 return true;
7432 }
7433 }
7434
7435 if (MouseEventToolbar(event)) return true;
7436
7437 if (MouseEventChartBar(event)) return true;
7438
7439 if (MouseEventMUIBar(event)) return true;
7440
7441 if (MouseEventIENCBar(event)) return true;
7442 }
7443 return false;
7444}
7445
7446bool ChartCanvas::MouseEventChartBar(wxMouseEvent &event) {
7447 if (!g_bShowChartBar) return false;
7448
7449 if (!m_Piano->MouseEvent(event)) return false;
7450
7451 cursor_region = CENTER;
7452 if (!g_btouch) SetCanvasCursor(event);
7453 return true;
7454}
7455
7456bool ChartCanvas::MouseEventToolbar(wxMouseEvent &event) {
7457 if (!IsPrimaryCanvas()) return false;
7458
7459 if (g_MainToolbar) {
7460 if (!g_MainToolbar->MouseEvent(event))
7461 return false;
7462 else
7463 g_MainToolbar->RefreshToolbar();
7464 }
7465
7466 cursor_region = CENTER;
7467 if (!g_btouch) SetCanvasCursor(event);
7468 return true;
7469}
7470
7471bool ChartCanvas::MouseEventIENCBar(wxMouseEvent &event) {
7472 if (!IsPrimaryCanvas()) return false;
7473
7474 if (g_iENCToolbar) {
7475 if (!g_iENCToolbar->MouseEvent(event))
7476 return false;
7477 else {
7478 g_iENCToolbar->RefreshToolbar();
7479 return true;
7480 }
7481 }
7482 return false;
7483}
7484
7485bool ChartCanvas::MouseEventMUIBar(wxMouseEvent &event) {
7486 if (m_muiBar) {
7487 if (!m_muiBar->MouseEvent(event)) return false;
7488 }
7489
7490 cursor_region = CENTER;
7491 if (!g_btouch) SetCanvasCursor(event);
7492 if (m_muiBar)
7493 return true;
7494 else
7495 return false;
7496}
7497
7498bool ChartCanvas::MouseEventSetup(wxMouseEvent &event, bool b_handle_dclick) {
7499 int x, y;
7500
7501 bool bret = false;
7502
7503 event.GetPosition(&x, &y);
7504
7505 x *= m_displayScale;
7506 y *= m_displayScale;
7507
7508 m_MouseDragging = event.Dragging();
7509
7510 // Some systems produce null drag events, where the pointer position has not
7511 // changed from the previous value. Detect this case, and abort further
7512 // processing (FS#1748)
7513#ifdef __WXMSW__
7514 if (event.Dragging()) {
7515 if ((x == mouse_x) && (y == mouse_y)) return true;
7516 }
7517#endif
7518
7519 mouse_x = x;
7520 mouse_y = y;
7521 mouse_leftisdown = event.LeftDown();
7523
7524 // Establish the event region
7525 cursor_region = CENTER;
7526
7527 int chartbar_height = GetChartbarHeight();
7528
7529 if (m_Compass && m_Compass->IsShown() &&
7530 m_Compass->GetRect().Contains(event.GetPosition())) {
7531 cursor_region = CENTER;
7532 } else if (x > xr_margin) {
7533 cursor_region = MID_RIGHT;
7534 } else if (x < xl_margin) {
7535 cursor_region = MID_LEFT;
7536 } else if (y > yb_margin - chartbar_height &&
7537 y < m_canvas_height - chartbar_height) {
7538 cursor_region = MID_TOP;
7539 } else if (y < yt_margin) {
7540 cursor_region = MID_BOT;
7541 } else {
7542 cursor_region = CENTER;
7543 }
7544
7545 if (!g_btouch) SetCanvasCursor(event);
7546
7547 // Protect from leftUp's coming from event handlers in child
7548 // windows who return focus to the canvas.
7549 leftIsDown = event.LeftDown();
7550
7551#ifndef __WXOSX__
7552 if (event.LeftDown()) {
7553 if (g_bShowMenuBar == false && g_bTempShowMenuBar == true) {
7554 // The menu bar is temporarily visible due to alt having been pressed.
7555 // Clicking will hide it, and do nothing else.
7556 g_bTempShowMenuBar = false;
7557 parent_frame->ApplyGlobalSettings(false);
7558 return (true);
7559 }
7560 }
7561#endif
7562
7563 // Update modifiers here; some window managers never send the key event
7564 m_modkeys = 0;
7565 if (event.ControlDown()) m_modkeys |= wxMOD_CONTROL;
7566 if (event.AltDown()) m_modkeys |= wxMOD_ALT;
7567
7568#ifdef __WXMSW__
7569 // TODO Test carefully in other platforms, remove ifdef....
7570 if (event.ButtonDown() && !HasCapture()) CaptureMouse();
7571 if (event.ButtonUp() && HasCapture()) ReleaseMouse();
7572#endif
7573
7574 event.SetEventObject(this);
7575 if (SendMouseEventToPlugins(event))
7576 return (true); // PlugIn did something, and does not want the canvas to
7577 // do anything else
7578
7579 // Capture LeftUp's and time them, unless it already came from the timer.
7580
7581 // Detect end of chart dragging
7582 if (g_btouch && m_bChartDragging && event.LeftUp()) {
7583 StartChartDragInertia();
7584 }
7585
7586 if (b_handle_dclick && event.LeftUp() && !singleClickEventIsValid) {
7587 // Ignore the second LeftUp after the DClick.
7588 if (m_DoubleClickTimer->IsRunning()) {
7589 m_DoubleClickTimer->Stop();
7590 return (true);
7591 }
7592
7593 // Save the event for later running if there is no DClick.
7594 m_DoubleClickTimer->Start(350, wxTIMER_ONE_SHOT);
7595 singleClickEvent = event;
7596 singleClickEventIsValid = true;
7597 return (true);
7598 }
7599
7600 // This logic is necessary on MSW to handle the case where
7601 // a context (right-click) menu is dismissed without action
7602 // by clicking on the chart surface.
7603 // We need to avoid an unintentional pan by eating some clicks...
7604#ifdef __WXMSW__
7605 if (event.LeftDown() || event.LeftUp() || event.Dragging()) {
7606 if (g_click_stop > 0) {
7607 g_click_stop--;
7608 return (true);
7609 }
7610 }
7611#endif
7612
7613 // Kick off the Rotation control timer
7614 if (GetUpMode() == COURSE_UP_MODE) {
7615 m_b_rot_hidef = false;
7616 pRotDefTimer->Start(500, wxTIMER_ONE_SHOT);
7617 } else
7618 pRotDefTimer->Stop();
7619
7620 // Retrigger the route leg / AIS target popup timer
7621 bool bRoll = !g_btouch;
7622#ifdef __ANDROID__
7623 bRoll = g_bRollover;
7624#endif
7625 if (bRoll) {
7626 if ((m_pRouteRolloverWin && m_pRouteRolloverWin->IsActive()) ||
7627 (m_pTrackRolloverWin && m_pTrackRolloverWin->IsActive()) ||
7628 (m_pAISRolloverWin && m_pAISRolloverWin->IsActive()))
7629 m_RolloverPopupTimer.Start(
7630 10,
7631 wxTIMER_ONE_SHOT); // faster response while the rollover is turned on
7632 else
7633 m_RolloverPopupTimer.Start(m_rollover_popup_timer_msec, wxTIMER_ONE_SHOT);
7634 }
7635
7636 // Retrigger the cursor tracking timer
7637 pCurTrackTimer->Start(m_curtrack_timer_msec, wxTIMER_ONE_SHOT);
7638
7639// Show cursor position on Status Bar, if present
7640// except for GTK, under which status bar updates are very slow
7641// due to Update() call.
7642// In this case, as a workaround, update the status window
7643// after an interval timer (pCurTrackTimer) pops, which will happen
7644// whenever the mouse has stopped moving for specified interval.
7645// See the method OnCursorTrackTimerEvent()
7646#if !defined(__WXGTK__) && !defined(__WXQT__)
7647 SetCursorStatus(m_cursor_lat, m_cursor_lon);
7648#endif
7649
7650 // Send the current cursor lat/lon to all PlugIns requesting it
7651 if (g_pi_manager) {
7652 // Occasionally, MSW will produce nonsense events on right click....
7653 // This results in an error in cursor geo position, so we skip this case
7654 if ((x >= 0) && (y >= 0))
7655 SendCursorLatLonToAllPlugIns(m_cursor_lat, m_cursor_lon);
7656 }
7657
7658 if (!g_btouch) {
7659 if ((m_bMeasure_Active && (m_nMeasureState >= 2)) || (m_routeState > 1)) {
7660 wxPoint p = ClientToScreen(wxPoint(x, y));
7661 }
7662 }
7663
7664 if (1 ) {
7665 // Route Creation Rubber Banding
7666 if (m_routeState >= 2) {
7667 r_rband.x = x;
7668 r_rband.y = y;
7669 m_bDrawingRoute = true;
7670
7671 if (!g_btouch) CheckEdgePan(x, y, event.Dragging(), 5, 2);
7672 Refresh(false);
7673 }
7674
7675 // Measure Tool Rubber Banding
7676 if (m_bMeasure_Active && (m_nMeasureState >= 2)) {
7677 r_rband.x = x;
7678 r_rband.y = y;
7679 m_bDrawingRoute = true;
7680
7681 if (!g_btouch) CheckEdgePan(x, y, event.Dragging(), 5, 2);
7682 Refresh(false);
7683 }
7684 }
7685 return bret;
7686}
7687
7688void ChartCanvas::CallPopupMenu(int x, int y) {
7689 int mx, my;
7690 mx = x;
7691 my = y;
7692
7693 last_drag.x = mx;
7694 last_drag.y = my;
7695 if (m_routeState) { // creating route?
7696 InvokeCanvasMenu(x, y, SELTYPE_ROUTECREATE);
7697 return;
7698 }
7699 // General Right Click
7700 // Look for selectable objects
7701 double slat, slon;
7702 slat = m_cursor_lat;
7703 slon = m_cursor_lon;
7704
7705#if defined(__WXMAC__) || defined(__ANDROID__)
7706 wxScreenDC sdc;
7707 ocpnDC dc(sdc);
7708#else
7709 wxClientDC cdc(GetParent());
7710 ocpnDC dc(cdc);
7711#endif
7712
7713 SelectItem *pFindAIS;
7714 SelectItem *pFindRP;
7715 SelectItem *pFindRouteSeg;
7716 SelectItem *pFindTrackSeg;
7717 SelectItem *pFindCurrent = NULL;
7718 SelectItem *pFindTide = NULL;
7719
7720 // Deselect any current objects
7721 if (m_pSelectedRoute) {
7722 m_pSelectedRoute->m_bRtIsSelected = false; // Only one selection at a time
7723 m_pSelectedRoute->DeSelectRoute();
7724#ifdef ocpnUSE_GL
7725 if (g_bopengl && m_glcc) {
7726 InvalidateGL();
7727 Update();
7728 } else
7729#endif
7730 RouteGui(*m_pSelectedRoute).Draw(dc, this, GetVP().GetBBox());
7731 }
7732
7733 if (m_pFoundRoutePoint) {
7734 m_pFoundRoutePoint->m_bPtIsSelected = false;
7735 RoutePointGui(*m_pFoundRoutePoint).Draw(dc, this);
7736 RefreshRect(m_pFoundRoutePoint->CurrentRect_in_DC);
7737 }
7738
7741 if (g_btouch && m_pRoutePointEditTarget) {
7742 m_pRoutePointEditTarget->m_bRPIsBeingEdited = false;
7743 m_pRoutePointEditTarget->m_bPtIsSelected = false;
7744 RoutePointGui(*m_pRoutePointEditTarget).EnableDragHandle(false);
7745 }
7746
7747 // Get all the selectable things at the cursor
7748 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7749 pFindAIS = pSelectAIS->FindSelection(ctx, slat, slon, SELTYPE_AISTARGET);
7750 pFindRP = pSelect->FindSelection(ctx, slat, slon, SELTYPE_ROUTEPOINT);
7751 pFindRouteSeg = pSelect->FindSelection(ctx, slat, slon, SELTYPE_ROUTESEGMENT);
7752 pFindTrackSeg = pSelect->FindSelection(ctx, slat, slon, SELTYPE_TRACKSEGMENT);
7753
7754 if (m_bShowCurrent)
7755 pFindCurrent =
7756 pSelectTC->FindSelection(ctx, slat, slon, SELTYPE_CURRENTPOINT);
7757
7758 if (m_bShowTide) // look for tide stations
7759 pFindTide = pSelectTC->FindSelection(ctx, slat, slon, SELTYPE_TIDEPOINT);
7760
7761 int seltype = 0;
7762
7763 // Try for AIS targets first
7764 if (pFindAIS) {
7765 m_FoundAIS_MMSI = pFindAIS->GetUserData();
7766
7767 // Make sure the target data is available
7768 if (g_pAIS->Get_Target_Data_From_MMSI(m_FoundAIS_MMSI))
7769 seltype |= SELTYPE_AISTARGET;
7770 }
7771
7772 // Now the various Route Parts
7773
7774 m_pFoundRoutePoint = NULL;
7775 if (pFindRP) {
7776 RoutePoint *pFirstVizPoint = NULL;
7777 RoutePoint *pFoundActiveRoutePoint = NULL;
7778 RoutePoint *pFoundVizRoutePoint = NULL;
7779 Route *pSelectedActiveRoute = NULL;
7780 Route *pSelectedVizRoute = NULL;
7781
7782 // There is at least one routepoint, so get the whole list
7783 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7784 SelectableItemList SelList =
7785 pSelect->FindSelectionList(ctx, slat, slon, SELTYPE_ROUTEPOINT);
7786 wxSelectableItemListNode *node = SelList.GetFirst();
7787 while (node) {
7788 SelectItem *pFindSel = node->GetData();
7789
7790 RoutePoint *prp = (RoutePoint *)pFindSel->m_pData1; // candidate
7791
7792 // Get an array of all routes using this point
7793 wxArrayPtrVoid *proute_array = g_pRouteMan->GetRouteArrayContaining(prp);
7794
7795 // Use route array (if any) to determine actual visibility for this point
7796 bool brp_viz = false;
7797 if (proute_array) {
7798 for (unsigned int ir = 0; ir < proute_array->GetCount(); ir++) {
7799 Route *pr = (Route *)proute_array->Item(ir);
7800 if (pr->IsVisible()) {
7801 brp_viz = true;
7802 break;
7803 }
7804 }
7805 if (!brp_viz && prp->IsShared()) // is not visible as part of route,
7806 // but still exists as a waypoint
7807 brp_viz = prp->IsVisible(); // so treat as isolated point
7808
7809 } else
7810 brp_viz = prp->IsVisible(); // isolated point
7811
7812 if ((NULL == pFirstVizPoint) && brp_viz) pFirstVizPoint = prp;
7813
7814 // Use route array to choose the appropriate route
7815 // Give preference to any active route, otherwise select the first visible
7816 // route in the array for this point
7817 m_pSelectedRoute = NULL;
7818 if (proute_array) {
7819 for (unsigned int ir = 0; ir < proute_array->GetCount(); ir++) {
7820 Route *pr = (Route *)proute_array->Item(ir);
7821 if (pr->m_bRtIsActive) {
7822 pSelectedActiveRoute = pr;
7823 pFoundActiveRoutePoint = prp;
7824 break;
7825 }
7826 }
7827
7828 if (NULL == pSelectedVizRoute) {
7829 for (unsigned int ir = 0; ir < proute_array->GetCount(); ir++) {
7830 Route *pr = (Route *)proute_array->Item(ir);
7831 if (pr->IsVisible()) {
7832 pSelectedVizRoute = pr;
7833 pFoundVizRoutePoint = prp;
7834 break;
7835 }
7836 }
7837 }
7838
7839 delete proute_array;
7840 }
7841
7842 node = node->GetNext();
7843 }
7844
7845 // Now choose the "best" selections
7846 if (pFoundActiveRoutePoint) {
7847 m_pFoundRoutePoint = pFoundActiveRoutePoint;
7848 m_pSelectedRoute = pSelectedActiveRoute;
7849 } else if (pFoundVizRoutePoint) {
7850 m_pFoundRoutePoint = pFoundVizRoutePoint;
7851 m_pSelectedRoute = pSelectedVizRoute;
7852 } else
7853 // default is first visible point in list
7854 m_pFoundRoutePoint = pFirstVizPoint;
7855
7856 if (m_pSelectedRoute) {
7857 if (m_pSelectedRoute->IsVisible()) seltype |= SELTYPE_ROUTEPOINT;
7858 } else if (m_pFoundRoutePoint)
7859 seltype |= SELTYPE_MARKPOINT;
7860
7861 // Highlite the selected point, to verify the proper right click
7862 // selection
7863 if (m_pFoundRoutePoint) {
7864 m_pFoundRoutePoint->m_bPtIsSelected = true;
7865 wxRect wp_rect;
7866 RoutePointGui(*m_pFoundRoutePoint)
7867 .CalculateDCRect(m_dc_route, this, &wp_rect);
7868 RefreshRect(wp_rect, true);
7869 }
7870 }
7871
7872 // Note here that we use SELTYPE_ROUTESEGMENT to select tracks as well as
7873 // routes But call the popup handler with identifier appropriate to the type
7874 if (pFindRouteSeg) // there is at least one select item
7875 {
7876 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7877 SelectableItemList SelList =
7878 pSelect->FindSelectionList(ctx, slat, slon, SELTYPE_ROUTESEGMENT);
7879
7880 if (NULL == m_pSelectedRoute) // the case where a segment only is selected
7881 {
7882 // Choose the first visible route containing segment in the list
7883 wxSelectableItemListNode *node = SelList.GetFirst();
7884 while (node) {
7885 SelectItem *pFindSel = node->GetData();
7886
7887 Route *pr = (Route *)pFindSel->m_pData3;
7888 if (pr->IsVisible()) {
7889 m_pSelectedRoute = pr;
7890 break;
7891 }
7892 node = node->GetNext();
7893 }
7894 }
7895
7896 if (m_pSelectedRoute) {
7897 if (NULL == m_pFoundRoutePoint)
7898 m_pFoundRoutePoint = (RoutePoint *)pFindRouteSeg->m_pData1;
7899
7900 m_pSelectedRoute->m_bRtIsSelected = !(seltype & SELTYPE_ROUTEPOINT);
7901 if (m_pSelectedRoute->m_bRtIsSelected) {
7902#ifdef ocpnUSE_GL
7903 if (g_bopengl && m_glcc) {
7904 InvalidateGL();
7905 Update();
7906 } else
7907#endif
7908 RouteGui(*m_pSelectedRoute).Draw(dc, this, GetVP().GetBBox());
7909 }
7910
7911 seltype |= SELTYPE_ROUTESEGMENT;
7912 }
7913 }
7914
7915 if (pFindTrackSeg) {
7916 m_pSelectedTrack = NULL;
7917 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7918 SelectableItemList SelList =
7919 pSelect->FindSelectionList(ctx, slat, slon, SELTYPE_TRACKSEGMENT);
7920
7921 // Choose the first visible track containing segment in the list
7922 wxSelectableItemListNode *node = SelList.GetFirst();
7923 while (node) {
7924 SelectItem *pFindSel = node->GetData();
7925
7926 Track *pt = (Track *)pFindSel->m_pData3;
7927 if (pt->IsVisible()) {
7928 m_pSelectedTrack = pt;
7929 break;
7930 }
7931 node = node->GetNext();
7932 }
7933
7934 if (m_pSelectedTrack) seltype |= SELTYPE_TRACKSEGMENT;
7935 }
7936
7937 bool bseltc = false;
7938 // if(0 == seltype)
7939 {
7940 if (pFindCurrent) {
7941 // There may be multiple current entries at the same point.
7942 // For example, there often is a current substation (with directions
7943 // specified) co-located with its master. We want to select the
7944 // substation, so that the direction will be properly indicated on the
7945 // graphic. So, we search the select list looking for IDX_type == 'c' (i.e
7946 // substation)
7947 IDX_entry *pIDX_best_candidate;
7948
7949 SelectItem *pFind = NULL;
7950 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
7951 SelectableItemList SelList = pSelectTC->FindSelectionList(
7952 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_CURRENTPOINT);
7953
7954 // Default is first entry
7955 wxSelectableItemListNode *node = SelList.GetFirst();
7956 pFind = node->GetData();
7957 pIDX_best_candidate = (IDX_entry *)(pFind->m_pData1);
7958
7959 if (SelList.GetCount() > 1) {
7960 node = node->GetNext();
7961 while (node) {
7962 pFind = node->GetData();
7963 IDX_entry *pIDX_candidate = (IDX_entry *)(pFind->m_pData1);
7964 if (pIDX_candidate->IDX_type == 'c') {
7965 pIDX_best_candidate = pIDX_candidate;
7966 break;
7967 }
7968
7969 node = node->GetNext();
7970 } // while (node)
7971 } else {
7972 wxSelectableItemListNode *node = SelList.GetFirst();
7973 pFind = node->GetData();
7974 pIDX_best_candidate = (IDX_entry *)(pFind->m_pData1);
7975 }
7976
7977 m_pIDXCandidate = pIDX_best_candidate;
7978
7979 if (0 == seltype) {
7980 DrawTCWindow(x, y, (void *)pIDX_best_candidate);
7981 Refresh(false);
7982 bseltc = true;
7983 } else
7984 seltype |= SELTYPE_CURRENTPOINT;
7985 }
7986
7987 else if (pFindTide) {
7988 m_pIDXCandidate = (IDX_entry *)pFindTide->m_pData1;
7989
7990 if (0 == seltype) {
7991 DrawTCWindow(x, y, (void *)pFindTide->m_pData1);
7992 Refresh(false);
7993 bseltc = true;
7994 } else
7995 seltype |= SELTYPE_TIDEPOINT;
7996 }
7997 }
7998
7999 if (0 == seltype) seltype |= SELTYPE_UNKNOWN;
8000
8001 if (!bseltc) {
8002 InvokeCanvasMenu(x, y, seltype);
8003
8004 // Clean up if not deleted in InvokeCanvasMenu
8005 if (m_pSelectedRoute && g_pRouteMan->IsRouteValid(m_pSelectedRoute)) {
8006 m_pSelectedRoute->m_bRtIsSelected = false;
8007 }
8008
8009 m_pSelectedRoute = NULL;
8010
8011 if (m_pFoundRoutePoint) {
8012 if (pSelect->IsSelectableRoutePointValid(m_pFoundRoutePoint))
8013 m_pFoundRoutePoint->m_bPtIsSelected = false;
8014 }
8015 m_pFoundRoutePoint = NULL;
8016
8017 Refresh(true);
8018 }
8019
8020 // Seth: Is this refresh needed?
8021 Refresh(false); // needed for MSW, not GTK Why??
8022}
8023bool ChartCanvas::MouseEventProcessObjects(wxMouseEvent &event) {
8024 // For now just bail out completely if the point clicked is not on the chart
8025 if (std::isnan(m_cursor_lat)) return false;
8026
8027 // Mouse Clicks
8028 bool ret = false; // return true if processed
8029
8030 int x, y, mx, my;
8031 event.GetPosition(&x, &y);
8032 mx = x;
8033 my = y;
8034
8035 // Calculate meaningful SelectRadius
8036 float SelectRadius;
8037 SelectRadius = g_Platform->GetSelectRadiusPix() /
8038 (m_true_scale_ppm * 1852 * 60); // Degrees, approximately
8039
8041 // We start with Double Click processing. The first left click just starts a
8042 // timer and is remembered, then we actually do something if there is a
8043 // LeftDClick. If there is, the two single clicks are ignored.
8044
8045 if (event.LeftDClick() && (cursor_region == CENTER)) {
8046 m_DoubleClickTimer->Start();
8047 singleClickEventIsValid = false;
8048
8049 double zlat, zlon;
8050 GetCanvasPixPoint(x * g_current_monitor_dip_px_ratio,
8051 y * g_current_monitor_dip_px_ratio, zlat, zlon);
8052
8053 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
8054 if (m_bShowAIS) {
8055 SelectItem *pFindAIS;
8056 pFindAIS = pSelectAIS->FindSelection(ctx, zlat, zlon, SELTYPE_AISTARGET);
8057
8058 if (pFindAIS) {
8059 m_FoundAIS_MMSI = pFindAIS->GetUserData();
8060 if (g_pAIS->Get_Target_Data_From_MMSI(m_FoundAIS_MMSI)) {
8061 ShowAISTargetQueryDialog(this, m_FoundAIS_MMSI);
8062 }
8063 return true;
8064 }
8065 }
8066
8067 SelectableItemList rpSelList =
8068 pSelect->FindSelectionList(ctx, zlat, zlon, SELTYPE_ROUTEPOINT);
8069 wxSelectableItemListNode *node = rpSelList.GetFirst();
8070 bool b_onRPtarget = false;
8071 while (node) {
8072 SelectItem *pFind = node->GetData();
8073 RoutePoint *frp = (RoutePoint *)pFind->m_pData1;
8074 if (m_pRoutePointEditTarget && (frp == m_pRoutePointEditTarget)) {
8075 b_onRPtarget = true;
8076 break;
8077 }
8078 node = node->GetNext();
8079 }
8080
8081 // Double tap with selected RoutePoint or Mark
8082
8083 if (m_pRoutePointEditTarget) {
8084 if (b_onRPtarget) {
8085 ShowMarkPropertiesDialog(m_pRoutePointEditTarget);
8086 return true;
8087 } else {
8088 m_pRoutePointEditTarget->m_bRPIsBeingEdited = false;
8089 m_pRoutePointEditTarget->m_bPtIsSelected = false;
8090 if (g_btouch)
8091 RoutePointGui(*m_pRoutePointEditTarget).EnableDragHandle(false);
8092 wxRect wp_rect;
8093 RoutePointGui(*m_pRoutePointEditTarget)
8094 .CalculateDCRect(m_dc_route, this, &wp_rect);
8095 m_pRoutePointEditTarget = NULL; // cancel selection
8096 RefreshRect(wp_rect, true);
8097 return true;
8098 }
8099 } else {
8100 node = rpSelList.GetFirst();
8101 if (node) {
8102 SelectItem *pFind = node->GetData();
8103 RoutePoint *frp = (RoutePoint *)pFind->m_pData1;
8104 if (frp) {
8105 wxArrayPtrVoid *proute_array =
8106 g_pRouteMan->GetRouteArrayContaining(frp);
8107
8108 // Use route array (if any) to determine actual visibility for this
8109 // point
8110 bool brp_viz = false;
8111 if (proute_array) {
8112 for (unsigned int ir = 0; ir < proute_array->GetCount(); ir++) {
8113 Route *pr = (Route *)proute_array->Item(ir);
8114 if (pr->IsVisible()) {
8115 brp_viz = true;
8116 break;
8117 }
8118 }
8119 if (!brp_viz &&
8120 frp->IsShared()) // is not visible as part of route, but still
8121 // exists as a waypoint
8122 brp_viz = frp->IsVisible(); // so treat as isolated point
8123 } else
8124 brp_viz = frp->IsVisible(); // isolated point
8125
8126 if (brp_viz) {
8127 ShowMarkPropertiesDialog(frp);
8128 return true;
8129 }
8130 }
8131 }
8132 }
8133
8134 SelectItem *cursorItem;
8135 cursorItem = pSelect->FindSelection(ctx, zlat, zlon, SELTYPE_ROUTESEGMENT);
8136
8137 if (cursorItem) {
8138 Route *pr = (Route *)cursorItem->m_pData3;
8139 if (pr->IsVisible()) {
8140 ShowRoutePropertiesDialog(_("Route Properties"), pr);
8141 return true;
8142 }
8143 }
8144
8145 cursorItem = pSelect->FindSelection(ctx, zlat, zlon, SELTYPE_TRACKSEGMENT);
8146
8147 if (cursorItem) {
8148 Track *pt = (Track *)cursorItem->m_pData3;
8149 if (pt->IsVisible()) {
8150 ShowTrackPropertiesDialog(pt);
8151 return true;
8152 }
8153 }
8154
8155 // Found no object to act on, so show chart info.
8156
8157 ShowObjectQueryWindow(x, y, zlat, zlon);
8158 return true;
8159 }
8160
8162 if (event.LeftDown()) {
8163 // This really should not be needed, but....
8164 // on Windows, when using wxAUIManager, sometimes the focus is lost
8165 // when clicking into another pane, e.g.the AIS target list, and then back
8166 // to this pane. Oddly, some mouse events are not lost, however. Like this
8167 // one....
8168 SetFocus();
8169
8170 last_drag.x = mx;
8171 last_drag.y = my;
8172 leftIsDown = true;
8173
8174 if (!g_btouch) {
8175 if (m_routeState) // creating route?
8176 {
8177 double rlat, rlon;
8178 bool appending = false;
8179 bool inserting = false;
8180 Route *tail = 0;
8181
8182 SetCursor(*pCursorPencil);
8183 rlat = m_cursor_lat;
8184 rlon = m_cursor_lon;
8185
8186 m_bRouteEditing = true;
8187
8188 if (m_routeState == 1) {
8189 m_pMouseRoute = new Route();
8190 pRouteList->Append(m_pMouseRoute);
8191 r_rband.x = x;
8192 r_rband.y = y;
8193 }
8194
8195 // Check to see if there is a nearby point which may be reused
8196 RoutePoint *pMousePoint = NULL;
8197
8198 // Calculate meaningful SelectRadius
8199 double nearby_radius_meters =
8200 g_Platform->GetSelectRadiusPix() / m_true_scale_ppm;
8201
8202 RoutePoint *pNearbyPoint =
8203 pWayPointMan->GetNearbyWaypoint(rlat, rlon, nearby_radius_meters);
8204 if (pNearbyPoint && (pNearbyPoint != m_prev_pMousePoint) &&
8205 !pNearbyPoint->m_bIsInLayer && pNearbyPoint->IsVisible()) {
8206 wxArrayPtrVoid *proute_array =
8207 g_pRouteMan->GetRouteArrayContaining(pNearbyPoint);
8208
8209 // Use route array (if any) to determine actual visibility for this
8210 // point
8211 bool brp_viz = false;
8212 if (proute_array) {
8213 for (unsigned int ir = 0; ir < proute_array->GetCount(); ir++) {
8214 Route *pr = (Route *)proute_array->Item(ir);
8215 if (pr->IsVisible()) {
8216 brp_viz = true;
8217 break;
8218 }
8219 }
8220 if (!brp_viz &&
8221 pNearbyPoint->IsShared()) // is not visible as part of route,
8222 // but still exists as a waypoint
8223 brp_viz =
8224 pNearbyPoint->IsVisible(); // so treat as isolated point
8225 } else
8226 brp_viz = pNearbyPoint->IsVisible(); // isolated point
8227
8228 if (brp_viz) {
8229 wxString msg = _("Use nearby waypoint?");
8230 // Don't add a mark without name to the route. Name it if needed
8231 const bool noname(pNearbyPoint->GetName() == "");
8232 if (noname) {
8233 msg =
8234 _("Use nearby nameless waypoint and name it M with"
8235 " a unique number?");
8236 }
8237 // Avoid route finish on focus change for message dialog
8238 m_FinishRouteOnKillFocus = false;
8239 int dlg_return =
8240 OCPNMessageBox(this, msg, _("OpenCPN Route Create"),
8241 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8242 m_FinishRouteOnKillFocus = true;
8243 if (dlg_return == wxID_YES) {
8244 if (noname) {
8245 if (m_pMouseRoute) {
8246 int last_wp_num = m_pMouseRoute->GetnPoints();
8247 // AP-ECRMB will truncate to 6 characters
8248 wxString guid_short = m_pMouseRoute->GetGUID().Left(2);
8249 wxString wp_name = wxString::Format(
8250 "M%002i-%s", last_wp_num + 1, guid_short);
8251 pNearbyPoint->SetName(wp_name);
8252 } else
8253 pNearbyPoint->SetName("WPXX");
8254 }
8255 pMousePoint = pNearbyPoint;
8256
8257 // Using existing waypoint, so nothing to delete for undo.
8258 if (m_routeState > 1)
8259 undo->BeforeUndoableAction(Undo_AppendWaypoint, pMousePoint,
8260 Undo_HasParent, NULL);
8261
8262 tail =
8263 g_pRouteMan->FindVisibleRouteContainingWaypoint(pMousePoint);
8264 bool procede = false;
8265 if (tail) {
8266 procede = true;
8267 // if (pMousePoint == tail->GetLastPoint()) procede = false;
8268 if (m_routeState > 1 && m_pMouseRoute && tail == m_pMouseRoute)
8269 procede = false;
8270 }
8271
8272 if (procede) {
8273 int dlg_return;
8274 m_FinishRouteOnKillFocus = false;
8275 if (m_routeState ==
8276 1) { // first point in new route, preceeding route to be
8277 // added? Not touch case
8278
8279 wxString dmsg =
8280 _("Insert first part of this route in the new route?");
8281 if (tail->GetIndexOf(pMousePoint) ==
8282 tail->GetnPoints()) // Starting on last point of another
8283 // route?
8284 dmsg = _("Insert this route in the new route?");
8285
8286 if (tail->GetIndexOf(pMousePoint) != 1) { // Anything to do?
8287 dlg_return = OCPNMessageBox(
8288 this, dmsg, _("OpenCPN Route Create"),
8289 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8290 m_FinishRouteOnKillFocus = true;
8291
8292 if (dlg_return == wxID_YES) {
8293 inserting = true; // part of the other route will be
8294 // preceeding the new route
8295 }
8296 }
8297 } else {
8298 wxString dmsg =
8299 _("Append last part of this route to the new route?");
8300 if (tail->GetIndexOf(pMousePoint) == 1)
8301 dmsg = _(
8302 "Append this route to the new route?"); // Picking the
8303 // first point
8304 // of another
8305 // route?
8306
8307 if (tail->GetLastPoint() != pMousePoint) { // Anything to do?
8308 dlg_return = OCPNMessageBox(
8309 this, dmsg, _("OpenCPN Route Create"),
8310 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8311 m_FinishRouteOnKillFocus = true;
8312
8313 if (dlg_return == wxID_YES) {
8314 appending = true; // part of the other route will be
8315 // appended to the new route
8316 }
8317 }
8318 }
8319 }
8320
8321 // check all other routes to see if this point appears in any
8322 // other route If it appears in NO other route, then it should e
8323 // considered an isolated mark
8324 if (!FindRouteContainingWaypoint(pMousePoint))
8325 pMousePoint->SetShared(true);
8326 }
8327 }
8328 }
8329
8330 if (NULL == pMousePoint) { // need a new point
8331 pMousePoint = new RoutePoint(rlat, rlon, g_default_routepoint_icon,
8332 _T(""), wxEmptyString);
8333 pMousePoint->SetNameShown(false);
8334
8335 pConfig->AddNewWayPoint(pMousePoint, -1); // use auto next num
8336 pSelect->AddSelectableRoutePoint(rlat, rlon, pMousePoint);
8337
8338 if (m_routeState > 1)
8339 undo->BeforeUndoableAction(Undo_AppendWaypoint, pMousePoint,
8340 Undo_IsOrphanded, NULL);
8341 }
8342
8343 if (m_pMouseRoute) {
8344 if (m_routeState == 1) {
8345 // First point in the route.
8346 m_pMouseRoute->AddPoint(pMousePoint);
8347 } else {
8348 if (m_pMouseRoute->m_NextLegGreatCircle) {
8349 double rhumbBearing, rhumbDist, gcBearing, gcDist;
8350 DistanceBearingMercator(rlat, rlon, m_prev_rlat, m_prev_rlon,
8351 &rhumbBearing, &rhumbDist);
8352 Geodesic::GreatCircleDistBear(m_prev_rlon, m_prev_rlat, rlon,
8353 rlat, &gcDist, &gcBearing, NULL);
8354 double gcDistNM = gcDist / 1852.0;
8355
8356 // Empirically found expression to get reasonable route segments.
8357 int segmentCount = (3.0 + (rhumbDist - gcDistNM)) /
8358 pow(rhumbDist - gcDistNM - 1, 0.5);
8359
8360 wxString msg;
8361 msg << _("For this leg the Great Circle route is ")
8362 << FormatDistanceAdaptive(rhumbDist - gcDistNM)
8363 << _(" shorter than rhumbline.\n\n")
8364 << _("Would you like include the Great Circle routing points "
8365 "for this leg?");
8366
8367 m_FinishRouteOnKillFocus = false;
8368 m_disable_edge_pan = true; // This helps on OS X if MessageBox
8369 // does not fully capture mouse
8370
8371 int answer = OCPNMessageBox(this, msg, _("OpenCPN Route Create"),
8372 wxYES_NO | wxNO_DEFAULT);
8373
8374 m_disable_edge_pan = false;
8375 m_FinishRouteOnKillFocus = true;
8376
8377 if (answer == wxID_YES) {
8378 RoutePoint *gcPoint;
8379 RoutePoint *prevGcPoint = m_prev_pMousePoint;
8380 wxRealPoint gcCoord;
8381
8382 for (int i = 1; i <= segmentCount; i++) {
8383 double fraction = (double)i * (1.0 / (double)segmentCount);
8384 Geodesic::GreatCircleTravel(m_prev_rlon, m_prev_rlat,
8385 gcDist * fraction, gcBearing,
8386 &gcCoord.x, &gcCoord.y, NULL);
8387
8388 if (i < segmentCount) {
8389 gcPoint = new RoutePoint(gcCoord.y, gcCoord.x, _T("xmblue"),
8390 _T(""), wxEmptyString);
8391 gcPoint->SetNameShown(false);
8392 pConfig->AddNewWayPoint(gcPoint, -1);
8393 pSelect->AddSelectableRoutePoint(gcCoord.y, gcCoord.x,
8394 gcPoint);
8395 } else {
8396 gcPoint = pMousePoint; // Last point, previously exsisting!
8397 }
8398
8399 m_pMouseRoute->AddPoint(gcPoint);
8400 pSelect->AddSelectableRouteSegment(
8401 prevGcPoint->m_lat, prevGcPoint->m_lon, gcPoint->m_lat,
8402 gcPoint->m_lon, prevGcPoint, gcPoint, m_pMouseRoute);
8403 prevGcPoint = gcPoint;
8404 }
8405
8406 undo->CancelUndoableAction(true);
8407
8408 } else {
8409 m_pMouseRoute->AddPoint(pMousePoint);
8410 pSelect->AddSelectableRouteSegment(
8411 m_prev_rlat, m_prev_rlon, rlat, rlon, m_prev_pMousePoint,
8412 pMousePoint, m_pMouseRoute);
8413 undo->AfterUndoableAction(m_pMouseRoute);
8414 }
8415 } else {
8416 // Ordinary rhumblinesegment.
8417 m_pMouseRoute->AddPoint(pMousePoint);
8418 pSelect->AddSelectableRouteSegment(m_prev_rlat, m_prev_rlon, rlat,
8419 rlon, m_prev_pMousePoint,
8420 pMousePoint, m_pMouseRoute);
8421 undo->AfterUndoableAction(m_pMouseRoute);
8422 }
8423 }
8424 }
8425 m_prev_rlat = rlat;
8426 m_prev_rlon = rlon;
8427 m_prev_pMousePoint = pMousePoint;
8428 if (m_pMouseRoute)
8429 m_pMouseRoute->m_lastMousePointIndex = m_pMouseRoute->GetnPoints();
8430
8431 m_routeState++;
8432
8433 if (appending ||
8434 inserting) { // Appending a route or making a new route
8435 int connect = tail->GetIndexOf(pMousePoint);
8436 if (connect == 1) {
8437 inserting = false; // there is nothing to insert
8438 appending = true; // so append
8439 }
8440 int length = tail->GetnPoints();
8441
8442 int i;
8443 int start, stop;
8444 if (appending) {
8445 start = connect + 1;
8446 stop = length;
8447 } else { // inserting
8448 start = 1;
8449 stop = connect;
8450 m_pMouseRoute->RemovePoint(
8451 m_pMouseRoute
8452 ->GetLastPoint()); // Remove the first and only point
8453 }
8454 for (i = start; i <= stop; i++) {
8455 m_pMouseRoute->AddPointAndSegment(tail->GetPoint(i), false);
8456 if (m_pMouseRoute)
8457 m_pMouseRoute->m_lastMousePointIndex =
8458 m_pMouseRoute->GetnPoints();
8459 m_routeState++;
8460 gFrame->RefreshAllCanvas();
8461 ret = true;
8462 }
8463 m_prev_rlat =
8464 m_pMouseRoute->GetPoint(m_pMouseRoute->GetnPoints())->m_lat;
8465 m_prev_rlon =
8466 m_pMouseRoute->GetPoint(m_pMouseRoute->GetnPoints())->m_lon;
8467 m_pMouseRoute->FinalizeForRendering();
8468 }
8469 gFrame->RefreshAllCanvas();
8470 ret = true;
8471 }
8472
8473 else if (m_bMeasure_Active && (m_nMeasureState >= 1)) // measure tool?
8474 {
8475 SetCursor(*pCursorPencil);
8476
8477 if (!m_pMeasureRoute) {
8478 m_pMeasureRoute = new Route();
8479 pRouteList->Append(m_pMeasureRoute);
8480 }
8481
8482 if (m_nMeasureState == 1) {
8483 r_rband.x = x;
8484 r_rband.y = y;
8485 }
8486
8487 RoutePoint *pMousePoint = new RoutePoint(m_cursor_lat, m_cursor_lon,
8488 wxString(_T ( "circle" )),
8489 wxEmptyString, wxEmptyString);
8490 pMousePoint->m_bShowName = false;
8491 pMousePoint->SetShowWaypointRangeRings(false);
8492
8493 m_pMeasureRoute->AddPoint(pMousePoint);
8494
8495 m_prev_rlat = m_cursor_lat;
8496 m_prev_rlon = m_cursor_lon;
8497 m_prev_pMousePoint = pMousePoint;
8498 m_pMeasureRoute->m_lastMousePointIndex = m_pMeasureRoute->GetnPoints();
8499
8500 m_nMeasureState++;
8501 gFrame->RefreshAllCanvas();
8502 ret = true;
8503 }
8504
8505 else {
8506 FindRoutePointsAtCursor(SelectRadius, true); // Not creating Route
8507 }
8508 } // !g_btouch
8509 else { // g_btouch
8510
8511 if ((m_bMeasure_Active && m_nMeasureState) || (m_routeState)) {
8512 // if near screen edge, pan with injection
8513 // if( CheckEdgePan( x, y, true, 5, 10 ) ) {
8514 // return;
8515 // }
8516 }
8517 }
8518
8519 if (ret) return true;
8520 }
8521
8522 if (event.Dragging()) {
8523 // in touch screen mode ensure the finger/cursor is on the selected point's
8524 // radius to allow dragging
8525 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
8526 if (g_btouch) {
8527 if (m_pRoutePointEditTarget && !m_bIsInRadius) {
8528 SelectItem *pFind = NULL;
8529 SelectableItemList SelList = pSelect->FindSelectionList(
8530 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_ROUTEPOINT);
8531 wxSelectableItemListNode *node = SelList.GetFirst();
8532 while (node) {
8533 pFind = node->GetData();
8534 RoutePoint *frp = (RoutePoint *)pFind->m_pData1;
8535 if (m_pRoutePointEditTarget == frp) m_bIsInRadius = true;
8536 node = node->GetNext();
8537 }
8538 }
8539
8540 // Check for use of dragHandle
8541 if (m_pRoutePointEditTarget &&
8542 m_pRoutePointEditTarget->IsDragHandleEnabled()) {
8543 SelectItem *pFind = NULL;
8544 SelectableItemList SelList = pSelect->FindSelectionList(
8545 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_DRAGHANDLE);
8546 wxSelectableItemListNode *node = SelList.GetFirst();
8547 while (node) {
8548 pFind = node->GetData();
8549 RoutePoint *frp = (RoutePoint *)pFind->m_pData1;
8550 if (m_pRoutePointEditTarget == frp) {
8551 m_bIsInRadius = true;
8552 break;
8553 }
8554 node = node->GetNext();
8555 }
8556
8557 if (!m_dragoffsetSet) {
8558 RoutePointGui(*m_pRoutePointEditTarget)
8559 .PresetDragOffset(this, mouse_x, mouse_y);
8560 m_dragoffsetSet = true;
8561 }
8562 }
8563 }
8564
8565 if (m_bRouteEditing && m_pRoutePointEditTarget) {
8566 bool DraggingAllowed = g_btouch ? m_bIsInRadius : true;
8567
8568 if (NULL == g_pMarkInfoDialog) {
8569 if (g_bWayPointPreventDragging) DraggingAllowed = false;
8570 } else if (!g_pMarkInfoDialog->IsShown() && g_bWayPointPreventDragging)
8571 DraggingAllowed = false;
8572
8573 if (m_pRoutePointEditTarget &&
8574 (m_pRoutePointEditTarget->GetIconName() == _T("mob")))
8575 DraggingAllowed = false;
8576
8577 if (m_pRoutePointEditTarget->m_bIsInLayer) DraggingAllowed = false;
8578
8579 if (DraggingAllowed) {
8580 if (!undo->InUndoableAction()) {
8581 undo->BeforeUndoableAction(Undo_MoveWaypoint, m_pRoutePointEditTarget,
8582 Undo_NeedsCopy, m_pFoundPoint);
8583 }
8584
8585 // Get the update rectangle for the union of the un-edited routes
8586 wxRect pre_rect;
8587
8588 if (!g_bopengl && m_pEditRouteArray) {
8589 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount(); ir++) {
8590 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
8591 // Need to validate route pointer
8592 // Route may be gone due to drgging close to ownship with
8593 // "Delete On Arrival" state set, as in the case of
8594 // navigating to an isolated waypoint on a temporary route
8595 if (g_pRouteMan->IsRouteValid(pr)) {
8596 wxRect route_rect;
8597 RouteGui(*pr).CalculateDCRect(m_dc_route, this, &route_rect);
8598 pre_rect.Union(route_rect);
8599 }
8600 }
8601 }
8602
8603 double new_cursor_lat = m_cursor_lat;
8604 double new_cursor_lon = m_cursor_lon;
8605
8606 if (CheckEdgePan(x, y, true, 5, 2))
8607 GetCanvasPixPoint(x, y, new_cursor_lat, new_cursor_lon);
8608
8609 // update the point itself
8610 if (g_btouch) {
8611 // m_pRoutePointEditTarget->SetPointFromDraghandlePoint(VPoint,
8612 // new_cursor_lat, new_cursor_lon);
8613 RoutePointGui(*m_pRoutePointEditTarget)
8614 .SetPointFromDraghandlePoint(this, mouse_x, mouse_y);
8615 // update the Drag Handle entry in the pSelect list
8616 pSelect->ModifySelectablePoint(new_cursor_lat, new_cursor_lon,
8617 m_pRoutePointEditTarget,
8618 SELTYPE_DRAGHANDLE);
8619 m_pFoundPoint->m_slat =
8620 m_pRoutePointEditTarget->m_lat; // update the SelectList entry
8621 m_pFoundPoint->m_slon = m_pRoutePointEditTarget->m_lon;
8622 } else {
8623 m_pRoutePointEditTarget->m_lat =
8624 new_cursor_lat; // update the RoutePoint entry
8625 m_pRoutePointEditTarget->m_lon = new_cursor_lon;
8626 m_pRoutePointEditTarget->m_wpBBox.Invalidate();
8627 m_pFoundPoint->m_slat =
8628 new_cursor_lat; // update the SelectList entry
8629 m_pFoundPoint->m_slon = new_cursor_lon;
8630 }
8631
8632 // Update the MarkProperties Dialog, if currently shown
8633 if ((NULL != g_pMarkInfoDialog) && (g_pMarkInfoDialog->IsShown())) {
8634 if (m_pRoutePointEditTarget == g_pMarkInfoDialog->GetRoutePoint())
8635 g_pMarkInfoDialog->UpdateProperties(true);
8636 }
8637
8638 if (g_bopengl) {
8639 // InvalidateGL();
8640 Refresh(false);
8641 } else {
8642 // Get the update rectangle for the edited route
8643 wxRect post_rect;
8644
8645 if (m_pEditRouteArray) {
8646 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
8647 ir++) {
8648 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
8649 if (g_pRouteMan->IsRouteValid(pr)) {
8650 wxRect route_rect;
8651 RouteGui(*pr).CalculateDCRect(m_dc_route, this, &route_rect);
8652 post_rect.Union(route_rect);
8653 }
8654 }
8655 }
8656
8657 // Invalidate the union region
8658 pre_rect.Union(post_rect);
8659 RefreshRect(pre_rect, false);
8660 }
8661 gFrame->RefreshCanvasOther(this);
8662 m_bRoutePoinDragging = true;
8663 }
8664 ret = true;
8665 } // if Route Editing
8666
8667 else if (m_bMarkEditing && m_pRoutePointEditTarget) {
8668 bool DraggingAllowed = g_btouch ? m_bIsInRadius : true;
8669
8670 if (NULL == g_pMarkInfoDialog) {
8671 if (g_bWayPointPreventDragging) DraggingAllowed = false;
8672 } else if (!g_pMarkInfoDialog->IsShown() && g_bWayPointPreventDragging)
8673 DraggingAllowed = false;
8674
8675 if (m_pRoutePointEditTarget &&
8676 (m_pRoutePointEditTarget->GetIconName() == _T("mob")))
8677 DraggingAllowed = false;
8678
8679 if (m_pRoutePointEditTarget->m_bIsInLayer) DraggingAllowed = false;
8680
8681 if (DraggingAllowed) {
8682 if (!undo->InUndoableAction()) {
8683 undo->BeforeUndoableAction(Undo_MoveWaypoint, m_pRoutePointEditTarget,
8684 Undo_NeedsCopy, m_pFoundPoint);
8685 }
8686
8687 // The mark may be an anchorwatch
8688 double lpp1 = 0.;
8689 double lpp2 = 0.;
8690 double lppmax;
8691
8692 if (pAnchorWatchPoint1 == m_pRoutePointEditTarget) {
8693 lpp1 = fabs(GetAnchorWatchRadiusPixels(pAnchorWatchPoint1));
8694 }
8695 if (pAnchorWatchPoint2 == m_pRoutePointEditTarget) {
8696 lpp2 = fabs(GetAnchorWatchRadiusPixels(pAnchorWatchPoint2));
8697 }
8698 lppmax = wxMax(lpp1 + 10, lpp2 + 10); // allow for cruft
8699
8700 // Get the update rectangle for the un-edited mark
8701 wxRect pre_rect;
8702 if (!g_bopengl) {
8703 RoutePointGui(*m_pRoutePointEditTarget)
8704 .CalculateDCRect(m_dc_route, this, &pre_rect);
8705 if ((lppmax > pre_rect.width / 2) || (lppmax > pre_rect.height / 2))
8706 pre_rect.Inflate((int)(lppmax - (pre_rect.width / 2)),
8707 (int)(lppmax - (pre_rect.height / 2)));
8708 }
8709
8710 // update the point itself
8711 if (g_btouch) {
8712 // m_pRoutePointEditTarget->SetPointFromDraghandlePoint(VPoint,
8713 // m_cursor_lat, m_cursor_lon);
8714 RoutePointGui(*m_pRoutePointEditTarget)
8715 .SetPointFromDraghandlePoint(this, mouse_x, mouse_y);
8716 // update the Drag Handle entry in the pSelect list
8717 pSelect->ModifySelectablePoint(m_cursor_lat, m_cursor_lon,
8718 m_pRoutePointEditTarget,
8719 SELTYPE_DRAGHANDLE);
8720 m_pFoundPoint->m_slat =
8721 m_pRoutePointEditTarget->m_lat; // update the SelectList entry
8722 m_pFoundPoint->m_slon = m_pRoutePointEditTarget->m_lon;
8723 } else {
8724 m_pRoutePointEditTarget->m_lat =
8725 m_cursor_lat; // update the RoutePoint entry
8726 m_pRoutePointEditTarget->m_lon = m_cursor_lon;
8727 m_pRoutePointEditTarget->m_wpBBox.Invalidate();
8728 m_pFoundPoint->m_slat = m_cursor_lat; // update the SelectList entry
8729 m_pFoundPoint->m_slon = m_cursor_lon;
8730 }
8731
8732 // Update the MarkProperties Dialog, if currently shown
8733 if ((NULL != g_pMarkInfoDialog) && (g_pMarkInfoDialog->IsShown())) {
8734 if (m_pRoutePointEditTarget == g_pMarkInfoDialog->GetRoutePoint())
8735 g_pMarkInfoDialog->UpdateProperties(true);
8736 }
8737
8738 // Invalidate the union region
8739 if (g_bopengl) {
8740 if (!g_btouch) InvalidateGL();
8741 Refresh(false);
8742 } else {
8743 // Get the update rectangle for the edited mark
8744 wxRect post_rect;
8745 RoutePointGui(*m_pRoutePointEditTarget)
8746 .CalculateDCRect(m_dc_route, this, &post_rect);
8747 if ((lppmax > post_rect.width / 2) || (lppmax > post_rect.height / 2))
8748 post_rect.Inflate((int)(lppmax - (post_rect.width / 2)),
8749 (int)(lppmax - (post_rect.height / 2)));
8750
8751 // Invalidate the union region
8752 pre_rect.Union(post_rect);
8753 RefreshRect(pre_rect, false);
8754 }
8755 gFrame->RefreshCanvasOther(this);
8756 m_bRoutePoinDragging = true;
8757 }
8758 ret = true;
8759 }
8760
8761 if (ret) return true;
8762 } // dragging
8763
8764 if (event.LeftUp()) {
8765 bool b_startedit_route = false;
8766 m_dragoffsetSet = false;
8767
8768 if (g_btouch) {
8769 m_bChartDragging = false;
8770 m_bIsInRadius = false;
8771
8772 if (m_routeState) // creating route?
8773 {
8774 if (m_bedge_pan) {
8775 m_bedge_pan = false;
8776 return false;
8777 }
8778
8779 double rlat, rlon;
8780 bool appending = false;
8781 bool inserting = false;
8782 Route *tail = 0;
8783
8784 rlat = m_cursor_lat;
8785 rlon = m_cursor_lon;
8786
8787 if (m_pRoutePointEditTarget) {
8788 m_pRoutePointEditTarget->m_bRPIsBeingEdited = false;
8789 m_pRoutePointEditTarget->m_bPtIsSelected = false;
8790 if (!g_bopengl) {
8791 wxRect wp_rect;
8792 RoutePointGui(*m_pRoutePointEditTarget)
8793 .CalculateDCRect(m_dc_route, this, &wp_rect);
8794 RefreshRect(wp_rect, true);
8795 }
8796 m_pRoutePointEditTarget = NULL;
8797 }
8798 m_bRouteEditing = true;
8799
8800 if (m_routeState == 1) {
8801 m_pMouseRoute = new Route();
8802 m_pMouseRoute->SetHiLite(50);
8803 pRouteList->Append(m_pMouseRoute);
8804 r_rband.x = x;
8805 r_rband.y = y;
8806 }
8807
8808 // Check to see if there is a nearby point which may be reused
8809 RoutePoint *pMousePoint = NULL;
8810
8811 // Calculate meaningful SelectRadius
8812 double nearby_radius_meters =
8813 g_Platform->GetSelectRadiusPix() / m_true_scale_ppm;
8814
8815 RoutePoint *pNearbyPoint =
8816 pWayPointMan->GetNearbyWaypoint(rlat, rlon, nearby_radius_meters);
8817 if (pNearbyPoint && (pNearbyPoint != m_prev_pMousePoint) &&
8818 !pNearbyPoint->m_bIsInLayer && pNearbyPoint->IsVisible()) {
8819 int dlg_return;
8820#ifndef __WXOSX__
8821 m_FinishRouteOnKillFocus =
8822 false; // Avoid route finish on focus change for message dialog
8823 dlg_return = OCPNMessageBox(
8824 this, _("Use nearby waypoint?"), _("OpenCPN Route Create"),
8825 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8826 m_FinishRouteOnKillFocus = true;
8827#else
8828 dlg_return = wxID_YES;
8829#endif
8830 if (dlg_return == wxID_YES) {
8831 pMousePoint = pNearbyPoint;
8832
8833 // Using existing waypoint, so nothing to delete for undo.
8834 if (m_routeState > 1)
8835 undo->BeforeUndoableAction(Undo_AppendWaypoint, pMousePoint,
8836 Undo_HasParent, NULL);
8837 tail = g_pRouteMan->FindVisibleRouteContainingWaypoint(pMousePoint);
8838
8839 bool procede = false;
8840 if (tail) {
8841 procede = true;
8842 // if (pMousePoint == tail->GetLastPoint()) procede = false;
8843 if (m_routeState > 1 && m_pMouseRoute && tail == m_pMouseRoute)
8844 procede = false;
8845 }
8846
8847 if (procede) {
8848 int dlg_return;
8849 m_FinishRouteOnKillFocus = false;
8850 if (m_routeState == 1) { // first point in new route, preceeding
8851 // route to be added? touch case
8852
8853 wxString dmsg =
8854 _("Insert first part of this route in the new route?");
8855 if (tail->GetIndexOf(pMousePoint) ==
8856 tail->GetnPoints()) // Starting on last point of another
8857 // route?
8858 dmsg = _("Insert this route in the new route?");
8859
8860 if (tail->GetIndexOf(pMousePoint) != 1) { // Anything to do?
8861 dlg_return =
8862 OCPNMessageBox(this, dmsg, _("OpenCPN Route Create"),
8863 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8864 m_FinishRouteOnKillFocus = true;
8865
8866 if (dlg_return == wxID_YES) {
8867 inserting = true; // part of the other route will be
8868 // preceeding the new route
8869 }
8870 }
8871 } else {
8872 wxString dmsg =
8873 _("Append last part of this route to the new route?");
8874 if (tail->GetIndexOf(pMousePoint) == 1)
8875 dmsg = _(
8876 "Append this route to the new route?"); // Picking the
8877 // first point of
8878 // another route?
8879
8880 if (tail->GetLastPoint() != pMousePoint) { // Anything to do?
8881 dlg_return =
8882 OCPNMessageBox(this, dmsg, _("OpenCPN Route Create"),
8883 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
8884 m_FinishRouteOnKillFocus = true;
8885
8886 if (dlg_return == wxID_YES) {
8887 appending = true; // part of the other route will be
8888 // appended to the new route
8889 }
8890 }
8891 }
8892 }
8893
8894 // check all other routes to see if this point appears in any other
8895 // route If it appears in NO other route, then it should e
8896 // considered an isolated mark
8897 if (!FindRouteContainingWaypoint(pMousePoint))
8898 pMousePoint->SetShared(true);
8899 }
8900 }
8901
8902 if (NULL == pMousePoint) { // need a new point
8903 pMousePoint = new RoutePoint(rlat, rlon, g_default_routepoint_icon,
8904 _T(""), wxEmptyString);
8905 pMousePoint->SetNameShown(false);
8906
8907 pConfig->AddNewWayPoint(pMousePoint, -1); // use auto next num
8908 pSelect->AddSelectableRoutePoint(rlat, rlon, pMousePoint);
8909
8910 if (m_routeState > 1)
8911 undo->BeforeUndoableAction(Undo_AppendWaypoint, pMousePoint,
8912 Undo_IsOrphanded, NULL);
8913 }
8914
8915 if (m_routeState == 1) {
8916 // First point in the route.
8917 m_pMouseRoute->AddPoint(pMousePoint);
8918 } else {
8919 if (m_pMouseRoute->m_NextLegGreatCircle) {
8920 double rhumbBearing, rhumbDist, gcBearing, gcDist;
8921 DistanceBearingMercator(rlat, rlon, m_prev_rlat, m_prev_rlon,
8922 &rhumbBearing, &rhumbDist);
8923 Geodesic::GreatCircleDistBear(m_prev_rlon, m_prev_rlat, rlon, rlat,
8924 &gcDist, &gcBearing, NULL);
8925 double gcDistNM = gcDist / 1852.0;
8926
8927 // Empirically found expression to get reasonable route segments.
8928 int segmentCount = (3.0 + (rhumbDist - gcDistNM)) /
8929 pow(rhumbDist - gcDistNM - 1, 0.5);
8930
8931 wxString msg;
8932 msg << _("For this leg the Great Circle route is ")
8933 << FormatDistanceAdaptive(rhumbDist - gcDistNM)
8934 << _(" shorter than rhumbline.\n\n")
8935 << _("Would you like include the Great Circle routing points "
8936 "for this leg?");
8937
8938#ifndef __WXOSX__
8939 m_FinishRouteOnKillFocus = false;
8940 int answer = OCPNMessageBox(this, msg, _("OpenCPN Route Create"),
8941 wxYES_NO | wxNO_DEFAULT);
8942 m_FinishRouteOnKillFocus = true;
8943#else
8944 int answer = wxID_NO;
8945#endif
8946
8947 if (answer == wxID_YES) {
8948 RoutePoint *gcPoint;
8949 RoutePoint *prevGcPoint = m_prev_pMousePoint;
8950 wxRealPoint gcCoord;
8951
8952 for (int i = 1; i <= segmentCount; i++) {
8953 double fraction = (double)i * (1.0 / (double)segmentCount);
8954 Geodesic::GreatCircleTravel(m_prev_rlon, m_prev_rlat,
8955 gcDist * fraction, gcBearing,
8956 &gcCoord.x, &gcCoord.y, NULL);
8957
8958 if (i < segmentCount) {
8959 gcPoint = new RoutePoint(gcCoord.y, gcCoord.x, _T("xmblue"),
8960 _T(""), wxEmptyString);
8961 gcPoint->SetNameShown(false);
8962 pConfig->AddNewWayPoint(gcPoint, -1);
8963 pSelect->AddSelectableRoutePoint(gcCoord.y, gcCoord.x,
8964 gcPoint);
8965 } else {
8966 gcPoint = pMousePoint; // Last point, previously exsisting!
8967 }
8968
8969 m_pMouseRoute->AddPoint(gcPoint);
8970 pSelect->AddSelectableRouteSegment(
8971 prevGcPoint->m_lat, prevGcPoint->m_lon, gcPoint->m_lat,
8972 gcPoint->m_lon, prevGcPoint, gcPoint, m_pMouseRoute);
8973 prevGcPoint = gcPoint;
8974 }
8975
8976 undo->CancelUndoableAction(true);
8977
8978 } else {
8979 m_pMouseRoute->AddPoint(pMousePoint);
8980 pSelect->AddSelectableRouteSegment(m_prev_rlat, m_prev_rlon, rlat,
8981 rlon, m_prev_pMousePoint,
8982 pMousePoint, m_pMouseRoute);
8983 undo->AfterUndoableAction(m_pMouseRoute);
8984 }
8985 } else {
8986 // Ordinary rhumblinesegment.
8987 m_pMouseRoute->AddPoint(pMousePoint);
8988 pSelect->AddSelectableRouteSegment(m_prev_rlat, m_prev_rlon, rlat,
8989 rlon, m_prev_pMousePoint,
8990 pMousePoint, m_pMouseRoute);
8991 undo->AfterUndoableAction(m_pMouseRoute);
8992 }
8993 }
8994
8995 m_prev_rlat = rlat;
8996 m_prev_rlon = rlon;
8997 m_prev_pMousePoint = pMousePoint;
8998 m_pMouseRoute->m_lastMousePointIndex = m_pMouseRoute->GetnPoints();
8999
9000 m_routeState++;
9001
9002 if (appending ||
9003 inserting) { // Appending a route or making a new route
9004 int connect = tail->GetIndexOf(pMousePoint);
9005 if (connect == 1) {
9006 inserting = false; // there is nothing to insert
9007 appending = true; // so append
9008 }
9009 int length = tail->GetnPoints();
9010
9011 int i;
9012 int start, stop;
9013 if (appending) {
9014 start = connect + 1;
9015 stop = length;
9016 } else { // inserting
9017 start = 1;
9018 stop = connect;
9019 m_pMouseRoute->RemovePoint(
9020 m_pMouseRoute
9021 ->GetLastPoint()); // Remove the first and only point
9022 }
9023 for (i = start; i <= stop; i++) {
9024 m_pMouseRoute->AddPointAndSegment(tail->GetPoint(i), false);
9025 if (m_pMouseRoute)
9026 m_pMouseRoute->m_lastMousePointIndex =
9027 m_pMouseRoute->GetnPoints();
9028 m_routeState++;
9029 gFrame->RefreshAllCanvas();
9030 ret = true;
9031 }
9032 m_prev_rlat =
9033 m_pMouseRoute->GetPoint(m_pMouseRoute->GetnPoints())->m_lat;
9034 m_prev_rlon =
9035 m_pMouseRoute->GetPoint(m_pMouseRoute->GetnPoints())->m_lon;
9036 m_pMouseRoute->FinalizeForRendering();
9037 }
9038
9039 Refresh(true);
9040 ret = true;
9041 } else if (m_bMeasure_Active && m_nMeasureState) // measure tool?
9042 {
9043 if (m_bedge_pan) {
9044 m_bedge_pan = false;
9045 return false;
9046 }
9047
9048 if (m_nMeasureState == 1) {
9049 m_pMeasureRoute = new Route();
9050 pRouteList->Append(m_pMeasureRoute);
9051 r_rband.x = x;
9052 r_rband.y = y;
9053 }
9054
9055 if (m_pMeasureRoute) {
9056 RoutePoint *pMousePoint = new RoutePoint(
9057 m_cursor_lat, m_cursor_lon, wxString(_T ( "circle" )),
9058 wxEmptyString, wxEmptyString);
9059 pMousePoint->m_bShowName = false;
9060
9061 m_pMeasureRoute->AddPoint(pMousePoint);
9062
9063 m_prev_rlat = m_cursor_lat;
9064 m_prev_rlon = m_cursor_lon;
9065 m_prev_pMousePoint = pMousePoint;
9066 m_pMeasureRoute->m_lastMousePointIndex =
9067 m_pMeasureRoute->GetnPoints();
9068
9069 m_nMeasureState++;
9070 } else {
9071 CancelMeasureRoute();
9072 }
9073
9074 Refresh(true);
9075 ret = true;
9076 } else {
9077 bool bSelectAllowed = true;
9078 if (NULL == g_pMarkInfoDialog) {
9079 if (g_bWayPointPreventDragging) bSelectAllowed = false;
9080 } else if (!g_pMarkInfoDialog->IsShown() && g_bWayPointPreventDragging)
9081 bSelectAllowed = false;
9082
9083 /*if this left up happens at the end of a route point dragging and if
9084 the cursor/thumb is on the draghandle icon, not on the point iself a new
9085 selection will select nothing and the drag will never be ended, so the
9086 legs around this point never selectable. At this step we don't need a
9087 new selection, just keep the previoulsly selected and dragged point */
9088 if (m_bRoutePoinDragging) bSelectAllowed = false;
9089
9090 if (bSelectAllowed) {
9091 bool b_was_editing_mark = m_bMarkEditing;
9092 bool b_was_editing_route = m_bRouteEditing;
9093 FindRoutePointsAtCursor(SelectRadius,
9094 true); // Possibly selecting a point in a
9095 // route for later dragging
9096
9097 /*route and a mark points in layer can't be dragged so should't be
9098 * selected and no draghandle icon*/
9099 if (m_pRoutePointEditTarget && m_pRoutePointEditTarget->m_bIsInLayer)
9100 m_pRoutePointEditTarget = NULL;
9101
9102 if (!b_was_editing_route) {
9103 if (m_pEditRouteArray) {
9104 b_startedit_route = true;
9105
9106 // Hide the track and route rollover during route point edit, not
9107 // needed, and may be confusing
9108 if (m_pTrackRolloverWin && m_pTrackRolloverWin->IsActive()) {
9109 m_pTrackRolloverWin->IsActive(false);
9110 }
9111 if (m_pRouteRolloverWin && m_pRouteRolloverWin->IsActive()) {
9112 m_pRouteRolloverWin->IsActive(false);
9113 }
9114
9115 wxRect pre_rect;
9116 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9117 ir++) {
9118 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9119 // Need to validate route pointer
9120 // Route may be gone due to drgging close to ownship with
9121 // "Delete On Arrival" state set, as in the case of
9122 // navigating to an isolated waypoint on a temporary route
9123 if (g_pRouteMan->IsRouteValid(pr)) {
9124 // pr->SetHiLite(50);
9125 wxRect route_rect;
9126 RouteGui(*pr).CalculateDCRect(m_dc_route, this, &route_rect);
9127 pre_rect.Union(route_rect);
9128 }
9129 }
9130 RefreshRect(pre_rect, true);
9131 }
9132 } else {
9133 b_startedit_route = false;
9134 }
9135
9136 // Mark editing
9137 if (m_pRoutePointEditTarget) {
9138 if (b_was_editing_mark ||
9139 b_was_editing_route) { // kill previous hilight
9140 if (m_lastRoutePointEditTarget) {
9141 m_lastRoutePointEditTarget->m_bRPIsBeingEdited = false;
9142 m_lastRoutePointEditTarget->m_bPtIsSelected = false;
9143 RoutePointGui(*m_lastRoutePointEditTarget)
9144 .EnableDragHandle(false);
9145 pSelect->DeleteSelectablePoint(m_lastRoutePointEditTarget,
9146 SELTYPE_DRAGHANDLE);
9147 }
9148 }
9149
9150 if (m_pRoutePointEditTarget) {
9151 m_pRoutePointEditTarget->m_bRPIsBeingEdited = true;
9152 m_pRoutePointEditTarget->m_bPtIsSelected = true;
9153 RoutePointGui(*m_pRoutePointEditTarget).EnableDragHandle(true);
9154 wxPoint2DDouble dragHandlePoint =
9155 RoutePointGui(*m_pRoutePointEditTarget)
9156 .GetDragHandlePoint(this);
9157 pSelect->AddSelectablePoint(
9158 dragHandlePoint.m_y, dragHandlePoint.m_x,
9159 m_pRoutePointEditTarget, SELTYPE_DRAGHANDLE);
9160 }
9161 } else { // Deselect everything
9162 if (m_lastRoutePointEditTarget) {
9163 m_lastRoutePointEditTarget->m_bRPIsBeingEdited = false;
9164 m_lastRoutePointEditTarget->m_bPtIsSelected = false;
9165 RoutePointGui(*m_lastRoutePointEditTarget)
9166 .EnableDragHandle(false);
9167 pSelect->DeleteSelectablePoint(m_lastRoutePointEditTarget,
9168 SELTYPE_DRAGHANDLE);
9169
9170 // Clear any routes being edited, probably orphans
9171 wxArrayPtrVoid *lastEditRouteArray =
9172 g_pRouteMan->GetRouteArrayContaining(
9173 m_lastRoutePointEditTarget);
9174 if (lastEditRouteArray) {
9175 for (unsigned int ir = 0; ir < lastEditRouteArray->GetCount();
9176 ir++) {
9177 Route *pr = (Route *)lastEditRouteArray->Item(ir);
9178 if (g_pRouteMan->IsRouteValid(pr)) {
9179 pr->m_bIsBeingEdited = false;
9180 }
9181 }
9182 }
9183 }
9184 }
9185
9186 // Do the refresh
9187
9188 if (g_bopengl) {
9189 InvalidateGL();
9190 Refresh(false);
9191 } else {
9192 if (m_lastRoutePointEditTarget) {
9193 wxRect wp_rect;
9194 RoutePointGui(*m_lastRoutePointEditTarget)
9195 .CalculateDCRect(m_dc_route, this, &wp_rect);
9196 RefreshRect(wp_rect, true);
9197 }
9198
9199 if (m_pRoutePointEditTarget) {
9200 wxRect wp_rect;
9201 RoutePointGui(*m_pRoutePointEditTarget)
9202 .CalculateDCRect(m_dc_route, this, &wp_rect);
9203 RefreshRect(wp_rect, true);
9204 }
9205 }
9206 }
9207 } // bSelectAllowed
9208
9209 // Check to see if there is a route or AIS target under the cursor
9210 // If so, start the rollover timer which creates the popup
9211 SelectCtx ctx(m_bShowNavobjects, GetCanvasTrueScale(), GetScaleValue());
9212 bool b_start_rollover = false;
9213 if (g_pAIS && g_pAIS->GetNumTargets() && m_bShowAIS) {
9214 SelectItem *pFind = pSelectAIS->FindSelection(
9215 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_AISTARGET);
9216 if (pFind) b_start_rollover = true;
9217 }
9218
9219 if (!b_start_rollover && !b_startedit_route) {
9220 SelectableItemList SelList = pSelect->FindSelectionList(
9221 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_ROUTESEGMENT);
9222 wxSelectableItemListNode *node = SelList.GetFirst();
9223 while (node) {
9224 SelectItem *pFindSel = node->GetData();
9225
9226 Route *pr = (Route *)pFindSel->m_pData3; // candidate
9227
9228 if (pr && pr->IsVisible()) {
9229 b_start_rollover = true;
9230 break;
9231 }
9232 node = node->GetNext();
9233 } // while
9234 }
9235
9236 if (!b_start_rollover && !b_startedit_route) {
9237 SelectableItemList SelList = pSelect->FindSelectionList(
9238 ctx, m_cursor_lat, m_cursor_lon, SELTYPE_TRACKSEGMENT);
9239 wxSelectableItemListNode *node = SelList.GetFirst();
9240 while (node) {
9241 SelectItem *pFindSel = node->GetData();
9242
9243 Track *tr = (Track *)pFindSel->m_pData3; // candidate
9244
9245 if (tr && tr->IsVisible()) {
9246 b_start_rollover = true;
9247 break;
9248 }
9249 node = node->GetNext();
9250 } // while
9251 }
9252
9253 if (b_start_rollover)
9254 m_RolloverPopupTimer.Start(m_rollover_popup_timer_msec,
9255 wxTIMER_ONE_SHOT);
9256 Route *tail = 0;
9257 Route *current = 0;
9258 bool appending = false;
9259 bool inserting = false;
9260 int connect = 0;
9261 if (m_bRouteEditing /* && !b_startedit_route*/) { // End of RoutePoint
9262 // drag
9263 if (m_pRoutePointEditTarget) {
9264 // Check to see if there is a nearby point which may replace the
9265 // dragged one
9266 RoutePoint *pMousePoint = NULL;
9267
9268 int index_last;
9269 if (m_bRoutePoinDragging && !m_pRoutePointEditTarget->m_bIsActive) {
9270 double nearby_radius_meters =
9271 g_Platform->GetSelectRadiusPix() / m_true_scale_ppm;
9272 RoutePoint *pNearbyPoint = pWayPointMan->GetOtherNearbyWaypoint(
9273 m_pRoutePointEditTarget->m_lat, m_pRoutePointEditTarget->m_lon,
9274 nearby_radius_meters, m_pRoutePointEditTarget->m_GUID);
9275 if (pNearbyPoint && !pNearbyPoint->m_bIsInLayer &&
9276 pWayPointMan->IsReallyVisible(pNearbyPoint)) {
9277 bool duplicate =
9278 false; // ensure we won't create duplicate point in routes
9279 if (m_pEditRouteArray && !pNearbyPoint->m_bIsolatedMark) {
9280 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9281 ir++) {
9282 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9283 if (pr && pr->pRoutePointList) {
9284 if (pr->pRoutePointList->IndexOf(pNearbyPoint) !=
9285 wxNOT_FOUND) {
9286 duplicate = true;
9287 break;
9288 }
9289 }
9290 }
9291 }
9292
9293 // Special case:
9294 // Allow "re-use" of a route's waypoints iff it is a simple
9295 // isolated route. This allows, for instance, creation of a closed
9296 // polygon route
9297 if (m_pEditRouteArray->GetCount() == 1) duplicate = false;
9298
9299 if (!duplicate) {
9300 int dlg_return;
9301 dlg_return =
9302 OCPNMessageBox(this,
9303 _("Replace this RoutePoint by the nearby "
9304 "Waypoint?"),
9305 _("OpenCPN RoutePoint change"),
9306 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9307 if (dlg_return == wxID_YES) {
9308 /*double confirmation if the dragged point has been manually
9309 * created which can be important and could be deleted
9310 * unintentionally*/
9311
9312 tail = g_pRouteMan->FindVisibleRouteContainingWaypoint(
9313 pNearbyPoint);
9314 current =
9315 FindRouteContainingWaypoint(m_pRoutePointEditTarget);
9316
9317 if (tail && current && (tail != current)) {
9318 int dlg_return1;
9319 connect = tail->GetIndexOf(pNearbyPoint);
9320 int index_current_route =
9321 current->GetIndexOf(m_pRoutePointEditTarget);
9322 index_last = current->GetIndexOf(current->GetLastPoint());
9323 dlg_return1 = wxID_NO;
9324 if (index_last ==
9325 index_current_route) { // we are dragging the last
9326 // point of the route
9327 if (connect != tail->GetnPoints()) { // anything to do?
9328
9329 wxString dmsg(
9330 _("Last part of route to be appended to dragged "
9331 "route?"));
9332 if (connect == 1)
9333 dmsg =
9334 _("Full route to be appended to dragged route?");
9335
9336 dlg_return1 = OCPNMessageBox(
9337 this, dmsg, _("OpenCPN Route Create"),
9338 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9339 if (dlg_return1 == wxID_YES) {
9340 appending = true;
9341 }
9342 }
9343 } else if (index_current_route ==
9344 1) { // dragging the first point of the route
9345 if (connect != 1) { // anything to do?
9346
9347 wxString dmsg(
9348 _("First part of route to be inserted into dragged "
9349 "route?"));
9350 if (connect == tail->GetnPoints())
9351 dmsg = _(
9352 "Full route to be inserted into dragged route?");
9353
9354 dlg_return1 = OCPNMessageBox(
9355 this, dmsg, _("OpenCPN Route Create"),
9356 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9357 if (dlg_return1 == wxID_YES) {
9358 inserting = true;
9359 }
9360 }
9361 }
9362 }
9363
9364 if (m_pRoutePointEditTarget->IsShared()) {
9365 // dlg_return = wxID_NO;
9366 dlg_return = OCPNMessageBox(
9367 this,
9368 _("Do you really want to delete and replace this "
9369 "WayPoint") +
9370 _T("\n") + _("which has been created manually?"),
9371 ("OpenCPN RoutePoint warning"),
9372 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9373 }
9374 }
9375 if (dlg_return == wxID_YES) {
9376 pMousePoint = pNearbyPoint;
9377 if (pMousePoint->m_bIsolatedMark) {
9378 pMousePoint->SetShared(true);
9379 }
9380 pMousePoint->m_bIsolatedMark =
9381 false; // definitely no longer isolated
9382 pMousePoint->m_bIsInRoute = true;
9383 }
9384 }
9385 }
9386 }
9387 if (!pMousePoint)
9388 pSelect->UpdateSelectableRouteSegments(m_pRoutePointEditTarget);
9389
9390 if (m_pEditRouteArray) {
9391 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9392 ir++) {
9393 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9394 if (g_pRouteMan->IsRouteValid(pr)) {
9395 if (pMousePoint) { // remove the dragged point and insert the
9396 // nearby
9397 int nRP =
9398 pr->pRoutePointList->IndexOf(m_pRoutePointEditTarget);
9399
9400 pSelect->DeleteAllSelectableRoutePoints(pr);
9401 pSelect->DeleteAllSelectableRouteSegments(pr);
9402
9403 pr->pRoutePointList->Insert(nRP, pMousePoint);
9404 pr->pRoutePointList->DeleteObject(m_pRoutePointEditTarget);
9405
9406 pSelect->AddAllSelectableRouteSegments(pr);
9407 pSelect->AddAllSelectableRoutePoints(pr);
9408 }
9409 pr->FinalizeForRendering();
9410 pr->UpdateSegmentDistances();
9411 if (m_bRoutePoinDragging) pConfig->UpdateRoute(pr);
9412 }
9413 }
9414 }
9415
9416 // Update the RouteProperties Dialog, if currently shown
9417 if (pRoutePropDialog && pRoutePropDialog->IsShown()) {
9418 if (m_pEditRouteArray) {
9419 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9420 ir++) {
9421 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9422 if (g_pRouteMan->IsRouteValid(pr)) {
9423 if (pRoutePropDialog->GetRoute() == pr) {
9424 pRoutePropDialog->SetRouteAndUpdate(pr, true);
9425 }
9426 /* cannot edit track points anyway
9427 else if ( ( NULL !=
9428 pTrackPropDialog ) && ( pTrackPropDialog->IsShown() ) &&
9429 pTrackPropDialog->m_pTrack == pr ) {
9430 pTrackPropDialog->SetTrackAndUpdate(
9431 pr );
9432 }
9433 */
9434 }
9435 }
9436 }
9437 }
9438 if (pMousePoint) { // clear all about the dragged point
9439 pConfig->DeleteWayPoint(m_pRoutePointEditTarget);
9440 pWayPointMan->RemoveRoutePoint(m_pRoutePointEditTarget);
9441 // Hide mark properties dialog if open on the replaced point
9442 if ((NULL != g_pMarkInfoDialog) && (g_pMarkInfoDialog->IsShown()))
9443 if (m_pRoutePointEditTarget == g_pMarkInfoDialog->GetRoutePoint())
9444 g_pMarkInfoDialog->Hide();
9445
9446 delete m_pRoutePointEditTarget;
9447 m_lastRoutePointEditTarget = NULL;
9448 m_pRoutePointEditTarget = NULL;
9449 undo->AfterUndoableAction(pMousePoint);
9450 undo->InvalidateUndo();
9451 }
9452 }
9453 }
9454
9455 else if (m_bMarkEditing) { // End of way point drag
9456 if (m_pRoutePointEditTarget)
9457 if (m_bRoutePoinDragging)
9458 pConfig->UpdateWayPoint(m_pRoutePointEditTarget);
9459 }
9460
9461 if (m_pRoutePointEditTarget)
9462 undo->AfterUndoableAction(m_pRoutePointEditTarget);
9463
9464 if (!m_pRoutePointEditTarget) {
9465 delete m_pEditRouteArray;
9466 m_pEditRouteArray = NULL;
9467 m_bRouteEditing = false;
9468 }
9469 m_bRoutePoinDragging = false;
9470
9471 if (appending) { // Appending to the route of which the last point is
9472 // dragged onto another route
9473
9474 // copy tail from connect until length to end of current after dragging
9475
9476 int length = tail->GetnPoints();
9477 for (int i = connect + 1; i <= length; i++) {
9478 current->AddPointAndSegment(tail->GetPoint(i), false);
9479 if (current) current->m_lastMousePointIndex = current->GetnPoints();
9480 m_routeState++;
9481 gFrame->RefreshAllCanvas();
9482 ret = true;
9483 }
9484 current->FinalizeForRendering();
9485 current->m_bIsBeingEdited = false;
9486 FinishRoute();
9487 g_pRouteMan->DeleteRoute(tail, NavObjectChanges::getInstance());
9488 }
9489 if (inserting) {
9490 pSelect->DeleteAllSelectableRoutePoints(current);
9491 pSelect->DeleteAllSelectableRouteSegments(current);
9492 for (int i = 1; i < connect; i++) { // numbering in the tail route
9493 current->InsertPointAndSegment(tail->GetPoint(i), i - 1, false);
9494 }
9495 pSelect->AddAllSelectableRouteSegments(current);
9496 pSelect->AddAllSelectableRoutePoints(current);
9497 current->FinalizeForRendering();
9498 current->m_bIsBeingEdited = false;
9499 g_pRouteMan->DeleteRoute(tail, NavObjectChanges::getInstance());
9500 }
9501
9502 // Update the RouteProperties Dialog, if currently shown
9503 if (pRoutePropDialog && pRoutePropDialog->IsShown()) {
9504 if (m_pEditRouteArray) {
9505 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount(); ir++) {
9506 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9507 if (g_pRouteMan->IsRouteValid(pr)) {
9508 if (pRoutePropDialog->GetRoute() == pr) {
9509 pRoutePropDialog->SetRouteAndUpdate(pr, true);
9510 }
9511 }
9512 }
9513 }
9514 }
9515
9516 } // g_btouch
9517
9518 else { // !g_btouch
9519 if (m_bRouteEditing) { // End of RoutePoint drag
9520 Route *tail = 0;
9521 Route *current = 0;
9522 bool appending = false;
9523 bool inserting = false;
9524 int connect = 0;
9525 int index_last;
9526 if (m_pRoutePointEditTarget) {
9527 m_pRoutePointEditTarget->m_bBlink = false;
9528 // Check to see if there is a nearby point which may replace the
9529 // dragged one
9530 RoutePoint *pMousePoint = NULL;
9531 if (m_bRoutePoinDragging && !m_pRoutePointEditTarget->m_bIsActive) {
9532 double nearby_radius_meters =
9533 g_Platform->GetSelectRadiusPix() / m_true_scale_ppm;
9534 RoutePoint *pNearbyPoint = pWayPointMan->GetOtherNearbyWaypoint(
9535 m_pRoutePointEditTarget->m_lat, m_pRoutePointEditTarget->m_lon,
9536 nearby_radius_meters, m_pRoutePointEditTarget->m_GUID);
9537 if (pNearbyPoint && !pNearbyPoint->m_bIsInLayer &&
9538 pWayPointMan->IsReallyVisible(pNearbyPoint)) {
9539 bool duplicate = false; // don't create duplicate point in routes
9540 if (m_pEditRouteArray && !pNearbyPoint->m_bIsolatedMark) {
9541 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9542 ir++) {
9543 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9544 if (pr && pr->pRoutePointList) {
9545 if (pr->pRoutePointList->IndexOf(pNearbyPoint) !=
9546 wxNOT_FOUND) {
9547 duplicate = true;
9548 break;
9549 }
9550 }
9551 }
9552 }
9553
9554 // Special case:
9555 // Allow "re-use" of a route's waypoints iff it is a simple
9556 // isolated route. This allows, for instance, creation of a closed
9557 // polygon route
9558 if (m_pEditRouteArray->GetCount() == 1) duplicate = false;
9559
9560 if (!duplicate) {
9561 int dlg_return;
9562 dlg_return =
9563 OCPNMessageBox(this,
9564 _("Replace this RoutePoint by the nearby "
9565 "Waypoint?"),
9566 _("OpenCPN RoutePoint change"),
9567 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9568 if (dlg_return == wxID_YES) {
9569 /*double confirmation if the dragged point has been manually
9570 * created which can be important and could be deleted
9571 * unintentionally*/
9572 tail = g_pRouteMan->FindVisibleRouteContainingWaypoint(
9573 pNearbyPoint);
9574 current =
9575 FindRouteContainingWaypoint(m_pRoutePointEditTarget);
9576
9577 if (tail && current && (tail != current)) {
9578 int dlg_return1;
9579 connect = tail->GetIndexOf(pNearbyPoint);
9580 int index_current_route =
9581 current->GetIndexOf(m_pRoutePointEditTarget);
9582 index_last = current->GetIndexOf(current->GetLastPoint());
9583 dlg_return1 = wxID_NO;
9584 if (index_last ==
9585 index_current_route) { // we are dragging the last
9586 // point of the route
9587 if (connect != tail->GetnPoints()) { // anything to do?
9588
9589 wxString dmsg(
9590 _("Last part of route to be appended to dragged "
9591 "route?"));
9592 if (connect == 1)
9593 dmsg =
9594 _("Full route to be appended to dragged route?");
9595
9596 dlg_return1 = OCPNMessageBox(
9597 this, dmsg, _("OpenCPN Route Create"),
9598 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9599 if (dlg_return1 == wxID_YES) {
9600 appending = true;
9601 }
9602 }
9603 } else if (index_current_route ==
9604 1) { // dragging the first point of the route
9605 if (connect != 1) { // anything to do?
9606
9607 wxString dmsg(
9608 _("First part of route to be inserted into dragged "
9609 "route?"));
9610 if (connect == tail->GetnPoints())
9611 dmsg = _(
9612 "Full route to be inserted into dragged route?");
9613
9614 dlg_return1 = OCPNMessageBox(
9615 this, dmsg, _("OpenCPN Route Create"),
9616 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9617 if (dlg_return1 == wxID_YES) {
9618 inserting = true;
9619 }
9620 }
9621 }
9622 }
9623
9624 if (m_pRoutePointEditTarget->IsShared()) {
9625 dlg_return = wxID_NO;
9626 dlg_return = OCPNMessageBox(
9627 this,
9628 _("Do you really want to delete and replace this "
9629 "WayPoint") +
9630 _T("\n") + _("which has been created manually?"),
9631 ("OpenCPN RoutePoint warning"),
9632 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
9633 }
9634 }
9635 if (dlg_return == wxID_YES) {
9636 pMousePoint = pNearbyPoint;
9637 if (pMousePoint->m_bIsolatedMark) {
9638 pMousePoint->SetShared(true);
9639 }
9640 pMousePoint->m_bIsolatedMark =
9641 false; // definitely no longer isolated
9642 pMousePoint->m_bIsInRoute = true;
9643 }
9644 }
9645 }
9646 }
9647 if (!pMousePoint)
9648 pSelect->UpdateSelectableRouteSegments(m_pRoutePointEditTarget);
9649
9650 if (m_pEditRouteArray) {
9651 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9652 ir++) {
9653 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9654 if (g_pRouteMan->IsRouteValid(pr)) {
9655 if (pMousePoint) { // replace dragged point by nearby one
9656 int nRP =
9657 pr->pRoutePointList->IndexOf(m_pRoutePointEditTarget);
9658
9659 pSelect->DeleteAllSelectableRoutePoints(pr);
9660 pSelect->DeleteAllSelectableRouteSegments(pr);
9661
9662 pr->pRoutePointList->Insert(nRP, pMousePoint);
9663 pr->pRoutePointList->DeleteObject(m_pRoutePointEditTarget);
9664
9665 pSelect->AddAllSelectableRouteSegments(pr);
9666 pSelect->AddAllSelectableRoutePoints(pr);
9667 }
9668 pr->FinalizeForRendering();
9669 pr->UpdateSegmentDistances();
9670 pr->m_bIsBeingEdited = false;
9671
9672 if (m_bRoutePoinDragging) pConfig->UpdateRoute(pr);
9673
9674 pr->SetHiLite(0);
9675 }
9676 }
9677 Refresh(false);
9678 }
9679
9680 if (appending) {
9681 // copy tail from connect until length to end of current after
9682 // dragging
9683
9684 int length = tail->GetnPoints();
9685 for (int i = connect + 1; i <= length; i++) {
9686 current->AddPointAndSegment(tail->GetPoint(i), false);
9687 if (current)
9688 current->m_lastMousePointIndex = current->GetnPoints();
9689 m_routeState++;
9690 gFrame->RefreshAllCanvas();
9691 ret = true;
9692 }
9693 current->FinalizeForRendering();
9694 current->m_bIsBeingEdited = false;
9695 FinishRoute();
9696 g_pRouteMan->DeleteRoute(tail, NavObjectChanges::getInstance());
9697 }
9698 if (inserting) {
9699 pSelect->DeleteAllSelectableRoutePoints(current);
9700 pSelect->DeleteAllSelectableRouteSegments(current);
9701 for (int i = 1; i < connect; i++) { // numbering in the tail route
9702 current->InsertPointAndSegment(tail->GetPoint(i), i - 1, false);
9703 }
9704 pSelect->AddAllSelectableRouteSegments(current);
9705 pSelect->AddAllSelectableRoutePoints(current);
9706 current->FinalizeForRendering();
9707 current->m_bIsBeingEdited = false;
9708 g_pRouteMan->DeleteRoute(tail, NavObjectChanges::getInstance());
9709 }
9710
9711 // Update the RouteProperties Dialog, if currently shown
9712 if (pRoutePropDialog && pRoutePropDialog->IsShown()) {
9713 if (m_pEditRouteArray) {
9714 for (unsigned int ir = 0; ir < m_pEditRouteArray->GetCount();
9715 ir++) {
9716 Route *pr = (Route *)m_pEditRouteArray->Item(ir);
9717 if (g_pRouteMan->IsRouteValid(pr)) {
9718 if (pRoutePropDialog->GetRoute() == pr) {
9719 pRoutePropDialog->SetRouteAndUpdate(pr, true);
9720 }
9721 }
9722 }
9723 }
9724 }
9725
9726 if (pMousePoint) {
9727 pConfig->DeleteWayPoint(m_pRoutePointEditTarget);
9728 pWayPointMan->RemoveRoutePoint(m_pRoutePointEditTarget);
9729 // Hide mark properties dialog if open on the replaced point
9730 if ((NULL != g_pMarkInfoDialog) && (g_pMarkInfoDialog->IsShown()))
9731 if (m_pRoutePointEditTarget == g_pMarkInfoDialog->GetRoutePoint())
9732 g_pMarkInfoDialog->Hide();
9733
9734 delete m_pRoutePointEditTarget;
9735 m_lastRoutePointEditTarget = NULL;
9736 undo->AfterUndoableAction(pMousePoint);
9737 undo->InvalidateUndo();
9738 } else {
9739 m_pRoutePointEditTarget->m_bPtIsSelected = false;
9740 m_pRoutePointEditTarget->m_bRPIsBeingEdited = false;
9741
9742 undo->AfterUndoableAction(m_pRoutePointEditTarget);
9743 }
9744
9745 delete m_pEditRouteArray;
9746 m_pEditRouteArray = NULL;
9747 }
9748
9749 InvalidateGL();
9750 m_bRouteEditing = false;
9751 m_pRoutePointEditTarget = NULL;
9752
9753 // if (m_toolBar && !m_toolBar->IsToolbarShown()) SurfaceToolbar();
9754 ret = true;
9755 }
9756
9757 else if (m_bMarkEditing) { // end of Waypoint drag
9758 if (m_pRoutePointEditTarget) {
9759 if (m_bRoutePoinDragging)
9760 pConfig->UpdateWayPoint(m_pRoutePointEditTarget);
9761 undo->AfterUndoableAction(m_pRoutePointEditTarget);
9762 m_pRoutePointEditTarget->m_bRPIsBeingEdited = false;
9763 if (!g_bopengl) {
9764 wxRect wp_rect;
9765 RoutePointGui(*m_pRoutePointEditTarget)
9766 .CalculateDCRect(m_dc_route, this, &wp_rect);
9767 m_pRoutePointEditTarget->m_bPtIsSelected = false;
9768 RefreshRect(wp_rect, true);
9769 }
9770 }
9771 m_pRoutePointEditTarget = NULL;
9772 m_bMarkEditing = false;
9773 // if (m_toolBar && !m_toolBar->IsToolbarShown()) SurfaceToolbar();
9774 ret = true;
9775 }
9776
9777 else if (leftIsDown) { // left click for chart center
9778 leftIsDown = false;
9779 ret = false;
9780
9781 if (!g_btouch) {
9782 if (!m_bChartDragging && !m_bMeasure_Active) {
9783 } else {
9784 m_bChartDragging = false;
9785 }
9786 }
9787 }
9788 m_bRoutePoinDragging = false;
9789 } // !btouch
9790
9791 if (ret) return true;
9792 } // left up
9793
9794 if (event.RightDown()) {
9795 SetFocus(); // This is to let a plugin know which canvas is right-clicked
9796 last_drag.x = mx;
9797 last_drag.y = my;
9798
9799 if (g_btouch) {
9800 // if( m_pRoutePointEditTarget )
9801 // return false;
9802 }
9803
9804 ret = true;
9805 m_FinishRouteOnKillFocus = false;
9806 CallPopupMenu(mx, my);
9807 m_FinishRouteOnKillFocus = true;
9808 } // Right down
9809
9810 return ret;
9811}
9812
9813bool panleftIsDown;
9814
9815bool ChartCanvas::MouseEventProcessCanvas(wxMouseEvent &event) {
9816 // Skip all mouse processing if shift is held.
9817 // This allows plugins to implement shift+drag behaviors.
9818 if (event.ShiftDown()) {
9819 return false;
9820 }
9821 int x, y;
9822 event.GetPosition(&x, &y);
9823
9824 x *= m_displayScale;
9825 y *= m_displayScale;
9826
9827 // Check for wheel rotation
9828 // ideally, should be just longer than the time between
9829 // processing accumulated mouse events from the event queue
9830 // as would happen during screen redraws.
9831 int wheel_dir = event.GetWheelRotation();
9832
9833 if (wheel_dir) {
9834 int mouse_wheel_oneshot = abs(wheel_dir) * 4; // msec
9835 wheel_dir = wheel_dir > 0 ? 1 : -1; // normalize
9836
9837 double factor = g_mouse_zoom_sensitivity;
9838 if (wheel_dir < 0) factor = 1 / factor;
9839
9840 if (g_bsmoothpanzoom) {
9841 if ((m_wheelstopwatch.Time() < m_wheelzoom_stop_oneshot)) {
9842 if (wheel_dir == m_last_wheel_dir) {
9843 m_wheelzoom_stop_oneshot += mouse_wheel_oneshot;
9844 // m_zoom_target /= factor;
9845 } else
9846 StopMovement();
9847 } else {
9848 m_wheelzoom_stop_oneshot = mouse_wheel_oneshot;
9849 m_wheelstopwatch.Start(0);
9850 // m_zoom_target = VPoint.chart_scale / factor;
9851 }
9852 }
9853
9854 m_last_wheel_dir = wheel_dir;
9855
9856 ZoomCanvas(factor, true, false);
9857 }
9858
9859 if (event.LeftDown()) {
9860 // Skip the first left click if it will cause a canvas focus shift
9861 if ((GetCanvasCount() > 1) && (this != g_focusCanvas)) {
9862 // printf("focus shift\n");
9863 return false;
9864 }
9865
9866 last_drag.x = x, last_drag.y = y;
9867 panleftIsDown = true;
9868 }
9869
9870 if (event.LeftUp()) {
9871 if (panleftIsDown) { // leftUp for chart center, but only after a leftDown
9872 // seen here.
9873 panleftIsDown = false;
9874
9875 if (!g_btouch) {
9876 if (!m_bChartDragging && !m_bMeasure_Active) {
9877 switch (cursor_region) {
9878 case MID_RIGHT: {
9879 PanCanvas(100, 0);
9880 break;
9881 }
9882
9883 case MID_LEFT: {
9884 PanCanvas(-100, 0);
9885 break;
9886 }
9887
9888 case MID_TOP: {
9889 PanCanvas(0, 100);
9890 break;
9891 }
9892
9893 case MID_BOT: {
9894 PanCanvas(0, -100);
9895 break;
9896 }
9897
9898 case CENTER: {
9899 PanCanvas(x - GetVP().pix_width / 2, y - GetVP().pix_height / 2);
9900 break;
9901 }
9902 }
9903 } else {
9904 m_bChartDragging = false;
9905 }
9906 }
9907 }
9908 }
9909
9910 if (event.Dragging() && event.LeftIsDown()) {
9911 /*
9912 * fixed dragging.
9913 * On my Surface Pro 3 running Arch Linux there is no mouse down event
9914 * before the drag event. Hence, as there is no mouse down event, last_drag
9915 * is not reset before the drag. And that results in one single drag
9916 * session, meaning you cannot drag the map a few miles north, lift your
9917 * finger, and the go even further north. Instead, the map resets itself
9918 * always to the very first drag start (since there is not reset of
9919 * last_drag).
9920 *
9921 * Besides, should not left down and dragging be enough of a situation to
9922 * start a drag procedure?
9923 *
9924 * Anyways, guarded it to be active in touch situations only.
9925 */
9926
9927 if (g_btouch) {
9928 struct timespec now;
9929 clock_gettime(CLOCK_MONOTONIC, &now);
9930 uint64_t tnow = (1e9 * now.tv_sec) + now.tv_nsec;
9931
9932 if (false == m_bChartDragging) {
9933 // Reset drag calculation members
9934 last_drag.x = x, last_drag.y = y;
9935 m_bChartDragging = true;
9936 m_chart_drag_total_time = 0;
9937 m_chart_drag_total_x = 0;
9938 m_chart_drag_total_y = 0;
9939 m_inertia_last_drag_x = x;
9940 m_inertia_last_drag_y = y;
9941 m_drag_vec_x.clear();
9942 m_drag_vec_y.clear();
9943 m_drag_vec_t.clear();
9944 m_last_drag_time = tnow;
9945 }
9946
9947 // Calculate and store drag dynamics.
9948 uint64_t delta_t = tnow - m_last_drag_time;
9949 double delta_tf = delta_t / 1e9;
9950
9951 m_chart_drag_total_time += delta_tf;
9952 m_chart_drag_total_x += m_inertia_last_drag_x - x;
9953 m_chart_drag_total_y += m_inertia_last_drag_y - y;
9954
9955 m_drag_vec_x.push_back(m_inertia_last_drag_x - x);
9956 m_drag_vec_y.push_back(m_inertia_last_drag_y - y);
9957 m_drag_vec_t.push_back(delta_tf);
9958
9959 m_inertia_last_drag_x = x;
9960 m_inertia_last_drag_y = y;
9961 m_last_drag_time = tnow;
9962
9963 if ((last_drag.x != x) || (last_drag.y != y)) {
9964 if (!m_routeState) { // Correct fault on wx32/gtk3, uncommanded
9965 // dragging on route create.
9966 // github #2994
9967 m_bChartDragging = true;
9968 StartTimedMovement();
9969 m_pan_drag.x += last_drag.x - x;
9970 m_pan_drag.y += last_drag.y - y;
9971 last_drag.x = x, last_drag.y = y;
9972 }
9973 }
9974 } else {
9975 if ((last_drag.x != x) || (last_drag.y != y)) {
9976 if (!m_routeState) { // Correct fault on wx32/gtk3, uncommanded
9977 // dragging on route create.
9978 // github #2994
9979 m_bChartDragging = true;
9980 StartTimedMovement();
9981 m_pan_drag.x += last_drag.x - x;
9982 m_pan_drag.y += last_drag.y - y;
9983 last_drag.x = x, last_drag.y = y;
9984 }
9985 }
9986 }
9987
9988 // Handle some special cases
9989 if (g_btouch) {
9990 if ((m_bMeasure_Active && m_nMeasureState) || (m_routeState)) {
9991 // deactivate next LeftUp to ovoid creating an unexpected point
9992 m_DoubleClickTimer->Start();
9993 singleClickEventIsValid = false;
9994 }
9995 }
9996 }
9997
9998 return true;
9999}
10000
10001void ChartCanvas::MouseEvent(wxMouseEvent &event) {
10002 if (MouseEventOverlayWindows(event)) return;
10003
10004 if (MouseEventSetup(event)) return; // handled, no further action required
10005
10006 if (!MouseEventProcessObjects(event)) MouseEventProcessCanvas(event);
10007}
10008
10009void ChartCanvas::SetCanvasCursor(wxMouseEvent &event) {
10010 // Switch to the appropriate cursor on mouse movement
10011
10012 wxCursor *ptarget_cursor = pCursorArrow;
10013 if (!pPlugIn_Cursor) {
10014 ptarget_cursor = pCursorArrow;
10015 if ((!m_routeState) &&
10016 (!m_bMeasure_Active) /*&& ( !m_bCM93MeasureOffset_Active )*/) {
10017 if (cursor_region == MID_RIGHT) {
10018 ptarget_cursor = pCursorRight;
10019 } else if (cursor_region == MID_LEFT) {
10020 ptarget_cursor = pCursorLeft;
10021 } else if (cursor_region == MID_TOP) {
10022 ptarget_cursor = pCursorDown;
10023 } else if (cursor_region == MID_BOT) {
10024 ptarget_cursor = pCursorUp;
10025 } else {
10026 ptarget_cursor = pCursorArrow;
10027 }
10028 } else if (m_bMeasure_Active ||
10029 m_routeState) // If Measure tool use Pencil Cursor
10030 ptarget_cursor = pCursorPencil;
10031 } else {
10032 ptarget_cursor = pPlugIn_Cursor;
10033 }
10034
10035 SetCursor(*ptarget_cursor);
10036}
10037
10038void ChartCanvas::LostMouseCapture(wxMouseCaptureLostEvent &event) {
10039 SetCursor(*pCursorArrow);
10040}
10041
10042void ChartCanvas::ShowObjectQueryWindow(int x, int y, float zlat, float zlon) {
10043 ChartPlugInWrapper *target_plugin_chart = NULL;
10044 s57chart *Chs57 = NULL;
10045 wxFileName file;
10046 wxArrayString files;
10047
10048 ChartBase *target_chart = GetChartAtCursor();
10049 if (target_chart) {
10050 file.Assign(target_chart->GetFullPath());
10051 if ((target_chart->GetChartType() == CHART_TYPE_PLUGIN) &&
10052 (target_chart->GetChartFamily() == CHART_FAMILY_VECTOR))
10053 target_plugin_chart = dynamic_cast<ChartPlugInWrapper *>(target_chart);
10054 else
10055 Chs57 = dynamic_cast<s57chart *>(target_chart);
10056 } else { // target_chart = null, might be mbtiles
10057 std::vector<int> stackIndexArray = m_pQuilt->GetExtendedStackIndexArray();
10058 unsigned int im = stackIndexArray.size();
10059 int scale = 2147483647; // max 32b integer
10060 if (VPoint.b_quilt && im > 0) {
10061 for (unsigned int is = 0; is < im; is++) {
10062 if (ChartData->GetDBChartType(stackIndexArray[is]) ==
10063 CHART_TYPE_MBTILES) {
10064 if (IsTileOverlayIndexInNoShow(stackIndexArray[is])) continue;
10065 double lat, lon;
10066 VPoint.GetLLFromPix(wxPoint(mouse_x, mouse_y), &lat, &lon);
10067 if (ChartData->GetChartTableEntry(stackIndexArray[is])
10068 .GetBBox()
10069 .Contains(lat, lon)) {
10070 if (ChartData->GetChartTableEntry(stackIndexArray[is]).GetScale() <
10071 scale) {
10072 scale =
10073 ChartData->GetChartTableEntry(stackIndexArray[is]).GetScale();
10074 file.Assign(ChartData->GetDBChartFileName(stackIndexArray[is]));
10075 }
10076 }
10077 }
10078 }
10079 }
10080 }
10081
10082 std::vector<Ais8_001_22 *> area_notices;
10083
10084 if (g_pAIS && m_bShowAIS && g_bShowAreaNotices) {
10085 float vp_scale = GetVPScale();
10086
10087 for (const auto &target : g_pAIS->GetAreaNoticeSourcesList()) {
10088 auto target_data = target.second;
10089 if (!target_data->area_notices.empty()) {
10090 for (auto &ani : target_data->area_notices) {
10091 Ais8_001_22 &area_notice = ani.second;
10092
10093 BoundingBox bbox;
10094
10095 for (Ais8_001_22_SubAreaList::iterator sa =
10096 area_notice.sub_areas.begin();
10097 sa != area_notice.sub_areas.end(); ++sa) {
10098 switch (sa->shape) {
10099 case AIS8_001_22_SHAPE_CIRCLE: {
10100 wxPoint target_point;
10101 GetCanvasPointPix(sa->latitude, sa->longitude, &target_point);
10102 bbox.Expand(target_point);
10103 if (sa->radius_m > 0.0) bbox.EnLarge(sa->radius_m * vp_scale);
10104 break;
10105 }
10106 case AIS8_001_22_SHAPE_RECT: {
10107 wxPoint target_point;
10108 GetCanvasPointPix(sa->latitude, sa->longitude, &target_point);
10109 bbox.Expand(target_point);
10110 if (sa->e_dim_m > sa->n_dim_m)
10111 bbox.EnLarge(sa->e_dim_m * vp_scale);
10112 else
10113 bbox.EnLarge(sa->n_dim_m * vp_scale);
10114 break;
10115 }
10116 case AIS8_001_22_SHAPE_POLYGON:
10117 case AIS8_001_22_SHAPE_POLYLINE: {
10118 for (int i = 0; i < 4; ++i) {
10119 double lat = sa->latitude;
10120 double lon = sa->longitude;
10121 ll_gc_ll(lat, lon, sa->angles[i], sa->dists_m[i] / 1852.0,
10122 &lat, &lon);
10123 wxPoint target_point;
10124 GetCanvasPointPix(lat, lon, &target_point);
10125 bbox.Expand(target_point);
10126 }
10127 break;
10128 }
10129 case AIS8_001_22_SHAPE_SECTOR: {
10130 double lat1 = sa->latitude;
10131 double lon1 = sa->longitude;
10132 double lat, lon;
10133 wxPoint target_point;
10134 GetCanvasPointPix(lat1, lon1, &target_point);
10135 bbox.Expand(target_point);
10136 for (int i = 0; i < 18; ++i) {
10137 ll_gc_ll(
10138 lat1, lon1,
10139 sa->left_bound_deg +
10140 i * (sa->right_bound_deg - sa->left_bound_deg) / 18,
10141 sa->radius_m / 1852.0, &lat, &lon);
10142 GetCanvasPointPix(lat, lon, &target_point);
10143 bbox.Expand(target_point);
10144 }
10145 ll_gc_ll(lat1, lon1, sa->right_bound_deg, sa->radius_m / 1852.0,
10146 &lat, &lon);
10147 GetCanvasPointPix(lat, lon, &target_point);
10148 bbox.Expand(target_point);
10149 break;
10150 }
10151 }
10152 }
10153
10154 if (bbox.GetValid() && bbox.PointInBox(x, y)) {
10155 area_notices.push_back(&area_notice);
10156 }
10157 }
10158 }
10159 }
10160 }
10161
10162 if (target_chart || !area_notices.empty() || file.HasName()) {
10163 // Go get the array of all objects at the cursor lat/lon
10164 int sel_rad_pix = 5;
10165 float SelectRadius = sel_rad_pix / (GetVP().view_scale_ppm * 1852 * 60);
10166
10167 // Make sure we always get the lights from an object, even if we are
10168 // currently not displaying lights on the chart.
10169
10170 SetCursor(wxCURSOR_WAIT);
10171 bool lightsVis = m_encShowLights; // gFrame->ToggleLights( false );
10172 if (!lightsVis) SetShowENCLights(true);
10173 ;
10174
10175 ListOfObjRazRules *rule_list = NULL;
10176 ListOfPI_S57Obj *pi_rule_list = NULL;
10177 if (Chs57)
10178 rule_list =
10179 Chs57->GetObjRuleListAtLatLon(zlat, zlon, SelectRadius, &GetVP());
10180 else if (target_plugin_chart)
10181 pi_rule_list = g_pi_manager->GetPlugInObjRuleListAtLatLon(
10182 target_plugin_chart, zlat, zlon, SelectRadius, GetVP());
10183
10184 ListOfObjRazRules *overlay_rule_list = NULL;
10185 ChartBase *overlay_chart = GetOverlayChartAtCursor();
10186 s57chart *CHs57_Overlay = dynamic_cast<s57chart *>(overlay_chart);
10187
10188 if (CHs57_Overlay) {
10189 overlay_rule_list = CHs57_Overlay->GetObjRuleListAtLatLon(
10190 zlat, zlon, SelectRadius, &GetVP());
10191 }
10192
10193 if (!lightsVis) SetShowENCLights(false);
10194
10195 wxString objText;
10196 wxFont *dFont = FontMgr::Get().GetFont(_("ObjectQuery"));
10197 wxString face = dFont->GetFaceName();
10198
10199 if (NULL == g_pObjectQueryDialog) {
10200 g_pObjectQueryDialog =
10201 new S57QueryDialog(this, -1, _("Object Query"), wxDefaultPosition,
10202 wxSize(g_S57_dialog_sx, g_S57_dialog_sy));
10203 }
10204
10205 wxColor bg = g_pObjectQueryDialog->GetBackgroundColour();
10206 wxColor fg = FontMgr::Get().GetFontColor(_("ObjectQuery"));
10207
10208#ifdef __WXOSX__
10209 // Auto Adjustment for dark mode
10210 fg = g_pObjectQueryDialog->GetForegroundColour();
10211#endif
10212
10213 objText.Printf(
10214 _T("<html><body bgcolor=#%02x%02x%02x><font color=#%02x%02x%02x>"),
10215 bg.Red(), bg.Green(), bg.Blue(), fg.Red(), fg.Green(), fg.Blue());
10216
10217#ifdef __WXOSX__
10218 int points = dFont->GetPointSize();
10219#else
10220 int points = dFont->GetPointSize() + 1;
10221#endif
10222
10223 int sizes[7];
10224 for (int i = -2; i < 5; i++) {
10225 sizes[i + 2] = points + i + (i > 0 ? i : 0);
10226 }
10227 g_pObjectQueryDialog->m_phtml->SetFonts(face, face, sizes);
10228
10229 if (wxFONTSTYLE_ITALIC == dFont->GetStyle()) objText += _T("<i>");
10230
10231 if (overlay_rule_list && CHs57_Overlay) {
10232 objText << CHs57_Overlay->CreateObjDescriptions(overlay_rule_list);
10233 objText << _T("<hr noshade>");
10234 }
10235
10236 for (std::vector<Ais8_001_22 *>::iterator an = area_notices.begin();
10237 an != area_notices.end(); ++an) {
10238 objText << _T( "<b>AIS Area Notice:</b> " );
10239 objText << ais8_001_22_notice_names[(*an)->notice_type];
10240 for (std::vector<Ais8_001_22_SubArea>::iterator sa =
10241 (*an)->sub_areas.begin();
10242 sa != (*an)->sub_areas.end(); ++sa)
10243 if (!sa->text.empty()) objText << sa->text;
10244 objText << _T( "<br>expires: " ) << (*an)->expiry_time.Format();
10245 objText << _T( "<hr noshade>" );
10246 }
10247
10248 if (Chs57)
10249 objText << Chs57->CreateObjDescriptions(rule_list);
10250 else if (target_plugin_chart)
10251 objText << g_pi_manager->CreateObjDescriptions(target_plugin_chart,
10252 pi_rule_list);
10253
10254 if (wxFONTSTYLE_ITALIC == dFont->GetStyle()) objText << _T("</i>");
10255
10256 // Add the additional info files
10257 wxString AddFiles, filenameOK;
10258 int filecount = 0;
10259 if (!target_plugin_chart) { // plugincharts shoud take care of this in the
10260 // plugin
10261
10262 AddFiles = wxString::Format(
10263 _T("<hr noshade><br><b>Additional info files attached to: </b> ")
10264 _T("<font ")
10265 _T("size=-2>%s</font><br><table border=0 cellspacing=0 ")
10266 _T("cellpadding=3>"),
10267 file.GetFullName());
10268 file.Normalize();
10269 file.Assign(file.GetPath(), wxT(""));
10270 wxDir dir(file.GetFullPath());
10271 wxString filename;
10272 bool cont = dir.GetFirst(&filename, "", wxDIR_FILES);
10273 while (cont) {
10274 file.Assign(dir.GetNameWithSep().append(filename));
10275 wxString FormatString =
10276 _T("<td valign=top><font size=-2><a ")
10277 _T("href=\"%s\">%s</a></font></td>");
10278 if (g_ObjQFileExt.Find(file.GetExt().Lower()) != wxNOT_FOUND) {
10279 filenameOK = file.GetFullPath(); // remember last valid name
10280 // we are making a 3 columns table. New row only every third file
10281 if (3 * ((int)filecount / 3) == filecount)
10282 FormatString.Prepend(_T("<tr>")); // new row
10283 else
10284 FormatString.Prepend(
10285 _T("<td>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp</td>")); // an empty
10286 // spacer column
10287
10288 AddFiles << wxString::Format(FormatString, file.GetFullPath(),
10289 file.GetFullName());
10290 filecount++;
10291 }
10292 cont = dir.GetNext(&filename);
10293 }
10294 objText << AddFiles << _T("</table>");
10295 }
10296 objText << _T("</font>");
10297 objText << _T("</body></html>");
10298
10299 if (Chs57 || target_plugin_chart || (filecount > 1)) {
10300 g_pObjectQueryDialog->SetHTMLPage(objText);
10301 g_pObjectQueryDialog->Show();
10302 }
10303 if ((!Chs57 && filecount == 1)) { // only one file?, show direktly
10304 // generate an event to avoid double code
10305 wxHtmlLinkInfo hli(filenameOK);
10306 wxHtmlLinkEvent hle(1, hli);
10307 g_pObjectQueryDialog->OnHtmlLinkClicked(hle);
10308 }
10309
10310 if (rule_list) rule_list->Clear();
10311 delete rule_list;
10312
10313 if (overlay_rule_list) overlay_rule_list->Clear();
10314 delete overlay_rule_list;
10315
10316 if (pi_rule_list) pi_rule_list->Clear();
10317 delete pi_rule_list;
10318
10319 SetCursor(wxCURSOR_ARROW);
10320 }
10321}
10322
10323void ChartCanvas::ShowMarkPropertiesDialog(RoutePoint *markPoint) {
10324 bool bNew = false;
10325 if (!g_pMarkInfoDialog) { // There is one global instance of the MarkProp
10326 // Dialog
10327 g_pMarkInfoDialog = new MarkInfoDlg(this);
10328 bNew = true;
10329 }
10330
10331 if (1 /*g_bresponsive*/) {
10332 wxSize canvas_size = GetSize();
10333
10334 int best_size_y = wxMin(400 / OCPN_GetWinDIPScaleFactor(), canvas_size.y);
10335 g_pMarkInfoDialog->SetMinSize(wxSize(-1, best_size_y));
10336
10337 g_pMarkInfoDialog->Layout();
10338
10339 wxPoint canvas_pos = GetPosition();
10340 wxSize fitted_size = g_pMarkInfoDialog->GetSize();
10341
10342 bool newFit = false;
10343 if (canvas_size.x < fitted_size.x) {
10344 fitted_size.x = canvas_size.x - 40;
10345 if (canvas_size.y < fitted_size.y)
10346 fitted_size.y -= 40; // scrollbar added
10347 }
10348 if (canvas_size.y < fitted_size.y) {
10349 fitted_size.y = canvas_size.y - 40;
10350 if (canvas_size.x < fitted_size.x)
10351 fitted_size.x -= 40; // scrollbar added
10352 }
10353
10354 if (newFit) {
10355 g_pMarkInfoDialog->SetSize(fitted_size);
10356 g_pMarkInfoDialog->Centre();
10357 }
10358 }
10359
10360 markPoint->m_bRPIsBeingEdited = false;
10361
10362 wxString title_base = _("Mark Properties");
10363 if (markPoint->m_bIsInRoute) {
10364 title_base = _("Waypoint Properties");
10365 }
10366 g_pMarkInfoDialog->SetRoutePoint(markPoint);
10367 g_pMarkInfoDialog->UpdateProperties();
10368 if (markPoint->m_bIsInLayer) {
10369 wxString caption(wxString::Format(_T("%s, %s: %s"), title_base, _("Layer"),
10370 GetLayerName(markPoint->m_LayerID)));
10371 g_pMarkInfoDialog->SetDialogTitle(caption);
10372 } else
10373 g_pMarkInfoDialog->SetDialogTitle(title_base);
10374
10375 g_pMarkInfoDialog->Show();
10376 g_pMarkInfoDialog->Raise();
10377 g_pMarkInfoDialog->InitialFocus();
10378 if (bNew) g_pMarkInfoDialog->CenterOnScreen();
10379}
10380
10381void ChartCanvas::ShowRoutePropertiesDialog(wxString title, Route *selected) {
10382 pRoutePropDialog = RoutePropDlgImpl::getInstance(this);
10383 pRoutePropDialog->SetRouteAndUpdate(selected);
10384 // pNew->UpdateProperties();
10385 pRoutePropDialog->Show();
10386 pRoutePropDialog->Raise();
10387 return;
10388 pRoutePropDialog = RoutePropDlgImpl::getInstance(
10389 this); // There is one global instance of the RouteProp Dialog
10390
10391 if (g_bresponsive) {
10392 wxSize canvas_size = GetSize();
10393 wxPoint canvas_pos = GetPosition();
10394 wxSize fitted_size = pRoutePropDialog->GetSize();
10395 ;
10396
10397 if (canvas_size.x < fitted_size.x) {
10398 fitted_size.x = canvas_size.x;
10399 if (canvas_size.y < fitted_size.y)
10400 fitted_size.y -= 20; // scrollbar added
10401 }
10402 if (canvas_size.y < fitted_size.y) {
10403 fitted_size.y = canvas_size.y;
10404 if (canvas_size.x < fitted_size.x)
10405 fitted_size.x -= 20; // scrollbar added
10406 }
10407
10408 pRoutePropDialog->SetSize(fitted_size);
10409 pRoutePropDialog->Centre();
10410
10411 // int xp = (canvas_size.x - fitted_size.x)/2;
10412 // int yp = (canvas_size.y - fitted_size.y)/2;
10413
10414 wxPoint xxp = ClientToScreen(canvas_pos);
10415 // pRoutePropDialog->Move(xxp.x + xp, xxp.y + yp);
10416 }
10417
10418 pRoutePropDialog->SetRouteAndUpdate(selected);
10419
10420 pRoutePropDialog->Show();
10421
10422 Refresh(false);
10423}
10424
10425void ChartCanvas::ShowTrackPropertiesDialog(Track *selected) {
10426 pTrackPropDialog = TrackPropDlg::getInstance(
10427 this); // There is one global instance of the RouteProp Dialog
10428
10429 pTrackPropDialog->SetTrackAndUpdate(selected);
10430 pTrackPropDialog->UpdateProperties();
10431
10432 pTrackPropDialog->Show();
10433
10434 Refresh(false);
10435}
10436
10437void pupHandler_PasteWaypoint() {
10438 Kml kml;
10439
10440 int pasteBuffer = kml.ParsePasteBuffer();
10441 RoutePoint *pasted = kml.GetParsedRoutePoint();
10442 if (!pasted) return;
10443
10444 double nearby_radius_meters =
10445 g_Platform->GetSelectRadiusPix() /
10446 gFrame->GetPrimaryCanvas()->GetCanvasTrueScale();
10447
10448 RoutePoint *nearPoint = pWayPointMan->GetNearbyWaypoint(
10449 pasted->m_lat, pasted->m_lon, nearby_radius_meters);
10450
10451 int answer = wxID_NO;
10452 if (nearPoint && !nearPoint->m_bIsInLayer) {
10453 wxString msg;
10454 msg << _(
10455 "There is an existing waypoint at the same location as the one you are "
10456 "pasting. Would you like to merge the pasted data with it?\n\n");
10457 msg << _("Answering 'No' will create a new waypoint at the same location.");
10458 answer = OCPNMessageBox(NULL, msg, _("Merge waypoint?"),
10459 (long)wxYES_NO | wxCANCEL | wxNO_DEFAULT);
10460 }
10461
10462 if (answer == wxID_YES) {
10463 nearPoint->SetName(pasted->GetName());
10464 nearPoint->m_MarkDescription = pasted->m_MarkDescription;
10465 if (pRouteManagerDialog && pRouteManagerDialog->IsShown())
10466 pRouteManagerDialog->UpdateWptListCtrl();
10467 }
10468
10469 if (answer == wxID_NO) {
10470 RoutePoint *newPoint = new RoutePoint(pasted);
10471 newPoint->m_bIsolatedMark = true;
10472 pSelect->AddSelectableRoutePoint(newPoint->m_lat, newPoint->m_lon,
10473 newPoint);
10474 pConfig->AddNewWayPoint(newPoint, -1);
10475 pWayPointMan->AddRoutePoint(newPoint);
10476 if (pRouteManagerDialog && pRouteManagerDialog->IsShown())
10477 pRouteManagerDialog->UpdateWptListCtrl();
10478 if (RoutePointGui(*newPoint).IsVisibleSelectable(g_focusCanvas))
10479 RoutePointGui(*newPoint).ShowScaleWarningMessage(g_focusCanvas);
10480 }
10481
10482 gFrame->InvalidateAllGL();
10483 gFrame->RefreshAllCanvas(false);
10484}
10485
10486void pupHandler_PasteRoute() {
10487 Kml kml;
10488
10489 int pasteBuffer = kml.ParsePasteBuffer();
10490 Route *pasted = kml.GetParsedRoute();
10491 if (!pasted) return;
10492
10493 double nearby_radius_meters =
10494 g_Platform->GetSelectRadiusPix() /
10495 gFrame->GetPrimaryCanvas()->GetCanvasTrueScale();
10496
10497 RoutePoint *curPoint;
10498 RoutePoint *nearPoint;
10499 RoutePoint *prevPoint = NULL;
10500
10501 bool mergepoints = false;
10502 bool createNewRoute = true;
10503 int existingWaypointCounter = 0;
10504
10505 for (int i = 1; i <= pasted->GetnPoints(); i++) {
10506 curPoint = pasted->GetPoint(i); // NB! n starts at 1 !
10507 nearPoint = pWayPointMan->GetNearbyWaypoint(
10508 curPoint->m_lat, curPoint->m_lon, nearby_radius_meters);
10509 if (nearPoint) {
10510 mergepoints = true;
10511 existingWaypointCounter++;
10512 // Small hack here to avoid both extending RoutePoint and repeating all
10513 // the GetNearbyWaypoint calculations. Use existin data field in
10514 // RoutePoint as temporary storage.
10515 curPoint->m_bPtIsSelected = true;
10516 }
10517 }
10518
10519 int answer = wxID_NO;
10520 if (mergepoints) {
10521 wxString msg;
10522 msg << _(
10523 "There are existing waypoints at the same location as some of the ones "
10524 "you are pasting. Would you like to just merge the pasted data into "
10525 "them?\n\n");
10526 msg << _("Answering 'No' will create all new waypoints for this route.");
10527 answer = OCPNMessageBox(NULL, msg, _("Merge waypoints?"),
10528 (long)wxYES_NO | wxCANCEL | wxYES_DEFAULT);
10529
10530 if (answer == wxID_CANCEL) {
10531 return;
10532 }
10533 }
10534
10535 // If all waypoints exist since before, and a route with the same name, we
10536 // don't create a new route.
10537 if (mergepoints && answer == wxID_YES &&
10538 existingWaypointCounter == pasted->GetnPoints()) {
10539 wxRouteListNode *route_node = pRouteList->GetFirst();
10540 while (route_node) {
10541 Route *proute = route_node->GetData();
10542
10543 if (pasted->m_RouteNameString == proute->m_RouteNameString) {
10544 createNewRoute = false;
10545 break;
10546 }
10547 route_node = route_node->GetNext();
10548 }
10549 }
10550
10551 Route *newRoute = 0;
10552 RoutePoint *newPoint = 0;
10553
10554 if (createNewRoute) {
10555 newRoute = new Route();
10556 newRoute->m_RouteNameString = pasted->m_RouteNameString;
10557 }
10558
10559 for (int i = 1; i <= pasted->GetnPoints(); i++) {
10560 curPoint = pasted->GetPoint(i);
10561 if (answer == wxID_YES && curPoint->m_bPtIsSelected) {
10562 curPoint->m_bPtIsSelected = false;
10563 newPoint = pWayPointMan->GetNearbyWaypoint(
10564 curPoint->m_lat, curPoint->m_lon, nearby_radius_meters);
10565 newPoint->SetName(curPoint->GetName());
10566 newPoint->m_MarkDescription = curPoint->m_MarkDescription;
10567
10568 if (createNewRoute) newRoute->AddPoint(newPoint);
10569 } else {
10570 curPoint->m_bPtIsSelected = false;
10571
10572 newPoint = new RoutePoint(curPoint);
10573 newPoint->m_bIsolatedMark = false;
10574 newPoint->SetIconName(_T("circle"));
10575 newPoint->m_bIsVisible = true;
10576 newPoint->m_bShowName = false;
10577 newPoint->SetShared(false);
10578
10579 newRoute->AddPoint(newPoint);
10580 pSelect->AddSelectableRoutePoint(newPoint->m_lat, newPoint->m_lon,
10581 newPoint);
10582 pConfig->AddNewWayPoint(newPoint, -1);
10583 pWayPointMan->AddRoutePoint(newPoint);
10584 }
10585 if (i > 1 && createNewRoute)
10586 pSelect->AddSelectableRouteSegment(prevPoint->m_lat, prevPoint->m_lon,
10587 curPoint->m_lat, curPoint->m_lon,
10588 prevPoint, newPoint, newRoute);
10589 prevPoint = newPoint;
10590 }
10591
10592 if (createNewRoute) {
10593 pRouteList->Append(newRoute);
10594 pConfig->AddNewRoute(newRoute); // use auto next num
10595
10596 if (pRoutePropDialog && pRoutePropDialog->IsShown()) {
10597 pRoutePropDialog->SetRouteAndUpdate(newRoute);
10598 }
10599
10600 if (pRouteManagerDialog && pRouteManagerDialog->IsShown()) {
10601 pRouteManagerDialog->UpdateRouteListCtrl();
10602 pRouteManagerDialog->UpdateWptListCtrl();
10603 }
10604 gFrame->InvalidateAllGL();
10605 gFrame->RefreshAllCanvas(false);
10606 }
10607 if (RoutePointGui(*newPoint).IsVisibleSelectable(g_focusCanvas))
10608 RoutePointGui(*newPoint).ShowScaleWarningMessage(g_focusCanvas);
10609}
10610
10611void pupHandler_PasteTrack() {
10612 Kml kml;
10613
10614 int pasteBuffer = kml.ParsePasteBuffer();
10615 Track *pasted = kml.GetParsedTrack();
10616 if (!pasted) return;
10617
10618 TrackPoint *curPoint;
10619
10620 Track *newTrack = new Track();
10621 TrackPoint *newPoint;
10622 TrackPoint *prevPoint = NULL;
10623
10624 newTrack->SetName(pasted->GetName());
10625
10626 for (int i = 0; i < pasted->GetnPoints(); i++) {
10627 curPoint = pasted->GetPoint(i);
10628
10629 newPoint = new TrackPoint(curPoint);
10630
10631 wxDateTime now = wxDateTime::Now();
10632 newPoint->SetCreateTime(curPoint->GetCreateTime());
10633
10634 newTrack->AddPoint(newPoint);
10635
10636 if (prevPoint)
10637 pSelect->AddSelectableTrackSegment(prevPoint->m_lat, prevPoint->m_lon,
10638 newPoint->m_lat, newPoint->m_lon,
10639 prevPoint, newPoint, newTrack);
10640
10641 prevPoint = newPoint;
10642 }
10643
10644 g_TrackList.push_back(newTrack);
10645 pConfig->AddNewTrack(newTrack);
10646
10647 gFrame->InvalidateAllGL();
10648 gFrame->RefreshAllCanvas(false);
10649}
10650
10651bool ChartCanvas::InvokeCanvasMenu(int x, int y, int seltype) {
10652 m_canvasMenu = new CanvasMenuHandler(this, m_pSelectedRoute, m_pSelectedTrack,
10653 m_pFoundRoutePoint, m_FoundAIS_MMSI,
10654 m_pIDXCandidate);
10655
10656 Connect(
10657 wxEVT_COMMAND_MENU_SELECTED,
10658 (wxObjectEventFunction)(wxEventFunction)&ChartCanvas::PopupMenuHandler);
10659
10660 m_canvasMenu->CanvasPopupMenu(x, y, seltype);
10661
10662 Disconnect(
10663 wxEVT_COMMAND_MENU_SELECTED,
10664 (wxObjectEventFunction)(wxEventFunction)&ChartCanvas::PopupMenuHandler);
10665
10666 delete m_canvasMenu;
10667 m_canvasMenu = NULL;
10668
10669#ifdef __WXQT__
10670 // gFrame->SurfaceToolbar();
10671 // g_MainToolbar->Raise();
10672#endif
10673
10674 return true;
10675}
10676
10677void ChartCanvas::PopupMenuHandler(wxCommandEvent &event) {
10678 // Pass menu events from the canvas to the menu handler
10679 // This is necessarily in ChartCanvas since that is the menu's parent.
10680 if (m_canvasMenu) {
10681 m_canvasMenu->PopupMenuHandler(event);
10682 }
10683 return;
10684}
10685
10686void ChartCanvas::StartRoute(void) {
10687 // Do not allow more than one canvas to create a route at one time.
10688 if (g_brouteCreating) return;
10689
10690 if (g_MainToolbar) g_MainToolbar->DisableTooltips();
10691
10692 g_brouteCreating = true;
10693 m_routeState = 1;
10694 m_bDrawingRoute = false;
10695 SetCursor(*pCursorPencil);
10696 // SetCanvasToolbarItemState(ID_ROUTE, true);
10697 gFrame->SetMasterToolbarItemState(ID_MENU_ROUTE_NEW, true);
10698
10699 HideGlobalToolbar();
10700
10701#ifdef __ANDROID__
10702 androidSetRouteAnnunciator(true);
10703#endif
10704}
10705
10706void ChartCanvas::FinishRoute(void) {
10707 m_routeState = 0;
10708 m_prev_pMousePoint = NULL;
10709 m_bDrawingRoute = false;
10710
10711 // SetCanvasToolbarItemState(ID_ROUTE, false);
10712 gFrame->SetMasterToolbarItemState(ID_MENU_ROUTE_NEW, false);
10713#ifdef __ANDROID__
10714 androidSetRouteAnnunciator(false);
10715#endif
10716
10717 SetCursor(*pCursorArrow);
10718
10719 if (m_pMouseRoute) {
10720 if (m_bAppendingRoute)
10721 pConfig->UpdateRoute(m_pMouseRoute);
10722 else {
10723 if (m_pMouseRoute->GetnPoints() > 1) {
10724 pConfig->AddNewRoute(m_pMouseRoute);
10725 } else {
10726 g_pRouteMan->DeleteRoute(m_pMouseRoute,
10727 NavObjectChanges::getInstance());
10728 m_pMouseRoute = NULL;
10729 }
10730 }
10731 if (m_pMouseRoute) m_pMouseRoute->SetHiLite(0);
10732
10733 if (RoutePropDlgImpl::getInstanceFlag() && pRoutePropDialog &&
10734 (pRoutePropDialog->IsShown())) {
10735 pRoutePropDialog->SetRouteAndUpdate(m_pMouseRoute, true);
10736 }
10737
10738 if (RouteManagerDialog::getInstanceFlag() && pRouteManagerDialog) {
10739 if (pRouteManagerDialog && pRouteManagerDialog->IsShown())
10740 pRouteManagerDialog->UpdateRouteListCtrl();
10741 }
10742 }
10743 m_bAppendingRoute = false;
10744 m_pMouseRoute = NULL;
10745
10746 m_pSelectedRoute = NULL;
10747
10748 undo->InvalidateUndo();
10749 gFrame->RefreshAllCanvas(true);
10750
10751 if (g_MainToolbar) g_MainToolbar->EnableTooltips();
10752
10753 ShowGlobalToolbar();
10754
10755 g_brouteCreating = false;
10756}
10757
10758void ChartCanvas::HideGlobalToolbar() {
10759 if (m_canvasIndex == 0) {
10760 m_last_TBviz = gFrame->SetGlobalToolbarViz(false);
10761 }
10762}
10763
10764void ChartCanvas::ShowGlobalToolbar() {
10765 if (m_canvasIndex == 0) {
10766 if (m_last_TBviz) gFrame->SetGlobalToolbarViz(true);
10767 }
10768}
10769
10770void ChartCanvas::ShowAISTargetList(void) {
10771 if (NULL == g_pAISTargetList) { // There is one global instance of the Dialog
10772 g_pAISTargetList = new AISTargetListDialog(parent_frame, g_pauimgr, g_pAIS);
10773 }
10774
10775 g_pAISTargetList->UpdateAISTargetList();
10776}
10777
10778void ChartCanvas::RenderAllChartOutlines(ocpnDC &dc, ViewPort &vp) {
10779 if (!m_bShowOutlines) return;
10780
10781 if (!ChartData) return;
10782
10783 int nEntry = ChartData->GetChartTableEntries();
10784
10785 for (int i = 0; i < nEntry; i++) {
10786 ChartTableEntry *pt = (ChartTableEntry *)&ChartData->GetChartTableEntry(i);
10787
10788 // Check to see if the candidate chart is in the currently active group
10789 bool b_group_draw = false;
10790 if (m_groupIndex > 0) {
10791 for (unsigned int ig = 0; ig < pt->GetGroupArray().size(); ig++) {
10792 int index = pt->GetGroupArray()[ig];
10793 if (m_groupIndex == index) {
10794 b_group_draw = true;
10795 break;
10796 }
10797 }
10798 } else
10799 b_group_draw = true;
10800
10801 if (b_group_draw) RenderChartOutline(dc, i, vp);
10802 }
10803
10804 // On CM93 Composite Charts, draw the outlines of the next smaller
10805 // scale cell
10806 cm93compchart *pcm93 = NULL;
10807 if (VPoint.b_quilt) {
10808 for (ChartBase *pch = GetFirstQuiltChart(); pch; pch = GetNextQuiltChart())
10809 if (pch->GetChartType() == CHART_TYPE_CM93COMP) {
10810 pcm93 = (cm93compchart *)pch;
10811 break;
10812 }
10813 } else if (m_singleChart &&
10814 (m_singleChart->GetChartType() == CHART_TYPE_CM93COMP))
10815 pcm93 = (cm93compchart *)m_singleChart;
10816
10817 if (pcm93) {
10818 double chart_native_ppm = m_canvas_scale_factor / pcm93->GetNativeScale();
10819 double zoom_factor = GetVP().view_scale_ppm / chart_native_ppm;
10820
10821 if (zoom_factor > 8.0) {
10822 wxPen mPen(GetGlobalColor(_T("UINFM")), 2, wxPENSTYLE_SHORT_DASH);
10823 dc.SetPen(mPen);
10824 } else {
10825 wxPen mPen(GetGlobalColor(_T("UINFM")), 1, wxPENSTYLE_SOLID);
10826 dc.SetPen(mPen);
10827 }
10828
10829 pcm93->RenderNextSmallerCellOutlines(dc, vp, this);
10830 }
10831}
10832
10833void ChartCanvas::RenderChartOutline(ocpnDC &dc, int dbIndex, ViewPort &vp) {
10834#ifdef ocpnUSE_GL
10835 if (g_bopengl && m_glcc) {
10836 /* opengl version specially optimized */
10837 m_glcc->RenderChartOutline(dc, dbIndex, vp);
10838 return;
10839 }
10840#endif
10841
10842 if (ChartData->GetDBChartType(dbIndex) == CHART_TYPE_PLUGIN) {
10843 if (!ChartData->IsChartAvailable(dbIndex)) return;
10844 }
10845
10846 float plylat, plylon;
10847 float plylat1, plylon1;
10848
10849 int pixx, pixy, pixx1, pixy1;
10850
10851 LLBBox box;
10852 ChartData->GetDBBoundingBox(dbIndex, box);
10853
10854 // Don't draw an outline in the case where the chart covers the entire world
10855 // */
10856 if (box.GetLonRange() == 360) return;
10857
10858 double lon_bias = 0;
10859 // chart is outside of viewport lat/lon bounding box
10860 if (box.IntersectOutGetBias(vp.GetBBox(), lon_bias)) return;
10861
10862 int nPly = ChartData->GetDBPlyPoint(dbIndex, 0, &plylat, &plylon);
10863
10864 if (ChartData->GetDBChartType(dbIndex) == CHART_TYPE_CM93)
10865 dc.SetPen(wxPen(GetGlobalColor(_T ( "YELO1" )), 1, wxPENSTYLE_SOLID));
10866
10867 else if (ChartData->GetDBChartFamily(dbIndex) == CHART_FAMILY_VECTOR)
10868 dc.SetPen(wxPen(GetGlobalColor(_T ( "UINFG" )), 1, wxPENSTYLE_SOLID));
10869
10870 else
10871 dc.SetPen(wxPen(GetGlobalColor(_T ( "UINFR" )), 1, wxPENSTYLE_SOLID));
10872
10873 // Are there any aux ply entries?
10874 int nAuxPlyEntries = ChartData->GetnAuxPlyEntries(dbIndex);
10875 if (0 == nAuxPlyEntries) // There are no aux Ply Point entries
10876 {
10877 wxPoint r, r1;
10878
10879 ChartData->GetDBPlyPoint(dbIndex, 0, &plylat, &plylon);
10880 plylon += lon_bias;
10881
10882 GetCanvasPointPix(plylat, plylon, &r);
10883 pixx = r.x;
10884 pixy = r.y;
10885
10886 for (int i = 0; i < nPly - 1; i++) {
10887 ChartData->GetDBPlyPoint(dbIndex, i + 1, &plylat1, &plylon1);
10888 plylon1 += lon_bias;
10889
10890 GetCanvasPointPix(plylat1, plylon1, &r1);
10891 pixx1 = r1.x;
10892 pixy1 = r1.y;
10893
10894 int pixxs1 = pixx1;
10895 int pixys1 = pixy1;
10896
10897 bool b_skip = false;
10898
10899 if (vp.chart_scale > 5e7) {
10900 // calculate projected distance between these two points in meters
10901 double dist = sqrt(pow((double)(pixx1 - pixx), 2) +
10902 pow((double)(pixy1 - pixy), 2)) /
10903 vp.view_scale_ppm;
10904
10905 if (dist > 0.0) {
10906 // calculate GC distance between these two points in meters
10907 double distgc =
10908 DistGreatCircle(plylat, plylon, plylat1, plylon1) * 1852.;
10909
10910 // If the distances are nonsense, it means that the scale is very
10911 // small and the segment wrapped the world So skip it....
10912 // TODO improve this to draw two segments
10913 if (fabs(dist - distgc) > 10000. * 1852.) // lotsa miles
10914 b_skip = true;
10915 } else
10916 b_skip = true;
10917 }
10918
10919 ClipResult res = cohen_sutherland_line_clip_i(
10920 &pixx, &pixy, &pixx1, &pixy1, 0, vp.pix_width, 0, vp.pix_height);
10921 if (res != Invisible && !b_skip)
10922 dc.DrawLine(pixx, pixy, pixx1, pixy1, false);
10923
10924 plylat = plylat1;
10925 plylon = plylon1;
10926 pixx = pixxs1;
10927 pixy = pixys1;
10928 }
10929
10930 ChartData->GetDBPlyPoint(dbIndex, 0, &plylat1, &plylon1);
10931 plylon1 += lon_bias;
10932
10933 GetCanvasPointPix(plylat1, plylon1, &r1);
10934 pixx1 = r1.x;
10935 pixy1 = r1.y;
10936
10937 ClipResult res = cohen_sutherland_line_clip_i(
10938 &pixx, &pixy, &pixx1, &pixy1, 0, vp.pix_width, 0, vp.pix_height);
10939 if (res != Invisible) dc.DrawLine(pixx, pixy, pixx1, pixy1, false);
10940 }
10941
10942 else // Use Aux PlyPoints
10943 {
10944 wxPoint r, r1;
10945
10946 int nAuxPlyEntries = ChartData->GetnAuxPlyEntries(dbIndex);
10947 for (int j = 0; j < nAuxPlyEntries; j++) {
10948 int nAuxPly =
10949 ChartData->GetDBAuxPlyPoint(dbIndex, 0, j, &plylat, &plylon);
10950 GetCanvasPointPix(plylat, plylon, &r);
10951 pixx = r.x;
10952 pixy = r.y;
10953
10954 for (int i = 0; i < nAuxPly - 1; i++) {
10955 ChartData->GetDBAuxPlyPoint(dbIndex, i + 1, j, &plylat1, &plylon1);
10956
10957 GetCanvasPointPix(plylat1, plylon1, &r1);
10958 pixx1 = r1.x;
10959 pixy1 = r1.y;
10960
10961 int pixxs1 = pixx1;
10962 int pixys1 = pixy1;
10963
10964 bool b_skip = false;
10965
10966 if (vp.chart_scale > 5e7) {
10967 // calculate projected distance between these two points in meters
10968 double dist = sqrt((double)((pixx1 - pixx) * (pixx1 - pixx)) +
10969 ((pixy1 - pixy) * (pixy1 - pixy))) /
10970 vp.view_scale_ppm;
10971 if (dist > 0.0) {
10972 // calculate GC distance between these two points in meters
10973 double distgc =
10974 DistGreatCircle(plylat, plylon, plylat1, plylon1) * 1852.;
10975
10976 // If the distances are nonsense, it means that the scale is very
10977 // small and the segment wrapped the world So skip it....
10978 // TODO improve this to draw two segments
10979 if (fabs(dist - distgc) > 10000. * 1852.) // lotsa miles
10980 b_skip = true;
10981 } else
10982 b_skip = true;
10983 }
10984
10985 ClipResult res = cohen_sutherland_line_clip_i(
10986 &pixx, &pixy, &pixx1, &pixy1, 0, vp.pix_width, 0, vp.pix_height);
10987 if (res != Invisible && !b_skip) dc.DrawLine(pixx, pixy, pixx1, pixy1);
10988
10989 plylat = plylat1;
10990 plylon = plylon1;
10991 pixx = pixxs1;
10992 pixy = pixys1;
10993 }
10994
10995 ChartData->GetDBAuxPlyPoint(dbIndex, 0, j, &plylat1, &plylon1);
10996 GetCanvasPointPix(plylat1, plylon1, &r1);
10997 pixx1 = r1.x;
10998 pixy1 = r1.y;
10999
11000 ClipResult res = cohen_sutherland_line_clip_i(
11001 &pixx, &pixy, &pixx1, &pixy1, 0, vp.pix_width, 0, vp.pix_height);
11002 if (res != Invisible) dc.DrawLine(pixx, pixy, pixx1, pixy1, false);
11003 }
11004 }
11005}
11006
11007static void RouteLegInfo(ocpnDC &dc, wxPoint ref_point, const wxString &first,
11008 const wxString &second) {
11009 wxFont *dFont = FontMgr::Get().GetFont(_("RouteLegInfoRollover"));
11010
11011 int pointsize = dFont->GetPointSize();
11012 pointsize /= OCPN_GetWinDIPScaleFactor();
11013
11014 wxFont *psRLI_font = FontMgr::Get().FindOrCreateFont(
11015 pointsize, dFont->GetFamily(), dFont->GetStyle(), dFont->GetWeight(),
11016 false, dFont->GetFaceName());
11017
11018 dc.SetFont(*psRLI_font);
11019
11020 int w1, h1;
11021 int w2 = 0;
11022 int h2 = 0;
11023 int h, w;
11024
11025 int xp, yp;
11026 int hilite_offset = 3;
11027#ifdef __WXMAC__
11028 wxScreenDC sdc;
11029 sdc.GetTextExtent(first, &w1, &h1, NULL, NULL, psRLI_font);
11030 if (second.Len()) sdc.GetTextExtent(second, &w2, &h2, NULL, NULL, psRLI_font);
11031#else
11032 dc.GetTextExtent(first, &w1, &h1);
11033 if (second.Len()) dc.GetTextExtent(second, &w2, &h2);
11034#endif
11035
11036 h1 *= (OCPN_GetWinDIPScaleFactor() * 100.) / 100;
11037 h2 *= (OCPN_GetWinDIPScaleFactor() * 100.) / 100;
11038
11039 w = wxMax(w1, w2) + (h1 / 2); // Add a little right pad
11040 w *= (OCPN_GetWinDIPScaleFactor() * 100.) / 100;
11041
11042 h = h1 + h2;
11043
11044 xp = ref_point.x - w;
11045 yp = ref_point.y;
11046 yp += hilite_offset;
11047
11048 AlphaBlending(dc, xp, yp, w, h, 0.0, GetGlobalColor(_T ( "YELO1" )), 172);
11049
11050 dc.SetPen(wxPen(GetGlobalColor(_T ( "UBLCK" ))));
11051 dc.SetTextForeground(GetGlobalColor(_T ( "UBLCK" )));
11052
11053 dc.DrawText(first, xp, yp);
11054 if (second.Len()) dc.DrawText(second, xp, yp + h1);
11055}
11056
11057void ChartCanvas::RenderShipToActive(ocpnDC &dc, bool Use_Opengl) {
11058 if (!g_bAllowShipToActive) return;
11059
11060 Route *rt = g_pRouteMan->GetpActiveRoute();
11061 if (!rt) return;
11062
11063 if (RoutePoint *rp = g_pRouteMan->GetpActivePoint()) {
11064 wxPoint2DDouble pa, pb;
11065 GetDoubleCanvasPointPix(gLat, gLon, &pa);
11066 GetDoubleCanvasPointPix(rp->m_lat, rp->m_lon, &pb);
11067
11068 // set pen
11069 int width =
11070 g_pRouteMan->GetRoutePen()->GetWidth(); // get default route pen with
11071 if (rt->m_width != wxPENSTYLE_INVALID)
11072 width = rt->m_width; // set route pen style if any
11073 wxPenStyle style = (wxPenStyle)::StyleValues[wxMin(
11074 g_shipToActiveStyle, 5)]; // get setting pen style
11075 if (style == wxPENSTYLE_INVALID) style = wxPENSTYLE_SOLID; // default style
11076 wxColour color =
11077 g_shipToActiveColor > 0 ? GpxxColors[wxMin(g_shipToActiveColor - 1, 15)]
11078 : // set setting route pen color
11079 g_pRouteMan->GetActiveRoutePen()->GetColour(); // default color
11080 wxPen *mypen = wxThePenList->FindOrCreatePen(color, width, style);
11081
11082 dc.SetPen(*mypen);
11083 dc.SetBrush(wxBrush(color, wxBRUSHSTYLE_SOLID));
11084
11085 if (!Use_Opengl)
11086 RouteGui(*rt).RenderSegment(dc, (int)pa.m_x, (int)pa.m_y, (int)pb.m_x,
11087 (int)pb.m_y, GetVP(), true);
11088
11089#ifdef ocpnUSE_GL
11090 else {
11091#ifdef USE_ANDROID_GLES2
11092 dc.DrawLine(pa.m_x, pa.m_y, pb.m_x, pb.m_y);
11093#else
11094 if (style != wxPENSTYLE_SOLID) {
11095 if (glChartCanvas::dash_map.find(style) !=
11096 glChartCanvas::dash_map.end()) {
11097 mypen->SetDashes(2, &glChartCanvas::dash_map[style][0]);
11098 dc.SetPen(*mypen);
11099 }
11100 }
11101 dc.DrawLine(pa.m_x, pa.m_y, pb.m_x, pb.m_y);
11102#endif
11103
11104 RouteGui(*rt).RenderSegmentArrowsGL(dc, (int)pa.m_x, (int)pa.m_y,
11105 (int)pb.m_x, (int)pb.m_y, GetVP());
11106 }
11107#endif
11108 }
11109}
11110
11111void ChartCanvas::RenderRouteLegs(ocpnDC &dc) {
11112 Route *route = 0;
11113 if (m_routeState >= 2) route = m_pMouseRoute;
11114 if (m_pMeasureRoute && m_bMeasure_Active && (m_nMeasureState >= 2))
11115 route = m_pMeasureRoute;
11116
11117 if (!route) return;
11118
11119 // Validate route pointer
11120 if (!g_pRouteMan->IsRouteValid(route)) return;
11121
11122 double render_lat = m_cursor_lat;
11123 double render_lon = m_cursor_lon;
11124
11125 int np = route->GetnPoints();
11126 if (np) {
11127 if (g_btouch && (np > 1)) np--;
11128 RoutePoint rp = route->GetPoint(np);
11129 render_lat = rp.m_lat;
11130 render_lon = rp.m_lon;
11131 }
11132
11133 double rhumbBearing, rhumbDist;
11134 DistanceBearingMercator(m_cursor_lat, m_cursor_lon, render_lat, render_lon,
11135 &rhumbBearing, &rhumbDist);
11136 double brg = rhumbBearing;
11137 double dist = rhumbDist;
11138
11139 // Skip GreatCircle rubberbanding on touch devices.
11140 if (!g_btouch) {
11141 double gcBearing, gcBearing2, gcDist;
11142 Geodesic::GreatCircleDistBear(render_lon, render_lat, m_cursor_lon,
11143 m_cursor_lat, &gcDist, &gcBearing,
11144 &gcBearing2);
11145 double gcDistm = gcDist / 1852.0;
11146
11147 if ((render_lat == m_cursor_lat) && (render_lon == m_cursor_lon))
11148 rhumbBearing = 90.;
11149
11150 wxPoint destPoint, lastPoint;
11151
11152 route->m_NextLegGreatCircle = false;
11153 int milesDiff = rhumbDist - gcDistm;
11154 if (milesDiff > 1) {
11155 brg = gcBearing;
11156 dist = gcDistm;
11157 route->m_NextLegGreatCircle = true;
11158 }
11159
11160 // FIXME (MacOS, the first segment is rendered wrong)
11161 RouteGui(*route).DrawPointWhich(dc, this, route->m_lastMousePointIndex,
11162 &lastPoint);
11163
11164 if (route->m_NextLegGreatCircle) {
11165 for (int i = 1; i <= milesDiff; i++) {
11166 double p = (double)i * (1.0 / (double)milesDiff);
11167 double pLat, pLon;
11168 Geodesic::GreatCircleTravel(render_lon, render_lat, gcDist * p, brg,
11169 &pLon, &pLat, &gcBearing2);
11170 destPoint = VPoint.GetPixFromLL(pLat, pLon);
11171 RouteGui(*route).DrawSegment(dc, this, &lastPoint, &destPoint, GetVP(),
11172 false);
11173 lastPoint = destPoint;
11174 }
11175 } else {
11176 if (r_rband.x && r_rband.y) { // RubberBand disabled?
11177 RouteGui(*route).DrawSegment(dc, this, &lastPoint, &r_rband, GetVP(),
11178 false);
11179 if (m_bMeasure_DistCircle) {
11180 double distanceRad = sqrtf(powf((float)(r_rband.x - lastPoint.x), 2) +
11181 powf((float)(r_rband.y - lastPoint.y), 2));
11182
11183 dc.SetPen(*g_pRouteMan->GetRoutePen());
11184 dc.SetBrush(*wxTRANSPARENT_BRUSH);
11185 dc.StrokeCircle(lastPoint.x, lastPoint.y, distanceRad);
11186 }
11187 }
11188 }
11189 }
11190
11191 wxString routeInfo;
11192 if (g_bShowTrue)
11193 routeInfo << wxString::Format(wxString("%03d%c(T) ", wxConvUTF8), (int)brg,
11194 0x00B0);
11195
11196 if (g_bShowMag) {
11197 double latAverage = (m_cursor_lat + render_lat) / 2;
11198 double lonAverage = (m_cursor_lon + render_lon) / 2;
11199 double varBrg = gFrame->GetMag(brg, latAverage, lonAverage);
11200
11201 routeInfo << wxString::Format(wxString("%03d%c(M) ", wxConvUTF8),
11202 (int)varBrg, 0x00B0);
11203 }
11204
11205 routeInfo << _T(" ") << FormatDistanceAdaptive(dist);
11206
11207 wxString s0;
11208 if (!route->m_bIsInLayer)
11209 s0.Append(_("Route") + _T(": "));
11210 else
11211 s0.Append(_("Layer Route: "));
11212
11213 double disp_length = route->m_route_length;
11214 if (!g_btouch) disp_length += dist; // Add in the to-be-created leg.
11215 s0 += FormatDistanceAdaptive(disp_length);
11216
11217 RouteLegInfo(dc, r_rband, routeInfo, s0);
11218
11219 m_brepaint_piano = true;
11220}
11221
11222void ChartCanvas::RenderVisibleSectorLights(ocpnDC &dc) {
11223 if (!m_bShowVisibleSectors) return;
11224
11225 if (g_bDeferredInitDone) {
11226 // need to re-evaluate sectors?
11227 double rhumbBearing, rhumbDist;
11228 DistanceBearingMercator(gLat, gLon, m_sector_glat, m_sector_glon,
11229 &rhumbBearing, &rhumbDist);
11230
11231 if (rhumbDist > 0.05) // miles
11232 {
11233 s57_GetVisibleLightSectors(this, gLat, gLon, GetVP(),
11234 m_sectorlegsVisible);
11235 m_sector_glat = gLat;
11236 m_sector_glon = gLon;
11237 }
11238 s57_DrawExtendedLightSectors(dc, VPoint, m_sectorlegsVisible);
11239 }
11240}
11241
11242void ChartCanvas::WarpPointerDeferred(int x, int y) {
11243 warp_x = x;
11244 warp_y = y;
11245 warp_flag = true;
11246}
11247
11248int s_msg;
11249
11250void ChartCanvas::UpdateCanvasS52PLIBConfig() {
11251 if (!ps52plib) return;
11252
11253 if (VPoint.b_quilt) { // quilted
11254 if (!m_pQuilt || !m_pQuilt->IsComposed()) return; // not ready
11255
11256 if (m_pQuilt->IsQuiltVector()) {
11257 if (ps52plib->GetStateHash() != m_s52StateHash) {
11258 UpdateS52State();
11259 m_s52StateHash = ps52plib->GetStateHash();
11260 }
11261 }
11262 } else {
11263 if (ps52plib->GetStateHash() != m_s52StateHash) {
11264 UpdateS52State();
11265 m_s52StateHash = ps52plib->GetStateHash();
11266 }
11267 }
11268
11269 // Plugin charts
11270 bool bSendPlibState = true;
11271 if (VPoint.b_quilt) { // quilted
11272 if (!m_pQuilt->DoesQuiltContainPlugins()) bSendPlibState = false;
11273 }
11274
11275 if (bSendPlibState) {
11276 wxJSONValue v;
11277 v[_T("OpenCPN Version Major")] = VERSION_MAJOR;
11278 v[_T("OpenCPN Version Minor")] = VERSION_MINOR;
11279 v[_T("OpenCPN Version Patch")] = VERSION_PATCH;
11280 v[_T("OpenCPN Version Date")] = VERSION_DATE;
11281 v[_T("OpenCPN Version Full")] = VERSION_FULL;
11282
11283 // S52PLIB state
11284 v[_T("OpenCPN S52PLIB ShowText")] = GetShowENCText();
11285 v[_T("OpenCPN S52PLIB ShowSoundings")] = GetShowENCDepth();
11286 v[_T("OpenCPN S52PLIB ShowLights")] = GetShowENCLights();
11287 v[_T("OpenCPN S52PLIB ShowAnchorConditions")] = m_encShowAnchor;
11288 v[_T("OpenCPN S52PLIB ShowQualityOfData")] = GetShowENCDataQual();
11289 v[_T("OpenCPN S52PLIB ShowATONLabel")] = GetShowENCBuoyLabels();
11290 v[_T("OpenCPN S52PLIB ShowLightDescription")] = GetShowENCLightDesc();
11291
11292 v[_T("OpenCPN S52PLIB DisplayCategory")] = GetENCDisplayCategory();
11293
11294 v[_T("OpenCPN S52PLIB SoundingsFactor")] = g_ENCSoundingScaleFactor;
11295 v[_T("OpenCPN S52PLIB TextFactor")] = g_ENCTextScaleFactor;
11296
11297 // Global S52 options
11298
11299 v[_T("OpenCPN S52PLIB MetaDisplay")] = ps52plib->m_bShowMeta;
11300 v[_T("OpenCPN S52PLIB DeclutterText")] = ps52plib->m_bDeClutterText;
11301 v[_T("OpenCPN S52PLIB ShowNationalText")] = ps52plib->m_bShowNationalTexts;
11302 v[_T("OpenCPN S52PLIB ShowImportantTextOnly")] =
11303 ps52plib->m_bShowS57ImportantTextOnly;
11304 v[_T("OpenCPN S52PLIB UseSCAMIN")] = ps52plib->m_bUseSCAMIN;
11305 v[_T("OpenCPN S52PLIB UseSUPER_SCAMIN")] = ps52plib->m_bUseSUPER_SCAMIN;
11306 v[_T("OpenCPN S52PLIB SymbolStyle")] = ps52plib->m_nSymbolStyle;
11307 v[_T("OpenCPN S52PLIB BoundaryStyle")] = ps52plib->m_nBoundaryStyle;
11308 v[_T("OpenCPN S52PLIB ColorShades")] =
11309 S52_getMarinerParam(S52_MAR_TWO_SHADES);
11310
11311 // Some global GUI parameters, for completeness
11312 v[_T("OpenCPN Zoom Mod Vector")] = g_chart_zoom_modifier_vector;
11313 v[_T("OpenCPN Zoom Mod Other")] = g_chart_zoom_modifier_raster;
11314 v[_T("OpenCPN Scale Factor Exp")] =
11315 g_Platform->GetChartScaleFactorExp(g_ChartScaleFactor);
11316 v[_T("OpenCPN Display Width")] = (int)g_display_size_mm;
11317
11318 wxJSONWriter w;
11319 wxString out;
11320 w.Write(v, out);
11321
11322 if (!g_lastS52PLIBPluginMessage.IsSameAs(out)) {
11323 SendMessageToAllPlugins(wxString("OpenCPN Config"), out);
11324 g_lastS52PLIBPluginMessage = out;
11325 }
11326 }
11327}
11328int spaint;
11329int s_in_update;
11330void ChartCanvas::OnPaint(wxPaintEvent &event) {
11331 wxPaintDC dc(this);
11332
11333 // GetToolbar()->Show( m_bToolbarEnable );
11334
11335 // Paint updates may have been externally disabled (temporarily, to avoid
11336 // Yield() recursion performance loss) It is important that the wxPaintDC is
11337 // built, even if we elect to not process this paint message. Otherwise, the
11338 // paint message may not be removed from the message queue, esp on Windows.
11339 // (FS#1213) This would lead to a deadlock condition in ::wxYield()
11340
11341 if (!m_b_paint_enable) {
11342 return;
11343 }
11344
11345 // If necessary, reconfigure the S52 PLIB
11346 UpdateCanvasS52PLIBConfig();
11347
11348#ifdef ocpnUSE_GL
11349 if (!g_bdisable_opengl && m_glcc) m_glcc->Show(g_bopengl);
11350
11351 if (m_glcc && g_bopengl) {
11352 if (!s_in_update) { // no recursion allowed, seen on lo-spec Mac
11353 s_in_update++;
11354 m_glcc->Update();
11355 s_in_update--;
11356 }
11357
11358 return;
11359 }
11360#endif
11361
11362 if ((GetVP().pix_width == 0) || (GetVP().pix_height == 0)) return;
11363
11364 wxRegion ru = GetUpdateRegion();
11365
11366 int rx, ry, rwidth, rheight;
11367 ru.GetBox(rx, ry, rwidth, rheight);
11368 // printf("%d Onpaint update region box: %d %d %d %d\n", spaint++, rx, ry,
11369 // rwidth, rheight);
11370
11371#ifdef ocpnUSE_DIBSECTION
11372 ocpnMemDC temp_dc;
11373#else
11374 wxMemoryDC temp_dc;
11375#endif
11376
11377 long height = GetVP().pix_height;
11378
11379#ifdef __WXMAC__
11380 // On OS X we have to explicitly extend the region for the piano area
11381 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
11382 if (!style->chartStatusWindowTransparent && g_bShowChartBar)
11383 height += m_Piano->GetHeight();
11384#endif // __WXMAC__
11385 wxRegion rgn_chart(0, 0, GetVP().pix_width, height);
11386
11387 // In case Thumbnail is shown, set up dc clipper and blt iterator regions
11388 if (pthumbwin) {
11389 int thumbx, thumby, thumbsx, thumbsy;
11390 pthumbwin->GetPosition(&thumbx, &thumby);
11391 pthumbwin->GetSize(&thumbsx, &thumbsy);
11392 wxRegion rgn_thumbwin(thumbx, thumby, thumbsx - 1, thumbsy - 1);
11393
11394 if (pthumbwin->IsShown()) {
11395 rgn_chart.Subtract(rgn_thumbwin);
11396 ru.Subtract(rgn_thumbwin);
11397 }
11398 }
11399
11400 // subtract the chart bar if it isn't transparent, and determine if we need to
11401 // paint it
11402 wxRegion rgn_blit = ru;
11403 if (g_bShowChartBar) {
11404 wxRect chart_bar_rect(0, GetClientSize().y - m_Piano->GetHeight(),
11405 GetClientSize().x, m_Piano->GetHeight());
11406
11407 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
11408 if (ru.Contains(chart_bar_rect) != wxOutRegion) {
11409 if (style->chartStatusWindowTransparent)
11410 m_brepaint_piano = true;
11411 else
11412 ru.Subtract(chart_bar_rect);
11413 }
11414 }
11415
11416 if (m_Compass && m_Compass->IsShown()) {
11417 wxRect compassRect = m_Compass->GetRect();
11418 if (ru.Contains(compassRect) != wxOutRegion) {
11419 ru.Subtract(compassRect);
11420 }
11421 }
11422
11423 // Is this viewpoint the same as the previously painted one?
11424 bool b_newview = true;
11425
11426 if ((m_cache_vp.view_scale_ppm == VPoint.view_scale_ppm) &&
11427 (m_cache_vp.rotation == VPoint.rotation) &&
11428 (m_cache_vp.clat == VPoint.clat) && (m_cache_vp.clon == VPoint.clon) &&
11429 m_cache_vp.IsValid()) {
11430 b_newview = false;
11431 }
11432
11433 // If the ViewPort is skewed or rotated, we may be able to use the cached
11434 // rotated bitmap.
11435 bool b_rcache_ok = false;
11436 if (fabs(VPoint.skew) > 0.01 || fabs(VPoint.rotation) > 0.01)
11437 b_rcache_ok = !b_newview;
11438
11439 // Make a special VP
11440 if (VPoint.b_MercatorProjectionOverride)
11441 VPoint.SetProjectionType(PROJECTION_MERCATOR);
11442 ViewPort svp = VPoint;
11443
11444 svp.pix_width = svp.rv_rect.width;
11445 svp.pix_height = svp.rv_rect.height;
11446
11447 // printf("Onpaint pix %d %d\n", VPoint.pix_width, VPoint.pix_height);
11448 // printf("OnPaint rv_rect %d %d\n", VPoint.rv_rect.width,
11449 // VPoint.rv_rect.height);
11450
11451 OCPNRegion chart_get_region(wxRect(0, 0, svp.pix_width, svp.pix_height));
11452
11453 // If we are going to use the cached rotated image, there is no need to fetch
11454 // any chart data and this will do it...
11455 if (b_rcache_ok) chart_get_region.Clear();
11456
11457 // Blit pan acceleration
11458 if (VPoint.b_quilt) // quilted
11459 {
11460 if (!m_pQuilt || !m_pQuilt->IsComposed()) return; // not ready
11461
11462 bool bvectorQuilt = m_pQuilt->IsQuiltVector();
11463
11464 bool busy = false;
11465 if (bvectorQuilt && (m_cache_vp.view_scale_ppm != VPoint.view_scale_ppm ||
11466 m_cache_vp.rotation != VPoint.rotation)) {
11467 AbstractPlatform::ShowBusySpinner();
11468 busy = true;
11469 }
11470
11471 if ((m_working_bm.GetWidth() != svp.pix_width) ||
11472 (m_working_bm.GetHeight() != svp.pix_height))
11473 m_working_bm.Create(svp.pix_width, svp.pix_height,
11474 -1); // make sure the target is big enoug
11475
11476 if (fabs(VPoint.rotation) < 0.01) {
11477 bool b_save = true;
11478
11479 if (g_SencThreadManager) {
11480 if (g_SencThreadManager->GetJobCount()) {
11481 b_save = false;
11482 m_cache_vp.Invalidate();
11483 }
11484 }
11485
11486 // If the saved wxBitmap from last OnPaint is useable
11487 // calculate the blit parameters
11488
11489 // We can only do screen blit painting if subsequent ViewPorts differ by
11490 // whole pixels So, in small scale bFollow mode, force the full screen
11491 // render. This seems a hack....There may be better logic here.....
11492
11493 // if(m_bFollow)
11494 // b_save = false;
11495
11496 if (m_bm_cache_vp.IsValid() && m_cache_vp.IsValid() /*&& !m_bFollow*/) {
11497 if (b_newview) {
11498 wxPoint c_old = VPoint.GetPixFromLL(VPoint.clat, VPoint.clon);
11499 wxPoint c_new = m_bm_cache_vp.GetPixFromLL(VPoint.clat, VPoint.clon);
11500
11501 int dy = c_new.y - c_old.y;
11502 int dx = c_new.x - c_old.x;
11503
11504 // printf("In OnPaint Trying Blit dx: %d
11505 // dy:%d\n\n", dx, dy);
11506
11507 if (m_pQuilt->IsVPBlittable(VPoint, dx, dy, true)) {
11508 if (dx || dy) {
11509 // Blit the reuseable portion of the cached wxBitmap to a working
11510 // bitmap
11511 temp_dc.SelectObject(m_working_bm);
11512
11513 wxMemoryDC cache_dc;
11514 cache_dc.SelectObject(m_cached_chart_bm);
11515
11516 if (dy > 0) {
11517 if (dx > 0) {
11518 temp_dc.Blit(0, 0, VPoint.pix_width - dx,
11519 VPoint.pix_height - dy, &cache_dc, dx, dy);
11520 } else {
11521 temp_dc.Blit(-dx, 0, VPoint.pix_width + dx,
11522 VPoint.pix_height - dy, &cache_dc, 0, dy);
11523 }
11524
11525 } else {
11526 if (dx > 0) {
11527 temp_dc.Blit(0, -dy, VPoint.pix_width - dx,
11528 VPoint.pix_height + dy, &cache_dc, dx, 0);
11529 } else {
11530 temp_dc.Blit(-dx, -dy, VPoint.pix_width + dx,
11531 VPoint.pix_height + dy, &cache_dc, 0, 0);
11532 }
11533 }
11534
11535 OCPNRegion update_region;
11536 if (dy) {
11537 if (dy > 0)
11538 update_region.Union(
11539 wxRect(0, VPoint.pix_height - dy, VPoint.pix_width, dy));
11540 else
11541 update_region.Union(wxRect(0, 0, VPoint.pix_width, -dy));
11542 }
11543
11544 if (dx) {
11545 if (dx > 0)
11546 update_region.Union(
11547 wxRect(VPoint.pix_width - dx, 0, dx, VPoint.pix_height));
11548 else
11549 update_region.Union(wxRect(0, 0, -dx, VPoint.pix_height));
11550 }
11551
11552 // Render the new region
11553 m_pQuilt->RenderQuiltRegionViewOnDCNoText(temp_dc, svp,
11554 update_region);
11555 cache_dc.SelectObject(wxNullBitmap);
11556 } else {
11557 // No sensible (dx, dy) change in the view, so use the cached
11558 // member bitmap
11559 temp_dc.SelectObject(m_cached_chart_bm);
11560 b_save = false;
11561 }
11562 m_pQuilt->ComputeRenderRegion(svp, chart_get_region);
11563
11564 } else // not blitable
11565 {
11566 temp_dc.SelectObject(m_working_bm);
11567 m_pQuilt->RenderQuiltRegionViewOnDCNoText(temp_dc, svp,
11568 chart_get_region);
11569 }
11570 } else {
11571 // No change in the view, so use the cached member bitmap2
11572 temp_dc.SelectObject(m_cached_chart_bm);
11573 b_save = false;
11574 }
11575 } else // cached bitmap is not yet valid
11576 {
11577 temp_dc.SelectObject(m_working_bm);
11578 m_pQuilt->RenderQuiltRegionViewOnDCNoText(temp_dc, svp,
11579 chart_get_region);
11580 }
11581
11582 // Save the fully rendered quilt image as a wxBitmap member of this class
11583 if (b_save) {
11584 // if((m_cached_chart_bm.GetWidth() !=
11585 // svp.pix_width) ||
11586 // (m_cached_chart_bm.GetHeight() !=
11587 // svp.pix_height))
11588 // m_cached_chart_bm.Create(svp.pix_width,
11589 // svp.pix_height, -1); // target wxBitmap
11590 // is big enough
11591 wxMemoryDC scratch_dc_0;
11592 scratch_dc_0.SelectObject(m_cached_chart_bm);
11593 scratch_dc_0.Blit(0, 0, svp.pix_width, svp.pix_height, &temp_dc, 0, 0);
11594
11595 scratch_dc_0.SelectObject(wxNullBitmap);
11596
11597 m_bm_cache_vp =
11598 VPoint; // save the ViewPort associated with the cached wxBitmap
11599 }
11600 }
11601
11602 else // quilted, rotated
11603 {
11604 temp_dc.SelectObject(m_working_bm);
11605 OCPNRegion chart_get_all_region(
11606 wxRect(0, 0, svp.pix_width, svp.pix_height));
11607 m_pQuilt->RenderQuiltRegionViewOnDCNoText(temp_dc, svp,
11608 chart_get_all_region);
11609 }
11610
11611 AbstractPlatform::HideBusySpinner();
11612
11613 }
11614
11615 else // not quilted
11616 {
11617 if (!m_singleChart) {
11618 dc.SetBackground(wxBrush(*wxLIGHT_GREY));
11619 dc.Clear();
11620 return;
11621 }
11622
11623 if (!chart_get_region.IsEmpty()) {
11624 m_singleChart->RenderRegionViewOnDC(temp_dc, svp, chart_get_region);
11625 }
11626 }
11627
11628 if (temp_dc.IsOk()) {
11629 // Arrange to render the World Chart vector data behind the rendered
11630 // current chart so that uncovered canvas areas show at least the world
11631 // chart.
11632 OCPNRegion chartValidRegion;
11633 if (!VPoint.b_quilt) {
11634 // Make a region covering the current chart on the canvas
11635
11636 if (m_singleChart->GetChartFamily() == CHART_FAMILY_VECTOR)
11637 m_singleChart->GetValidCanvasRegion(svp, &chartValidRegion);
11638 else {
11639 // The raster calculations in ChartBaseBSB::ComputeSourceRectangle
11640 // require that the viewport passed here have pix_width and pix_height
11641 // set to the actual display, not the virtual (rv_rect) sizes
11642 // (the vector calculations require the virtual sizes in svp)
11643
11644 m_singleChart->GetValidCanvasRegion(VPoint, &chartValidRegion);
11645 chartValidRegion.Offset(-VPoint.rv_rect.x, -VPoint.rv_rect.y);
11646 }
11647 } else
11648 chartValidRegion = m_pQuilt->GetFullQuiltRenderedRegion();
11649
11650 temp_dc.DestroyClippingRegion();
11651
11652 // Copy current chart region
11653 OCPNRegion backgroundRegion(wxRect(0, 0, svp.pix_width, svp.pix_height));
11654
11655 if (chartValidRegion.IsOk()) backgroundRegion.Subtract(chartValidRegion);
11656
11657 if (!backgroundRegion.IsEmpty()) {
11658 // Draw the Background Chart only in the areas NOT covered by the
11659 // current chart view
11660
11661 /* unfortunately wxDC::DrawRectangle and wxDC::Clear do not respect
11662 clipping regions with more than 1 rectangle so... */
11663 wxColour water = pWorldBackgroundChart->water;
11664 if (water.IsOk()) {
11665 temp_dc.SetPen(*wxTRANSPARENT_PEN);
11666 temp_dc.SetBrush(wxBrush(water));
11667 OCPNRegionIterator upd(backgroundRegion); // get the update rect list
11668 while (upd.HaveRects()) {
11669 wxRect rect = upd.GetRect();
11670 temp_dc.DrawRectangle(rect);
11671 upd.NextRect();
11672 }
11673 }
11674 // Associate with temp_dc
11675 wxRegion *clip_region = backgroundRegion.GetNew_wxRegion();
11676 temp_dc.SetDeviceClippingRegion(*clip_region);
11677 delete clip_region;
11678
11679 ocpnDC bgdc(temp_dc);
11680 double r = VPoint.rotation;
11681 SetVPRotation(VPoint.skew);
11682
11683 // pWorldBackgroundChart->RenderViewOnDC(bgdc, VPoint);
11684 gShapeBasemap.RenderViewOnDC(bgdc, VPoint);
11685
11686 SetVPRotation(r);
11687 }
11688 } // temp_dc.IsOk();
11689
11690 wxMemoryDC *pChartDC = &temp_dc;
11691 wxMemoryDC rotd_dc;
11692
11693 if (((fabs(GetVP().rotation) > 0.01)) || (fabs(GetVP().skew) > 0.01)) {
11694 // Can we use the current rotated image cache?
11695 if (!b_rcache_ok) {
11696#ifdef __WXMSW__
11697 wxMemoryDC tbase_dc;
11698 wxBitmap bm_base(svp.pix_width, svp.pix_height, -1);
11699 tbase_dc.SelectObject(bm_base);
11700 tbase_dc.Blit(0, 0, svp.pix_width, svp.pix_height, &temp_dc, 0, 0);
11701 tbase_dc.SelectObject(wxNullBitmap);
11702#else
11703 const wxBitmap &bm_base = temp_dc.GetSelectedBitmap();
11704#endif
11705
11706 wxImage base_image;
11707 if (bm_base.IsOk()) base_image = bm_base.ConvertToImage();
11708
11709 // Use a local static image rotator to improve wxWidgets code profile
11710 // Especially, on GTK the wxRound and wxRealPoint functions are very
11711 // expensive.....
11712
11713 double angle = GetVP().skew - GetVP().rotation;
11714 wxImage ri;
11715 bool b_rot_ok = false;
11716 if (base_image.IsOk()) {
11717 ViewPort rot_vp = GetVP();
11718
11719 m_b_rot_hidef = false;
11720
11721 ri = Image_Rotate(
11722 base_image, angle,
11723 wxPoint(GetVP().rv_rect.width / 2, GetVP().rv_rect.height / 2),
11724 m_b_rot_hidef, &m_roffset);
11725
11726 if ((rot_vp.view_scale_ppm == VPoint.view_scale_ppm) &&
11727 (rot_vp.rotation == VPoint.rotation) &&
11728 (rot_vp.clat == VPoint.clat) && (rot_vp.clon == VPoint.clon) &&
11729 rot_vp.IsValid() && (ri.IsOk())) {
11730 b_rot_ok = true;
11731 }
11732 }
11733
11734 if (b_rot_ok) {
11735 delete m_prot_bm;
11736 m_prot_bm = new wxBitmap(ri);
11737 }
11738
11739 m_roffset.x += VPoint.rv_rect.x;
11740 m_roffset.y += VPoint.rv_rect.y;
11741 }
11742
11743 if (m_prot_bm && m_prot_bm->IsOk()) {
11744 rotd_dc.SelectObject(*m_prot_bm);
11745 pChartDC = &rotd_dc;
11746 } else {
11747 pChartDC = &temp_dc;
11748 m_roffset = wxPoint(0, 0);
11749 }
11750 } else { // unrotated
11751 pChartDC = &temp_dc;
11752 m_roffset = wxPoint(0, 0);
11753 }
11754
11755 wxPoint offset = m_roffset;
11756
11757 // Save the PixelCache viewpoint for next time
11758 m_cache_vp = VPoint;
11759
11760 // Set up a scratch DC for overlay objects
11761 wxMemoryDC mscratch_dc;
11762 mscratch_dc.SelectObject(*pscratch_bm);
11763
11764 mscratch_dc.ResetBoundingBox();
11765 mscratch_dc.DestroyClippingRegion();
11766 mscratch_dc.SetDeviceClippingRegion(rgn_chart);
11767
11768 // Blit the externally invalidated areas of the chart onto the scratch dc
11769 wxRegionIterator upd(rgn_blit); // get the update rect list
11770 while (upd) {
11771 wxRect rect = upd.GetRect();
11772
11773 mscratch_dc.Blit(rect.x, rect.y, rect.width, rect.height, pChartDC,
11774 rect.x - offset.x, rect.y - offset.y);
11775 upd++;
11776 }
11777
11778 // If multi-canvas, indicate which canvas has keyboard focus
11779 // by drawing a simple blue bar at the top.
11780 if (m_show_focus_bar && (g_canvasConfig != 0)) { // multi-canvas?
11781 if (this == wxWindow::FindFocus()) {
11782 g_focusCanvas = this;
11783
11784 wxColour colour = GetGlobalColor(_T("BLUE4"));
11785 mscratch_dc.SetPen(wxPen(colour));
11786 mscratch_dc.SetBrush(wxBrush(colour));
11787
11788 wxRect activeRect(0, 0, GetClientSize().x, m_focus_indicator_pix);
11789 mscratch_dc.DrawRectangle(activeRect);
11790 }
11791 }
11792
11793 // Any MBtiles?
11794 std::vector<int> stackIndexArray = m_pQuilt->GetExtendedStackIndexArray();
11795 unsigned int im = stackIndexArray.size();
11796 if (VPoint.b_quilt && im > 0) {
11797 std::vector<int> tiles_to_show;
11798 for (unsigned int is = 0; is < im; is++) {
11799 const ChartTableEntry &cte =
11800 ChartData->GetChartTableEntry(stackIndexArray[is]);
11801 if (IsTileOverlayIndexInNoShow(stackIndexArray[is])) {
11802 continue;
11803 }
11804 if (cte.GetChartType() == CHART_TYPE_MBTILES) {
11805 tiles_to_show.push_back(stackIndexArray[is]);
11806 }
11807 }
11808
11809 if (tiles_to_show.size())
11810 SetAlertString(_("MBTile requires OpenGL to be enabled"));
11811 }
11812
11813 // May get an unexpected OnPaint call while switching display modes
11814 // Guard for that.
11815 if (!g_bopengl) {
11816 // Draw the rest of the overlay objects directly on the scratch dc
11817 ocpnDC scratch_dc(mscratch_dc);
11818 DrawOverlayObjects(scratch_dc, ru);
11819
11820 if (m_bShowTide) {
11821 RebuildTideSelectList(GetVP().GetBBox());
11822 DrawAllTidesInBBox(scratch_dc, GetVP().GetBBox());
11823 }
11824
11825 if (m_bShowCurrent) {
11826 RebuildCurrentSelectList(GetVP().GetBBox());
11827 DrawAllCurrentsInBBox(scratch_dc, GetVP().GetBBox());
11828 }
11829
11830 if (m_brepaint_piano && g_bShowChartBar) {
11831 m_Piano->Paint(GetClientSize().y - m_Piano->GetHeight(), mscratch_dc);
11832 }
11833
11834 if (m_Compass) m_Compass->Paint(scratch_dc);
11835
11836 RenderAlertMessage(mscratch_dc, GetVP());
11837 }
11838
11839 // quiting?
11840 if (g_bquiting) {
11841#ifdef ocpnUSE_DIBSECTION
11842 ocpnMemDC q_dc;
11843#else
11844 wxMemoryDC q_dc;
11845#endif
11846 wxBitmap qbm(GetVP().pix_width, GetVP().pix_height);
11847 q_dc.SelectObject(qbm);
11848
11849 // Get a copy of the screen
11850 q_dc.Blit(0, 0, GetVP().pix_width, GetVP().pix_height, &mscratch_dc, 0, 0);
11851
11852 // Draw a rectangle over the screen with a stipple brush
11853 wxBrush qbr(*wxBLACK, wxBRUSHSTYLE_FDIAGONAL_HATCH);
11854 q_dc.SetBrush(qbr);
11855 q_dc.DrawRectangle(0, 0, GetVP().pix_width, GetVP().pix_height);
11856
11857 // Blit back into source
11858 mscratch_dc.Blit(0, 0, GetVP().pix_width, GetVP().pix_height, &q_dc, 0, 0,
11859 wxCOPY);
11860
11861 q_dc.SelectObject(wxNullBitmap);
11862 }
11863#if 0
11864 // It is possible that this two-step method may be reuired for some platforms.
11865 // So, retain in the code base to aid recovery if necessary
11866
11867 // Create and Render the Vector quilt decluttered text overlay, omitting CM93 composite
11868 if( VPoint.b_quilt ) {
11869 if(m_pQuilt->IsQuiltVector() && ps52plib && ps52plib->GetShowS57Text()){
11870 ChartBase *chart = m_pQuilt->GetRefChart();
11871 if( chart && chart->GetChartType() != CHART_TYPE_CM93COMP){
11872
11873 // Clear the text Global declutter list
11874 ChartPlugInWrapper *ChPI = dynamic_cast<ChartPlugInWrapper*>( chart );
11875 if(ChPI)
11876 ChPI->ClearPLIBTextList();
11877 else{
11878 if(ps52plib)
11879 ps52plib->ClearTextList();
11880 }
11881
11882 wxMemoryDC t_dc;
11883 wxBitmap qbm( GetVP().pix_width, GetVP().pix_height );
11884
11885 wxColor maskBackground = wxColour(1,0,0);
11886 t_dc.SelectObject( qbm );
11887 t_dc.SetBackground(wxBrush(maskBackground));
11888 t_dc.Clear();
11889
11890 // Copy the scratch DC into the new bitmap
11891 t_dc.Blit( 0, 0, GetVP().pix_width, GetVP().pix_height, scratch_dc.GetDC(), 0, 0, wxCOPY );
11892
11893 // Render the text to the new bitmap
11894 OCPNRegion chart_all_text_region( wxRect( 0, 0, GetVP().pix_width, GetVP().pix_height ) );
11895 m_pQuilt->RenderQuiltRegionViewOnDCTextOnly( t_dc, svp, chart_all_text_region );
11896
11897 // Copy the new bitmap back to the scratch dc
11898 wxRegionIterator upd_final( ru );
11899 while( upd_final ) {
11900 wxRect rect = upd_final.GetRect();
11901 scratch_dc.GetDC()->Blit( rect.x, rect.y, rect.width, rect.height, &t_dc, rect.x, rect.y, wxCOPY, true );
11902 upd_final++;
11903 }
11904
11905 t_dc.SelectObject( wxNullBitmap );
11906 }
11907 }
11908 }
11909#endif
11910 // Direct rendering model...
11911 if (VPoint.b_quilt) {
11912 if (m_pQuilt->IsQuiltVector() && ps52plib && ps52plib->GetShowS57Text()) {
11913 ChartBase *chart = m_pQuilt->GetRefChart();
11914 if (chart && chart->GetChartType() != CHART_TYPE_CM93COMP) {
11915 // Clear the text Global declutter list
11916 ChartPlugInWrapper *ChPI = dynamic_cast<ChartPlugInWrapper *>(chart);
11917 if (ChPI)
11918 ChPI->ClearPLIBTextList();
11919 else {
11920 if (ps52plib) ps52plib->ClearTextList();
11921 }
11922
11923 // Render the text directly to the scratch bitmap
11924 OCPNRegion chart_all_text_region(
11925 wxRect(0, 0, GetVP().pix_width, GetVP().pix_height));
11926
11927 if (g_bShowChartBar && m_Piano) {
11928 wxRect chart_bar_rect(0, GetVP().pix_height - m_Piano->GetHeight(),
11929 GetVP().pix_width, m_Piano->GetHeight());
11930
11931 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
11932 if (!style->chartStatusWindowTransparent)
11933 chart_all_text_region.Subtract(chart_bar_rect);
11934 }
11935
11936 if (m_Compass && m_Compass->IsShown()) {
11937 wxRect compassRect = m_Compass->GetRect();
11938 if (chart_all_text_region.Contains(compassRect) != wxOutRegion) {
11939 chart_all_text_region.Subtract(compassRect);
11940 }
11941 }
11942
11943 mscratch_dc.DestroyClippingRegion();
11944
11945 m_pQuilt->RenderQuiltRegionViewOnDCTextOnly(mscratch_dc, svp,
11946 chart_all_text_region);
11947 }
11948 }
11949 }
11950
11951 // And finally, blit the scratch dc onto the physical dc
11952 wxRegionIterator upd_final(rgn_blit);
11953 while (upd_final) {
11954 wxRect rect = upd_final.GetRect();
11955 dc.Blit(rect.x, rect.y, rect.width, rect.height, &mscratch_dc, rect.x,
11956 rect.y);
11957 upd_final++;
11958 }
11959
11960 // Test code to validate the dc drawing rectangle....
11961 /*
11962 wxRegionIterator upd_ru ( rgn_blit ); // get the update rect list
11963 while ( upd_ru )
11964 {
11965 wxRect rect = upd_ru.GetRect();
11966
11967 dc.SetPen(wxPen(*wxRED));
11968 dc.SetBrush(wxBrush(*wxRED, wxTRANSPARENT));
11969 dc.DrawRectangle(rect);
11970 upd_ru ++ ;
11971 }
11972 */
11973
11974 // Deselect the chart bitmap from the temp_dc, so that it will not be
11975 // destroyed in the temp_dc dtor
11976 temp_dc.SelectObject(wxNullBitmap);
11977 // And for the scratch bitmap
11978 mscratch_dc.SelectObject(wxNullBitmap);
11979
11980 dc.DestroyClippingRegion();
11981
11982 PaintCleanup();
11983}
11984
11985void ChartCanvas::PaintCleanup() {
11986 // Handle the current graphic window, if present
11987
11988 if (pCwin) {
11989 pCwin->Show();
11990 if (m_bTCupdate) {
11991 pCwin->Refresh();
11992 pCwin->Update();
11993 }
11994 }
11995
11996 // And set flags for next time
11997 m_bTCupdate = false;
11998
11999 // Handle deferred WarpPointer
12000 if (warp_flag) {
12001 WarpPointer(warp_x, warp_y);
12002 warp_flag = false;
12003 }
12004
12005 // Start movement timers, this runs nearly immediately.
12006 // the reason we cannot simply call it directly is the
12007 // refresh events it emits may be blocked from this paint event
12008 pMovementTimer->Start(1, wxTIMER_ONE_SHOT);
12009 m_VPMovementTimer.Start(1, wxTIMER_ONE_SHOT);
12010}
12011
12012#if 0
12013wxColour GetErrorGraphicColor(double val)
12014{
12015 /*
12016 double valm = wxMin(val_max, val);
12017
12018 unsigned char green = (unsigned char)(255 * (1 - (valm/val_max)));
12019 unsigned char red = (unsigned char)(255 * (valm/val_max));
12020
12021 wxImage::HSVValue hv = wxImage::RGBtoHSV(wxImage::RGBValue(red, green, 0));
12022
12023 hv.saturation = 1.0;
12024 hv.value = 1.0;
12025
12026 wxImage::RGBValue rv = wxImage::HSVtoRGB(hv);
12027 return wxColour(rv.red, rv.green, rv.blue);
12028 */
12029
12030 // HTML colors taken from NOAA WW3 Web representation
12031 wxColour c;
12032 if((val > 0) && (val < 1)) c.Set(_T("#002ad9"));
12033 else if((val >= 1) && (val < 2)) c.Set(_T("#006ed9"));
12034 else if((val >= 2) && (val < 3)) c.Set(_T("#00b2d9"));
12035 else if((val >= 3) && (val < 4)) c.Set(_T("#00d4d4"));
12036 else if((val >= 4) && (val < 5)) c.Set(_T("#00d9a6"));
12037 else if((val >= 5) && (val < 7)) c.Set(_T("#00d900"));
12038 else if((val >= 7) && (val < 9)) c.Set(_T("#95d900"));
12039 else if((val >= 9) && (val < 12)) c.Set(_T("#d9d900"));
12040 else if((val >= 12) && (val < 15)) c.Set(_T("#d9ae00"));
12041 else if((val >= 15) && (val < 18)) c.Set(_T("#d98300"));
12042 else if((val >= 18) && (val < 21)) c.Set(_T("#d95700"));
12043 else if((val >= 21) && (val < 24)) c.Set(_T("#d90000"));
12044 else if((val >= 24) && (val < 27)) c.Set(_T("#ae0000"));
12045 else if((val >= 27) && (val < 30)) c.Set(_T("#8c0000"));
12046 else if((val >= 30) && (val < 36)) c.Set(_T("#870000"));
12047 else if((val >= 36) && (val < 42)) c.Set(_T("#690000"));
12048 else if((val >= 42) && (val < 48)) c.Set(_T("#550000"));
12049 else if( val >= 48) c.Set(_T("#410000"));
12050
12051 return c;
12052}
12053
12054void ChartCanvas::RenderGeorefErrorMap( wxMemoryDC *pmdc, ViewPort *vp)
12055{
12056 wxImage gr_image(vp->pix_width, vp->pix_height);
12057 gr_image.InitAlpha();
12058
12059 double maxval = -10000;
12060 double minval = 10000;
12061
12062 double rlat, rlon;
12063 double glat, glon;
12064
12065 GetCanvasPixPoint(0, 0, rlat, rlon);
12066
12067 for(int i=1; i < vp->pix_height-1; i++)
12068 {
12069 for(int j=0; j < vp->pix_width; j++)
12070 {
12071 // Reference mercator value
12072// vp->GetMercatorLLFromPix(wxPoint(j, i), &rlat, &rlon);
12073
12074 // Georef value
12075 GetCanvasPixPoint(j, i, glat, glon);
12076
12077 maxval = wxMax(maxval, (glat - rlat));
12078 minval = wxMin(minval, (glat - rlat));
12079
12080 }
12081 rlat = glat;
12082 }
12083
12084 GetCanvasPixPoint(0, 0, rlat, rlon);
12085 for(int i=1; i < vp->pix_height-1; i++)
12086 {
12087 for(int j=0; j < vp->pix_width; j++)
12088 {
12089 // Reference mercator value
12090// vp->GetMercatorLLFromPix(wxPoint(j, i), &rlat, &rlon);
12091
12092 // Georef value
12093 GetCanvasPixPoint(j, i, glat, glon);
12094
12095 double f = ((glat - rlat)-minval)/(maxval - minval);
12096
12097 double dy = (f * 40);
12098
12099 wxColour c = GetErrorGraphicColor(dy);
12100 unsigned char r = c.Red();
12101 unsigned char g = c.Green();
12102 unsigned char b = c.Blue();
12103
12104 gr_image.SetRGB(j, i, r,g,b);
12105 if((glat - rlat )!= 0)
12106 gr_image.SetAlpha(j, i, 128);
12107 else
12108 gr_image.SetAlpha(j, i, 255);
12109
12110 }
12111 rlat = glat;
12112 }
12113
12114 // Create a Bitmap
12115 wxBitmap *pbm = new wxBitmap(gr_image);
12116 wxMask *gr_mask = new wxMask(*pbm, wxColour(0,0,0));
12117 pbm->SetMask(gr_mask);
12118
12119 pmdc->DrawBitmap(*pbm, 0,0);
12120
12121 delete pbm;
12122
12123}
12124
12125#endif
12126
12127void ChartCanvas::CancelMouseRoute() {
12128 m_routeState = 0;
12129 m_pMouseRoute = NULL;
12130 m_bDrawingRoute = false;
12131}
12132
12133int ChartCanvas::GetNextContextMenuId() {
12134 return CanvasMenuHandler::GetNextContextMenuId();
12135}
12136
12137bool ChartCanvas::SetCursor(const wxCursor &c) {
12138#ifdef ocpnUSE_GL
12139 if (g_bopengl && m_glcc)
12140 return m_glcc->SetCursor(c);
12141 else
12142#endif
12143 return wxWindow::SetCursor(c);
12144}
12145
12146void ChartCanvas::Refresh(bool eraseBackground, const wxRect *rect) {
12147 if (g_bquiting) return;
12148 // Keep the mouse position members up to date
12149 GetCanvasPixPoint(mouse_x, mouse_y, m_cursor_lat, m_cursor_lon);
12150
12151 // Retrigger the route leg popup timer
12152 // This handles the case when the chart is moving in auto-follow mode,
12153 // but no user mouse input is made. The timer handler may Hide() the
12154 // popup if the chart moved enough n.b. We use slightly longer oneshot
12155 // value to allow this method's Refresh() to complete before potentially
12156 // getting another Refresh() in the popup timer handler.
12157 if (!m_RolloverPopupTimer.IsRunning() &&
12158 ((m_pRouteRolloverWin && m_pRouteRolloverWin->IsActive()) ||
12159 (m_pTrackRolloverWin && m_pTrackRolloverWin->IsActive()) ||
12160 (m_pAISRolloverWin && m_pAISRolloverWin->IsActive())))
12161 m_RolloverPopupTimer.Start(500, wxTIMER_ONE_SHOT);
12162
12163#ifdef ocpnUSE_GL
12164 if (m_glcc && g_bopengl) {
12165 // We need to invalidate the FBO cache to ensure repaint of "grounded"
12166 // overlay objects.
12167 if (eraseBackground && m_glcc->UsingFBO()) m_glcc->Invalidate();
12168
12169 m_glcc->Refresh(eraseBackground,
12170 NULL); // We always are going to render the entire screen
12171 // anyway, so make
12172 // sure that the window managers understand the invalid area
12173 // is actually the entire client area.
12174
12175 // We need to selectively Refresh some child windows, if they are visible.
12176 // Note that some children are refreshed elsewhere on timer ticks, so don't
12177 // need attention here.
12178
12179 // Thumbnail chart
12180 if (pthumbwin && pthumbwin->IsShown()) {
12181 pthumbwin->Raise();
12182 pthumbwin->Refresh(false);
12183 }
12184
12185 // ChartInfo window
12186 if (m_pCIWin && m_pCIWin->IsShown()) {
12187 m_pCIWin->Raise();
12188 m_pCIWin->Refresh(false);
12189 }
12190
12191 // if(g_MainToolbar)
12192 // g_MainToolbar->UpdateRecoveryWindow(g_bshowToolbar);
12193
12194 } else
12195#endif
12196 wxWindow::Refresh(eraseBackground, rect);
12197}
12198
12199void ChartCanvas::Update() {
12200 if (m_glcc && g_bopengl) {
12201#ifdef ocpnUSE_GL
12202 m_glcc->Update();
12203#endif
12204 } else
12205 wxWindow::Update();
12206}
12207
12208void ChartCanvas::DrawEmboss(ocpnDC &dc, emboss_data *pemboss) {
12209 if (!pemboss) return;
12210 int x = pemboss->x, y = pemboss->y;
12211 const double factor = 200;
12212
12213 wxASSERT_MSG(dc.GetDC(), wxT("DrawEmboss has no dc (opengl?)"));
12214 wxMemoryDC *pmdc = dynamic_cast<wxMemoryDC *>(dc.GetDC());
12215 wxASSERT_MSG(pmdc, wxT("dc to EmbossCanvas not a memory dc"));
12216
12217 // Grab a snipped image out of the chart
12218 wxMemoryDC snip_dc;
12219 wxBitmap snip_bmp(pemboss->width, pemboss->height, -1);
12220 snip_dc.SelectObject(snip_bmp);
12221
12222 snip_dc.Blit(0, 0, pemboss->width, pemboss->height, pmdc, x, y);
12223 snip_dc.SelectObject(wxNullBitmap);
12224
12225 wxImage snip_img = snip_bmp.ConvertToImage();
12226
12227 // Apply Emboss map to the snip image
12228 unsigned char *pdata = snip_img.GetData();
12229 if (pdata) {
12230 for (int y = 0; y < pemboss->height; y++) {
12231 int map_index = (y * pemboss->width);
12232 for (int x = 0; x < pemboss->width; x++) {
12233 double val = (pemboss->pmap[map_index] * factor) / 256.;
12234
12235 int nred = (int)((*pdata) + val);
12236 nred = nred > 255 ? 255 : (nred < 0 ? 0 : nred);
12237 *pdata++ = (unsigned char)nred;
12238
12239 int ngreen = (int)((*pdata) + val);
12240 ngreen = ngreen > 255 ? 255 : (ngreen < 0 ? 0 : ngreen);
12241 *pdata++ = (unsigned char)ngreen;
12242
12243 int nblue = (int)((*pdata) + val);
12244 nblue = nblue > 255 ? 255 : (nblue < 0 ? 0 : nblue);
12245 *pdata++ = (unsigned char)nblue;
12246
12247 map_index++;
12248 }
12249 }
12250 }
12251
12252 // Convert embossed snip to a bitmap
12253 wxBitmap emb_bmp(snip_img);
12254
12255 // Map to another memoryDC
12256 wxMemoryDC result_dc;
12257 result_dc.SelectObject(emb_bmp);
12258
12259 // Blit to target
12260 pmdc->Blit(x, y, pemboss->width, pemboss->height, &result_dc, 0, 0);
12261
12262 result_dc.SelectObject(wxNullBitmap);
12263}
12264
12265emboss_data *ChartCanvas::EmbossOverzoomIndicator(ocpnDC &dc) {
12266 double zoom_factor = GetVP().ref_scale / GetVP().chart_scale;
12267
12268 if (GetQuiltMode()) {
12269 // disable Overzoom indicator for MBTiles
12270 int refIndex = GetQuiltRefChartdbIndex();
12271 if (refIndex >= 0) {
12272 const ChartTableEntry &cte = ChartData->GetChartTableEntry(refIndex);
12273 ChartTypeEnum current_type = (ChartTypeEnum)cte.GetChartType();
12274 if (current_type == CHART_TYPE_MBTILES) {
12275 ChartBase *pChart = m_pQuilt->GetRefChart();
12276 ChartMbTiles *ptc = dynamic_cast<ChartMbTiles *>(pChart);
12277 if (ptc) {
12278 zoom_factor = ptc->GetZoomFactor();
12279 }
12280 }
12281 }
12282
12283 if (zoom_factor <= 3.9) return NULL;
12284 } else {
12285 if (m_singleChart) {
12286 if (zoom_factor <= 3.9) return NULL;
12287 } else
12288 return NULL;
12289 }
12290
12291 if (m_pEM_OverZoom) {
12292 m_pEM_OverZoom->x = 4;
12293 m_pEM_OverZoom->y = 0;
12294 if (g_MainToolbar && IsPrimaryCanvas()) {
12295 wxRect masterToolbarRect = g_MainToolbar->GetToolbarRect();
12296 m_pEM_OverZoom->x = masterToolbarRect.width + 4;
12297 }
12298 }
12299 return m_pEM_OverZoom;
12300}
12301
12302void ChartCanvas::DrawOverlayObjects(ocpnDC &dc, const wxRegion &ru) {
12303 GridDraw(dc);
12304
12305 // bool pluginOverlayRender = true;
12306 //
12307 // if(g_canvasConfig > 0){ // Multi canvas
12308 // if(IsPrimaryCanvas())
12309 // pluginOverlayRender = false;
12310 // }
12311
12312 g_overlayCanvas = this;
12313
12314 if (g_pi_manager) {
12315 g_pi_manager->SendViewPortToRequestingPlugIns(GetVP());
12316 g_pi_manager->RenderAllCanvasOverlayPlugIns(dc, GetVP(), m_canvasIndex,
12317 OVERLAY_LEGACY);
12318 }
12319
12320 AISDrawAreaNotices(dc, GetVP(), this);
12321
12322 wxDC *pdc = dc.GetDC();
12323 if (pdc) {
12324 pdc->DestroyClippingRegion();
12325 wxDCClipper(*pdc, ru);
12326 }
12327
12328 if (m_bShowNavobjects) {
12329 DrawAllTracksInBBox(dc, GetVP().GetBBox());
12330 DrawAllRoutesInBBox(dc, GetVP().GetBBox());
12331 DrawAllWaypointsInBBox(dc, GetVP().GetBBox());
12332 DrawAnchorWatchPoints(dc);
12333 } else {
12334 DrawActiveTrackInBBox(dc, GetVP().GetBBox());
12335 DrawActiveRouteInBBox(dc, GetVP().GetBBox());
12336 }
12337
12338 AISDraw(dc, GetVP(), this);
12339 ShipDraw(dc);
12340 AlertDraw(dc);
12341
12342 RenderVisibleSectorLights(dc);
12343
12344 RenderAllChartOutlines(dc, GetVP());
12345 RenderRouteLegs(dc);
12346 RenderShipToActive(dc, false);
12347 ScaleBarDraw(dc);
12348 s57_DrawExtendedLightSectors(dc, VPoint, extendedSectorLegs);
12349 if (g_pi_manager) {
12350 g_pi_manager->RenderAllCanvasOverlayPlugIns(dc, GetVP(), m_canvasIndex,
12351 OVERLAY_OVER_SHIPS);
12352 }
12353
12354 DrawEmboss(dc, EmbossDepthScale());
12355 DrawEmboss(dc, EmbossOverzoomIndicator(dc));
12356 if (g_pi_manager) {
12357 g_pi_manager->RenderAllCanvasOverlayPlugIns(dc, GetVP(), m_canvasIndex,
12358 OVERLAY_OVER_EMBOSS);
12359 }
12360 if (!g_PrintingInProgress) {
12361 if (IsPrimaryCanvas()) {
12362 if (g_MainToolbar) g_MainToolbar->DrawDC(dc, 1.0);
12363 }
12364
12365 if (IsPrimaryCanvas()) {
12366 if (g_iENCToolbar) g_iENCToolbar->DrawDC(dc, 1.0);
12367 }
12368
12369 if (m_muiBar) m_muiBar->DrawDC(dc, 1.0);
12370
12371 if (m_pTrackRolloverWin) {
12372 m_pTrackRolloverWin->Draw(dc);
12373 m_brepaint_piano = true;
12374 }
12375
12376 if (m_pRouteRolloverWin) {
12377 m_pRouteRolloverWin->Draw(dc);
12378 m_brepaint_piano = true;
12379 }
12380
12381 if (m_pAISRolloverWin) {
12382 m_pAISRolloverWin->Draw(dc);
12383 m_brepaint_piano = true;
12384 }
12385 }
12386 if (g_pi_manager) {
12387 g_pi_manager->RenderAllCanvasOverlayPlugIns(dc, GetVP(), m_canvasIndex,
12388 OVERLAY_OVER_UI);
12389 }
12390}
12391
12392emboss_data *ChartCanvas::EmbossDepthScale() {
12393 if (!m_bShowDepthUnits) return NULL;
12394
12395 int depth_unit_type = DEPTH_UNIT_UNKNOWN;
12396
12397 if (GetQuiltMode()) {
12398 wxString s = m_pQuilt->GetQuiltDepthUnit();
12399 s.MakeUpper();
12400 if (s == _T("FEET"))
12401 depth_unit_type = DEPTH_UNIT_FEET;
12402 else if (s.StartsWith(_T("FATHOMS")))
12403 depth_unit_type = DEPTH_UNIT_FATHOMS;
12404 else if (s.StartsWith(_T("METERS")))
12405 depth_unit_type = DEPTH_UNIT_METERS;
12406 else if (s.StartsWith(_T("METRES")))
12407 depth_unit_type = DEPTH_UNIT_METERS;
12408 else if (s.StartsWith(_T("METRIC")))
12409 depth_unit_type = DEPTH_UNIT_METERS;
12410 else if (s.StartsWith(_T("METER")))
12411 depth_unit_type = DEPTH_UNIT_METERS;
12412
12413 } else {
12414 if (m_singleChart) {
12415 depth_unit_type = m_singleChart->GetDepthUnitType();
12416 if (m_singleChart->GetChartFamily() == CHART_FAMILY_VECTOR)
12417 depth_unit_type = ps52plib->m_nDepthUnitDisplay + 1;
12418 }
12419 }
12420
12421 emboss_data *ped = NULL;
12422 switch (depth_unit_type) {
12423 case DEPTH_UNIT_FEET:
12424 ped = m_pEM_Feet;
12425 break;
12426 case DEPTH_UNIT_METERS:
12427 ped = m_pEM_Meters;
12428 break;
12429 case DEPTH_UNIT_FATHOMS:
12430 ped = m_pEM_Fathoms;
12431 break;
12432 default:
12433 return NULL;
12434 }
12435
12436 ped->x = (GetVP().pix_width - ped->width);
12437
12438 if (m_Compass && m_bShowCompassWin && g_bShowCompassWin) {
12439 wxRect r = m_Compass->GetRect();
12440 ped->y = r.y + r.height;
12441 } else {
12442 ped->y = 40;
12443 }
12444 return ped;
12445}
12446
12447void ChartCanvas::CreateDepthUnitEmbossMaps(ColorScheme cs) {
12448 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
12449 wxFont font;
12450 if (style->embossFont == wxEmptyString) {
12451 wxFont *dFont = FontMgr::Get().GetFont(_("Dialog"), 0);
12452 font = *dFont;
12453 font.SetPointSize(60);
12454 font.SetWeight(wxFONTWEIGHT_BOLD);
12455 } else
12456 font = wxFont(style->embossHeight, wxFONTFAMILY_SWISS, wxFONTSTYLE_NORMAL,
12457 wxFONTWEIGHT_BOLD, false, style->embossFont);
12458
12459 int emboss_width = 500;
12460 int emboss_height = 200;
12461
12462 // Free any existing emboss maps
12463 delete m_pEM_Feet;
12464 delete m_pEM_Meters;
12465 delete m_pEM_Fathoms;
12466
12467 // Create the 3 DepthUnit emboss map structures
12468 m_pEM_Feet =
12469 CreateEmbossMapData(font, emboss_width, emboss_height, _("Feet"), cs);
12470 m_pEM_Meters =
12471 CreateEmbossMapData(font, emboss_width, emboss_height, _("Meters"), cs);
12472 m_pEM_Fathoms =
12473 CreateEmbossMapData(font, emboss_width, emboss_height, _("Fathoms"), cs);
12474}
12475
12476#define OVERZOOM_TEXT _("OverZoom")
12477
12478void ChartCanvas::SetOverzoomFont() {
12479 ocpnStyle::Style *style = g_StyleManager->GetCurrentStyle();
12480 int w, h;
12481
12482 wxFont font;
12483 if (style->embossFont == wxEmptyString) {
12484 wxFont *dFont = FontMgr::Get().GetFont(_("Dialog"), 0);
12485 font = *dFont;
12486 font.SetPointSize(40);
12487 font.SetWeight(wxFONTWEIGHT_BOLD);
12488 } else
12489 font = wxFont(style->embossHeight, wxFONTFAMILY_SWISS, wxFONTSTYLE_NORMAL,
12490 wxFONTWEIGHT_BOLD, false, style->embossFont);
12491
12492 wxClientDC dc(this);
12493 dc.SetFont(font);
12494 dc.GetTextExtent(OVERZOOM_TEXT, &w, &h);
12495
12496 while (font.GetPointSize() > 10 && (w > 500 || h > 100)) {
12497 font.SetPointSize(font.GetPointSize() - 1);
12498 dc.SetFont(font);
12499 dc.GetTextExtent(OVERZOOM_TEXT, &w, &h);
12500 }
12501 m_overzoomFont = font;
12502 m_overzoomTextWidth = w;
12503 m_overzoomTextHeight = h;
12504}
12505
12506void ChartCanvas::CreateOZEmbossMapData(ColorScheme cs) {
12507 delete m_pEM_OverZoom;
12508
12509 if (m_overzoomTextWidth > 0 && m_overzoomTextHeight > 0)
12510 m_pEM_OverZoom =
12511 CreateEmbossMapData(m_overzoomFont, m_overzoomTextWidth + 10,
12512 m_overzoomTextHeight + 10, OVERZOOM_TEXT, cs);
12513}
12514
12515emboss_data *ChartCanvas::CreateEmbossMapData(wxFont &font, int width,
12516 int height, const wxString &str,
12517 ColorScheme cs) {
12518 int *pmap;
12519
12520 // Create a temporary bitmap
12521 wxBitmap bmp(width, height, -1);
12522
12523 // Create a memory DC
12524 wxMemoryDC temp_dc;
12525 temp_dc.SelectObject(bmp);
12526
12527 // Paint on it
12528 temp_dc.SetBackground(*wxWHITE_BRUSH);
12529 temp_dc.SetTextBackground(*wxWHITE);
12530 temp_dc.SetTextForeground(*wxBLACK);
12531
12532 temp_dc.Clear();
12533
12534 temp_dc.SetFont(font);
12535
12536 int str_w, str_h;
12537 temp_dc.GetTextExtent(str, &str_w, &str_h);
12538 // temp_dc.DrawText( str, width - str_w - 10, 10 );
12539 temp_dc.DrawText(str, 1, 1);
12540
12541 // Deselect the bitmap
12542 temp_dc.SelectObject(wxNullBitmap);
12543
12544 // Convert bitmap the wxImage for manipulation
12545 wxImage img = bmp.ConvertToImage();
12546
12547 int image_width = str_w * 105 / 100;
12548 int image_height = str_h * 105 / 100;
12549 wxRect r(0, 0, wxMin(image_width, img.GetWidth()),
12550 wxMin(image_height, img.GetHeight()));
12551 wxImage imgs = img.GetSubImage(r);
12552
12553 double val_factor;
12554 switch (cs) {
12555 case GLOBAL_COLOR_SCHEME_DAY:
12556 default:
12557 val_factor = 1;
12558 break;
12559 case GLOBAL_COLOR_SCHEME_DUSK:
12560 val_factor = .5;
12561 break;
12562 case GLOBAL_COLOR_SCHEME_NIGHT:
12563 val_factor = .25;
12564 break;
12565 }
12566
12567 int val;
12568 int index;
12569 const int w = imgs.GetWidth();
12570 const int h = imgs.GetHeight();
12571 pmap = (int *)calloc(w * h * sizeof(int), 1);
12572 // Create emboss map by differentiating the emboss image
12573 // and storing integer results in pmap
12574 // n.b. since the image is B/W, it is sufficient to check
12575 // one channel (i.e. red) only
12576 for (int y = 1; y < h - 1; y++) {
12577 for (int x = 1; x < w - 1; x++) {
12578 val =
12579 img.GetRed(x + 1, y + 1) - img.GetRed(x - 1, y - 1); // range +/- 256
12580 val = (int)(val * val_factor);
12581 index = (y * w) + x;
12582 pmap[index] = val;
12583 }
12584 }
12585
12586 emboss_data *pret = new emboss_data;
12587 pret->pmap = pmap;
12588 pret->width = w;
12589 pret->height = h;
12590
12591 return pret;
12592}
12593
12594void ChartCanvas::DrawAllTracksInBBox(ocpnDC &dc, LLBBox &BltBBox) {
12595 Track *active_track = NULL;
12596 for (Track *pTrackDraw : g_TrackList) {
12597 if (g_pActiveTrack == pTrackDraw) {
12598 active_track = pTrackDraw;
12599 continue;
12600 }
12601
12602 TrackGui(*pTrackDraw).Draw(this, dc, GetVP(), BltBBox);
12603 }
12604
12605 if (active_track) TrackGui(*active_track).Draw(this, dc, GetVP(), BltBBox);
12606}
12607
12608void ChartCanvas::DrawActiveTrackInBBox(ocpnDC &dc, LLBBox &BltBBox) {
12609 Track *active_track = NULL;
12610 for (Track *pTrackDraw : g_TrackList) {
12611 if (g_pActiveTrack == pTrackDraw) {
12612 active_track = pTrackDraw;
12613 break;
12614 }
12615 }
12616 if (active_track) TrackGui(*active_track).Draw(this, dc, GetVP(), BltBBox);
12617}
12618
12619void ChartCanvas::DrawAllRoutesInBBox(ocpnDC &dc, LLBBox &BltBBox) {
12620 Route *active_route = NULL;
12621
12622 for (wxRouteListNode *node = pRouteList->GetFirst(); node;
12623 node = node->GetNext()) {
12624 Route *pRouteDraw = node->GetData();
12625 if (pRouteDraw->IsActive() || pRouteDraw->IsSelected()) {
12626 active_route = pRouteDraw;
12627 continue;
12628 }
12629
12630 // if(m_canvasIndex == 1)
12631 RouteGui(*pRouteDraw).Draw(dc, this, BltBBox);
12632 }
12633
12634 // Draw any active or selected route (or track) last, so that is is always on
12635 // top
12636 if (active_route) RouteGui(*active_route).Draw(dc, this, BltBBox);
12637}
12638
12639void ChartCanvas::DrawActiveRouteInBBox(ocpnDC &dc, LLBBox &BltBBox) {
12640 Route *active_route = NULL;
12641
12642 for (wxRouteListNode *node = pRouteList->GetFirst(); node;
12643 node = node->GetNext()) {
12644 Route *pRouteDraw = node->GetData();
12645 if (pRouteDraw->IsActive() || pRouteDraw->IsSelected()) {
12646 active_route = pRouteDraw;
12647 break;
12648 }
12649 }
12650 if (active_route) RouteGui(*active_route).Draw(dc, this, BltBBox);
12651}
12652
12653void ChartCanvas::DrawAllWaypointsInBBox(ocpnDC &dc, LLBBox &BltBBox) {
12654 if (!pWayPointMan) return;
12655
12656 wxRoutePointListNode *node = pWayPointMan->GetWaypointList()->GetFirst();
12657
12658 while (node) {
12659 RoutePoint *pWP = node->GetData();
12660 if (pWP) {
12661 if (pWP->m_bIsInRoute) {
12662 node = node->GetNext();
12663 continue;
12664 }
12665
12666 /* technically incorrect... waypoint has bounding box */
12667 if (BltBBox.Contains(pWP->m_lat, pWP->m_lon))
12668 RoutePointGui(*pWP).Draw(dc, this, NULL);
12669 else {
12670 // Are Range Rings enabled?
12671 if (pWP->GetShowWaypointRangeRings() &&
12672 (pWP->GetWaypointRangeRingsNumber() > 0)) {
12673 double factor = 1.00;
12674 if (pWP->GetWaypointRangeRingsStepUnits() ==
12675 1) // convert kilometers to NMi
12676 factor = 1 / 1.852;
12677
12678 double radius = factor * pWP->GetWaypointRangeRingsNumber() *
12679 pWP->GetWaypointRangeRingsStep() / 60.;
12680 radius *= 2; // Fudge factor
12681
12682 LLBBox radar_box;
12683 radar_box.Set(pWP->m_lat - radius, pWP->m_lon - radius,
12684 pWP->m_lat + radius, pWP->m_lon + radius);
12685 if (!BltBBox.IntersectOut(radar_box)) {
12686 RoutePointGui(*pWP).Draw(dc, this, NULL);
12687 }
12688 }
12689 }
12690 }
12691
12692 node = node->GetNext();
12693 }
12694}
12695
12696void ChartCanvas::DrawBlinkObjects(void) {
12697 // All RoutePoints
12698 wxRect update_rect;
12699
12700 if (!pWayPointMan) return;
12701
12702 wxRoutePointListNode *node = pWayPointMan->GetWaypointList()->GetFirst();
12703
12704 while (node) {
12705 RoutePoint *pWP = node->GetData();
12706 if (pWP) {
12707 if (pWP->m_bBlink) {
12708 update_rect.Union(pWP->CurrentRect_in_DC);
12709 }
12710 }
12711
12712 node = node->GetNext();
12713 }
12714 if (!update_rect.IsEmpty()) RefreshRect(update_rect);
12715}
12716
12717void ChartCanvas::DrawAnchorWatchPoints(ocpnDC &dc) {
12718 // draw anchor watch rings, if activated
12719
12720 if (pAnchorWatchPoint1 || pAnchorWatchPoint2) {
12721 wxPoint r1, r2;
12722 wxPoint lAnchorPoint1, lAnchorPoint2;
12723 double lpp1 = 0.0;
12724 double lpp2 = 0.0;
12725 if (pAnchorWatchPoint1) {
12726 lpp1 = GetAnchorWatchRadiusPixels(pAnchorWatchPoint1);
12727 GetCanvasPointPix(pAnchorWatchPoint1->m_lat, pAnchorWatchPoint1->m_lon,
12728 &lAnchorPoint1);
12729 }
12730 if (pAnchorWatchPoint2) {
12731 lpp2 = GetAnchorWatchRadiusPixels(pAnchorWatchPoint2);
12732 GetCanvasPointPix(pAnchorWatchPoint2->m_lat, pAnchorWatchPoint2->m_lon,
12733 &lAnchorPoint2);
12734 }
12735
12736 wxPen ppPeng(GetGlobalColor(_T ( "UGREN" )), 2);
12737 wxPen ppPenr(GetGlobalColor(_T ( "URED" )), 2);
12738
12739 wxBrush *ppBrush = wxTheBrushList->FindOrCreateBrush(
12740 wxColour(0, 0, 0), wxBRUSHSTYLE_TRANSPARENT);
12741 dc.SetBrush(*ppBrush);
12742
12743 if (lpp1 > 0) {
12744 dc.SetPen(ppPeng);
12745 dc.StrokeCircle(lAnchorPoint1.x, lAnchorPoint1.y, fabs(lpp1));
12746 }
12747
12748 if (lpp2 > 0) {
12749 dc.SetPen(ppPeng);
12750 dc.StrokeCircle(lAnchorPoint2.x, lAnchorPoint2.y, fabs(lpp2));
12751 }
12752
12753 if (lpp1 < 0) {
12754 dc.SetPen(ppPenr);
12755 dc.StrokeCircle(lAnchorPoint1.x, lAnchorPoint1.y, fabs(lpp1));
12756 }
12757
12758 if (lpp2 < 0) {
12759 dc.SetPen(ppPenr);
12760 dc.StrokeCircle(lAnchorPoint2.x, lAnchorPoint2.y, fabs(lpp2));
12761 }
12762 }
12763}
12764
12765double ChartCanvas::GetAnchorWatchRadiusPixels(RoutePoint *pAnchorWatchPoint) {
12766 double lpp = 0.;
12767 wxPoint r1;
12768 wxPoint lAnchorPoint;
12769 double d1 = 0.0;
12770 double dabs;
12771 double tlat1, tlon1;
12772
12773 if (pAnchorWatchPoint) {
12774 (pAnchorWatchPoint->GetName()).ToDouble(&d1);
12775 d1 = AnchorDistFix(d1, AnchorPointMinDist, g_nAWMax);
12776 dabs = fabs(d1 / 1852.);
12777 ll_gc_ll(pAnchorWatchPoint->m_lat, pAnchorWatchPoint->m_lon, 0, dabs,
12778 &tlat1, &tlon1);
12779 GetCanvasPointPix(tlat1, tlon1, &r1);
12780 GetCanvasPointPix(pAnchorWatchPoint->m_lat, pAnchorWatchPoint->m_lon,
12781 &lAnchorPoint);
12782 lpp = sqrt(pow((double)(lAnchorPoint.x - r1.x), 2) +
12783 pow((double)(lAnchorPoint.y - r1.y), 2));
12784
12785 // This is an entry watch
12786 if (d1 < 0) lpp = -lpp;
12787 }
12788 return lpp;
12789}
12790
12791//------------------------------------------------------------------------------------------
12792// Tides Support
12793//------------------------------------------------------------------------------------------
12794void ChartCanvas::RebuildTideSelectList(LLBBox &BBox) {
12795 if (!ptcmgr) return;
12796
12797 pSelectTC->DeleteAllSelectableTypePoints(SELTYPE_TIDEPOINT);
12798
12799 for (int i = 1; i < ptcmgr->Get_max_IDX() + 1; i++) {
12800 const IDX_entry *pIDX = ptcmgr->GetIDX_entry(i);
12801 double lon = pIDX->IDX_lon;
12802 double lat = pIDX->IDX_lat;
12803
12804 char type = pIDX->IDX_type; // Entry "TCtcIUu" identifier
12805 if ((type == 't') || (type == 'T')) {
12806 if (BBox.Contains(lat, lon)) {
12807 // Manage the point selection list
12808 pSelectTC->AddSelectablePoint(lat, lon, pIDX, SELTYPE_TIDEPOINT);
12809 }
12810 }
12811 }
12812}
12813
12814extern wxDateTime gTimeSource;
12815
12816void ChartCanvas::DrawAllTidesInBBox(ocpnDC &dc, LLBBox &BBox) {
12817 if (!ptcmgr) return;
12818
12819 wxDateTime this_now = gTimeSource;
12820 bool cur_time = !gTimeSource.IsValid();
12821 if (cur_time) this_now = wxDateTime::Now();
12822 time_t t_this_now = this_now.GetTicks();
12823
12824 wxPen *pblack_pen = wxThePenList->FindOrCreatePen(
12825 GetGlobalColor(_T ( "UINFD" )), 1, wxPENSTYLE_SOLID);
12826 wxPen *pyelo_pen = wxThePenList->FindOrCreatePen(
12827 GetGlobalColor(cur_time ? _T ( "YELO1" ) : _T ( "YELO2" )), 1,
12828 wxPENSTYLE_SOLID);
12829 wxPen *pblue_pen = wxThePenList->FindOrCreatePen(
12830 GetGlobalColor(cur_time ? _T ( "BLUE2" ) : _T ( "BLUE3" )), 1,
12831 wxPENSTYLE_SOLID);
12832
12833 wxBrush *pgreen_brush = wxTheBrushList->FindOrCreateBrush(
12834 GetGlobalColor(_T ( "GREEN1" )), wxBRUSHSTYLE_SOLID);
12835 // wxBrush *pblack_brush = wxTheBrushList->FindOrCreateBrush (
12836 // GetGlobalColor ( _T ( "UINFD" ) ), wxSOLID );
12837 wxBrush *pblue_brush = wxTheBrushList->FindOrCreateBrush(
12838 GetGlobalColor(cur_time ? _T ( "BLUE2" ) : _T ( "BLUE3" )),
12839 wxBRUSHSTYLE_SOLID);
12840 wxBrush *pyelo_brush = wxTheBrushList->FindOrCreateBrush(
12841 GetGlobalColor(cur_time ? _T ( "YELO1" ) : _T ( "YELO2" )),
12842 wxBRUSHSTYLE_SOLID);
12843
12844 wxFont *dFont = FontMgr::Get().GetFont(_("ExtendedTideIcon"));
12845 dc.SetTextForeground(FontMgr::Get().GetFontColor(_("ExtendedTideIcon")));
12846 int font_size = wxMax(10, dFont->GetPointSize());
12847 font_size /= g_Platform->GetDisplayDIPMult(this);
12848 wxFont *plabelFont = FontMgr::Get().FindOrCreateFont(
12849 font_size, dFont->GetFamily(), dFont->GetStyle(), dFont->GetWeight(),
12850 false, dFont->GetFaceName());
12851
12852 dc.SetPen(*pblack_pen);
12853 dc.SetBrush(*pgreen_brush);
12854
12855 wxBitmap bm;
12856 switch (m_cs) {
12857 case GLOBAL_COLOR_SCHEME_DAY:
12858 bm = m_bmTideDay;
12859 break;
12860 case GLOBAL_COLOR_SCHEME_DUSK:
12861 bm = m_bmTideDusk;
12862 break;
12863 case GLOBAL_COLOR_SCHEME_NIGHT:
12864 bm = m_bmTideNight;
12865 break;
12866 default:
12867 bm = m_bmTideDay;
12868 break;
12869 }
12870
12871 int bmw = bm.GetWidth();
12872 int bmh = bm.GetHeight();
12873
12874 float scale_factor = 1.0;
12875
12876 // Set the onscreen size of the symbol
12877 // Compensate for various display resolutions
12878 float icon_pixelRefDim = 45;
12879
12880#if 0
12881 float nominal_icon_size_mm = g_Platform->GetDisplaySizeMM() *25 / 1000; // Intended physical rendered size onscreen
12882 nominal_icon_size_mm = wxMax(nominal_icon_size_mm, 8);
12883 nominal_icon_size_mm = wxMin(nominal_icon_size_mm, 15);
12884 float nominal_icon_size_pixels = wxMax(4.0, floor(g_Platform->GetDisplayDPmm() * nominal_icon_size_mm)); // nominal size, but not less than 4 pixel
12885#endif
12886
12887#ifndef __ANDROID__
12888 // another method is simply to declare that the icon shall be x times the size
12889 // of a raster symbol (e.g.BOYLAT)
12890 // This is a bit of a hack that will suffice until until we get fully
12891 // scalable ENC symbol sets
12892 // float nominal_icon_size_pixels = 48; // 3 x 16
12893 // float pix_factor = nominal_icon_size_pixels / icon_pixelRefDim;
12894
12895 // or, x times size of text font
12896 wxScreenDC sdc;
12897 int height;
12898 sdc.GetTextExtent("M", NULL, &height, NULL, NULL, plabelFont);
12899 height *= g_Platform->GetDisplayDIPMult(this);
12900 float nominal_icon_size_pixels = 48; // 3 x 16
12901 float pix_factor = (2 * height) / nominal_icon_size_pixels;
12902
12903#else
12904 // Yet another method goes like this:
12905 // Set the onscreen size of the symbol
12906 // Compensate for various display resolutions
12907 // Develop empirically, making a symbol about 16 mm tall
12908 double symHeight =
12909 icon_pixelRefDim /
12910 GetPixPerMM(); // from draw instructions, symbol is xx pix high
12911 double targetHeight0 = 16.0;
12912
12913 // But we want to scale the size down for smaller displays
12914 double displaySize = m_display_size_mm;
12915 displaySize = wxMax(displaySize, 100);
12916
12917 float targetHeight = wxMin(targetHeight0, displaySize / 15);
12918
12919 double pix_factor = targetHeight / symHeight;
12920#endif
12921
12922 scale_factor *= pix_factor;
12923
12924 float user_scale_factor = g_ChartScaleFactorExp;
12925 if (g_ChartScaleFactorExp > 1.0)
12926 user_scale_factor = (log(g_ChartScaleFactorExp) + 1.0) *
12927 1.2; // soften the scale factor a bit
12928
12929 scale_factor *= user_scale_factor;
12930 scale_factor *= GetContentScaleFactor();
12931
12932 {
12933 double marge = 0.05;
12934 std::vector<LLBBox> drawn_boxes;
12935 for (int i = 1; i < ptcmgr->Get_max_IDX() + 1; i++) {
12936 const IDX_entry *pIDX = ptcmgr->GetIDX_entry(i);
12937
12938 char type = pIDX->IDX_type; // Entry "TCtcIUu" identifier
12939 if ((type == 't') || (type == 'T')) // only Tides
12940 {
12941 double lon = pIDX->IDX_lon;
12942 double lat = pIDX->IDX_lat;
12943
12944 if (BBox.ContainsMarge(lat, lon, marge)) {
12945 // Avoid drawing detailed graphic for duplicate tide stations
12946 if (GetVP().chart_scale < 500000) {
12947 bool bdrawn = false;
12948 for (size_t i = 0; i < drawn_boxes.size(); i++) {
12949 if (drawn_boxes[i].Contains(lat, lon)) {
12950 bdrawn = true;
12951 break;
12952 }
12953 }
12954 if (bdrawn) continue; // the station loop
12955
12956 LLBBox this_box;
12957 this_box.Set(lat, lon, lat, lon);
12958 this_box.EnLarge(.005);
12959 drawn_boxes.push_back(this_box);
12960 }
12961
12962 wxPoint r;
12963 GetCanvasPointPix(lat, lon, &r);
12964 // draw standard icons
12965 if (GetVP().chart_scale > 500000) {
12966 dc.DrawBitmap(bm, r.x - bmw / 2, r.y - bmh / 2, true);
12967 }
12968 // draw "extended" icons
12969 else {
12970 dc.SetFont(*plabelFont);
12971 {
12972 {
12973 float val, nowlev;
12974 float ltleve = 0.;
12975 float htleve = 0.;
12976 time_t tctime;
12977 time_t lttime = 0;
12978 time_t httime = 0;
12979 bool wt;
12980 // define if flood or ebb in the last ten minutes and verify if
12981 // data are useable
12982 if (ptcmgr->GetTideFlowSens(
12983 t_this_now, BACKWARD_TEN_MINUTES_STEP,
12984 pIDX->IDX_rec_num, nowlev, val, wt)) {
12985 // search forward the first HW or LW near "now" ( starting at
12986 // "now" - ten minutes )
12987 ptcmgr->GetHightOrLowTide(
12988 t_this_now + BACKWARD_TEN_MINUTES_STEP,
12989 FORWARD_TEN_MINUTES_STEP, FORWARD_ONE_MINUTES_STEP, val,
12990 wt, pIDX->IDX_rec_num, val, tctime);
12991 if (wt) {
12992 httime = tctime;
12993 htleve = val;
12994 } else {
12995 lttime = tctime;
12996 ltleve = val;
12997 }
12998 wt = !wt;
12999
13000 // then search opposite tide near "now"
13001 if (tctime > t_this_now) // search backward
13002 ptcmgr->GetHightOrLowTide(
13003 t_this_now, BACKWARD_TEN_MINUTES_STEP,
13004 BACKWARD_ONE_MINUTES_STEP, nowlev, wt,
13005 pIDX->IDX_rec_num, val, tctime);
13006 else
13007 // or search forward
13008 ptcmgr->GetHightOrLowTide(
13009 t_this_now, FORWARD_TEN_MINUTES_STEP,
13010 FORWARD_ONE_MINUTES_STEP, nowlev, wt, pIDX->IDX_rec_num,
13011 val, tctime);
13012 if (wt) {
13013 httime = tctime;
13014 htleve = val;
13015 } else {
13016 lttime = tctime;
13017 ltleve = val;
13018 }
13019
13020 // draw the tide rectangle:
13021
13022 // tide icon rectangle has default pre-scaled width = 12 ,
13023 // height = 45
13024 int width = (int)(12 * scale_factor + 0.5);
13025 int height = (int)(45 * scale_factor + 0.5);
13026 int linew = wxMax(1, (int)(scale_factor));
13027 int xDraw = r.x - (width / 2);
13028 int yDraw = r.y - (height / 2);
13029
13030 // process tide state ( %height and flow sens )
13031 float ts = 1 - ((nowlev - ltleve) / (htleve - ltleve));
13032 int hs = (httime > lttime) ? -4 : 4;
13033 hs *= (int)(scale_factor + 0.5);
13034 if (ts > 0.995 || ts < 0.005) hs = 0;
13035 int ht_y = (int)(height * ts);
13036
13037 // draw yellow tide rectangle outlined in black
13038 pblack_pen->SetWidth(linew);
13039 dc.SetPen(*pblack_pen);
13040 dc.SetBrush(*pyelo_brush);
13041 dc.DrawRectangle(xDraw, yDraw, width, height);
13042
13043 // draw blue rectangle as water height, smaller in width than
13044 // yellow rectangle
13045 dc.SetPen(*pblue_pen);
13046 dc.SetBrush(*pblue_brush);
13047 dc.DrawRectangle((xDraw + 2 * linew), yDraw + ht_y,
13048 (width - (4 * linew)), height - ht_y);
13049
13050 // draw sens arrows (ensure they are not "under-drawn" by top
13051 // line of blue rectangle )
13052 int hl;
13053 wxPoint arrow[3];
13054 arrow[0].x = xDraw + 2 * linew;
13055 arrow[1].x = xDraw + width / 2;
13056 arrow[2].x = xDraw + width - 2 * linew;
13057 pyelo_pen->SetWidth(linew);
13058 pblue_pen->SetWidth(linew);
13059 if (ts > 0.35 || ts < 0.15) // one arrow at 3/4 hight tide
13060 {
13061 hl = (int)(height * 0.25) + yDraw;
13062 arrow[0].y = hl;
13063 arrow[1].y = hl + hs;
13064 arrow[2].y = hl;
13065 if (ts < 0.15)
13066 dc.SetPen(*pyelo_pen);
13067 else
13068 dc.SetPen(*pblue_pen);
13069 dc.DrawLines(3, arrow);
13070 }
13071 if (ts > 0.60 || ts < 0.40) // one arrow at 1/2 hight tide
13072 {
13073 hl = (int)(height * 0.5) + yDraw;
13074 arrow[0].y = hl;
13075 arrow[1].y = hl + hs;
13076 arrow[2].y = hl;
13077 if (ts < 0.40)
13078 dc.SetPen(*pyelo_pen);
13079 else
13080 dc.SetPen(*pblue_pen);
13081 dc.DrawLines(3, arrow);
13082 }
13083 if (ts < 0.65 || ts > 0.85) // one arrow at 1/4 Hight tide
13084 {
13085 hl = (int)(height * 0.75) + yDraw;
13086 arrow[0].y = hl;
13087 arrow[1].y = hl + hs;
13088 arrow[2].y = hl;
13089 if (ts < 0.65)
13090 dc.SetPen(*pyelo_pen);
13091 else
13092 dc.SetPen(*pblue_pen);
13093 dc.DrawLines(3, arrow);
13094 }
13095 // draw tide level text
13096 wxString s;
13097 s.Printf(_T("%3.1f"), nowlev);
13098 Station_Data *pmsd = pIDX->pref_sta_data; // write unit
13099 if (pmsd) s.Append(wxString(pmsd->units_abbrv, wxConvUTF8));
13100 int wx1;
13101 dc.GetTextExtent(s, &wx1, NULL);
13102 wx1 *= g_Platform->GetDisplayDIPMult(this);
13103 dc.DrawText(s, r.x - (wx1 / 2), yDraw + height);
13104 }
13105 }
13106 }
13107 }
13108 }
13109 }
13110 }
13111 }
13112}
13113
13114//------------------------------------------------------------------------------------------
13115// Currents Support
13116//------------------------------------------------------------------------------------------
13117
13118void ChartCanvas::RebuildCurrentSelectList(LLBBox &BBox) {
13119 if (!ptcmgr) return;
13120
13121 pSelectTC->DeleteAllSelectableTypePoints(SELTYPE_CURRENTPOINT);
13122
13123 for (int i = 1; i < ptcmgr->Get_max_IDX() + 1; i++) {
13124 const IDX_entry *pIDX = ptcmgr->GetIDX_entry(i);
13125 double lon = pIDX->IDX_lon;
13126 double lat = pIDX->IDX_lat;
13127
13128 char type = pIDX->IDX_type; // Entry "TCtcIUu" identifier
13129 if (((type == 'c') || (type == 'C')) && (!pIDX->b_skipTooDeep)) {
13130 if ((BBox.Contains(lat, lon))) {
13131 // Manage the point selection list
13132 pSelectTC->AddSelectablePoint(lat, lon, pIDX, SELTYPE_CURRENTPOINT);
13133 }
13134 }
13135 }
13136}
13137
13138void ChartCanvas::DrawAllCurrentsInBBox(ocpnDC &dc, LLBBox &BBox) {
13139 if (!ptcmgr) return;
13140
13141 float tcvalue, dir;
13142 bool bnew_val;
13143 char sbuf[20];
13144 wxFont *pTCFont;
13145 double lon_last = 0.;
13146 double lat_last = 0.;
13147 // arrow size for Raz Blanchard : 12 knots north
13148 double marge = 0.2;
13149 bool cur_time = !gTimeSource.IsValid();
13150
13151 double true_scale_display = floor(VPoint.chart_scale / 100.) * 100.;
13152 bDrawCurrentValues = true_scale_display < g_Show_Target_Name_Scale;
13153
13154 wxPen *pblack_pen = wxThePenList->FindOrCreatePen(
13155 GetGlobalColor(_T ( "UINFD" )), 1, wxPENSTYLE_SOLID);
13156 wxPen *porange_pen = wxThePenList->FindOrCreatePen(
13157 GetGlobalColor(cur_time ? _T ( "UINFO" ) : _T ( "UINFB" )), 1,
13158 wxPENSTYLE_SOLID);
13159 wxBrush *porange_brush = wxTheBrushList->FindOrCreateBrush(
13160 GetGlobalColor(cur_time ? _T ( "UINFO" ) : _T ( "UINFB" )),
13161 wxBRUSHSTYLE_SOLID);
13162 wxBrush *pgray_brush = wxTheBrushList->FindOrCreateBrush(
13163 GetGlobalColor(_T ( "UIBDR" )), wxBRUSHSTYLE_SOLID);
13164 wxBrush *pblack_brush = wxTheBrushList->FindOrCreateBrush(
13165 GetGlobalColor(_T ( "UINFD" )), wxBRUSHSTYLE_SOLID);
13166
13167 double skew_angle = GetVPRotation();
13168
13169 wxFont *dFont = FontMgr::Get().GetFont(_("CurrentValue"));
13170 dc.SetTextForeground(FontMgr::Get().GetFontColor(_("CurrentValue")));
13171 int font_size = wxMax(10, dFont->GetPointSize());
13172 font_size /= g_Platform->GetDisplayDIPMult(this);
13173 pTCFont = FontMgr::Get().FindOrCreateFont(
13174 font_size, dFont->GetFamily(), dFont->GetStyle(), dFont->GetWeight(),
13175 false, dFont->GetFaceName());
13176
13177 float scale_factor = 1.0;
13178
13179 // Set the onscreen size of the symbol
13180 // Compensate for various display resolutions
13181
13182#if 0
13183 float nominal_icon_size_mm = g_Platform->GetDisplaySizeMM() *3 / 1000; // Intended physical rendered size onscreen
13184 nominal_icon_size_mm = wxMax(nominal_icon_size_mm, 2);
13185 nominal_icon_size_mm = wxMin(nominal_icon_size_mm, 4);
13186 float nominal_icon_size_pixels = wxMax(4.0, floor(g_Platform->GetDisplayDPmm() * nominal_icon_size_mm)); // nominal size, but not less than 4 pixel
13187#endif
13188
13189#if 0
13190 // another method is simply to declare that the icon shall be x times the size of a raster symbol (e.g.BOYLAT)
13191 // This is a bit of a hack that will suffice until until we get fully scalable ENC symbol sets
13192 float nominal_icon_size_pixels = 6; // 16 / 3
13193 float pix_factor = nominal_icon_size_pixels / icon_pixelRefDim;
13194#endif
13195
13196#ifndef __ANDROID__
13197 // or, x times size of text font
13198 wxScreenDC sdc;
13199 int height;
13200 sdc.GetTextExtent("M", NULL, &height, NULL, NULL, pTCFont);
13201 height *= g_Platform->GetDisplayDIPMult(this);
13202 float nominal_icon_size_pixels = 15;
13203 float pix_factor = (1 * height) / nominal_icon_size_pixels;
13204
13205#else
13206 // Yet another method goes like this:
13207 // Set the onscreen size of the symbol
13208 // Compensate for various display resolutions
13209 // Develop empirically....
13210 float icon_pixelRefDim = 5;
13211
13212 double symHeight =
13213 icon_pixelRefDim /
13214 GetPixPerMM(); // from draw instructions, symbol is xx pix high
13215 double targetHeight0 = 2.0;
13216
13217 // But we want to scale the size down for smaller displays
13218 double displaySize = m_display_size_mm;
13219 displaySize = wxMax(displaySize, 100);
13220
13221 float targetHeight = wxMin(targetHeight0, displaySize / 50);
13222 double pix_factor = targetHeight / symHeight;
13223#endif
13224
13225 scale_factor *= pix_factor;
13226
13227 float user_scale_factor = g_ChartScaleFactorExp;
13228 if (g_ChartScaleFactorExp > 1.0)
13229 user_scale_factor = (log(g_ChartScaleFactorExp) + 1.0) *
13230 1.2; // soften the scale factor a bit
13231
13232 scale_factor *= user_scale_factor;
13233
13234 scale_factor *= GetContentScaleFactor();
13235
13236 {
13237 for (int i = 1; i < ptcmgr->Get_max_IDX() + 1; i++) {
13238 const IDX_entry *pIDX = ptcmgr->GetIDX_entry(i);
13239 double lon = pIDX->IDX_lon;
13240 double lat = pIDX->IDX_lat;
13241
13242 char type = pIDX->IDX_type; // Entry "TCtcIUu" identifier
13243 if (((type == 'c') || (type == 'C')) && (1 /*pIDX->IDX_Useable*/)) {
13244 if (!pIDX->b_skipTooDeep && (BBox.ContainsMarge(lat, lon, marge))) {
13245 wxPoint r;
13246 GetCanvasPointPix(lat, lon, &r);
13247
13248 wxPoint d[4]; // points of a diamond at the current station location
13249 int dd = (int)(5.0 * scale_factor + 0.5);
13250 d[0].x = r.x;
13251 d[0].y = r.y + dd;
13252 d[1].x = r.x + dd;
13253 d[1].y = r.y;
13254 d[2].x = r.x;
13255 d[2].y = r.y - dd;
13256 d[3].x = r.x - dd;
13257 d[3].y = r.y;
13258
13259 if (1) {
13260 pblack_pen->SetWidth(wxMax(2, (int)(scale_factor + 0.5)));
13261 dc.SetPen(*pblack_pen);
13262 dc.SetBrush(*porange_brush);
13263 dc.DrawPolygon(4, d);
13264
13265 if (type == 'C') {
13266 dc.SetBrush(*pblack_brush);
13267 dc.DrawCircle(r.x, r.y, (int)(2 * scale_factor));
13268 }
13269
13270 if (GetVP().chart_scale < 1000000) {
13271 if (!ptcmgr->GetTideOrCurrent15(0, i, tcvalue, dir, bnew_val))
13272 continue;
13273 } else
13274 continue;
13275
13276 if (1 /*type == 'c'*/) {
13277 {
13278 // Get the display pixel location of the current station
13279 int pixxc, pixyc;
13280 pixxc = r.x;
13281 pixyc = r.y;
13282
13283 // Adjust drawing size using logarithmic scale. tcvalue is
13284 // current in knots
13285 double a1 = fabs(tcvalue) * 10.;
13286 // Current values <= 0.1 knot will have no arrow
13287 a1 = wxMax(1.0, a1);
13288 double a2 = log10(a1);
13289
13290 float cscale = scale_factor * a2 * 0.4;
13291
13292 porange_pen->SetWidth(wxMax(2, (int)(scale_factor + 0.5)));
13293 dc.SetPen(*porange_pen);
13294 DrawArrow(dc, pixxc, pixyc, dir - 90 + (skew_angle * 180. / PI),
13295 cscale);
13296 // Draw text, if enabled
13297
13298 if (bDrawCurrentValues) {
13299 dc.SetFont(*pTCFont);
13300 snprintf(sbuf, 19, "%3.1f", fabs(tcvalue));
13301 dc.DrawText(wxString(sbuf, wxConvUTF8), pixxc, pixyc);
13302 }
13303 }
13304 } // scale
13305 }
13306 /* This is useful for debugging the TC database
13307 else
13308 {
13309 dc.SetPen ( *porange_pen );
13310 dc.SetBrush ( *pgray_brush );
13311 dc.DrawPolygon ( 4, d );
13312 }
13313 */
13314 }
13315 lon_last = lon;
13316 lat_last = lat;
13317 }
13318 }
13319 }
13320}
13321
13322void ChartCanvas::DrawTCWindow(int x, int y, void *pvIDX) {
13323 pCwin = new TCWin(this, x, y, pvIDX);
13324}
13325
13326#define NUM_CURRENT_ARROW_POINTS 9
13327static wxPoint CurrentArrowArray[NUM_CURRENT_ARROW_POINTS] = {
13328 wxPoint(0, 0), wxPoint(0, -10), wxPoint(55, -10),
13329 wxPoint(55, -25), wxPoint(100, 0), wxPoint(55, 25),
13330 wxPoint(55, 10), wxPoint(0, 10), wxPoint(0, 0)};
13331
13332void ChartCanvas::DrawArrow(ocpnDC &dc, int x, int y, double rot_angle,
13333 double scale) {
13334 if (scale > 1e-2) {
13335 float sin_rot = sin(rot_angle * PI / 180.);
13336 float cos_rot = cos(rot_angle * PI / 180.);
13337
13338 // Move to the first point
13339
13340 float xt = CurrentArrowArray[0].x;
13341 float yt = CurrentArrowArray[0].y;
13342
13343 float xp = (xt * cos_rot) - (yt * sin_rot);
13344 float yp = (xt * sin_rot) + (yt * cos_rot);
13345 int x1 = (int)(xp * scale);
13346 int y1 = (int)(yp * scale);
13347
13348 // Walk thru the point list
13349 for (int ip = 1; ip < NUM_CURRENT_ARROW_POINTS; ip++) {
13350 xt = CurrentArrowArray[ip].x;
13351 yt = CurrentArrowArray[ip].y;
13352
13353 float xp = (xt * cos_rot) - (yt * sin_rot);
13354 float yp = (xt * sin_rot) + (yt * cos_rot);
13355 int x2 = (int)(xp * scale);
13356 int y2 = (int)(yp * scale);
13357
13358 dc.DrawLine(x1 + x, y1 + y, x2 + x, y2 + y);
13359
13360 x1 = x2;
13361 y1 = y2;
13362 }
13363 }
13364}
13365
13366wxString ChartCanvas::FindValidUploadPort() {
13367 wxString port;
13368 // Try to use the saved persistent upload port first
13369 if (!g_uploadConnection.IsEmpty() &&
13370 g_uploadConnection.StartsWith(_T("Serial"))) {
13371 port = g_uploadConnection;
13372 }
13373
13374 else {
13375 // If there is no persistent upload port recorded (yet)
13376 // then use the first available serial connection which has output defined.
13377 for (auto *cp : TheConnectionParams()) {
13378 if ((cp->IOSelect != DS_TYPE_INPUT) && cp->Type == SERIAL)
13379 port << _T("Serial:") << cp->Port;
13380 }
13381 }
13382 return port;
13383}
13384
13385void ShowAISTargetQueryDialog(wxWindow *win, int mmsi) {
13386 if (!win) return;
13387
13388 if (NULL == g_pais_query_dialog_active) {
13389 int pos_x = g_ais_query_dialog_x;
13390 int pos_y = g_ais_query_dialog_y;
13391
13392 if (g_pais_query_dialog_active) {
13393 g_pais_query_dialog_active->Destroy();
13394 g_pais_query_dialog_active = new AISTargetQueryDialog();
13395 } else {
13396 g_pais_query_dialog_active = new AISTargetQueryDialog();
13397 }
13398
13399 g_pais_query_dialog_active->Create(win, -1, _("AIS Target Query"),
13400 wxPoint(pos_x, pos_y));
13401
13402 g_pais_query_dialog_active->SetAutoCentre(g_btouch);
13403 g_pais_query_dialog_active->SetAutoSize(g_bresponsive);
13404 g_pais_query_dialog_active->SetMMSI(mmsi);
13405 g_pais_query_dialog_active->UpdateText();
13406 wxSize sz = g_pais_query_dialog_active->GetSize();
13407
13408 bool b_reset_pos = false;
13409#ifdef __WXMSW__
13410 // Support MultiMonitor setups which an allow negative window positions.
13411 // If the requested window title bar does not intersect any installed
13412 // monitor, then default to simple primary monitor positioning.
13413 RECT frame_title_rect;
13414 frame_title_rect.left = pos_x;
13415 frame_title_rect.top = pos_y;
13416 frame_title_rect.right = pos_x + sz.x;
13417 frame_title_rect.bottom = pos_y + 30;
13418
13419 if (NULL == MonitorFromRect(&frame_title_rect, MONITOR_DEFAULTTONULL))
13420 b_reset_pos = true;
13421#else
13422
13423 // Make sure drag bar (title bar) of window intersects wxClient Area of
13424 // screen, with a little slop...
13425 wxRect window_title_rect; // conservative estimate
13426 window_title_rect.x = pos_x;
13427 window_title_rect.y = pos_y;
13428 window_title_rect.width = sz.x;
13429 window_title_rect.height = 30;
13430
13431 wxRect ClientRect = wxGetClientDisplayRect();
13432 ClientRect.Deflate(
13433 60, 60); // Prevent the new window from being too close to the edge
13434 if (!ClientRect.Intersects(window_title_rect)) b_reset_pos = true;
13435
13436#endif
13437
13438 if (b_reset_pos) g_pais_query_dialog_active->Move(50, 200);
13439
13440 } else {
13441 g_pais_query_dialog_active->SetMMSI(mmsi);
13442 g_pais_query_dialog_active->UpdateText();
13443 }
13444
13445 g_pais_query_dialog_active->Show();
13446}
13447
13448void ChartCanvas::ToggleCanvasQuiltMode(void) {
13449 bool cur_mode = GetQuiltMode();
13450
13451 if (!GetQuiltMode())
13452 SetQuiltMode(true);
13453 else if (GetQuiltMode()) {
13454 SetQuiltMode(false);
13455 g_sticky_chart = GetQuiltReferenceChartIndex();
13456 }
13457
13458 if (cur_mode != GetQuiltMode()) {
13459 SetupCanvasQuiltMode();
13460 DoCanvasUpdate();
13461 InvalidateGL();
13462 Refresh();
13463 }
13464 // TODO What to do about this?
13465 // g_bQuiltEnable = GetQuiltMode();
13466
13467 // Recycle the S52 PLIB so that vector charts will flush caches and re-render
13468 if (ps52plib) ps52plib->GenerateStateHash();
13469
13470 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
13471 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
13472}
13473
13474void ChartCanvas::DoCanvasStackDelta(int direction) {
13475 if (!GetQuiltMode()) {
13476 int current_stack_index = GetpCurrentStack()->CurrentStackEntry;
13477 if ((current_stack_index + direction) >= GetpCurrentStack()->nEntry) return;
13478 if ((current_stack_index + direction) < 0) return;
13479
13480 if (m_bpersistent_quilt /*&& g_bQuiltEnable*/) {
13481 int new_dbIndex =
13482 GetpCurrentStack()->GetDBIndex(current_stack_index + direction);
13483
13484 if (IsChartQuiltableRef(new_dbIndex)) {
13485 ToggleCanvasQuiltMode();
13486 SelectQuiltRefdbChart(new_dbIndex);
13487 m_bpersistent_quilt = false;
13488 }
13489 } else {
13490 SelectChartFromStack(current_stack_index + direction);
13491 }
13492 } else {
13493 std::vector<int> piano_chart_index_array =
13494 GetQuiltExtendedStackdbIndexArray();
13495 int refdb = GetQuiltRefChartdbIndex();
13496
13497 // Find the ref chart in the stack
13498 int current_index = -1;
13499 for (unsigned int i = 0; i < piano_chart_index_array.size(); i++) {
13500 if (refdb == piano_chart_index_array[i]) {
13501 current_index = i;
13502 break;
13503 }
13504 }
13505 if (current_index == -1) return;
13506
13507 const ChartTableEntry &ctet = ChartData->GetChartTableEntry(refdb);
13508 int target_family = ctet.GetChartFamily();
13509
13510 int new_index = -1;
13511 int check_index = current_index + direction;
13512 bool found = false;
13513 int check_dbIndex = -1;
13514 int new_dbIndex = -1;
13515
13516 // When quilted. switch within the same chart family
13517 while (!found &&
13518 (unsigned int)check_index < piano_chart_index_array.size() &&
13519 (check_index >= 0)) {
13520 check_dbIndex = piano_chart_index_array[check_index];
13521 const ChartTableEntry &cte = ChartData->GetChartTableEntry(check_dbIndex);
13522 if (target_family == cte.GetChartFamily()) {
13523 found = true;
13524 new_index = check_index;
13525 new_dbIndex = check_dbIndex;
13526 break;
13527 }
13528
13529 check_index += direction;
13530 }
13531
13532 if (!found) return;
13533
13534 if (!IsChartQuiltableRef(new_dbIndex)) {
13535 ToggleCanvasQuiltMode();
13536 SelectdbChart(new_dbIndex);
13537 m_bpersistent_quilt = true;
13538 } else {
13539 SelectQuiltRefChart(new_index);
13540 }
13541 }
13542
13543 gFrame->UpdateGlobalMenuItems(); // update the state of the menu items
13544 // (checkmarks etc)
13545 SetQuiltChartHiLiteIndex(-1);
13546
13547 ReloadVP();
13548}
13549
13550//--------------------------------------------------------------------------------------------------------
13551//
13552// Toolbar support
13553//
13554//--------------------------------------------------------------------------------------------------------
13555
13556void ChartCanvas::OnToolLeftClick(wxCommandEvent &event) {
13557 // Handle the per-canvas toolbar clicks here
13558
13559 switch (event.GetId()) {
13560 case ID_ZOOMIN: {
13561 StopMovement();
13562 ZoomCanvasSimple(g_plus_minus_zoom_factor);
13563 break;
13564 }
13565
13566 case ID_ZOOMOUT: {
13567 StopMovement();
13568 ZoomCanvasSimple(1.0 / g_plus_minus_zoom_factor);
13569 break;
13570 }
13571
13572 case ID_STKUP:
13573 DoCanvasStackDelta(1);
13574 DoCanvasUpdate();
13575 break;
13576
13577 case ID_STKDN:
13578 DoCanvasStackDelta(-1);
13579 DoCanvasUpdate();
13580 break;
13581
13582 case ID_FOLLOW: {
13583 TogglebFollow();
13584 break;
13585 }
13586
13587 case ID_CURRENT: {
13588 ShowCurrents(!GetbShowCurrent());
13589 ReloadVP();
13590 Refresh(false);
13591 break;
13592 }
13593
13594 case ID_TIDE: {
13595 ShowTides(!GetbShowTide());
13596 ReloadVP();
13597 Refresh(false);
13598 break;
13599 }
13600
13601 case ID_ROUTE: {
13602 if (0 == m_routeState) {
13603 StartRoute();
13604 } else {
13605 FinishRoute();
13606 }
13607
13608#ifdef __ANDROID__
13609 androidSetRouteAnnunciator(m_routeState == 1);
13610#endif
13611 break;
13612 }
13613
13614 case ID_AIS: {
13615 SetAISCanvasDisplayStyle(-1);
13616 break;
13617 }
13618
13619 default:
13620 break;
13621 }
13622
13623 // And then let gFrame handle the rest....
13624 event.Skip();
13625}
13626
13627void ChartCanvas::SetShowAIS(bool show) {
13628 m_bShowAIS = show;
13629 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
13630 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
13631}
13632
13633void ChartCanvas::SetAttenAIS(bool show) {
13634 m_bShowAISScaled = show;
13635 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
13636 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
13637}
13638
13639void ChartCanvas::SetAISCanvasDisplayStyle(int StyleIndx) {
13640 // make some arrays to hold the dfferences between cycle steps
13641 // show all, scaled, hide all
13642 bool bShowAIS_Array[3] = {true, true, false};
13643 bool bShowScaled_Array[3] = {false, true, true};
13644 wxString ToolShortHelp_Array[3] = {_("Show all AIS Targets"),
13645 _("Attenuate less critical AIS targets"),
13646 _("Hide AIS Targets")};
13647 wxString iconName_Array[3] = {_T("AIS"), _T("AIS_Suppressed"),
13648 _T("AIS_Disabled")};
13649 int ArraySize = 3;
13650 int AIS_Toolbar_Switch = 0;
13651 if (StyleIndx == -1) { // -1 means coming from toolbar button
13652 // find current state of switch
13653 for (int i = 1; i < ArraySize; i++) {
13654 if ((bShowAIS_Array[i] == m_bShowAIS) &&
13655 (bShowScaled_Array[i] == m_bShowAISScaled))
13656 AIS_Toolbar_Switch = i;
13657 }
13658 AIS_Toolbar_Switch++; // we did click so continu with next item
13659 if ((!g_bAllowShowScaled) && (AIS_Toolbar_Switch == 1))
13660 AIS_Toolbar_Switch++;
13661
13662 } else { // coming from menu bar.
13663 AIS_Toolbar_Switch = StyleIndx;
13664 }
13665 // make sure we are not above array
13666 if (AIS_Toolbar_Switch >= ArraySize) AIS_Toolbar_Switch = 0;
13667
13668 int AIS_Toolbar_Switch_Next =
13669 AIS_Toolbar_Switch + 1; // Find out what will happen at next click
13670 if ((!g_bAllowShowScaled) && (AIS_Toolbar_Switch_Next == 1))
13671 AIS_Toolbar_Switch_Next++;
13672 if (AIS_Toolbar_Switch_Next >= ArraySize)
13673 AIS_Toolbar_Switch_Next = 0; // If at end of cycle start at 0
13674
13675 // Set found values to global and member variables
13676 m_bShowAIS = bShowAIS_Array[AIS_Toolbar_Switch];
13677 m_bShowAISScaled = bShowScaled_Array[AIS_Toolbar_Switch];
13678}
13679
13680void ChartCanvas::TouchAISToolActive(void) {}
13681
13682void ChartCanvas::UpdateAISTBTool(void) {}
13683
13684//---------------------------------------------------------------------------------
13685//
13686// Compass/GPS status icon support
13687//
13688//---------------------------------------------------------------------------------
13689
13690void ChartCanvas::UpdateGPSCompassStatusBox(bool b_force_new) {
13691 // Look for change in overlap or positions
13692 bool b_update = false;
13693 int cc1_edge_comp = 2;
13694 wxRect rect = m_Compass->GetRect();
13695 wxSize parent_size = GetSize();
13696
13697 parent_size *= m_displayScale;
13698
13699 // check to see if it would overlap if it was in its home position (upper
13700 // right)
13701 wxPoint tentative_pt(parent_size.x - rect.width - cc1_edge_comp,
13702 g_StyleManager->GetCurrentStyle()->GetCompassYOffset());
13703 wxRect tentative_rect(tentative_pt, rect.GetSize());
13704
13705 m_Compass->Move(tentative_pt);
13706
13707 if (m_Compass && m_Compass->IsShown())
13708 m_Compass->UpdateStatus(b_force_new | b_update);
13709
13710 if (b_force_new | b_update) Refresh();
13711}
13712
13713void ChartCanvas::SelectChartFromStack(int index, bool bDir,
13714 ChartTypeEnum New_Type,
13715 ChartFamilyEnum New_Family) {
13716 if (!GetpCurrentStack()) return;
13717 if (!ChartData) return;
13718
13719 if (index < GetpCurrentStack()->nEntry) {
13720 // Open the new chart
13721 ChartBase *pTentative_Chart;
13722 pTentative_Chart = ChartData->OpenStackChartConditional(
13723 GetpCurrentStack(), index, bDir, New_Type, New_Family);
13724
13725 if (pTentative_Chart) {
13726 if (m_singleChart) m_singleChart->Deactivate();
13727
13728 m_singleChart = pTentative_Chart;
13729 m_singleChart->Activate();
13730
13731 GetpCurrentStack()->CurrentStackEntry = ChartData->GetStackEntry(
13732 GetpCurrentStack(), m_singleChart->GetFullPath());
13733 }
13734
13735 // Setup the view
13736 double zLat, zLon;
13737 if (m_bFollow) {
13738 zLat = gLat;
13739 zLon = gLon;
13740 } else {
13741 zLat = m_vLat;
13742 zLon = m_vLon;
13743 }
13744
13745 double best_scale_ppm = GetBestVPScale(m_singleChart);
13746 double rotation = GetVPRotation();
13747 double oldskew = GetVPSkew();
13748 double newskew = m_singleChart->GetChartSkew() * PI / 180.0;
13749
13750 if (!g_bskew_comp && (GetUpMode() == NORTH_UP_MODE)) {
13751 if (fabs(oldskew) > 0.0001) rotation = 0.0;
13752 if (fabs(newskew) > 0.0001) rotation = newskew;
13753 }
13754
13755 SetViewPoint(zLat, zLon, best_scale_ppm, newskew, rotation);
13756
13757 UpdateGPSCompassStatusBox(true); // Pick up the rotation
13758 }
13759
13760 // refresh Piano
13761 int idx = GetpCurrentStack()->GetCurrentEntrydbIndex();
13762 if (idx < 0) return;
13763
13764 std::vector<int> piano_active_chart_index_array;
13765 piano_active_chart_index_array.push_back(
13766 GetpCurrentStack()->GetCurrentEntrydbIndex());
13767 m_Piano->SetActiveKeyArray(piano_active_chart_index_array);
13768}
13769
13770void ChartCanvas::SelectdbChart(int dbindex) {
13771 if (!GetpCurrentStack()) return;
13772 if (!ChartData) return;
13773
13774 if (dbindex >= 0) {
13775 // Open the new chart
13776 ChartBase *pTentative_Chart;
13777 pTentative_Chart = ChartData->OpenChartFromDB(dbindex, FULL_INIT);
13778
13779 if (pTentative_Chart) {
13780 if (m_singleChart) m_singleChart->Deactivate();
13781
13782 m_singleChart = pTentative_Chart;
13783 m_singleChart->Activate();
13784
13785 GetpCurrentStack()->CurrentStackEntry = ChartData->GetStackEntry(
13786 GetpCurrentStack(), m_singleChart->GetFullPath());
13787 }
13788
13789 // Setup the view
13790 double zLat, zLon;
13791 if (m_bFollow) {
13792 zLat = gLat;
13793 zLon = gLon;
13794 } else {
13795 zLat = m_vLat;
13796 zLon = m_vLon;
13797 }
13798
13799 double best_scale_ppm = GetBestVPScale(m_singleChart);
13800
13801 if (m_singleChart)
13802 SetViewPoint(zLat, zLon, best_scale_ppm,
13803 m_singleChart->GetChartSkew() * PI / 180., GetVPRotation());
13804
13805 // SetChartUpdatePeriod( );
13806
13807 // UpdateGPSCompassStatusBox(); // Pick up the rotation
13808 }
13809
13810 // TODO refresh_Piano();
13811}
13812
13813void ChartCanvas::selectCanvasChartDisplay(int type, int family) {
13814 double target_scale = GetVP().view_scale_ppm;
13815
13816 if (!GetQuiltMode()) {
13817 if (GetpCurrentStack()) {
13818 int stack_index = -1;
13819 for (int i = 0; i < GetpCurrentStack()->nEntry; i++) {
13820 int check_dbIndex = GetpCurrentStack()->GetDBIndex(i);
13821 if (check_dbIndex < 0) continue;
13822 const ChartTableEntry &cte =
13823 ChartData->GetChartTableEntry(check_dbIndex);
13824 if (type == cte.GetChartType()) {
13825 stack_index = i;
13826 break;
13827 } else if (family == cte.GetChartFamily()) {
13828 stack_index = i;
13829 break;
13830 }
13831 }
13832
13833 if (stack_index >= 0) {
13834 SelectChartFromStack(stack_index);
13835 }
13836 }
13837 } else {
13838 int sel_dbIndex = -1;
13839 std::vector<int> piano_chart_index_array =
13840 GetQuiltExtendedStackdbIndexArray();
13841 for (unsigned int i = 0; i < piano_chart_index_array.size(); i++) {
13842 int check_dbIndex = piano_chart_index_array[i];
13843 const ChartTableEntry &cte = ChartData->GetChartTableEntry(check_dbIndex);
13844 if (type == cte.GetChartType()) {
13845 if (IsChartQuiltableRef(check_dbIndex)) {
13846 sel_dbIndex = check_dbIndex;
13847 break;
13848 }
13849 } else if (family == cte.GetChartFamily()) {
13850 if (IsChartQuiltableRef(check_dbIndex)) {
13851 sel_dbIndex = check_dbIndex;
13852 break;
13853 }
13854 }
13855 }
13856
13857 if (sel_dbIndex >= 0) {
13858 SelectQuiltRefdbChart(sel_dbIndex, false); // no autoscale
13859 // Re-qualify the quilt reference chart selection
13860 AdjustQuiltRefChart();
13861 }
13862
13863 // Now reset the scale to the target...
13864 SetVPScale(target_scale);
13865 }
13866
13867 SetQuiltChartHiLiteIndex(-1);
13868
13869 ReloadVP();
13870}
13871
13872bool ChartCanvas::IsTileOverlayIndexInYesShow(int index) {
13873 return std::find(m_tile_yesshow_index_array.begin(),
13874 m_tile_yesshow_index_array.end(),
13875 index) != m_tile_yesshow_index_array.end();
13876}
13877
13878bool ChartCanvas::IsTileOverlayIndexInNoShow(int index) {
13879 return std::find(m_tile_noshow_index_array.begin(),
13880 m_tile_noshow_index_array.end(),
13881 index) != m_tile_noshow_index_array.end();
13882}
13883
13884void ChartCanvas::AddTileOverlayIndexToNoShow(int index) {
13885 if (std::find(m_tile_noshow_index_array.begin(),
13886 m_tile_noshow_index_array.end(),
13887 index) == m_tile_noshow_index_array.end()) {
13888 m_tile_noshow_index_array.push_back(index);
13889 }
13890}
13891
13892//-------------------------------------------------------------------------------------------------------
13893//
13894// Piano support
13895//
13896//-------------------------------------------------------------------------------------------------------
13897
13898void ChartCanvas::HandlePianoClick(
13899 int selected_index, const std::vector<int> &selected_dbIndex_array) {
13900 if (g_options && g_options->IsShown())
13901 return; // Piano might be invalid due to chartset updates.
13902 if (!m_pCurrentStack) return;
13903 if (!ChartData) return;
13904
13905 // stop movement or on slow computer we may get something like :
13906 // zoom out with the wheel (timer is set)
13907 // quickly click and display a chart, which may zoom in
13908 // but the delayed timer fires first and it zooms out again!
13909 StopMovement();
13910
13911 // When switching by piano key click, we may appoint the new target chart to
13912 // be any chart in the composite array.
13913 // As an improvement to UX, find the chart that is "closest" to the current
13914 // vp,
13915 // and select that chart. This will cause a jump to the centroid of that
13916 // chart
13917
13918 double distance = 25000; // RTW
13919 int closest_index = -1;
13920 for (int chart_index : selected_dbIndex_array) {
13921 const ChartTableEntry &cte = ChartData->GetChartTableEntry(chart_index);
13922 double chart_lat = (cte.GetLatMax() + cte.GetLatMin()) / 2;
13923 double chart_lon = (cte.GetLonMax() + cte.GetLonMin()) / 2;
13924
13925 // measure distance as Manhattan style
13926 double test_distance = abs(m_vLat - chart_lat) + abs(m_vLon - chart_lon);
13927 if (test_distance < distance) {
13928 distance = test_distance;
13929 closest_index = chart_index;
13930 }
13931 }
13932
13933 int selected_dbIndex = selected_dbIndex_array[0];
13934 if (closest_index >= 0) selected_dbIndex = closest_index;
13935
13936 if (!GetQuiltMode()) {
13937 if (m_bpersistent_quilt /* && g_bQuiltEnable*/) {
13938 if (IsChartQuiltableRef(selected_dbIndex)) {
13939 ToggleCanvasQuiltMode();
13940 SelectQuiltRefdbChart(selected_dbIndex);
13941 m_bpersistent_quilt = false;
13942 } else {
13943 SelectChartFromStack(selected_index);
13944 }
13945 } else {
13946 SelectChartFromStack(selected_index);
13947 g_sticky_chart = selected_dbIndex;
13948 }
13949
13950 if (m_singleChart)
13951 GetVP().SetProjectionType(m_singleChart->GetChartProjectionType());
13952 } else {
13953 // Handle MBTiles overlays first
13954 // Left click simply toggles the noshow array index entry
13955 if (CHART_TYPE_MBTILES == ChartData->GetDBChartType(selected_dbIndex)) {
13956 bool bfound = false;
13957 for (unsigned int i = 0; i < m_tile_noshow_index_array.size(); i++) {
13958 if (m_tile_noshow_index_array[i] ==
13959 selected_dbIndex) { // chart is in the noshow list
13960 m_tile_noshow_index_array.erase(m_tile_noshow_index_array.begin() +
13961 i); // erase it
13962 bfound = true;
13963 break;
13964 }
13965 }
13966 if (!bfound) {
13967 m_tile_noshow_index_array.push_back(selected_dbIndex);
13968 }
13969
13970 // If not already present, add this tileset to the "yes_show" array.
13971 if (!IsTileOverlayIndexInYesShow(selected_dbIndex))
13972 m_tile_yesshow_index_array.push_back(selected_dbIndex);
13973 }
13974
13975 else {
13976 if (IsChartQuiltableRef(selected_dbIndex)) {
13977 // if( ChartData ) ChartData->PurgeCache();
13978
13979 // If the chart is a vector chart, and of very large scale,
13980 // then we had better set the new scale directly to avoid excessive
13981 // underzoom on, eg, Inland ENCs
13982 bool set_scale = false;
13983 if (CHART_TYPE_S57 == ChartData->GetDBChartType(selected_dbIndex)) {
13984 if (ChartData->GetDBChartScale(selected_dbIndex) < 5000) {
13985 set_scale = true;
13986 }
13987 }
13988
13989 if (!set_scale) {
13990 SelectQuiltRefdbChart(selected_dbIndex, true); // autoscale
13991 } else {
13992 SelectQuiltRefdbChart(selected_dbIndex, false); // no autoscale
13993
13994 // Adjust scale so that the selected chart is underzoomed/overzoomed
13995 // by a controlled amount
13996 ChartBase *pc =
13997 ChartData->OpenChartFromDB(selected_dbIndex, FULL_INIT);
13998 if (pc) {
13999 double proposed_scale_onscreen =
14001
14002 if (g_bPreserveScaleOnX) {
14003 proposed_scale_onscreen =
14004 wxMin(proposed_scale_onscreen,
14005 100 * pc->GetNormalScaleMax(GetCanvasScaleFactor(),
14006 GetCanvasWidth()));
14007 } else {
14008 proposed_scale_onscreen =
14009 wxMin(proposed_scale_onscreen,
14010 20 * pc->GetNormalScaleMax(GetCanvasScaleFactor(),
14011 GetCanvasWidth()));
14012
14013 proposed_scale_onscreen =
14014 wxMax(proposed_scale_onscreen,
14015 pc->GetNormalScaleMin(GetCanvasScaleFactor(),
14016 g_b_overzoom_x));
14017 }
14018
14019 SetVPScale(GetCanvasScaleFactor() / proposed_scale_onscreen);
14020 }
14021 }
14022 } else {
14023 ToggleCanvasQuiltMode();
14024 SelectdbChart(selected_dbIndex);
14025 m_bpersistent_quilt = true;
14026 }
14027 }
14028 }
14029
14030 SetQuiltChartHiLiteIndex(-1);
14031 gFrame->UpdateGlobalMenuItems(); // update the state of the menu items
14032 // (checkmarks etc)
14033 HideChartInfoWindow();
14034 DoCanvasUpdate();
14035 ReloadVP(); // Pick up the new selections
14036}
14037
14038void ChartCanvas::HandlePianoRClick(
14039 int x, int y, int selected_index,
14040 const std::vector<int> &selected_dbIndex_array) {
14041 if (g_options && g_options->IsShown())
14042 return; // Piano might be invalid due to chartset updates.
14043 if (!GetpCurrentStack()) return;
14044
14045 PianoPopupMenu(x, y, selected_index, selected_dbIndex_array);
14046 UpdateCanvasControlBar();
14047
14048 SetQuiltChartHiLiteIndex(-1);
14049}
14050
14051void ChartCanvas::HandlePianoRollover(
14052 int selected_index, const std::vector<int> &selected_dbIndex_array,
14053 int n_charts, int scale) {
14054 if (g_options && g_options->IsShown())
14055 return; // Piano might be invalid due to chartset updates.
14056 if (!GetpCurrentStack()) return;
14057 if (!ChartData) return;
14058
14059 if (ChartData->IsBusy()) return;
14060
14061 wxPoint key_location = m_Piano->GetKeyOrigin(selected_index);
14062
14063 if (!GetQuiltMode()) {
14064 ShowChartInfoWindow(key_location.x, selected_dbIndex_array[0]);
14065 } else {
14066 // Select the correct vector
14067 std::vector<int> piano_chart_index_array;
14068 if (m_Piano->GetPianoMode() == PIANO_MODE_LEGACY) {
14069 piano_chart_index_array = GetQuiltExtendedStackdbIndexArray();
14070 if ((GetpCurrentStack()->nEntry > 1) ||
14071 (piano_chart_index_array.size() >= 1)) {
14072 ShowChartInfoWindow(key_location.x, selected_dbIndex_array[0]);
14073
14074 SetQuiltChartHiLiteIndexArray(selected_dbIndex_array);
14075 ReloadVP(false); // no VP adjustment allowed
14076 } else if (GetpCurrentStack()->nEntry == 1) {
14077 const ChartTableEntry &cte =
14078 ChartData->GetChartTableEntry(GetpCurrentStack()->GetDBIndex(0));
14079 if (CHART_TYPE_CM93COMP != cte.GetChartType()) {
14080 ShowChartInfoWindow(key_location.x, selected_dbIndex_array[0]);
14081 ReloadVP(false);
14082 } else if ((-1 == selected_index) &&
14083 (0 == selected_dbIndex_array.size())) {
14084 ShowChartInfoWindow(key_location.x, -1);
14085 }
14086 }
14087 } else {
14088 piano_chart_index_array = GetQuiltFullScreendbIndexArray();
14089
14090 if ((GetpCurrentStack()->nEntry > 1) ||
14091 (piano_chart_index_array.size() >= 1)) {
14092 if (n_charts > 1)
14093 ShowCompositeInfoWindow(key_location.x, n_charts, scale,
14094 selected_dbIndex_array);
14095 else if (n_charts == 1)
14096 ShowChartInfoWindow(key_location.x, selected_dbIndex_array[0]);
14097
14098 SetQuiltChartHiLiteIndexArray(selected_dbIndex_array);
14099 ReloadVP(false); // no VP adjustment allowed
14100 }
14101 }
14102 }
14103}
14104
14105void ChartCanvas::ClearPianoRollover() {
14106 ClearQuiltChartHiLiteIndexArray();
14107 ShowChartInfoWindow(0, -1);
14108 std::vector<int> vec;
14109 ShowCompositeInfoWindow(0, 0, 0, vec);
14110 ReloadVP(false);
14111}
14112
14113void ChartCanvas::UpdateCanvasControlBar(void) {
14114 if (m_pianoFrozen) return;
14115
14116 if (!GetpCurrentStack()) return;
14117 if (!ChartData) return;
14118 if (!g_bShowChartBar) return;
14119
14120 int sel_type = -1;
14121 int sel_family = -1;
14122
14123 std::vector<int> piano_chart_index_array;
14124 std::vector<int> empty_piano_chart_index_array;
14125
14126 wxString old_hash = m_Piano->GetStoredHash();
14127
14128 if (GetQuiltMode()) {
14129 m_Piano->SetKeyArray(GetQuiltExtendedStackdbIndexArray(),
14130 GetQuiltFullScreendbIndexArray());
14131
14132 std::vector<int> piano_active_chart_index_array =
14133 GetQuiltCandidatedbIndexArray();
14134 m_Piano->SetActiveKeyArray(piano_active_chart_index_array);
14135
14136 std::vector<int> piano_eclipsed_chart_index_array =
14137 GetQuiltEclipsedStackdbIndexArray();
14138 m_Piano->SetEclipsedIndexArray(piano_eclipsed_chart_index_array);
14139
14140 m_Piano->SetNoshowIndexArray(m_quilt_noshow_index_array);
14141 m_Piano->AddNoshowIndexArray(m_tile_noshow_index_array);
14142
14143 sel_type = ChartData->GetDBChartType(GetQuiltReferenceChartIndex());
14144 sel_family = ChartData->GetDBChartFamily(GetQuiltReferenceChartIndex());
14145 } else {
14146 piano_chart_index_array = ChartData->GetCSArray(GetpCurrentStack());
14147 m_Piano->SetKeyArray(piano_chart_index_array, piano_chart_index_array);
14148 // TODO refresh_Piano();
14149
14150 if (m_singleChart) {
14151 sel_type = m_singleChart->GetChartType();
14152 sel_family = m_singleChart->GetChartFamily();
14153 }
14154 }
14155
14156 // Set up the TMerc and Skew arrays
14157 std::vector<int> piano_skew_chart_index_array;
14158 std::vector<int> piano_tmerc_chart_index_array;
14159 std::vector<int> piano_poly_chart_index_array;
14160
14161 for (unsigned int ino = 0; ino < piano_chart_index_array.size(); ino++) {
14162 const ChartTableEntry &ctei =
14163 ChartData->GetChartTableEntry(piano_chart_index_array[ino]);
14164 double skew_norm = ctei.GetChartSkew();
14165 if (skew_norm > 180.) skew_norm -= 360.;
14166
14167 if (ctei.GetChartProjectionType() == PROJECTION_TRANSVERSE_MERCATOR)
14168 piano_tmerc_chart_index_array.push_back(piano_chart_index_array[ino]);
14169
14170 // Polyconic skewed charts should show as skewed
14171 else if (ctei.GetChartProjectionType() == PROJECTION_POLYCONIC) {
14172 if (fabs(skew_norm) > 1.)
14173 piano_skew_chart_index_array.push_back(piano_chart_index_array[ino]);
14174 else
14175 piano_poly_chart_index_array.push_back(piano_chart_index_array[ino]);
14176 } else if (fabs(skew_norm) > 1.)
14177 piano_skew_chart_index_array.push_back(piano_chart_index_array[ino]);
14178 }
14179 m_Piano->SetSkewIndexArray(piano_skew_chart_index_array);
14180 m_Piano->SetTmercIndexArray(piano_tmerc_chart_index_array);
14181 m_Piano->SetPolyIndexArray(piano_poly_chart_index_array);
14182
14183 wxString new_hash = m_Piano->GenerateAndStoreNewHash();
14184 if (new_hash != old_hash) {
14185 m_Piano->FormatKeys();
14186 HideChartInfoWindow();
14187 m_Piano->ResetRollover();
14188 SetQuiltChartHiLiteIndex(-1);
14189 m_brepaint_piano = true;
14190 }
14191
14192 // Create a bitmask int that describes what Family/Type of charts are shown in
14193 // the bar, and notify the platform.
14194 int mask = 0;
14195 for (unsigned int ino = 0; ino < piano_chart_index_array.size(); ino++) {
14196 const ChartTableEntry &ctei =
14197 ChartData->GetChartTableEntry(piano_chart_index_array[ino]);
14198 ChartFamilyEnum e = (ChartFamilyEnum)ctei.GetChartFamily();
14199 ChartTypeEnum t = (ChartTypeEnum)ctei.GetChartType();
14200 if (e == CHART_FAMILY_RASTER) mask |= 1;
14201 if (e == CHART_FAMILY_VECTOR) {
14202 if (t == CHART_TYPE_CM93COMP)
14203 mask |= 4;
14204 else
14205 mask |= 2;
14206 }
14207 }
14208
14209 wxString s_indicated;
14210 if (sel_type == CHART_TYPE_CM93COMP)
14211 s_indicated = _T("cm93");
14212 else {
14213 if (sel_family == CHART_FAMILY_RASTER)
14214 s_indicated = _T("raster");
14215 else if (sel_family == CHART_FAMILY_VECTOR)
14216 s_indicated = _T("vector");
14217 }
14218
14219 g_Platform->setChartTypeMaskSel(mask, s_indicated);
14220}
14221
14222void ChartCanvas::FormatPianoKeys(void) { m_Piano->FormatKeys(); }
14223
14224void ChartCanvas::PianoPopupMenu(
14225 int x, int y, int selected_index,
14226 const std::vector<int> &selected_dbIndex_array) {
14227 if (!GetpCurrentStack()) return;
14228
14229 // No context menu if quilting is disabled
14230 if (!GetQuiltMode()) return;
14231
14232 m_piano_ctx_menu = new wxMenu();
14233
14234 if (m_Piano->GetPianoMode() == PIANO_MODE_COMPOSITE) {
14235 // m_piano_ctx_menu->Append(ID_PIANO_EXPAND_PIANO, _("Legacy chartbar"));
14236 // Connect(ID_PIANO_EXPAND_PIANO, wxEVT_COMMAND_MENU_SELECTED,
14237 // wxCommandEventHandler(ChartCanvas::OnPianoMenuExpandChartbar));
14238 } else {
14239 // m_piano_ctx_menu->Append(ID_PIANO_CONTRACT_PIANO, _("Fullscreen
14240 // chartbar")); Connect(ID_PIANO_CONTRACT_PIANO,
14241 // wxEVT_COMMAND_MENU_SELECTED,
14242 // wxCommandEventHandler(ChartCanvas::OnPianoMenuContractChartbar));
14243
14244 menu_selected_dbIndex = selected_dbIndex_array[0];
14245 menu_selected_index = selected_index;
14246
14247 // Search the no-show array
14248 bool b_is_in_noshow = false;
14249 for (unsigned int i = 0; i < m_quilt_noshow_index_array.size(); i++) {
14250 if (m_quilt_noshow_index_array[i] ==
14251 menu_selected_dbIndex) // chart is in the noshow list
14252 {
14253 b_is_in_noshow = true;
14254 break;
14255 }
14256 }
14257
14258 if (b_is_in_noshow) {
14259 m_piano_ctx_menu->Append(ID_PIANO_ENABLE_QUILT_CHART,
14260 _("Show This Chart"));
14261 Connect(ID_PIANO_ENABLE_QUILT_CHART, wxEVT_COMMAND_MENU_SELECTED,
14262 wxCommandEventHandler(ChartCanvas::OnPianoMenuEnableChart));
14263 } else if (GetpCurrentStack()->nEntry > 1) {
14264 m_piano_ctx_menu->Append(ID_PIANO_DISABLE_QUILT_CHART,
14265 _("Hide This Chart"));
14266 Connect(ID_PIANO_DISABLE_QUILT_CHART, wxEVT_COMMAND_MENU_SELECTED,
14267 wxCommandEventHandler(ChartCanvas::OnPianoMenuDisableChart));
14268 }
14269 }
14270
14271 wxPoint pos = wxPoint(x, y - 30);
14272
14273 // Invoke the drop-down menu
14274 if (m_piano_ctx_menu->GetMenuItems().GetCount())
14275 PopupMenu(m_piano_ctx_menu, pos);
14276
14277 delete m_piano_ctx_menu;
14278 m_piano_ctx_menu = NULL;
14279
14280 HideChartInfoWindow();
14281 m_Piano->ResetRollover();
14282
14283 SetQuiltChartHiLiteIndex(-1);
14284 ClearQuiltChartHiLiteIndexArray();
14285
14286 ReloadVP();
14287}
14288
14289void ChartCanvas::OnPianoMenuEnableChart(wxCommandEvent &event) {
14290 for (unsigned int i = 0; i < m_quilt_noshow_index_array.size(); i++) {
14291 if (m_quilt_noshow_index_array[i] ==
14292 menu_selected_dbIndex) // chart is in the noshow list
14293 {
14294 m_quilt_noshow_index_array.erase(m_quilt_noshow_index_array.begin() + i);
14295 break;
14296 }
14297 }
14298}
14299
14300void ChartCanvas::OnPianoMenuDisableChart(wxCommandEvent &event) {
14301 if (!GetpCurrentStack()) return;
14302 if (!ChartData) return;
14303
14304 RemoveChartFromQuilt(menu_selected_dbIndex);
14305
14306 // It could happen that the chart being disabled is the reference
14307 // chart....
14308 if (menu_selected_dbIndex == GetQuiltRefChartdbIndex()) {
14309 int type = ChartData->GetDBChartType(menu_selected_dbIndex);
14310
14311 int i = menu_selected_index + 1; // select next smaller scale chart
14312 bool b_success = false;
14313 while (i < GetpCurrentStack()->nEntry - 1) {
14314 int dbIndex = GetpCurrentStack()->GetDBIndex(i);
14315 if (type == ChartData->GetDBChartType(dbIndex)) {
14316 SelectQuiltRefChart(i);
14317 b_success = true;
14318 break;
14319 }
14320 i++;
14321 }
14322
14323 // If that did not work, try to select the next larger scale compatible
14324 // chart
14325 if (!b_success) {
14326 i = menu_selected_index - 1;
14327 while (i > 0) {
14328 int dbIndex = GetpCurrentStack()->GetDBIndex(i);
14329 if (type == ChartData->GetDBChartType(dbIndex)) {
14330 SelectQuiltRefChart(i);
14331 b_success = true;
14332 break;
14333 }
14334 i--;
14335 }
14336 }
14337 }
14338}
14339
14340void ChartCanvas::RemoveChartFromQuilt(int dbIndex) {
14341 // Remove the item from the list (if it appears) to avoid multiple addition
14342 for (unsigned int i = 0; i < m_quilt_noshow_index_array.size(); i++) {
14343 if (m_quilt_noshow_index_array[i] ==
14344 dbIndex) // chart is already in the noshow list
14345 {
14346 m_quilt_noshow_index_array.erase(m_quilt_noshow_index_array.begin() + i);
14347 break;
14348 }
14349 }
14350
14351 m_quilt_noshow_index_array.push_back(dbIndex);
14352}
14353
14354bool ChartCanvas::UpdateS52State() {
14355 bool retval = false;
14356 // printf(" update %d\n", IsPrimaryCanvas());
14357
14358 if (ps52plib) {
14359 ps52plib->SetShowS57Text(m_encShowText);
14360 ps52plib->SetDisplayCategory((DisCat)m_encDisplayCategory);
14361 ps52plib->m_bShowSoundg = m_encShowDepth;
14362 ps52plib->m_bShowAtonText = m_encShowBuoyLabels;
14363 ps52plib->m_bShowLdisText = m_encShowLightDesc;
14364
14365 // Lights
14366 if (!m_encShowLights) // On, going off
14367 ps52plib->AddObjNoshow("LIGHTS");
14368 else // Off, going on
14369 ps52plib->RemoveObjNoshow("LIGHTS");
14370 ps52plib->SetLightsOff(!m_encShowLights);
14371 ps52plib->m_bExtendLightSectors = true;
14372
14373 // TODO ps52plib->m_bShowAtons = m_encShowBuoys;
14374 ps52plib->SetAnchorOn(m_encShowAnchor);
14375 ps52plib->SetQualityOfData(m_encShowDataQual);
14376 }
14377
14378 return retval;
14379}
14380
14381void ChartCanvas::SetShowENCDataQual(bool show) {
14382 m_encShowDataQual = show;
14383 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14384 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14385
14386 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14387}
14388
14389void ChartCanvas::SetShowENCText(bool show) {
14390 m_encShowText = show;
14391 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14392 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14393
14394 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14395}
14396
14397void ChartCanvas::SetENCDisplayCategory(int category) {
14398 m_encDisplayCategory = category;
14399 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14400}
14401
14402void ChartCanvas::SetShowENCDepth(bool show) {
14403 m_encShowDepth = show;
14404 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14405 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14406
14407 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14408}
14409
14410void ChartCanvas::SetShowENCLightDesc(bool show) {
14411 m_encShowLightDesc = show;
14412 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14413 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14414
14415 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14416}
14417
14418void ChartCanvas::SetShowENCBuoyLabels(bool show) {
14419 m_encShowBuoyLabels = show;
14420 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14421}
14422
14423void ChartCanvas::SetShowENCLights(bool show) {
14424 m_encShowLights = show;
14425 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14426 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14427
14428 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14429}
14430
14431void ChartCanvas::SetShowENCAnchor(bool show) {
14432 m_encShowAnchor = show;
14433 if (GetMUIBar() && GetMUIBar()->GetCanvasOptions())
14434 GetMUIBar()->GetCanvasOptions()->RefreshControlValues();
14435
14436 m_s52StateHash = 0; // Force a S52 PLIB re-configure
14437}
14438
14439wxRect ChartCanvas::GetMUIBarRect() {
14440 wxRect rv;
14441 if (m_muiBar) {
14442 rv = m_muiBar->GetRect();
14443 }
14444
14445 return rv;
14446}
14447
14448void ChartCanvas::RenderAlertMessage(wxDC &dc, const ViewPort &vp) {
14449 if (!GetAlertString().IsEmpty()) {
14450 wxFont *pfont = wxTheFontList->FindOrCreateFont(
14451 10, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL);
14452
14453 dc.SetFont(*pfont);
14454 dc.SetPen(*wxTRANSPARENT_PEN);
14455
14456 dc.SetBrush(wxColour(243, 229, 47));
14457 int w, h;
14458 dc.GetMultiLineTextExtent(GetAlertString(), &w, &h);
14459 h += 2;
14460 // int yp = vp.pix_height - 20 - h;
14461
14462 wxRect sbr = GetScaleBarRect();
14463 int xp = sbr.x + sbr.width + 10;
14464 int yp = (sbr.y + sbr.height) - h;
14465
14466 int wdraw = w + 10;
14467 dc.DrawRectangle(xp, yp, wdraw, h);
14468 dc.DrawLabel(GetAlertString(), wxRect(xp, yp, wdraw, h),
14469 wxALIGN_CENTRE_HORIZONTAL | wxALIGN_CENTRE_VERTICAL);
14470 }
14471}
14472
14473//--------------------------------------------------------------------------------------------------------
14474// Screen Brightness Control Support Routines
14475//
14476//--------------------------------------------------------------------------------------------------------
14477
14478#ifdef __UNIX__
14479#define BRIGHT_XCALIB
14480#define __OPCPN_USEICC__
14481#endif
14482
14483#ifdef __OPCPN_USEICC__
14484int CreateSimpleICCProfileFile(const char *file_name, double co_red,
14485 double co_green, double co_blue);
14486
14487wxString temp_file_name;
14488#endif
14489
14490#if 0
14491class ocpnCurtain: public wxDialog
14492{
14493 DECLARE_CLASS( ocpnCurtain )
14494 DECLARE_EVENT_TABLE()
14495
14496public:
14497 ocpnCurtain( wxWindow *parent, wxPoint position, wxSize size, long wstyle );
14498 ~ocpnCurtain( );
14499 bool ProcessEvent(wxEvent& event);
14500
14501};
14502
14503IMPLEMENT_CLASS ( ocpnCurtain, wxDialog )
14504
14505BEGIN_EVENT_TABLE(ocpnCurtain, wxDialog)
14506END_EVENT_TABLE()
14507
14508ocpnCurtain::ocpnCurtain( wxWindow *parent, wxPoint position, wxSize size, long wstyle )
14509{
14510 wxDialog::Create( parent, -1, _T("ocpnCurtain"), position, size, wxNO_BORDER | wxSTAY_ON_TOP );
14511}
14512
14513ocpnCurtain::~ocpnCurtain()
14514{
14515}
14516
14517bool ocpnCurtain::ProcessEvent(wxEvent& event)
14518{
14519 GetParent()->GetEventHandler()->SetEvtHandlerEnabled(true);
14520 return GetParent()->GetEventHandler()->ProcessEvent(event);
14521}
14522#endif
14523
14524#ifdef _WIN32
14525#include <windows.h>
14526
14527HMODULE hGDI32DLL;
14528typedef BOOL(WINAPI *SetDeviceGammaRamp_ptr_type)(HDC hDC, LPVOID lpRampTable);
14529typedef BOOL(WINAPI *GetDeviceGammaRamp_ptr_type)(HDC hDC, LPVOID lpRampTable);
14530SetDeviceGammaRamp_ptr_type
14531 g_pSetDeviceGammaRamp; // the API entry points in the dll
14532GetDeviceGammaRamp_ptr_type g_pGetDeviceGammaRamp;
14533
14534WORD *g_pSavedGammaMap;
14535
14536#endif
14537
14538int InitScreenBrightness(void) {
14539#ifdef _WIN32
14540#ifdef ocpnUSE_GL
14541 if (gFrame->GetPrimaryCanvas()->GetglCanvas() && g_bopengl) {
14542 HDC hDC;
14543 BOOL bbr;
14544
14545 if (NULL == hGDI32DLL) {
14546 hGDI32DLL = LoadLibrary(TEXT("gdi32.dll"));
14547
14548 if (NULL != hGDI32DLL) {
14549 // Get the entry points of the required functions
14550 g_pSetDeviceGammaRamp = (SetDeviceGammaRamp_ptr_type)GetProcAddress(
14551 hGDI32DLL, "SetDeviceGammaRamp");
14552 g_pGetDeviceGammaRamp = (GetDeviceGammaRamp_ptr_type)GetProcAddress(
14553 hGDI32DLL, "GetDeviceGammaRamp");
14554
14555 // If the functions are not found, unload the DLL and return false
14556 if ((NULL == g_pSetDeviceGammaRamp) ||
14557 (NULL == g_pGetDeviceGammaRamp)) {
14558 FreeLibrary(hGDI32DLL);
14559 hGDI32DLL = NULL;
14560 return 0;
14561 }
14562 }
14563 }
14564
14565 // Interface is ready, so....
14566 // Get some storage
14567 if (!g_pSavedGammaMap) {
14568 g_pSavedGammaMap = (WORD *)malloc(3 * 256 * sizeof(WORD));
14569
14570 hDC = GetDC(NULL); // Get the full screen DC
14571 bbr = g_pGetDeviceGammaRamp(
14572 hDC, g_pSavedGammaMap); // Get the existing ramp table
14573 ReleaseDC(NULL, hDC); // Release the DC
14574 }
14575
14576 // On Windows hosts, try to adjust the registry to allow full range
14577 // setting of Gamma table This is an undocumented Windows hack.....
14578 wxRegKey *pRegKey = new wxRegKey(
14579 _T("HKEY_LOCAL_MACHINE\\SOFTWARE\\Microsoft\\Windows ")
14580 _T("NT\\CurrentVersion\\ICM"));
14581 if (!pRegKey->Exists()) pRegKey->Create();
14582 pRegKey->SetValue(_T("GdiIcmGammaRange"), 256);
14583
14584 g_brightness_init = true;
14585 return 1;
14586 }
14587#endif
14588
14589 {
14590 if (NULL == g_pcurtain) {
14591 if (gFrame->CanSetTransparent()) {
14592 // Build the curtain window
14593 g_pcurtain = new wxDialog(gFrame->GetPrimaryCanvas(), -1, _T(""),
14594 wxPoint(0, 0), ::wxGetDisplaySize(),
14595 wxNO_BORDER | wxTRANSPARENT_WINDOW |
14596 wxSTAY_ON_TOP | wxDIALOG_NO_PARENT);
14597
14598 // g_pcurtain = new ocpnCurtain(gFrame,
14599 // wxPoint(0,0),::wxGetDisplaySize(),
14600 // wxNO_BORDER | wxTRANSPARENT_WINDOW
14601 // |wxSTAY_ON_TOP | wxDIALOG_NO_PARENT);
14602
14603 g_pcurtain->Hide();
14604
14605 HWND hWnd = GetHwndOf(g_pcurtain);
14606 SetWindowLong(hWnd, GWL_EXSTYLE,
14607 GetWindowLong(hWnd, GWL_EXSTYLE) | ~WS_EX_APPWINDOW);
14608 g_pcurtain->SetBackgroundColour(wxColour(0, 0, 0));
14609 g_pcurtain->SetTransparent(0);
14610
14611 g_pcurtain->Maximize();
14612 g_pcurtain->Show();
14613
14614 // All of this is obtuse, but necessary for Windows...
14615 g_pcurtain->Enable();
14616 g_pcurtain->Disable();
14617
14618 gFrame->Disable();
14619 gFrame->Enable();
14620 // SetFocus();
14621 }
14622 }
14623 g_brightness_init = true;
14624
14625 return 1;
14626 }
14627#else
14628 // Look for "xcalib" application
14629 wxString cmd(_T ( "xcalib -version" ));
14630
14631 wxArrayString output;
14632 long r = wxExecute(cmd, output);
14633 if (0 != r)
14634 wxLogMessage(
14635 _T(" External application \"xcalib\" not found. Screen brightness ")
14636 _T("not changed."));
14637
14638 g_brightness_init = true;
14639 return 0;
14640#endif
14641}
14642
14643int RestoreScreenBrightness(void) {
14644#ifdef _WIN32
14645
14646 if (g_pSavedGammaMap) {
14647 HDC hDC = GetDC(NULL); // Get the full screen DC
14648 g_pSetDeviceGammaRamp(hDC,
14649 g_pSavedGammaMap); // Restore the saved ramp table
14650 ReleaseDC(NULL, hDC); // Release the DC
14651
14652 free(g_pSavedGammaMap);
14653 g_pSavedGammaMap = NULL;
14654 }
14655
14656 if (g_pcurtain) {
14657 g_pcurtain->Close();
14658 g_pcurtain->Destroy();
14659 g_pcurtain = NULL;
14660 }
14661
14662 g_brightness_init = false;
14663 return 1;
14664
14665#endif
14666
14667#ifdef BRIGHT_XCALIB
14668 if (g_brightness_init) {
14669 wxString cmd;
14670 cmd = _T("xcalib -clear");
14671 wxExecute(cmd, wxEXEC_ASYNC);
14672 g_brightness_init = false;
14673 }
14674
14675 return 1;
14676#endif
14677
14678 return 0;
14679}
14680
14681// Set brightness. [0..100]
14682int SetScreenBrightness(int brightness) {
14683#ifdef _WIN32
14684
14685 // Under Windows, we use the SetDeviceGammaRamp function which exists in
14686 // some (most modern?) versions of gdi32.dll Load the required library dll,
14687 // if not already in place
14688#ifdef ocpnUSE_GL
14689 if (gFrame->GetPrimaryCanvas()->GetglCanvas() && g_bopengl) {
14690 if (g_pcurtain) {
14691 g_pcurtain->Close();
14692 g_pcurtain->Destroy();
14693 g_pcurtain = NULL;
14694 }
14695
14696 InitScreenBrightness();
14697
14698 if (NULL == hGDI32DLL) {
14699 // Unicode stuff.....
14700 wchar_t wdll_name[80];
14701 MultiByteToWideChar(0, 0, "gdi32.dll", -1, wdll_name, 80);
14702 LPCWSTR cstr = wdll_name;
14703
14704 hGDI32DLL = LoadLibrary(cstr);
14705
14706 if (NULL != hGDI32DLL) {
14707 // Get the entry points of the required functions
14708 g_pSetDeviceGammaRamp = (SetDeviceGammaRamp_ptr_type)GetProcAddress(
14709 hGDI32DLL, "SetDeviceGammaRamp");
14710 g_pGetDeviceGammaRamp = (GetDeviceGammaRamp_ptr_type)GetProcAddress(
14711 hGDI32DLL, "GetDeviceGammaRamp");
14712
14713 // If the functions are not found, unload the DLL and return false
14714 if ((NULL == g_pSetDeviceGammaRamp) ||
14715 (NULL == g_pGetDeviceGammaRamp)) {
14716 FreeLibrary(hGDI32DLL);
14717 hGDI32DLL = NULL;
14718 return 0;
14719 }
14720 }
14721 }
14722
14723 HDC hDC = GetDC(NULL); // Get the full screen DC
14724
14725 /*
14726 int cmcap = GetDeviceCaps(hDC, COLORMGMTCAPS);
14727 if (cmcap != CM_GAMMA_RAMP)
14728 {
14729 wxLogMessage(_T(" Video hardware does not support brightness control by
14730 gamma ramp adjustment.")); return false;
14731 }
14732 */
14733
14734 int increment = brightness * 256 / 100;
14735
14736 // Build the Gamma Ramp table
14737 WORD GammaTable[3][256];
14738
14739 int table_val = 0;
14740 for (int i = 0; i < 256; i++) {
14741 GammaTable[0][i] = r_gamma_mult * (WORD)table_val;
14742 GammaTable[1][i] = g_gamma_mult * (WORD)table_val;
14743 GammaTable[2][i] = b_gamma_mult * (WORD)table_val;
14744
14745 table_val += increment;
14746
14747 if (table_val > 65535) table_val = 65535;
14748 }
14749
14750 g_pSetDeviceGammaRamp(hDC, GammaTable); // Set the ramp table
14751 ReleaseDC(NULL, hDC); // Release the DC
14752
14753 return 1;
14754 }
14755#endif
14756
14757 {
14758 if (g_pSavedGammaMap) {
14759 HDC hDC = GetDC(NULL); // Get the full screen DC
14760 g_pSetDeviceGammaRamp(hDC,
14761 g_pSavedGammaMap); // Restore the saved ramp table
14762 ReleaseDC(NULL, hDC); // Release the DC
14763 }
14764
14765 if (brightness < 100) {
14766 if (NULL == g_pcurtain) InitScreenBrightness();
14767
14768 if (g_pcurtain) {
14769 int sbrite = wxMax(1, brightness);
14770 sbrite = wxMin(100, sbrite);
14771
14772 g_pcurtain->SetTransparent((100 - sbrite) * 256 / 100);
14773 }
14774 } else {
14775 if (g_pcurtain) {
14776 g_pcurtain->Close();
14777 g_pcurtain->Destroy();
14778 g_pcurtain = NULL;
14779 }
14780 }
14781
14782 return 1;
14783 }
14784
14785#endif
14786
14787#ifdef BRIGHT_XCALIB
14788
14789 if (!g_brightness_init) {
14790 last_brightness = 100;
14791 g_brightness_init = true;
14792 temp_file_name = wxFileName::CreateTempFileName(_T(""));
14793 InitScreenBrightness();
14794 }
14795
14796#ifdef __OPCPN_USEICC__
14797 // Create a dead simple temporary ICC profile file, with gamma ramps set as
14798 // desired, and then activate this temporary profile using xcalib <filename>
14799 if (!CreateSimpleICCProfileFile(
14800 (const char *)temp_file_name.fn_str(), brightness * r_gamma_mult,
14801 brightness * g_gamma_mult, brightness * b_gamma_mult)) {
14802 wxString cmd(_T ( "xcalib " ));
14803 cmd += temp_file_name;
14804
14805 wxExecute(cmd, wxEXEC_ASYNC);
14806 }
14807
14808#else
14809 // Or, use "xcalib -co" to set overall contrast value
14810 // This is not as nice, since the -co parameter wants to be a fraction of
14811 // the current contrast, and values greater than 100 are not allowed. As a
14812 // result, increases of contrast must do a "-clear" step first, which
14813 // produces objectionable flashing.
14814 if (brightness > last_brightness) {
14815 wxString cmd;
14816 cmd = _T("xcalib -clear");
14817 wxExecute(cmd, wxEXEC_ASYNC);
14818
14819 ::wxMilliSleep(10);
14820
14821 int brite_adj = wxMax(1, brightness);
14822 cmd.Printf(_T("xcalib -co %2d -a"), brite_adj);
14823 wxExecute(cmd, wxEXEC_ASYNC);
14824 } else {
14825 int brite_adj = wxMax(1, brightness);
14826 int factor = (brite_adj * 100) / last_brightness;
14827 factor = wxMax(1, factor);
14828 wxString cmd;
14829 cmd.Printf(_T("xcalib -co %2d -a"), factor);
14830 wxExecute(cmd, wxEXEC_ASYNC);
14831 }
14832
14833#endif
14834
14835 last_brightness = brightness;
14836
14837#endif
14838
14839 return 0;
14840}
14841
14842#ifdef __OPCPN_USEICC__
14843
14844#define MLUT_TAG 0x6d4c5554L
14845#define VCGT_TAG 0x76636774L
14846
14847int GetIntEndian(unsigned char *s) {
14848 int ret;
14849 unsigned char *p;
14850 int i;
14851
14852 p = (unsigned char *)&ret;
14853
14854 if (1)
14855 for (i = sizeof(int) - 1; i > -1; --i) *p++ = s[i];
14856 else
14857 for (i = 0; i < (int)sizeof(int); ++i) *p++ = s[i];
14858
14859 return ret;
14860}
14861
14862unsigned short GetShortEndian(unsigned char *s) {
14863 unsigned short ret;
14864 unsigned char *p;
14865 int i;
14866
14867 p = (unsigned char *)&ret;
14868
14869 if (1)
14870 for (i = sizeof(unsigned short) - 1; i > -1; --i) *p++ = s[i];
14871 else
14872 for (i = 0; i < (int)sizeof(unsigned short); ++i) *p++ = s[i];
14873
14874 return ret;
14875}
14876
14877// Create a very simple Gamma correction file readable by xcalib
14878int CreateSimpleICCProfileFile(const char *file_name, double co_red,
14879 double co_green, double co_blue) {
14880 FILE *fp;
14881
14882 if (file_name) {
14883 fp = fopen(file_name, "wb");
14884 if (!fp) return -1; /* file can not be created */
14885 } else
14886 return -1; /* filename char pointer not valid */
14887
14888 // Write header
14889 char header[128];
14890 for (int i = 0; i < 128; i++) header[i] = 0;
14891
14892 fwrite(header, 128, 1, fp);
14893
14894 // Num tags
14895 int numTags0 = 1;
14896 int numTags = GetIntEndian((unsigned char *)&numTags0);
14897 fwrite(&numTags, 1, 4, fp);
14898
14899 int tagName0 = VCGT_TAG;
14900 int tagName = GetIntEndian((unsigned char *)&tagName0);
14901 fwrite(&tagName, 1, 4, fp);
14902
14903 int tagOffset0 = 128 + 4 * sizeof(int);
14904 int tagOffset = GetIntEndian((unsigned char *)&tagOffset0);
14905 fwrite(&tagOffset, 1, 4, fp);
14906
14907 int tagSize0 = 1;
14908 int tagSize = GetIntEndian((unsigned char *)&tagSize0);
14909 fwrite(&tagSize, 1, 4, fp);
14910
14911 fwrite(&tagName, 1, 4, fp); // another copy of tag
14912
14913 fwrite(&tagName, 1, 4, fp); // dummy
14914
14915 // Table type
14916
14917 /* VideoCardGammaTable (The simplest type) */
14918 int gammatype0 = 0;
14919 int gammatype = GetIntEndian((unsigned char *)&gammatype0);
14920 fwrite(&gammatype, 1, 4, fp);
14921
14922 int numChannels0 = 3;
14923 unsigned short numChannels = GetShortEndian((unsigned char *)&numChannels0);
14924 fwrite(&numChannels, 1, 2, fp);
14925
14926 int numEntries0 = 256;
14927 unsigned short numEntries = GetShortEndian((unsigned char *)&numEntries0);
14928 fwrite(&numEntries, 1, 2, fp);
14929
14930 int entrySize0 = 1;
14931 unsigned short entrySize = GetShortEndian((unsigned char *)&entrySize0);
14932 fwrite(&entrySize, 1, 2, fp);
14933
14934 unsigned char ramp[256];
14935
14936 // Red ramp
14937 for (int i = 0; i < 256; i++) ramp[i] = i * co_red / 100.;
14938 fwrite(ramp, 256, 1, fp);
14939
14940 // Green ramp
14941 for (int i = 0; i < 256; i++) ramp[i] = i * co_green / 100.;
14942 fwrite(ramp, 256, 1, fp);
14943
14944 // Blue ramp
14945 for (int i = 0; i < 256; i++) ramp[i] = i * co_blue / 100.;
14946 fwrite(ramp, 256, 1, fp);
14947
14948 fclose(fp);
14949
14950 return 0;
14951}
14952#endif // __OPCPN_USEICC__
Global state for AIS decoder.
Dialog for displaying a list of AIS targets.
Dialog for querying detailed information about an AIS target.
bool Create(wxWindow *parent, wxWindowID id=wxID_ANY, const wxString &caption=_("Object Query"), const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxDefaultSize, long style=AIS_TARGET_QUERY_STYLE)
Creation.
double GetDisplayDIPMult(wxWindow *win)
Get the display scaling factor for DPI-aware rendering.
Represents an active track that is currently being recorded.
Definition track.h:194
Handles context menu events for the chart canvas.
Definition canvasMenu.h:82
A custom panel for displaying chart information.
Definition ChInfoWin.h:36
Base class for BSB (Maptech/NOS) format nautical charts.
Definition chartimg.h:131
Base class for all chart types.
Definition chartbase.h:119
ChartCanvas - Main chart display and interaction component.
Definition chcanv.h:148
bool GetCanvasPointPixVP(ViewPort &vp, double rlat, double rlon, wxPoint *r)
Convert latitude/longitude to canvas pixel coordinates rounded to nearest integer using specified vie...
Definition chcanv.cpp:4539
bool GetCanvasPointPix(double rlat, double rlon, wxPoint *r)
Convert latitude/longitude to canvas pixel coordinates (physical pixels) rounded to nearest integer.
Definition chcanv.cpp:4535
void GetDoubleCanvasPointPixVP(ViewPort &vp, double rlat, double rlon, wxPoint2DDouble *r)
Convert latitude/longitude to canvas pixel coordinates (physical pixels) with double precision,...
Definition chcanv.cpp:4485
double m_cursor_lat
The latitude in degrees corresponding to the most recently processed cursor position.
Definition chcanv.h:742
double GetCanvasScaleFactor()
Return the number of logical pixels per meter for the screen.
Definition chcanv.h:458
double GetPixPerMM()
Get the number of logical pixels per millimeter on the screen.
Definition chcanv.h:489
void SetDisplaySizeMM(double size)
Set the width of the screen in millimeters.
Definition chcanv.cpp:2378
bool MouseEventSetup(wxMouseEvent &event, bool b_handle_dclick=true)
Definition chcanv.cpp:7498
bool PanCanvas(double dx, double dy)
Pans (moves) the canvas by the specified physical pixels in x and y directions.
Definition chcanv.cpp:5036
float GetVPScale()
Return the ViewPort scale factor, in physical pixels per meter.
Definition chcanv.h:447
void DoZoomCanvas(double factor, bool can_zoom_to_cursor=true)
Internal function that implements the actual zoom operation.
Definition chcanv.cpp:4641
double GetCanvasTrueScale()
Return the physical pixels per meter at chart center, accounting for latitude distortion.
Definition chcanv.h:463
void ZoomCanvasSimple(double factor)
Perform an immediate zoom operation without smooth transitions.
Definition chcanv.cpp:4616
bool SetVPScale(double sc, bool b_refresh=true)
Sets the viewport scale while maintaining the center point.
Definition chcanv.cpp:5310
double m_cursor_lon
The longitude in degrees corresponding to the most recently processed cursor position.
Definition chcanv.h:726
void GetCanvasPixPoint(double x, double y, double &lat, double &lon)
Convert canvas pixel coordinates (physical pixels) to latitude/longitude.
Definition chcanv.cpp:4560
void ZoomCanvas(double factor, bool can_zoom_to_cursor=true, bool stoptimer=true)
Perform a smooth zoom operation on the chart canvas by the specified factor.
Definition chcanv.cpp:4621
void GetDoubleCanvasPointPix(double rlat, double rlon, wxPoint2DDouble *r)
Convert latitude/longitude to canvas pixel coordinates (physical pixels) with double precision.
Definition chcanv.cpp:4480
bool SetViewPoint(double lat, double lon)
Set the viewport center point.
Definition chcanv.cpp:5329
bool MouseEventProcessCanvas(wxMouseEvent &event)
Processes mouse events for core chart panning and zooming operations.
Definition chcanv.cpp:9815
Manages the chart database and provides access to chart data.
Definition chartdb.h:95
Represents a user-defined collection of logically related charts.
Definition chartdbs.h:464
Represents an MBTiles format chart.
Definition mbtiles.h:69
Wrapper class for plugin-based charts.
Definition chartimg.h:392
Primary navigation console display for route and vessel tracking.
Definition concanv.h:127
wxFont * FindOrCreateFont(int point_size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underline=false, const wxString &facename=wxEmptyString, wxFontEncoding encoding=wxFONTENCODING_DEFAULT)
Creates or finds a matching font in the font cache.
Definition FontMgr.cpp:450
wxColour GetFontColor(const wxString &TextElement) const
Gets the text color for a UI element.
Definition FontMgr.cpp:117
wxFont * GetFont(const wxString &TextElement, int requested_font_size=0)
Gets a font object for a UI element.
Definition FontMgr.cpp:186
Modal dialog displaying hotkeys.
Definition hotkeys_dlg.h:29
Represents an index entry for tidal and current data.
Definition IDX_entry.h:49
char IDX_type
Entry type identifier "TCtcIUu".
Definition IDX_entry.h:61
double IDX_lat
Latitude of the station (in degrees, +North)
Definition IDX_entry.h:65
double IDX_lon
Longitude of the station (in degrees, +East)
Definition IDX_entry.h:64
bool b_skipTooDeep
Flag to skip processing if depth exceeds limit.
Definition IDX_entry.h:110
Station_Data * pref_sta_data
Pointer to the reference station data.
Definition IDX_entry.h:97
int IDX_rec_num
Record number for multiple entries with same name.
Definition IDX_entry.h:60
Definition kml.h:54
Modern User Interface Control Bar for OpenCPN.
Definition MUIBar.h:63
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.
wxSize getDisplaySize()
Get the display size in logical pixels.
double GetDisplaySizeMM()
Get the width of the screen in millimeters.
An iterator class for OCPNRegion.
Definition OCPNRegion.h:156
A wrapper class for wxRegion with additional functionality.
Definition OCPNRegion.h:56
Definition piano.h:65
A popup frame containing a detail slider for chart display.
Definition Quilt.h:83
bool Compose(const ViewPort &vp)
Definition Quilt.cpp:1695
Represents a waypoint or mark within the navigation system.
Definition route_point.h:68
bool m_bIsolatedMark
Flag indicating if the waypoint is a standalone mark.
Represents a navigational route in the navigation system.
Definition route.h:96
bool ActivateNextPoint(Route *pr, bool skipped)
Definition routeman.cpp:395
bool DeleteRoute(Route *pRoute, NavObjectChanges *nav_obj_changes)
Definition routeman.cpp:835
Dialog for displaying query results of S57 objects.
Manager for S57 chart SENC creation threads.
Manages a set of ShapeBaseChart objects at different resolutions.
Definition tcmgr.h:86
Definition TCWin.h:46
Window for displaying chart thumbnails.
Definition thumbwin.h:56
Represents a single point in a track.
Definition track.h:53
Class TrackPropDlg.
bool UpdateProperties()
Represents a track, which is a series of connected track points.
Definition track.h:79
Definition undo.h:60
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
void SetPixelScale(double scale)
Set the physical to logical pixel ratio for the display.
Definition viewport.cpp:133
int pix_width
Width of the viewport in physical pixels.
Definition viewport.h:256
wxPoint2DDouble GetDoublePixFromLL(double lat, double lon)
Convert latitude and longitude on the ViewPort to physical pixel coordinates with double precision.
Definition viewport.cpp:145
double tilt
Tilt angle for perspective view in radians.
Definition viewport.h:241
double skew
Angular distortion (shear transform) applied to the viewport in radians.
Definition viewport.h:237
void GetLLFromPix(const wxPoint &p, double *lat, double *lon)
Convert physical pixel coordinates on the ViewPort to latitude and longitude.
Definition viewport.h:105
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
wxPoint GetPixFromLL(double lat, double lon)
Convert latitude and longitude on the ViewPort to physical pixel coordinates.
Definition viewport.cpp:136
double chart_scale
Chart scale denominator (e.g., 50000 for a 1:50000 scale).
Definition viewport.h:244
bool AddRoutePoint(RoutePoint *prp)
Add a point to list which owns it.
bool RemoveRoutePoint(RoutePoint *prp)
Remove a routepoint from list if present, deallocate it all cases.
Encapsulates persistent canvas configuration.
double iLat
Latitude of the center of the chart, in degrees.
bool bShowOutlines
Display chart outlines.
bool bShowDepthUnits
Display depth unit indicators.
double iLon
Longitude of the center of the chart, in degrees.
double iRotation
Initial rotation angle in radians.
bool bCourseUp
Orient display to course up.
bool bQuilt
Enable chart quilting.
bool bFollow
Enable vessel following mode.
double iScale
Initial chart scale factor.
bool bShowENCText
Display ENC text elements.
bool bShowAIS
Display AIS targets.
bool bShowGrid
Display coordinate grid.
bool bShowCurrents
Display current information.
bool bShowTides
Display tide information.
bool bLookahead
Enable lookahead mode.
bool bHeadUp
Orient display to heading up.
bool bAttenAIS
Enable AIS target attenuation.
Represents a composite CM93 chart covering multiple scales.
Definition cm93.h:424
Stores emboss effect data for textures.
Definition emboss_data.h:35
OpenGL chart rendering canvas.
Floating toolbar for iENC (International Electronic Navigational Chart) functionality.
Definition iENCToolbar.h:43
Represents a compass display in the OpenCPN navigation system.
Definition compass.h:37
wxRect GetRect(void) const
Return the coordinates of the compass widget, in physical pixels relative to the canvas window.
Definition compass.h:61
wxRect GetLogicalRect(void) const
Return the coordinates of the compass widget, in logical pixels.
Definition compass.cpp:201
Device context class that can use either wxDC or OpenGL for drawing.
Definition ocpndc.h:64
void DrawLine(wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2, bool b_hiqual=true)
Draw a line between two points using either wxDC or OpenGL.
Definition ocpndc.cpp:476
Floating toolbar dialog for OpenCPN.
Definition toolbar.h:386
Represents an S57 format electronic navigational chart in OpenCPN.
Definition s57chart.h:120
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.
Global variables reflecting command line options and arguments.
Hooks into gui available in model.
int GetChartbarHeight(void)
Gets height of chart bar in pixels.
int GetCanvasCount()
Gets total number of chart canvases.
PI_DisCat GetENCDisplayCategory(int CanvasIndex)
Gets current ENC display category.
void EnableTenHertzUpdate(bool enable)
Enable or disable 10 Hz update rate.
bool GetEnableTenHertzUpdate()
Check if 10 Hz update rate is enabled.
double OCPN_GetWinDIPScaleFactor()
Gets Windows-specific DPI scaling factor.
Tools to send data to plugins.
Route validators for dialog validation.
Represents an entry in the chart table, containing information about a single chart.
Definition chartdbs.h:181