27#include "wx/tokenzr.h"
28#include "wx/datetime.h"
30#include <wx/wfstream.h>
32#include <wx/filename.h>
34#include <wx/graphics.h>
37#include <wx/stdpaths.h>
49#include <wx/arrimpl.cpp>
52#include "android_jvm.h"
56double m_cursor_lat, m_cursor_lon;
92 return ((x - i) >= 0.5) ? (i + 1) : (i);
94 return (-x + i >= 0.5) ? (i - 1) : (i);
104#if wxCHECK_VERSION(2, 9, 4)
105#define SetBitmapLabelLabel SetBitmap
108#define DEFAULT_STYLE \
109 = wxCAPTION | wxCLOSE_BOX | wxSYSTEM_MENU | wxTAB_TRAVERSAL
111WX_DEFINE_OBJARRAY(ArrayOfGribRecordSets);
114static int CompareFileStringTime(
const wxString &first,
115 const wxString &second) {
117 wxFileName s(second);
118 wxTimeSpan sp = s.GetModificationTime() - f.GetModificationTime();
119 return sp.GetMinutes();
125wxWindow *GetGRIBCanvas() {
148GribTimelineRecordSet::~GribTimelineRecordSet() {
153void GribTimelineRecordSet::ClearCachedData() {
157 for (
unsigned int j = 0; j <
m_IsobarArray[i]->GetCount(); j++) {
171GRIBUICtrlBar::GRIBUICtrlBar(wxWindow *parent, wxWindowID
id,
172 const wxString &title,
const wxPoint &pos,
173 const wxSize &size,
long style,
grib_pi *ppi,
181 pReq_Dialog =
nullptr;
182 m_bGRIBActiveFile =
nullptr;
183 m_pTimelineSet =
nullptr;
184 m_gCursorData =
nullptr;
185 m_gGRIBUICData =
nullptr;
186 m_gtk_started =
false;
191 m_fgCtrlGrabberSize->Add(m_gGrabber, 0, wxALL, 0);
193 this->SetSizer(m_fgCtrlBarSizer);
195 m_fgCtrlBarSizer->Fit(
this);
198 m_tFormatRefresh.Connect(
199 wxEVT_TIMER, wxTimerEventHandler(GRIBUICtrlBar::OnFormatRefreshTimer),
204 wxDateTime referenceDate(1, wxDateTime::Jan, 2021, 12, 0, 0);
208 m_tFormatRefresh.Start(5000);
211 pConf->SetPath(_T (
"/Settings/GRIB" ));
212 pConf->Read(_T (
"WindPlot" ), &m_bDataPlot[GribOverlaySettings::WIND],
214 pConf->Read(_T (
"WindGustPlot" ),
215 &m_bDataPlot[GribOverlaySettings::WIND_GUST],
false);
216 pConf->Read(_T (
"PressurePlot" ),
217 &m_bDataPlot[GribOverlaySettings::PRESSURE],
false);
218 pConf->Read(_T (
"WavePlot" ), &m_bDataPlot[GribOverlaySettings::WAVE],
220 pConf->Read(_T (
"CurrentPlot" ),
221 &m_bDataPlot[GribOverlaySettings::CURRENT],
false);
222 pConf->Read(_T (
"PrecipitationPlot" ),
223 &m_bDataPlot[GribOverlaySettings::PRECIPITATION],
false);
224 pConf->Read(_T (
"CloudPlot" ), &m_bDataPlot[GribOverlaySettings::CLOUD],
226 pConf->Read(_T (
"AirTemperaturePlot" ),
227 &m_bDataPlot[GribOverlaySettings::AIR_TEMPERATURE],
false);
228 pConf->Read(_T (
"SeaTemperaturePlot" ),
229 &m_bDataPlot[GribOverlaySettings::SEA_TEMPERATURE],
false);
230 pConf->Read(_T (
"CAPEPlot" ), &m_bDataPlot[GribOverlaySettings::CAPE],
232 pConf->Read(_T (
"CompReflectivityPlot" ),
233 &m_bDataPlot[GribOverlaySettings::COMP_REFL],
false);
235 pConf->Read(_T (
"CursorDataShown" ), &m_CDataIsShown,
true);
237 pConf->Read(_T (
"lastdatatype" ), &m_lastdatatype, 0);
239 pConf->SetPath(_T (
"/Settings/GRIB/FileNames" ));
240 m_file_names.Clear();
241 if (pConf->GetNumberOfEntries()) {
244 bool bCont = pConf->GetFirstEntry(str, dummy);
246 pConf->Read(str, &val);
247 m_file_names.Add(val);
248 bCont = pConf->GetNextEntry(str, dummy);
252 wxStandardPathsBase &spath = wxStandardPaths::Get();
254 pConf->SetPath(_T (
"/Directories" ));
255 pConf->Read(_T (
"GRIBDirectory" ), &m_grib_dir);
257 pConf->SetPath(_T(
"/PlugIns/GRIB" ));
261 pConf->SetPath(_T (
"/Settings/GRIB/XyGrib" ));
262 pConf->Read(_T(
"AtmModelIndex" ), &xyGribConfig.atmModelIndex, 0);
263 pConf->Read(_T(
"WaveModelIndex" ), &xyGribConfig.waveModelIndex, 0);
264 pConf->Read(_T(
"ResolutionIndex" ), &xyGribConfig.resolutionIndex, 0);
265 pConf->Read(_T(
"DurationIndex" ), &xyGribConfig.durationIndex, 0);
266 pConf->Read(_T(
"RunIndex" ), &xyGribConfig.runIndex, 0);
267 pConf->Read(_T(
"IntervalIndex" ), &xyGribConfig.intervalIndex, 0);
268 pConf->Read(_T(
"Wind" ), &xyGribConfig.wind,
true);
269 pConf->Read(_T(
"Gust" ), &xyGribConfig.gust,
true);
270 pConf->Read(_T(
"Pressure" ), &xyGribConfig.pressure,
false);
271 pConf->Read(_T(
"Temperature" ), &xyGribConfig.temperature,
true);
272 pConf->Read(_T(
"Cape" ), &xyGribConfig.cape,
false);
273 pConf->Read(_T(
"Reflectivity" ), &xyGribConfig.reflectivity,
false);
274 pConf->Read(_T(
"CloudCover" ), &xyGribConfig.cloudCover,
true);
275 pConf->Read(_T(
"Precipitation" ), &xyGribConfig.precipitation,
true);
276 pConf->Read(_T(
"WaveHeight" ), &xyGribConfig.waveHeight,
true);
277 pConf->Read(_T(
"WindWaves" ), &xyGribConfig.windWaves,
true);
283 m_tPlayStop.Connect(wxEVT_TIMER,
284 wxTimerEventHandler(GRIBUICtrlBar::OnPlayStopTimer),
287 Connect(wxEVT_MOVE, wxMoveEventHandler(GRIBUICtrlBar::OnMove));
289 m_OverlaySettings.Read();
294 SetMinSize(GetBestSize());
295 if (m_ProjectBoatPanel) {
296 m_ProjectBoatPanel->SetSpeed(pPlugIn->m_boat_sog);
297 m_ProjectBoatPanel->SetCourse(pPlugIn->m_boat_cog);
299 m_highlight_latmax = 0;
300 m_highlight_lonmax = 0;
301 m_highlight_latmin = 0;
302 m_highlight_lonmin = 0;
304#ifndef __OCPN__ANDROID__
310 createRequestDialog();
314GRIBUICtrlBar::~GRIBUICtrlBar() {
319 pConf->SetPath(_T (
"/Settings/GRIB" ));
320 pConf->Write(_T (
"WindPlot" ), m_bDataPlot[GribOverlaySettings::WIND]);
321 pConf->Write(_T (
"WindGustPlot" ),
322 m_bDataPlot[GribOverlaySettings::WIND_GUST]);
323 pConf->Write(_T (
"PressurePlot" ),
324 m_bDataPlot[GribOverlaySettings::PRESSURE]);
325 pConf->Write(_T (
"WavePlot" ), m_bDataPlot[GribOverlaySettings::WAVE]);
326 pConf->Write(_T (
"CurrentPlot" ),
327 m_bDataPlot[GribOverlaySettings::CURRENT]);
328 pConf->Write(_T (
"PrecipitationPlot" ),
329 m_bDataPlot[GribOverlaySettings::PRECIPITATION]);
330 pConf->Write(_T (
"CloudPlot" ), m_bDataPlot[GribOverlaySettings::CLOUD]);
331 pConf->Write(_T (
"AirTemperaturePlot" ),
332 m_bDataPlot[GribOverlaySettings::AIR_TEMPERATURE]);
333 pConf->Write(_T (
"SeaTemperaturePlot" ),
334 m_bDataPlot[GribOverlaySettings::SEA_TEMPERATURE]);
335 pConf->Write(_T (
"CAPEPlot" ), m_bDataPlot[GribOverlaySettings::CAPE]);
336 pConf->Write(_T (
"CompReflectivityPlot" ),
337 m_bDataPlot[GribOverlaySettings::COMP_REFL]);
339 pConf->Write(_T (
"CursorDataShown" ), m_CDataIsShown);
341 pConf->Write(_T (
"lastdatatype" ), m_lastdatatype);
343 pConf->SetPath(_T (
"/Settings/GRIB/FileNames" ));
344 int iFileMax = pConf->GetNumberOfEntries();
348 for (
int i = 0; i < iFileMax; i++) {
349 if (pConf->GetFirstEntry(key, dummy)) pConf->DeleteEntry(key,
false);
353 for (
unsigned int i = 0; i <
m_file_names.GetCount(); i++) {
355 key.Printf(_T(
"Filename%d"), i);
359 pConf->SetPath(_T (
"/Directories" ));
360 pConf->Write(_T (
"GRIBDirectory" ),
m_grib_dir);
363 pConf->SetPath(_T (
"/Settings/GRIB/XyGrib" ));
364 pConf->Write(_T(
"AtmModelIndex" ), xyGribConfig.atmModelIndex);
365 pConf->Write(_T(
"WaveModelIndex" ), xyGribConfig.waveModelIndex);
366 pConf->Write(_T(
"ResolutionIndex" ), xyGribConfig.resolutionIndex);
367 pConf->Write(_T(
"DurationIndex" ), xyGribConfig.durationIndex);
368 pConf->Write(_T(
"RunIndex" ), xyGribConfig.runIndex);
369 pConf->Write(_T(
"IntervalIndex" ), xyGribConfig.intervalIndex);
370 pConf->Write(_T(
"Wind" ), xyGribConfig.wind);
371 pConf->Write(_T(
"Gust" ), xyGribConfig.gust);
372 pConf->Write(_T(
"Pressure" ), xyGribConfig.pressure);
373 pConf->Write(_T(
"Temperature" ), xyGribConfig.temperature);
374 pConf->Write(_T(
"Cape" ), xyGribConfig.cape);
375 pConf->Write(_T(
"Reflectivity" ), xyGribConfig.reflectivity);
376 pConf->Write(_T(
"CloudCover" ), xyGribConfig.cloudCover);
377 pConf->Write(_T(
"Precipitation" ), xyGribConfig.precipitation);
378 pConf->Write(_T(
"WaveHeight" ), xyGribConfig.waveHeight);
379 pConf->Write(_T(
"WindWaves" ), xyGribConfig.windWaves);
385void GRIBUICtrlBar::SetScaledBitmap(
double factor) {
387 m_ScaledFactor = wxRound(factor * 4.0) / 4.0;
389 m_bpPrev->SetBitmapLabel(
390 GetScaledBitmap(wxBitmap(prev), _T(
"prev"), m_ScaledFactor));
391 m_bpNext->SetBitmapLabel(
392 GetScaledBitmap(wxBitmap(next), _T(
"next"), m_ScaledFactor));
393 m_bpAltitude->SetBitmapLabel(
394 GetScaledBitmap(wxBitmap(altitude), _T(
"altitude"), m_ScaledFactor));
395 m_bpNow->SetBitmapLabel(
396 GetScaledBitmap(wxBitmap(now), _T(
"now"), m_ScaledFactor));
397 m_bpZoomToCenter->SetBitmapLabel(
398 GetScaledBitmap(wxBitmap(zoomto), _T(
"zoomto"), m_ScaledFactor));
399 m_bpPlay->SetBitmapLabel(
400 GetScaledBitmap(wxBitmap(play), _T(
"play"), m_ScaledFactor));
401 m_bpShowCursorData->SetBitmapLabel(GetScaledBitmap(
402 wxBitmap(m_CDataIsShown ? curdata : ncurdata),
403 m_CDataIsShown ? _T(
"curdata") : _T(
"ncurdata"), m_ScaledFactor));
405 m_bpOpenFile->SetBitmapLabel(
406 GetScaledBitmap(wxBitmap(openfile), _T(
"openfile"), m_ScaledFactor));
407 m_bpSettings->SetBitmapLabel(
408 GetScaledBitmap(wxBitmap(setting), _T(
"setting"), m_ScaledFactor));
414#ifdef __OCPN__ANDROID__
415 m_sTimeline->SetSize(wxSize(20 * m_ScaledFactor, -1));
416 m_sTimeline->SetMinSize(wxSize(20 * m_ScaledFactor, -1));
418 m_sTimeline->SetSize(wxSize(90 * m_ScaledFactor, -1));
419 m_sTimeline->SetMinSize(wxSize(90 * m_ScaledFactor, -1));
424 if (
nullptr == m_bpRequest)
return;
425 m_bpRequest->SetBitmapLabel(
426 GetScaledBitmap(wxBitmap(request), _T(
"request"), m_ScaledFactor));
427 m_bpRequest->SetToolTip(_(
"Start a download request"));
430void GRIBUICtrlBar::OpenFile(
bool newestFile) {
431 m_bpPlay->SetBitmapLabel(
432 GetScaledBitmap(wxBitmap(play), _T(
"play"), m_ScaledFactor));
433 m_cRecordForecast->Clear();
434 pPlugIn->GetGRIBOverlayFactory()->ClearParticles();
440 m_sTimeline->SetValue(0);
442 m_InterpolateMode =
false;
444 m_SelectionIsSaved =
false;
445 m_HasAltitude =
false;
466 title = (_(
"File: "));
467 title.Append(fn.GetFullName());
468 if (rsa->GetCount() == 0) {
471 title.Prepend(_(
"Error! ")).Append(_(
" contains no valid data!"));
473 PopulateComboDataList();
479 if (rsa->GetCount() > 1) {
481 &last = rsa->Item(rsa->GetCount() - 1);
484 wxTimeSpan span = wxDateTime(last.m_Reference_Time) -
486 m_TimeLineHours = span.GetHours();
489 int halfintermin(wxTimeSpan(wxDateTime(second.m_Reference_Time) -
493 for (m_FileIntervalIndex = 0;; m_FileIntervalIndex++) {
498 if (m_FileIntervalIndex > 0) m_FileIntervalIndex--;
506 title = _(
"No valid GRIB file");
508 pPlugIn->GetGRIBOverlayFactory()->SetMessage(title);
510 SetTimeLineMax(
false);
512 if (
pPlugIn->GetStartOptions() &&
513 m_TimeLineHours != 0)
514 ComputeBestForecastForNow();
520 for (
int i = 1; i < 5; i++) {
525 m_HasAltitude =
true;
530#ifdef __OCPN__ANDROID__
531 m_bpSettings->Enable(
true);
537 m_sTimeline->Enable(
m_pTimelineSet !=
nullptr && m_TimeLineHours);
556 bool bconfigOK =
false;
557 if (m_bDataPlot[GribOverlaySettings::WIND] &&
560 if (m_bDataPlot[GribOverlaySettings::WIND_GUST] &&
563 if (m_bDataPlot[GribOverlaySettings::PRESSURE] &&
566 if (m_bDataPlot[GribOverlaySettings::WAVE] &&
569 if (m_bDataPlot[GribOverlaySettings::WAVE] &&
572 if (m_bDataPlot[GribOverlaySettings::CURRENT] &&
576 if (m_bDataPlot[GribOverlaySettings::PRECIPITATION] &&
579 if (m_bDataPlot[GribOverlaySettings::CLOUD] &&
582 if (m_bDataPlot[GribOverlaySettings::AIR_TEMPERATURE] &&
585 if (m_bDataPlot[GribOverlaySettings::SEA_TEMPERATURE] &&
588 if (m_bDataPlot[GribOverlaySettings::CAPE] &&
591 if (m_bDataPlot[GribOverlaySettings::COMP_REFL] &&
598 for (
int i = 0; i < (int)GribOverlaySettings::GEO_ALTITUDE; i++) {
600 m_bDataPlot[i] =
true;
607 double *latmin,
double *latmax,
608 double *lonmin,
double *lonmax) {
611 double ltmi = -GRIB_NOTDEF, ltma = GRIB_NOTDEF, lnmi = -GRIB_NOTDEF,
613 for (
unsigned int i = 0; i <
Idx_COUNT; i++) {
616 if (pGRA->getLatMin() < ltmi) ltmi = pGRA->getLatMin();
617 if (pGRA->getLatMax() > ltma) ltma = pGRA->getLatMax();
618 if (pGRA->getLonMin() < lnmi) lnmi = pGRA->getLonMin();
619 if (pGRA->getLonMax() > lnma) lnma = pGRA->getLonMax();
621 if (ltmi == -GRIB_NOTDEF || lnmi == -GRIB_NOTDEF || ltma == GRIB_NOTDEF ||
625 if (latmin) *latmin = ltmi;
626 if (latmax) *latmax = ltma;
627 if (lonmin) *lonmin = lnmi;
628 if (lonmax) *lonmax = lnma;
635 : m_files(files), m_pattern(pattern) {}
636 virtual wxDirTraverseResult OnFile(
const wxString &filename) {
637 if (m_pattern.Matches(filename)) m_files.Add(filename);
638 return wxDIR_CONTINUE;
640 virtual wxDirTraverseResult OnDir(
const wxString &WXUNUSED(dirname)) {
645 wxArrayString &m_files;
646 const wxRegEx &m_pattern;
649wxArrayString GRIBUICtrlBar::GetFilesInDirectory() {
650 wxArrayString file_array;
651 if (!wxDir::Exists(
m_grib_dir))
return file_array;
655 wxRegEx pattern(_T(
".+\\.gri?b2?(\\.(bz2|gz))?$"),
656 wxRE_EXTENDED | wxRE_ICASE | wxRE_NOSUB);
659 dir.Traverse(collector);
661 CompareFileStringTime);
665void GRIBUICtrlBar::SetCursorLatLon(
double lat,
double lon) {
669 if (m_vpMouse && ((lat > m_vpMouse->
lat_min) && (lat < m_vpMouse->lat_max)) &&
670 ((lon > m_vpMouse->
lon_min) && (lon < m_vpMouse->lon_max)))
675 if (!m_CDataIsShown)
return;
677 if (m_DialogStyle >> 1 == SEPARATED) {
678 if (m_gGRIBUICData) {
679 if (!m_gGRIBUICData->m_gCursorData->m_tCursorTrackTimer.IsRunning())
680 m_gGRIBUICData->m_gCursorData->m_tCursorTrackTimer.Start(
681 50, wxTIMER_ONE_SHOT);
685 if (!m_gCursorData->m_tCursorTrackTimer.IsRunning())
686 m_gCursorData->m_tCursorTrackTimer.Start(50, wxTIMER_ONE_SHOT);
691void GRIBUICtrlBar::OnShowCursorData(wxCommandEvent &event) {
692 m_CDataIsShown = !m_CDataIsShown;
693 m_bpShowCursorData->SetBitmapLabel(GetScaledBitmap(
694 wxBitmap(m_CDataIsShown ? curdata : ncurdata),
695 m_CDataIsShown ? _T(
"curdata") : _T(
"ncurdata"), m_ScaledFactor));
696 SetDialogsStyleSizePosition(
true);
699void GRIBUICtrlBar::SetDialogsStyleSizePosition(
bool force_recompute) {
704 if (!force_recompute &&
705 (m_old_DialogStyle == m_DialogStyle
707 (m_old_DialogStyle >> 1 == ATTACHED && m_DialogStyle >> 1 == ATTACHED)))
710 bool m_HasCaption = GetWindowStyleFlag() == (wxCAPTION | wxCLOSE_BOX |
711 wxSYSTEM_MENU | wxTAB_TRAVERSAL);
716 int state = (m_DialogStyle >> 1 == ATTACHED && m_CDataIsShown) ? 0 : 1;
719 bool vis = i > 0 ? true : m_HasAltitude ? true :
false;
720 if (FindWindow(i + ID_CTRLALTITUDE))
721 FindWindow(i + ID_CTRLALTITUDE)
727 m_bpShowCursorData->SetToolTip(m_CDataIsShown ? _(
"Hide data at cursor")
728 : _(
"Show data at cursor"));
729 m_bpPlay->SetToolTip(_(
"Start play back"));
734 m_gCursorData->Hide();
735 m_fgCDataSizer->Detach(m_gCursorData);
738 SetMinSize(wxSize(0, 0));
742 if (m_gGRIBUICData) {
743 m_gGRIBUICData->Destroy();
744 m_gGRIBUICData =
nullptr;
747 if ((m_DialogStyle >> 1 == SEPARATED || !m_CDataIsShown) &&
754 if (m_CDataIsShown) {
755 if (m_DialogStyle >> 1 == ATTACHED) {
757 if (!m_gCursorData) m_gCursorData =
new CursorData(
this, *
this);
758 pPlugIn->SetDialogFont(m_gCursorData);
759 m_gCursorData->PopulateTrackingControls(
false);
761 if (m_fgCDataSizer->GetItem(m_gCursorData) ==
nullptr)
762 m_fgCDataSizer->Add(m_gCursorData, 0);
763 m_gCursorData->Show();
765 }
else if (m_DialogStyle >> 1 == SEPARATED) {
768 m_gGRIBUICData->m_gCursorData->PopulateTrackingControls(
769 m_DialogStyle == SEPARATED_VERTICAL);
770 pPlugIn->SetDialogFont(m_gGRIBUICData->m_gCursorData);
771 m_gGRIBUICData->Fit();
772 m_gGRIBUICData->Update();
773 m_gGRIBUICData->Show();
775 m_gGRIBUICData->Layout();
776 m_gGRIBUICData->Fit();
781 wxSize sd = GetSize();
783 if (!m_gtk_started && m_HasCaption ) {
785 m_gtk_started =
true;
788 SetSize(wxSize(sd.x, sd.y));
789 SetMinSize(wxSize(sd.x, sd.y));
790#ifdef __OCPN__ANDROID__
796 wxPoint pNew =
pPlugIn->GetCtrlBarXY();
797 pNew.x = tbRect.x + tbRect.width + 4;
805 if (sd.x > widthAvail) {
808 int target_char_width = (float)widthAvail / 28;
817 wxFONTSTYLE_NORMAL, wxFONTWEIGHT_NORMAL,
819 dc.GetTextExtent(_T(
"W"), &width, &height,
nullptr,
nullptr, sFont);
820 if (width <= target_char_width) bOK =
true;
822 if (pointSize <= 10) bOK =
true;
825 m_cRecordForecast->SetFont(*sFont);
830 SetSize(wxSize(widthAvail, sd.y));
831 SetMinSize(wxSize(widthAvail, sd.y));
835 wxPoint pNow =
pPlugIn->GetCtrlBarXY();
842 m_old_DialogStyle = m_DialogStyle;
845void GRIBUICtrlBar::OnAltitude(wxCommandEvent &event) {
846 if (!m_HasAltitude)
return;
848 wxMenu *amenu =
new wxMenu();
849 amenu->Connect(wxEVT_COMMAND_MENU_SELECTED,
850 wxMenuEventHandler(GRIBUICtrlBar::OnMenuEvent),
nullptr,
this);
852 for (
int i = 0; i < 5; i++) {
860 amenu, ID_CTRLALTITUDE + 1000 + i,
869 amenu->Check(ID_CTRLALTITUDE + 1000 + m_Altitude,
true);
876void GRIBUICtrlBar::OnMove(wxMoveEvent &event) {
878 GetScreenPosition(&w, &h);
879 pPlugIn->SetCtrlBarXY(wxPoint(w, h));
882void GRIBUICtrlBar::OnMenuEvent(wxMenuEvent &event) {
883 int id =
event.GetId();
886 int alt = m_Altitude;
889 case ID_CTRLALTITUDE + 1000:
892 case ID_CTRLALTITUDE + 1001:
895 case ID_CTRLALTITUDE + 1002:
898 case ID_CTRLALTITUDE + 1003:
901 case ID_CTRLALTITUDE + 1004:
909 OnZoomToCenterClick(evt);
911 case ID_BTNSHOWCDATA:
912 OnShowCursorData(evt);
924 OnRequestForecastData(evt);
926 if (alt != m_Altitude) {
927 SetDialogsStyleSizePosition(
true);
932void GRIBUICtrlBar::MenuAppend(wxMenu *menu,
int id, wxString label,
933 wxItemKind kind, wxBitmap bitmap,
935 wxMenuItem *item =
new wxMenuItem(menu,
id, label, _T(
""), kind);
938 item->SetSubMenu(submenu);
948#if defined(__WXMSW__) || defined(__WXGTK__)
949 if (!bitmap.IsSameAs(wxNullBitmap)) item->SetBitmap(bitmap);
955void GRIBUICtrlBar::OnMouseEvent(wxMouseEvent &event) {
956 if (event.RightDown()) {
958 wxMenu *xmenu =
new wxMenu();
959 xmenu->Connect(wxEVT_COMMAND_MENU_SELECTED,
960 wxMenuEventHandler(GRIBUICtrlBar::OnMenuEvent),
nullptr,
964 wxMenu *smenu =
new wxMenu();
965 smenu->Connect(wxEVT_COMMAND_MENU_SELECTED,
966 wxMenuEventHandler(GRIBUICtrlBar::OnMenuEvent),
nullptr,
969 for (
int i = 0; i < 5; i++) {
977 smenu, ID_CTRLALTITUDE + 1000 + i,
984 smenu->Check(ID_CTRLALTITUDE + 1000 + m_Altitude,
true);
986 xmenu, wxID_ANY, _(
"Select geopotential altitude"), wxITEM_NORMAL,
987 GetScaledBitmap(wxBitmap(altitude), _T(
"altitude"), m_ScaledFactor),
990 MenuAppend(xmenu, ID_BTNNOW, _(
"Now"), wxITEM_NORMAL,
991 GetScaledBitmap(wxBitmap(now), _T(
"now"), m_ScaledFactor));
992 MenuAppend(xmenu, ID_BTNZOOMTC, _(
"Zoom To Center"), wxITEM_NORMAL,
993 GetScaledBitmap(wxBitmap(zoomto), _T(
"zoomto"), m_ScaledFactor));
995 xmenu, ID_BTNSHOWCDATA,
996 m_CDataIsShown ? _(
"Hide data at cursor") : _(
"Show data at cursor"),
998 GetScaledBitmap(wxBitmap(m_CDataIsShown ? curdata : ncurdata),
999 m_CDataIsShown ? _T(
"curdata") : _T(
"ncurdata"),
1003 m_tPlayStop.IsRunning() ? _(
"Stop play back") : _(
"Start play back"),
1005 GetScaledBitmap(wxBitmap(
m_tPlayStop.IsRunning() ? stop : play),
1006 m_tPlayStop.IsRunning() ? _T(
"stop") : _T(
"play"),
1009 xmenu, ID_BTNOPENFILE, _(
"Open a new file"), wxITEM_NORMAL,
1010 GetScaledBitmap(wxBitmap(openfile), _T(
"openfile"), m_ScaledFactor));
1012 xmenu, ID_BTNSETTING, _(
"Settings"), wxITEM_NORMAL,
1013 GetScaledBitmap(wxBitmap(setting), _T(
"setting"), m_ScaledFactor));
1019 requeststate1 ? _(
"Request forecast data")
1021 ? _(
"Draw requested Area or Click here to stop request")
1022 : _(
"Valid Area and Continue"),
1024 GetScaledBitmap(wxBitmap(requeststate1 ? request
1025 : requeststate3 ? selzone
1027 requeststate1 ? _T(
"request")
1028 : requeststate3 ? _T(
"selzone")
1029 : _T(
"request_end"),
1039 if (m_DialogStyle >> 1 == SEPARATED)
return;
1040 wxMouseEvent evt(event);
1043#ifndef __OCPN__ANDROID__
1044 if (m_gCursorData && m_CDataIsShown) {
1045 m_gCursorData->OnMouseEvent(evt);
1050void GRIBUICtrlBar::ContextMenuItemCallback(
int id) {
1052 bool dataisshown = m_CDataIsShown;
1053 m_CDataIsShown =
false;
1066 m_CDataIsShown = dataisshown;
1071 if (m_vpMouse == vp)
return;
1083void GRIBUICtrlBar::OnClose(wxCloseEvent &event) {
1085 if (m_gGRIBUICData) m_gGRIBUICData->Hide();
1088 pReq_Dialog->StopGraphicalZoneSelection();
1092 pPlugIn->SendTimelineMessage(wxInvalidDateTime);
1094 pPlugIn->OnGribCtrlBarClose();
1097void GRIBUICtrlBar::OnSize(wxSizeEvent &event) {
1099 wxSize p =
event.GetSize();
1105void GRIBUICtrlBar::OnPaint(wxPaintEvent &event) {
1106 wxWindowListNode *node = this->GetChildren().GetFirst();
1109 wxWindow *win = node->GetData();
1110 if (
dynamic_cast<wxBitmapButton *
>(win))
1111 dc.DrawBitmap(
dynamic_cast<wxBitmapButton *
>(win)->GetBitmap(), 5, 5,
1113 node = node->GetNext();
1117void GRIBUICtrlBar::createRequestDialog() {
1118 ::wxBeginBusyCursor();
1123 pPlugIn->SetDialogFont(pReq_Dialog);
1124 pPlugIn->SetDialogFont(pReq_Dialog->m_sScrolledDialog);
1126 pReq_Dialog->SetRequestDialogSize();
1127 if (::wxIsBusy()) ::wxEndBusyCursor();
1130void GRIBUICtrlBar::OnRequestForecastData(wxCommandEvent &event) {
1135 if (pReq_Dialog && pReq_Dialog->IsShown())
return;
1138 createRequestDialog();
1141 ::wxDisplaySize(&w,
nullptr);
1142 pReq_Dialog->Move((w - pReq_Dialog->GetSize().GetX()) / 2, 30);
1143 pReq_Dialog->Show();
1148void GRIBUICtrlBar::OnSettings(wxCommandEvent &event) {
1152 ::wxBeginBusyCursor();
1158 pPlugIn->SetDialogFont(dialog);
1159 for (
size_t i = 0; i < dialog->m_nSettingsBook->GetPageCount(); i++) {
1160 wxScrolledWindow *sc =
1161 ((wxScrolledWindow *)dialog->m_nSettingsBook->GetPage(i));
1165 dialog->m_nSettingsBook->ChangeSelection(dialog->GetPageIndex());
1166 dialog->SetSettingsDialogSize();
1169 ::wxDisplaySize(&w,
nullptr);
1170 dialog->Move((w - dialog->GetSize().GetX()) / 2, 30);
1173 ::wxEndBusyCursor();
1175 if (dialog->ShowModal() == wxID_OK) {
1176 dialog->WriteSettings();
1179 initSettings.Settings[GribOverlaySettings::WIND].m_Units &&
1181 GribOverlaySettings::BFS ||
1182 initSettings.Settings[GribOverlaySettings::WIND].m_Units ==
1183 GribOverlaySettings::BFS))
1185 STARTING_STATE_STYLE;
1189 m_DialogStyle = initSettings.m_iCtrlandDataStyle;
1191 ::wxBeginBusyCursor();
1193 dialog->SaveLastPage();
1195 m_InterpolateMode =
false;
1196 SetTimeLineMax(
true);
1197 SetFactoryOptions();
1199 SetDialogsStyleSizePosition(
true);
1205#ifdef __OCPN__ANDROID__
1206wxString callActivityMethod_ss(
const char *method, wxString parm);
1209void GRIBUICtrlBar::OnCompositeDialog(wxCommandEvent &event) {
1212 initSettings.Read();
1215 wxString json_begin = initSettings.SettingsToJSON(json);
1216 wxLogMessage(json_begin);
1225 double lon_min = wxRound(current_vp.
lon_min) - 1;
1226 double lon_max = wxRound(current_vp.
lon_max) + 1;
1227 double lat_min = wxRound(current_vp.
lat_min) - 1;
1228 double lat_max = wxRound(current_vp.
lat_max) + 1;
1232 int numErrors = reader.
Parse(json_begin, &v);
1233 if (numErrors > 0) {
1237 v[_T(
"latMin")] = lat_min;
1238 v[_T(
"latMax")] = lat_max;
1239 v[_T(
"lonMin")] = lon_min;
1240 v[_T(
"lonMax")] = lon_max;
1244 v[_T(
"grib_file")] = _T(
"");
1247 wxString json_final;
1248 w.
Write(v, json_final);
1249 wxLogMessage(json_final);
1251#ifdef __OCPN__ANDROID__
1252 wxString ret = callActivityMethod_ss(
"doGRIBActivity", json_final);
1259void GRIBUICtrlBar::OpenFileFromJSON(wxString json) {
1265 int numErrors = reader.
Parse(json, &root);
1266 if (numErrors > 0) {
1270 wxString file = root[(_T(
"grib_file"))].AsString();
1272 if (file.Length() && wxFileExists(file)) {
1273 wxFileName fn(file);
1281void GRIBUICtrlBar::OnPlayStop(wxCommandEvent &event) {
1285 m_bpPlay->SetBitmapLabel(
1286 GetScaledBitmap(wxBitmap(stop), _T(
"stop"), m_ScaledFactor));
1287 m_bpPlay->SetToolTip(_(
"Stop play back"));
1289 wxTIMER_CONTINUOUS);
1294void GRIBUICtrlBar::OnPlayStopTimer(wxTimerEvent &event) {
1295 if (m_sTimeline->GetValue() >= m_sTimeline->GetMax()) {
1298 ComputeBestForecastForNow();
1299 if (m_sTimeline->GetValue() >= m_sTimeline->GetMax())
1303 m_sTimeline->SetValue(0);
1310 ? GetNearestValue(GetNow(), 1)
1311 : GetNearestIndex(GetNow(), 2)
1312 : m_sTimeline->GetValue();
1313 m_sTimeline->SetValue(value + 1);
1317 if (!m_InterpolateMode)
1318 m_cRecordForecast->SetSelection(m_sTimeline->GetValue());
1322void GRIBUICtrlBar::StopPlayBack() {
1325 m_bpPlay->SetBitmapLabel(
1326 GetScaledBitmap(wxBitmap(play), _T(
"play"), m_ScaledFactor));
1327 m_bpPlay->SetToolTip(_(
"Start play back"));
1331void GRIBUICtrlBar::TimelineChanged() {
1333 pPlugIn->GetGRIBOverlayFactory()->SetGribTimelineRecordSet(
nullptr);
1337 RestaureSelectionString();
1343 if (!m_InterpolateMode) {
1346 GribRecordSet &sel = rsa->Item(m_cRecordForecast->GetCurrentSelection());
1349 ? wxTimeSpan(t - MinTime()).GetMinutes() /
1352 : m_cRecordForecast->GetCurrentSelection());
1354 m_cRecordForecast->SetSelection(GetNearestIndex(time, 2));
1355 SaveSelectionString();
1357 m_cRecordForecast->SetString(m_Selection_index,
1360 m_cRecordForecast->SetStringSelection(
1366 pPlugIn->SendTimelineMessage(time);
1370void GRIBUICtrlBar::RestaureSelectionString() {
1371 if (!m_SelectionIsSaved)
return;
1373 int sel = m_cRecordForecast->GetSelection();
1374 m_cRecordForecast->SetString(m_Selection_index, m_Selection_label);
1375 m_cRecordForecast->SetSelection(sel);
1376 m_SelectionIsSaved =
false;
1379int GRIBUICtrlBar::GetNearestIndex(wxDateTime time,
int model) {
1384 wxDateTime itime, ip1time;
1385 for (i = 0; i < rsa->GetCount() - 1; i++) {
1386 itime = rsa->Item(i).m_Reference_Time;
1387 ip1time = rsa->Item(i + 1).m_Reference_Time;
1388 if (ip1time >= time)
break;
1390 if (!model)
return (time - itime > (ip1time - time) * 3) ? i + 1 : i;
1392 return model == 1 ? time == ip1time ? i : i + 1 : time == ip1time ? i + 1 : i;
1395int GRIBUICtrlBar::GetNearestValue(wxDateTime time,
int model) {
1397 if (m_TimeLineHours == 0)
return 0;
1398 wxDateTime itime, ip1time;
1401 wxTimeSpan span = time - MinTime();
1402 int t = span.GetMinutes() / stepmin;
1404 wxTimeSpan(t * stepmin / 60, (t * stepmin) % 60);
1405 ip1time = itime + wxTimeSpan(stepmin / 60, stepmin % 60);
1407 if (model == 1)
return time == ip1time ? t + 1 : t;
1409 return (time - itime > (ip1time - time) * 3) ? t + 1 : t;
1412wxDateTime GRIBUICtrlBar::GetNow() {
1413 wxDateTime now = wxDateTime::Now();
1419 now = (now > rsa->Item(rsa->GetCount() - 1).m_Reference_Time)
1420 ? rsa->Item(rsa->GetCount() - 1).m_Reference_Time
1421 : (now < rsa->Item(0).m_Reference_Time) ? rsa->Item(0).m_Reference_Time
1427 if (m_InterpolateMode) {
1428 int tl = (m_TimeLineHours == 0) ? 0 : m_sTimeline->GetValue();
1431 return MinTime() + wxTimeSpan(tl * stepmin / 60, (tl * stepmin) % 60);
1435 unsigned int index = m_cRecordForecast->GetCurrentSelection() < 1
1437 : m_cRecordForecast->GetCurrentSelection();
1438 if (rsa && index < rsa->GetCount())
return rsa->Item(index).m_Reference_Time;
1440 return wxDateTime::Now();
1443wxDateTime GRIBUICtrlBar::MinTime() {
1445 if (rsa && rsa->GetCount()) {
1449 return wxDateTime::Now();
1456 if (rsa->GetCount() == 0)
return nullptr;
1463 wxDateTime GR1time, GR2time;
1469 for (j = 0; j < rsa->GetCount(); j++) {
1475 if (curtime <= time) GR1time = curtime, GRS1 = GRS, GR1 = GR;
1477 if (curtime >= time) {
1478 GR2time = curtime, GRS2 = GRS, GR2 = GR;
1483 if (!GR1 || !GR2)
continue;
1485 wxDateTime mintime = MinTime();
1486 double minute2 = (GR2time - mintime).GetMinutes();
1487 double minute1 = (GR1time - mintime).GetMinutes();
1488 double nminute = (time - mintime).GetMinutes();
1490 if (minute2 < minute1 || nminute < minute1 || nminute > minute2)
continue;
1492 double interp_const;
1493 if (minute1 == minute2) {
1498 interp_const = (nminute - minute1) / (minute2 - minute1);
1529 *GR1, *GR2, interp_const, i ==
Idx_WVDIR));
1540 auto sog = m_ProjectBoatPanel->GetSpeed();
1541 auto cog = m_ProjectBoatPanel->GetCourse();
1543 static_cast<double>(now.GetTicks() -
pPlugIn->m_boat_time) * sog / 3600.0;
1545 pPlugIn->m_boat_lon, cog, dist,
1546 &m_projected_lat, &m_projected_lon);
1554double GRIBUICtrlBar::getTimeInterpolatedValue(
int idx,
double lon,
double lat,
1559 if (rsa->GetCount() == 0)
return GRIB_NOTDEF;
1561 GribRecord *before =
nullptr, *after =
nullptr;
1564 time_t t = time.GetTicks();
1565 for (j = 0; j < rsa->GetCount(); j++) {
1570 time_t curtime = GR->getRecordCurrentDate();
1573 if (curtime < t) before = GR;
1581 if (!before || !after)
return GRIB_NOTDEF;
1583 time_t t1 = before->getRecordCurrentDate();
1584 time_t t2 = after->getRecordCurrentDate();
1588 double v2 = after->getInterpolatedValue(lon, lat);
1589 if (v1 != GRIB_NOTDEF && v2 != GRIB_NOTDEF) {
1590 double k = fabs((
double)(t - t1) / (t2 - t1));
1591 return (1.0 - k) * v1 + k * v2;
1597bool GRIBUICtrlBar::getTimeInterpolatedValues(
double &M,
double &A,
int idx1,
1598 int idx2,
double lon,
double lat,
1606 if (rsa->GetCount() == 0)
return false;
1608 GribRecord *beforeX =
nullptr, *afterX =
nullptr;
1609 GribRecord *beforeY =
nullptr, *afterY =
nullptr;
1612 time_t t = time.GetTicks();
1613 for (j = 0; j < rsa->GetCount(); j++) {
1617 if (!GX || !GY)
continue;
1619 time_t curtime = GX->getRecordCurrentDate();
1634 if (!beforeX || !afterX)
return false;
1636 time_t t1 = beforeX->getRecordCurrentDate();
1637 time_t t2 = afterX->getRecordCurrentDate();
1642 double v1m, v2m, v1a, v2a;
1651 if (v1m == GRIB_NOTDEF || v2m == GRIB_NOTDEF || v1a == GRIB_NOTDEF ||
1655 double k = fabs((
double)(t - t1) / (t2 - t1));
1656 M = (1.0 - k) * v1m + k * v2m;
1657 A = (1.0 - k) * v1a + k * v2a;
1661void GRIBUICtrlBar::OnTimeline(wxScrollEvent &event) {
1664 if (!m_InterpolateMode)
1665 m_cRecordForecast->SetSelection(m_sTimeline->GetValue());
1670void GRIBUICtrlBar::OnOpenFile(wxCommandEvent &event) {
1674#ifndef __OCPN__ANDROID__
1676 wxStandardPathsBase &path = wxStandardPaths::Get();
1677 wxString l_grib_dir = path.GetDocumentsDir();
1681 wxFileDialog *dialog =
1682 new wxFileDialog(
nullptr, _(
"Select a GRIB file"), l_grib_dir, _T(
""),
1684 "(*.grb;*.bz2;*.gz;*.grib2;*.grb2)|*.grb;*.bz2;*.gz;"
1685 "*.grib2;*.grb2|All files (*)|*.*"),
1686 wxFD_OPEN | wxFD_FILE_MUST_EXIST | wxFD_MULTIPLE,
1687 wxDefaultPosition, wxDefaultSize, _T(
"File Dialog"));
1689 if (dialog->ShowModal() == wxID_OK) {
1690 ::wxBeginBusyCursor();
1696 if (g_pi->m_bZoomToCenterAtInit) DoZoomToCenter();
1698 SetDialogsStyleSizePosition(
true);
1703 wxStandardPathsBase &path = wxStandardPaths::Get();
1709 nullptr, &file, _(
"Select a GRIB file"),
m_grib_dir, _T(
""), _T(
"*.*"));
1711 if (response == wxID_OK) {
1712 wxFileName fn(file);
1717 SetDialogsStyleSizePosition(
true);
1722void GRIBUICtrlBar::CreateActiveFileFromNames(
const wxArrayString &filenames) {
1723 if (filenames.GetCount() != 0) {
1730void GRIBUICtrlBar::PopulateComboDataList() {
1732 if (m_cRecordForecast->GetCount()) {
1733 index = m_cRecordForecast->GetCurrentSelection();
1734 m_cRecordForecast->Clear();
1738 for (
size_t i = 0; i < rsa->GetCount(); i++) {
1739 wxDateTime t(rsa->Item(i).m_Reference_Time);
1742 m_cRecordForecast->SetSelection(index);
1745void GRIBUICtrlBar::OnZoomToCenterClick(wxCommandEvent &event) {
1750 double latmin,latmax,lonmin,lonmax;
1751 if(!GetGribZoneLimits(
m_pTimelineSet, &latmin, &latmax, &lonmin, &lonmax ))
1757 double width = lonmax - lonmin;
1758 double height = latmax - latmin;
1761 double clat = latmin + height / 2;
1762 double clon = lonmin + width / 2;
1766 lonmin = clon - 60.;
1767 lonmax = clon + 60.;
1770 latmin = clat - 60.;
1771 latmax = clat + 60.;
1781 int w =
pPlugIn->GetGRIBOverlayFactory()->m_ParentSize.GetWidth();
1782 int h =
pPlugIn->GetGRIBOverlayFactory()->m_ParentSize.GetHeight();
1786 ppm = wxMin(w/(ow*1852), h/(oh*1852)) * ( 100 - fabs( clat ) ) / 90;
1788 ppm = wxMin(ppm, 1.0);
1796void GRIBUICtrlBar::DoZoomToCenter() {
1799 double latmin, latmax, lonmin, lonmax;
1800 if (!GetGribZoneLimits(
m_pTimelineSet, &latmin, &latmax, &lonmin, &lonmax))
1806 double width = lonmax - lonmin;
1807 double height = latmax - latmin;
1810 double clat = latmin + height / 2;
1811 double clon = lonmin + width / 2;
1815 lonmin = clon - 60.;
1816 lonmax = clon + 60.;
1818 if (height > 120.) {
1819 latmin = clat - 60.;
1820 latmax = clat + 60.;
1828 wxWindow *wx = GetGRIBCanvas();
1830 int w = wx->GetSize().x;
1831 int h = wx->GetSize().y;
1835 ppm = wxMin(w / (ow * 1852), h / (oh * 1852)) * (100 - fabs(clat)) / 90;
1837 ppm = wxMin(ppm, 1.0);
1842void GRIBUICtrlBar::OnPrev(wxCommandEvent &event) {
1846 RestaureSelectionString();
1850 selection = GetNearestIndex(GetNow(), 1);
1851 else if (m_InterpolateMode)
1855 selection = m_cRecordForecast->GetCurrentSelection();
1858 m_InterpolateMode =
false;
1860 m_cRecordForecast->SetSelection(selection < 1 ? 0 : selection - 1);
1865void GRIBUICtrlBar::OnNext(wxCommandEvent &event) {
1869 RestaureSelectionString();
1873 selection = GetNearestIndex(GetNow(), 2);
1874 else if (m_InterpolateMode)
1878 selection = m_cRecordForecast->GetCurrentSelection();
1880 m_cRecordForecast->SetSelection(selection);
1883 m_InterpolateMode =
false;
1885 if (selection == (
int)m_cRecordForecast->GetCount() - 1)
1888 m_cRecordForecast->SetSelection(selection + 1);
1893void GRIBUICtrlBar::ComputeBestForecastForNow() {
1895 pPlugIn->GetGRIBOverlayFactory()->SetGribTimelineRecordSet(
nullptr);
1899 wxDateTime now = GetNow();
1902 m_sTimeline->SetValue(GetNearestValue(now, 0));
1904 m_cRecordForecast->SetSelection(GetNearestIndex(now, 0));
1905 m_sTimeline->SetValue(m_cRecordForecast->GetCurrentSelection());
1908 if (
pPlugIn->GetStartOptions() !=
1915 m_InterpolateMode =
true;
1917 SetGribTimelineRecordSet(
1920 RestaureSelectionString();
1922 m_cRecordForecast->SetSelection(GetNearestIndex(now, 2));
1923 SaveSelectionString();
1925 m_cRecordForecast->SetString(m_Selection_index,
1928 m_cRecordForecast->SetStringSelection(nowTime);
1932 pPlugIn->SendTimelineMessage(now);
1936void GRIBUICtrlBar::SetGribTimelineRecordSet(
1941 if (!
pPlugIn->GetGRIBOverlayFactory())
return;
1946void GRIBUICtrlBar::SetTimeLineMax(
bool SetValue) {
1947 int oldmax = wxMax(m_sTimeline->GetMax(), 1),
1948 oldval = m_sTimeline->GetValue();
1953 m_sTimeline->SetMax(m_TimeLineHours * 60 / stepmin);
1957 m_sTimeline->SetMax(rsa->GetCount() - 1);
1962 if (SetValue && m_sTimeline->GetMax() != 0) {
1964 ComputeBestForecastForNow();
1966 m_sTimeline->SetValue(m_sTimeline->GetMax() * oldval / oldmax);
1970void GRIBUICtrlBar::SetFactoryOptions() {
1973 pPlugIn->GetGRIBOverlayFactory()->ClearCachedData();
1979void GRIBUICtrlBar::OnFormatRefreshTimer(wxTimerEvent &event) {
1982 wxDateTime referenceDate(1, wxDateTime::Jan, 2021, 12, 0, 0);
1985 if (currentFormat != m_sLastTimeFormat) {
1987 m_sLastTimeFormat = currentFormat;
1991 PopulateComboDataList();
2005unsigned int GRIBFile::ID = 0;
2011 m_pGribReader =
nullptr;
2012 m_last_message = wxEmptyString;
2013 for (
unsigned int i = 0; i < file_names.GetCount(); i++) {
2014 wxString file_name = file_names[i];
2015 if (::wxFileExists(file_name)) m_bOK =
true;
2018 if (m_bOK ==
false) {
2019 m_last_message = _(
" files don't exist!");
2028 for (
unsigned int i = 0; i < file_names.GetCount(); i++) {
2029 file_name = file_names[i];
2030 m_pGribReader->openFile(file_name);
2032 if (m_pGribReader->isOk()) {
2039 if (m_bOK ==
false) {
2040 m_last_message = _(
" can't be read!");
2045 m_FileNames.Clear();
2046 m_FileNames.Add(file_name);
2048 m_FileNames = file_names;
2052 m_pGribReader->computeAccumulationRecords(GRB_PRECIP_TOT, LV_GND_SURF, 0);
2053 m_pGribReader->computeAccumulationRecords(GRB_PRECIP_RATE, LV_GND_SURF, 0);
2054 m_pGribReader->computeAccumulationRecords(GRB_CLOUD_TOT, LV_ATMOS_ALL, 0);
2062 m_nGribRecords = m_pGribReader->getTotalNumberOfGribRecords();
2066 std::set<time_t>::iterator iter;
2067 std::set<time_t> date_list = m_pGribReader->getListDates();
2068 for (iter = date_list.begin(); iter != date_list.end(); iter++) {
2070 time_t reftime = *iter;
2072 m_GribRecordSetArray.Add(t);
2080 bool polarWind(
false);
2081 bool polarCurrent(
false);
2082 bool sigWave(
false);
2085 std::map<std::string, std::vector<GribRecord *> *> *p_map =
2086 m_pGribReader->getGribMap();
2089 std::map<std::string, std::vector<GribRecord *> *>::iterator it;
2090 for (it = p_map->begin(); it != p_map->end(); it++) {
2091 std::vector<GribRecord *> *ls = (*it).second;
2092 for (zuint i = 0; i < ls->size(); i++) {
2095 time_t thistime = pRec->getRecordCurrentDate();
2098 for (
unsigned int j = 0; j < m_GribRecordSetArray.GetCount(); j++) {
2099 if (m_GribRecordSetArray.Item(j).m_Reference_Time == thistime) {
2100 int idx = -1, mdx = -1;
2124 case GRB_WIND_SPEED:
2147 polarCurrent =
true;
2153 polarCurrent =
true;
2185 case GRB_PRECIP_RATE:
2186 case GRB_PRECIP_TOT:
2211 mdx = 1000 + NORWAY_METNO;
2241 case GRB_GEOPOT_HGT:
2267 if (m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[idx]) {
2270 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[idx];
2292 if (oRec->
getDataType() == GRB_HTSGW) skip =
true;
2302 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[idx] = pRec;
2303 if (m_GribIdxArray.Index(idx) == wxNOT_FOUND)
2304 m_GribIdxArray.Add(idx, 1);
2305 if (mdx != -1 && m_GribIdxArray.Index(mdx) == wxNOT_FOUND)
2306 m_GribIdxArray.Add(mdx, 1);
2314 if (polarWind || polarCurrent) {
2315 for (
unsigned int j = 0; j < m_GribRecordSetArray.GetCount(); j++) {
2316 for (
unsigned int i = 0; i <
Idx_COUNT; i++) {
2320 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[i];
2322 if (pRec !=
nullptr && pRec->
getDataType() == GRB_WIND_DIR) {
2344 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[idx];
2345 if (pRec1 !=
nullptr && pRec1->
getDataType() == GRB_WIND_SPEED)
2353 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[i];
2355 if (pRec !=
nullptr && pRec->
getDataType() == GRB_CUR_DIR) {
2365 m_GribRecordSetArray.Item(j).m_GribRecordPtrArray[idx];
2366 if (pRec1 !=
nullptr && pRec1->
getDataType() == GRB_CUR_SPEED)
2377 pRec->getRecordRefDate();
2380GRIBFile::~GRIBFile() {
delete m_pGribReader; }
2387 :
GRIBUICDataBase(parent.pParent, CURSOR_DATA, _(
"GRIB Display Control"),
2388 wxDefaultPosition, wxDefaultSize,
2389 wxSYSTEM_MENU | wxNO_BORDER | wxSTAY_ON_TOP)
2392 wxDefaultPosition, wxDefaultSize,
2393 wxSYSTEM_MENU | wxNO_BORDER)
2396 m_gpparent(parent) {
2400 m_gCursorData =
new CursorData(
this, m_gpparent);
2401 m_fgCdataSizer->Add(m_gCursorData, 0, wxALL, 0);
2403 Connect(wxEVT_MOVE, wxMoveEventHandler(GRIBUICData::OnMove));
2406void GRIBUICData::OnMove(wxMoveEvent &event) {
2408 GetScreenPosition(&w, &h);
2409 m_gpparent.
pPlugIn->SetCursorDataXY(wxPoint(w, h));
2415#ifdef __OCPN__ANDROID__
2417#include <QtAndroidExtras/QAndroidJniObject>
2419bool CheckPendingJNIException() {
2426 if (java_vm->GetEnv((
void **)&jenv, JNI_VERSION_1_6) != JNI_OK) {
2431 if ((jenv)->ExceptionCheck() == JNI_TRUE) {
2439wxString callActivityMethod_ss(
const char *method, wxString parm) {
2445 if (CheckPendingJNIException())
return _T(
"NOK");
2447 wxString return_string;
2448 QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod(
2449 "org/qtproject/qt5/android/QtNative",
"activity",
2450 "()Landroid/app/Activity;");
2451 if (CheckPendingJNIException())
return _T(
"NOK");
2453 if (!activity.isValid()) {
2455 return return_string;
2460 if (java_vm->GetEnv((
void **)&jenv, JNI_VERSION_1_6) != JNI_OK) {
2462 return "jenv Error";
2465 jstring p = (jenv)->NewStringUTF(parm.c_str());
2471 QAndroidJniObject data = activity.callObjectMethod(
2472 method,
"(Ljava/lang/String;)Ljava/lang/String;", p);
2473 if (CheckPendingJNIException())
return _T(
"NOK");
2477 jstring s = data.object<jstring>();
2479 if ((jenv)->GetStringLength(s)) {
2480 const char *ret_string = (jenv)->GetStringUTFChars(s,
nullptr);
2481 return_string = wxString(ret_string, wxConvUTF8);
2484 return return_string;
@ Idx_AIR_TEMP850
Air temperature at 850 hPa in Kelvin (K)
@ Idx_COMP_REFL
Composite radar reflectivity in dBZ (decibel relative to Z)
@ Idx_PRECIP_TOT
Precipitation data in millimeters per hour.
@ Idx_AIR_TEMP
Air temperature at 2m in Kelvin (K)
@ Idx_PRESSURE
Surface pressure in Pascal (Pa)
@ Idx_WVDIR
Wave direction.
@ Idx_HUMID_RE850
Relative humidity at 850 hPa in % (percent, range 0-100%)
@ Idx_CLOUD_TOT
Total cloud cover in % (percent, range 0-100%)
@ Idx_WIND_GUST
Wind gust speed at surface in m/s.
@ Idx_WIND_VX
Surface wind velocity X component in m/s.
@ Idx_AIR_TEMP300
Air temperature at 300 hPa in Kelvin (K)
@ Idx_COUNT
Number of supported GRIB record types.
@ Idx_WIND_VY850
Wind velocity Y component at 850 hPa in m/s.
@ Idx_HUMID_RE500
Relative humidity at 500 hPa in % (percent, range 0-100%)
@ Idx_WIND_VX300
Wind velocity X component at 300 hPa in m/s.
@ Idx_HUMID_RE300
Relative humidity at 300 hPa in % (percent, range 0-100%)
@ Idx_WIND_VX850
Wind velocity X component at 850 hPa in m/s.
@ Idx_WIND_VY300
Wind velocity Y component at 300 hPa in m/s.
@ Idx_WIND_VX700
Wind velocity X component at 700 hPa in m/s.
@ Idx_HTSIGW
Significant wave height in meters.
@ Idx_AIR_TEMP700
Air temperature at 700 hPa in Kelvin (K)
@ Idx_WIND_VY500
Wind velocity Y component at 500 hPa in m/s.
@ Idx_GEOP_HGT500
Geopotential height at 500 hPa in gpm (geopotential meters)
@ Idx_SEACURRENT_VY
Sea current velocity Y component in m/s.
@ Idx_WIND_VX500
Wind velocity X component at 500 hPa in m/s.
@ Idx_GEOP_HGT300
Geopotential height at 300 hPa in gpm (geopotential meters)
@ Idx_GEOP_HGT700
Geopotential height at 700 hPa in gpm (geopotential meters)
@ Idx_WIND_VY700
Wind velocity Y component at 700 hPa in m/s.
@ Idx_SEA_TEMP
Sea surface temperature in Kelvin (K)
@ Idx_WIND_VY
Surface wind velocity Y component in m/s.
@ Idx_GEOP_HGT850
Geopotential height at 850 hPa in gpm (geopotential meters)
@ Idx_SEACURRENT_VX
Sea current velocity X component in m/s.
@ Idx_AIR_TEMP500
Air temperature at 500 hPa in Kelvin (K)
@ Idx_HUMID_RE700
Relative humidity at 700 hPa in % (percent, range 0-100%)
@ Idx_CAPE
Convective Available Potential Energy in J/kg (Joules per kilogram)
int m_SavedZoneSelMode
Persisted version of the GRIB area selection mode.
int m_ZoneSelMode
Tracks the current state of GRIB area selection for zone coordinates.
GRIB Data Table View and Export Interface.
#define ID_BTNREQUEST
ID of button for requesting/downloading GRIB data./*#end#*/.
int m_SavedZoneSelMode
Persisted version of the GRIB area selection mode.
int m_ZoneSelMode
Tracks the current state of GRIB area selection for zone coordinates.
GRIB Weather Data Control Interface.
@ AUTO_SELECTION
Area automatically set from current viewport bounds.
@ DRAW_SELECTION
Manual mode has been selected.
@ START_SELECTION
User has clicked Shift + Left click and is drawing the bounding box by dragging the mouse.
@ SAVED_SELECTION
Area loaded from previously saved coordinates.
Tracks and displays GRIB meteorological data at cursor position.
Manages multiple GRIB record sets from one or more GRIB files.
GRIBFile(const wxArrayString &file_names, bool CumRec, bool WaveRec, bool newestFile=false)
Creates a new GRIBFile by parsing one or more GRIB files.
time_t GetRefDateTime(void)
Returns the reference datetime of the GRIB data, as the number of seconds since the epoch.
ArrayOfGribRecordSets * GetRecordSetArrayPtr(void)
Gets pointer to array of record sets organized by timestamp.
wxArrayString & GetFileNames(void)
Gets the list of source filenames being used.
bool IsOK(void)
Checks if file loading and parsing was successful.
Dialog showing GRIB data in a table format.
void SetTableSizePosition(int vpWidth, int vpHeight)
Set the table size and position relative to viewport.
void InitGribTable(ArrayOfGribRecordSets *rsa)
Initialize the GRIB data table.
void SetViewPortWithFocus(PlugIn_ViewPort *vp)
Set the ViewPort that has the focus.
void GetProjectedLatLon(int &x, int &y, PlugIn_ViewPort *vp)
Gets the projected position of vessel based on current course, speed and forecast time.
GribOverlaySettings m_OverlaySettings
Settings that control how GRIB data is displayed and overlaid.
void SetRequestButtonBitmap(int type)
Set the icon and tooltip for the download request button.
wxTimer m_tPlayStop
Timer for controlling GRIB animation playback.
void SetViewPortUnderMouse(PlugIn_ViewPort *vp)
Set the ViewPort under the mouse.
wxString m_grib_dir
Directory containing GRIB files.
grib_pi * pPlugIn
Plugin instance that owns this control bar.
GRIBFile * m_bGRIBActiveFile
Currently active GRIB file being displayed.
wxDateTime TimelineTime()
Returns the selected time in the GRIB timeline widget.
void UpdateTrackingControl()
Schedules an update of the GRIB data values display at current cursor position.
GribTimelineRecordSet * m_pTimelineSet
Current set of GRIB records for timeline playback.
wxArrayString m_file_names
List of GRIB filenames being displayed.
GribTimelineRecordSet * GetTimeLineRecordSet(wxDateTime time)
Retrieves or creates a temporally interpolated GRIB record set for a specific timestamp.
void copyMissingWaveRecords()
Fills gaps in wave-related data fields by propagating known values across missing time periods.
void copyFirstCumulativeRecord()
Initializes cumulative meteorological parameters by copying their first record values.
Manages a collection of GribRecord objects representing multiple meteorological parameters at a singl...
void SetUnRefGribRecord(int i, GribRecord *pGR)
Sets a GRIB record that this set owns and will be responsible for deleting.
GribRecord * m_GribRecordPtrArray[Idx_COUNT]
Array of pointers to GRIB records representing different meteorological parameters.
time_t m_Reference_Time
Reference time for this set of records, as the number of seconds since the epoch.
Represents a meteorological data grid from a GRIB (Gridded Binary) file.
zuchar getTimeRange() const
Returns the time range indicator that defines how P1 and P2 should be interpreted.
static void Polar2UV(GribRecord *pDIR, GribRecord *pSPEED)
Converts wind or current values from polar (direction/speed) to cartesian (U/V) components.
static GribRecord * Interpolated2DRecord(GribRecord *&rety, const GribRecord &rec1x, const GribRecord &rec1y, const GribRecord &rec2x, const GribRecord &rec2y, double d)
Creates temporally interpolated records for vector fields (wind, currents).
zuint getDataCenterModel() const
Returns the numerical weather prediction model/center that produced this data.
double getInterpolatedValue(double px, double py, bool numericalInterpolation=true, bool dir=false) const
Get spatially interpolated value at exact lat/lon position.
static GribRecord * InterpolatedRecord(const GribRecord &rec1, const GribRecord &rec2, double d, bool dir=false)
Creates a new GribRecord by temporally interpolating between two time points.
static bool getInterpolatedValues(double &M, double &A, const GribRecord *GRX, const GribRecord *GRY, double px, double py, bool numericalInterpolation=true)
Gets spatially interpolated wind or current vector values at a specific latitude/longitude point.
zuint getLevelValue() const
Returns the numeric value associated with the level type.
zuchar getLevelType() const
Returns the type of vertical level for this grid's data.
zuchar getDataType() const
Returns the type of meteorological parameter stored in this grid.
Manages GRIB file request configuration and downloads.
void OnVpUnderMouseChange(PlugIn_ViewPort *vp)
Callback invoked when the view port under mouse has changed.
void OnVpWithFocusChange(PlugIn_ViewPort *vp)
Callback invoked when the focused view port has changed, such as in multi-chart mode when user switch...
A specialized GribRecordSet that represents temporally interpolated weather data with isobar renderin...
wxArrayPtrVoid * m_IsobarArray[Idx_COUNT]
Array of cached isobar calculations for each data type (wind, pressure, etc).
GribTimelineRecordSet(unsigned int cnt)
Creates a timeline record set containing temporally interpolated GRIB records.
Contains view parameters and status information for a chart display viewport.
int pix_width
Viewport width in pixels.
double lon_max
Maximum longitude of the viewport.
double lat_max
Maximum latitude of the viewport.
int pix_height
Viewport height in pixels.
double lon_min
Minimum longitude of the viewport.
double lat_min
Minimum latitude of the viewport.
bool GetCopyFirstCumRec()
Returns true if cumulative parameters like precipitation and cloud cover should initialize their star...
bool GetCopyMissWaveRec()
Returns true if wave data should be propagated across time periods where wave records are missing.
int Parse(const wxString &doc, wxJSONValue *val)
Parse the JSON document.
The JSON value class implementation.
The JSON document writer.
void Write(const wxJSONValue &value, wxString &str)
Write the JSONvalue object to a JSON text.
Email Request System for GRIB Data.
GRIB Weather Data Plugin for OpenCPN.
PlugIn Object Definition/API.
wxWindow * GetOCPNCanvasWindow()
Gets OpenCPN's main canvas window.
void SetCanvasContextMenuItemViz(int item, bool viz)
Temporarily changes context menu item visibility.
int GetCanvasCount()
Gets total number of chart canvases.
wxFileConfig * GetOCPNConfigObject()
Gets OpenCPN's configuration object.
wxFont * FindOrCreateFont_PlugIn(int point_size, wxFontFamily family, wxFontStyle style, wxFontWeight weight, bool underline, const wxString &facename, wxFontEncoding encoding)
Creates or finds a font in the font cache.
void PositionBearingDistanceMercator_Plugin(double lat, double lon, double brg, double dist, double *dlat, double *dlon)
Calculates destination point given starting point, bearing and distance.
void JumpToPosition(double lat, double lon, double scale)
Centers chart display on specified position at given scale.
int PlatformFileSelectorDialog(wxWindow *parent, wxString *file_spec, wxString Title, wxString initDir, wxString suggestedName, wxString wildcard)
Shows platform-optimized file selector dialog.
wxRect GetMasterToolbarRect()
Gets bounding rectangle of master toolbar.
void CanvasJumpToPosition(wxWindow *canvas, double lat, double lon, double scale)
Centers specified canvas on given position at given scale.
void GetCanvasPixLL(PlugIn_ViewPort *vp, wxPoint *pp, double lat, double lon)
Converts lat/lon to canvas physical pixel coordinates.
void DimeWindow(wxWindow *win)
Applies system color scheme to window.
wxWindow * GetCanvasByIndex(int canvasIndex)
Gets chart canvas window by index.
void RequestRefresh(wxWindow *win)
Requests window refresh.
void DistanceBearingMercator_Plugin(double lat0, double lon0, double lat1, double lon1, double *brg, double *dist)
Calculates bearing and distance between two points using Mercator projection.
OpenGL Platform Abstraction Layer.
wxString toUsrDateTimeFormat_Plugin(const wxDateTime date_time, const DateTimeFormatOptions &options)
Format a date/time to a localized string representation, conforming to the global date/time format an...