31#include <wx/glcanvas.h>
32#include <wx/graphics.h>
33#include <wx/progdlg.h>
35#include "pi_shaders.h"
46extern double g_ContentScaleFactor;
47float g_piGLMinSymbolLineWidth = 0.9;
49enum GRIB_OVERLAP { _GIN, _GON, _GOUT };
56 double lat_max,
double lon_min,
double lon_max,
58 if (((vp->
lon_min - Marge) > (lon_max + Marge)) ||
59 ((vp->
lon_max + Marge) < (lon_min - Marge)) ||
60 ((vp->
lat_max + Marge) < (lat_min - Marge)) ||
61 ((vp->
lat_min - Marge) > (lat_max + Marge)))
77 if (y < m_miny || y > m_maxy)
return FALSE;
82 if (x < m_maxx - 360.)
84 else if (x > m_minx + 360.)
87 if (x < m_minx || x > m_maxx)
return FALSE;
93static wxString MToString(
int DataCenterModel )
95 switch( DataCenterModel ) {
96 case NOAA_GFS:
return _T(
"NOAA_GFS");
97 case NOAA_NCEP_WW3:
return _T(
"NOAA_NCEP_WW3");
98 case NOAA_NCEP_SST:
return _T(
"NOAA_NCEP_SST");
99 case NOAA_RTOFS:
return _T(
"NOAA_RTOFS");
100 case FNMOC_WW3_GLB:
return _T(
"FNMOC_WW3");
101 case FNMOC_WW3_MED:
return _T(
"FNMOC_WW3");
102 case NORWAY_METNO:
return _T(
"NORWAY_METNO");
103 default :
return _T(
"OTHER_DATA_CENTER");
109static GLuint texture_format = 0;
113static GLboolean QueryExtension(
const char *extName )
125 extNameLen = strlen( extName );
127 p = (
char *) glGetString( GL_EXTENSIONS );
132 end = p + strlen( p );
135 int n = strcspn( p,
" " );
136 if( ( extNameLen == n ) && ( strncmp( extName, p, n ) == 0 ) ) {
144#if defined(__WXMSW__)
145#define systemGetProcAddress(ADDR) wglGetProcAddress(ADDR)
146#elif defined(__WXOSX__)
148#define systemGetProcAddress(ADDR) dlsym(RTLD_DEFAULT, ADDR)
150#define systemGetProcAddress(ADDR) glXGetProcAddress((const GLubyte *)ADDR)
155void LineBuffer::pushLine(
float x0,
float y0,
float x1,
float y1) {
156 buffer.push_back(x0);
157 buffer.push_back(y0);
158 buffer.push_back(x1);
159 buffer.push_back(y1);
162void LineBuffer::pushPetiteBarbule(
int b,
int l) {
163 int tilt = (l * 100) / 250;
164 pushLine(b, 0, b + tilt, -l);
167void LineBuffer::pushGrandeBarbule(
int b,
int l) {
168 int tilt = (l * 100) / 250;
169 pushLine(b, 0, b + tilt, -l);
172void LineBuffer::pushTriangle(
int b,
int l) {
173 int dim = (l * 100) / 250;
174 pushLine(b, 0, b + dim, -l);
175 pushLine(b + (dim * 2), 0, b + dim, -l);
178void LineBuffer::Finalize() {
179 count = buffer.size() / 4;
180 lines =
new float[buffer.size()];
182 for (std::list<float>::iterator it = buffer.begin(); it != buffer.end(); it++)
186int adjustSpacing(
int dialogSetSpacing) {
187#ifdef __OCPN__ANDROID__
192 int sizeMin = wxMin(sz.x, sz.y);
193 int space = ((double)dialogSetSpacing) * (sizeMin / 2) / 100;
198 return dialogSetSpacing;
206 : m_dlg(dlg), m_Settings(dlg.m_OverlaySettings) {
207 if (wxGetDisplaySize().x > 0) {
214 wxMax(wxGetDisplaySize().x, wxGetDisplaySize().y);
216 m_pixelMM = wxMax(.02, m_pixelMM);
222 m_pGribTimelineRecordSet =
nullptr;
223 m_last_vp_scale = 0.;
226#if wxUSE_GRAPHICS_CONTEXT
229 m_Font_Message =
nullptr;
232 for (
int i = 0; i < GribOverlaySettings::SETTINGS_COUNT; i++)
233 m_pOverlay[i] =
nullptr;
235 m_ParticleMap =
nullptr;
236 m_tParticleTimer.Connect(
237 wxEVT_TIMER, wxTimerEventHandler(GRIBOverlayFactory::OnParticleTimer),
239 m_bUpdateParticles =
false;
243 if (m_pixelMM < 0.2) {
244 windArrowSize = 5.0 / m_pixelMM;
245 windArrowSize = wxMin(
246 windArrowSize, wxMax(wxGetDisplaySize().x, wxGetDisplaySize().y) / 20);
251 double s = 2 * M_PI / 10.;
252 for (
double a = 0; a < 2 * M_PI; a += s)
253 m_WindArrowCache[0].pushLine(r * sin(a), r * cos(a), r * sin(a + s),
256 int dec = -windArrowSize / 2;
257 int pointerLength = windArrowSize / 3;
260 for (i = 1; i < 14; i++) {
263 arrow.pushLine(dec, 0, dec + windArrowSize, 0);
264 arrow.pushLine(dec, 0, dec + pointerLength, pointerLength / 2);
265 arrow.pushLine(dec, 0, dec + pointerLength,
266 -(pointerLength / 2));
269 int featherPosition = windArrowSize / 6;
272 dec + windArrowSize - featherPosition;
273 int b2 = dec + windArrowSize;
275 int lpetite = windArrowSize / 5;
276 int lgrande = lpetite * 2;
279 m_WindArrowCache[1].pushPetiteBarbule(b1, lpetite);
281 m_WindArrowCache[2].pushGrandeBarbule(b2, lgrande);
283 m_WindArrowCache[3].pushGrandeBarbule(b2, lgrande);
284 m_WindArrowCache[3].pushPetiteBarbule(b2 - featherPosition, lpetite);
286 m_WindArrowCache[4].pushGrandeBarbule(b2, lgrande);
287 m_WindArrowCache[4].pushGrandeBarbule(b2 - featherPosition, lgrande);
289 m_WindArrowCache[5].pushGrandeBarbule(b2, lgrande);
290 m_WindArrowCache[5].pushGrandeBarbule(b2 - featherPosition, lgrande);
291 m_WindArrowCache[5].pushPetiteBarbule(b2 - featherPosition * 2, lpetite);
293 m_WindArrowCache[6].pushGrandeBarbule(b2, lgrande);
294 m_WindArrowCache[6].pushGrandeBarbule(b2 - featherPosition, lgrande);
295 m_WindArrowCache[6].pushGrandeBarbule(b2 - featherPosition * 2, lgrande);
297 m_WindArrowCache[7].pushGrandeBarbule(b2, lgrande);
298 m_WindArrowCache[7].pushGrandeBarbule(b2 - featherPosition, lgrande);
299 m_WindArrowCache[7].pushGrandeBarbule(b2 - featherPosition * 2, lgrande);
300 m_WindArrowCache[7].pushPetiteBarbule(b2 - featherPosition * 3, lpetite);
302 m_WindArrowCache[8].pushGrandeBarbule(b2, lgrande);
303 m_WindArrowCache[8].pushGrandeBarbule(b2 - featherPosition, lgrande);
304 m_WindArrowCache[8].pushGrandeBarbule(b2 - featherPosition * 2, lgrande);
305 m_WindArrowCache[8].pushGrandeBarbule(b2 - featherPosition * 3, lgrande);
307 m_WindArrowCache[9].pushTriangle(b1 - featherPosition, lgrande);
309 m_WindArrowCache[10].pushTriangle(b1 - featherPosition, lgrande);
310 m_WindArrowCache[10].pushGrandeBarbule(b1 - featherPosition * 2, lgrande);
312 m_WindArrowCache[11].pushTriangle(b1 - featherPosition, lgrande);
313 m_WindArrowCache[11].pushGrandeBarbule(b1 - featherPosition * 2, lgrande);
314 m_WindArrowCache[11].pushGrandeBarbule(b1 - featherPosition * 3, lgrande);
316 m_WindArrowCache[12].pushTriangle(b1 - featherPosition, lgrande);
317 m_WindArrowCache[12].pushGrandeBarbule(b1 - featherPosition * 2, lgrande);
318 m_WindArrowCache[12].pushGrandeBarbule(b1 - featherPosition * 3, lgrande);
319 m_WindArrowCache[12].pushGrandeBarbule(b1 - featherPosition * 4, lgrande);
321 m_WindArrowCache[13].pushTriangle(b1 - featherPosition, lgrande);
322 m_WindArrowCache[13].pushTriangle(b1 - featherPosition * 3, lgrande);
324 for (i = 0; i < 14; i++) m_WindArrowCache[i].Finalize();
327 for (
int i = 0; i < 2; i++) {
333 if (m_pixelMM > 0.2) {
334 arrowSize = 5.0 / m_pixelMM;
336 arrowSize, wxMax(wxGetDisplaySize().x, wxGetDisplaySize().y) / 20);
337 dec1 = arrowSize / 6;
338 dec2 = arrowSize / 8;
344 dec = -arrowSize / 2;
346 m_SingleArrow[i].pushLine(dec, 0, dec + arrowSize, 0);
347 m_SingleArrow[i].pushLine(dec - 2, 0, dec + dec1, dec1 + 1);
348 m_SingleArrow[i].pushLine(dec - 2, 0, dec + dec1, -(dec1 + 1));
349 m_SingleArrow[i].Finalize();
351 m_DoubleArrow[i].pushLine(dec, -dec2, dec + arrowSize, -dec2);
352 m_DoubleArrow[i].pushLine(dec, dec2, dec + arrowSize, +dec2);
354 m_DoubleArrow[i].pushLine(dec - 2, 0, dec + dec1, dec1 + 1);
355 m_DoubleArrow[i].pushLine(dec - 2, 0, dec + dec1, -(dec1 + 1));
356 m_DoubleArrow[i].Finalize();
360GRIBOverlayFactory::~GRIBOverlayFactory() {
365 if (m_oDC)
delete m_oDC;
366 if (m_Font_Message)
delete m_Font_Message;
369void GRIBOverlayFactory::Reset() {
370 m_pGribTimelineRecordSet =
nullptr;
375void GRIBOverlayFactory::SetMessageFont() {
384 if (m_Font_Message)
delete m_Font_Message;
385 m_Font_Message =
new wxFont(fo);
388void GRIBOverlayFactory::SetGribTimelineRecordSet(
391 m_pGribTimelineRecordSet = pGribTimelineRecordSet;
394void GRIBOverlayFactory::ClearCachedData(
void) {
396 for (
int i = 0; i < GribOverlaySettings::SETTINGS_COUNT; i++) {
397 delete m_pOverlay[i];
398 m_pOverlay[i] =
nullptr;
402#ifdef __OCPN__ANDROID__
403#include "pi_shaders.h"
406bool GRIBOverlayFactory::RenderGLGribOverlay(wxGLContext *pcontext,
408 if (g_bpause)
return false;
412 if (!m_oDC || !m_oDC->UsesGL()) {
419#ifndef USE_ANDROID_GLES2
420 glGetIntegerv(GL_SMOOTH_LINE_WIDTH_RANGE, &parms[0]);
422 glGetIntegerv(GL_ALIASED_LINE_WIDTH_RANGE, &parms[0]);
424 g_piGLMinSymbolLineWidth = wxMax(parms[0], 1);
430 m_oDC->SetDC(
nullptr);
434 bool rv = DoRenderGribOverlay(vp);
441bool GRIBOverlayFactory::RenderGribOverlay(wxDC &dc,
PlugIn_ViewPort *vp) {
442 if (!m_oDC || m_oDC->UsesGL()) {
454#if wxUSE_GRAPHICS_CONTEXT
456 pmdc =
dynamic_cast<wxMemoryDC*
>(&dc);
457 wxGraphicsContext *pgc = wxGraphicsContext::Create( *pmdc );
462 bool rv = DoRenderGribOverlay(vp);
467void GRIBOverlayFactory::SettingsIdToGribId(
int i,
int &idx,
int &idy,
472 case GribOverlaySettings::WIND:
475 case GribOverlaySettings::WIND_GUST:
480 case GribOverlaySettings::PRESSURE:
485 case GribOverlaySettings::WAVE:
490 case GribOverlaySettings::CURRENT:
495 case GribOverlaySettings::PRECIPITATION:
500 case GribOverlaySettings::CLOUD:
505 case GribOverlaySettings::AIR_TEMPERATURE:
510 case GribOverlaySettings::SEA_TEMPERATURE:
515 case GribOverlaySettings::CAPE:
520 case GribOverlaySettings::COMP_REFL:
529 if (!m_pGribTimelineRecordSet) {
537 m_TexFontNumbers.Build(*m_Font_Message);
539 if (m_oDC) m_oDC->SetFont(*m_Font_Message);
542 m_Message_Hiden.Empty();
545 if (m_pdc && vp->
view_scale_ppm != m_last_vp_scale) ClearCachedData();
551 wxArrayPtrVoid **pIA = m_pGribTimelineRecordSet->
m_IsobarArray;
553 for (
int overlay = 1; overlay >= 0; overlay--) {
554 for (
int i = 0; i < GribOverlaySettings::SETTINGS_COUNT; i++) {
555 if (i == GribOverlaySettings::WIND) {
557 if (m_dlg.m_bDataPlot[i]) RenderGribOverlayMap(i, pGR, vp);
559 if (m_dlg.m_bDataPlot[i]) {
560 RenderGribBarbedArrows(i, pGR, vp);
561 RenderGribIsobar(i, pGR, pIA, vp);
562 RenderGribNumbers(i, pGR, vp);
563 RenderGribParticles(i, pGR, vp);
565 if (m_Settings.Settings[i].m_iBarbedVisibility)
566 RenderGribBarbedArrows(i, pGR, vp);
571 if (i == GribOverlaySettings::PRESSURE) {
573 if (m_dlg.m_bDataPlot[i]) {
574 RenderGribIsobar(i, pGR, pIA, vp);
575 RenderGribNumbers(i, pGR, vp);
577 if (m_Settings.Settings[i].m_iIsoBarVisibility)
578 RenderGribIsobar(i, pGR, pIA, vp);
583 if (m_dlg.InDataPlot(i) && !m_dlg.m_bDataPlot[i])
continue;
586 RenderGribOverlayMap(i, pGR, vp);
588 RenderGribBarbedArrows(i, pGR, vp);
589 RenderGribIsobar(i, pGR, pIA, vp);
590 RenderGribDirectionArrows(i, pGR, vp);
591 RenderGribNumbers(i, pGR, vp);
592 RenderGribParticles(i, pGR, vp);
597 if (!m_Message_Hiden.IsEmpty()) m_Message_Hiden.Append(_T(
"\n"));
598 m_Message_Hiden.Append(_(
"Warning : Data at Geopotential Height"))
600 .Append(m_Settings.GetAltitudeFromIndex(
602 m_Settings.Settings[GribOverlaySettings::PRESSURE].m_Units))
604 .Append(m_Settings.GetUnitSymbol(GribOverlaySettings::PRESSURE))
607 if (m_dlg.ProjectionEnabled()) {
610 DrawProjectedPosition(x, y);
612 if (!m_Message_Hiden.IsEmpty()) m_Message_Hiden.Append(_T(
"\n"));
613 m_Message_Hiden.Append(m_Message);
617 if (m_dlg.m_highlight_latmax - m_dlg.m_highlight_latmin > 0.01 &&
618 m_dlg.m_highlight_lonmax - m_dlg.m_highlight_lonmin > 0.01) {
620 GetCanvasPixLL(vp, &p1, m_dlg.m_highlight_latmin, m_dlg.m_highlight_lonmin);
621 GetCanvasPixLL(vp, &p2, m_dlg.m_highlight_latmax, m_dlg.m_highlight_lonmax);
623 m_pdc->SetPen(wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT)));
625 wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT),
626 wxBRUSHSTYLE_CROSSDIAG_HATCH));
627 m_pdc->DrawRectangle(p1.x, p1.y, p2.x - p1.x, p2.y - p1.y);
631 m_oDC->SetPen(wxPen(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT)));
633 wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT),
634 wxBRUSHSTYLE_CROSSDIAG_HATCH));
635 m_oDC->DrawRectangle(p1.x, p1.y, p2.x - p1.x, p2.y - p1.y);
643static inline bool isClearSky(
int settings,
double v) {
644 return ((settings == GribOverlaySettings::PRECIPITATION) ||
645 (settings == GribOverlaySettings::CLOUD)) &&
650void GRIBOverlayFactory::GetCalibratedGraphicColor(
int settings,
double val_in,
651 unsigned char *data) {
652 unsigned char r, g, b, a;
653 a = m_Settings.m_iOverlayTransparency;
655 if (val_in != GRIB_NOTDEF) {
656 val_in = m_Settings.CalibrateValue(settings, val_in);
659 if ((settings == GribOverlaySettings::PRECIPITATION ||
660 settings == GribOverlaySettings::CLOUD) &&
663 if ((settings == GribOverlaySettings::COMP_REFL) && val_in < 5) a = 0;
665 GetGraphicColor(settings, val_in, r, g, b);
667 r = 255, g = 255, b = 255, a = 0;
675bool GRIBOverlayFactory::CreateGribGLTexture(
GribOverlay *pGO,
int settings,
678 pGR->getLonMin() == 0 && pGR->getLonMax() + pGR->
getDi() >= 360.;
681 int tw, th, samples = 1;
684 if (pGR->
getNi() > 1024 || pGR->
getNj() > 1024) {
690 dw = (tw > 1022) ? 1022. / tw : 1.;
691 dh = (th > 1022) ? 1022. / th : 1.;
692 delta = wxMin(dw, dh);
700 tw = samples * (pGR->
getNi() - 1) + 1 + 2 * !repeat;
701 th = samples * (pGR->
getNj() - 1) + 1 + 2;
702 if (tw >= 512 || th >= 512 || samples == 16)
break;
707 if (tw > 1024 || th > 1024)
return false;
709 pGO->m_iTexDataDim[0] = tw;
710 pGO->m_iTexDataDim[1] = th;
712#ifdef USE_ANDROID_GLES2
721 if (((xp != 0) && !(xp & (xp - 1))))
733 if (((xp != 0) && !(xp & (xp - 1))))
749 unsigned char *data =
new unsigned char[tw * th * 4];
751 for (
int j = 0; j < pGR->
getNj(); j++) {
752 for (
int i = 0; i < pGR->
getNi(); i++) {
754 int y = (j + 1) * delta;
755 int x = (i + !repeat) * delta;
756 int doff = 4 * (y * tw + x);
757 GetCalibratedGraphicColor(settings, v, data + doff);
760 }
else if (samples == 1) {
761 for (
int j = 0; j < pGR->
getNj(); j++) {
762 for (
int i = 0; i < pGR->
getNi(); i++) {
766 int doff = 4 * (y * tw + x);
767 GetCalibratedGraphicColor(settings, v, data + doff);
771 for (
int j = 0; j < pGR->
getNj(); j++) {
772 for (
int i = 0; i < pGR->
getNi(); i++) {
773 double v00 = pGR->
getValue(i, j), v01 = GRIB_NOTDEF;
774 double v10 = GRIB_NOTDEF, v11 = GRIB_NOTDEF;
775 if (i < pGR->getNi() - 1) {
777 if (j < pGR->getNj() - 1) v11 = pGR->
getValue(i + 1, j + 1);
779 if (j < pGR->getNj() - 1) v10 = pGR->
getValue(i, j + 1);
781 for (
int ys = 0; ys < samples; ys++) {
782 int y = j * samples + ys + 1;
783 double yd = (double)ys / samples;
785 double a0 = 1, a1 = 1;
786 if (v10 == GRIB_NOTDEF) {
788 if (v00 == GRIB_NOTDEF)
792 }
else if (v00 == GRIB_NOTDEF)
795 v0 = (1 - yd) * v00 + yd * v10;
796 if (v11 == GRIB_NOTDEF) {
798 if (v01 == GRIB_NOTDEF)
802 }
else if (v01 == GRIB_NOTDEF)
805 v1 = (1 - yd) * v01 + yd * v11;
807 for (
int xs = 0; xs < samples; xs++) {
808 int x = i * samples + xs + !repeat;
809 double xd = (double)xs / samples;
811 if (v1 == GRIB_NOTDEF)
812 v = v0, a = (1 - xd) * a0;
813 else if (v0 == GRIB_NOTDEF)
816 v = (1 - xd) * v0 + xd * v1;
817 a = (1 - xd) * a0 + xd * a1;
820 int doff = 4 * (y * tw + x);
821 GetCalibratedGraphicColor(settings, v, data + doff);
824 if (i == pGR->
getNi() - 1)
break;
826 if (j == pGR->
getNj() - 1)
break;
833 memcpy(data, data + 4 * tw * 1, 4 * tw);
834 memcpy(data + 4 * tw * (th - 1), data + 4 * tw * (th - 2), 4 * tw);
835 for (
int x = 0; x < tw; x++) {
838 doff = 4 * ((th - 1) * tw + x);
843 for (
int y = 0; y < th; y++) {
844 int doff = 4 * y * tw, soff = doff + 4;
845 memcpy(data + doff, data + soff, 4);
847 doff = 4 * (y * tw + tw - 1), soff = doff - 4;
848 memcpy(data + doff, data + soff, 4);
853 glGenTextures(1, &texture);
854 glBindTexture(texture_format, texture);
856 glTexParameteri(texture_format, GL_TEXTURE_WRAP_S,
857 repeat ? GL_REPEAT : GL_CLAMP_TO_EDGE);
858 glTexParameteri(texture_format, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
859 glTexParameteri(texture_format, GL_TEXTURE_MAG_FILTER, GL_LINEAR);
860 glTexParameteri(texture_format, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
863 glPushClientAttrib(GL_CLIENT_PIXEL_STORE_BIT);
865 glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
866 glPixelStorei(GL_UNPACK_SKIP_PIXELS, 0);
867 glPixelStorei(GL_UNPACK_SKIP_ROWS, 0);
868 glPixelStorei(GL_UNPACK_ROW_LENGTH, tw);
870 glTexImage2D(texture_format, 0, GL_RGBA, tw, th, 0, GL_RGBA, GL_UNSIGNED_BYTE,
875 glTexImage2D(texture_format, 0, GL_RGBA, tw, th, 0, GL_RGBA, GL_UNSIGNED_BYTE,
881 pGO->m_iTexture = texture;
882 pGO->m_iTextureDim[0] = tw;
883 pGO->m_iTextureDim[1] = th;
889wxImage GRIBOverlayFactory::CreateGribImage(
int settings,
GribRecord *pGR,
892 const wxPoint &porg) {
898 int width = abs(pmax.x - pmin.x);
899 int height = abs(pmax.y - pmin.y);
903 if (width > m_ParentSize.GetWidth() || height > m_ParentSize.GetHeight())
907 wxImage gr_image(width, height);
908 gr_image.InitAlpha();
911 for (
int ipix = 0; ipix < (width - grib_pixel_size + 1);
912 ipix += grib_pixel_size) {
913 for (
int jpix = 0; jpix < (height - grib_pixel_size + 1);
914 jpix += grib_pixel_size) {
921 if (v != GRIB_NOTDEF) {
922 v = m_Settings.CalibrateValue(settings, v);
923 wxColour c = GetGraphicColor(settings, v);
927 isClearSky(settings, v) ? 0 : m_Settings.m_iOverlayTransparency;
929 unsigned char r = c.Red();
930 unsigned char g = c.Green();
931 unsigned char b = c.Blue();
933 for (
int xp = 0; xp < grib_pixel_size; xp++)
934 for (
int yp = 0; yp < grib_pixel_size; yp++) {
935 gr_image.SetRGB(ipix + xp, jpix + yp, r, g, b);
936 gr_image.SetAlpha(ipix + xp, jpix + yp, a);
939 for (
int xp = 0; xp < grib_pixel_size; xp++)
940 for (
int yp = 0; yp < grib_pixel_size; yp++)
941 gr_image.SetAlpha(ipix + xp, jpix + yp, 0);
946 return gr_image.Blur(4);
958 {0, _T(
"#d90000")}, {1, _T(
"#d92a00")}, {2, _T(
"#d96e00")},
959 {3, _T(
"#d9b200")}, {4, _T(
"#d4d404")}, {5, _T(
"#a6d906")},
960 {7, _T(
"#06d9a0")}, {9, _T(
"#00d9b0")}, {12, _T(
"#00d9c0")},
961 {15, _T(
"#00aed0")}, {18, _T(
"#0083e0")}, {21, _T(
"#0057e0")},
962 {24, _T(
"#0000f0")}, {27, _T(
"#0400f0")}, {30, _T(
"#1c00f0")},
963 {36, _T(
"#4800f0")}, {42, _T(
"#6900f0")}, {48, _T(
"#a000f0")},
964 {56, _T(
"#f000f0")}};
967 {0, _T(
"#00d900")}, {1, _T(
"#2ad900")}, {2, _T(
"#6ed900")},
968 {3, _T(
"#b2d900")}, {4, _T(
"#d4d400")}, {5, _T(
"#d9a600")},
969 {7, _T(
"#d90000")}, {9, _T(
"#d90040")}, {12, _T(
"#d90060")},
970 {15, _T(
"#ae0080")}, {18, _T(
"#8300a0")}, {21, _T(
"#5700c0")},
971 {24, _T(
"#0000d0")}, {27, _T(
"#0400e0")}, {30, _T(
"#0800e0")},
972 {36, _T(
"#a000e0")}, {42, _T(
"#c004c0")}, {48, _T(
"#c008a0")},
973 {56, _T(
"#c0a008")}};
977 {0, _T(
"#288CFF")}, {3, _T(
"#00AFFF")}, {6, _T(
"#00DCE1")},
978 {9, _T(
"#00F7B0")}, {12, _T(
"#00EA9C")}, {15, _T(
"#82F059")},
979 {18, _T(
"#F0F503")}, {21, _T(
"#FFED00")}, {24, _T(
"#FFDB00")},
980 {27, _T(
"#FFC700")}, {30, _T(
"#FFB400")}, {33, _T(
"#FF9800")},
981 {36, _T(
"#FF7E00")}, {39, _T(
"#F77800")}, {42, _T(
"#EC7814")},
982 {45, _T(
"#E4711E")}, {48, _T(
"#E06128")}, {51, _T(
"#DC5132")},
983 {54, _T(
"#D5453C")}, {57, _T(
"#CD3A46")}, {60, _T(
"#BE2C50")},
984 {63, _T(
"#B41A5A")}, {66, _T(
"#AA1464")}, {70, _T(
"#962878")},
985 {75, _T(
"#8C328C")}};
989 {0, _T(
"#283282")}, {5, _T(
"#273c8c")}, {10, _T(
"#264696")},
990 {14, _T(
"#2350a0")}, {18, _T(
"#1f5aaa")}, {22, _T(
"#1a64b4")},
991 {26, _T(
"#136ec8")}, {29, _T(
"#0c78e1")}, {32, _T(
"#0382e6")},
992 {35, _T(
"#0091e6")}, {38, _T(
"#009ee1")}, {41, _T(
"#00a6dc")},
993 {44, _T(
"#00b2d7")}, {47, _T(
"#00bed2")}, {50, _T(
"#28c8c8")},
994 {53, _T(
"#78d2aa")}, {56, _T(
"#8cdc78")}, {59, _T(
"#a0eb5f")},
995 {62, _T(
"#c8f550")}, {65, _T(
"#f3fb02")}, {68, _T(
"#ffed00")},
996 {71, _T(
"#ffdd00")}, {74, _T(
"#ffc900")}, {78, _T(
"#ffab00")},
997 {82, _T(
"#ff8100")}, {86, _T(
"#f1780c")}, {90, _T(
"#e26a23")},
998 {95, _T(
"#d5453c")}, {100, _T(
"#b53c59")}};
1003 {-2, _T(
"#cc04ae")}, {2, _T(
"#8f06e4")}, {6, _T(
"#486afa")},
1004 {10, _T(
"#00ffff")}, {15, _T(
"#00d54b")}, {19, _T(
"#59d800")},
1005 {23, _T(
"#f2fc00")}, {27, _T(
"#ff1500")}, {32, _T(
"#ff0000")},
1006 {36, _T(
"#d80000")}, {40, _T(
"#a90000")}, {44, _T(
"#870000")},
1007 {48, _T(
"#690000")}, {52, _T(
"#550000")}, {56, _T(
"#330000")}};
1010static ColorMap PrecipitationMap[] = {
1011 {0, _T(
"#ffffff")}, {.01, _T(
"#c8f0ff")}, {.02, _T(
"#b4e6ff")},
1012 {.05, _T(
"#8cd3ff")}, {.07, _T(
"#78caff")}, {.1, _T(
"#6ec1ff")},
1013 {.2, _T(
"#64b8ff")}, {.5, _T(
"#50a6ff")}, {.7, _T(
"#469eff")},
1014 {1.0, _T(
"#3c96ff")}, {2.0, _T(
"#328eff")}, {5.0, _T(
"#1e7eff")},
1015 {7.0, _T(
"#1476f0")}, {10, _T(
"#0a6edc")}, {20, _T(
"#0064c8")},
1016 {50, _T(
"#0052aa")}};
1020 {0, _T(
"#ffffff")}, {1, _T(
"#f0f0e6")}, {10, _T(
"#e6e6dc")},
1021 {20, _T(
"#dcdcd2")}, {30, _T(
"#c8c8b4")}, {40, _T(
"#aaaa8c")},
1022 {50, _T(
"#969678")}, {60, _T(
"#787864")}, {70, _T(
"#646450")},
1023 {80, _T(
"#5a5a46")}, {90, _T(
"#505036")}};
1026 {0, _T(
"#ffffff")}, {5, _T(
"#06E8E4")}, {10, _T(
"#009BE9")},
1027 {15, _T(
"#0400F3")}, {20, _T(
"#00F924")}, {25, _T(
"#06C200")},
1028 {30, _T(
"#009100")}, {35, _T(
"#FAFB00")}, {40, _T(
"#EBB608")},
1029 {45, _T(
"#FF9400")}, {50, _T(
"#FD0002")}, {55, _T(
"#D70000")},
1030 {60, _T(
"#C20300")}, {65, _T(
"#F900FE")}, {70, _T(
"#945AC8")}};
1033 {0, _T(
"#0046c8")}, {5, _T(
"#0050f0")}, {10, _T(
"#005aff")},
1034 {15, _T(
"#0069ff")}, {20, _T(
"#0078ff")}, {30, _T(
"#000cff")},
1035 {45, _T(
"#00a1ff")}, {60, _T(
"#00b6fa")}, {100, _T(
"#00c9ee")},
1036 {150, _T(
"#00e0da")}, {200, _T(
"#00e6b4")}, {300, _T(
"#82e678")},
1037 {500, _T(
"#9bff3b")}, {700, _T(
"#ffdc00")}, {1000, _T(
"#ffb700")},
1038 {1500, _T(
"#f37800")}, {2000, _T(
"#d4440c")}, {2500, _T(
"#c8201c")},
1039 {3000, _T(
"#ad0430")},
1043 {0, _T(
"#6271B7")}, {3, _T(
"#3961A9")}, {6, _T(
"#4A94A9")},
1044 {9, _T(
"#4D8D7B")}, {12, _T(
"#53A553")}, {15, _T(
"#53A553")},
1045 {18, _T(
"#359F35")}, {21, _T(
"#A79D51")}, {24, _T(
"#9F7F3A")},
1046 {27, _T(
"#A16C5C")}, {30, _T(
"#A16C5C")}, {33, _T(
"#813A4E")},
1047 {36, _T(
"#AF5088")}, {39, _T(
"#AF5088")}, {42, _T(
"#754A93")},
1048 {45, _T(
"#754A93")}, {48, _T(
"#6D61A3")}, {51, _T(
"#44698D")},
1049 {54, _T(
"#44698D")}, {57, _T(
"#5C9098")}, {60, _T(
"#7D44A5")},
1050 {63, _T(
"#7D44A5")}, {66, _T(
"#7D44A5")}, {69, _T(
"#E7D7D7")},
1051 {72, _T(
"#E7D7D7")}, {75, _T(
"#E7D7D7")}, {78, _T(
"#DBD483")},
1052 {81, _T(
"#DBD483")}, {84, _T(
"#DBD483")}, {87, _T(
"#CDC470")},
1053 {90, _T(
"#CDC470")}, {93, _T(
"#CDC470")}, {96, _T(
"#CDC470")},
1054 {99, _T(
"#808080")}};
1057static ColorMap *ColorMaps[] = {CurrentMap, GenericMap, WindMap, AirTempMap, SeaTempMap, PrecipitationMap, CloudMap};
1061 GENERIC_GRAPHIC_INDEX,
1063 AIRTEMP__GRAPHIC_INDEX,
1064 SEATEMP_GRAPHIC_INDEX,
1065 PRECIPITATION_GRAPHIC_INDEX,
1066 CLOUD_GRAPHIC_INDEX,
1067 CURRENT_GRAPHIC_INDEX,
1073static void InitColor(
ColorMap *map,
size_t maplen) {
1075 for (
size_t i = 0; i < maplen; i++) {
1078 map[i].g = c.Green();
1079 map[i].b = c.Blue();
1083void GRIBOverlayFactory::InitColorsTable() {
1084 InitColor(CurrentMap, (
sizeof CurrentMap) / (
sizeof *CurrentMap));
1085 InitColor(GenericMap, (
sizeof GenericMap) / (
sizeof *GenericMap));
1086 InitColor(WindMap, (
sizeof WindMap) / (
sizeof *WindMap));
1087 InitColor(AirTempMap, (
sizeof AirTempMap) / (
sizeof *AirTempMap));
1088 InitColor(SeaTempMap, (
sizeof SeaTempMap) / (
sizeof *SeaTempMap));
1089 InitColor(PrecipitationMap,
1090 (
sizeof PrecipitationMap) / (
sizeof *PrecipitationMap));
1091 InitColor(CloudMap, (
sizeof CloudMap) / (
sizeof *CloudMap));
1092 InitColor(CAPEMap, (
sizeof CAPEMap) / (
sizeof *CAPEMap));
1093 InitColor(REFCMap, (
sizeof REFCMap) / (
sizeof *REFCMap));
1094 InitColor(WindyMap, (
sizeof WindyMap) / (
sizeof *WindyMap));
1097void GRIBOverlayFactory::GetGraphicColor(
int settings,
double val_in,
1098 unsigned char &r,
unsigned char &g,
1100 int colormap_index = m_Settings.Settings[settings].m_iOverlayMapColors;
1105 double min = m_Settings.GetMin(settings), max = m_Settings.GetMax(settings);
1108 val_in /= max - min;
1110 switch (colormap_index) {
1111 case CURRENT_GRAPHIC_INDEX:
1113 maplen = (
sizeof CurrentMap) / (
sizeof *CurrentMap);
1115 case GENERIC_GRAPHIC_INDEX:
1117 maplen = (
sizeof GenericMap) / (
sizeof *GenericMap);
1119 case WIND_GRAPHIC_INDEX:
1121 maplen = (
sizeof WindMap) / (
sizeof *WindMap);
1123 case AIRTEMP__GRAPHIC_INDEX:
1125 maplen = (
sizeof AirTempMap) / (
sizeof *AirTempMap);
1127 case SEATEMP_GRAPHIC_INDEX:
1129 maplen = (
sizeof SeaTempMap) / (
sizeof *SeaTempMap);
1131 case PRECIPITATION_GRAPHIC_INDEX:
1132 map = PrecipitationMap;
1133 maplen = (
sizeof PrecipitationMap) / (
sizeof *PrecipitationMap);
1135 case CLOUD_GRAPHIC_INDEX:
1137 maplen = (
sizeof CloudMap) / (
sizeof *CloudMap);
1139 case CAPE_GRAPHIC_INDEX:
1141 maplen = (
sizeof CAPEMap) / (
sizeof *CAPEMap);
1143 case REFC_GRAPHIC_INDEX:
1145 maplen = (
sizeof REFCMap) / (
sizeof *REFCMap);
1147 case WINDY_GRAPHIC_INDEX:
1149 maplen = (
sizeof WindyMap) / (
sizeof *WindyMap);
1156 double cmax = map[maplen - 1].val;
1158 for (
int i = 1; i < maplen; i++) {
1159 double nmapvala = map[i - 1].val / cmax;
1160 double nmapvalb = map[i].val / cmax;
1161 if (nmapvalb > val_in || i == maplen - 1) {
1162 if (m_bGradualColors) {
1163 double d = (val_in - nmapvala) / (nmapvalb - nmapvala);
1164 r = (1 - d) * map[i - 1].r + d * map[i].r;
1165 g = (1 - d) * map[i - 1].g + d * map[i].g;
1166 b = (1 - d) * map[i - 1].b + d * map[i].b;
1178wxColour GRIBOverlayFactory::GetGraphicColor(
int settings,
double val_in) {
1179 unsigned char r, g, b;
1180 GetGraphicColor(settings, val_in, r, g, b);
1181 return wxColour(r, g, b);
1184wxString GRIBOverlayFactory::getLabelString(
double value,
int settings) {
1186 wxString f = _T(
"%.*f");
1189 case GribOverlaySettings::PRESSURE:
1191 if (m_Settings.Settings[settings].m_Units == 2)
1193 else if (m_Settings.Settings[settings].m_Units == 0 &&
1194 m_Settings.Settings[settings].m_bAbbrIsoBarsNumbers) {
1195 value -= floor(value / 100.) * 100.;
1199 case GribOverlaySettings::WAVE:
1200 case GribOverlaySettings::CURRENT:
1201 case GribOverlaySettings::AIR_TEMPERATURE:
1202 case GribOverlaySettings::SEA_TEMPERATURE:
1205 case GribOverlaySettings::PRECIPITATION:
1206 p = value < 100. ? 2 : value < 10. ? 1 : 0;
1207 p += m_Settings.Settings[settings].m_Units == 1 ? 1 : 0;
1212 return wxString::Format(f, p, value);
1216wxImage &GRIBOverlayFactory::getLabel(
double value,
int settings,
1217 wxColour back_color) {
1218 std::map<double, wxImage>::iterator it;
1219 it = m_labelCache.find(value);
1220 if (it != m_labelCache.end())
return m_labelCache[value];
1222 wxString labels = getLabelString(value, settings);
1224 wxColour text_color;
1225 GetGlobalColor(_T (
"UBLCK" ), &text_color);
1226 wxPen penText(text_color);
1228 wxBrush backBrush(back_color);
1230 wxFont mfont(9, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL,
1231 wxFONTWEIGHT_NORMAL);
1235 sdc.GetTextExtent(labels, &w, &h,
nullptr,
nullptr, &mfont);
1237 int label_offset = 5;
1239 wxBitmap bm(w + label_offset * 2, h + 2);
1244 mdc.SetPen(penText);
1245 mdc.SetBrush(backBrush);
1246 mdc.SetTextForeground(text_color);
1247 mdc.SetTextBackground(back_color);
1252 mdc.DrawRectangle(xd, yd, w + (label_offset * 2), h + 2);
1253 mdc.DrawText(labels, label_offset + xd, yd + 1);
1255 mdc.SelectObject(wxNullBitmap);
1257 m_labelCache[value] = bm.ConvertToImage();
1259 m_labelCache[value].InitAlpha();
1261 return m_labelCache[value];
1264double square(
double x) {
return x * x; }
1266void GRIBOverlayFactory::RenderGribBarbedArrows(
int settings,
GribRecord **pGR,
1268 if (!m_Settings.Settings[settings].m_bBarbedArrows)
return;
1274 SettingsIdToGribId(settings, idx, idy, polar);
1275 if (idx < 0 || idy < 0)
return;
1280 if (!pGRX || !pGRY)
return;
1283 GetGlobalColor(_T (
"YELO2" ), &colour);
1287#ifndef __OCPN__ANDROID__
1289 glEnable(GL_LINE_SMOOTH);
1291 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1292 glHint(GL_LINE_SMOOTH_HINT, GL_NICEST);
1298 glEnableClientState(GL_VERTEX_ARRAY);
1302 if (m_Settings.Settings[settings].m_bBarbArrFixSpac) {
1305 adjustSpacing(m_Settings.Settings[settings].m_iBarbArrSpacing);
1307 int total_spacing = space_pixels + arrowSize;
1313 double center_lat, center_lon;
1317 wxPoint offset_point(center.x + total_spacing, center.y + total_spacing);
1318 double offset_lat, offset_lon;
1322 double lat_spacing = fabs(center_lat - offset_lat);
1323 double lon_spacing = fabs(center_lon - offset_lon);
1327 double start_lat = floor(vp->
lat_min / lat_spacing) * lat_spacing;
1328 double start_lon = floor(vp->
lon_min / lon_spacing) * lon_spacing;
1331 double end_lat = vp->
lat_max + lat_spacing;
1332 double end_lon = vp->
lon_max + lon_spacing;
1335 for (
double lat = start_lat; lat <= end_lat; lat += lat_spacing) {
1336 for (
double lon = start_lon; lon <= end_lon; lon += lon_spacing) {
1344 drawWindArrowWithBarbs(settings, p.x, p.y, vkn * 3.6 / 1.852,
1345 (ang - 90) * M_PI / 180, (lat < 0.), colour,
1352 double minspace = wxMax(m_Settings.Settings[settings].m_iBarbArrSpacing,
1353 windArrowSize * 1.2);
1354 double minspace2 = square(minspace);
1357 int imax = pGRX->
getNi();
1358 int jmax = pGRX->
getNj();
1360 wxPoint firstpx(-1000, -1000);
1361 wxPoint oldpx(-1000, -1000);
1362 wxPoint oldpy(-1000, -1000);
1364 for (
int i = 0; i < imax; i++) {
1369 pGRX->
getXY(i, pGRX->
getNj() / 2, &lonl, &latl);
1373 if (pl.x <= firstpx.x &&
1374 square(pl.x - firstpx.x) + square(pl.y - firstpx.y) <
1378 if (square(pl.x - oldpx.x) + square(pl.y - oldpx.y) < minspace2)
continue;
1381 if (i == 0) firstpx = pl;
1384 for (
int j = 0; j < jmax; j++) {
1385 double lat = pGRX->
getY(j);
1387 if (!PointInLLBox(vp, lon, lat))
continue;
1392 if (square(p.x - oldpy.x) + square(p.y - oldpy.y) < minspace2)
continue;
1396 if (lon > 180) lon -= 360;
1401 if (vx != GRIB_NOTDEF && vy != GRIB_NOTDEF) {
1403 vkn = sqrt(vx * vx + vy * vy);
1404 ang = atan2(vy, -vx);
1405 drawWindArrowWithBarbs(settings, p.x, p.y, vkn * 3.6 / 1.852, ang,
1413 if (!m_pdc) glDisableClientState(GL_VERTEX_ARRAY);
1417void GRIBOverlayFactory::RenderGribIsobar(
int settings,
GribRecord **pGR,
1418 wxArrayPtrVoid **pIsobarArray,
1420 if (!m_Settings.Settings[settings].m_bIsoBars)
return;
1425 SettingsIdToGribId(settings, idx, idy, polar);
1426 if (idx < 0)
return;
1428 GribRecord *pGRA = pGR[idx], *pGRM =
nullptr;
1432 wxColour back_color;
1433 GetGlobalColor(_T (
"DILG1" ), &back_color);
1436 if (!pIsobarArray[idx]) {
1438 if (idy >= 0 && !polar && pGR[idy]) {
1439 pGRM = GribRecord::MagnitudeRecord(*pGR[idx], *pGR[idy]);
1440 if (!pGRM->isOk()) {
1441 m_Message_Hiden.Append(_(
"IsoBar Unable to compute record magnitude"));
1448 pIsobarArray[idx] =
new wxArrayPtrVoid;
1451 wxGenericProgressDialog *progressdialog =
nullptr;
1452 wxDateTime start = wxDateTime::Now();
1454 double min = m_Settings.GetMin(settings);
1455 double max = m_Settings.GetMax(settings);
1458 double factor = (settings == GribOverlaySettings::PRESSURE &&
1459 m_Settings.Settings[settings].m_Units == 2)
1463 for (
double press = min; press <= max;
1464 press += (m_Settings.Settings[settings].m_iIsoBarSpacing * factor)) {
1466 progressdialog->Update(press - min);
1468 wxDateTime now = wxDateTime::Now();
1469 if ((now - start).GetSeconds() > 3 && press - min < (max - min) / 2) {
1470 progressdialog =
new wxGenericProgressDialog(
1471 _(
"Building Isobar map"), _(
"Wind"), max - min + 1,
nullptr,
1472 wxPD_SMOOTH | wxPD_ELAPSED_TIME | wxPD_REMAINING_TIME);
1477 m_Settings.CalibrationFactor(settings, press,
true),
1478 m_Settings.CalibrationOffset(settings), pGRA);
1480 pIsobarArray[idx]->Add(piso);
1482 delete progressdialog;
1488 for (
unsigned int i = 0; i < pIsobarArray[idx]->GetCount(); i++) {
1490 piso->drawIsoLine(
this, m_pdc, vp,
true);
1497 piso->drawIsoLineLabels(
this, m_pdc, vp, density, first,
1498 getLabel(piso->getValue(), settings, back_color));
1500 piso->drawIsoLineLabelsGL(
this, vp, density, first,
1501 getLabelString(piso->getValue(), settings),
1502 back_color, m_TexFontNumbers);
1506void GRIBOverlayFactory::FillGrid(
GribRecord *pGR) {
1508 int imax = pGR->
getNi();
1509 int jmax = pGR->
getNj();
1511 for (
int i = 0; i < imax; i++) {
1512 for (
int j = 1; j < jmax - 1; j++) {
1513 if (pGR->
getValue(i, j) == GRIB_NOTDEF) {
1516 if (pGR->
getValue(i, j - 1) != GRIB_NOTDEF) {
1520 if (pGR->
getValue(i, j + 1) != GRIB_NOTDEF) {
1524 if (div > 1) pGR->setValue(i, j, acc / div);
1529 for (
int j = 0; j < jmax; j++) {
1530 for (
int i = 1; i < imax - 1; i++) {
1531 if (pGR->
getValue(i, j) == GRIB_NOTDEF) {
1534 if (pGR->
getValue(i - 1, j) != GRIB_NOTDEF) {
1538 if (pGR->
getValue(i + 1, j) != GRIB_NOTDEF) {
1542 if (div > 1) pGR->setValue(i, j, acc / div);
1547 pGR->setFilled(
true);
1550void GRIBOverlayFactory::RenderGribDirectionArrows(
int settings,
1553 if (!m_Settings.Settings[settings].m_bDirectionArrows)
return;
1558 SettingsIdToGribId(settings, idx, idy, polar);
1559 if (idx < 0 || idy < 0)
return;
1563 if (!pGRX || !pGRY)
return;
1564 if (!pGRX->isFilled()) FillGrid(pGRX);
1565 if (!pGRY->isFilled()) FillGrid(pGRY);
1570 arrowSizeIdx = m_Settings.Settings[settings].m_iDirectionArrowSize;
1571 if (arrowSizeIdx == 0) {
1572 if (m_pixelMM > 0.2)
1575 arrowSize = 5. / m_pixelMM;
1581 GetGlobalColor(_T (
"DILG3" ), &colour);
1585 if (m_pixelMM > 0.2) {
1587 glEnable(GL_LINE_SMOOTH);
1589 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
1590 glHint(GL_LINE_SMOOTH_HINT, GL_NICEST);
1592 if (m_Settings.Settings[settings].m_iDirectionArrowForm == 0)
1598 glEnableClientState(GL_VERTEX_ARRAY);
1602 if (m_Settings.Settings[settings].m_bDirArrFixSpac) {
1605 adjustSpacing(m_Settings.Settings[settings].m_iBarbArrSpacing);
1607 int total_spacing = space_pixels + arrowSize;
1613 double center_lat, center_lon;
1617 wxPoint offset_point(center.x + total_spacing, center.y + total_spacing);
1618 double offset_lat, offset_lon;
1622 double lat_spacing = fabs(center_lat - offset_lat);
1623 double lon_spacing = fabs(center_lon - offset_lon);
1627 double start_lat = floor(vp->
lat_min / lat_spacing) * lat_spacing;
1628 double start_lon = floor(vp->
lon_min / lon_spacing) * lon_spacing;
1631 double end_lat = vp->
lat_max + lat_spacing;
1632 double end_lon = vp->
lon_max + lon_spacing;
1635 for (
double lat = start_lat; lat <= end_lat; lat += lat_spacing) {
1636 for (
double lon = start_lon; lon <= end_lon; lon += lon_spacing) {
1648 if (dir == GRIB_NOTDEF || sh == GRIB_NOTDEF)
continue;
1652 scale = wxMax(1.0, sh);
1655 dir = (dir - 90) * M_PI / 180.;
1658 if (m_Settings.Settings[settings].m_iDirectionArrowForm == 0)
1659 drawSingleArrow(p.x, p.y, dir + vp->
rotation, colour, arrowWidth,
1660 arrowSizeIdx,
scale);
1661 else if (m_Settings.Settings[settings].m_iDirectionArrowForm == 1)
1662 drawDoubleArrow(p.x, p.y, dir + vp->
rotation, colour, arrowWidth,
1663 arrowSizeIdx,
scale);
1665 drawSingleArrow(p.x, p.y, dir + vp->
rotation, colour,
1666 wxMax(1, wxMin(8, (
int)(sh + 0.5))), arrowSizeIdx,
1675 wxMax(m_Settings.Settings[settings].m_iDirArrSpacing,
1676 m_Settings.Settings[settings].m_iDirectionArrowSize * 1.2);
1677 double minspace2 = square(minspace);
1680 int imax = pGRX->
getNi();
1681 int jmax = pGRX->
getNj();
1683 wxPoint firstpx(-1000, -1000);
1684 wxPoint oldpx(-1000, -1000);
1685 wxPoint oldpy(-1000, -1000);
1687 for (
int i = 0; i < imax; i++) {
1689 pGRX->
getXY(i, pGRX->
getNj() / 2, &lonl, &latl);
1694 if (pl.x <= firstpx.x &&
1695 square(pl.x - firstpx.x) + square(pl.y - firstpx.y) <
1699 if (square(pl.x - oldpx.x) + square(pl.y - oldpx.y) < minspace2)
continue;
1702 if (i == 0) firstpx = pl;
1704 for (
int j = 0; j < jmax; j++) {
1706 pGRX->
getXY(i, j, &lon, &lat);
1711 if (square(p.x - oldpy.x) + square(p.y - oldpy.y) >= minspace2) {
1714 if (lon > 180) lon -= 360;
1716 if (PointInLLBox(vp, lon, lat)) {
1717 double sh, dir, wdh;
1723 if (dir == GRIB_NOTDEF || sh == GRIB_NOTDEF)
continue;
1731 wdh = (8 / 2.5 * sh) + 0.5;
1732 scale = wxMax(1.0, sh);
1735 dir = (dir - 90) * M_PI / 180.;
1738 if (m_Settings.Settings[settings].m_iDirectionArrowForm == 0)
1739 drawSingleArrow(p.x, p.y, dir + vp->
rotation, colour, arrowWidth,
1740 arrowSizeIdx,
scale);
1741 else if (m_Settings.Settings[settings].m_iDirectionArrowForm == 1)
1742 drawDoubleArrow(p.x, p.y, dir + vp->
rotation, colour, arrowWidth,
1743 arrowSizeIdx,
scale);
1745 drawSingleArrow(p.x, p.y, dir + vp->
rotation, colour,
1746 wxMax(1, wxMin(8, (
int)wdh)), arrowSizeIdx,
1755 if (!m_pdc) glDisableClientState(GL_VERTEX_ARRAY);
1759void GRIBOverlayFactory::RenderGribOverlayMap(
int settings,
GribRecord **pGR,
1761 if (!m_Settings.Settings[settings].m_bOverlayMap)
return;
1763 const int grib_pixel_size = 4;
1766 SettingsIdToGribId(settings, idx, idy, polar);
1767 if (idx < 0 || !pGR[idx])
return;
1769 GribRecord *pGRA = pGR[idx], *pGRM =
nullptr;
1772 if (idy >= 0 && !polar && pGR[idy]) {
1773 pGRM = GribRecord::MagnitudeRecord(*pGR[idx], *pGR[idy]);
1774 if (!pGRM->isOk()) {
1775 m_Message_Hiden.Append(
1776 _(
"OverlayMap Unable to compute record magnitude"));
1783 if (!pGRA->isFilled()) FillGrid(pGRA);
1791 if (Intersect(vp, pGRA->getLatMin(), pGRA->getLatMax(), pGRA->getLonMin(),
1792 pGRA->getLonMax(), 0.) != _GOUT)
1794 if (Intersect(vp, pGRA->getLatMin(), pGRA->getLatMax(),
1795 pGRA->getLonMin() - 360., pGRA->getLonMax() - 360.,
1801 if (!m_pOverlay[settings]) m_pOverlay[settings] =
new GribOverlay;
1809 texture_format = GL_TEXTURE_2D;
1811 if (!texture_format)
1813 m_Message_Hiden.Append(
1814 _(
"Overlays not supported by this graphics hardware (Disable "
1817 if (!pGO->m_iTexture) CreateGribGLTexture(pGO, settings, pGRA);
1819 if (pGO->m_iTexture)
1820 DrawGLTexture(pGO, pGRA, vp);
1822 m_Message_Hiden.IsEmpty()
1824 .Append(_(
"Overlays too wide and can't be displayed:"))
1826 .Append(GribOverlaySettings::NameFromIndex(settings))
1827 : m_Message_Hiden.Append(_T(
",")).Append(
1828 GribOverlaySettings::NameFromIndex(settings));
1834 m_Message_Hiden.Append(_(
1835 "overlays suppressed if not north-up in DC mode (enable OpenGL)"));
1837 if (!pGO->m_pDCBitmap) {
1839 CreateGribImage(settings, pGRA, vp, grib_pixel_size, porg);
1840 if (bl_image.IsOk()) {
1842 pGO->m_pDCBitmap =
new wxBitmap(bl_image);
1844 new wxMask(*(pGO->m_pDCBitmap), wxColour(0, 0, 0));
1845 pGO->m_pDCBitmap->SetMask(gr_mask);
1849 if (pGO->m_pDCBitmap)
1850 m_pdc->DrawBitmap(*(pGO->m_pDCBitmap), porg.x, porg.y,
true);
1852 m_Message_Hiden.IsEmpty()
1855 "Please Zoom or Scale Out to view invisible overlays:"))
1857 .Append(GribOverlaySettings::NameFromIndex(settings))
1858 : m_Message_Hiden.Append(_T(
",")).Append(
1859 GribOverlaySettings::NameFromIndex(settings));
1867void GRIBOverlayFactory::RenderGribNumbers(
int settings,
GribRecord **pGR,
1869 if (!m_Settings.Settings[settings].m_bNumbers)
return;
1874 SettingsIdToGribId(settings, idx, idy, polar);
1875 if (idx < 0)
return;
1877 GribRecord *pGRA = pGR[idx], *pGRM =
nullptr;
1882 if (idy >= 0 && !polar && pGR[idy]) {
1883 pGRM = GribRecord::MagnitudeRecord(*pGR[idx], *pGR[idy]);
1884 if (!pGRM->isOk()) {
1885 m_Message_Hiden.Append(
1886 _(
"GribNumbers Unable to compute record magnitude"));
1895 m_TexFontNumbers.GetTextExtent(_T(
"1234"), &wstring,
nullptr);
1897 if (m_Settings.Settings[settings].m_bNumFixSpac) {
1900 int space = adjustSpacing(m_Settings.Settings[settings].m_iNumbersSpacing);
1910 if (ptl.x >= pbr.x) {
1913 pbr.x = m_ParentSize.GetWidth();
1916 for (
int i = wxMax(ptl.x, 0); i < wxMin(pbr.x, m_ParentSize.GetWidth());
1917 i += (space + wstring)) {
1918 for (
int j = wxMax(ptl.y, 0); j < wxMin(pbr.y, m_ParentSize.GetHeight());
1919 j += (space + wstring)) {
1920 double lat, lon, val;
1923 if (val != GRIB_NOTDEF) {
1924 double value = m_Settings.CalibrateValue(settings, val);
1925 wxColour back_color = GetGraphicColor(settings, value);
1927 DrawNumbers(wxPoint(i, j), value, settings, back_color);
1934 wxMax(m_Settings.Settings[settings].m_iNumbersSpacing, wstring * 1.2);
1935 double minspace2 = square(minspace);
1938 int imax = pGRA->
getNi();
1939 int jmax = pGRA->
getNj();
1941 wxPoint firstpx(-1000, -1000);
1942 wxPoint oldpx(-1000, -1000);
1943 wxPoint oldpy(-1000, -1000);
1945 for (
int i = 0; i < imax; i++) {
1947 pGRA->
getXY(i, pGRA->
getNj() / 2, &lonl, &latl);
1952 if (pl.x <= firstpx.x &&
1953 square(pl.x - firstpx.x) + square(pl.y - firstpx.y) <
1957 if (square(pl.x - oldpx.x) + square(pl.y - oldpx.y) >= minspace2) {
1959 if (i == 0) firstpx = pl;
1961 for (
int j = 0; j < jmax; j++) {
1963 pGRA->
getXY(i, j, &lon, &lat);
1968 if (square(p.x - oldpy.x) + square(p.y - oldpy.y) >= minspace2) {
1971 if (lon > 180) lon -= 360;
1973 if (PointInLLBox(vp, lon, lat)) {
1976 if (mag != GRIB_NOTDEF) {
1977 double value = m_Settings.CalibrateValue(settings, mag);
1978 wxColour back_color = GetGraphicColor(settings, value);
1980 DrawNumbers(p, value, settings, back_color);
1992void GRIBOverlayFactory::DrawNumbers(wxPoint p,
double value,
int settings,
1993 wxColour back_color) {
1995 wxImage &label = getLabel(value, settings, back_color);
1997 int w = label.GetWidth(), h = label.GetHeight();
1998 for (
int y = 0; y < h; y++)
1999 for (
int x = 0; x < w; x++)
2000 label.SetAlpha(x, y, m_Settings.m_iOverlayTransparency);
2002 m_pdc->DrawBitmap(label, p.x, p.y,
true);
2008 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
2009 glColor4ub(back_color.Red(), back_color.Green(), back_color.Blue(),
2010 m_Settings.m_iOverlayTransparency);
2014 wxString label = getLabelString(value, settings);
2016 m_TexFontNumbers.GetTextExtent(label, &w, &h);
2018 int label_offsetx = 5, label_offsety = 1;
2019 int x = p.x - label_offsetx, y = p.y - label_offsety;
2020 w += 2 * label_offsetx, h += 2 * label_offsety;
2025 glVertex2i(x + w, y);
2026 glVertex2i(x + w, y + h);
2027 glVertex2i(x, y + h);
2030 glColor4ub(0, 0, 0, m_Settings.m_iOverlayTransparency);
2032 glBegin(GL_LINE_LOOP);
2034 glVertex2i(x + w, y);
2035 glVertex2i(x + w, y + h);
2036 glVertex2i(x, y + h);
2039 glEnable(GL_TEXTURE_2D);
2040 m_TexFontNumbers.RenderString(label, p.x, p.y);
2041 glDisable(GL_TEXTURE_2D);
2047 wxFont font(9, wxFONTFAMILY_DEFAULT, wxFONTSTYLE_NORMAL,
2048 wxFONTWEIGHT_NORMAL);
2051 wxString label = getLabelString(value, settings);
2053 m_oDC->SetFont(font);
2055 m_oDC->GetTextExtent(label, &w, &h);
2057 int label_offsetx = 5, label_offsety = 1;
2058 int x = p.x - label_offsetx, y = p.y - label_offsety;
2059 w += 2 * label_offsetx, h += 2 * label_offsety;
2061 m_oDC->SetBrush(wxBrush(back_color));
2062 m_oDC->DrawRoundedRectangle(x, y, w, h, 0);
2065 m_oDC->SetPen(wxPen(wxColour(0, 0, 0), 1));
2066 m_oDC->DrawLine(x, y, x + w, y);
2067 m_oDC->DrawLine(x + w, y, x + w, y + h);
2068 m_oDC->DrawLine(x + w, y + h, x, y + h);
2069 m_oDC->DrawLine(x, y + h, x, y);
2071 m_oDC->DrawText(label, p.x, p.y);
2078void GRIBOverlayFactory::RenderGribParticles(
int settings,
GribRecord **pGR,
2080 if (!m_Settings.Settings[settings].m_bParticles)
return;
2086 SettingsIdToGribId(settings, idx, idy, polar);
2087 if (idx < 0 || idy < 0)
return;
2092 if (!pGRX || !pGRY)
return;
2097 if (m_ParticleMap && m_ParticleMap->m_Setting != settings) ClearParticles();
2099 if (!m_ParticleMap) m_ParticleMap =
new ParticleMap(settings);
2101 std::vector<Particle> &particles = m_ParticleMap->m_Particles;
2103 const int max_duration = 50;
2104 const int run_count = 6;
2106 double density = m_Settings.Settings[settings].m_dParticleDensity;
2109 int history_size = 27 / sqrt(density);
2110 history_size = wxMin(history_size, MAX_PARTICLE_HISTORY);
2112 std::vector<Particle>::iterator it;
2114 if (m_ParticleMap->history_size != history_size) {
2115 for (
unsigned int i = 0; i < particles.size(); i++) {
2117 if (m_ParticleMap->history_size > history_size &&
2118 it.m_HistoryPos >= history_size) {
2119 it = particles[particles.size() - 1];
2120 particles.pop_back();
2125 it.m_HistorySize = it.m_HistoryPos + 1;
2127 m_ParticleMap->history_size = history_size;
2135 for (it = particles.begin(); it != particles.end(); it++)
2136 for (
int i = 0; i < it->m_HistorySize; i++) {
2138 float(&p)[2] = n.m_Pos;
2139 if (p[0] == -10000)
continue;
2143 n.m_Screen[0] = ps.x;
2144 n.m_Screen[1] = ps.y;
2156 for (it = particles.begin(); it != particles.end(); it++)
2157 for (
int i = 0; i < it->m_HistorySize; i++) {
2159 float(&p)[2] = n.m_Pos;
2160 if (p[0] == -10000)
continue;
2162 n.m_Screen[0] += p1.x;
2163 n.m_Screen[1] += p1.y;
2171 if (m_bUpdateParticles) {
2172 for (
unsigned int i = 0; i < particles.size(); i++) {
2176 if (++it.m_Run < run_count)
continue;
2181 it = particles[particles.size() - 1];
2182 particles.pop_back();
2189 float(&pp)[2] = it.m_History[it.m_HistoryPos].m_Pos;
2192 if (++it.m_HistorySize > history_size) it.m_HistorySize = history_size;
2194 if (++it.m_HistoryPos >= history_size) it.m_HistoryPos = 0;
2197 float(&p)[2] = n.m_Pos;
2198 double vkn = 0, ang;
2200 if (it.
m_Duration < max_duration - history_size &&
2203 vkn > 0 && vkn < 100) {
2204 vkn = m_Settings.CalibrateValue(settings, vkn);
2206 if (settings == GribOverlaySettings::CURRENT)
2207 d = vkn * run_count;
2209 d = vkn * run_count / 4;
2221 float angr = ang / 180 * M_PI;
2222 p[0] = pp[0] + sinf(angr) * d / 60;
2223 p[1] = pp[1] + cosf(angr) * d / 60;
2225 float angr = ang / 180 * M_PI;
2226 float latr = pp[1] * M_PI / 180;
2228 float sD = sinf(D), cD = cosf(D);
2229 float sy = sinf(latr), cy = cosf(latr);
2230 float sa = sinf(angr), ca = cosf(angr);
2232 p[0] = pp[0] + asinf(sa * sD / cy) * 180 / M_PI;
2233 p[1] = asinf(sy * cD + cy * sD * ca) * 180 / M_PI;
2238 n.m_Screen[0] = ps.x;
2239 n.m_Screen[1] = ps.y;
2241 wxColor c = GetGraphicColor(settings, vkn);
2243 n.m_Color[0] = c.Red();
2244 n.m_Color[1] = c.Green();
2245 n.m_Color[2] = c.Blue();
2251 m_bUpdateParticles =
false;
2253 int total_particles = density * pGRX->
getNi() * pGRX->
getNj();
2256 if (total_particles > 60000) total_particles = 60000;
2259 int remove_particles = ((int)particles.size() - total_particles) / 16;
2260 for (
int i = 0; i < remove_particles; i++) particles.pop_back();
2264 int new_particles = (total_particles - (int)particles.size()) / 64;
2266 for (
int npi = 0; npi < new_particles; npi++) {
2269 for (
int i = 0; i < 20; i++) {
2272 (float)rand() / RAND_MAX * (pGRX->getLonMax() - pGRX->getLonMin()) +
2275 (float)rand() / RAND_MAX * (pGRX->getLatMax() - pGRX->getLatMin()) +
2279 vkn > 0 && vkn < 100)
2280 vkn = m_Settings.CalibrateValue(settings, vkn);
2286 if (settings != GribOverlaySettings::CURRENT || vkn > 1 - (
double)i / 20)
2292 np.m_HistoryPos = 0;
2293 np.m_HistorySize = 1;
2295 if (run == run_count) run = 0;
2297 memcpy(np.m_History[np.m_HistoryPos].m_Pos, p,
sizeof p);
2301 np.m_History[np.m_HistoryPos].m_Screen[0] = ps.x;
2302 np.m_History[np.m_HistoryPos].m_Screen[1] = ps.y;
2304 wxColour c = GetGraphicColor(settings, vkn);
2305 np.m_History[np.m_HistoryPos].m_Color[0] = c.Red();
2306 np.m_History[np.m_HistoryPos].m_Color[1] = c.Green();
2307 np.m_History[np.m_HistoryPos].m_Color[2] = c.Blue();
2309 particles.push_back(np);
2315 glEnable(GL_LINE_SMOOTH);
2317 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
2318 glHint(GL_LINE_SMOOTH_HINT, GL_NICEST);
2323 unsigned char *&ca = m_ParticleMap->color_array;
2324 float *&va = m_ParticleMap->vertex_array;
2325 float *&caf = m_ParticleMap->color_float_array;
2327 if (m_ParticleMap->array_size < particles.size() && !m_pdc) {
2328 m_ParticleMap->array_size = 2 * particles.size();
2334 new unsigned char[m_ParticleMap->array_size * MAX_PARTICLE_HISTORY * 8];
2335 caf =
new float[m_ParticleMap->array_size * MAX_PARTICLE_HISTORY * 8];
2336 va =
new float[m_ParticleMap->array_size * MAX_PARTICLE_HISTORY * 4];
2340 for (std::vector<Particle>::iterator it = particles.begin();
2341 it != particles.end(); it++) {
2342 wxUint8 alpha = 250;
2344 int i = it->m_HistoryPos;
2346 bool lip_valid =
false;
2347 float *lp =
nullptr, lip[2];
2352 float(&dp)[2] = it->m_History[i].m_Pos;
2353 if (dp[0] != -10000) {
2354 float(&sp)[2] = it->m_History[i].m_Screen;
2355 wxUint8(&ci)[3] = it->m_History[i].m_Color;
2357 wxUint8 c[4] = {ci[0], ci[1], (
unsigned char)(ci[2] + 240 - alpha / 2),
2360 cf[0] = ci[0] / 256.;
2361 cf[1] = ci[1] / 256.;
2362 cf[2] = ((
unsigned char)(ci[2] + 240 - alpha / 2)) / 256.;
2363 cf[3] = alpha / 256.;
2365 if (lp && fabsf(lp[0] - sp[0]) < vp->
pix_width) {
2370 float d = (float)it->m_Run / run_count;
2371 for (
int j = 0; j < 2; j++) sip[j] = d * lp[j] + (1 - d) * sp[j];
2373 if (lip_valid && fabsf(lip[0] - sip[0]) < vp->
pix_width) {
2375 m_pdc->SetPen(wxPen(wxColour(c[0], c[1], c[2]), 2));
2376 m_pdc->DrawLine(sip[0], sip[1], lip[0], lip[1]);
2378 memcpy(ca + 4 * cnt, c,
sizeof lc);
2379 memcpy(caf + 4 * cnt, cf,
sizeof lcf);
2380 memcpy(va + 2 * cnt, lip,
sizeof sp);
2382 memcpy(ca + 4 * cnt, lc,
sizeof c);
2383 memcpy(caf + 4 * cnt, lcf,
sizeof cf);
2384 memcpy(va + 2 * cnt, sip,
sizeof sp);
2389 memcpy(lip, sip,
sizeof lip);
2393 memcpy(lc, c,
sizeof lc);
2394 memcpy(lcf, cf,
sizeof lcf);
2400 i = history_size - 1;
2401 if (i >= it->m_HistorySize)
break;
2404 if (i == it->m_HistoryPos)
break;
2406 alpha -= 240 / history_size;
2412 m_oDC->DrawGLLineArray(cnt, va, caf, ca,
false);
2421 if (!m_pdc) glFlush();
2424 int time = sw.Time();
2428 m_tParticleTimer.Start(wxMax(50 - time, 2 * time), wxTIMER_ONE_SHOT);
2431 static int total_time;
2433 static int total_count;
2434 if(++total_count == 100) {
2435 printf(
"time: %.2f\n", (
double)total_time / total_count);
2436 total_time = total_count = 0;
2441void GRIBOverlayFactory::OnParticleTimer(wxTimerEvent &event) {
2442 m_bUpdateParticles =
true;
2451void GRIBOverlayFactory::DrawProjectedPosition(
int x,
int y) {
2454 dc.SetPen(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT));
2455 dc.SetBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
2456 dc.DrawRectangle(x, y, 20, 20);
2457 dc.DrawLine(x, y, x + 20, y + 20);
2458 dc.DrawLine(x, y + 20, x + 20, y);
2461 m_oDC->SetPen(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOWTEXT));
2462 m_oDC->SetBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
2463 m_oDC->DrawRectangle(x - 10, y - 10, 20, 20);
2464 m_oDC->StrokeLine(x - 10, y - 10, x + 10, y + 10);
2465 m_oDC->StrokeLine(x - 10, y + 10, x + 10, y - 10);
2470void GRIBOverlayFactory::DrawMessageWindow(wxString msg,
int x,
int y,
2472 if (msg.empty())
return;
2474 int ScaleBare_H = 30;
2479 dc.SetPen(*wxTRANSPARENT_PEN);
2481 dc.SetBrush(wxColour(243, 229, 47));
2483 dc.GetMultiLineTextExtent(msg, &w, &h);
2487 int label_offset = 10;
2488 int wdraw = w + (label_offset * 2);
2489 dc.DrawRectangle(0, yp, wdraw, h);
2490 dc.DrawLabel(msg, wxRect(label_offset, yp, wdraw, h),
2491 wxALIGN_LEFT | wxALIGN_CENTRE_VERTICAL);
2494 m_oDC->SetFont(*mfont);
2495 m_oDC->SetPen(*wxTRANSPARENT_PEN);
2497 m_oDC->SetBrush(wxColour(243, 229, 47));
2499 m_oDC->GetTextExtent(msg, &w, &h);
2502 int label_offset = 10;
2503 int wdraw = w + (label_offset * 2);
2504 wdraw *= g_ContentScaleFactor;
2505 h *= g_ContentScaleFactor;
2508 m_oDC->DrawRectangle(0, yp, wdraw, h);
2509 m_oDC->DrawText(msg, label_offset, yp);
2538void GRIBOverlayFactory::drawDoubleArrow(
int x,
int y,
double ang,
2539 wxColour arrowColor,
int arrowWidth,
2540 int arrowSizeIdx,
double scale) {
2542 wxPen pen(arrowColor, 2);
2544 m_pdc->SetBrush(*wxTRANSPARENT_BRUSH);
2545#if wxUSE_GRAPHICS_CONTEXT
2546 if (m_hiDefGraphics && m_gdc) m_gdc->SetPen(pen);
2550 wxPen pen(arrowColor, arrowWidth);
2555 drawLineBuffer(m_DoubleArrow[arrowSizeIdx], x, y, ang,
scale);
2558void GRIBOverlayFactory::drawSingleArrow(
int x,
int y,
double ang,
2559 wxColour arrowColor,
int arrowWidth,
2560 int arrowSizeIdx,
double scale) {
2562 wxPen pen(arrowColor, arrowWidth);
2564 m_pdc->SetBrush(*wxTRANSPARENT_BRUSH);
2565#if wxUSE_GRAPHICS_CONTEXT
2566 if (m_hiDefGraphics && m_gdc) m_gdc->SetPen(pen);
2570 wxPen pen(arrowColor, arrowWidth);
2575 drawLineBuffer(m_SingleArrow[arrowSizeIdx], x, y, ang,
scale);
2578void GRIBOverlayFactory::drawWindArrowWithBarbs(
int settings,
int x,
int y,
2579 double vkn,
double ang,
2580 bool south, wxColour arrowColor,
2581 double rotate_angle) {
2582 if (m_Settings.Settings[settings].m_iBarbedColour == 1)
2583 arrowColor = GetGraphicColor(settings, vkn);
2589 float penWidth = .6 / m_pixelMM;
2591 float penWidth = .4 / m_pixelMM;
2593 penWidth = wxMin(penWidth, 3.0);
2596 wxPen pen(arrowColor, 2);
2598 m_pdc->SetBrush(*wxTRANSPARENT_BRUSH);
2600#if wxUSE_GRAPHICS_CONTEXT
2601 if (m_hiDefGraphics && m_gdc) m_gdc->SetPen(pen);
2607 wxPen pen(arrowColor, penWidth);
2623 cacheidx = (int)(vkn + 2.5) / 5;
2625 cacheidx = (int)(vkn + 5) / 10 + 4;
2629 ang += rotate_angle;
2631 drawLineBuffer(m_WindArrowCache[cacheidx], x, y, ang, 1.0, south,
2632 m_bDrawBarbedArrowHead);
2635void GRIBOverlayFactory::drawLineBuffer(
LineBuffer &buffer,
int x,
int y,
2636 double ang,
double scale,
bool south,
2639 float six = sinf(ang), cox = cosf(ang), siy, coy;
2641 siy = -six, coy = -cox;
2643 siy = six, coy = cox;
2646 int count = buffer.count;
2651 wxASSERT(
sizeof vertexes /
sizeof *vertexes >= (
unsigned)count * 4);
2652 for (
int i = 0; i < 2 * count; i++) {
2654 if (!head && i > 1) j += 4;
2655 float *k = buffer.lines + 2 * j;
2656 vertexes[2 * i + 0] = k[0] * cox *
scale + k[1] * siy *
scale + x;
2657 vertexes[2 * i + 1] = k[0] * six *
scale - k[1] * coy *
scale + y;
2661 for (
int i = 0; i < count; i++) {
2662 float *l = vertexes + 4 * i;
2663#if wxUSE_GRAPHICS_CONTEXT
2664 if (m_hiDefGraphics && m_gdc)
2665 m_gdc->StrokeLine(l[0], l[1], l[2], l[3]);
2668 m_pdc->DrawLine(l[0], l[1], l[2], l[3]);
2673 for (
int i = 0; i < count; i++) {
2674 float *l = vertexes + 4 * i;
2675 if (m_hiDefGraphics)
2676 m_oDC->StrokeLine(l[0], l[1], l[2], l[3]);
2678 m_oDC->DrawLine(l[0], l[1], l[2], l[3]);
2693 double uv[],
double x,
double y,
2694 double width,
double height) {
2697 glEnable(texture_format);
2700 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
2702 glTexEnvi(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_BLEND);
2707 coords[1] = -height;
2709 coords[3] = -height;
2715 extern int pi_texture_2D_shader_program;
2716 glUseProgram(pi_texture_2D_shader_program);
2719 GLint mPosAttrib = glGetAttribLocation(pi_texture_2D_shader_program,
"aPos");
2720 GLint mUvAttrib = glGetAttribLocation(pi_texture_2D_shader_program,
"aUV");
2723 GLint texUni = glGetUniformLocation(pi_texture_2D_shader_program,
"uTex");
2724 glUniform1i(texUni, 0);
2727 glBindBuffer(GL_ARRAY_BUFFER, 0);
2728 glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
2731 glVertexAttribPointer(mPosAttrib, 2, GL_FLOAT, GL_FALSE, 0, coords);
2733 glEnableVertexAttribArray(mPosAttrib);
2736 glVertexAttribPointer(mUvAttrib, 2, GL_FLOAT, GL_FALSE, 0, uv);
2738 glEnableVertexAttribArray(mUvAttrib);
2744 mat4x4_rotate_Z(Q, I, angle);
2751 glGetUniformLocation(pi_texture_2D_shader_program,
"TransformMatrix");
2752 glUniformMatrix4fv(matloc, 1, GL_FALSE, (
const GLfloat *)Q);
2755 glActiveTexture(GL_TEXTURE0);
2762 GLushort indices1[] = {0,1,3,2};
2763 glDrawElements(GL_TRIANGLE_STRIP, 4, GL_UNSIGNED_SHORT, indices1);
2786 glVertexAttribPointer(mPosAttrib, 2, GL_FLOAT, GL_FALSE, 0, co1);
2787 glVertexAttribPointer(mUvAttrib, 2, GL_FLOAT, GL_FALSE, 0, tco1);
2789 glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
2791 glDisable(GL_BLEND);
2792 glDisable(texture_format);
2796 glUniformMatrix4fv(matloc, 1, GL_FALSE, (
const GLfloat *)I);
2802 glColor4f(1, 1, 1, 1);
2803 glTexEnvi(GL_TEXTURE_ENV, GL_TEXTURE_ENV_MODE, GL_BLEND);
2805 if (texture_format != GL_TEXTURE_2D) {
2806 for (
int i = 0; i < 4; i++) {
2807 uv[i * 2] *= pGR->
getNi();
2808 uv[(i * 2) + 1] *= pGR->
getNj();
2813 glTexCoord2d(uv[0], uv[1]), glVertex2f(x - width, y - height);
2814 glTexCoord2d(uv[2], uv[3]), glVertex2f(x, y - height);
2815 glTexCoord2d(uv[4], uv[5]), glVertex2f(x, y);
2816 glTexCoord2d(uv[6], uv[7]), glVertex2f(x - width, y);
2824 glEnable(texture_format);
2825 glBindTexture(texture_format, pGO->m_iTexture);
2828 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
2830 double lat_min = pGR->getLatMin(), lon_min = pGR->getLonMin();
2832 bool repeat = pGR->getLonMin() == 0 && pGR->getLonMax() + pGR->
getDi() == 360;
2855 xsquares = wxMax(xsquares, 2);
2856 ysquares = wxMax(ysquares, 2);
2859 double xs = vp->
pix_width / double(xsquares),
2862 typedef double mx[2][2];
2864 mx *lva =
new mx[xsquares + 1];
2865 int tw = pGO->m_iTextureDim[0], th = pGO->m_iTextureDim[1];
2866 double latstep = fabs(pGR->
getDj()) / (th - 2 - 1) * (pGR->
getNj() - 1);
2867 double lonstep = pGR->
getDi() / (tw - 2 * !repeat - 1) * (pGR->
getNi() - 1);
2869 double potNormX = (double)pGO->m_iTexDataDim[0] / tw;
2870 double potNormY = (double)pGO->m_iTexDataDim[1] / th;
2872 double clon = (lon_min + pGR->getLonMax()) / 2;
2874 for (
double y = 0; y < vp->
pix_height + ys / 2; y += ys) {
2877 for (
double x = 0; x < vp->
pix_width + xs / 2; x += xs) {
2883 if (clon - lon > 180)
2885 else if (lon - clon > 180)
2890 (((lon - lon_min) / lonstep - repeat + 1.5) / tw) * potNormX;
2891 lva[i][j][1] = (((lat - lat_min) / latstep + 1.5) / th) * potNormY;
2893 if (pGR->
getDj() < 0) lva[i][j][1] = 1 - lva[i][j][1];
2895 if (x > 0 && y > 0) {
2896 double u0 = lva[i - 1][!j][0], v0 = lva[i - 1][!j][1];
2897 double u1 = lva[i][!j][0], v1 = lva[i][!j][1];
2898 double u2 = lva[i][j][0], v2 = lva[i][j][1];
2899 double u3 = lva[i - 1][j][0], v3 = lva[i - 1][j][1];
2904 else if (u0 - u1 > .5)
2908 else if (u0 - u2 > .5)
2912 else if (u0 - u3 > .5)
2917 ((u0 >= 0 || u1 >= 0 || u2 >= 0 || u3 >= 0) &&
2918 (u0 <= 1 || u1 <= 1 || u2 <= 1 || u3 <= 1))) &&
2919 (v0 >= 0 || v1 >= 0 || v2 >= 0 || v3 >= 0) &&
2920 (v0 <= 1 || v1 <= 1 || v2 <= 1 || v3 <= 1)) {
2932 DrawSingleGLTexture(pGO, pGR, uv, x, y, xs, ys);
2943 glDisable(GL_BLEND);
2944 glDisable(texture_format);
GRIB Data Visualization and Rendering Factory.
@ Idx_COMP_REFL
Composite radar reflectivity.
@ Idx_PRECIP_TOT
Total precipitation.
@ Idx_AIR_TEMP
Air temperature at 2m.
@ Idx_PRESSURE
Surface pressure.
@ Idx_WVDIR
Wave direction.
@ Idx_CLOUD_TOT
Total cloud cover.
@ Idx_WIND_GUST
Wind gust speed at surface.
@ Idx_WIND_VX
Surface wind velocity X component.
@ Idx_HTSIGW
Significant wave height.
@ Idx_SEACURRENT_VY
Sea current velocity Y component.
@ Idx_SEA_TEMP
Sea surface temperature.
@ Idx_WIND_VY
Surface wind velocity Y component.
@ Idx_SEACURRENT_VX
Sea current velocity X component.
@ Idx_CAPE
Convective Available Potential Energy.
GRIB Weather Data Control Interface.
void GetProjectedLatLon(int &x, int &y, PlugIn_ViewPort *vp)
Gets the projected position of vessel based on current course, speed and forecast time.
Container for rendered GRIB data visualizations in texture or bitmap form.
GribRecord * m_GribRecordPtrArray[Idx_COUNT]
Array of pointers to GRIB records representing different meteorological parameters.
Represents a meteorological data grid from a GRIB (Gridded Binary) file.
double getDi() const
Returns the grid spacing in longitude (i) direction in degrees.
void getXY(int i, int j, double *x, double *y) const
Converts grid indices to longitude/latitude coordinates.
double getDj() const
Returns the grid spacing in latitude (j) direction in degrees.
double getInterpolatedValue(double px, double py, bool numericalInterpolation=true, bool dir=false) const
Get spatially interpolated value at exact lat/lon position.
int getNi() const
Returns the number of points in the longitude (i) direction of the grid.
int getNj() const
Returns the number of points in the latitude (j) direction of the grid.
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.
double getY(int j) const
Converts grid index j to latitude in degrees.
double getValue(int i, int j) const
Returns the data value at a specific grid point.
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).
Assembles input characters to lines.
Contains view parameters and status information for a chart display viewport.
double view_scale_ppm
Display scale in pixels per meter.
int pix_width
Viewport width in pixels.
double lon_max
Maximum longitude of the viewport.
double clon
Center longitude of the viewport in decimal degrees.
double lat_max
Maximum latitude of the viewport.
int pix_height
Viewport height in pixels.
double clat
Center latitude of the viewport in decimal degrees.
double skew
Display skew angle in radians.
double rotation
Display rotation angle in radians.
bool bValid
True if this viewport is valid and can be used for rendering.
double lon_min
Minimum longitude of the viewport.
double lat_min
Minimum latitude of the viewport.
int m_projection_type
Chart projection type (PROJECTION_MERCATOR, etc.)
@ PI_PROJECTION_MERCATOR
Mercator projection, standard for navigation charts.
int GetChartbarHeight(void)
Gets height of chart bar in pixels.
wxWindow * GetOCPNCanvasWindow()
Gets OpenCPN's main canvas window.
int GetCanvasCount()
Gets total number of chart canvases.
wxFont * OCPNGetFont(wxString TextElement, int default_size)
Gets a font for UI elements.
wxFont GetOCPNGUIScaledFont_PlugIn(wxString item)
Gets a uniquely scaled font copy for responsive UI elements.
void PositionBearingDistanceMercator_Plugin(double lat, double lon, double brg, double dist, double *dlat, double *dlon)
Calculates destination point given start point, bearing and distance.
double PlugInGetDisplaySizeMM()
Gets physical display size in millimeters.
void GetCanvasPixLL(PlugIn_ViewPort *vp, wxPoint *pp, double lat, double lon)
Converts lat/lon to canvas physical pixel coordinates.
wxWindow * GetCanvasByIndex(int canvasIndex)
Gets chart canvas window by index.
double OCPN_GetWinDIPScaleFactor()
Gets Windows-specific DPI scaling factor.
void GetCanvasLLPix(PlugIn_ViewPort *vp, wxPoint p, double *plat, double *plon)
Converts canvas physical pixel coordinates to lat/lon.
OpenGL Platform Abstraction Layer.
Manager for particle animation system.
Individual particle for wind/current animation.
int m_Duration
Duration this particle should exist in animation cycles.