OpenCPN Partial API docs
Loading...
Searching...
No Matches
viewport.cpp
1/***************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: ViewPort
5 * Author: David Register
6 *
7 ***************************************************************************
8 * Copyright (C) 2015 by David S. Register *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25
26// For compilers that support precompilation, includes "wx.h".
27#include <wx/wxprec.h>
28
29#ifndef WX_PRECOMP
30#include <wx/wx.h>
31#endif // precompiled headers
32#include <wx/image.h>
33#include <wx/graphics.h>
34#include <wx/listbook.h>
35#include <wx/clipbrd.h>
36#include <wx/aui/aui.h>
37
38#if defined(__OCPN__ANDROID__)
39#include <GLES2/gl2.h>
40#elif defined(__WXQT__) || defined(__WXGTK__)
41#include <GL/glew.h>
42#endif
43
44#include "config.h"
45
46#include "dychart.h"
47
48#include <wx/listimpl.cpp>
49
50#include "chcanv.h"
51#include "TCWin.h"
52#include "model/geodesic.h"
53#include "styles.h"
54#include "model/routeman.h"
55#include "navutil.h"
56#include "kml.h"
57#include "concanv.h"
58#include "thumbwin.h"
59#include "chartdb.h"
60#include "chartimg.h"
61#include "model/cutil.h"
62#include "TrackPropDlg.h"
63#include "tcmgr.h"
64#include "routemanagerdialog.h"
65#include "pluginmanager.h"
66#include "ocpn_pixel.h"
67#include "ocpndc.h"
68#include "undo.h"
69#include "model/multiplexer.h"
70#include "timers.h"
71#include "tide_time.h"
72#include "glTextureDescriptor.h"
73#include "ChInfoWin.h"
74#include "Quilt.h"
75#include "model/select_item.h"
76#include "model/select.h"
77#include "FontMgr.h"
78#include "model/ais_decoder.h"
79#include "model/ais_target_data.h"
80#include "AISTargetAlertDialog.h"
81#include "SendToGpsDlg.h"
82#include "OCPNRegion.h"
83#include "gshhs.h"
84
85#ifdef ocpnUSE_GL
86#include "glChartCanvas.h"
87#endif
88
89#include "cm93.h" // for chart outline draw
90#include "s57chart.h" // for ArrayOfS57Obj
91#include "s52plib.h"
92
93#include "ais.h"
94
95#ifdef __VISUALC__
96#include <wx/msw/msvcrt.h>
97#endif
98
99#ifndef __WXMSW__
100#include <signal.h>
101#include <setjmp.h>
102
103extern struct sigaction sa_all_old;
104
105extern sigjmp_buf env; // the context saved by sigsetjmp();
106#endif
107
108#include <vector>
109
110// ----------------------------------------------------------------------------
111// Useful Prototypes
112// ----------------------------------------------------------------------------
113
114extern void catch_signals(int signo);
115
116//------------------------------------------------------------------------------
117// ViewPort Implementation
118//------------------------------------------------------------------------------
119ViewPort::ViewPort() {
120 bValid = false;
121 skew = 0.;
122 view_scale_ppm = 1;
123 rotation = 0.;
124 tilt = 0.;
125 b_quilt = false;
126 pix_height = pix_width = 0;
127 b_MercatorProjectionOverride = false;
128 lat0_cache = NAN;
129 m_projection_type = PROJECTION_MERCATOR;
130 m_displayScale = 1.0;
131}
132
133void ViewPort::SetPixelScale(double scale) { m_displayScale = scale; }
134
135// TODO: eliminate the use of this function
136wxPoint ViewPort::GetPixFromLL(double lat, double lon) {
137 wxPoint2DDouble p = GetDoublePixFromLL(lat, lon);
138 if (wxFinite(p.m_x) && wxFinite(p.m_y)) {
139 if ((abs(p.m_x) < 1e6) && (abs(p.m_y) < 1e6))
140 return wxPoint(wxRound(p.m_x), wxRound(p.m_y));
141 }
142 return wxPoint(INVALID_COORD, INVALID_COORD);
143}
144
145wxPoint2DDouble ViewPort::GetDoublePixFromLL(double lat, double lon) {
146 double easting = 0;
147 double northing = 0;
148 double xlon = lon;
149
150 /* Make sure lon and lon0 are same phase */
151 if (xlon * clon < 0.) {
152 if (xlon < 0.)
153 xlon += 360.;
154 else
155 xlon -= 360.;
156 }
157
158 if (fabs(xlon - clon) > 180.) {
159 if (xlon > clon)
160 xlon -= 360.;
161 else
162 xlon += 360.;
163 }
164
165 // update cache of trig functions used for projections
166 if (clat != lat0_cache) {
167 lat0_cache = clat;
168 switch (m_projection_type) {
169 case PROJECTION_MERCATOR:
170 case PROJECTION_WEB_MERCATOR:
171 cache0 = toSMcache_y30(clat);
172 break;
173 case PROJECTION_POLAR:
174 cache0 = toPOLARcache_e(clat);
175 break;
176 case PROJECTION_ORTHOGRAPHIC:
177 case PROJECTION_STEREOGRAPHIC:
178 case PROJECTION_GNOMONIC:
179 cache_phi0(clat, &cache0, &cache1);
180 break;
181 }
182 }
183
184 switch (m_projection_type) {
185 case PROJECTION_MERCATOR:
186 case PROJECTION_WEB_MERCATOR:
187 toSMcache(lat, xlon, cache0, clon, &easting, &northing);
188 break;
189
190 case PROJECTION_TRANSVERSE_MERCATOR:
191 // We calculate northings as referenced to the equator
192 // And eastings as though the projection point is midscreen.
193
194 double tmeasting, tmnorthing;
195 double tmceasting, tmcnorthing;
196 toTM(clat, clon, 0., clon, &tmceasting, &tmcnorthing);
197 toTM(lat, xlon, 0., clon, &tmeasting, &tmnorthing);
198
199 northing = tmnorthing - tmcnorthing;
200 easting = tmeasting - tmceasting;
201 break;
202
203 case PROJECTION_POLYCONIC:
204
205 // We calculate northings as referenced to the equator
206 // And eastings as though the projection point is midscreen.
207 double pceasting, pcnorthing;
208 toPOLY(clat, clon, 0., clon, &pceasting, &pcnorthing);
209
210 double peasting, pnorthing;
211 toPOLY(lat, xlon, 0., clon, &peasting, &pnorthing);
212
213 easting = peasting;
214 northing = pnorthing - pcnorthing;
215 break;
216
217 case PROJECTION_ORTHOGRAPHIC:
218 toORTHO(lat, xlon, cache0, cache1, clon, &easting, &northing);
219 break;
220
221 case PROJECTION_POLAR:
222 toPOLAR(lat, xlon, cache0, clat, clon, &easting, &northing);
223 break;
224
225 case PROJECTION_STEREOGRAPHIC:
226 toSTEREO(lat, xlon, cache0, cache1, clon, &easting, &northing);
227 break;
228
229 case PROJECTION_GNOMONIC:
230 toGNO(lat, xlon, cache0, cache1, clon, &easting, &northing);
231 break;
232
233 case PROJECTION_EQUIRECTANGULAR:
234 toEQUIRECT(lat, xlon, clat, clon, &easting, &northing);
235 break;
236
237 default:
238 printf("unhandled projection\n");
239 }
240
241 if (!wxFinite(easting) || !wxFinite(northing))
242 return wxPoint2DDouble(easting, northing);
243
244 double epix = easting * view_scale_ppm;
245 double npix = northing * view_scale_ppm;
246 double dxr = epix;
247 double dyr = npix;
248
249 // Apply VP Rotation
250 double angle = rotation;
251
252 if (angle) {
253 dxr = epix * cos(angle) + npix * sin(angle);
254 dyr = npix * cos(angle) - epix * sin(angle);
255 }
256 double x = (pix_width / 2.0) + dxr;
257 double y = (pix_height / 2.0) - dyr;
258 if (!g_bopengl) {
259 // Convert from physical to logical pixels when not using OpenGL.
260 // This ensures that the viewport corner coordinates remain the
261 // same in OpenGL and no-OpenGL mode (especially in MacOS).
262 x /= m_displayScale;
263 y /= m_displayScale;
264 }
265 return wxPoint2DDouble(x, y);
266}
267
268void ViewPort::GetLLFromPix(const wxPoint2DDouble &p, double *lat,
269 double *lon) {
270 // Calculate distance from the center of the viewport to the given point in
271 // physical pixels.
272 double dx = p.m_x - (pix_width / 2.0);
273 double dy = (pix_height / 2.0) - p.m_y;
274
275 double xpr = dx;
276 double ypr = dy;
277
278 // Apply VP Rotation
279 double angle = rotation;
280
281 if (angle) {
282 xpr = (dx * cos(angle)) - (dy * sin(angle));
283 ypr = (dy * cos(angle)) + (dx * sin(angle));
284 }
285 double d_east = xpr / view_scale_ppm;
286 double d_north = ypr / view_scale_ppm;
287
288 double slat = 0.0, slon = 0.0;
289 switch (m_projection_type) {
290 case PROJECTION_MERCATOR:
291 case PROJECTION_WEB_MERCATOR:
292 // TODO This could be fromSM_ECC to better match some Raster charts
293 // However, it seems that cm93 (and S57) prefer no eccentricity
294 // correction Think about it....
295 fromSM(d_east, d_north, clat, clon, &slat, &slon);
296 break;
297
298 case PROJECTION_TRANSVERSE_MERCATOR: {
299 double tmceasting, tmcnorthing;
300 toTM(clat, clon, 0., clon, &tmceasting, &tmcnorthing);
301
302 fromTM(d_east, d_north + tmcnorthing, 0., clon, &slat, &slon);
303 } break;
304
305 case PROJECTION_POLYCONIC: {
306 double polyeasting, polynorthing;
307 toPOLY(clat, clon, 0., clon, &polyeasting, &polynorthing);
308
309 fromPOLY(d_east, d_north + polynorthing, 0., clon, &slat, &slon);
310 } break;
311
312 case PROJECTION_ORTHOGRAPHIC:
313 fromORTHO(d_east, d_north, clat, clon, &slat, &slon);
314 break;
315
316 case PROJECTION_POLAR:
317 fromPOLAR(d_east, d_north, clat, clon, &slat, &slon);
318 break;
319
320 case PROJECTION_STEREOGRAPHIC:
321 fromSTEREO(d_east, d_north, clat, clon, &slat, &slon);
322 break;
323
324 case PROJECTION_GNOMONIC:
325 fromGNO(d_east, d_north, clat, clon, &slat, &slon);
326 break;
327
328 case PROJECTION_EQUIRECTANGULAR:
329 fromEQUIRECT(d_east, d_north, clat, clon, &slat, &slon);
330 break;
331
332 default:
333 printf("unhandled projection\n");
334 }
335
336 *lat = slat;
337
338 if (slon < -180.)
339 slon += 360.;
340 else if (slon > 180.)
341 slon -= 360.;
342 *lon = slon;
343}
344
345LLRegion ViewPort::GetLLRegion(const OCPNRegion &region) {
346 // todo: for these projecetions, improve this calculation by using the
347 // method in SetBoxes here
348#ifndef ocpnUSE_GL
349 return LLRegion(GetBBox());
350#else
351
352 if (!glChartCanvas::CanClipViewport(*this)) return LLRegion(GetBBox());
353
354 OCPNRegionIterator it(region);
355 LLRegion r;
356 while (it.HaveRects()) {
357 wxRect rect = it.GetRect();
358
359 int x1 = rect.x, y1 = rect.y, x2 = x1 + rect.width, y2 = y1 + rect.height;
360 int p[8] = {x1, y1, x2, y1, x2, y2, x1, y2};
361 double pll[2896]; // Max splits is 180, ((180 * 2) + 2) * 8 = 2896.
362 int j;
363
364 /* if the viewport is rotated, we must split the segments as straight lines
365 in lat/lon coordinates map to curves in projected coordinate space */
366 if (fabs(rotation) >= 0.0001) {
367 j = 0;
368 double lastlat, lastlon;
369 int li = 6;
370 GetLLFromPix(wxPoint(p[li], p[li + 1]), &lastlat, &lastlon);
371 for (int i = 0; i < 8; i += 2) {
372 double lat, lon;
373 GetLLFromPix(wxPoint(p[i], p[i + 1]), &lat, &lon);
374
375 // use 2 degree grid
376 double grid = 2;
377 int lat_splits = floor(fabs(lat - lastlat) / grid);
378 double lond = fabs(lon - lastlon);
379 int lon_splits = floor((lond > 180 ? 360 - lond : lond) / grid);
380 int splits = wxMax(lat_splits, lon_splits) + 1;
381
382 for (int k = 1; k < splits; k++) {
383 float d = (float)k / splits;
384 GetLLFromPix(wxPoint((1 - d) * p[li] + d * p[i],
385 (1 - d) * p[li + 1] + d * p[i + 1]),
386 pll + j, pll + j + 1);
387 j += 2;
388 }
389 pll[j++] = lat;
390 pll[j++] = lon;
391 li = i;
392 lastlat = lat, lastlon = lon;
393 }
394 } else {
395 j = 8;
396 for (int i = 0; i < j; i += 2)
397 GetLLFromPix(wxPoint(p[i], p[i + 1]), pll + i, pll + i + 1);
398 }
399
400 // resolve (this works even if rectangle crosses both 0 and 180)
401 for (int i = 0; i < j; i += 2) {
402 if (pll[i + 1] <= clon - 180)
403 pll[i + 1] += 360;
404 else if (pll[i + 1] >= clon + 180)
405 pll[i + 1] -= 360;
406 }
407
408 r.Union(LLRegion(j / 2, pll));
409 it.NextRect();
410 }
411 return r;
412#endif
413}
414
416 double maxlat;
417 bool subtract;
418 OCPNRegion r;
419};
420
422 const LLRegion &llregion,
423 int chart_native_scale) {
424 double rotation_save = rotation;
425 rotation = 0;
426
427 std::list<ContourRegion> cregions;
428 for (std::list<poly_contour>::const_iterator i = llregion.contours.begin();
429 i != llregion.contours.end(); i++) {
430 float *contour_points = new float[2 * i->size()];
431 int idx = 0;
432 std::list<contour_pt>::const_iterator j;
433 for (j = i->begin(); j != i->end(); j++) {
434 contour_points[idx++] = j->y;
435 contour_points[idx++] = j->x;
436 }
437
438 double total = 0, maxlat = -90;
439 int pl = idx - 2;
440 double x0 = contour_points[0] - contour_points[pl + 0];
441 double y0 = contour_points[1] - contour_points[pl + 1];
442 // determine winding direction of this contour
443 for (int p = 0; p < idx; p += 2) {
444 maxlat = wxMax(maxlat, contour_points[p]);
445 int pn = p < idx - 2 ? p + 2 : 0;
446 double x1 = contour_points[pn + 0] - contour_points[p + 0];
447 double y1 = contour_points[pn + 1] - contour_points[p + 1];
448 total += x1 * y0 - x0 * y1;
449 x0 = x1, y0 = y1;
450 }
451
453 s.maxlat = maxlat;
454 s.subtract = total < 0;
455 s.r = GetVPRegionIntersect(region, i->size(), contour_points,
456 chart_native_scale, NULL);
457 delete[] contour_points;
458
459 std::list<ContourRegion>::iterator k = cregions.begin();
460 while (k != cregions.end()) {
461 if (k->maxlat < s.maxlat) break;
462 k++;
463 }
464 cregions.insert(k, s);
465 }
466
467 OCPNRegion r;
468 for (std::list<ContourRegion>::iterator k = cregions.begin();
469 k != cregions.end(); k++) {
470 if (k->r.Ok()) {
471 if (k->subtract)
472 r.Subtract(k->r);
473 else
474 r.Union(k->r);
475 }
476 }
477
478 rotation = rotation_save;
479 return r;
480}
481
483 float *llpoints,
484 int chart_native_scale,
485 wxPoint *ppoints) {
486 // Calculate the intersection between a given OCPNRegion (Region) and a
487 // polygon specified by lat/lon points.
488
489 // If the viewpoint is highly overzoomed wrt to chart native scale, the
490 // polygon region may be huge. This can be very expensive, and lead to
491 // crashes on some platforms (gtk in particular) So, look for this case and
492 // handle appropriately with respect to the given Region
493
494 if (chart_scale < chart_native_scale / 10) {
495 // Scan the points one-by-one, so that we can get min/max to make a bbox
496 float *pfp = llpoints;
497 float lon_max = -10000.;
498 float lon_min = 10000.;
499 float lat_max = -10000.;
500 float lat_min = 10000.;
501
502 for (int ip = 0; ip < nPoints; ip++) {
503 lon_max = wxMax(lon_max, pfp[1]);
504 lon_min = wxMin(lon_min, pfp[1]);
505 lat_max = wxMax(lat_max, pfp[0]);
506 lat_min = wxMin(lat_min, pfp[0]);
507
508 pfp += 2;
509 }
510
511 LLBBox chart_box;
512 chart_box.Set(lat_min, lon_min, lat_max, lon_max);
513
514 // Case: vpBBox is completely outside the chart box, or vice versa
515 // Return an empty region
516 if (chart_box.IntersectOut(vpBBox)) return OCPNRegion();
517
518 // Case: vpBBox is completely inside the chart box
519 // Note that this test is not perfect, and will fail for some charts.
520 // The chart coverage may be essentially triangular, and the viewport
521 // box may be in the "cut off" segment of the chart_box, and not
522 // actually exhibit any true overlap. Results will be reported
523 // incorrectly. How to fix: maybe scrub the chart points and see if it
524 // is likely that a region may be safely built and intersection tested.
525
526 if (chart_box.IntersectIn(vpBBox)) return Region;
527
528 wxPoint p1 = GetPixFromLL(lat_max, lon_min); // upper left
529 wxPoint p2 = GetPixFromLL(lat_min, lon_max); // lower right
530
531 OCPNRegion r(p1, p2);
532 r.Intersect(Region);
533 return r;
534 }
535
536 // More "normal" case
537
538 wxPoint *pp;
539
540 // Use the passed point buffer if available
541 if (ppoints == NULL)
542 pp = new wxPoint[nPoints];
543 else
544 pp = ppoints;
545
546 float *pfp = llpoints;
547
548 wxPoint p = GetPixFromLL(pfp[0], pfp[1]);
549 int poly_x_max = INVALID_COORD, poly_y_max = INVALID_COORD,
550 poly_x_min = INVALID_COORD, poly_y_min = INVALID_COORD;
551
552 bool valid = false;
553 int npPoints = 0;
554 for (int ip = 0; ip < nPoints; ip++) {
555 wxPoint p = GetPixFromLL(pfp[0], pfp[1]);
556 if (p.x == INVALID_COORD) continue;
557
558 pp[npPoints++] = p;
559
560 if (valid) {
561 poly_x_max = wxMax(poly_x_max, p.x);
562 poly_y_max = wxMax(poly_y_max, p.y);
563 poly_x_min = wxMin(poly_x_min, p.x);
564 poly_y_min = wxMin(poly_y_min, p.y);
565 } else {
566 poly_x_max = p.x;
567 poly_y_max = p.y;
568 poly_x_min = p.x;
569 poly_y_min = p.y;
570 valid = true;
571 }
572 pfp += 2;
573 }
574
575 if (!valid) {
576 if (ppoints == NULL) delete[] pp;
577 return OCPNRegion(); // empty;
578 }
579
580 // We want to avoid processing regions with very large rectangle counts,
581 // so make some tests for special cases
582
583 float_2Dpt p0, p1, p2, p3;
584
585 // First, calculate whether any segment of the input polygon intersects the
586 // specified Region
587 int nrect = 0;
588 bool b_intersect = false;
589 OCPNRegionIterator screen_region_it1(Region);
590 while (screen_region_it1.HaveRects()) {
591 wxRect rect = screen_region_it1.GetRect();
592
593 double lat, lon;
594
595 // The screen region corners
596 GetLLFromPix(wxPoint(rect.x, rect.y), &lat, &lon);
597 p0.y = lat;
598 p0.x = lon;
599
600 GetLLFromPix(wxPoint(rect.x + rect.width, rect.y), &lat, &lon);
601 p1.y = lat;
602 p1.x = lon;
603
604 GetLLFromPix(wxPoint(rect.x + rect.width, rect.y + rect.height), &lat,
605 &lon);
606 p2.y = lat;
607 p2.x = lon;
608
609 GetLLFromPix(wxPoint(rect.x, rect.y + rect.height), &lat, &lon);
610 p3.y = lat;
611 p3.x = lon;
612
613 for (int i = 0; i < npPoints - 1; i++) {
614 // Quick check on y dimension
615 int y0 = pp[i].y;
616 int y1 = pp[i + 1].y;
617
618 if (((y0 < rect.y) && (y1 < rect.y)) ||
619 ((y0 > rect.y + rect.height) && (y1 > rect.y + rect.height)))
620 continue; // both ends of line outside of box, top or bottom
621
622 // Look harder
623 float_2Dpt f0;
624 f0.y = llpoints[i * 2];
625 f0.x = llpoints[(i * 2) + 1];
626 float_2Dpt f1;
627 f1.y = llpoints[(i + 1) * 2];
628 f1.x = llpoints[((i + 1) * 2) + 1];
629 b_intersect |= Intersect_FL(p0, p1, f0, f1) != 0;
630 if (b_intersect) break;
631 b_intersect |= Intersect_FL(p1, p2, f0, f1) != 0;
632 if (b_intersect) break;
633 b_intersect |= Intersect_FL(p2, p3, f0, f1) != 0;
634 if (b_intersect) break;
635 b_intersect |= Intersect_FL(p3, p0, f0, f1) != 0;
636 if (b_intersect) break;
637
638 // Must check the case where the input polygon has been pre-normalized,
639 // eg (0 < lon < 360), as cm93
640 f0.x -= 360.;
641 f1.x -= 360.;
642 b_intersect |= Intersect_FL(p0, p1, f0, f1) != 0;
643 if (b_intersect) break;
644 b_intersect |= Intersect_FL(p1, p2, f0, f1) != 0;
645 if (b_intersect) break;
646 b_intersect |= Intersect_FL(p2, p3, f0, f1) != 0;
647 if (b_intersect) break;
648 b_intersect |= Intersect_FL(p3, p0, f0, f1) != 0;
649 if (b_intersect) break;
650 }
651
652 // Check segment, last point back to first point
653 if (!b_intersect) {
654 float_2Dpt f0;
655 f0.y = llpoints[(nPoints - 1) * 2];
656 f0.x = llpoints[((nPoints - 1) * 2) + 1];
657 float_2Dpt f1;
658 f1.y = llpoints[0];
659 f1.x = llpoints[1];
660 b_intersect |= Intersect_FL(p0, p1, f0, f1) != 0;
661 b_intersect |= Intersect_FL(p1, p2, f0, f1) != 0;
662 b_intersect |= Intersect_FL(p2, p3, f0, f1) != 0;
663 b_intersect |= Intersect_FL(p3, p0, f0, f1) != 0;
664
665 f0.x -= 360.;
666 f1.x -= 360.;
667 b_intersect |= Intersect_FL(p0, p1, f0, f1) != 0;
668 b_intersect |= Intersect_FL(p1, p2, f0, f1) != 0;
669 b_intersect |= Intersect_FL(p2, p3, f0, f1) != 0;
670 b_intersect |= Intersect_FL(p3, p0, f0, f1) != 0;
671 }
672
673 screen_region_it1.NextRect();
674 nrect++;
675 }
676
677 // If there is no itersection, we need to consider the case where
678 // the subject polygon is entirely within the Region
679 bool b_contained = false;
680 if (!b_intersect) {
681 OCPNRegionIterator screen_region_it2(Region);
682 while (screen_region_it2.HaveRects()) {
683 wxRect rect = screen_region_it2.GetRect();
684
685 for (int i = 0; i < npPoints - 1; i++) {
686 int x0 = pp[i].x;
687 int y0 = pp[i].y;
688
689 if ((x0 < rect.x) || (x0 > rect.x + rect.width) || (y0 < rect.y) ||
690 (y0 > rect.y + rect.height))
691 continue;
692
693 b_contained = true;
694 break;
695 }
696 screen_region_it2.NextRect();
697 }
698 }
699
700#if 1
701 // and here is the payoff
702 if (!b_contained && !b_intersect) {
703 // Two cases to consider
704 wxRect rpoly(poly_x_min, poly_y_min, poly_x_max - poly_x_min,
705 poly_y_max - poly_y_min);
706 wxRect rRegion = Region.GetBox();
707 if (rpoly.Contains(rRegion)) {
708 // subject poygon may be large enough to fully encompass the target
709 // Region, but it might not, especially for irregular or concave charts.
710 // So we cannot directly shortcut here
711 // Better check....
712
713#if 1
714 if (nrect == 1) { // most common case
715 // If the subject polygon contains the center of the target rectangle,
716 // then the intersection must be the target rectangle
717 float rlat = (p0.y + p2.y) / 2.;
718 float rlon = (p0.x + p1.x) / 2.;
719
720 if (G_PtInPolygon_FL((float_2Dpt *)llpoints, nPoints, rlon, rlat)) {
721 if (NULL == ppoints) delete[] pp;
722 return Region;
723 }
724 rlon += 360.;
725 if (G_PtInPolygon_FL((float_2Dpt *)llpoints, nPoints, rlon, rlat)) {
726 if (NULL == ppoints) delete[] pp;
727 return Region;
728 }
729
730 // otherwise, there is no intersection
731 else {
732 if (NULL == ppoints) delete[] pp;
733 wxRegion r;
734 return r;
735 }
736 }
737
738#endif
739
740 } else {
741 // Subject polygon is entirely outside of target Region
742 // so the intersection must be empty.
743 if (NULL == ppoints) delete[] pp;
744 wxRegion r;
745 return r;
746 }
747 } else if (b_contained && !b_intersect) {
748 // subject polygon is entirely withing the target Region,
749 // so the intersection is the subject polygon
750 OCPNRegion r = OCPNRegion(npPoints, pp);
751 if (NULL == ppoints) delete[] pp;
752 return r;
753 }
754
755#endif
756
757#ifdef __WXGTK__
758 sigaction(SIGSEGV, NULL,
759 &sa_all_old); // save existing action for this signal
760
761 struct sigaction temp;
762 sigaction(SIGSEGV, NULL, &temp); // inspect existing action for this signal
763
764 temp.sa_handler = catch_signals; // point to my handler
765 sigemptyset(&temp.sa_mask); // make the blocking set
766 // empty, so that all
767 // other signals will be
768 // unblocked during my handler
769 temp.sa_flags = 0;
770 sigaction(SIGSEGV, &temp, NULL);
771
772 if (sigsetjmp(env, 1)) // Something in the below code block faulted....
773 {
774 sigaction(SIGSEGV, &sa_all_old, NULL); // reset signal handler
775
776 return Region;
777
778 }
779
780 else {
781 OCPNRegion r = OCPNRegion(npPoints, pp);
782 if (NULL == ppoints) delete[] pp;
783
784 sigaction(SIGSEGV, &sa_all_old, NULL); // reset signal handler
785 r.Intersect(Region);
786 return r;
787 }
788
789#else
790 OCPNRegion r = OCPNRegion(npPoints, pp);
791
792 if (NULL == ppoints) delete[] pp;
793
794 r.Intersect(Region);
795 return r;
796
797#endif
798}
799
800wxRect ViewPort::GetVPRectIntersect(size_t n, float *llpoints) {
801 // Calculate the intersection between the currect VP screen
802 // and the bounding box of a polygon specified by lat/lon points.
803
804 float *pfp = llpoints;
805
806 BoundingBox point_box;
807 for (unsigned int ip = 0; ip < n; ip++) {
808 point_box.Expand(pfp[1], pfp[0]);
809 pfp += 2;
810 }
811
812 wxPoint pul = GetPixFromLL(point_box.GetMaxY(), point_box.GetMinX());
813 wxPoint plr = GetPixFromLL(point_box.GetMinY(), point_box.GetMaxX());
814
815 OCPNRegion r(pul, plr);
816 OCPNRegion rs(rv_rect);
817
818 r.Intersect(rs);
819
820 return r.GetBox();
821}
822
824 // In the case where canvas rotation is applied, we need to define a larger
825 // "virtual" pixel window size to ensure that enough chart data is fatched
826 // and available to fill the rotated screen.
827 rv_rect = wxRect(0, 0, pix_width, pix_height);
828
829 // Specify the minimum required rectangle in unrotated screen space which
830 // will supply full screen data after specified rotation
831 if ((fabs(skew) > .0001) || (fabs(rotation) > .0001)) {
832 double rotator = rotation;
833 double lpixh = pix_height;
834 double lpixw = pix_width;
835
836 lpixh = wxMax(lpixh,
837 (fabs(pix_height * cos(skew)) + fabs(pix_width * sin(skew))));
838 lpixw = wxMax(lpixw,
839 (fabs(pix_width * cos(skew)) + fabs(pix_height * sin(skew))));
840
841 int dy = wxRound(fabs(lpixh * cos(rotator)) + fabs(lpixw * sin(rotator)));
842 int dx = wxRound(fabs(lpixw * cos(rotator)) + fabs(lpixh * sin(rotator)));
843
844 // It is important for MSW build that viewport pixel dimensions be
845 // multiples of 4.....
846 if (dy % 4) dy += 4 - (dy % 4);
847 if (dx % 4) dx += 4 - (dx % 4);
848
849 int inflate_x = wxMax((dx - pix_width) / 2, 0);
850 int inflate_y = wxMax((dy - pix_height) / 2, 0);
851
852 // Grow the source rectangle appropriately
853 rv_rect.Inflate(inflate_x, inflate_y);
854 }
855
856 // Compute Viewport lat/lon reference points for co-ordinate hit testing
857
858 // This must be done in unrotated space with respect to full unrotated screen
859 // space calculated above
860 double rotation_save = rotation;
861 SetRotationAngle(0.0);
862
863 wxPoint ul(rv_rect.x, rv_rect.y); // Upper left.
864 wxPoint lr(rv_rect.x + rv_rect.width,
865 rv_rect.y + rv_rect.height); // Lower right.
866 double dlat_min, dlat_max, dlon_min, dlon_max;
867
868 bool hourglass = false;
869 switch (m_projection_type) {
870 case PROJECTION_TRANSVERSE_MERCATOR:
871 case PROJECTION_STEREOGRAPHIC:
872 case PROJECTION_GNOMONIC:
873 hourglass = true;
874 // fall through
875 case PROJECTION_POLYCONIC:
876 case PROJECTION_POLAR:
877 case PROJECTION_ORTHOGRAPHIC: {
878 double d;
879
880 if (clat > 0) { // north polar
881 wxPoint u(rv_rect.x + rv_rect.width / 2, rv_rect.y);
882 wxPoint ur(rv_rect.x + rv_rect.width, rv_rect.y);
883 GetLLFromPix(ul, &d, &dlon_min);
884 GetLLFromPix(ur, &d, &dlon_max);
885 GetLLFromPix(lr, &dlat_min, &d);
886 GetLLFromPix(u, &dlat_max, &d);
887
888 if (fabs(fabs(d - clon) - 180) < 1) { // the pole is onscreen
889 dlat_max = 90;
890 dlon_min = -180;
891 dlon_max = 180;
892 } else if (std::isnan(dlat_max))
893 dlat_max = 90;
894
895 if (hourglass) {
896 // near equator, center may be less
897 wxPoint l(rv_rect.x + rv_rect.width / 2, rv_rect.y + rv_rect.height);
898 double dlat_min2;
899 GetLLFromPix(l, &dlat_min2, &d);
900 dlat_min = wxMin(dlat_min, dlat_min2);
901 }
902
903 if (std::isnan(dlat_min)) // world is off-screen
904 dlat_min = clat - 90;
905 } else { // south polar
906 wxPoint l(rv_rect.x + rv_rect.width / 2, rv_rect.y + rv_rect.height);
907 wxPoint ll(rv_rect.x, rv_rect.y + rv_rect.height);
908 GetLLFromPix(ul, &dlat_max, &d);
909 GetLLFromPix(lr, &d, &dlon_max);
910 GetLLFromPix(ll, &d, &dlon_min);
911 GetLLFromPix(l, &dlat_min, &d);
912
913 if (fabs(fabs(d - clon) - 180) < 1) { // the pole is onscreen
914 dlat_min = -90;
915 dlon_min = -180;
916 dlon_max = 180;
917 } else if (std::isnan(dlat_min))
918 dlat_min = -90;
919
920 if (hourglass) {
921 // near equator, center may be less
922 wxPoint u(rv_rect.x + rv_rect.width / 2, rv_rect.y);
923 double dlat_max2;
924 GetLLFromPix(u, &dlat_max2, &d);
925 dlat_max = wxMax(dlat_max, dlat_max2);
926 }
927
928 if (std::isnan(dlat_max)) // world is off-screen
929 dlat_max = clat + 90;
930 }
931
932 if (std::isnan(dlon_min)) {
933 // if neither pole is visible, but left and right of the screen are in
934 // space we can avoid drawing the far side of the earth
935 if (dlat_max < 90 && dlat_min > -90) {
936 dlon_min =
937 clon - 90 -
938 fabs(clat); // this logic is not optimal, is it always correct?
939 dlon_max = clon + 90 + fabs(clat);
940 } else {
941 dlon_min = -180;
942 dlon_max = 180;
943 }
944 }
945 } break;
946
947 default: // works for mercator and equirectangular
948 {
949 GetLLFromPix(ul, &dlat_max, &dlon_min);
950 GetLLFromPix(lr, &dlat_min, &dlon_max);
951 }
952 }
953
954 if (clon < dlon_min)
955 dlon_min -= 360;
956 else if (clon > dlon_max)
957 dlon_max += 360;
958
959 // Set the viewport lat/lon bounding box appropriately
960 vpBBox.Set(dlat_min, dlon_min, dlat_max, dlon_max);
961
962 // Restore the rotation angle
963 SetRotationAngle(rotation_save);
964}
965
966void ViewPort::SetBBoxDirect(double latmin, double lonmin, double latmax,
967 double lonmax) {
968 vpBBox.Set(latmin, lonmin, latmax, lonmax);
969}
970bool ViewPort::ContainsIDL() {
971 if ((vpBBox.GetMinLon() <= -180.) && (vpBBox.GetMaxLon() > -180.))
972 return true;
973 if ((vpBBox.GetMinLon() <= 180.) && (vpBBox.GetMaxLon() > 180.)) return true;
974 return false;
975}
976
977ViewPort ViewPort::BuildExpandedVP(int width, int height) {
978 ViewPort new_vp = *this;
979
980 new_vp.pix_width = width;
981 new_vp.pix_height = height;
982 new_vp.SetBoxes();
983
984 return new_vp;
985}
986
987void ViewPort::SetVPTransformMatrix() {
988 mat4x4 m;
989 mat4x4_identity(m);
990 mat4x4_scale_aniso((float(*)[4])vp_matrix_transform, m,
991 2.0 / (float)pix_width, -2.0 / (float)pix_height, 1.0);
992 mat4x4_translate_in_place((float(*)[4])vp_matrix_transform, -pix_width / 2,
993 -pix_height / 2, 0);
994}
An iterator class for OCPNRegion.
Definition OCPNRegion.h:156
A wrapper class for wxRegion with additional functionality.
Definition OCPNRegion.h:56
ViewPort - Core geographic projection and coordinate transformation engine.
Definition viewport.h:81
double view_scale_ppm
Requested view scale in physical pixels per meter (ppm), before applying projections.
Definition viewport.h:229
int pix_height
Height of the viewport in physical pixels.
Definition viewport.h:258
void SetBoxes(void)
Computes the bounding box coordinates for the current viewport.
Definition viewport.cpp:823
double rotation
Rotation angle of the viewport in radians.
Definition viewport.h:239
void SetPixelScale(double scale)
Set the physical to logical pixel ratio for the display.
Definition viewport.cpp:133
wxRect GetVPRectIntersect(size_t n, float *llpoints)
Get the viewport rectangle intersecting with a set of lat/lon points.
Definition viewport.cpp:800
int pix_width
Width of the viewport in physical pixels.
Definition viewport.h:256
wxPoint2DDouble GetDoublePixFromLL(double lat, double lon)
Convert latitude and longitude on the ViewPort to physical pixel coordinates with double precision.
Definition viewport.cpp:145
double tilt
Tilt angle for perspective view in radians.
Definition viewport.h:241
double skew
Angular distortion (shear transform) applied to the viewport in radians.
Definition viewport.h:237
void GetLLFromPix(const wxPoint &p, double *lat, double *lon)
Convert physical pixel coordinates on the ViewPort to latitude and longitude.
Definition viewport.h:105
OCPNRegion GetVPRegionIntersect(const OCPNRegion &region, const LLRegion &llregion, int chart_native_scale)
Get the intersection of the viewport with a given region.
Definition viewport.cpp:421
double clon
Center longitude of the viewport in degrees.
Definition viewport.h:224
double clat
Center latitude of the viewport in degrees.
Definition viewport.h:222
wxPoint GetPixFromLL(double lat, double lon)
Convert latitude and longitude on the ViewPort to physical pixel coordinates.
Definition viewport.cpp:136
double chart_scale
Chart scale denominator (e.g., 50000 for a 1:50000 scale).
Definition viewport.h:244