30#include <wx/graphics.h>
36#ifdef __OCPN__ANDROID__
42wxList ocpn_wx_spline_point_list;
44#include <wx/listimpl.cpp>
45WX_DEFINE_LIST(MySegList);
46WX_DEFINE_LIST(MySegListList);
56 double xmin, xmax, ymin, ymax;
58void CompOutCode(
double x,
double y, outcode *code,
63 *code = 1L << ((long)TOP);
64 else if (y < LINK->ymin)
65 *code = 1L << ((long)BOTTOM);
67 *code |= 1L << ((long)RIGHT);
68 else if (x < LINK->xmin)
69 *code |= 1L << ((long)LEFT);
72ClipResult cohen_sutherland_line_clip_d(
double *x0,
double *y0,
double *x1,
73 double *y1,
double xmin_,
double xmax_,
74 double ymin_,
double ymax_) {
79 ClipResult clip = Visible;
80 outcode outcode0, outcode1, outcodeOut;
82 double x = 0., y = 0.;
88 CompOutCode(*x0, *y0, &outcode0, &V);
89 CompOutCode(*x1, *y1, &outcode1, &V);
91 if (outcode0 == 0 && outcode1 == 0) {
93 }
else if ((outcode0 & outcode1) != 0) {
104 outcodeOut = outcode0;
106 outcodeOut = outcode1;
110 if (((1L << ((
long)TOP)) & outcodeOut) != 0) {
112 x = *x0 + (*x1 - *x0) * (V.ymax - *y0) / (*y1 - *y0);
114 }
else if (((1L << ((
long)BOTTOM)) & outcodeOut) != 0) {
116 x = *x0 + (*x1 - *x0) * (V.ymin - *y0) / (*y1 - *y0);
118 }
else if (((1L << ((
long)RIGHT)) & outcodeOut) != 0) {
120 y = *y0 + (*y1 - *y0) * (V.xmax - *x0) / (*x1 - *x0);
122 }
else if (((1L << ((
long)LEFT)) & outcodeOut) != 0) {
124 y = *y0 + (*y1 - *y0) * (V.xmin - *x0) / (*x1 - *x0);
129 if (outcodeOut == outcode0) {
132 CompOutCode(*x0, *y0, &outcode0, &V);
136 CompOutCode(*x1, *y1, &outcode1, &V);
143ClipResult cohen_sutherland_line_clip_i(
int *x0_,
int *y0_,
int *x1_,
int *y1_,
144 int xmin_,
int xmax_,
int ymin_,
147 double x0, y0, x1, y1;
153 cohen_sutherland_line_clip_d(&x0, &y0, &x1, &y1, (
double)xmin_,
154 (
double)xmax_, (
double)ymin_, (
double)ymax_);
162double round_msvc(
double x) {
return (floor(x + 0.5)); }
167 if (wxGetDisplaySize().x > 0) {
169 m_pixelMM = wxMax(.02, m_pixelMM);
173 value = val / coeff - offset;
181 extractIsoLine(rec_);
185 if (trace.size() == 0)
return;
191 std::list<Segment *>::iterator it;
192 for (it = trace.begin(); it != trace.end(); it++) {
195 m_seglist.Append(*it);
202 MySegList *ps = BuildContinuousSegment();
204 m_SegListList.Append(ps);
206 MySegList::Node *node;
211 node = m_seglist.GetFirst();
213 seg = node->GetData();
215 m_seglist.Erase(node);
216 node = m_seglist.GetFirst();
218 node = node->GetNext();
221 if (0 == m_seglist.GetCount()) bdone =
true;
232 std::list<Segment *>::iterator it;
233 for (it = trace.begin(); it != trace.end(); it++) {
239 m_SegListList.DeleteContents(
true);
240 m_SegListList.Clear();
243MySegList *IsoLine::BuildContinuousSegment(
void) {
244 MySegList::Node *node;
247 MySegList *ret_list =
new MySegList;
254 node = m_seglist.GetFirst();
255 Segment *seg0 = node->GetData();
257 segjoin2.Append(seg0);
264 node = m_seglist.GetFirst();
266 seg = node->GetData();
268 if ((!seg->bUsed) && (seg->py1 == tseg->py2) &&
269 (seg->px1 == tseg->px2))
272 segjoin2.Append(seg);
275 }
else if ((!seg->bUsed) && (seg->py2 == tseg->py2) &&
276 (seg->px2 == tseg->px2))
285 segjoin2.Append(seg);
290 node = node->GetNext();
303 node = m_seglist.GetFirst();
304 seg0 = node->GetData();
306 segjoin1.Append(seg0);
312 node = m_seglist.GetFirst();
314 seg = node->GetData();
316 if ((!seg->bUsed) && (seg->py2 == tseg->py1) &&
317 (seg->px2 == tseg->px1))
320 segjoin1.Append(seg);
323 }
else if ((!seg->bUsed) && (seg->py1 == tseg->py1) &&
324 (seg->px1 == tseg->px1))
333 segjoin1.Append(seg);
338 node = node->GetNext();
351 int n1 = segjoin1.GetCount();
352 for (
int i = n1 - 1; i > 0; i--) {
353 node = segjoin1.Item(i);
354 seg = node->GetData();
355 ret_list->Append(seg);
359 int n2 = segjoin2.GetCount();
360 for (
int i = 0; i < n2; i++) {
361 node = segjoin2.Item(i);
362 seg = node->GetData();
363 ret_list->Append(seg);
374 int nsegs = trace.size();
375 if (nsegs < 1)
return;
377 GetGlobalColor(_T (
"UITX1" ), &isoLineColor);
379#if wxUSE_GRAPHICS_CONTEXT
380 wxGraphicsContext *pgc =
nullptr;
384 wxPen ppISO(isoLineColor, 2);
386#if wxUSE_GRAPHICS_CONTEXT
388 pmdc =
dynamic_cast<wxMemoryDC *
>(dc);
389 pgc = wxGraphicsContext::Create(*pmdc);
411 wxPen ppISO(isoLineColor, 2);
412 pof->m_oDC->SetPen(ppISO);
417 std::list<Segment *>::iterator it;
422 for (it = trace.begin(); it != trace.end(); it++) {
428 double sx1 = seg->px1, sx2 = seg->px2;
431 else if (sx1 - sx2 > 180)
434 if ((sx1 + 180 < vp->
clon && sx2 + 180 > vp->
clon) ||
435 (sx1 + 180 > vp->
clon && sx2 + 180 < vp->
clon) ||
436 (sx1 - 180 < vp->
clon && sx2 - 180 > vp->
clon) ||
437 (sx1 - 180 > vp->
clon && sx2 - 180 < vp->
clon))
447#if wxUSE_GRAPHICS_CONTEXT
449 pgc->StrokeLine(ab.x, ab.y, cd.x, cd.y);
452 dc->DrawLine(ab.x, ab.y, cd.x, cd.y);
457 pof->m_oDC->DrawLine(ab.x, ab.y, cd.x, cd.y);
464#if wxUSE_GRAPHICS_CONTEXT
479 std::list<Segment *>::iterator it;
487 for (it = trace.begin(); it != trace.end(); it++, nb++) {
488 if (nb % density == 0) {
499 int w = imageLabel.GetWidth();
500 int h = imageLabel.GetHeight();
502 int label_offset = 6;
503 int xd = (ab.x + cd.x - (w + label_offset * 2)) / 2;
504 int yd = (ab.y + cd.y - h) / 2;
506 int x = xd - label_offset;
507 wxRect r(x, yd, w, h);
509 if (!prev.Intersects(r)) {
514 wxImage img(w, h, imageLabel.GetData(),
true);
515 dc->DrawBitmap(img, xd, yd,
false);
523 int density,
int first, wxString label,
524 wxColour &color,
TexFont &texfont)
527 std::list<Segment *>::iterator it;
532 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
538 for (it = trace.begin(); it != trace.end(); it++, nb++) {
539 if (nb % density == 0) {
551 texfont.GetTextExtent(label, &w, &h);
553 int label_offsetx = 6, label_offsety = 1;
554 int xd = (ab.x + cd.x - (w + label_offsetx * 2)) / 2;
555 int yd = (ab.y + cd.y - h) / 2;
556 int x = xd - label_offsetx, y = yd - label_offsety;
557 w += 2 * label_offsetx, h += 2 * label_offsety;
559 wxRect r(x, y, w, h);
561 if (!prev.Intersects(r)) {
566 pof->m_oDC->SetPen(*wxBLACK_PEN);
567 pof->m_oDC->SetBrush(color);
568 pof->m_oDC->DrawRectangle(x, y, w, h);
569 pof->m_oDC->DrawText(label, xd, yd);
574 glColor4ub(color.Red(), color.Green(), color.Blue(), color.Alpha());
579 glVertex2i(x + w, y);
580 glVertex2i(x + w, y + h);
581 glVertex2i(x, y + h);
586 glBegin(GL_LINE_LOOP);
588 glVertex2i(x + w, y);
589 glVertex2i(x + w, y + h);
590 glVertex2i(x, y + h);
593 glEnable(GL_TEXTURE_2D);
594 texfont.RenderString(label, xd, yd);
595 glDisable(GL_TEXTURE_2D);
608Segment::Segment(
int I,
int w,
int J,
char c1,
char c2,
char c3,
char c4,
610 traduitCode(I, w, J, c1, i, j);
611 traduitCode(I, w, J, c2, k, l);
612 traduitCode(I, w, J, c3, m, n);
613 traduitCode(I, w, J, c4, o, p);
615 intersectionAreteGrille(i, j, k, l, &px1, &py1, rec, pressure);
616 intersectionAreteGrille(m, n, o, p, &px2, &py2, rec, pressure);
619void Segment::intersectionAreteGrille(
int i,
int j,
int k,
int l,
double *x,
622 double xa, xb, ya, yb, pa, pb, dec;
626 rec->
getXY(i, j, &xa, &ya);
627 rec->
getXY(k, l, &xb, &yb);
631 dec = (pressure - pa) / (pb - pa);
634 if (fabs(dec) > 1) dec = 0.5;
644 dec = (pressure - pa) / (pb - pa);
647 if (fabs(dec) > 1) dec = 0.5;
648 *y = ya + (yb - ya) * dec;
651void Segment::traduitCode(
int I,
int w,
int J,
char c1,
int &i,
int &j) {
652 int Im1 = I ? I - 1 : w - 1;
680void IsoLine::extractIsoLine(
const GribRecord *rec) {
687 if (rec->getLonMax() + rec->
getDi() - rec->getLonMin() == 360) We++;
689 for (j = 1; j < H; j++)
693 for (i = 1; i < We; i++, a = b, c = d) {
702 if (a == GRIB_NOTDEF || b == GRIB_NOTDEF || c == GRIB_NOTDEF ||
706 if ((a < value && b < value && c < value && d < value) ||
707 (a > value && b > value && c > value && d > value))
716 if ((a <= value && b <= value && c <= value && d > value) ||
717 (a > value && b > value && c > value && d <= value))
718 trace.push_back(
new Segment(ni, W, j,
'c',
'd',
'b',
'd', rec, value));
719 else if ((a <= value && c <= value && d <= value && b > value) ||
720 (a > value && c > value && d > value && b <= value))
721 trace.push_back(
new Segment(ni, W, j,
'a',
'b',
'b',
'd', rec, value));
722 else if ((c <= value && d <= value && b <= value && a > value) ||
723 (c > value && d > value && b > value && a <= value))
724 trace.push_back(
new Segment(ni, W, j,
'a',
'b',
'a',
'c', rec, value));
725 else if ((a <= value && b <= value && d <= value && c > value) ||
726 (a > value && b > value && d > value && c <= value))
727 trace.push_back(
new Segment(ni, W, j,
'a',
'c',
'c',
'd', rec, value));
731 else if ((a <= value && b <= value && c > value && d > value) ||
732 (a > value && b > value && c <= value && d <= value))
733 trace.push_back(
new Segment(ni, W, j,
'a',
'c',
'b',
'd', rec, value));
734 else if ((a <= value && c <= value && b > value && d > value) ||
735 (a > value && c > value && b <= value && d <= value))
736 trace.push_back(
new Segment(ni, W, j,
'a',
'b',
'c',
'd', rec, value));
740 else if (a <= value && d <= value && c > value && b > value) {
741 trace.push_back(
new Segment(ni, W, j,
'a',
'b',
'b',
'd', rec, value));
742 trace.push_back(
new Segment(ni, W, j,
'a',
'c',
'c',
'd', rec, value));
743 }
else if (a > value && d > value && c <= value && b <= value) {
744 trace.push_back(
new Segment(ni, W, j,
'a',
'b',
'a',
'c', rec, value));
745 trace.push_back(
new Segment(ni, W, j,
'b',
'd',
'c',
'd', rec, value));
758void ocpn_wx_quadratic_spline(
double a1,
double b1,
double a2,
double b2,
759 double a3,
double b3,
double a4,
double b4);
760void ocpn_wx_clear_stack();
761int ocpn_wx_spline_pop(
double *x1,
double *y1,
double *x2,
double *y2,
762 double *x3,
double *y3,
double *x4,
double *y4);
763void ocpn_wx_spline_push(
double x1,
double y1,
double x2,
double y2,
double x3,
764 double y3,
double x4,
double y4);
765static bool ocpn_wx_spline_add_point(
double x,
double y);
767#define half(z1, z2) ((z1 + z2) / 2.0)
772void ocpn_wx_quadratic_spline(
double a1,
double b1,
double a2,
double b2,
773 double a3,
double b3,
double a4,
double b4) {
775 double x1, y1, x2, y2, x3, y3, x4, y4;
777 ocpn_wx_clear_stack();
778 ocpn_wx_spline_push(a1, b1, a2, b2, a3, b3, a4, b4);
780 while (ocpn_wx_spline_pop(&x1, &y1, &x2, &y2, &x3, &y3, &x4, &y4)) {
781 xmid = (double)half(x2, x3);
782 ymid = (double)half(y2, y3);
783 if (fabs(x1 - xmid) < THRESHOLD && fabs(y1 - ymid) < THRESHOLD &&
784 fabs(xmid - x4) < THRESHOLD && fabs(ymid - y4) < THRESHOLD) {
785 ocpn_wx_spline_add_point(x1, y1);
786 ocpn_wx_spline_add_point(xmid, ymid);
788 ocpn_wx_spline_push(xmid, ymid, (
double)half(xmid, x3),
789 (
double)half(ymid, y3), (
double)half(x3, x4),
790 (
double)half(y3, y4), x4, y4);
791 ocpn_wx_spline_push(x1, y1, (
double)half(x1, x2), (
double)half(y1, y2),
792 (
double)half(x2, xmid), (
double)half(y2, ymid), xmid,
801 double x1, y1, x2, y2, x3, y3, x4, y4;
804#define SPLINE_STACK_DEPTH 20
805static Stack ocpn_wx_spline_stack[SPLINE_STACK_DEPTH];
806static Stack *ocpn_wx_stack_top;
807static int ocpn_wx_stack_count;
809void ocpn_wx_clear_stack() {
810 ocpn_wx_stack_top = ocpn_wx_spline_stack;
811 ocpn_wx_stack_count = 0;
814void ocpn_wx_spline_push(
double x1,
double y1,
double x2,
double y2,
double x3,
815 double y3,
double x4,
double y4) {
816 ocpn_wx_stack_top->x1 = x1;
817 ocpn_wx_stack_top->y1 = y1;
818 ocpn_wx_stack_top->x2 = x2;
819 ocpn_wx_stack_top->y2 = y2;
820 ocpn_wx_stack_top->x3 = x3;
821 ocpn_wx_stack_top->y3 = y3;
822 ocpn_wx_stack_top->x4 = x4;
823 ocpn_wx_stack_top->y4 = y4;
825 ocpn_wx_stack_count++;
828int ocpn_wx_spline_pop(
double *x1,
double *y1,
double *x2,
double *y2,
829 double *x3,
double *y3,
double *x4,
double *y4) {
830 if (ocpn_wx_stack_count == 0)
return (0);
832 ocpn_wx_stack_count--;
833 *x1 = ocpn_wx_stack_top->x1;
834 *y1 = ocpn_wx_stack_top->y1;
835 *x2 = ocpn_wx_stack_top->x2;
836 *y2 = ocpn_wx_stack_top->y2;
837 *x3 = ocpn_wx_stack_top->x3;
838 *y3 = ocpn_wx_stack_top->y3;
839 *x4 = ocpn_wx_stack_top->x4;
840 *y4 = ocpn_wx_stack_top->y4;
844static bool ocpn_wx_spline_add_point(
double x,
double y) {
845 wxPoint *point =
new wxPoint;
848 ocpn_wx_spline_point_list.Append((wxObject *)point);
852void GenSpline(wxList *points) {
854 double cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4;
855 double x1, y1, x2, y2;
857 wxList::compatibility_iterator node = points->GetFirst();
862 p = (wxPoint *)node->GetData();
867 node = node->GetNext();
868 p = (wxPoint *)node->GetData();
872 cx1 = (double)((x1 + x2) / 2);
873 cy1 = (double)((y1 + y2) / 2);
874 cx2 = (double)((cx1 + x2) / 2);
875 cy2 = (double)((cy1 + y2) / 2);
877 ocpn_wx_spline_add_point(x1, y1);
879 while ((node = node->GetNext())
884 p = (wxPoint *)node->GetData();
889 cx4 = (double)(x1 + x2) / 2;
890 cy4 = (double)(y1 + y2) / 2;
891 cx3 = (double)(x1 + cx4) / 2;
892 cy3 = (double)(y1 + cy4) / 2;
894 ocpn_wx_quadratic_spline(cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4);
898 cx2 = (double)(cx1 + x2) / 2;
899 cy2 = (double)(cy1 + y2) / 2;
902 ocpn_wx_spline_add_point(cx1, cy1);
903 ocpn_wx_spline_add_point(x2, y2);
907static void GenerateSpline(
int n, wxPoint points[])
910 for (
int i =0; i < n; i++)
912 list.Append((wxObject*)&points[i]);
918static void ClearSplineList()
920 wxList::compatibility_iterator node = ocpn_wx_spline_point_list.GetFirst();
923 wxPoint *point = (wxPoint *)node->GetData();
925 ocpn_wx_spline_point_list.Erase(node);
926 node = ocpn_wx_spline_point_list.GetFirst();
GRIB Data Visualization and Rendering Factory.
GRIB Display Settings Configuration Interface.
GRIB Isobar and Isoline Generation System.
Factory class for creating and managing GRIB data visualizations.
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.
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.
double getValue(int i, int j) const
Returns the data value at a specific grid point.
IsoLine(double val, double coeff, double offset, const GribRecord *rec)
Contains view parameters and status information for a chart display viewport.
double clon
Center longitude of the viewport in decimal degrees.
int m_projection_type
Chart projection type (PROJECTION_MERCATOR, etc.)
@ PI_PROJECTION_MERCATOR
Mercator projection, standard for navigation charts.
@ PI_PROJECTION_EQUIRECTANGULAR
Equirectangular/Plate Carrée projection, simple lat/lon grid.
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.