OpenCPN Partial API docs
Loading...
Searching...
No Matches
MUIBar.cpp
1/******************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: MUI Control Bar
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 *
27 */
28
29#include <wx/wxprec.h>
30
31#ifndef WX_PRECOMP
32#include <wx/wx.h>
33#endif // precompiled headers
34
35#include <wx/statline.h>
36#include "ssfn.h"
37
38#include "chcanv.h"
39#include "MUIBar.h"
40#include "OCPNPlatform.h"
41#include "CanvasOptions.h"
42#include "DetailSlider.h"
43#include "GoToPositionDialog.h"
44#include "styles.h"
45#include "navutil.h"
46#include "svg_utils.h"
47#include "model/idents.h"
48#include "color_handler.h"
49#include "navutil.h"
50#include "pluginmanager.h"
51
52#ifdef ocpnUSE_GL
53#include "glChartCanvas.h"
54#endif
55
56#ifdef __OCPN__ANDROID__
57#include "androidUTIL.h"
58#include "qdebug.h"
59#endif
60
61#ifdef ocpnUSE_GL
62extern GLenum g_texture_rectangle_format;
63#endif
64
65//------------------------------------------------------------------------------
66// External Static Storage
67//------------------------------------------------------------------------------
68
69extern OCPNPlatform* g_Platform;
70extern ChartCanvas* g_focusCanvas;
71extern ocpnStyle::StyleManager* g_StyleManager;
72extern bool g_bShowMuiZoomButtons;
73extern bool g_bopengl;
74
75ssfn_t ctx; /* the renderer context */
76ssfn_glyph_t* glyph; /* the returned rasterized bitmap */
77
78double getValue(int animationType, double t);
79
80// Helper classes
81
82#define ID_SCALE_CANCEL 8301
83#define ID_SCALE_OK 8302
84#define ID_SCALECTRL 8303
85
86class SetScaleDialog : public wxDialog {
87 DECLARE_EVENT_TABLE()
88
89public:
92 SetScaleDialog(wxWindow* parent, wxWindowID id = SYMBOL_GOTOPOS_IDNAME,
93 const wxString& caption = _("Set scale"),
94 const wxPoint& pos = wxDefaultPosition,
95 const wxSize& size = wxDefaultSize,
96 long style = wxDEFAULT_DIALOG_STYLE);
97
99
101 bool Create(wxWindow* parent, wxWindowID id = wxID_ANY,
102 const wxString& caption = _("Set scale"),
103 const wxPoint& pos = wxDefaultPosition,
104 const wxSize& size = wxDefaultSize,
105 long style = wxDEFAULT_DIALOG_STYLE);
106
107 void SetColorScheme(ColorScheme cs);
108
109 void CreateControls();
110
111 void OnSetScaleCancelClick(wxCommandEvent& event);
112 void OnSetScaleOKClick(wxCommandEvent& event);
113
115
116 wxTextCtrl* m_ScaleCtl;
117 wxButton* m_CancelButton;
118 wxButton* m_OKButton;
119};
120
121BEGIN_EVENT_TABLE(SetScaleDialog, wxDialog)
122EVT_BUTTON(ID_GOTOPOS_CANCEL, SetScaleDialog::OnSetScaleCancelClick)
123EVT_BUTTON(ID_GOTOPOS_OK, SetScaleDialog::OnSetScaleOKClick)
124END_EVENT_TABLE()
125
126
131
132SetScaleDialog::SetScaleDialog(wxWindow* parent, wxWindowID id,
133 const wxString& caption, const wxPoint& pos,
134 const wxSize& size, long style) {
135 long wstyle =
136 wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER | wxFRAME_FLOAT_ON_PARENT;
137
138 Create(parent, id, caption, pos, size, wstyle);
139}
140
141SetScaleDialog::~SetScaleDialog() {}
142
147bool SetScaleDialog::Create(wxWindow* parent, wxWindowID id,
148 const wxString& caption, const wxPoint& pos,
149 const wxSize& size, long style) {
150 SetExtraStyle(GetExtraStyle() | wxWS_EX_BLOCK_EVENTS);
151 wxDialog::Create(parent, id, caption, pos, size, style);
152
154 GetSizer()->SetSizeHints(this);
155 Centre();
156
157 return TRUE;
158}
159
165 SetScaleDialog* itemDialog1 = this;
166
167 wxBoxSizer* itemBoxSizer2 = new wxBoxSizer(wxVERTICAL);
168 itemDialog1->SetSizer(itemBoxSizer2);
169
170 wxStaticBox* itemStaticBoxSizer4Static =
171 new wxStaticBox(itemDialog1, wxID_ANY, _("Chart Scale"));
172
173 wxStaticBoxSizer* itemStaticBoxSizer4 =
174 new wxStaticBoxSizer(itemStaticBoxSizer4Static, wxVERTICAL);
175 itemBoxSizer2->Add(itemStaticBoxSizer4, 0, wxEXPAND | wxALL, 5);
176
177 wxStaticText* itemStaticText5 = new wxStaticText(
178 itemDialog1, wxID_STATIC, _T(""), wxDefaultPosition, wxDefaultSize, 0);
179 itemStaticBoxSizer4->Add(itemStaticText5, 0,
180 wxALIGN_LEFT | wxLEFT | wxRIGHT | wxTOP, 5);
181
182 m_ScaleCtl = new wxTextCtrl(itemDialog1, ID_SCALECTRL, _T(""),
183 wxDefaultPosition, wxSize(180, -1), 0);
184 itemStaticBoxSizer4->Add(
185 m_ScaleCtl, 0, wxALIGN_LEFT | wxLEFT | wxRIGHT | wxBOTTOM | wxEXPAND, 5);
186
187 wxBoxSizer* itemBoxSizer16 = new wxBoxSizer(wxHORIZONTAL);
188 itemBoxSizer2->Add(itemBoxSizer16, 0, wxALIGN_RIGHT | wxALL, 5);
189
190 m_CancelButton = new wxButton(itemDialog1, ID_GOTOPOS_CANCEL, _("Cancel"),
191 wxDefaultPosition, wxDefaultSize, 0);
192 itemBoxSizer16->Add(m_CancelButton, 0, wxALIGN_CENTER_VERTICAL | wxALL, 5);
193
194 m_OKButton = new wxButton(itemDialog1, ID_GOTOPOS_OK, _("OK"),
195 wxDefaultPosition, wxDefaultSize, 0);
196 itemBoxSizer16->Add(m_OKButton, 0, wxALIGN_CENTER_VERTICAL | wxALL, 5);
197 m_OKButton->SetDefault();
198
199 SetColorScheme((ColorScheme)0);
200}
201
202void SetScaleDialog::SetColorScheme(ColorScheme cs) { DimeControl(this); }
203
204void SetScaleDialog::OnSetScaleCancelClick(wxCommandEvent& event) {
205 Close();
206 event.Skip();
207}
208
209void SetScaleDialog::OnSetScaleOKClick(wxCommandEvent& event) {
210 SetReturnCode(wxID_OK);
211 EndModal(wxID_OK);
212 return;
213}
214
215//------------------------------------------------------------------------------
216// MUIButton
217//------------------------------------------------------------------------------
218
220 wxSize DoGetBestSize() const;
221
222public:
223 MUIButton();
224 MUIButton(wxWindow* parent, wxWindowID id = wxID_ANY,
225 float scale_factor = 1.0,
226 const wxString& bitmapState0 = wxEmptyString,
227 const wxString& bitmapState1 = wxEmptyString,
228 const wxString& bitmapState2 = wxEmptyString,
229 const wxPoint& pos = wxDefaultPosition,
230 const wxSize& size = wxDefaultSize, long style = wxNO_BORDER);
231
232 bool Create(wxWindow* parent, wxWindowID id = wxID_ANY,
233 float scale_factor = 1.0,
234 const wxString& bitmapState0 = wxEmptyString,
235 const wxString& bitmapState1 = wxEmptyString,
236 const wxString& bitmapState2 = wxEmptyString,
237 const wxPoint& pos = wxDefaultPosition,
238 const wxSize& size = wxDefaultSize, long style = wxNO_BORDER);
239
240 ~MUIButton();
241
242 void Init();
243 void CreateControls();
244
245 void SetState(int state);
246 int GetState() { return mState; }
247
248 void SetColorScheme(ColorScheme cs);
249 void OnSize(wxSizeEvent& event);
250 void OnLeftDown(wxMouseEvent& event);
251 void OnLeftUp(wxMouseEvent& event);
252
253 wxBitmap GetBitmapResource(const wxString& name);
254
255 wxIcon GetIconResource(const wxString& name);
256 wxBitmap GetButtonBitmap() { return m_bitmap; }
257
259 static bool ShowToolTips();
260 wxSize m_size;
261 wxPoint m_position;
262
263private:
264 wxString m_bitmapFileState0;
265 wxString m_bitmapFileState1;
266 wxString m_bitmapFileState2;
267 wxBitmap m_bitmap;
268 wxBitmap m_bitmapState0;
269 wxBitmap m_bitmapState1;
270 wxBitmap m_bitmapState2;
271
272 int mState;
273 float m_scaleFactor;
274 wxSize m_styleToolSize;
275 ColorScheme m_cs;
276};
277
278MUIButton::MUIButton() { Init(); }
279
280MUIButton::MUIButton(wxWindow* parent, wxWindowID id, float scale_factor,
281 const wxString& bitmap, const wxString& bitmapState1,
282 const wxString& bitmapState2, const wxPoint& pos,
283 const wxSize& size, long style) {
284 Init();
285
286 Create(parent, id, scale_factor, bitmap, bitmapState1, bitmapState2, pos,
287 size, style);
288}
289
290bool MUIButton::Create(wxWindow* parent, wxWindowID id, float scale_factor,
291 const wxString& bitmap, const wxString& bitmapState1,
292 const wxString& bitmapState2, const wxPoint& pos,
293 const wxSize& size, long style) {
294 m_bitmapFileState0 = bitmap;
295 m_bitmapFileState1 = bitmapState1;
296 m_bitmapFileState2 = bitmapState2;
297
298 m_scaleFactor = scale_factor;
299
300 m_styleToolSize = g_StyleManager->GetCurrentStyle()->GetToolSize();
301
302 // Arbitrarily boost the MUIButton default size above the style defined size.
303 // No good reason.....
304 m_styleToolSize = wxSize(m_styleToolSize.x * 1.25, m_styleToolSize.y * 1.25);
305
306 m_size = wxSize(m_styleToolSize.x * m_scaleFactor,
307 m_styleToolSize.y * m_scaleFactor);
308
309 CreateControls();
310 return true;
311}
312
313MUIButton::~MUIButton() {}
314
315void MUIButton::Init() {
316 mState = 0;
317 m_cs = (ColorScheme)-1; // GLOBAL_COLOR_SCHEME_RGB;
318}
319
320void MUIButton::CreateControls() {
321 // this->SetForegroundColour(wxColour(255, 255, 255));
322
323 // wxColour backColor = GetGlobalColor(_T("GREY3"));
324 // SetBackgroundColour(backColor);
325
326 // this->SetFont(wxFont(8, wxFONTFAMILY_SWISS, wxFONTSTYLE_NORMAL,
327 // wxFONTWEIGHT_NORMAL, false, wxT("Tahoma")));
328}
329
330void MUIButton::SetColorScheme(ColorScheme cs) {
331#if 1
332 if (m_cs != cs) {
333 wxColour backColor = GetGlobalColor(_T("GREY3"));
334 // SetBackgroundColour(backColor);
335
336 ocpnStyle::Style* style = g_StyleManager->GetCurrentStyle();
337
338 wxBitmap bmp = LoadSVG(m_bitmapFileState0, m_size.x, m_size.y);
339 m_bitmapState0 = style->SetBitmapBrightness(bmp, cs);
340
341 bmp = LoadSVG(m_bitmapFileState1, m_size.x, m_size.y);
342 if (bmp.IsOk())
343 m_bitmapState1 = style->SetBitmapBrightness(bmp, cs);
344 else
345 m_bitmapState1 = m_bitmapState0;
346
347 bmp = LoadSVG(m_bitmapFileState2, m_size.x, m_size.y);
348 if (bmp.IsOk())
349 m_bitmapState2 = style->SetBitmapBrightness(bmp, cs);
350 else
351 m_bitmapState2 = m_bitmapState0;
352
353 switch (mState) {
354 case 0:
355 default:
356 m_bitmap = m_bitmapState0;
357 break;
358
359 case 1:
360 m_bitmap = m_bitmapState1;
361 break;
362
363 case 2:
364 m_bitmap = m_bitmapState2;
365 break;
366 }
367
368 m_cs = cs;
369 }
370#endif
371}
372
373bool MUIButton::ShowToolTips() { return true; }
374
375void MUIButton::SetState(int state) {
376 switch (state) {
377 case 0:
378 default:
379 m_bitmap = m_bitmapState0;
380 break;
381
382 case 1:
383 m_bitmap = m_bitmapState1;
384 break;
385
386 case 2:
387 m_bitmap = m_bitmapState2;
388 break;
389 }
390
391 mState = state;
392}
393
394void MUIButton::OnSize(wxSizeEvent& event) {
395 if (m_bitmap.IsOk()) {
396 if (event.GetSize() == m_bitmap.GetSize()) return;
397 }
398
399 if (!m_bitmapFileState0.IsEmpty())
400 m_bitmapState0 =
401 LoadSVG(m_bitmapFileState0, event.GetSize().x, event.GetSize().y);
402
403 if (!m_bitmapFileState1.IsEmpty())
404 m_bitmapState1 =
405 LoadSVG(m_bitmapFileState1, event.GetSize().x, event.GetSize().y);
406 if (!m_bitmapState1.IsOk() || m_bitmapFileState1.IsEmpty())
407 m_bitmapState1 = m_bitmapState0;
408
409 if (!m_bitmapFileState2.IsEmpty())
410 m_bitmapState2 =
411 LoadSVG(m_bitmapFileState2, event.GetSize().x, event.GetSize().y);
412 if (!m_bitmapState2.IsOk() || m_bitmapFileState2.IsEmpty())
413 m_bitmapState2 = m_bitmapState0;
414
415 switch (mState) {
416 case 0:
417 default:
418 m_bitmap = m_bitmapState0;
419 break;
420
421 case 1:
422 m_bitmap = m_bitmapState1;
423 break;
424
425 case 2:
426 m_bitmap = m_bitmapState2;
427 break;
428 }
429}
430
431wxSize MUIButton::DoGetBestSize() const {
432 // wxSize labelSize = wxDefaultSize;
433 // GetTextExtent(m_Label, &labelSize.x, &labelSize.y);
434 // return wxSize(wxMax(40, labelSize.x + 20), wxMax(20, labelSize.y +
435 // 10));
436 return wxSize(m_styleToolSize.x * m_scaleFactor,
437 m_styleToolSize.y * m_scaleFactor);
438}
439
440//-----------------------------------------------------------------------
441// SSFN Utils
442//-----------------------------------------------------------------------
443static std::unordered_map<char, ssfn_glyph_t*> ssfn_glyph_map;
444
448ssfn_font_t* load_font(const char* filename) {
449 char* fontdata = NULL;
450 long int size;
451 FILE* f;
452#if HAS_ZLIB
453 unsigned char hdr[2];
454 gzFile g;
455#endif
456
457 f = fopen(filename, "rb");
458 if (!f) {
459 fprintf(stderr, "unable to load %s\n", filename);
460 return NULL;
461 }
462 size = 0;
463#if HAS_ZLIB
464 fread(&hdr, 2, 1, f);
465 if (hdr[0] == 0x1f && hdr[1] == 0x8b) {
466 fseek(f, -4L, SEEK_END);
467 fread(&size, 4, 1, f);
468 } else {
469 fseek(f, 0, SEEK_END);
470 size = ftell(f);
471 }
472 fclose(f);
473 g = gzopen(filename, "r");
474#else
475 fseek(f, 0, SEEK_END);
476 size = ftell(f);
477 fseek(f, 0, SEEK_SET);
478#endif
479 if (!size) {
480 fprintf(stderr, "unable to load %s\n", filename);
481 exit(3);
482 }
483 fontdata = (char*)malloc(size);
484 if (!fontdata) {
485 fprintf(stderr, "memory allocation error\n");
486 exit(2);
487 }
488#if HAS_ZLIB
489 gzread(g, fontdata, size);
490 gzclose(g);
491#else
492 fread(fontdata, size, 1, f);
493 fclose(f);
494#endif
495
496 return (ssfn_font_t*)fontdata;
497}
498
499bool RenderGlyphToImageBuffer(unsigned char* buffer, ssfn_glyph_t* glyph,
500 int x_offset, int w, int h, int nominal_baseline,
501 wxColour& color, wxColour& back_color) {
502 unsigned char* src = glyph->data;
503 for (int i = 0; i < h; i++) {
504 for (int j = 0; j < glyph->w; j++) {
505 size_t index = i * w + j + x_offset;
506 index += (nominal_baseline - glyph->baseline) * w;
507 if (index > (size_t)h * (w - 1)) continue;
508 size_t didx = index * 3;
509
510 size_t sidx = i * glyph->pitch + j;
511 unsigned char d = src[sidx];
512 float dn = d / 256.;
513 buffer[didx] = (color.Red() * dn) + (back_color.Red() * (1 - dn));
514 buffer[didx + 1] = (color.Green() * dn) + (back_color.Green() * (1 - dn));
515 buffer[didx + 2] = (color.Blue() * dn) + (back_color.Blue() * (1 - dn));
516 }
517 }
518 return true;
519}
520
521bool RenderStringToBuffer(unsigned char* buffer, std::string s, int wbox,
522 int hbox, int nominal_baseline, wxColour color,
523 wxColour& back_color) {
524 int xpos = 0;
525 for (unsigned int i = 0; i < s.size(); i++) {
526 char key = s[i];
527 ssfn_glyph_t* glyph = 0;
528 // find the required glyph
529 if (auto findit = ssfn_glyph_map.find(key);
530 findit != ssfn_glyph_map.end()) {
531 glyph = findit->second;
532 } else {
533 glyph = ssfn_render(&ctx, key);
534 ssfn_glyph_map[key] = glyph;
535 }
536 RenderGlyphToImageBuffer(buffer, glyph, xpos, wbox, hbox, nominal_baseline,
537 color, back_color);
538 xpos += glyph->adv_x;
539 }
540 return true;
541}
542
543//------------------------------------------------------------------------------
544// MUITextButton
545//------------------------------------------------------------------------------
546
548public:
550 MUITextButton(wxWindow* parent, wxWindowID id, float scale_factor,
551 wxColor backColor, const wxString& text = wxEmptyString,
552 const wxPoint& pos = wxDefaultPosition,
553 const wxSize& size = wxDefaultSize, long style = wxNO_BORDER);
554
555 bool Create(wxWindow* parent, wxWindowID id, float scale_factor = 1.0,
556 const wxString& text = wxEmptyString,
557 const wxPoint& pos = wxDefaultPosition,
558 const wxSize& size = wxDefaultSize, long style = wxNO_BORDER);
559
561
562 void Init();
563 wxSize GetSize() { return m_size; }
564 void SetState(int state);
565
566 void SetColorScheme(ColorScheme cs);
567 void OnSize(wxSizeEvent& event);
568 void OnLeftDown(wxMouseEvent& event);
569 void OnLeftUp(wxMouseEvent& event);
570 void SetText(const wxString& text);
571
572 wxBitmap GetButtonBitmap() { return m_bitmap; }
573 wxSize m_size;
574 wxPoint m_position;
575
576private:
577 void BuildBitmap();
578
579 wxString m_text;
580 wxBitmap m_bitmap;
581
582 int mState;
583 float m_scaleFactor;
584 wxSize m_styleToolSize;
585 ColorScheme m_cs;
586 int m_pixel_height;
587 int m_ssfn_status;
588 wxColor m_backgrounColor;
589};
590
591MUITextButton::MUITextButton() { Init(); }
592
593MUITextButton::MUITextButton(wxWindow* parent, wxWindowID id,
594 float scale_factor, wxColor backColor,
595 const wxString& text, const wxPoint& pos,
596 const wxSize& size, long style) {
597 m_backgrounColor = backColor;
598 Init();
599 Create(parent, id, scale_factor, text, pos, size, style);
600}
601
602bool MUITextButton::Create(wxWindow* parent, wxWindowID id, float scale_factor,
603 const wxString& text, const wxPoint& pos,
604 const wxSize& size, long style) {
605 m_text = text;
606
607 m_scaleFactor = scale_factor;
608
609 m_styleToolSize = g_StyleManager->GetCurrentStyle()->GetToolSize();
610
611 // Arbitrarily boost the MUITextButton default size above the style defined
612 // size. No good reason.....
613 m_styleToolSize = wxSize(m_styleToolSize.x * 1.25, m_styleToolSize.y * 1.25);
614
615 int height_ref = m_styleToolSize.y;
616
617 /* initialize the ssfn library, the context is zerod out */
618 memset(&ctx, 0, sizeof(ssfn_t));
619 wxString font_file = g_Platform->GetSharedDataDir() + "ssfndata/FreeSans.sfn";
620 std::string sfont = font_file.ToStdString();
621 ssfn_font_t* pssfn_font = NULL;
622 pssfn_font = load_font(sfont.c_str());
623 if (pssfn_font) ssfn_load(&ctx, pssfn_font);
624
625 m_pixel_height = height_ref / 2;
626 m_pixel_height *= m_scaleFactor;
627
628 /* select the typeface to use */
629 m_ssfn_status =
630 ssfn_select(&ctx, SSFN_FAMILY_SANS, NULL, /* family */
631 SSFN_STYLE_REGULAR, m_pixel_height, /* style and size */
632 SSFN_MODE_ALPHA /* rendering mode */
633 );
634
635 int wbox = 20;
636 int hbox = 20;
637
638 if (m_ssfn_status == SSFN_OK) {
639 std::string t = "1:888888";
640 ssfn_bbox(&ctx, (char*)t.c_str(), 0, &wbox, &hbox);
641 }
642
643 m_size = wxSize(hbox * 1.5, m_styleToolSize.y);
644 m_size.y *= m_scaleFactor;
645
646 return true;
647}
648MUITextButton::~MUITextButton() {
649 for (const auto& [key, value] : ssfn_glyph_map) {
650 free(value);
651 }
652 ssfn_glyph_map.clear();
653}
654
655void MUITextButton::Init() {
656 mState = 0;
657 m_cs = (ColorScheme)-1; // GLOBAL_COLOR_SCHEME_RGB;
658}
659
660void MUITextButton::SetText(const wxString& text) {
661 if (!m_text.IsSameAs(text)) {
662 m_text = text;
663 BuildBitmap();
664 }
665}
666
667void MUITextButton::SetColorScheme(ColorScheme cs) {
668 if (m_cs != cs) {
669 // wxColour backColor = GetGlobalColor(_T("GREY3"));
670 // SetBackgroundColour(backColor);
671
672 m_cs = cs;
673 }
674 BuildBitmap();
675}
676
677void MUITextButton::SetState(int state) { mState = state; }
678
679void MUITextButton::OnSize(wxSizeEvent& event) {
680 BuildBitmap();
681 return;
682}
683
684void MUITextButton::BuildBitmap() {
685 int width = m_size.x;
686 int height = m_size.y;
687
688 if (m_ssfn_status != SSFN_OK) return;
689
690 int wbox = 0;
691 int hbox = 0;
692 std::string t = m_text.ToStdString();
693 ssfn_bbox(&ctx, (char*)t.c_str(), 0, &wbox, &hbox);
694
695 ssfn_glyph_t* glyph = ssfn_render(&ctx, '0');
696 int baseline = glyph->baseline;
697 free(glyph);
698
699 unsigned char* image_data = (unsigned char*)calloc(1, wbox * hbox * 3);
700 for (int i = 0; i < wbox * hbox; i++) {
701 int idx = i * 3;
702 image_data[idx] = m_backgrounColor.Red();
703 image_data[idx + 1] = m_backgrounColor.Green();
704 image_data[idx + 2] = m_backgrounColor.Blue();
705 }
706
707 RenderStringToBuffer(image_data, t, wbox, hbox, baseline,
708 GetGlobalColor("CHWHT"), m_backgrounColor);
709
710 wxImage fimage = wxImage(wbox, hbox, image_data);
711 wxSize clip_size = wxSize(wbox, baseline + 2);
712 wxRect clip_rect = wxRect(0, 0, clip_size.x, clip_size.y);
713 wxImage clip_image = fimage.GetSubImage(clip_rect);
714
715 m_bitmap = wxBitmap(clip_image);
716}
717
718#define CANVAS_OPTIONS_ANIMATION_TIMER_1 800
719#define CANVAS_OPTIONS_TIMER 801
720
721//------------------------------------------------------------------------------
722// MUIBar Implementation
723//------------------------------------------------------------------------------
724
725// Define a constructor
726MUIBar::MUIBar() {}
727
728MUIBar::MUIBar(ChartCanvas* parent, int orientation, float size_factor,
729 wxWindowID id, const wxPoint& pos, const wxSize& size,
730 long style, const wxString& name) {
731 m_parentCanvas = parent;
732 m_orientation = orientation;
733
734 m_scaleFactor = size_factor;
735 m_cs = (ColorScheme)-1;
736 wxColour backColor = wxColor(*wxBLACK); // GetGlobalColor(m_backcolorString);
737
738 Init();
739 CreateControls();
740}
741
742MUIBar::~MUIBar() {
743 if (m_canvasOptions) {
744 m_canvasOptions->Destroy();
745 m_canvasOptions = 0;
746 }
747 delete m_zinButton;
748 delete m_zoutButton;
749 delete m_followButton;
750 delete m_menuButton;
751 delete m_scaleButton;
752 InvalidateBitmap();
753}
754
755void MUIBar::Init() {
756 m_zinButton = NULL;
757 m_zoutButton = NULL;
758 m_followButton = NULL;
759 m_menuButton = NULL;
760 m_scaleButton = NULL;
761
762 m_canvasOptions = NULL;
763 m_canvasOptionsAnimationTimer.SetOwner(this,
764 CANVAS_OPTIONS_ANIMATION_TIMER_1);
765 m_backcolor = GetGlobalColor("GREY3");
766 m_capture_size_y = 0;
767
768 m_COTopOffset = 60; // TODO should be below GPS/Compass
769
770 CanvasOptionTimer.SetOwner(this, CANVAS_OPTIONS_TIMER);
771 m_coAnimateByBitmaps = false;
772 m_bEffects = false; // true;
773#ifdef __OCPN__ANDROID__
774 m_bEffects = false;
775#endif
776 m_texture = 0;
777 m_end_margin = m_parentCanvas->GetCharWidth() / 2;
778 m_scale = 0;
779}
780
781void MUIBar::SetColorScheme(ColorScheme cs) {
782 if (m_cs != cs) {
783 if (m_zinButton) m_zinButton->SetColorScheme(cs);
784 if (m_zoutButton) m_zoutButton->SetColorScheme(cs);
785 if (m_followButton) m_followButton->SetColorScheme(cs);
786 if (m_menuButton) m_menuButton->SetColorScheme(cs);
787
788 if (m_scaleButton) m_scaleButton->SetColorScheme(cs);
789
790 m_cs = cs;
791 InvalidateBitmap();
792 }
793}
794void MUIBar::InvalidateBitmap() {
795 m_bitmap = wxNullBitmap;
796
797#ifdef ocpnUSE_GL
798 if (g_bopengl) {
799 glDeleteTextures(1, &m_texture);
800 m_texture = 0;
801 }
802#endif
803}
804
805bool MUIBar::MouseEvent(wxMouseEvent& event) {
806 int x, y;
807 event.GetPosition(&x, &y);
808
809 // Check the regions
810 wxRect r = wxRect(m_screenPos, m_size);
811 if (r.Contains(x, y)) {
812 // Check buttons
813 if (event.LeftDown()) {
814 if (g_bShowMuiZoomButtons) {
815 wxRect rzin(m_zinButton->m_position.x, m_zinButton->m_position.y,
816 m_zinButton->m_size.x, m_zinButton->m_size.y);
817 rzin.Offset(m_screenPos);
818 if (rzin.Contains(x, y)) {
819 wxCommandEvent evt(wxEVT_COMMAND_MENU_SELECTED, ID_ZOOMIN);
820 m_parentCanvas->GetEventHandler()->AddPendingEvent(evt);
821 if (g_focusCanvas) g_focusCanvas->TriggerDeferredFocus();
822 return true;
823 }
824
825 wxRect rzout(m_zoutButton->m_position.x, m_zoutButton->m_position.y,
826 m_zoutButton->m_size.x, m_zoutButton->m_size.y);
827 rzout.Offset(m_screenPos);
828 if (rzout.Contains(x, y)) {
829 wxCommandEvent evt(wxEVT_COMMAND_MENU_SELECTED, ID_ZOOMOUT);
830 m_parentCanvas->GetEventHandler()->AddPendingEvent(evt);
831 if (g_focusCanvas) g_focusCanvas->TriggerDeferredFocus();
832 }
833 }
834
835 if (m_followButton) {
836 wxRect rfollow(m_followButton->m_position.x,
837 m_followButton->m_position.y, m_followButton->m_size.x,
838 m_followButton->m_size.y);
839 rfollow.Offset(m_screenPos);
840 if (rfollow.Contains(x, y)) {
841 wxCommandEvent evt(wxEVT_COMMAND_MENU_SELECTED, ID_FOLLOW);
842 m_parentCanvas->GetEventHandler()->AddPendingEvent(evt);
843 if (g_focusCanvas) g_focusCanvas->TriggerDeferredFocus();
844 }
845 }
846
847 wxRect rmenu(m_menuButton->m_position.x, m_menuButton->m_position.y,
848 m_menuButton->m_size.x, m_menuButton->m_size.y);
849 rmenu.Offset(m_screenPos);
850 if (rmenu.Contains(x, y)) {
851 HandleMenuClick();
852 if (g_focusCanvas) g_focusCanvas->TriggerDeferredFocus();
853 }
854 } else if (event.LeftUp()) {
855 if (m_scaleButton) {
856 wxRect rscale(m_scaleButton->m_position.x, m_scaleButton->m_position.y,
857 m_scaleButton->m_size.x, m_scaleButton->m_size.y);
858 rscale.Offset(m_screenPos);
859 if (rscale.Contains(x, y)) {
860 OnScaleSelected(event);
861 }
862 }
863 }
864 return true;
865 }
866 return false;
867}
868
869void MUIBar::OnScaleSelected(wxMouseEvent& event) {
870 auto pcc = dynamic_cast<ChartCanvas*>(m_parentCanvas);
871 if (!pcc) return;
872
873 SetScaleDialog dlg(pcc);
874 dlg.Centre();
875 dlg.ShowModal();
876 if (dlg.GetReturnCode() == wxID_OK) {
877 wxString newScale = dlg.m_ScaleCtl->GetValue();
878 if (newScale.Contains(':')) newScale = newScale.AfterFirst(':');
879 double dScale;
880 if (newScale.ToDouble(&dScale)) {
881 // Try to constrain the scale to something reasonable
882 dScale = wxMin(dScale, 3e6);
883 dScale = wxMax(dScale, 1000);
884 double displayScaleNow = pcc->GetScaleValue();
885 double factor = displayScaleNow / dScale;
886 pcc->DoZoomCanvas(factor, false);
887
888 // Run the calculation again, to reduce roundoff error in large scale
889 // jumps.
890 displayScaleNow = pcc->GetScaleValue();
891 factor = displayScaleNow / dScale;
892 pcc->DoZoomCanvas(factor, false);
893 }
894 }
895}
896
897void MUIBar::SetCanvasENCAvailable(bool avail) {
898 m_CanvasENCAvail = avail;
899 if (m_canvasOptions) m_canvasOptions->SetENCAvailable(avail);
900}
901
902void MUIBar::CreateControls() {
903 wxString iconDir = g_Platform->GetSharedDataDir() + _T("uidata/MUI_flat/");
904
905 // Build one button to get sizes
906 MUIButton* tb = new MUIButton(m_parentCanvas, ID_ZOOMIN, m_scaleFactor,
907 iconDir + _T("MUI_zoom-in.svg"));
908 wxSize button_size = tb->m_size;
909 delete tb;
910
911 if (m_orientation == wxHORIZONTAL) {
912 // Buttons
913
914 int xoff = 0;
915 if (g_bShowMuiZoomButtons) {
916 m_zinButton = new MUIButton(m_parentCanvas, ID_ZOOMIN, m_scaleFactor,
917 iconDir + _T("MUI_zoom-in.svg"));
918 m_zinButton->m_position = wxPoint(xoff, 0);
919 xoff += m_zinButton->m_size.x;
920
921 m_zoutButton = new MUIButton(m_parentCanvas, ID_ZOOMOUT, m_scaleFactor,
922 iconDir + _T("MUI_zoom-out.svg"));
923 m_zoutButton->m_position = wxPoint(xoff, 0);
924 xoff += m_zoutButton->m_size.x;
925 }
926
927#ifndef __OCPN__ANDROID__
928 // Scale
929
930 m_scaleButton = new MUITextButton(m_parentCanvas, wxID_ANY, m_scaleFactor,
931 GetBackgroundColor(), "1:400000");
932
933 m_scaleButton->m_position = wxPoint(xoff, 0);
934 if (m_scaleButton->GetButtonBitmap().IsOk()) {
935 int bm_pos_y = (m_scaleButton->GetSize().y -
936 m_scaleButton->GetButtonBitmap().GetHeight()) /
937 2;
938 m_scaleButton->m_position = wxPoint(xoff, bm_pos_y);
939 }
940 xoff += m_scaleButton->m_size.x;
941
942 m_followButton = new MUIButton(m_parentCanvas, ID_FOLLOW, m_scaleFactor,
943 iconDir + _T("MUI_follow.svg"),
944 iconDir + _T("MUI_follow_active.svg"),
945 iconDir + _T("MUI_follow_ahead.svg"));
946 m_followButton->m_position = wxPoint(xoff, 0);
947 xoff += m_followButton->m_size.x;
948#endif
949
950 m_menuButton = new MUIButton(m_parentCanvas, ID_MUI_MENU, m_scaleFactor,
951 iconDir + _T("MUI_menu.svg"));
952 m_menuButton->m_position = wxPoint(xoff, 0);
953 xoff += m_menuButton->m_size.x;
954 m_size.x = xoff;
955 m_size.y = button_size.y;
956
957 } else {
958 int yoff = 0;
959
960 // Buttons
961 if (g_bShowMuiZoomButtons) {
962 m_zinButton = new MUIButton(m_parentCanvas, ID_ZOOMIN, m_scaleFactor,
963 iconDir + _T("MUI_zoom-in.svg"));
964 m_zinButton->m_position = wxPoint(0, yoff);
965 yoff += m_zinButton->m_size.y;
966
967 m_zoutButton = new MUIButton(m_parentCanvas, ID_ZOOMOUT, m_scaleFactor,
968 iconDir + _T("MUI_zoom-out.svg"));
969 m_zoutButton->m_position = wxPoint(0, yoff);
970 yoff += m_zoutButton->m_size.y;
971 }
972
973#ifndef __OCPN__ANDROID__
974 m_followButton = new MUIButton(m_parentCanvas, ID_FOLLOW, m_scaleFactor,
975 iconDir + _T("MUI_follow.svg"),
976 iconDir + _T("MUI_follow_active.svg"),
977 iconDir + _T("MUI_follow_ahead.svg"));
978 m_followButton->m_position = wxPoint(0, yoff);
979 yoff += m_followButton->m_size.y;
980#endif
981
982 m_menuButton = new MUIButton(m_parentCanvas, ID_MUI_MENU, m_scaleFactor,
983 iconDir + _T("MUI_menu.svg"));
984 m_menuButton->m_position = wxPoint(0, yoff);
985 yoff += m_menuButton->m_size.y;
986
987 m_size.y = yoff;
988 m_size.x = button_size.x;
989 }
990}
991
992void MUIBar::SetBestPosition(void) {
993 int x =
994 (m_parentCanvas->GetClientSize().x - (m_size.x + (m_end_margin) * 2.00));
995
996 int bottomOffset = 6;
997
998 int y = m_parentCanvas->GetClientSize().y - m_size.y - bottomOffset;
999 // if ( g_bopengl){
1000 // y -= m_parentCanvas->GetClientSize().y % 1;
1001 // }
1002
1003 wxPoint position = wxPoint(x, y);
1004 m_screenPos = position;
1005
1006 if (m_canvasOptions) {
1007 m_canvasOptions->Destroy();
1008 m_canvasOptions = 0;
1009 }
1010}
1011
1012void MUIBar::UpdateDynamicValues() {
1013 if (!m_scaleButton) return;
1014
1015 wxString scaleString;
1016 int scale = m_parentCanvas->GetScaleValue();
1017
1018 if (scale != m_scale) InvalidateBitmap();
1019 m_scale = scale;
1020
1021 if (scale < 1e6)
1022 scaleString.Printf(_T("1:%d"), scale);
1023 else
1024 scaleString.Printf(_T("1:%4.1f M"), scale / 1e6);
1025
1026 if (m_scaleButton) m_scaleButton->SetText(scaleString);
1027}
1028
1029void MUIBar::SetFollowButtonState(int state) {
1030 if (m_followButton && m_followButton->GetState() != state) {
1031 m_followButton->SetState(state);
1032 InvalidateBitmap();
1033 }
1034}
1035
1036void MUIBar::HandleMenuClick() {
1037 if (!m_canvasOptions) {
1038 m_canvasOptions = new CanvasOptions(m_parentCanvas);
1039
1040 // calculate best size for CanvasOptions dialog
1041
1042 wxPoint parentClientUpperRight =
1043 m_parentCanvas->ClientToScreen(wxPoint(m_parentCanvas->GetSize().x, 0));
1044 wxPoint muibar_top = m_parentCanvas->ClientToScreen(m_screenPos);
1045 int size_y = muibar_top.y - (parentClientUpperRight.y + m_COTopOffset);
1046 size_y -= m_parentCanvas->GetCharHeight();
1047 size_y = wxMax(size_y, 100); // ensure always big enough to see
1048
1049 m_canvasOptions->SetSize(wxSize(-1, size_y));
1050 m_canvasOptionsFullSize = m_canvasOptions->GetSize();
1051 m_canvasOptionsFullSize.x +=
1052 m_canvasOptions->GetCharWidth(); // Allow for scroll bar, since
1053 // sizer won't do it.
1054
1055 m_currentCOPos = m_parentCanvas->ClientToScreen(
1056 wxPoint(m_parentCanvas->GetSize().x, m_COTopOffset));
1057
1058 m_canvasOptions->Move(m_currentCOPos);
1059 m_canvasOptions->Hide();
1060 }
1061
1062 m_canvasOptions->SetENCAvailable(m_CanvasENCAvail);
1063
1064 if (m_canvasOptions->IsShown())
1065 PushCanvasOptions(); // hide it
1066 else {
1067 // Grab the backing bitmap, if available
1068
1069 if (m_coAnimateByBitmaps && m_capture_size_y) {
1070 int overShoot_x = m_canvasOptions->GetSize().x * 2 / 10; // 20%
1071 m_backingPoint =
1072 wxPoint(m_capturePoint.x - overShoot_x, m_capturePoint.y);
1073
1074 m_backingBitmap = wxBitmap(m_canvasOptionsFullSize.x + overShoot_x,
1075 m_capture_size_y, -1);
1076 wxMemoryDC mdcb;
1077 mdcb.SelectObject(m_backingBitmap);
1078 wxScreenDC sdc;
1079 mdcb.Blit(0, 0, m_canvasOptionsFullSize.x + overShoot_x, m_capture_size_y,
1080 &sdc, m_capturePoint.x - overShoot_x, m_capturePoint.y, wxCOPY);
1081 mdcb.SelectObject(wxNullBitmap);
1082 }
1083 PullCanvasOptions();
1084 }
1085}
1086
1087void MUIBar::CaptureCanvasOptionsBitmap() {
1088 m_coSequence = 0;
1089 CanvasOptionTimer.Start(100, wxTIMER_ONE_SHOT);
1090}
1091
1092void MUIBar::CaptureCanvasOptionsBitmapChain(wxTimerEvent& event) {
1093 if (m_coSequence == 0) {
1094 if (!m_canvasOptions) m_canvasOptions = new CanvasOptions(m_parentCanvas);
1095
1096 wxPoint parentClientUpperRight =
1097 m_parentCanvas->ClientToScreen(wxPoint(m_parentCanvas->GetSize().x, 0));
1098 wxRect rmui = m_parentCanvas->GetMUIBarRect();
1099 int size_y = rmui.y - (parentClientUpperRight.y + m_COTopOffset);
1100 size_y -= m_parentCanvas->GetCharHeight();
1101 size_y = wxMax(size_y, 100); // ensure always big enough to see
1102 m_capture_size_y = size_y;
1103
1104 m_canvasOptions->SetSize(wxSize(-1, size_y));
1105
1106 m_capturePoint = m_parentCanvas->ClientToScreen(
1107 wxPoint(m_parentCanvas->GetSize().x, m_COTopOffset));
1108 m_canvasOptions->Move(m_capturePoint);
1109 m_canvasOptions->Show();
1110
1111 m_coSequence++;
1112 CanvasOptionTimer.Start(1, wxTIMER_ONE_SHOT);
1113 }
1114
1115 else if (m_coSequence == 1) {
1116 m_capturePoint = m_parentCanvas->ClientToScreen(
1117 wxPoint(m_parentCanvas->GetSize().x - m_canvasOptionsFullSize.x,
1118 m_COTopOffset));
1119 m_canvasOptions->Move(m_capturePoint);
1120
1121 m_coSequence++;
1122 CanvasOptionTimer.Start(1, wxTIMER_ONE_SHOT);
1123 }
1124
1125 else if (m_coSequence == 2) {
1126 m_animateBitmap =
1127 wxBitmap(m_canvasOptions->GetSize().x, m_capture_size_y, -1);
1128 wxMemoryDC mdc(m_animateBitmap);
1129
1130 wxScreenDC sdc;
1131
1132 mdc.Blit(0, 0, m_canvasOptions->GetSize().x, m_capture_size_y, &sdc,
1133 m_capturePoint.x, m_capturePoint.y, wxCOPY);
1134 mdc.SelectObject(wxNullBitmap);
1135
1136 // delete m_canvasOptions;
1137 // m_canvasOptions = NULL;
1138 }
1139}
1140
1141wxBitmap& MUIBar::CreateBitmap(double displayScale) {
1142 if (m_bitmap.IsOk()) return m_bitmap;
1143
1144 // Make the bitmap
1145 int width = m_size.x;
1146 int height = m_size.y;
1147
1148 wxMemoryDC mdc;
1149 wxBitmap bm(width, height);
1150 mdc.SelectObject(bm);
1151 mdc.SetBackground(wxBrush(GetBackgroundColor()));
1152
1153 mdc.Clear();
1154
1155 wxBitmap bmd;
1156 if (g_bShowMuiZoomButtons) {
1157 wxBitmap bmd = m_zinButton->GetButtonBitmap();
1158 if (bmd.IsOk())
1159 mdc.DrawBitmap(bmd, m_zinButton->m_position.x, m_zinButton->m_position.y,
1160 false);
1161
1162 bmd = m_zoutButton->GetButtonBitmap();
1163 if (bmd.IsOk())
1164 mdc.DrawBitmap(bmd, m_zoutButton->m_position.x,
1165 m_zoutButton->m_position.y, false);
1166 }
1167
1168 if (m_scaleButton) {
1169 bmd = m_scaleButton->GetButtonBitmap();
1170 if (bmd.IsOk()) {
1171 int bm_pos_y = (m_scaleButton->GetSize().y - bmd.GetHeight()) / 2;
1172 int bm_pos_x = m_scaleButton->m_position.x +
1173 (m_scaleButton->GetSize().x - bmd.GetWidth()) / 2;
1174
1175 mdc.DrawBitmap(bmd, bm_pos_x, bm_pos_y, false);
1176 }
1177 }
1178
1179 if (m_followButton) {
1180 bmd = m_followButton->GetButtonBitmap();
1181 if (bmd.IsOk())
1182 mdc.DrawBitmap(bmd, m_followButton->m_position.x,
1183 m_followButton->m_position.y, false);
1184 }
1185
1186 if (m_menuButton) {
1187 bmd = m_menuButton->GetButtonBitmap();
1188 if (bmd.IsOk())
1189 mdc.DrawBitmap(bmd, m_menuButton->m_position.x,
1190 m_menuButton->m_position.y, false);
1191 }
1192
1193 mdc.SelectObject(wxNullBitmap);
1194
1195 m_bitmap = bm;
1196 return m_bitmap;
1197}
1198
1199void MUIBar::DrawGL(ocpnDC& gldc, double displayScale) {
1200#ifdef ocpnUSE_GL
1201
1202 wxColour backColor = GetBackgroundColor();
1203 gldc.SetBrush(wxBrush(backColor));
1204 gldc.SetPen(wxPen(backColor));
1205
1206 wxRect r = wxRect(m_screenPos, m_size);
1207 if (m_orientation == wxHORIZONTAL)
1208 gldc.DrawRoundedRectangle(
1209 (r.x - m_end_margin / 2) * displayScale, (r.y - 1) * displayScale,
1210 (r.width + m_end_margin) * displayScale, (r.height + 2) * displayScale,
1211 (m_end_margin * 1) * displayScale);
1212 else
1213 gldc.DrawRoundedRectangle((r.x - 1) * displayScale,
1214 (r.y - m_end_margin / 2) * displayScale,
1215 (r.width + 2) * displayScale,
1216 (r.height + 2 * m_end_margin) * displayScale,
1217 (m_end_margin * 1.5) * displayScale);
1218
1219 int width = m_size.x;
1220 int height = m_size.y;
1221
1222 CreateBitmap(displayScale);
1223
1224 // Make a GL texture
1225 if (!m_texture) {
1226 glGenTextures(1, &m_texture);
1227
1228 glBindTexture(g_texture_rectangle_format, m_texture);
1229 glTexParameterf(g_texture_rectangle_format, GL_TEXTURE_MIN_FILTER,
1230 GL_NEAREST);
1231 glTexParameteri(g_texture_rectangle_format, GL_TEXTURE_MAG_FILTER,
1232 GL_NEAREST);
1233 glTexParameteri(g_texture_rectangle_format, GL_TEXTURE_WRAP_S,
1234 GL_CLAMP_TO_EDGE);
1235 glTexParameteri(g_texture_rectangle_format, GL_TEXTURE_WRAP_T,
1236 GL_CLAMP_TO_EDGE);
1237 } else {
1238 glBindTexture(g_texture_rectangle_format, m_texture);
1239 }
1240
1241 // fill texture data
1242 if (m_bitmap.IsOk()) {
1243 wxImage image = m_bitmap.ConvertToImage();
1244 if (image.IsOk()) {
1245 unsigned char* d = image.GetData();
1246 if (d) {
1247 unsigned char* e = new unsigned char[4 * width * height];
1248 for (int y = 0; y < height; y++)
1249 for (int x = 0; x < width; x++) {
1250 int i = y * width + x;
1251 memcpy(e + 4 * i, d + 3 * i, 3);
1252 e[4 * i + 3] =
1253 255; // d[3*i + 2] == 255 ? 0:255; //255 - d[3 * i + 2];
1254 }
1255 glTexImage2D(g_texture_rectangle_format, 0, GL_RGBA, width, height, 0,
1256 GL_RGBA, GL_UNSIGNED_BYTE, e);
1257 delete[] e;
1258 glDisable(g_texture_rectangle_format);
1259 glDisable(GL_BLEND);
1260 }
1261 }
1262 }
1263
1264 // Render the texture
1265 if (m_texture) {
1266 glEnable(g_texture_rectangle_format);
1267 glBindTexture(g_texture_rectangle_format, m_texture);
1268 glEnable(GL_BLEND);
1269
1270 int x0 = m_screenPos.x, x1 = x0 + width;
1271 int y0 = m_screenPos.y - 0, y1 = y0 + height;
1272 x0 *= displayScale;
1273 x1 *= displayScale;
1274 y0 *= displayScale;
1275 y1 *= displayScale;
1276
1277 float tx, ty;
1278 if (GL_TEXTURE_RECTANGLE_ARB == g_texture_rectangle_format)
1279 tx = width, ty = height;
1280 else
1281 tx = ty = 1;
1282
1283 float coords[8];
1284 float uv[8];
1285
1286 // normal uv
1287 uv[0] = 0;
1288 uv[1] = 0;
1289 uv[2] = tx;
1290 uv[3] = 0;
1291 uv[4] = tx;
1292 uv[5] = ty;
1293 uv[6] = 0;
1294 uv[7] = ty;
1295
1296 // pixels
1297 coords[0] = x0;
1298 coords[1] = y0;
1299 coords[2] = x1;
1300 coords[3] = y0;
1301 coords[4] = x1;
1302 coords[5] = y1;
1303 coords[6] = x0;
1304 coords[7] = y1;
1305
1306 m_parentCanvas->GetglCanvas()->RenderTextures(gldc, coords, uv, 4,
1307 m_parentCanvas->GetpVP());
1308
1309 glDisable(g_texture_rectangle_format);
1310 glBindTexture(g_texture_rectangle_format, 0);
1311 glDisable(GL_BLEND);
1312 }
1313#endif
1314
1315 return;
1316}
1317
1318void MUIBar::DrawDC(ocpnDC& dc, double displayScale) {
1319 CreateBitmap(1.0);
1320 dc.DrawBitmap(m_bitmap, m_screenPos.x, m_screenPos.y, false);
1321}
1322
1323void MUIBar::ResetCanvasOptions() {
1324 delete m_canvasOptions;
1325 m_canvasOptions = NULL;
1326}
1327
1328void MUIBar::PullCanvasOptions() {
1329 // Target position
1330 int cox = m_parentCanvas->GetSize().x - m_canvasOptionsFullSize.x;
1331 int coy = m_COTopOffset;
1332 m_targetCOPos = m_parentCanvas->ClientToScreen(wxPoint(cox, coy));
1333
1334 if (!m_bEffects) {
1335 m_canvasOptions->Move(m_targetCOPos);
1336 m_canvasOptions->Show();
1337 return;
1338 }
1339
1340 // Capture the animation bitmap, if required..
1341
1342 if (m_coAnimateByBitmaps && !m_animateBitmap.IsOk()) {
1343 m_canvasOptions->Move(m_targetCOPos);
1344 m_canvasOptions->Show();
1345 CaptureCanvasOptionsBitmap();
1346 return;
1347 }
1348
1349 // Setup animation parameters
1350 // Start Position
1351 m_startCOPos = m_canvasOptions->GetPosition();
1352
1353 // Present Position
1354 m_currentCOPos = m_startCOPos;
1355
1356 m_animationType = CO_ANIMATION_CUBIC_REVERSE; // CO_ANIMATION_CUBIC_BACK_IN;
1357 m_animateSteps = 10;
1358 m_animationTotalTime = 200; // msec
1359
1360 m_pushPull = CO_PULL;
1361 auto pcc = dynamic_cast<ChartCanvas*>(m_parentCanvas);
1362 pcc->m_b_paint_enable = false;
1363
1364 // Start the animation....
1365 m_animateStep = 0;
1366 m_canvasOptionsAnimationTimer.Start(10, true);
1367 m_canvasOptions->Move(m_targetCOPos);
1368 m_canvasOptions->Hide();
1369}
1370
1371void MUIBar::PushCanvasOptions() {
1372 if (!m_bEffects) {
1373 m_canvasOptions->Hide();
1374 return;
1375 }
1376
1377 // Setup animation parameters
1378
1379 // Target position
1380 int cox = m_parentCanvas->GetSize().x;
1381 int coy = 20;
1382
1383 if (1)
1384 m_targetCOPos = m_parentCanvas->ClientToScreen(wxPoint(cox, coy));
1385 else
1386 m_targetCOPos = wxPoint(cox, coy);
1387
1388 // Start Position
1389 m_startCOPos = m_canvasOptions->GetPosition();
1390
1391 // Present Position
1392 m_currentCOPos = m_startCOPos;
1393
1394 // Animation type
1395 m_animationType = CO_ANIMATION_LINEAR;
1396 m_animateSteps = 5;
1397 m_animationTotalTime = 100; // msec
1398 m_pushPull = CO_PUSH;
1399 auto pcc = dynamic_cast<ChartCanvas*>(m_parentCanvas);
1400
1401 // Start the animation....
1402 m_animateStep = 0;
1403 m_canvasOptionsAnimationTimer.Start(10, true);
1404 m_canvasOptions->Show();
1405}
1406
1407void MUIBar::onCanvasOptionsAnimationTimerEvent(wxTimerEvent& event) {
1408 double progress = m_animateStep / (double)m_animateSteps;
1409 double valueX = getValue(m_animationType, progress);
1410
1411 double dx = (m_targetCOPos.x - m_startCOPos.x) * valueX;
1412
1413 wxPoint newPos = wxPoint(m_startCOPos.x + dx, m_currentCOPos.y);
1414
1415 int size_x;
1416 if (m_pushPull == CO_PULL)
1417 size_x = abs(dx);
1418 else
1419 size_x = (m_targetCOPos.x - m_startCOPos.x) - abs(dx);
1420
1421 if (!m_coAnimateByBitmaps) {
1422 m_canvasOptions->SetSize(newPos.x, newPos.y, size_x, wxDefaultCoord,
1423 wxSIZE_USE_EXISTING);
1424 // m_canvasOptions->GetSizer()->Layout();
1425 m_canvasOptions->Show();
1426 } else {
1427 m_canvasOptions->Hide();
1428 wxScreenDC sdc;
1429
1430 if (1 /*m_pushPull == CO_PULL*/) {
1431 // restore Backing bitmap, to cover any overshoot
1432 if (m_backingBitmap.IsOk()) {
1433 wxMemoryDC mdc_back(m_backingBitmap);
1434 sdc.Blit(m_backingPoint.x, m_backingPoint.y,
1435 m_backingBitmap.GetWidth() - size_x,
1436 m_backingBitmap.GetHeight(), &mdc_back, 0, 0, wxCOPY);
1437 }
1438 }
1439
1440 wxMemoryDC mdc(m_animateBitmap);
1441 sdc.Blit(newPos.x, newPos.y, size_x, m_animateBitmap.GetHeight(), &mdc, 0,
1442 0, wxCOPY);
1443 mdc.SelectObject(wxNullBitmap);
1444 }
1445
1446 m_currentCOPos = newPos;
1447
1448 double dt = m_animationTotalTime / m_animateSteps;
1449
1450 if (m_animateStep++ < m_animateSteps + 1) {
1451 m_canvasOptionsAnimationTimer.Start(dt, true);
1452 } else {
1453 m_currentCOPos = m_targetCOPos;
1454 m_canvasOptions->Show(m_pushPull == CO_PULL);
1455
1456 auto pcc = dynamic_cast<ChartCanvas*>(m_parentCanvas);
1457 if (pcc) {
1458 pcc->m_b_paint_enable = true;
1459
1460 if (m_pushPull == CO_PUSH) {
1461 delete m_canvasOptions;
1462 m_canvasOptions = NULL;
1463 }
1464#ifdef __WXOSX__
1465 if (m_pushPull == CO_PUSH) pcc->TriggerDeferredFocus();
1466#else
1467 pcc->TriggerDeferredFocus();
1468#endif
1469
1470 pcc->Refresh();
1471 }
1472 }
1473}
1474
1475// Animation support
1476
1477double bounceMaker(double t, double c, double a) {
1478 if (t == 1.0) return c;
1479 if (t < (4 / 11.0)) {
1480 return c * (7.5625 * t * t);
1481 } else if (t < (8 / 11.0)) {
1482 t -= (6 / 11.0);
1483 return -a * (1. - (7.5625 * t * t + .75)) + c;
1484 } else if (t < (10 / 11.0)) {
1485 t -= (9 / 11.0);
1486 return -a * (1. - (7.5625 * t * t + .9375)) + c;
1487 } else {
1488 t -= (21 / 22.0);
1489 return -a * (1. - (7.5625 * t * t + .984375)) + c;
1490 }
1491}
1492
1493double getValue(int animationType, double t) {
1494 double value = 0;
1495 double s = 1;
1496 double tp;
1497 switch (animationType) {
1498 case CO_ANIMATION_LINEAR:
1499 default:
1500 value = t;
1501 break;
1502 case CO_ANIMATION_CUBIC:
1503 tp = t - 1.0;
1504 value = tp * tp * tp + 1;
1505 // value = t*t*t;
1506 break;
1507 case CO_ANIMATION_CUBIC_REVERSE:
1508 tp = t - 1.0;
1509 value = tp * tp * tp + 1;
1510 break;
1511 case CO_ANIMATION_CUBIC_BOUNCE_IN:
1512 value = bounceMaker(t, 1, s);
1513 break;
1514
1515 case CO_ANIMATION_CUBIC_BACK_IN:
1516 tp = t - 1.0;
1517 value = tp * tp * ((s + 1) * tp + s) + 1;
1518 break;
1519 }
1520
1521 return value;
1522}
Represents the Canvas Options dialog.
ChartCanvas - Main chart display and interaction component.
Definition chcanv.h:148
static bool ShowToolTips()
Should we show tooltips?
Definition MUIBar.cpp:373
Provides platform-specific support utilities for OpenCPN.
bool Create(wxWindow *parent, wxWindowID id=wxID_ANY, const wxString &caption=_("Set scale"), const wxPoint &pos=wxDefaultPosition, const wxSize &size=wxDefaultSize, long style=wxDEFAULT_DIALOG_STYLE)
Creation.
Definition MUIBar.cpp:147
void CreateControls()
Definition MUIBar.cpp:164
wxTextCtrl * m_ScaleCtl
Should we show tooltips?
Definition MUIBar.cpp:116
SetScaleDialog()
Constructors.
Definition MUIBar.cpp:130
Device context class that can use either wxDC or OpenGL for drawing.
Definition ocpndc.h:64
Definition ssfn.h:176