33#include <wx/wfstream.h>
36#include <unordered_map>
40enum { SAILDOCS, ZYGRIB };
41enum { GFS, COAMPS, RTOFS, HRRR, ICON, ECMWF };
43wxString toMailFormat(
int NEflag,
46 char c = NEflag == 1 ? a < 0 ?
'S' :
'N' : a < 0 ?
'W' :
'E';
48 s.Printf(_T (
"%01d%c" ), abs(a), c);
63 m_boundingBoxCanvasIndex = -1;
66#if defined(__WXOSX__) || defined(__WXGTK3__)
68 m_displayScale = GetContentScaleFactor();
73 m_downloading =
false;
74 m_bTransferSuccess =
true;
77 auto bg = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW)
78 .GetAsString(wxC2S_HTML_SYNTAX);
79 auto fg = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT)
80 .GetAsString(wxC2S_HTML_SYNTAX);
81 m_htmlWinWorld->SetBorders(10);
82 m_htmlWinWorld->SetPage(
83 "<html><body bgcolor="
92 _(
"<h1>OpenCPN ECMWF forecast</h1>"
93 "<p>Free service based on ECMWF Open Data published under the terms of "
94 "Creative Commons CC-4.0-BY license</p>"
95 "<p>The IFS model GRIB files include information about surface "
97 "atmospheric pressure, wind strength, wind direction, wave height and "
98 "direction for the whole world on a 0.25 degree resolution "
100 "step in the first 144 hours and 6 hour step up to 10 days.</p>"
101 "The AIFS model contains data for wind, pressure and temperature on a "
102 "0.25 degree grid with 6 hour step for up to 15 days"
103 "<p>The data is updated twice a day as soon as the 00z and 12z model "
104 "runs finish and the "
105 "results are published by ECMWF, which usually means new forecast data "
106 "is available shortly after 8AM and 8PM UTC.</p>"
107 "<p>The grib downloaded covers the area of the primary chart "
109 "<p>The service is provided on best effort basis and comes with no "
110 "guarantees. The server is hosted by a volunteer and the service is "
111 "provided free of charge and without accepting any liability "
112 "whatsoever for its continuous availability, or for any loss or damage "
113 "arising from its use. If you find the service useful, please "
114 "consider making a donation to the OpenCPN project.</p>"
115 "<p>This service is based on data and products of the European Centre "
116 "for Medium-Range Weather Forecasts (ECMWF).</p>"
117 "<p>Source: www.ecmwf.int</p>"
118 "<p>Disclaimer: ECMWF does not accept any liability whatsoever for any "
119 "error or omission in the data, their availability, or for any loss or "
120 "damage arising from their use.</p>"
121 "</font></body></html>"));
122 m_htmlInfoWin->SetBorders(10);
123 m_htmlInfoWin->SetPage(
124 "<html><body bgcolor="
133 _(
"<h1>Grib weather forecasts</h1>"
134 "<p>Collection of local weather models from various sources available "
135 "for download over the internet.</p>"
136 "</font></body></html>"));
138 m_bLocal_source_selected =
false;
139 EnableDownloadButtons();
141 m_selectedAtmModelIndex = 0;
142 m_selectedWaveModelIndex = 0;
143 InitializeXygribDialog();
146GribRequestSetting::~GribRequestSetting() {
148 OCPN_cancelDownloadFileBackground(m_download_handle);
152 wxEVT_DOWNLOAD_EVENT,
153 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
162void GribRequestSetting::SaveConfig() {
165 pConf->SetPath(_T(
"/PlugIns/GRIB" ));
167 pConf->Write(_T (
"MailRequestConfig" ), m_RequestConfigBase);
168 pConf->Write(_T(
"MailSenderAddress" ), m_pSenderAddress->GetValue());
169 pConf->Write(_T(
"MailRequestAddresses" ), m_MailToAddresses);
170 pConf->Write(_T(
"ZyGribLogin" ), m_pLogin->GetValue());
171 pConf->Write(_T(
"ZyGribCode" ), m_pCode->GetValue());
172 pConf->Write(_T(
"SendMailMethod" ), m_SendMethod);
173 pConf->Write(_T(
"MovingGribSpeed" ), m_sMovingSpeed->GetValue());
174 pConf->Write(_T(
"MovingGribCourse" ), m_sMovingCourse->GetValue());
179 pConf->Write(_T(
"ManualRequestZoneSizingMode" ),
m_ZoneSelMode);
182 pConf->Write(_T(
"RequestZoneMaxLat" ),
m_spMaxLat->GetValue());
183 pConf->Write(_T(
"RequestZoneMinLat" ),
m_spMinLat->GetValue());
184 pConf->Write(_T(
"RequestZoneMaxLon" ),
m_spMaxLon->GetValue());
185 pConf->Write(_T(
"RequestZoneMinLon" ),
m_spMinLon->GetValue());
189void GribRequestSetting::InitRequestConfig() {
193 pConf->SetPath(_T(
"/PlugIns/GRIB" ));
196 pConf->Read(_T(
"MailRequestConfig" ), &m_RequestConfigBase,
197 _T(
"000220XX........0" ));
198 pConf->Read(_T(
"MailSenderAddress" ), &l, _T(
""));
199 m_pSenderAddress->ChangeValue(l);
200 pConf->Read(_T(
"MailRequestAddresses" ), &m_MailToAddresses,
201 _T(
"query@saildocs.com;gribauto@zygrib.org"));
202 pConf->Read(_T(
"ZyGribLogin" ), &l, _T(
""));
203 m_pLogin->ChangeValue(l);
204 pConf->Read(_T(
"ZyGribCode" ), &l, _T(
""));
205 m_pCode->ChangeValue(l);
206 pConf->Read(_T(
"SendMailMethod" ), &m_SendMethod, 0);
207 pConf->Read(_T(
"MovingGribSpeed" ), &m, 0);
208 m_sMovingSpeed->SetValue(m);
209 pConf->Read(_T(
"MovingGribCourse" ), &m, 0);
210 m_sMovingCourse->SetValue(m);
211 pConf->Read(_T(
"ManualRequestZoneSizingMode" ), &m, 0);
213 pConf->Read(_T(
"ManualRequestZoneSizing" ), &m, 0);
220 UpdateAreaSelectionState();
221 pConf->Read(_T(
"RequestZoneMaxLat" ), &m, 0);
223 pConf->Read(_T(
"RequestZoneMinLat" ), &m, 0);
225 pConf->Read(_T(
"RequestZoneMaxLon" ), &m, 0);
227 pConf->Read(_T(
"RequestZoneMinLon" ), &m, 0);
230 SetCoordinatesText();
233 if (m_RequestConfigBase.Len() !=
234 wxString(_T(
"000220XX.............." )).Len())
235 m_RequestConfigBase = _T(
"000220XX.............." );
238 wxString s1[] = {_T(
"GFS"), _T(
"COAMPS"), _T(
"RTOFS"),
239 _T(
"HRRR"), _T(
"ICON"), _T(
"ECMWF")};
240 for (
unsigned int i = 0; i < (
sizeof(s1) /
sizeof(wxString)); i++)
241 m_pModel->Append(s1[i]);
242 wxString s2[] = {_T(
"Saildocs"), _T(
"zyGrib")};
243 for (
unsigned int i = 0; i < (
sizeof(s2) /
sizeof(wxString)); i++)
244 m_pMailTo->Append(s2[i]);
245 wxString s3[] = {_T(
"WW3-GLOBAL"), _T(
"WW3-MEDIT")};
246 for (
unsigned int i = 0; i < (
sizeof(s3) /
sizeof(wxString)); i++)
247 m_pWModel->Append(s3[i]);
250 m_tResUnit->SetLabel(wxString::Format(_T(
"\u00B0")));
251 m_sCourseUnit->SetLabel(wxString::Format(_T(
"\u00B0")));
255 GetTextExtent(_T(
"-360"), &w, &h, 0, 0,
259 m_sMovingSpeed->SetMinSize(wxSize(w, h));
260 m_sMovingCourse->SetMinSize(wxSize(w, h));
267 m_pSenderAddress->SetToolTip(
268 _(
"Address used to send request eMail. (Mandatory for LINUX)"));
269 m_pLogin->SetToolTip(_(
"This is your zyGrib's forum access Login"));
271 _(
"Get this Code in zyGrib's forum ( This is not your password! )"));
272 m_sMovingSpeed->SetToolTip(_(
"Enter your forescasted Speed (in Knots)"));
273 m_sMovingCourse->SetToolTip(_(
"Enter your forecasted Course"));
276 ((wxString)m_RequestConfigBase.GetChar(0)).ToLong(&i);
277 m_pMailTo->SetSelection(i);
278 ((wxString)m_RequestConfigBase.GetChar(1)).ToLong(&i);
279 m_pModel->SetSelection(i);
280 m_cMovingGribEnabled->SetValue(m_RequestConfigBase.GetChar(16) ==
282 ((wxString)m_RequestConfigBase.GetChar(2)).ToLong(&i);
283 ((wxString)m_RequestConfigBase.GetChar(3)).ToLong(&j);
284 ((wxString)m_RequestConfigBase.GetChar(4)).ToLong(&k, 16);
288 m_pSenderSizer->ShowItems(
false);
290 if (m_SendMethod == 0)
291 m_pSenderSizer->ShowItems(
false);
293 m_pSenderSizer->ShowItems(
297 m_tMouseEventTimer.Connect(
298 wxEVT_TIMER, wxTimerEventHandler(GribRequestSetting::OnMouseEventTimer),
303 ApplyRequestConfig(i, j, k);
305 ((wxString)m_RequestConfigBase.GetChar(5)).ToLong(&j);
306 m_pWModel->SetSelection(j);
308 m_pWind->Enable(
false);
309 m_pPress->Enable(
false);
314 m_MailImage->SetValue(WriteMail());
317wxWindow *GetGRIBCanvas();
318void GribRequestSetting::OnClose(wxCloseEvent &event) {
320 OCPN_cancelDownloadFileBackground(m_download_handle);
321 m_downloading =
false;
322 m_download_handle = 0;
326 wxEVT_DOWNLOAD_EVENT,
327 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
336 m_parent.m_highlight_latmax = 0;
337 m_parent.m_highlight_lonmax = 0;
338 m_parent.m_highlight_latmin = 0;
339 m_parent.m_highlight_lonmin = 0;
343void GribRequestSetting::SetRequestDialogSize() {
346 GetTextExtent(_T(
"abc"),
nullptr, &y, 0, 0,
OCPNGetFont(_(
"Dialog")));
347 m_MailImage->SetMinSize(
348 wxSize(-1, ((y * m_MailImage->GetNumberOfLines()) + 10)));
352 m_fgScrollSizer->Fit(m_sScrolledDialog);
355 SetMinSize(wxSize(0, 0));
358 wxWindow *frame = wxTheApp->GetTopWindow();
360 int w = frame->GetClientSize().x;
361 int h = frame->GetClientSize().y;
367 m_sScrolledDialog->SetMinSize(
368 wxSize(wxMin(w, scroll.x),
369 wxMin(h, scroll.y)));
374 wxSize sd = GetSize();
375 if (sd.y == GetClientSize().y) sd.y += 30;
376 SetSize(wxSize(sd.x, sd.y));
377 SetMinSize(wxSize(sd.x, sd.y));
385 if ((fabs(vp->
lat_max) < 90.) && (fabs(lonmax) < 360.)) {
386 if (lonmax < -180.) lonmax += 360.;
387 if (lonmax > 180.) lonmax -= 360.;
389 if ((fabs(vp->
lat_min) < 90.) && (fabs(lonmin) < 360.)) {
390 if (lonmin < -180.) lonmin += 360.;
391 if (lonmin > 180.) lonmin -= 360.;
394 bool bnew_val =
false;
396 if (
m_spMinLon->GetValue() != (
int)floor(lonmin)) bnew_val =
true;
398 if (
m_spMaxLon->GetValue() != (
int)ceil(lonmax)) bnew_val =
true;
406 SetCoordinatesText();
407 m_MailImage->SetValue(WriteMail());
412 if (!(event.ShiftDown() || m_bpManualSelection->GetValue())) {
417 if (event.LeftDown() && !event.Dragging()) {
432 int xm =
event.GetX() * m_displayScale;
433 int ym =
event.GetY() * m_displayScale;
435 if (event.LeftDown() && !event.Dragging()) {
436 m_parent.pParent->SetFocus();
449 SetCoordinatesText();
450 UpdateAreaSelectionState();
451 m_MailImage->SetValue(WriteMail());
454 UpdateGribSizeEstimate();
455 m_boundingBoxCanvasIndex = -1;
456 }
else if (event.Dragging()) {
471 if (!m_tMouseEventTimer.IsRunning())
472 m_tMouseEventTimer.Start(20, wxTIMER_ONE_SHOT);
477void GribRequestSetting::OnMouseEventTimer(wxTimerEvent &event) {
499void GribRequestSetting::SetCoordinatesText() {
500 m_stMaxLatNS->SetLabel(
m_spMaxLat->GetValue() < 0 ? _(
"S") : _(
"N"));
501 m_stMinLonEW->SetLabel(
m_spMinLon->GetValue() < 0 ? _(
"W") : _(
"E"));
502 m_stMaxLonEW->SetLabel(
m_spMaxLon->GetValue() < 0 ? _(
"W") : _(
"E"));
503 m_stMinLatNS->SetLabel(
m_spMinLat->GetValue() < 0 ? _(
"S") : _(
"N"));
506size_t LengthSelToHours(
int sel) {
520 switch (ev.getDLEventCondition()) {
524 Disconnect(wxEVT_DOWNLOAD_EVENT,
528 m_downloading =
false;
529 m_download_handle = 0;
534 if (ev.getTotal() != 0) {
535 switch (m_downloadType) {
537 m_staticTextInfo->SetLabelText(
538 wxString::Format(_(
"Downloading... %li / %li"),
539 ev.getTransferred(), ev.getTotal()));
543 m_stLocalDownloadInfo->SetLabelText(
544 wxString::Format(_(
"Downloading... %li / %li"),
545 ev.getTransferred(), ev.getTotal()));
549 m_xygribPanel->m_progress_gauge->SetValue(
550 100 * ev.getTransferred() / ev.getTotal());
552 m_xygribPanel->m_status_text->SetLabel(wxString::Format(
553 "%s (%ld kB / %ld kB)",
554 _(
"Downloading GRIB file").c_str().AsChar(),
555 ev.getTransferred() / 1024, ev.getTotal() / 1024));
561 if (ev.getTransferred() > 0) {
562 switch (m_downloadType) {
564 m_staticTextInfo->SetLabelText(wxString::Format(
565 _(
"Downloading... %li / ???"), ev.getTransferred()));
569 m_stLocalDownloadInfo->SetLabelText(wxString::Format(
570 _(
"Downloading... %li / ???"), ev.getTransferred()));
584void GribRequestSetting::OnWorldDownload(wxCommandEvent &event) {
586 OCPN_cancelDownloadFileBackground(m_download_handle);
587 m_downloading =
false;
588 m_download_handle = 0;
590 wxEVT_DOWNLOAD_EVENT,
591 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
593 m_btnDownloadWorld->SetLabelText(_(
"Download"));
594 m_staticTextInfo->SetLabelText(_(
"Download canceled"));
597 EnableDownloadButtons();
598 wxTheApp->ProcessPendingEvents();
603 m_downloading =
true;
605 EnableDownloadButtons();
606 m_btnDownloadWorld->SetLabelText(_(
"Cancel"));
607 m_staticTextInfo->SetLabelText(_(
"Preparing data on server..."));
610 switch (m_chECMWFResolution->GetSelection()) {
615 model =
"ecmwfaifs0p25";
621 std::ostringstream oss;
622 oss <<
"https://grib.bosun.io/grib?";
623 oss <<
"model=" << model;
628 oss <<
"&length=" << LengthSelToHours(m_chForecastLength->GetSelection());
630 wxString::Format(
"ocpn_%s_%li_%s.grb2", model.c_str(),
631 LengthSelToHours(m_chForecastLength->GetSelection()),
632 wxDateTime::Now().Format(
"%F-%H-%M"));
633 wxString path = m_parent.GetGribDir();
634 path.Append(wxFileName::GetPathSeparator());
635 path.Append(filename);
639 wxEVT_DOWNLOAD_EVENT,
640 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
643 OCPN_downloadFileBackground(oss.str(), path,
this, &m_download_handle);
644 while (m_downloading) {
645 wxTheApp->ProcessPendingEvents();
649 if (m_bTransferSuccess) {
650 m_staticTextInfo->SetLabelText(
651 wxString::Format(_(
"Download complete: %s"), path.c_str()));
658 if (m_parent.
pPlugIn->m_bZoomToCenterAtInit) m_parent.DoZoomToCenter();
660 m_parent.SetDialogsStyleSizePosition(
true);
664 m_staticTextInfo->SetLabelText(_(
"Download failed"));
667 m_btnDownloadWorld->SetLabelText(_(
"Download"));
669 EnableDownloadButtons();
672enum LocalSourceItem { SOURCE, AREA, GRIB };
674enum LocalGribDownloadType { DIRECT, MANIFEST, WEBPAGE };
677 GribCatalogInfo(LocalSourceItem type, wxString name, wxString description,
678 wxString url, wxString filename,
679 LocalGribDownloadType download_type,
double latmin,
680 double lonmin,
double latmax,
double lonmax)
683 description(description),
686 download_type(download_type),
691 LocalSourceItem type;
693 wxString description;
696 LocalGribDownloadType download_type;
703void GribRequestSetting::FillTreeCtrl(
wxJSONValue &data) {
704 m_SourcesTreeCtrl1->DeleteAllItems();
706 m_SourcesTreeCtrl1->AddRoot(_(
"Local high resolution forecasts"));
707 if (data.HasMember(
"sources") && data[
"sources"].IsArray()) {
708 for (
int i = 0; i < data[
"sources"].Size(); i++) {
711 LocalSourceItem::SOURCE, source[
"source"].AsString(),
712 source[
"description"].AsString(), source[
"url"].AsString(),
713 wxEmptyString, LocalGribDownloadType::WEBPAGE, 0, 0, 0, 0);
714 wxTreeItemId src_id = m_SourcesTreeCtrl1->AppendItem(
715 root, source[
"source"].AsString(), -1, -1, info);
716 if (source.
HasMember(
"areas") && source[
"areas"].IsArray()) {
717 for (
int j = 0; j < source[_T(
"areas")].
Size(); j++) {
720 LocalSourceItem::AREA, area[
"name"].AsString(),
721 source[
"description"].AsString(), source[
"url"].AsString(),
722 wxEmptyString, LocalGribDownloadType::WEBPAGE,
723 area[
"boundary"][
"lat_min"].AsDouble(),
724 area[
"boundary"][
"lon_min"].AsDouble(),
725 area[
"boundary"][
"lat_max"].AsDouble(),
726 area[
"boundary"][
"lon_max"].AsDouble());
727 m_SourcesTreeCtrl1->AppendItem(src_id, area[
"name"].AsString(), -1,
729 if (area.
HasMember(
"gribs") && area[
"gribs"].IsArray()) {
730 for (
int k = 0; k < area[
"gribs"].
Size(); k++) {
733 LocalSourceItem::GRIB, grib[
"name"].AsString(),
734 source[
"description"].AsString(),
735 grib.
HasMember(
"url") ? grib[
"url"].AsString()
736 : grib[
"cat_url"].AsString(),
737 grib.HasMember(
"filename") ? grib[
"filename"].AsString() :
"",
738 grib.HasMember(
"url") ? LocalGribDownloadType::DIRECT
739 : LocalGribDownloadType::MANIFEST,
740 area[
"boundary"][
"lat_min"].AsDouble(),
741 area[
"boundary"][
"lon_min"].AsDouble(),
742 area[
"boundary"][
"lat_max"].AsDouble(),
743 area[
"boundary"][
"lon_max"].AsDouble());
744 m_SourcesTreeCtrl1->AppendItem(
745 m_SourcesTreeCtrl1->GetLastChild(src_id),
746 grib[_T(
"name")].AsString(), -1, -1, info);
751 m_SourcesTreeCtrl1->CollapseAllChildren(src_id);
754 m_SourcesTreeCtrl1->Expand(root);
757void GribRequestSetting::ReadLocalCatalog() {
759 wxFileInputStream str(m_parent.
pPlugIn->m_local_sources_catalog);
761 reader.
Parse(str, &root);
765void GribRequestSetting::HighlightArea(
double latmax,
double lonmax,
766 double latmin,
double lonmin) {
767 m_parent.m_highlight_latmax = latmax;
768 m_parent.m_highlight_lonmax = lonmax;
769 m_parent.m_highlight_latmin = latmin;
770 m_parent.m_highlight_lonmin = lonmin;
773void GribRequestSetting::OnLocalTreeSelChanged(wxTreeEvent &event) {
774 wxTreeItemId item = m_SourcesTreeCtrl1->GetSelection();
777 if (src->type == LocalSourceItem::GRIB) {
778 m_stLocalDownloadInfo->SetLabelText(_(
"Download grib..."));
779 m_bLocal_source_selected =
true;
780 HighlightArea(src->latmax, src->lonmax, src->latmin, src->lonmin);
782 m_stLocalDownloadInfo->SetLabelText(_(
"Select grib..."));
783 m_bLocal_source_selected =
false;
784 HighlightArea(src->latmax, src->lonmax, src->latmin, src->lonmin);
787 EnableDownloadButtons();
790void GribRequestSetting::OnUpdateLocalCatalog(wxCommandEvent &event) {
792 OCPN_cancelDownloadFileBackground(m_download_handle);
793 m_downloading =
false;
794 m_download_handle = 0;
796 wxEVT_DOWNLOAD_EVENT,
797 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
799 m_btnDownloadLocal->SetLabelText(_(
"Download"));
800 m_stLocalDownloadInfo->SetLabelText(_(
"Download canceled"));
803 EnableDownloadButtons();
804 wxTheApp->ProcessPendingEvents();
809 m_downloading =
true;
811 EnableDownloadButtons();
812 m_btnDownloadLocal->SetLabelText(_(
"Cancel"));
813 m_staticTextInfo->SetLabelText(_(
"Downloading catalog update..."));
818 wxEVT_DOWNLOAD_EVENT,
819 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
821 auto res = OCPN_downloadFileBackground(
822 CATALOG_URL, m_parent.
pPlugIn->m_local_sources_catalog +
"new",
this,
824 while (m_downloading) {
825 wxTheApp->ProcessPendingEvents();
829 if (m_bTransferSuccess) {
830 wxRenameFile(m_parent.
pPlugIn->m_local_sources_catalog +
"new",
831 m_parent.
pPlugIn->m_local_sources_catalog,
true);
833 m_stLocalDownloadInfo->SetLabelText(_(
"Catalog update complete."));
835 m_stLocalDownloadInfo->SetLabelText(_(
"Download failed"));
838 m_btnDownloadLocal->SetLabelText(_(
"Download"));
840 EnableDownloadButtons();
843void GribRequestSetting::OnDownloadLocal(wxCommandEvent &event) {
845 OCPN_cancelDownloadFileBackground(m_download_handle);
846 m_downloading =
false;
847 m_download_handle = 0;
849 wxEVT_DOWNLOAD_EVENT,
850 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
852 m_btnDownloadLocal->SetLabelText(_(
"Download"));
853 m_stLocalDownloadInfo->SetLabelText(_(
"Download canceled"));
856 EnableDownloadButtons();
857 wxTheApp->ProcessPendingEvents();
862 m_downloading =
true;
864 EnableDownloadButtons();
865 m_btnDownloadLocal->SetLabelText(_(
"Cancel"));
866 m_staticTextInfo->SetLabelText(_(
"Downloading grib..."));
869 m_SourcesTreeCtrl1->GetSelection()));
870 if (!src || src->type != LocalSourceItem::GRIB || src->url.IsEmpty()) {
871 m_downloading =
false;
872 m_stLocalDownloadInfo->SetLabelText(_(
"Download can't be started."));
873 m_btnDownloadWorld->SetLabelText(_(
"Download"));
875 EnableDownloadButtons();
881 wxEVT_DOWNLOAD_EVENT,
882 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
884 wxString url = src->url;
887 if (src->download_type == LocalGribDownloadType::MANIFEST) {
888 wxString path = m_parent.GetGribDir();
889 path.Append(wxFileName::GetPathSeparator());
890 path.Append(
"grib_manifest.json");
891 auto res = OCPN_downloadFileBackground(url, path,
this, &m_download_handle);
892 while (m_downloading) {
893 wxTheApp->ProcessPendingEvents();
898 if (!m_bTransferSuccess) {
900 m_stLocalDownloadInfo->SetLabelText(_(
"Download failed"));
904 m_stLocalDownloadInfo->SetLabelText(_(
"Download canceled"));
908 wxFileInputStream str(path);
910 reader.
Parse(str, &root);
912 wxString parsed = root[
"url"].
AsString();
913 if (parsed.StartsWith(
"http")) {
916 m_stLocalDownloadInfo->SetLabelText(_(
"Download failed"));
923 m_downloading =
false;
924 m_download_handle = 0;
925 Disconnect(wxEVT_DOWNLOAD_EVENT,
929 m_btnDownloadLocal->SetLabelText(_(
"Download"));
930 m_stLocalDownloadInfo->SetLabelText(_(
"Download failed"));
933 EnableDownloadButtons();
934 wxTheApp->ProcessPendingEvents();
940 m_downloading =
true;
945 wxEVT_DOWNLOAD_EVENT,
946 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
949 if (!src->filename.IsEmpty()) {
950 filename = src->filename;
955 filename.Replace(
"?",
"_");
956 filename.Replace(
"&",
"_");
957 if (!(filename.Contains(
".grb2") || filename.Contains(
".grib2") ||
958 filename.Contains(
".grb") || filename.Contains(
".grib"))) {
959 filename.Append(
".grb");
963 wxString path = m_parent.GetGribDir();
964 path.Append(wxFileName::GetPathSeparator());
965 path.Append(filename);
966 auto res = OCPN_downloadFileBackground(url, path,
this, &m_download_handle);
967 while (m_downloading) {
968 wxTheApp->ProcessPendingEvents();
972 if (m_bTransferSuccess) {
973 m_stLocalDownloadInfo->SetLabelText(_(
"Grib download complete."));
974 m_stLocalDownloadInfo->SetLabelText(
975 wxString::Format(_(
"Download complete: %s"), path.c_str()));
983 if (m_parent.
pPlugIn->m_bZoomToCenterAtInit) m_parent.DoZoomToCenter();
985 m_parent.SetDialogsStyleSizePosition(
true);
989 m_stLocalDownloadInfo->SetLabelText(_(
"Download failed"));
992 m_btnDownloadWorld->SetLabelText(_(
"Download"));
994 EnableDownloadButtons();
997void GribRequestSetting::EnableDownloadButtons() {
998 switch (m_downloadType) {
1000 m_btnDownloadWorld->Enable(
true);
1001 m_btnDownloadLocal->Enable(
false);
1002 m_buttonUpdateCatalog->Enable(
false);
1006 m_btnDownloadWorld->Enable(
false);
1007 m_btnDownloadLocal->Enable(m_bLocal_source_selected || m_downloading);
1008 m_buttonUpdateCatalog->Enable(
false);
1011 m_xygribPanel->m_download_button->Enable(
true);
1014 m_btnDownloadWorld->Enable(
true);
1015 m_btnDownloadLocal->Enable(m_bLocal_source_selected || m_downloading);
1016 m_buttonUpdateCatalog->Enable(
true);
1017 m_xygribPanel->m_download_button->Enable(
true);
1022void GribRequestSetting::StopGraphicalZoneSelection() {
1042 if (!m_AllowSend)
return;
1048 UpdateGribSizeEstimate();
1051void GribRequestSetting::ApplyRequestConfig(
unsigned rs,
unsigned it,
1054 const wxString res[][RESOLUTIONS] = {
1055 {_T(
"0.25"), _T(
"0.5"), _T(
"1.0"), _T(
"2.0")},
1056 {_T(
"0.2"), _T(
"0.8"), _T(
"1.6"), wxEmptyString},
1057 {_T(
"0.08"), _T(
"0.24"), _T(
"1.0"), wxEmptyString},
1058 {_T(
"0.03"), _T(
"0.24"), _T(
"1.0"), wxEmptyString},
1059 {_T(
"0.0625"), _T(
"0.125"), wxEmptyString, wxEmptyString},
1060 {_T(
"0.4"), _T(
"1.0"), _T(
"2.0"), wxEmptyString}
1063 IsZYGRIB = m_pMailTo->GetCurrentSelection() == ZYGRIB;
1065 m_pModel->SetSelection(GFS);
1066 IsGFS = m_pModel->GetCurrentSelection() == GFS;
1067 bool IsRTOFS = m_pModel->GetCurrentSelection() == RTOFS;
1068 bool IsHRRR = m_pModel->GetCurrentSelection() == HRRR;
1069 bool IsICON = m_pModel->GetCurrentSelection() == ICON;
1070 bool IsECMWF = m_pModel->GetCurrentSelection() == ECMWF;
1073 m_pResolution->Clear();
1074 if (m_pModel->GetCurrentSelection() >= 0) {
1075 for (
int i = 0; i < RESOLUTIONS; i++) {
1076 if (res[m_pModel->GetCurrentSelection()][i] != wxEmptyString) {
1077 wxString s = res[m_pModel->GetCurrentSelection()][i];
1078 m_pResolution->Append(s);
1082 m_pResolution->SetSelection(rs);
1086 l = (IsGFS || IsRTOFS || IsICON || IsECMWF) ? 3 : IsHRRR ? 1 : 6;
1089 m = IsHRRR ? 2 : 25;
1091 m_pInterval->Clear();
1092 for (
unsigned i = l; i < m; i *= 2)
1093 m_pInterval->Append(wxString::Format(_T(
"%d"), i));
1094 m_pInterval->SetSelection(wxMin(it, m_pInterval->GetCount() - 1));
1104 m_pTimeRange->Clear();
1105 for (
unsigned i = 2; i < l + 1; i++)
1106 m_pTimeRange->Append(wxString::Format(_T(
"%d"), i));
1107 m_pTimeRange->SetSelection(wxMin(l - 2, tr));
1109 m_pModel->Enable(!IsZYGRIB);
1110 m_pWind->SetValue(!IsRTOFS);
1111 m_pPress->SetValue(!IsRTOFS);
1112 m_pWaves->SetValue(m_RequestConfigBase.GetChar(8) ==
'X' && IsGFS);
1113 m_pWaves->Enable(IsECMWF ||
1114 (IsGFS && m_pTimeRange->GetCurrentSelection() < 7));
1116 m_pRainfall->SetValue(m_RequestConfigBase.GetChar(9) ==
'X' &&
1118 m_pRainfall->Enable(IsGFS || IsHRRR);
1119 m_pCloudCover->SetValue(m_RequestConfigBase.GetChar(10) ==
'X' && IsGFS);
1120 m_pCloudCover->Enable(IsGFS);
1121 m_pAirTemp->SetValue(m_RequestConfigBase.GetChar(11) ==
'X' &&
1122 (IsGFS || IsHRRR || IsICON || IsECMWF));
1123 m_pAirTemp->Enable(IsGFS || IsHRRR || IsICON || IsECMWF);
1124 m_pSeaTemp->SetValue(m_RequestConfigBase.GetChar(12) ==
'X' &&
1125 ((!IsZYGRIB && IsGFS) || IsRTOFS || IsHRRR || IsICON));
1126 m_pSeaTemp->Enable(!IsZYGRIB && (IsGFS || IsHRRR || IsICON));
1127 m_pWindGust->SetValue(m_RequestConfigBase.GetChar(14) ==
'X' &&
1128 (IsGFS || IsHRRR || IsICON));
1129 m_pWindGust->Enable(IsGFS || IsHRRR || IsICON);
1130 m_pCAPE->SetValue(m_RequestConfigBase.GetChar(15) ==
'X' &&
1132 m_pCAPE->Enable(IsGFS || IsHRRR);
1133 m_pReflectivity->Enable(IsGFS || IsHRRR);
1135 m_pAltitudeData->SetValue(
1136 (IsGFS || IsICON || IsECMWF)
1137 ? m_RequestConfigBase.GetChar(17) ==
'X'
1139 m_pAltitudeData->Enable(IsGFS || IsICON || IsECMWF);
1140 m_p850hpa->SetValue(IsZYGRIB ? m_RequestConfigBase.GetChar(18) ==
'X'
1142 m_p850hpa->Enable(IsZYGRIB);
1143 m_p700hpa->SetValue(IsZYGRIB ? m_RequestConfigBase.GetChar(19) ==
'X'
1145 m_p700hpa->Enable(IsZYGRIB);
1146 m_p500hpa->SetValue((IsGFS || IsICON || IsECMWF)
1147 ? m_RequestConfigBase.GetChar(20) ==
'X'
1149 m_p500hpa->Enable(IsGFS || IsICON || IsECMWF);
1150 m_p300hpa->SetValue(IsZYGRIB ? m_RequestConfigBase.GetChar(21) ==
'X'
1152 m_p300hpa->Enable(IsZYGRIB);
1154 m_pCurrent->SetValue(IsRTOFS);
1155 m_pCurrent->Enable(
false);
1158 m_cMovingGribEnabled->Show(!IsZYGRIB);
1159 m_fgMovingParams->ShowItems(m_cMovingGribEnabled->IsChecked() &&
1160 m_cMovingGribEnabled->IsShown());
1162 m_fgLog->ShowItems(IsZYGRIB);
1164 m_pWModel->Show(IsZYGRIB && m_pWaves->IsChecked());
1166 m_fgAltitudeData->ShowItems(
1167 m_pAltitudeData->IsChecked());
1170void GribRequestSetting::OnTopChange(wxCommandEvent &event) {
1172 if (m_pMailTo->GetCurrentSelection() == ZYGRIB) {
1173 m_pMailTo->SetSelection(0);
1174 int mes = OCPNMessageBox_PlugIn(
1176 _(
"Sorry...\nZyGrib momentary stopped providing this service...\nOnly "
1177 "Saildocs option is available"),
1178 _(
"Warning"), wxOK);
1180 ApplyRequestConfig(m_pResolution->GetCurrentSelection(),
1181 m_pInterval->GetCurrentSelection(),
1182 m_pTimeRange->GetCurrentSelection());
1184 m_cMovingGribEnabled->Show(m_pMailTo->GetCurrentSelection() == SAILDOCS);
1186 if (m_AllowSend) m_MailImage->SetValue(WriteMail());
1188 SetRequestDialogSize();
1191void GribRequestSetting::UpdateAreaSelectionState() {
1200void GribRequestSetting::OnZoneSelectionModeChange(wxCommandEvent &event) {
1201 StopGraphicalZoneSelection();
1210 }
else if (event.GetId() ==
MANSELECT) {
1216 m_cUseSavedZone->SetValue(
false);
1217 }
else if (event.GetId() ==
SAVEDZONE) {
1225 UpdateAreaSelectionState();
1226 if (m_AllowSend) m_MailImage->SetValue(WriteMail());
1228 SetRequestDialogSize();
1242 center.x = x + (zw / 2);
1243 center.y = y + (zh / 2);
1249 wxColour pen_color, back_color;
1250 GetGlobalColor(_T (
"DASHR" ), &pen_color);
1251 GetGlobalColor(_T (
"YELO1" ), &back_color);
1253 int label_offsetx = 5, label_offsety = 1;
1256 EstimateFileSize(&size);
1258 wxString label(_(
"Coord. "));
1259 label.Append(toMailFormat(1,
m_spMaxLat->GetValue()) + _T(
" "));
1260 label.Append(toMailFormat(0,
m_spMinLon->GetValue()) + _T(
" "));
1261 label.Append(toMailFormat(1,
m_spMinLat->GetValue()) + _T(
" "));
1262 label.Append(toMailFormat(0,
m_spMaxLon->GetValue()) + _T(
"\n"));
1263 label.Append(_T(
"Estim. Size "))
1264 .Append((wxString::Format(_T(
"%1.2f " ), size) + _(
"MB")));
1267 wxPen pen(pen_color);
1270 m_pdc->SetBrush(*wxTRANSPARENT_BRUSH);
1271 m_pdc->DrawRectangle(x, y, zw, zh);
1276 sdc.GetMultiLineTextExtent(label, &w, &h, &sl, font);
1278 m_pdc->GetMultiLineTextExtent(label, &w, &h, &sl, font);
1282 w += 2 * label_offsetx, h += 2 * label_offsety;
1283 x = center.x - (w / 2);
1284 y = center.y - (h / 2);
1286 h *= m_displayScale;
1287 w *= m_displayScale;
1294 mdc.SetBrush(back_color);
1295 mdc.SetPen(*wxTRANSPARENT_PEN);
1296 mdc.SetTextForeground(wxColor(0, 0, 0));
1297 mdc.DrawRectangle(0, 0, w, h);
1298 mdc.DrawLabel(label, wxRect(label_offsetx, label_offsety, w, h));
1300 wxImage im = bm.ConvertToImage();
1302 w = im.GetWidth(), h = im.GetHeight();
1303 for (
int j = 0; j < h; j++)
1304 for (
int i = 0; i < w; i++) im.SetAlpha(i, j, 155);
1306 m_pdc->DrawBitmap(im, x, y,
true);
1310#ifndef USE_ANDROID_GLES2
1317 m_oDC->SetPen(wxPen(pen_color, 3));
1320 outline[0] = wxPoint(x, y);
1321 outline[1] = wxPoint(x + zw, y);
1322 outline[2] = wxPoint(x + zw, y + zh);
1323 outline[3] = wxPoint(x, y + zh);
1324 outline[4] = wxPoint(x, y);
1325 m_oDC->DrawLines(5, outline);
1327 m_oDC->SetFont(*font);
1329 m_oDC->GetTextExtent(label, &w, &h);
1330 h *= m_displayScale;
1331 w *= m_displayScale;
1333 m_oDC->GetTextExtent(
"W", &ww, &hw);
1335 int label_offsetx = ww, label_offsety = 1;
1336 int x = center.x - w / 2;
1337 int y = center.y - h / 2;
1339 w += 2 * label_offsetx, h += 2 * label_offsety;
1341 m_oDC->SetBrush(wxBrush(back_color));
1342 m_oDC->DrawRoundedRectangle(x, y, w, h, 0);
1345 m_oDC->SetPen(wxPen(wxColour(0, 0, 0), 1));
1347 outline[0] = wxPoint(x, y);
1348 outline[1] = wxPoint(x + w, y);
1349 outline[2] = wxPoint(x + w, y + h);
1350 outline[3] = wxPoint(x, y + h);
1351 outline[4] = wxPoint(x, y);
1352 m_oDC->DrawLines(5, outline);
1354 m_oDC->DrawText(label, x + label_offsetx, y + label_offsety);
1374void GribRequestSetting::OnMovingClick(wxCommandEvent &event) {
1375 m_fgMovingParams->ShowItems(m_cMovingGribEnabled->IsChecked() &&
1376 m_cMovingGribEnabled->IsShown());
1378 if (m_AllowSend) m_MailImage->SetValue(WriteMail());
1379 SetRequestDialogSize();
1384void GribRequestSetting::OnCoordinatesChange(wxSpinEvent &event) {
1385 SetCoordinatesText();
1387 StopGraphicalZoneSelection();
1389 if (!m_AllowSend)
return;
1391 m_MailImage->SetValue(WriteMail());
1394void GribRequestSetting::OnAnyChange(wxCommandEvent &event) {
1395 m_fgAltitudeData->ShowItems(m_pAltitudeData->IsChecked());
1397 m_pWModel->Show(IsZYGRIB && m_pWaves->IsChecked());
1399 if (m_AllowSend) m_MailImage->SetValue(WriteMail());
1401 SetRequestDialogSize();
1404void GribRequestSetting::OnTimeRangeChange(wxCommandEvent &event) {
1405 m_pWModel->Show(IsZYGRIB && m_pWaves->IsChecked());
1407 if (m_pModel->GetCurrentSelection() == 0) {
1408 if (m_pTimeRange->GetCurrentSelection() >
1410 m_pWaves->SetValue(0);
1411 m_pWaves->Enable(
false);
1412 OCPNMessageBox_PlugIn(
1414 _(
"You request a forecast for more than 8 days horizon.\nThis is "
1415 "conflicting with Wave data which will be removed from your "
1416 "request.\nDon't forget that beyond the first 8 days, the "
1417 "resolution will be only 2.5\u00B0x2.5\u00B0\nand the time "
1418 "intervall 12 hours."),
1421 m_pWaves->Enable(
true);
1424 if (m_AllowSend) m_MailImage->SetValue(WriteMail());
1426 SetRequestDialogSize();
1429void GribRequestSetting::OnOK(wxCommandEvent &event) {
1430 bool IsCOAMPS = m_pModel->GetCurrentSelection() == COAMPS;
1431 bool IsRTOFS = m_pModel->GetCurrentSelection() == RTOFS;
1432 bool IsICON = m_pModel->GetCurrentSelection() == ICON;
1433 bool IsECMWF = m_pModel->GetCurrentSelection() == ECMWF;
1434 m_RequestConfigBase.SetChar(
1435 0, (
char)(m_pMailTo->GetCurrentSelection() +
'0'));
1436 m_cMovingGribEnabled->IsChecked()
1437 ? m_RequestConfigBase.SetChar(16,
'X')
1438 : m_RequestConfigBase.SetChar(16,
'.');
1441 m_RequestConfigBase.SetChar(
1442 1, (
char)(m_pModel->GetCurrentSelection() +
'0'));
1444 m_RequestConfigBase.SetChar(
1445 2, (
char)(m_pResolution->GetCurrentSelection() +
'0'));
1447 m_RequestConfigBase.SetChar(3,
1448 (
char)(m_pInterval->GetCurrentSelection() +
'0'));
1451 range.Printf(_T(
"%x"), m_pTimeRange->GetCurrentSelection() +
1453 m_RequestConfigBase.SetChar(4, range.GetChar(0));
1455 if (IsZYGRIB && m_pWModel->IsShown())
1456 m_RequestConfigBase.SetChar(
1457 5, (
char)(m_pWModel->GetCurrentSelection() +
'0'));
1459 m_RequestConfigBase.SetChar(
1461 m_RequestConfigBase.SetChar(
1464 if (!IsCOAMPS && !IsRTOFS) {
1465 m_pWindGust->IsChecked() ? m_RequestConfigBase.SetChar(14,
'X')
1466 : m_RequestConfigBase.SetChar(14,
'.');
1467 m_pWaves->IsChecked() ? m_RequestConfigBase.SetChar(8,
'X')
1468 : m_RequestConfigBase.SetChar(8,
'.');
1469 m_pRainfall->IsChecked() ? m_RequestConfigBase.SetChar(9,
'X')
1470 : m_RequestConfigBase.SetChar(9,
'.');
1471 m_pCloudCover->IsChecked() ? m_RequestConfigBase.SetChar(10,
'X')
1472 : m_RequestConfigBase.SetChar(10,
'.');
1473 m_pAirTemp->IsChecked() ? m_RequestConfigBase.SetChar(11,
'X')
1474 : m_RequestConfigBase.SetChar(11,
'.');
1475 m_pSeaTemp->IsChecked() ? m_RequestConfigBase.SetChar(12,
'X')
1476 : m_RequestConfigBase.SetChar(12,
'.');
1477 m_pCAPE->IsChecked() ? m_RequestConfigBase.SetChar(15,
'X')
1478 : m_RequestConfigBase.SetChar(15,
'.');
1481 m_pCurrent->IsChecked() ? m_RequestConfigBase.SetChar(13,
'X')
1482 : m_RequestConfigBase.SetChar(13,
'.');
1484 if (IsGFS || IsICON || IsECMWF) {
1485 m_pAltitudeData->IsChecked()
1486 ? m_RequestConfigBase.SetChar(17,
'X')
1487 : m_RequestConfigBase.SetChar(17,
'.');
1488 m_p500hpa->IsChecked() ? m_RequestConfigBase.SetChar(20,
'X')
1489 : m_RequestConfigBase.SetChar(20,
'.');
1492 m_p850hpa->IsChecked() ? m_RequestConfigBase.SetChar(18,
'X')
1493 : m_RequestConfigBase.SetChar(18,
'.');
1494 m_p700hpa->IsChecked() ? m_RequestConfigBase.SetChar(19,
'X')
1495 : m_RequestConfigBase.SetChar(19,
'.');
1496 m_p300hpa->IsChecked() ? m_RequestConfigBase.SetChar(21,
'X')
1497 : m_RequestConfigBase.SetChar(21,
'.');
1504wxString GribRequestSetting::WriteMail() {
1506 int limit = IsZYGRIB ? 2 : 0;
1510 const wxString s[] = {_T(
","), _T(
" ")};
1511 const wxString p[][11] = {
1513 {_T(
"APCP"), _T(
"TCDC"), _T(
"AIRTMP"), _T(
"HTSGW,WVPER,WVDIR"),
1514 _T(
"SEATMP"), _T(
"GUST"), _T(
"CAPE"), wxEmptyString, wxEmptyString,
1515 _T(
"WIND500,HGT500"), wxEmptyString},
1520 {_T(
""), _T(
""), _T(
"AIRTMP"), _T(
""), _T(
"SFCTMP"), _T(
"GUST"), _T(
""),
1521 _T(
""), _T(
""), _T(
"WIND500,HGT500"), _T(
"")},
1523 {_T(
""), _T(
""), _T(
"TEMP"), _T(
"WAVES"), _T(
""), _T(
""), _T(
""), _T(
""),
1524 _T(
""), _T(
"WIND500,HGT500"), _T(
"")},
1526 {_T(
"PRECIP"), _T(
"CLOUD"), _T(
"TEMP"), _T(
"WVSIG WVWIND"), wxEmptyString,
1527 _T(
"GUST"), _T(
"CAPE"), _T(
"A850"), _T(
"A700"), _T(
"A500"), _T(
"A300")}};
1529 wxString r_topmess, r_parameters, r_zone;
1531 switch (m_pMailTo->GetCurrentSelection()) {
1533 r_zone = toMailFormat(1,
m_spMaxLat->GetValue()) + _T(
",") +
1534 toMailFormat(1,
m_spMinLat->GetValue()) + _T(
",") +
1535 toMailFormat(2,
m_spMinLon->GetValue()) + _T(
",") +
1537 r_topmess = wxT(
"send ");
1538 r_topmess.Append(m_pModel->GetStringSelection() + _T(
":"));
1539 r_topmess.Append(r_zone + _T(
"|"));
1540 r_topmess.Append(m_pResolution->GetStringSelection())
1542 .Append(m_pResolution->GetStringSelection())
1545 m_pInterval->GetStringSelection().ToDouble(&v);
1546 r_topmess.Append(wxString::Format(_T(
"0,%d,%d"), (
int)v, (
int)v * 2));
1547 m_pTimeRange->GetStringSelection().ToDouble(&v);
1548 r_topmess.Append(wxString::Format(_T(
"..%d"), (
int)v * 24) + _T(
"|=\n"));
1555 r_zone = toMailFormat(1,
m_spMinLat->GetValue()) +
1556 toMailFormat(2,
m_spMinLon->GetValue()) + _T(
" ") +
1558 toMailFormat(2, maxlon);
1559 r_topmess = wxT(
"login : ");
1560 r_topmess.Append(m_pLogin->GetValue() + _T(
"\n"));
1561 r_topmess.Append(wxT(
"code :"));
1562 r_topmess.Append(m_pCode->GetValue() + _T(
"\n"));
1563 r_topmess.Append(wxT(
"area : "));
1564 r_topmess.append(r_zone + _T(
"\n"));
1565 r_topmess.Append(wxT(
"resol : "));
1566 r_topmess.append(m_pResolution->GetStringSelection() + _T(
"\n"));
1567 r_topmess.Append(wxT(
"days : "));
1568 r_topmess.append(m_pTimeRange->GetStringSelection() + _T(
"\n"));
1569 r_topmess.Append(wxT(
"hours : "));
1570 r_topmess.append(m_pInterval->GetStringSelection() + _T(
"\n"));
1571 if (m_pWaves->IsChecked()) {
1572 r_topmess.Append(wxT(
"waves : "));
1573 r_topmess.append(m_pWModel->GetStringSelection() + _T(
"\n"));
1575 r_topmess.Append(wxT(
"meteo : "));
1576 r_topmess.append(m_pModel->GetStringSelection() + _T(
"\n"));
1577 if (m_pLogin->GetValue().IsEmpty() || m_pCode->GetValue().IsEmpty())
1582 int GFSZ = IsZYGRIB ? 6 : 0;
1583 switch (m_pModel->GetCurrentSelection()) {
1585 r_parameters = wxT(
"WIND") + s[m_pMailTo->GetCurrentSelection()] +
1587 if (m_pRainfall->IsChecked())
1588 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1590 if (m_pCloudCover->IsChecked())
1591 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1593 if (m_pAirTemp->IsChecked())
1594 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1596 if (m_pWaves->IsChecked())
1597 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1599 if (m_pSeaTemp->IsChecked())
1600 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1602 if (m_pWindGust->IsChecked())
1603 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1605 if (m_pCAPE->IsChecked())
1606 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1608 if (m_pAltitudeData->IsChecked()) {
1609 if (m_p850hpa->IsChecked())
1610 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1612 if (m_p700hpa->IsChecked())
1613 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1615 if (m_p500hpa->IsChecked())
1616 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1618 if (m_p300hpa->IsChecked())
1619 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1624 r_parameters = wxT(
"WIND,PRMSL");
1628 r_parameters = wxT(
"CUR,WTMP");
1631 r_parameters = wxT(
"WIND,PRMSL");
1633 if (m_pRainfall->IsChecked())
1634 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[GFS][0]);
1635 if (m_pAirTemp->IsChecked())
1636 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[GFS][2]);
1637 if (m_pSeaTemp->IsChecked())
1638 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[GFS][4]);
1639 if (m_pWindGust->IsChecked())
1640 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[GFS][5]);
1641 if (m_pCAPE->IsChecked())
1642 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[GFS][6]);
1645 r_parameters = wxT(
"WIND,PRMSL");
1647 if (m_pAirTemp->IsChecked())
1648 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ICON][2]);
1649 if (m_pSeaTemp->IsChecked())
1650 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ICON][4]);
1651 if (m_pWindGust->IsChecked())
1652 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ICON][5]);
1653 if (m_pAltitudeData->IsChecked()) {
1654 if (m_p500hpa->IsChecked())
1655 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ICON][9]);
1659 r_parameters = wxT(
"WIND,MSLP");
1661 if (m_pAirTemp->IsChecked())
1662 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ECMWF][2]);
1663 if (m_pAltitudeData->IsChecked()) {
1664 if (m_p500hpa->IsChecked())
1665 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] +
1668 if (m_pWaves->IsChecked())
1669 r_parameters.Append(s[m_pMailTo->GetCurrentSelection()] + p[ECMWF][3]);
1672 if (!IsZYGRIB && m_cMovingGribEnabled->IsChecked())
1673 r_parameters.Append(wxString::Format(
1674 _T(
"|%d,%d"), m_sMovingSpeed->GetValue(), m_sMovingCourse->GetValue()));
1678 char c = m_pMailTo->GetCurrentSelection() == SAILDOCS ?
',' :
' ';
1679 for (
size_t i = 0; i < r_parameters.Len(); i++) {
1680 if (r_parameters.GetChar(i) ==
'|')
1682 if (r_parameters.GetChar(i) == c) j++;
1684 r_parameters.insert(i + 1, m_pMailTo->GetCurrentSelection() == SAILDOCS
1692 m_MailError_Nb += EstimateFileSize(&size);
1694 m_tFileSize->SetLabel(wxString::Format(_T(
"%1.2f " ), size) + _(
"MB"));
1697 m_tLimit->SetLabel(wxString(_T(
"( ")) + _(
"Max") +
1698 wxString::Format(_T(
" %d "), limit) + _(
"MB") +
1700 if (size > limit) m_MailError_Nb += 2;
1702 m_tLimit->SetLabel(wxEmptyString);
1704 return wxString(r_topmess + r_parameters);
1707int GribRequestSetting::EstimateFileSize(
double *size) {
1708 if (!size)
return 0;
1712 double reso, time, inter;
1713 m_pResolution->GetStringSelection().ToDouble(&reso);
1714 m_pTimeRange->GetStringSelection().ToDouble(&time);
1715 m_pInterval->GetStringSelection().ToDouble(&inter);
1719 if (maxlat - minlat < 0)
return 3;
1720 double wlon = (maxlon > minlon ? 0 : 360) + maxlon - minlon;
1721 if (wlon > 180 || (maxlat - minlat > 180))
return 4;
1723 if (fabs(wlon) < 2 * reso || maxlat - minlat < 2 * reso)
1726 int npts = (int)(ceil(((
double)(maxlat - minlat) / reso)) *
1727 ceil(((
double)(wlon) / reso)));
1729 if (m_pModel->GetCurrentSelection() == COAMPS)
1730 npts = wxMin(npts, (
int)(ceil(40.0 / reso) * ceil(40.0 / reso)));
1733 int nbrec = (int)(time * 24 / inter) + 1;
1734 int nbPress = (m_pPress->IsChecked()) ? nbrec : 0;
1735 int nbWind = (m_pWind->IsChecked()) ? 2 * nbrec : 0;
1736 int nbwave = (m_pWaves->IsChecked()) ? 2 * nbrec : 0;
1737 int nbRain = (m_pRainfall->IsChecked()) ? nbrec - 1 : 0;
1738 int nbCloud = (m_pCloudCover->IsChecked()) ? nbrec - 1 : 0;
1739 int nbTemp = (m_pAirTemp->IsChecked()) ? nbrec : 0;
1740 int nbSTemp = (m_pSeaTemp->IsChecked()) ? nbrec : 0;
1741 int nbGUSTsfc = (m_pWindGust->IsChecked()) ? nbrec : 0;
1742 int nbCurrent = (m_pCurrent->IsChecked()) ? nbrec : 0;
1743 int nbCape = (m_pCAPE->IsChecked()) ? nbrec : 0;
1745 IsZYGRIB ? 5 * nbrec : 3 * nbrec;
1749 double estime = 0.0;
1753 estime += nbWind * (head + (nbits * npts) / 8 + 2);
1754 estime += nbCurrent * (head + (nbits * npts) / 8 + 2);
1757 estime += nbTemp * (head + (nbits * npts) / 8 + 2);
1758 estime += nbSTemp * (head + (nbits * npts) / 8 + 2);
1761 estime += nbRain * (head + (nbits * npts) / 8 + 2);
1764 estime += nbPress * (head + (nbits * npts) / 8 + 2);
1767 estime += nbCloud * (head + (nbits * npts) / 8 + 2);
1770 estime += nbGUSTsfc * (head + (nbits * npts) / 8 + 2);
1773 estime += nbCape * (head + (nbits * npts) / 8 + 2);
1776 estime += nbwave * (head + (nbits * npts) / 8 + 2);
1778 if (m_pAltitudeData->IsChecked()) {
1780 if (m_p850hpa->IsChecked()) nbalt++;
1781 if (m_p700hpa->IsChecked()) nbalt++;
1782 if (m_p500hpa->IsChecked()) nbalt++;
1783 if (m_p300hpa->IsChecked()) nbalt++;
1786 estime += nbAltitude * nbalt * (head + (nbits * npts) / 8 + 2);
1789 *size = estime / (1024. * 1024.);
1794const wxString EncodeURL(
const wxString &uri) {
1795 static std::unordered_map<int, wxString> sEncodeMap = {
1796 {(int)
'!',
"%21"}, {(int)
'#',
"%23"}, {(int)
'$',
"%24"},
1797 {(int)
'&',
"%26"}, {(int)
'\'',
"%27"}, {(int)
'(',
"%28"},
1798 {(int)
')',
"%29"}, {(int)
'*',
"%2A"}, {(int)
'+',
"%2B"},
1799 {(int)
',',
"%2C"}, {(int)
';',
"%3B"}, {(int)
'=',
"%3D"},
1800 {(int)
'?',
"%3F"}, {(int)
'@',
"%40"}, {(int)
'[',
"%5B"},
1801 {(int)
']',
"%5D"}, {(int)
' ',
"%20"}, {(int)
'|',
"%7C"},
1802 {(int)
':',
"%3A"}, {(int)
'\n',
"%0A"}};
1805 for (
size_t i = 0; i < uri.length(); ++i) {
1807 std::unordered_map<int, wxString>::iterator iter = sEncodeMap.find((
int)ch);
1808 if (iter != sEncodeMap.end()) {
1809 encoded << iter->second;
1817void GribRequestSetting::OnSendMaiL(wxCommandEvent &event) {
1818 StopGraphicalZoneSelection();
1825 m_MailImage->SetForegroundColour(
1829 m_MailImage->SetValue(WriteMail());
1830 SetRequestDialogSize();
1835 const wxString error[] = {
1837 _(
"Before sending an email to Zygrib you have to enter your Login and "
1838 "Code.\nPlease visit www.zygrib.org/ and follow instructions..."),
1839 _(
"Too big file! zyGrib limit is 2Mb!"),
1840 _(
"Error! Max Lat lower than Min Lat or Max Lon lower than Min Lon!"),
1841 _(
"Too large area! Each side must be less than 180\u00B0!"),
1842 _(
"Too small area for this resolution!")};
1844 ::wxBeginBusyCursor();
1846 m_MailImage->SetForegroundColour(wxColor(255, 0, 0));
1847 m_AllowSend =
false;
1849 if (m_MailError_Nb) {
1850 if (m_MailError_Nb > 7) {
1851 m_MailImage->SetValue(error[1] + error[0] + error[m_MailError_Nb - 6]);
1853 if (m_MailError_Nb == 6) m_MailError_Nb = 1;
1854 m_MailImage->SetValue(error[m_MailError_Nb]);
1858 SetRequestDialogSize();
1860 ::wxEndBusyCursor();
1871 (m_pMailTo->GetCurrentSelection() == SAILDOCS)
1872 ? _T(
"grib-request")
1874 (m_pMailTo->GetCurrentSelection() == SAILDOCS)
1875 ? m_MailToAddresses.BeforeFirst(_T(
';'))
1876 : m_MailToAddresses.AfterFirst(_T(
';')).BeforeFirst(_T(
';')),
1877 EncodeURL(WriteMail()),
1878 m_pSenderAddress->GetValue());
1881 (m_pMailTo->GetCurrentSelection() == SAILDOCS)
1882 ? _T(
"grib-request")
1884 (m_pMailTo->GetCurrentSelection() == SAILDOCS)
1885 ? m_MailToAddresses.BeforeFirst(_T(
';'))
1886 : m_MailToAddresses.AfterFirst(_T(
';')).BeforeFirst(_T(
';')),
1888 m_pSenderAddress->GetValue());
1892 if (mail.Send(*message, m_SendMethod)) {
1894 m_MailImage->SetValue(
1895 _(
"Your request is ready. An email is prepared in your email "
1896 "environment. \nYou have just to verify and send it...\nSave or "
1897 "Cancel to finish...or Continue..."));
1899 if (m_SendMethod == 0) {
1900 m_MailImage->SetValue(
1901 _(
"Your request is ready. An email is prepared in your email "
1902 "environment. \nYou have just to verify and send it...\nSave or "
1903 "Cancel to finish...or Continue..."));
1905 m_MailImage->SetValue(
1906 _(
"Your request was sent \n(if your system has an MTA configured and "
1907 "is able to send email).\nSave or Cancel to finish...or "
1912 m_MailImage->SetValue(
1913 _(
"Request can't be sent. Please verify your email systeme "
1914 "parameters.\nYou should also have a look at your log file.\nSave or "
1915 "Cancel to finish..."));
1919 SetRequestDialogSize();
1921 ::wxEndBusyCursor();
1926wxString GribRequestSetting::BuildXyGribUrl() {
1929 wxString::Format(
"http://grbsrv.opengribs.org/getmygribs2.php?");
1931 urlStr << wxString::Format(
"la1=%.0f", floor(
GetMinLat()));
1932 urlStr << wxString::Format(
"&la2=%.0f", ceil(
GetMaxLat()));
1933 urlStr << wxString::Format(
"&lo1=%.0f", floor(
GetMinLon()));
1934 urlStr << wxString::Format(
"&lo2=%.0f", ceil(
GetMaxLon()));
1937 urlStr << wxString::Format(
1939 xygribAtmModelList[m_selectedAtmModelIndex]
1940 ->reqName[m_xygribPanel->m_resolution_choice->GetSelection()]);
1942 urlStr << wxString::Format(
1944 xygribAtmModelList[m_selectedAtmModelIndex]
1945 ->interval[m_xygribPanel->m_interval_choice->GetSelection()]);
1947 urlStr << wxString::Format(
1948 "&days=%d", m_xygribPanel->m_duration_choice->GetSelection() + 1);
1952 wxString selStr = m_xygribPanel->m_run_choice->GetStringSelection();
1953 if (selStr.IsSameAs(
"18h",
false)) {
1954 urlStr <<
"&cyc=18";
1955 }
else if (selStr.IsSameAs(
"12h",
false)) {
1956 urlStr <<
"&cyc=12";
1957 }
else if (selStr.IsSameAs(
"6h",
false)) {
1958 urlStr <<
"&cyc=06";
1959 }
else if (selStr.IsSameAs(
"0h",
false)) {
1960 urlStr <<
"&cyc=00";
1962 urlStr <<
"&cyc=last";
1967 if (m_xygribPanel->m_wind_cbox->IsEnabled() &&
1968 m_xygribPanel->m_wind_cbox->IsChecked())
1970 if (m_xygribPanel->m_pressure_cbox->IsEnabled() &&
1971 m_xygribPanel->m_pressure_cbox->IsChecked()) {
1972 if (xygribAtmModelList[m_selectedAtmModelIndex]->altPressure) {
1978 if (m_xygribPanel->m_precipitation_cbox->IsEnabled() &&
1979 m_xygribPanel->m_precipitation_cbox->IsChecked())
1981 if (m_xygribPanel->m_cloudcover_cbox->IsEnabled() &&
1982 m_xygribPanel->m_cloudcover_cbox->IsChecked())
1984 if (m_xygribPanel->m_temperature_cbox->IsEnabled() &&
1985 m_xygribPanel->m_temperature_cbox->IsChecked())
1987 if (m_xygribPanel->m_cape_cbox->IsEnabled() &&
1988 m_xygribPanel->m_cape_cbox->IsChecked())
1990 if (m_xygribPanel->m_reflectivity_cbox->IsEnabled() &&
1991 m_xygribPanel->m_reflectivity_cbox->IsChecked())
1993 if (m_xygribPanel->m_gust_cbox->IsEnabled() &&
1994 m_xygribPanel->m_gust_cbox->IsChecked())
1998 if ((m_selectedWaveModelIndex >= 0) &&
1999 (xygribWaveModelList[m_selectedWaveModelIndex] !=
nullptr)) {
2000 wxString modelStr = wxString::Format(
2001 "&wmdl=%s", xygribWaveModelList[m_selectedWaveModelIndex]->reqName);
2002 wxString wParams =
"";
2003 if (m_xygribPanel->m_waveheight_cbox->IsChecked()) {
2006 if (m_xygribPanel->m_windwave_cbox->IsChecked()) {
2007 wParams <<
"h;d;p;";
2009 if (wParams.length() > 0) {
2010 urlStr << wxString::Format(
"%s&wpar=%s", modelStr.c_str(),
2013 urlStr <<
"&wmdl=none";
2016 urlStr <<
"&wmdl=none";
2024wxString GribRequestSetting::BuildGribFileName() {
2025 wxString selStr = m_xygribPanel->m_resolution_choice->GetStringSelection();
2026 selStr.Replace(
".",
"P");
2029 if (m_selectedWaveModelIndex < 0) {
2030 fileName = wxString::Format(
2031 "XyGrib_%s_%s_%s.grb2", wxDateTime::Now().Format(
"%F-%H-%M"),
2032 m_xygribPanel->m_atmmodel_choice->GetStringSelection(), selStr);
2034 fileName = wxString::Format(
2035 "XyGrib_%s_%s_%s_%s.grb2", wxDateTime::Now().Format(
"%F-%H-%M"),
2036 m_xygribPanel->m_atmmodel_choice->GetStringSelection(), selStr,
2037 m_xygribPanel->m_wavemodel_choice->GetStringSelection());
2049void GribRequestSetting::OnXyGribDownloadButton(wxCommandEvent &event) {
2051 if (m_downloading) {
2054 OCPN_cancelDownloadFileBackground(m_download_handle);
2055 m_downloading =
false;
2056 m_download_handle = 0;
2058 wxEVT_DOWNLOAD_EVENT,
2059 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
2060 m_connected =
false;
2061 m_xygribPanel->m_progress_gauge->SetValue(0);
2062 m_xygribPanel->m_download_button->SetLabelText(_(
"Download"));
2063 m_xygribPanel->m_status_text->SetLabelText(_(
"Download cancelled"));
2066 EnableDownloadButtons();
2067 wxTheApp->ProcessPendingEvents();
2072 if (m_gribSizeEstimate > XYGRIB_MAX_DOWNLOADABLE_GRIB_SIZE_MB * 1024 * 1024) {
2073 m_xygribPanel->m_status_text->SetLabelText(
2074 wxString::Format(_(
"Can't download GRIB file bigger than %d MB"),
2075 (
int)XYGRIB_MAX_DOWNLOADABLE_GRIB_SIZE_MB));
2081 MemorizeXyGribConfiguration();
2083 m_downloading =
true;
2085 EnableDownloadButtons();
2087 m_xygribPanel->m_download_button->SetLabelText(_(
"Cancel"));
2088 m_xygribPanel->m_status_text->SetLabelText(
2089 _(
"Preparing GRIB file on server..."));
2093 wxString requestUrl = BuildXyGribUrl();
2098 wxString filename = wxString::Format(
"ocpn_xygrib_%s.xml",
2099 wxDateTime::Now().Format(
"%F-%H-%M"));
2100 wxString path = m_parent.GetGribDir();
2101 path << wxFileName::GetPathSeparator();
2106 wxEVT_DOWNLOAD_EVENT,
2107 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
2109 auto res = OCPN_downloadFileBackground(requestUrl.c_str(), path,
this,
2110 &m_download_handle);
2115 while (m_downloading) {
2116 wxTheApp->ProcessPendingEvents();
2120 if ((m_canceled) || (!m_bTransferSuccess)) {
2122 m_xygribPanel->m_status_text->SetLabelText(_(
"Grib request failed"));
2123 m_xygribPanel->m_download_button->SetLabelText(_(
"Download"));
2125 EnableDownloadButtons();
2133 bool readOk = xmlFile.Open(path);
2135 readOk &= xmlFile.ReadAll(&strXml);
2143 if (readOk && (((
int)strXml.find(
"\"status\":true") == wxNOT_FOUND))) {
2146 int errPos = strXml.find(
"\"message\":\"");
2147 int errEnd = strXml.find(
"\"}");
2148 if ((errPos != wxNOT_FOUND) && (errEnd != wxNOT_FOUND)) {
2150 errorStr = strXml.Mid(errPos, errEnd - errPos);
2152 errorStr =
"Unknown server error";
2155 m_xygribPanel->m_status_text->SetLabelText(
2156 wxString::Format(
"%s (%s)", _(
"Server Error"), errorStr));
2157 m_xygribPanel->m_download_button->SetLabelText(_(
"Download"));
2159 EnableDownloadButtons();
2164 int urlPos = strXml.find(
"\"url\":\"http:");
2165 int urlEnd = strXml.find(
".grb2\"");
2166 if ((urlPos != wxNOT_FOUND) && (urlEnd != wxNOT_FOUND)) {
2169 url = strXml.Mid(urlPos, urlEnd - urlPos);
2170 url.Replace(
"\\/",
"/");
2181 m_xygribPanel->m_status_text->SetLabelText(
2182 _(
"Error parsing XML file from server"));
2183 m_xygribPanel->m_download_button->SetLabelText(_(
"Download"));
2185 EnableDownloadButtons();
2189 m_xygribPanel->m_status_text->SetLabelText(_(
"Downloading GRIB file"));
2192 path = m_parent.GetGribDir();
2193 path << wxFileName::GetPathSeparator();
2194 path << BuildGribFileName();
2198 m_downloading =
true;
2202 wxEVT_DOWNLOAD_EVENT,
2203 (wxObjectEventFunction)(wxEventFunction)&GribRequestSetting::onDLEvent);
2206 OCPN_downloadFileBackground(url.c_str(), path,
this, &m_download_handle);
2209 while (m_downloading) {
2210 wxTheApp->ProcessPendingEvents();
2215 m_xygribPanel->m_download_button->SetLabelText(_(
"Download"));
2218 if (m_bTransferSuccess) {
2220 m_xygribPanel->m_status_text->SetLabelText(_(
"Download complete"));
2221 wxFileName fn(path);
2225 m_parent.OpenFile();
2227 if (m_parent.
pPlugIn->m_bZoomToCenterAtInit) m_parent.DoZoomToCenter();
2229 m_parent.SetDialogsStyleSizePosition(
true);
2234 m_xygribPanel->m_status_text->SetLabelText(_(
"Download failed"));
2238 EnableDownloadButtons();
2244void GribRequestSetting::OnXyGribAtmModelChoice(wxCommandEvent &event) {
2248 wxString atmModel = m_xygribPanel->m_atmmodel_choice->GetStringSelection();
2250 while ((selectedAtmModel = xygribAtmModelList[modelIndex]) !=
nullptr) {
2251 if (selectedAtmModel->name.IsSameAs(atmModel,
true)) {
2258 if (selectedAtmModel ==
nullptr) {
2259 selectedAtmModel = xygribAtmModelList[0];
2264 if (selectedAtmModel->wind)
2265 m_xygribPanel->m_wind_cbox->Enable();
2267 m_xygribPanel->m_wind_cbox->Disable();
2269 if (selectedAtmModel->gust)
2270 m_xygribPanel->m_gust_cbox->Enable();
2272 m_xygribPanel->m_gust_cbox->Disable();
2274 if (selectedAtmModel->pressure)
2275 m_xygribPanel->m_pressure_cbox->Enable();
2277 m_xygribPanel->m_pressure_cbox->Disable();
2279 if (selectedAtmModel->temperature)
2280 m_xygribPanel->m_temperature_cbox->Enable();
2282 m_xygribPanel->m_temperature_cbox->Disable();
2284 if (selectedAtmModel->cape)
2285 m_xygribPanel->m_cape_cbox->Enable();
2287 m_xygribPanel->m_cape_cbox->Disable();
2289 if (selectedAtmModel->reflectivity)
2290 m_xygribPanel->m_reflectivity_cbox->Enable();
2292 m_xygribPanel->m_reflectivity_cbox->Disable();
2294 if (selectedAtmModel->cloudCover)
2295 m_xygribPanel->m_cloudcover_cbox->Enable();
2297 m_xygribPanel->m_cloudcover_cbox->Disable();
2299 if (selectedAtmModel->precipitation)
2300 m_xygribPanel->m_precipitation_cbox->Enable();
2302 m_xygribPanel->m_precipitation_cbox->Disable();
2305 m_xygribPanel->m_resolution_choice->Clear();
2306 for (
int i = 0; i < selectedAtmModel->nbRes; i++) {
2307 m_xygribPanel->m_resolution_choice->Insert(selectedAtmModel->resolution[i],
2312 m_xygribPanel->m_duration_choice->Clear();
2313 for (
int i = 0; i < selectedAtmModel->duration; i++) {
2314 m_xygribPanel->m_duration_choice->Insert(wxString::Format(
"%d", i + 1), i);
2318 m_xygribPanel->m_interval_choice->Clear();
2319 for (
int i = 0; i < selectedAtmModel->nbInterval; i++) {
2320 m_xygribPanel->m_interval_choice->Insert(
2321 wxString::Format(
"%dh", selectedAtmModel->interval[i]), i);
2325 m_xygribPanel->m_run_choice->Clear();
2326 if (selectedAtmModel->runMask == XYGRIB_RUN_0_12) {
2327 m_xygribPanel->m_run_choice->Insert(
"0h", 0);
2328 m_xygribPanel->m_run_choice->Insert(
"12h", 1);
2329 m_xygribPanel->m_run_choice->Insert(_(
"Latest"), 2);
2331 m_xygribPanel->m_run_choice->Insert(
"0h", 0);
2332 m_xygribPanel->m_run_choice->Insert(
"6h", 1);
2333 m_xygribPanel->m_run_choice->Insert(
"12h", 2);
2334 m_xygribPanel->m_run_choice->Insert(
"18h", 3);
2335 m_xygribPanel->m_run_choice->Insert(_(
"Latest"), 4);
2338 if (modelIndex == m_parent.xyGribConfig.atmModelIndex) {
2339 ApplyXyGribConfiguration();
2341 m_selectedAtmModelIndex = modelIndex;
2342 m_xygribPanel->m_resolution_choice->SetSelection(0);
2343 m_xygribPanel->m_duration_choice->SetSelection(
2344 m_xygribPanel->m_duration_choice->GetCount() - 1);
2345 m_xygribPanel->m_interval_choice->SetSelection(0);
2346 m_xygribPanel->m_run_choice->SetSelection(
2347 m_xygribPanel->m_run_choice->GetCount() - 1);
2349 MemorizeXyGribConfiguration();
2355void GribRequestSetting::OnXyGribWaveModelChoice(wxCommandEvent &event) {
2359 wxString waveModel = m_xygribPanel->m_wavemodel_choice->GetStringSelection();
2361 while ((selectedModel = xygribWaveModelList[modelIndex]) !=
nullptr) {
2362 if (selectedModel->name.IsSameAs(waveModel,
true)) {
2369 if (selectedModel ==
nullptr) {
2371 m_selectedWaveModelIndex = -1;
2372 m_xygribPanel->m_waveheight_cbox->Disable();
2373 m_xygribPanel->m_windwave_cbox->Disable();
2374 MemorizeXyGribConfiguration();
2378 m_selectedWaveModelIndex = modelIndex;
2381 if (selectedModel->significantHeight) {
2382 m_xygribPanel->m_waveheight_cbox->Enable();
2384 m_xygribPanel->m_waveheight_cbox->Disable();
2387 if (selectedModel->windWaves) {
2388 m_xygribPanel->m_windwave_cbox->Enable();
2390 m_xygribPanel->m_windwave_cbox->Disable();
2392 MemorizeXyGribConfiguration();
2398void GribRequestSetting::OnXyGribConfigChange(wxCommandEvent &event) {
2399 MemorizeXyGribConfiguration();
2403void GribRequestSetting::InitializeXygribDialog() {
2408 m_xygribPanel->m_atmmodel_choice->Clear();
2410 while ((selectedAtmModel = xygribAtmModelList[modelIndex]) !=
nullptr) {
2411 m_xygribPanel->m_atmmodel_choice->Insert(selectedAtmModel->name,
2416 m_selectedAtmModelIndex = m_parent.xyGribConfig.atmModelIndex;
2417 selectedAtmModel = xygribAtmModelList[m_selectedAtmModelIndex];
2420 m_xygribPanel->m_wavemodel_choice->Clear();
2422 while ((selectedWaveModel = xygribWaveModelList[modelIndex]) !=
nullptr) {
2423 m_xygribPanel->m_wavemodel_choice->Insert(selectedWaveModel->name,
2428 m_xygribPanel->m_wavemodel_choice->Insert(
"None", modelIndex);
2429 m_selectedWaveModelIndex = m_parent.xyGribConfig.waveModelIndex;
2430 selectedWaveModel = xygribWaveModelList[m_selectedWaveModelIndex];
2431 if (selectedWaveModel ==
nullptr) {
2432 m_selectedWaveModelIndex = -1;
2436 m_xygribPanel->m_resolution_choice->Clear();
2437 for (
int i = 0; i < selectedAtmModel->nbRes; i++) {
2438 m_xygribPanel->m_resolution_choice->Insert(selectedAtmModel->resolution[i],
2443 m_xygribPanel->m_duration_choice->Clear();
2444 for (
int i = 0; i < selectedAtmModel->duration; i++) {
2445 m_xygribPanel->m_duration_choice->Insert(wxString::Format(
"%d", i + 1), i);
2449 m_xygribPanel->m_interval_choice->Clear();
2450 for (
int i = 0; i < selectedAtmModel->nbInterval; i++) {
2451 m_xygribPanel->m_interval_choice->Insert(
2452 wxString::Format(
"%dh", selectedAtmModel->interval[i]), i);
2456 m_xygribPanel->m_run_choice->Clear();
2457 if (selectedAtmModel->runMask == XYGRIB_RUN_0_12) {
2458 m_xygribPanel->m_run_choice->Insert(
"0h", 0);
2459 m_xygribPanel->m_run_choice->Insert(
"12h", 1);
2460 m_xygribPanel->m_run_choice->Insert(_(
"Latest"), 2);
2462 m_xygribPanel->m_run_choice->Insert(
"0h", 0);
2463 m_xygribPanel->m_run_choice->Insert(
"6h", 1);
2464 m_xygribPanel->m_run_choice->Insert(
"12h", 2);
2465 m_xygribPanel->m_run_choice->Insert(
"18h", 3);
2466 m_xygribPanel->m_run_choice->Insert(_(
"Latest"), 4);
2469 if (selectedAtmModel->wind)
2470 m_xygribPanel->m_wind_cbox->Enable();
2472 m_xygribPanel->m_wind_cbox->Disable();
2474 if (selectedAtmModel->gust)
2475 m_xygribPanel->m_gust_cbox->Enable();
2477 m_xygribPanel->m_gust_cbox->Disable();
2479 if (selectedAtmModel->pressure)
2480 m_xygribPanel->m_pressure_cbox->Enable();
2482 m_xygribPanel->m_pressure_cbox->Disable();
2484 if (selectedAtmModel->temperature)
2485 m_xygribPanel->m_temperature_cbox->Enable();
2487 m_xygribPanel->m_temperature_cbox->Disable();
2489 if (selectedAtmModel->cape)
2490 m_xygribPanel->m_cape_cbox->Enable();
2492 m_xygribPanel->m_cape_cbox->Disable();
2494 if (selectedAtmModel->reflectivity)
2495 m_xygribPanel->m_reflectivity_cbox->Enable();
2497 m_xygribPanel->m_reflectivity_cbox->Disable();
2499 if (selectedAtmModel->cloudCover)
2500 m_xygribPanel->m_cloudcover_cbox->Enable();
2502 m_xygribPanel->m_cloudcover_cbox->Disable();
2504 if (selectedAtmModel->precipitation)
2505 m_xygribPanel->m_precipitation_cbox->Enable();
2507 m_xygribPanel->m_precipitation_cbox->Disable();
2509 if ((selectedWaveModel !=
nullptr) && (selectedWaveModel->significantHeight))
2510 m_xygribPanel->m_waveheight_cbox->Enable();
2512 m_xygribPanel->m_waveheight_cbox->Disable();
2514 if ((selectedWaveModel !=
nullptr) && (selectedWaveModel->windWaves))
2515 m_xygribPanel->m_windwave_cbox->Enable();
2517 m_xygribPanel->m_windwave_cbox->Disable();
2519 ApplyXyGribConfiguration();
2526void GribRequestSetting::ApplyXyGribConfiguration() {
2527 m_xygribPanel->m_atmmodel_choice->SetSelection(
2528 m_parent.xyGribConfig.atmModelIndex);
2529 m_xygribPanel->m_wavemodel_choice->SetSelection(
2530 m_parent.xyGribConfig.waveModelIndex);
2532 m_xygribPanel->m_wind_cbox->SetValue(m_parent.xyGribConfig.wind);
2533 m_xygribPanel->m_gust_cbox->SetValue(m_parent.xyGribConfig.gust);
2534 m_xygribPanel->m_pressure_cbox->SetValue(m_parent.xyGribConfig.pressure);
2535 m_xygribPanel->m_temperature_cbox->SetValue(
2536 m_parent.xyGribConfig.temperature);
2537 m_xygribPanel->m_cape_cbox->SetValue(m_parent.xyGribConfig.cape);
2538 m_xygribPanel->m_reflectivity_cbox->SetValue(
2539 m_parent.xyGribConfig.reflectivity);
2540 m_xygribPanel->m_cloudcover_cbox->SetValue(m_parent.xyGribConfig.cloudCover);
2541 m_xygribPanel->m_precipitation_cbox->SetValue(
2542 m_parent.xyGribConfig.precipitation);
2543 m_xygribPanel->m_waveheight_cbox->SetValue(m_parent.xyGribConfig.waveHeight);
2544 m_xygribPanel->m_windwave_cbox->SetValue(m_parent.xyGribConfig.windWaves);
2545 m_xygribPanel->m_resolution_choice->SetSelection(
2546 m_parent.xyGribConfig.resolutionIndex);
2547 m_xygribPanel->m_duration_choice->SetSelection(
2548 m_parent.xyGribConfig.durationIndex);
2549 m_xygribPanel->m_interval_choice->SetSelection(
2550 m_parent.xyGribConfig.intervalIndex);
2551 m_xygribPanel->m_run_choice->SetSelection(m_parent.xyGribConfig.runIndex);
2553 UpdateGribSizeEstimate();
2560void GribRequestSetting::MemorizeXyGribConfiguration() {
2561 m_parent.xyGribConfig.atmModelIndex =
2562 m_xygribPanel->m_atmmodel_choice->GetSelection();
2563 m_parent.xyGribConfig.waveModelIndex =
2564 m_xygribPanel->m_wavemodel_choice->GetSelection();
2566 m_parent.xyGribConfig.wind = m_xygribPanel->m_wind_cbox->IsChecked();
2567 m_parent.xyGribConfig.gust = m_xygribPanel->m_gust_cbox->IsChecked();
2568 m_parent.xyGribConfig.pressure = m_xygribPanel->m_pressure_cbox->IsChecked();
2569 m_parent.xyGribConfig.temperature =
2570 m_xygribPanel->m_temperature_cbox->IsChecked();
2571 m_parent.xyGribConfig.cape = m_xygribPanel->m_cape_cbox->IsChecked();
2572 m_parent.xyGribConfig.reflectivity =
2573 m_xygribPanel->m_reflectivity_cbox->IsChecked();
2574 m_parent.xyGribConfig.cloudCover =
2575 m_xygribPanel->m_cloudcover_cbox->IsChecked();
2576 m_parent.xyGribConfig.precipitation =
2577 m_xygribPanel->m_precipitation_cbox->IsChecked();
2578 m_parent.xyGribConfig.waveHeight =
2579 m_xygribPanel->m_waveheight_cbox->IsChecked();
2580 m_parent.xyGribConfig.windWaves = m_xygribPanel->m_windwave_cbox->IsChecked();
2582 m_parent.xyGribConfig.resolutionIndex =
2583 m_xygribPanel->m_resolution_choice->GetSelection();
2584 m_parent.xyGribConfig.durationIndex =
2585 m_xygribPanel->m_duration_choice->GetSelection();
2586 m_parent.xyGribConfig.intervalIndex =
2587 m_xygribPanel->m_interval_choice->GetSelection();
2588 m_parent.xyGribConfig.runIndex = m_xygribPanel->m_run_choice->GetSelection();
2590 UpdateGribSizeEstimate();
2596void GribRequestSetting::UpdateGribSizeEstimate() {
2601 if (!m_xygribPanel->m_resolution_choice->GetStringSelection().ToCDouble(
2603 m_xygribPanel->m_sizeestimate_text->SetLabel(
"Unknown");
2606 if (!m_xygribPanel->m_duration_choice->GetStringSelection().ToCLong(&days)) {
2607 m_xygribPanel->m_sizeestimate_text->SetLabel(
"Unknown");
2610 wxString intvStr = m_xygribPanel->m_interval_choice->GetStringSelection();
2611 intvStr.Replace(
"h",
"");
2612 if (!intvStr.ToCLong(&interval)) {
2613 m_xygribPanel->m_sizeestimate_text->SetLabel(
"Unknown");
2618 m_xygribPanel->m_sizeestimate_text->SetLabel(
"Unknown");
2627 int npts = (int)(ceil(fabs(xmax - xmin) / resolution) *
2628 ceil(fabs(ymax - ymin) / resolution));
2631 int nbrec = (int)days * 24 / interval + 1;
2633 int nbPress = (m_xygribPanel->m_pressure_cbox->IsChecked() &&
2634 m_xygribPanel->m_pressure_cbox->IsEnabled())
2637 int nbWind = (m_xygribPanel->m_wind_cbox->IsChecked() &&
2638 m_xygribPanel->m_wind_cbox->IsEnabled())
2641 int nbRain = (m_xygribPanel->m_precipitation_cbox->IsChecked() &&
2642 m_xygribPanel->m_precipitation_cbox->IsEnabled())
2645 int nbCloud = (m_xygribPanel->m_cloudcover_cbox->IsChecked() &&
2646 m_xygribPanel->m_cloudcover_cbox->IsEnabled())
2649 int nbTemp = (m_xygribPanel->m_temperature_cbox->IsChecked() &&
2650 m_xygribPanel->m_temperature_cbox->IsEnabled())
2654 int nbCAPEsfc = (m_xygribPanel->m_cape_cbox->IsChecked() &&
2655 m_xygribPanel->m_cape_cbox->IsEnabled())
2658 int nbReflectivity = (m_xygribPanel->m_reflectivity_cbox->IsChecked() &&
2659 m_xygribPanel->m_reflectivity_cbox->IsEnabled())
2662 int nbGUSTsfc = (m_xygribPanel->m_gust_cbox->IsChecked() &&
2663 m_xygribPanel->m_gust_cbox->IsEnabled())
2673 estimate += nbWind * (head + (nbits * npts) / 8 + 2);
2676 estimate += nbTemp * (head + (nbits * npts) / 8 + 2);
2682 estimate += nbRain * (head + 24 + (nbits * npts) / 8 + 2);
2685 estimate += nbPress * (head + (nbits * npts) / 8 + 2);
2688 estimate += nbCloud * (head + 24 + (nbits * npts) / 8 + 2);
2691 estimate += nbReflectivity * (head + (nbits * npts) / 8 + 2);
2694 estimate += nbCAPEsfc * (head + (nbits * npts) / 8 + 2);
2697 estimate += nbGUSTsfc * (head + (nbits * npts) / 8 + 2);
2701 int atmEstimate = estimate;
2705 if (m_xygribPanel->m_wavemodel_choice->GetStringSelection().IsSameAs(
2708 npts = (int)(ceil(fabs(xmax - xmin) / 0.5) * ceil(fabs(ymax - ymin) / 0.5));
2709 nbrec = (int)fmin(8, days) * 24 / interval + 1;
2710 }
else if (m_xygribPanel->m_wavemodel_choice->GetStringSelection().IsSameAs(
2714 (int)(ceil(fabs(xmax - xmin) / 0.25) * ceil(fabs(ymax - ymin) / 0.25));
2715 nbrec = (int)fmin(8, days) * 24 / interval + 1;
2716 }
else if (m_xygribPanel->m_wavemodel_choice->GetStringSelection().IsSameAs(
2720 (int)(ceil(fabs(xmax - xmin) / 0.05) * ceil(fabs(ymax - ymin) / 0.1));
2721 nbrec = (int)fmin(4, days) * 24 / interval + 1;
2727 if (m_xygribPanel->m_waveheight_cbox->IsChecked()) {
2729 estimate += nbrec * (head + (nbits * npts) / 8 + 2);
2732 if (m_xygribPanel->m_windwave_cbox->IsChecked()) {
2734 estimate += nbrec * (head + (nbits * npts) / 8 + 2);
2736 estimate += nbrec * (head + (nbits * npts) / 8 + 2);
2738 estimate += nbrec * (head + (nbits * npts) / 8 + 2);
2741 int wavEstimate = estimate - atmEstimate;
2743 wavEstimate = wavEstimate * 0.6;
2745 estimate = atmEstimate + wavEstimate;
2748 estimate = estimate * 0.65;
2750 wxString warningStr =
"";
2751 if (estimate / (1024 * 1024) > XYGRIB_MAX_DOWNLOADABLE_GRIB_SIZE_MB) {
2752 warningStr = wxString::Format(
"(Warning GRIB exceeds %d MB limit)",
2753 XYGRIB_MAX_DOWNLOADABLE_GRIB_SIZE_MB);
2756 m_xygribPanel->m_sizeestimate_text->SetLabel(
2757 wxString::Format(
"%d kB %s", estimate / 1024, warningStr));
2759 m_gribSizeEstimate = estimate;
GRIB Data Visualization and Rendering Factory.
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 Request and Download Management.
@ WORLD
Global forecast downloads (e.g., GFS)
@ XYGRIB
Downloads from XyGrib service.
@ NONE
No download source selected.
@ LOCAL_CATALOG
Downloads from configured catalogs.
@ LOCAL
Downloads from local sources.
#define SAVEDZONE
Save zone.
#define MANSELECT
Manual selection mode.
#define AUTOSELECT
Automatic selection mode.
@ 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.
@ COMPLETE_SELECTION
Selection box completed in manual mode, coordinates have been captured after the user has released th...
@ SAVED_SELECTION
Area loaded from previously saved coordinates.
XyGrib Model Configuration and Definitions.
void SetRequestButtonBitmap(int type)
Set the icon and tooltip for the download request button.
wxString m_grib_dir
Directory containing GRIB files.
grib_pi * pPlugIn
Plugin instance that owns this control bar.
wxArrayString m_file_names
List of GRIB filenames being displayed.
Class GribRequestSettingBase.
wxButton * m_rButtonCancel
Button to Cancel a request to download, close the dialog without saving the configuration.
wxSpinCtrl * m_spMaxLon
A spinner for the max longitude of the bounding box for downloads.
wxSpinCtrl * m_spMaxLat
A spinner for the max latitude of the bounding box for downloads.
wxRadioButton * m_rbManualSelect
Radio button selected to indicate the download area is based on the area selected by the user.
wxRadioButton * m_rbCurrentView
Radio button selected to indicate the download area is based on the visible area of the chart in the ...
wxSpinCtrl * m_spMinLat
A spinner for the min latitude of the bounding box for downloads.
wxButton * m_rButtonApply
Button to Save the "download request" configuration.
wxButton * m_rButtonYes
Button to Send a download request through e-mail.
wxSpinCtrl * m_spMinLon
A spinner for the min longitude of the bounding box for downloads.
Manages GRIB file request configuration and downloads.
double m_StartLat
The latitude at the starting point of the bounding box.
double GetMinLat() const
Get the minimum latitude of the bounding box for the download request.
double GetMaxLat() const
Get the maximum latitude of the bounding box for the download request.
double m_Lat
The latitude at the mouse cursor while drawing a bounding box.
void OnVpUnderMouseChange(PlugIn_ViewPort *vp)
Callback invoked when the view port under mouse has changed.
bool RenderGlZoneOverlay()
Renders the GRIB area selection overlay using OpenGL.
bool RenderZoneOverlay(wxDC &dc)
Renders the GRIB area selection overlay using standard device context.
double m_Lon
The longitude at the mouse cursor while drawing a bounding box.
wxPoint m_StartPoint
Starting point of the bounding box in physical pixels.
bool DoRenderZoneOverlay()
Draws the GRIB area selection overlay on the chart.
bool MouseEventHook(wxMouseEvent &event)
Intercepts mouse events to handle GRIB area selection.
void OnVpWithFocusChange(PlugIn_ViewPort *vp)
Callback invoked when the focused view port has changed, such as in multi-chart mode when user switch...
PlugIn_ViewPort * m_VpMouse
The viewport under the mouse.
ZoneSelectionRenderState m_RenderSelectionZoneState
Current state of the bounding box overlay rendering.
double GetMaxLon() const
Get the maximum longitude of the bounding box for the download request.
PlugIn_ViewPort * m_VpFocus
The viewport currently in focus.
double m_StartLon
The longitude at the starting point of the bounding box.
double GetMinLon() const
Get the minimum longitude of the bounding box for the download request.
Contains view parameters and status information for a chart display viewport.
double lon_max
Maximum longitude of the viewport.
double lat_max
Maximum latitude of the viewport.
double lon_min
Minimum longitude of the viewport.
double lat_min
Minimum latitude of the viewport.
int Parse(const wxString &doc, wxJSONValue *val)
Parse the JSON document.
The JSON value class implementation.
int Size() const
Return the size of the array or map stored in this value.
bool HasMember(unsigned index) const
Return TRUE if the object contains an element at the specified index.
wxString AsString() const
Return the stored value as a wxWidget's string.
Email Request System for GRIB Data.
GRIB Weather Data Plugin for OpenCPN.
@ OCPN_DL_EVENT_TYPE_PROGRESS
Download progress update.
@ OCPN_DL_EVENT_TYPE_END
Download has completed.
@ OCPN_DL_NO_ERROR
Download completed successfully.
wxFont * OCPNGetFont(wxString TextElement, int default_size)
Gets a font for UI elements.
int GetCanvasIndexUnderMouse(void)
Gets index of chart canvas under mouse cursor.
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.
double OCPN_GetWinDIPScaleFactor()
Gets Windows-specific DPI scaling factor.
void RequestRefresh(wxWindow *win)
Requests window refresh.
wxFileConfig * GetOCPNConfigObject(void)
Gets OpenCPN's configuration object.
void GetCanvasLLPix(PlugIn_ViewPort *vp, wxPoint p, double *plat, double *plon)
Converts canvas physical pixel coordinates to lat/lon.
OpenGL Platform Abstraction Layer.