OpenCPN Partial API docs
Loading...
Searching...
No Matches
IsoLine.cpp
Go to the documentation of this file.
1/**********************************************************************
2zyGrib: meteorological GRIB file viewer
3Copyright (C) 2008 - Jacques Zaninetti - http://www.zygrib.org
4
5This program is free software: you can redistribute it and/or modify
6it under the terms of the GNU General Public License as published by
7the Free Software Foundation, either version 3 of the License, or
8(at your option) any later version.
9
10This program is distributed in the hope that it will be useful,
11but WITHOUT ANY WARRANTY; without even the implied warranty of
12MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13GNU General Public License for more details.
14
15You should have received a copy of the GNU General Public License
16along with this program. If not, see <http://www.gnu.org/licenses/>.
17***********************************************************************/
22#include "wx/wxprec.h"
23
24#ifndef WX_PRECOMP
25#include "wx/wx.h"
26#endif // precompiled headers
27
28// #include "chcanv.h"
29// #include "model/georef.h"
30#include <wx/graphics.h>
31
32#include "IsoLine.h"
33#include "GribSettingsDialog.h"
34#include "GribOverlayFactory.h"
35
36#ifdef __OCPN__ANDROID__
37#include "qdebug.h"
38#endif
39
40// static void GenerateSpline(int n, wxPoint points[]);
41// static void ClearSplineList();
42wxList ocpn_wx_spline_point_list;
43
44#include <wx/listimpl.cpp>
45WX_DEFINE_LIST(MySegList);
46WX_DEFINE_LIST(MySegListList);
47
48#ifndef PI
49#define PI 3.14159
50#endif
51#define CTRUE -1
52#define CFALSE 0
53
54/* Local variables for cohen_sutherland_line_clip: */
56 double xmin, xmax, ymin, ymax;
57};
58void CompOutCode(double x, double y, outcode *code,
59 struct LOC_cohen_sutherland_line_clip *LINK) {
60 /*Compute outcode for the point (x,y) */
61 *code = 0;
62 if (y > LINK->ymax)
63 *code = 1L << ((long)TOP);
64 else if (y < LINK->ymin)
65 *code = 1L << ((long)BOTTOM);
66 if (x > LINK->xmax)
67 *code |= 1L << ((long)RIGHT);
68 else if (x < LINK->xmin)
69 *code |= 1L << ((long)LEFT);
70}
71
72ClipResult cohen_sutherland_line_clip_d(double *x0, double *y0, double *x1,
73 double *y1, double xmin_, double xmax_,
74 double ymin_, double ymax_) {
75 /* Cohen-Sutherland clipping algorithm for line P0=(x1,y0) to P1=(x1,y1)
76 * and clip rectangle with diagonal from (xmin,ymin) to (xmax,ymax).*/
78 int done = CFALSE;
79 ClipResult clip = Visible;
80 outcode outcode0, outcode1, outcodeOut;
81 /*Outcodes for P0,P1, and whichever point lies outside the clip rectangle*/
82 double x = 0., y = 0.;
83
84 V.xmin = xmin_;
85 V.xmax = xmax_;
86 V.ymin = ymin_;
87 V.ymax = ymax_;
88 CompOutCode(*x0, *y0, &outcode0, &V);
89 CompOutCode(*x1, *y1, &outcode1, &V);
90 do {
91 if (outcode0 == 0 && outcode1 == 0) { /*Trivial accept and exit*/
92 done = CTRUE;
93 } else if ((outcode0 & outcode1) != 0) {
94 clip = Invisible;
95 done = CTRUE;
96 }
97 /*Logical intersection is true, so trivial reject and exit.*/
98 else {
99 clip = Visible;
100 /*Failed both tests, so calculate the line segment to clip;
101 * from an outside point to an intersection with clip edge.*/
102 /*At least one endpoint is outside the clip rectangle; pick it.*/
103 if (outcode0 != 0)
104 outcodeOut = outcode0;
105 else
106 outcodeOut = outcode1;
107 /*Now find intersection point;
108 * use formulas y=y0+slope*(x-x0),x=x0+(1/slope)*(y-y0).*/
109
110 if (((1L << ((long)TOP)) & outcodeOut) != 0) {
111 /*Divide line at top of clip rectangle*/
112 x = *x0 + (*x1 - *x0) * (V.ymax - *y0) / (*y1 - *y0);
113 y = V.ymax;
114 } else if (((1L << ((long)BOTTOM)) & outcodeOut) != 0) {
115 /*Divide line at bottom of clip rectangle*/
116 x = *x0 + (*x1 - *x0) * (V.ymin - *y0) / (*y1 - *y0);
117 y = V.ymin;
118 } else if (((1L << ((long)RIGHT)) & outcodeOut) != 0) {
119 /*Divide line at right edge of clip rectangle*/
120 y = *y0 + (*y1 - *y0) * (V.xmax - *x0) / (*x1 - *x0);
121 x = V.xmax;
122 } else if (((1L << ((long)LEFT)) & outcodeOut) != 0) {
123 /*Divide line at left edge of clip rectangle*/
124 y = *y0 + (*y1 - *y0) * (V.xmin - *x0) / (*x1 - *x0);
125 x = V.xmin;
126 }
127 /*Now we move outside point to intersection point to clip,
128 * and get ready for next pass.*/
129 if (outcodeOut == outcode0) {
130 *x0 = x;
131 *y0 = y;
132 CompOutCode(*x0, *y0, &outcode0, &V);
133 } else {
134 *x1 = x;
135 *y1 = y;
136 CompOutCode(*x1, *y1, &outcode1, &V);
137 }
138 }
139 } while (!done);
140 return clip;
141}
142
143ClipResult cohen_sutherland_line_clip_i(int *x0_, int *y0_, int *x1_, int *y1_,
144 int xmin_, int xmax_, int ymin_,
145 int ymax_) {
146 ClipResult ret;
147 double x0, y0, x1, y1;
148 x0 = *x0_;
149 y0 = *y0_;
150 x1 = *x1_;
151 y1 = *y1_;
152 ret =
153 cohen_sutherland_line_clip_d(&x0, &y0, &x1, &y1, (double)xmin_,
154 (double)xmax_, (double)ymin_, (double)ymax_);
155 *x0_ = (int)x0;
156 *y0_ = (int)y0;
157 *x1_ = (int)x1;
158 *y1_ = (int)y1;
159 return ret;
160}
161
162double round_msvc(double x) { return (floor(x + 0.5)); }
163
164//---------------------------------------------------------------
165IsoLine::IsoLine(double val, double coeff, double offset,
166 const GribRecord *rec_) {
167 if (wxGetDisplaySize().x > 0) {
168 m_pixelMM = PlugInGetDisplaySizeMM() / wxGetDisplaySize().x;
169 m_pixelMM = wxMax(.02, m_pixelMM); // protect against bad data
170 } else
171 m_pixelMM = 0.27; // semi-standard number...
172
173 value = val / coeff - offset;
174
175 rec = rec_;
176 W = rec_->getNi();
177 H = rec_->getNj();
178
179 //---------------------------------------------------------
180 // Génère la liste des segments.
181 extractIsoLine(rec_);
182
183 value = val;
184
185 if (trace.size() == 0) return;
186
187 // Join the isoline segments into a nice list
188 // Which is end-to-end continuous and unidirectional
189
190 // Create a master wxList of the trace list
191 std::list<Segment *>::iterator it;
192 for (it = trace.begin(); it != trace.end(); it++) {
193 Segment *seg = *it;
194 seg->bUsed = false;
195 m_seglist.Append(*it);
196 }
197
198 // Isoline may be discontinuous....
199 // So build a list of continuous segments
200 bool bdone = false;
201 while (!bdone) {
202 MySegList *ps = BuildContinuousSegment();
203
204 m_SegListList.Append(ps);
205
206 MySegList::Node *node;
207 Segment *seg;
208
209 // recreate the master list, removing used segs
210
211 node = m_seglist.GetFirst();
212 while (node) {
213 seg = node->GetData();
214 if (seg->bUsed) {
215 m_seglist.Erase(node);
216 node = m_seglist.GetFirst();
217 } else
218 node = node->GetNext();
219 }
220
221 if (0 == m_seglist.GetCount()) bdone = true;
222 }
223
226}
227//---------------------------------------------------------------
228IsoLine::~IsoLine() {
229 // printf("delete Isobar : press=%4.0f long=%d\n", pressure/100,
230 // trace.size());
231
232 std::list<Segment *>::iterator it;
233 for (it = trace.begin(); it != trace.end(); it++) {
234 delete *it;
235 *it = nullptr;
236 }
237 trace.clear();
238
239 m_SegListList.DeleteContents(true);
240 m_SegListList.Clear();
241}
242
243MySegList *IsoLine::BuildContinuousSegment(void) {
244 MySegList::Node *node;
245 Segment *seg;
246
247 MySegList *ret_list = new MySegList;
248
249 // Build a chain extending from the "2" end of the target segment
250 // The joined list, side 2...
251 MySegList segjoin2;
252
253 // Add any first segment to the list
254 node = m_seglist.GetFirst();
255 Segment *seg0 = node->GetData();
256 seg0->bUsed = true;
257 segjoin2.Append(seg0);
258
259 Segment *tseg = seg0;
260
261 while (tseg) {
262 bool badded = false;
263 Segment *seg;
264 node = m_seglist.GetFirst();
265 while (node) {
266 seg = node->GetData();
267
268 if ((!seg->bUsed) && (seg->py1 == tseg->py2) &&
269 (seg->px1 == tseg->px2)) // fits without reverse
270 {
271 seg->bUsed = true;
272 segjoin2.Append(seg);
273 badded = true;
274 break;
275 } else if ((!seg->bUsed) && (seg->py2 == tseg->py2) &&
276 (seg->px2 == tseg->px2)) // fits, needs reverse
277 {
278 seg->bUsed = true;
279 double a = seg->px2;
280 seg->px2 = seg->px1;
281 seg->px1 = a;
282 double b = seg->py2;
283 seg->py2 = seg->py1;
284 seg->py1 = b;
285 segjoin2.Append(seg);
286 badded = true;
287 break;
288 }
289
290 node = node->GetNext();
291 }
292 if (badded == true)
293 tseg = seg;
294 else
295 tseg = nullptr;
296 }
297
298 // Build a chain extending from the "1" end of the target segment
299 // The joined list, side 1...
300 MySegList segjoin1;
301
302 // Add the same first segment to the list
303 node = m_seglist.GetFirst();
304 seg0 = node->GetData();
305 seg0->bUsed = true;
306 segjoin1.Append(seg0);
307
308 tseg = seg0;
309
310 while (tseg) {
311 bool badded = false;
312 node = m_seglist.GetFirst();
313 while (node) {
314 seg = node->GetData();
315
316 if ((!seg->bUsed) && (seg->py2 == tseg->py1) &&
317 (seg->px2 == tseg->px1)) // fits without reverse
318 {
319 seg->bUsed = true;
320 segjoin1.Append(seg);
321 badded = true;
322 break;
323 } else if ((!seg->bUsed) && (seg->py1 == tseg->py1) &&
324 (seg->px1 == tseg->px1)) // fits, needs reverse
325 {
326 seg->bUsed = true;
327 double a = seg->px2;
328 seg->px2 = seg->px1;
329 seg->px1 = a;
330 double b = seg->py2;
331 seg->py2 = seg->py1;
332 seg->py1 = b;
333 segjoin1.Append(seg);
334 badded = true;
335 break;
336 }
337
338 node = node->GetNext();
339 }
340 if (badded == true)
341 tseg = seg;
342 else
343 tseg = nullptr;
344 }
345
346 // Now have two lists...
347
348 // Start with "1" side list,
349 // starting from the end, and skipping the first segment
350
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);
356 }
357
358 // Now add the "2"side list
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);
364 }
365
366 // And there it is
367
368 return ret_list;
369}
370
371//---------------------------------------------------------------
372void IsoLine::drawIsoLine(GRIBOverlayFactory *pof, wxDC *dc,
373 PlugIn_ViewPort *vp, bool bHiDef) {
374 int nsegs = trace.size();
375 if (nsegs < 1) return;
376
377 GetGlobalColor(_T ( "UITX1" ), &isoLineColor);
378
379#if wxUSE_GRAPHICS_CONTEXT
380 wxGraphicsContext *pgc = nullptr;
381#endif
382
383 if (dc) {
384 wxPen ppISO(isoLineColor, 2);
385
386#if wxUSE_GRAPHICS_CONTEXT
387 wxMemoryDC *pmdc;
388 pmdc = dynamic_cast<wxMemoryDC *>(dc);
389 pgc = wxGraphicsContext::Create(*pmdc);
390 pgc->SetPen(ppISO);
391#endif
392 dc->SetPen(ppISO);
393 } else { /* opengl */
394#ifdef ocpnUSE_GL
395 // #ifndef USE_ANDROID_GLES2
396 // if(m_pixelMM > 0.2){ // pixel size large enough to
397 // render well
398 // // Enable anti-aliased lines, at best quality
399 // glEnable( GL_LINE_SMOOTH );
400 // glEnable( GL_BLEND );
401 // glBlendFunc( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA );
402 // glHint( GL_LINE_SMOOTH_HINT, GL_NICEST );
403 // glLineWidth( 2 );
404 // }
405 // else{
406 // glLineWidth( 0.4/m_pixelMM); // set a target line
407 // width by MM
408 // }
409 // #else
410 if (pof->m_oDC) {
411 wxPen ppISO(isoLineColor, 2);
412 pof->m_oDC->SetPen(ppISO);
413 }
414#endif
415 }
416
417 std::list<Segment *>::iterator it;
418
419 //---------------------------------------------------------
420 // Dessine les segments
421 //---------------------------------------------------------
422 for (it = trace.begin(); it != trace.end(); it++) {
423 Segment *seg = *it;
424
427 /* skip segments that go the wrong way around the world */
428 double sx1 = seg->px1, sx2 = seg->px2;
429 if (sx2 - sx1 > 180)
430 sx2 -= 360;
431 else if (sx1 - sx2 > 180)
432 sx1 -= 360;
433
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))
438 continue;
439 }
440
441 wxPoint ab;
442 GetCanvasPixLL(vp, &ab, seg->py1, seg->px1);
443 wxPoint cd;
444 GetCanvasPixLL(vp, &cd, seg->py2, seg->px2);
445
446 if (dc) {
447#if wxUSE_GRAPHICS_CONTEXT
448 if (bHiDef && pgc)
449 pgc->StrokeLine(ab.x, ab.y, cd.x, cd.y);
450 else
451#endif
452 dc->DrawLine(ab.x, ab.y, cd.x, cd.y);
453 } else { /* opengl */
454#ifdef ocpnUSE_GL
455
456 if (pof->m_oDC) {
457 pof->m_oDC->DrawLine(ab.x, ab.y, cd.x, cd.y);
458 }
459
460#endif
461 }
462 }
463
464#if wxUSE_GRAPHICS_CONTEXT
465 delete pgc;
466#endif
467
468 // if(!dc) /* opengl */
469 // glEnd();
470}
471
472//---------------------------------------------------------------
473
474void IsoLine::drawIsoLineLabels(GRIBOverlayFactory *pof, wxDC *dc,
475 PlugIn_ViewPort *vp, int density, int first,
476 wxImage &imageLabel)
477
478{
479 std::list<Segment *>::iterator it;
480 int nb = first;
481 wxString label;
482
483 //---------------------------------------------------------
484 // Ecrit les labels
485 //---------------------------------------------------------
486 wxRect prev;
487 for (it = trace.begin(); it != trace.end(); it++, nb++) {
488 if (nb % density == 0) {
489 Segment *seg = *it;
490
491 // if(vp->vpBBox.PointInBox((seg->px1 + seg->px2)/2., (seg->py1
492 // + seg->py2)/2., 0.))
493 {
494 wxPoint ab;
495 GetCanvasPixLL(vp, &ab, seg->py1, seg->px1);
496 wxPoint cd;
497 GetCanvasPixLL(vp, &cd, seg->py1, seg->px1);
498
499 int w = imageLabel.GetWidth();
500 int h = imageLabel.GetHeight();
501
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;
505
506 int x = xd - label_offset;
507 wxRect r(x, yd, w, h);
508 r.Inflate(w);
509 if (!prev.Intersects(r)) {
510 prev = r;
511
512 /* don't use alpha for isobars, for some reason draw bitmap ignores
513 the 4th argument (true or false has same result) */
514 wxImage img(w, h, imageLabel.GetData(), true);
515 dc->DrawBitmap(img, xd, yd, false);
516 }
517 }
518 }
519 }
520}
521
522void IsoLine::drawIsoLineLabelsGL(GRIBOverlayFactory *pof, PlugIn_ViewPort *vp,
523 int density, int first, wxString label,
524 wxColour &color, TexFont &texfont)
525
526{
527 std::list<Segment *>::iterator it;
528 int nb = first;
529
530#ifdef ocpnUSE_GL
531 glEnable(GL_BLEND);
532 glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
533
534 //---------------------------------------------------------
535 // Ecrit les labels
536 //---------------------------------------------------------
537 wxRect prev;
538 for (it = trace.begin(); it != trace.end(); it++, nb++) {
539 if (nb % density == 0) {
540 Segment *seg = *it;
541
542 // if(vp->vpBBox.PointInBox((seg->px1 + seg->px2)/2., (seg->py1
543 // + seg->py2)/2., 0.))
544 {
545 wxPoint ab;
546 GetCanvasPixLL(vp, &ab, seg->py1, seg->px1);
547 wxPoint cd;
548 GetCanvasPixLL(vp, &cd, seg->py1, seg->px1);
549
550 int w, h;
551 texfont.GetTextExtent(label, &w, &h);
552
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;
558
559 wxRect r(x, y, w, h);
560 r.Inflate(w);
561 if (!prev.Intersects(r)) {
562#if 1
563 prev = r;
564 if (pof->m_oDC) {
565 // pof->m_oDC->SetFont( *mfont );
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);
570 }
571
572#else
573 prev = r;
574 glColor4ub(color.Red(), color.Green(), color.Blue(), color.Alpha());
575
576 /* draw bounding rectangle */
577 glBegin(GL_QUADS);
578 glVertex2i(x, y);
579 glVertex2i(x + w, y);
580 glVertex2i(x + w, y + h);
581 glVertex2i(x, y + h);
582 glEnd();
583
584 glColor3ub(0, 0, 0);
585
586 glBegin(GL_LINE_LOOP);
587 glVertex2i(x, y);
588 glVertex2i(x + w, y);
589 glVertex2i(x + w, y + h);
590 glVertex2i(x, y + h);
591 glEnd();
592
593 glEnable(GL_TEXTURE_2D);
594 texfont.RenderString(label, xd, yd);
595 glDisable(GL_TEXTURE_2D);
596#endif
597 }
598 }
599 }
600 }
601 glDisable(GL_BLEND);
602#endif
603}
604
605//==================================================================================
606// Segment
607//==================================================================================
608Segment::Segment(int I, int w, int J, char c1, char c2, char c3, char c4,
609 const GribRecord *rec, double pressure) {
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);
614
615 intersectionAreteGrille(i, j, k, l, &px1, &py1, rec, pressure);
616 intersectionAreteGrille(m, n, o, p, &px2, &py2, rec, pressure);
617}
618//-----------------------------------------------------------------------
619void Segment::intersectionAreteGrille(int i, int j, int k, int l, double *x,
620 double *y, const GribRecord *rec,
621 double pressure) {
622 double xa, xb, ya, yb, pa, pb, dec;
623 pa = rec->getValue(i, j);
624 pb = rec->getValue(k, l);
625
626 rec->getXY(i, j, &xa, &ya);
627 rec->getXY(k, l, &xb, &yb);
628
629 // Abscisse
630 if (pb != pa)
631 dec = (pressure - pa) / (pb - pa);
632 else
633 dec = 0.5;
634 if (fabs(dec) > 1) dec = 0.5;
635 double xd = xb - xa;
636 if (xd < -180)
637 xd += 360;
638 else if (xd > 180)
639 xd -= 360;
640 *x = xa + xd * dec;
641
642 // Ordonnée
643 if (pb != pa)
644 dec = (pressure - pa) / (pb - pa);
645 else
646 dec = 0.5;
647 if (fabs(dec) > 1) dec = 0.5;
648 *y = ya + (yb - ya) * dec;
649}
650//---------------------------------------------------------------
651void Segment::traduitCode(int I, int w, int J, char c1, int &i, int &j) {
652 int Im1 = I ? I - 1 : w - 1;
653 switch (c1) {
654 case 'a':
655 i = Im1;
656 j = J - 1;
657 break;
658 case 'b':
659 i = I;
660 j = J - 1;
661 break;
662 case 'c':
663 i = Im1;
664 j = J;
665 break;
666 case 'd':
667 i = I;
668 j = J;
669 break;
670 default:
671 i = I;
672 j = J;
673 }
674}
675
676//-----------------------------------------------------------------------
677// Génère la liste des segments.
678// Les coordonnées sont les indices dans la grille du GribRecord
679//---------------------------------------------------------
680void IsoLine::extractIsoLine(const GribRecord *rec) {
681 int i, j, W, H;
682 double a, b, c, d;
683 W = rec->getNi();
684 H = rec->getNj();
685
686 int We = W;
687 if (rec->getLonMax() + rec->getDi() - rec->getLonMin() == 360) We++;
688
689 for (j = 1; j < H; j++) // !!!! 1 to end
690 {
691 a = rec->getValue(0, j - 1);
692 c = rec->getValue(0, j);
693 for (i = 1; i < We; i++, a = b, c = d) {
694 // x = rec->getX(i);
695 // y = rec->getY(j);
696
697 int ni = i;
698 if (i == W) ni = 0;
699 b = rec->getValue(ni, j - 1);
700 d = rec->getValue(ni, j);
701
702 if (a == GRIB_NOTDEF || b == GRIB_NOTDEF || c == GRIB_NOTDEF ||
703 d == GRIB_NOTDEF)
704 continue;
705
706 if ((a < value && b < value && c < value && d < value) ||
707 (a > value && b > value && c > value && d > value))
708 continue;
709
710 // Détermine si 1 ou 2 segments traversent la case ab-cd
711 // a b
712 // c d
713 //--------------------------------
714 // 1 segment en diagonale
715 //--------------------------------
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));
728 //--------------------------------
729 // 1 segment H ou V
730 //--------------------------------
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));
737 //--------------------------------
738 // 2 segments en diagonale
739 //--------------------------------
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));
746 }
747 }
748 }
749}
750
751// ----------------------------------------------------------------------------
752// splines code lifted from wxWidgets
753// ----------------------------------------------------------------------------
754
755// ----------------------------------- spline code
756// ----------------------------------------
757
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);
766
767#define half(z1, z2) ((z1 + z2) / 2.0)
768#define THRESHOLD 5
769
770/* iterative version */
771
772void ocpn_wx_quadratic_spline(double a1, double b1, double a2, double b2,
773 double a3, double b3, double a4, double b4) {
774 double xmid, ymid;
775 double x1, y1, x2, y2, x3, y3, x4, y4;
776
777 ocpn_wx_clear_stack();
778 ocpn_wx_spline_push(a1, b1, a2, b2, a3, b3, a4, b4);
779
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);
787 } else {
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,
793 ymid);
794 }
795 }
796}
797
798/* utilities used by spline drawing routines */
799
801 double x1, y1, x2, y2, x3, y3, x4, y4;
802} Stack;
803
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;
808
809void ocpn_wx_clear_stack() {
810 ocpn_wx_stack_top = ocpn_wx_spline_stack;
811 ocpn_wx_stack_count = 0;
812}
813
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;
824 ocpn_wx_stack_top++;
825 ocpn_wx_stack_count++;
826}
827
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);
831 ocpn_wx_stack_top--;
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;
841 return (1);
842}
843
844static bool ocpn_wx_spline_add_point(double x, double y) {
845 wxPoint *point = new wxPoint;
846 point->x = (int)x;
847 point->y = (int)y;
848 ocpn_wx_spline_point_list.Append((wxObject *)point);
849 return true;
850}
851
852void GenSpline(wxList *points) {
853 wxPoint *p;
854 double cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4;
855 double x1, y1, x2, y2;
856
857 wxList::compatibility_iterator node = points->GetFirst();
858 if (!node)
859 // empty list
860 return;
861
862 p = (wxPoint *)node->GetData();
863
864 x1 = p->x;
865 y1 = p->y;
866
867 node = node->GetNext();
868 p = (wxPoint *)node->GetData();
869
870 x2 = p->x;
871 y2 = p->y;
872 cx1 = (double)((x1 + x2) / 2);
873 cy1 = (double)((y1 + y2) / 2);
874 cx2 = (double)((cx1 + x2) / 2);
875 cy2 = (double)((cy1 + y2) / 2);
876
877 ocpn_wx_spline_add_point(x1, y1);
878
879 while ((node = node->GetNext())
880#if !wxUSE_STL
881 != nullptr
882#endif // !wxUSE_STL
883 ) {
884 p = (wxPoint *)node->GetData();
885 x1 = x2;
886 y1 = y2;
887 x2 = p->x;
888 y2 = p->y;
889 cx4 = (double)(x1 + x2) / 2;
890 cy4 = (double)(y1 + y2) / 2;
891 cx3 = (double)(x1 + cx4) / 2;
892 cy3 = (double)(y1 + cy4) / 2;
893
894 ocpn_wx_quadratic_spline(cx1, cy1, cx2, cy2, cx3, cy3, cx4, cy4);
895
896 cx1 = cx4;
897 cy1 = cy4;
898 cx2 = (double)(cx1 + x2) / 2;
899 cy2 = (double)(cy1 + y2) / 2;
900 }
901
902 ocpn_wx_spline_add_point(cx1, cy1);
903 ocpn_wx_spline_add_point(x2, y2);
904}
905
906#if 0
907static void GenerateSpline(int n, wxPoint points[])
908{
909 wxList list;
910 for (int i =0; i < n; i++)
911 {
912 list.Append((wxObject*)&points[i]);
913 }
914
915 GenSpline(&list);
916}
917
918static void ClearSplineList()
919{
920 wxList::compatibility_iterator node = ocpn_wx_spline_point_list.GetFirst();
921 while (node)
922 {
923 wxPoint *point = (wxPoint *)node->GetData();
924 delete point;
925 ocpn_wx_spline_point_list.Erase(node);
926 node = ocpn_wx_spline_point_list.GetFirst();
927 }
928}
929#endif
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.
Definition GribRecord.h:182
double getDi() const
Returns the grid spacing in longitude (i) direction in degrees.
Definition GribRecord.h:460
void getXY(int i, int j, double *x, double *y) const
Converts grid indices to longitude/latitude coordinates.
Definition GribRecord.h:562
int getNi() const
Returns the number of points in the longitude (i) direction of the grid.
Definition GribRecord.h:448
int getNj() const
Returns the number of points in the latitude (j) direction of the grid.
Definition GribRecord.h:454
double getValue(int i, int j) const
Returns the data value at a specific grid point.
Definition GribRecord.h:480
IsoLine(double val, double coeff, double offset, const GribRecord *rec)
Definition IsoLine.cpp:165
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.