OpenCPN Partial API docs
Loading...
Searching...
No Matches
shapefile_basemap.cpp
1/******************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: Shapefile basemap
5 *
6 ***************************************************************************
7 * Copyright (C) 2012-2023 by David S. Register *
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 * This program is distributed in the hope that it will be useful, *
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
17 * GNU General Public License for more details. *
18 * *
19 * You should have received a copy of the GNU General Public License *
20 * along with this program; if not, write to the *
21 * Free Software Foundation, Inc., *
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
23 ***************************************************************************
24 *
25 *
26 */
27
28// Include OCPNPlatform.h before shapefile_basemap.h to prevent obscure syntax
29// error when compiling with VS2022
30#include "OCPNPlatform.h"
31#include "shapefile_basemap.h"
32#include "chartbase.h"
33#include "glChartCanvas.h"
34
35#include "model/logger.h"
36
37#ifdef ocpnUSE_GL
38#include "shaders.h"
39#endif
40
41#ifdef __WXMSW__
42#define __CALL_CONVENTION //__stdcall
43#else
44#define __CALL_CONVENTION
45#endif
46
47extern OCPNPlatform *g_Platform;
48extern wxString gWorldShapefileLocation;
49
50#ifdef ocpnUSE_GL
51
52typedef union {
53 GLdouble data[6];
54 struct sGLvertex {
55 GLdouble x;
56 GLdouble y;
57 GLdouble z;
58 GLdouble r;
59 GLdouble g;
60 GLdouble b;
61 } info;
62} GLvertexshp;
63#include <list>
64
65static std::list<float_2Dpt> g_pvshp;
66static std::list<GLvertexshp *> g_vertexesshp;
67static int g_typeshp, g_posshp;
68static float_2Dpt g_p1shp, g_p2shp;
69
70void __CALL_CONVENTION shpscombineCallback(GLdouble coords[3],
71 GLdouble *vertex_data[4],
72 GLfloat weight[4],
73 GLdouble **dataOut) {
74 GLvertexshp *vertex;
75
76 vertex = new GLvertexshp();
77 g_vertexesshp.push_back(vertex);
78
79 vertex->info.x = coords[0];
80 vertex->info.y = coords[1];
81
82 *dataOut = vertex->data;
83}
84
85void __CALL_CONVENTION shpserrorCallback(GLenum errorCode) {
86 const GLubyte *estring;
87 estring = gluErrorString(errorCode);
88 // wxLogMessage( _T("OpenGL Tessellation Error: %s"), estring );
89}
90
91void __CALL_CONVENTION shpsbeginCallback(GLenum type) {
92 switch (type) {
93 case GL_TRIANGLES:
94 case GL_TRIANGLE_STRIP:
95 case GL_TRIANGLE_FAN:
96 g_typeshp = type;
97 break;
98 default:
99 printf("tess unhandled begin type: %d\n", type);
100 }
101
102 g_posshp = 0;
103}
104
105void __CALL_CONVENTION shpsendCallback() {}
106
107void __CALL_CONVENTION shpsvertexCallback(GLvoid *arg) {
108 GLvertexshp *vertex;
109 vertex = (GLvertexshp *)arg;
110 float_2Dpt p;
111 p.y = vertex->info.x;
112 p.x = vertex->info.y;
113
114 // convert strips and fans into triangles
115 if (g_typeshp != GL_TRIANGLES) {
116 if (g_posshp > 2) {
117 g_pvshp.push_back(g_p1shp);
118 g_pvshp.push_back(g_p2shp);
119 }
120
121 if (g_typeshp == GL_TRIANGLE_STRIP)
122 g_p1shp = g_p2shp;
123 else if (g_posshp == 0)
124 g_p1shp = p;
125 g_p2shp = p;
126 }
127
128 g_pvshp.push_back(p);
129 g_posshp++;
130}
131#endif
132
133ShapeBaseChartSet::ShapeBaseChartSet() : _loaded(false) {
134 land_color = wxColor(170, 175, 80);
135}
136void ShapeBaseChartSet::SetBasemapLandColor(wxColor color) {
137 land_color = color;
138}
139wxColor ShapeBaseChartSet::GetBasemapLandColor() { return land_color; }
140
141wxPoint2DDouble ShapeBaseChartSet::GetDoublePixFromLL(ViewPort &vp, double lat,
142 double lon) {
143 wxPoint2DDouble p = vp.GetDoublePixFromLL(lat, lon);
144 p.m_x -= vp.rv_rect.x, p.m_y -= vp.rv_rect.y;
145 return p;
146}
147
148ShapeBaseChart &ShapeBaseChartSet::LowestQualityBaseMap() {
149 if (_basemap_map.find(Quality::crude) != _basemap_map.end()) {
150 return _basemap_map.at(Quality::crude);
151 }
152 if (_basemap_map.find(Quality::low) != _basemap_map.end()) {
153 return _basemap_map.at(Quality::low);
154 }
155 if (_basemap_map.find(Quality::medium) != _basemap_map.end()) {
156 return _basemap_map.at(Quality::medium);
157 }
158 if (_basemap_map.find(Quality::high) != _basemap_map.end()) {
159 return _basemap_map.at(Quality::high);
160 }
161 return _basemap_map.at(Quality::full);
162}
163
164ShapeBaseChart &ShapeBaseChartSet::HighestQualityBaseMap() {
165 if (_basemap_map.find(Quality::full) != _basemap_map.end()) {
166 return _basemap_map.at(Quality::full);
167 }
168 if (_basemap_map.find(Quality::high) != _basemap_map.end()) {
169 return _basemap_map.at(Quality::high);
170 }
171 if (_basemap_map.find(Quality::medium) != _basemap_map.end()) {
172 return _basemap_map.at(Quality::medium);
173 }
174 if (_basemap_map.find(Quality::low) != _basemap_map.end()) {
175 return _basemap_map.at(Quality::low);
176 }
177 return _basemap_map.at(Quality::crude);
178}
179
180ShapeBaseChart &ShapeBaseChartSet::SelectBaseMap(const size_t &scale) {
181 if (_basemap_map.find(Quality::full) != _basemap_map.end() &&
182 _basemap_map.at(Quality::full).IsUsable() &&
183 scale <= _basemap_map.at(Quality::full).MinScale()) {
184 return _basemap_map.at(Quality::full);
185 } else if (_basemap_map.find(Quality::high) != _basemap_map.end() &&
186 _basemap_map.at(Quality::high).IsUsable() &&
187 scale <= _basemap_map.at(Quality::high).MinScale()) {
188 return _basemap_map.at(Quality::high);
189 } else if (_basemap_map.find(Quality::medium) != _basemap_map.end() &&
190 _basemap_map.at(Quality::medium).IsUsable() &&
191 scale <= _basemap_map.at(Quality::medium).MinScale()) {
192 return _basemap_map.at(Quality::medium);
193 } else if (_basemap_map.find(Quality::low) != _basemap_map.end() &&
194 _basemap_map.at(Quality::low).IsUsable() &&
195 scale <= _basemap_map.at(Quality::low).MinScale()) {
196 return _basemap_map.at(Quality::low);
197 }
198 return LowestQualityBaseMap();
199}
200
201void ShapeBaseChartSet::Reset() {
202 // Establish the Shapefile basemap location
203 wxString basemap_dir;
204 if (gWorldShapefileLocation.empty()) {
205 basemap_dir = g_Platform->GetSharedDataDir();
206 basemap_dir.Append("basemap_shp");
207 } else {
208 basemap_dir = gWorldShapefileLocation;
209 }
210
211 LoadBasemaps(basemap_dir.ToStdString());
212}
213void ShapeBaseChartSet::LoadBasemaps(const std::string &dir) {
214 _loaded = false;
215 _basemap_map.clear();
216
217 if (fs::exists(ShapeBaseChart::ConstructPath(dir, "crude_10x10"))) {
218 auto c = ShapeBaseChart(ShapeBaseChart::ConstructPath(dir, "crude_10x10"),
219 300000000, land_color);
220 c._dmod = 10;
221 _basemap_map.insert(std::make_pair(Quality::crude, c));
222 }
223
224 if (fs::exists(ShapeBaseChart::ConstructPath(dir, "low"))) {
225 _basemap_map.insert(std::make_pair(
226 Quality::low, ShapeBaseChart(ShapeBaseChart::ConstructPath(dir, "low"),
227 15000000, land_color)));
228 }
229 if (fs::exists(ShapeBaseChart::ConstructPath(dir, "medium"))) {
230 _basemap_map.insert(std::make_pair(
231 Quality::medium,
232 ShapeBaseChart(ShapeBaseChart::ConstructPath(dir, "medium"), 1000000,
233 land_color)));
234 }
235 if (fs::exists(ShapeBaseChart::ConstructPath(dir, "high"))) {
236 _basemap_map.insert(std::make_pair(
237 Quality::high,
238 ShapeBaseChart(ShapeBaseChart::ConstructPath(dir, "high"), 300000,
239 land_color)));
240 }
241 if (fs::exists(ShapeBaseChart::ConstructPath(dir, "full"))) {
242 _basemap_map.insert(std::make_pair(
243 Quality::full,
244 ShapeBaseChart(ShapeBaseChart::ConstructPath(dir, "full"), 50000,
245 land_color)));
246 }
247 _loaded = true;
248 // if(_basemap_map.size())
249 // LowestQualityBaseMap().LoadSHP();
250}
251
253 if (!fs::exists(_filename)) {
254 _is_usable = false;
255 return false;
256 }
257 std::unique_ptr<shp::ShapefileReader> temp_reader(
258 new shp::ShapefileReader(_filename));
259 if (!temp_reader->isOpen()) {
260 MESSAGE_LOG << "Shapefile " << _filename << " is not opened";
261 _is_usable = false;
262 return false;
263 }
264 if (!_loading) {
265 // Check if loading was cancelled
266 return false;
267 }
268 auto bounds = temp_reader->getBounds();
269 _is_usable = temp_reader->getCount() > 1 && bounds.getMaxX() <= 180 &&
270 bounds.getMinX() >= -180 && bounds.getMinY() >= -90 &&
271 bounds.getMaxY() <=
272 90; // TODO - Do we care whether the planet is covered?
273 if (!_loading) {
274 // Check if loading was cancelled
275 _is_usable = false;
276 return false;
277 }
278 _is_usable &= temp_reader->getGeometryType() == shp::GeometryType::Polygon;
279 bool has_x = false;
280 bool has_y = false;
281 for (auto field : temp_reader->getFields()) {
282 if (field.getName() == "x") {
283 has_x = true;
284 } else if (field.getName() == "y") {
285 has_y = true;
286 }
287 }
288 _is_tiled = (has_x && has_y);
289 if (_is_usable && _is_tiled) {
290 size_t feat{0};
291 for (auto const &feature : *temp_reader) {
292 if (!_loading) {
293 // Check if loading was cancelled
294 _is_usable = false;
295 return false;
296 }
297 auto f1 = feature.getAttributes();
298 // Create a LatLonKey using the 'y' (latitude) and 'x' (longitude)
299 // attributes These values represent the top-left corner of the tiles
300 _tiles[LatLonKey(std::any_cast<int>(feature.getAttributes()["y"]),
301 std::any_cast<int>(feature.getAttributes()["x"]))]
302 .push_back(feat);
303 feat++;
304 }
305 }
306 if (_loading) { // Only set reader if loading wasn't cancelled
307 _reader = temp_reader.release();
308 }
309 return _is_usable;
310}
311
312void ShapeBaseChart::DoDrawPolygonFilled(ocpnDC &pnt, ViewPort &vp,
313 const shp::Feature &feature) {
314 double old_x = -9999999.0, old_y = -9999999.0;
315 auto polygon = static_cast<shp::Polygon *>(feature.getGeometry());
316 pnt.SetBrush(_color);
317 for (auto &ring : polygon->getRings()) {
318 wxPoint *poly_pt = new wxPoint[ring.getPoints().size()];
319 size_t cnt{0};
320 auto bbox = vp.GetBBox();
321 for (auto &point : ring.getPoints()) {
322 // if (bbox.ContainsMarge(point.getY(), point.getX(), 0.05)) {
323 wxPoint2DDouble q =
324 ShapeBaseChartSet::GetDoublePixFromLL(vp, point.getY(), point.getX());
325 if (round(q.m_x) != round(old_x) || round(q.m_y) != round(old_y)) {
326 poly_pt[cnt].x = round(q.m_x);
327 poly_pt[cnt].y = round(q.m_y);
328 cnt++;
329 }
330 old_x = q.m_x;
331 old_y = q.m_y;
332 //}
333 }
334 if (cnt > 1) {
335 pnt.DrawPolygonTessellated(cnt, poly_pt, 0, 0);
336 }
337 delete[] poly_pt;
338 }
339}
340
341void ShapeBaseChart::AddPointToTessList(shp::Point &point, ViewPort &vp,
342 GLUtesselator *tobj, bool idl) {
343#ifdef ocpnUSE_GL
344
345 wxPoint2DDouble q;
346 if (glChartCanvas::HasNormalizedViewPort(vp)) {
347 q = ShapeBaseChartSet::GetDoublePixFromLL(vp, point.getY(), point.getX());
348 } else { // tesselation directly from lat/lon
349 q.m_x = point.getY(), q.m_y = point.getX();
350 }
351 GLvertexshp *vertex = new GLvertexshp();
352 g_vertexesshp.push_back(vertex);
353 if (vp.m_projection_type != PROJECTION_POLAR) {
354 // need to correctly pick +180 or -180 longitude for projections
355 // that have a discontiguous date line
356
357 if (idl && (point.getX() == 180)) {
358 if (vp.m_projection_type == PROJECTION_MERCATOR ||
359 vp.m_projection_type == PROJECTION_EQUIRECTANGULAR) {
360 // q.m_x -= 40058986 * 4096.0; // 360 degrees in normalized
361 // viewport
362 } else {
363 q.m_x -= 360; // lat/lon coordinates
364 }
365 }
366 }
367 vertex->info.x = q.m_x;
368 vertex->info.y = q.m_y;
369
370 gluTessVertex(tobj, (GLdouble *)vertex, (GLdouble *)vertex);
371#endif
372}
373
374void ShapeBaseChart::DoDrawPolygonFilledGL(ocpnDC &pnt, ViewPort &vp,
375 const shp::Feature &feature) {
376#ifdef ocpnUSE_GL
377
378 bool idl =
379 vp.GetBBox().GetMinLon() <= -180 || vp.GetBBox().GetMaxLon() >= 180;
380 auto polygon = static_cast<shp::Polygon *>(feature.getGeometry());
381 for (auto &ring : polygon->getRings()) {
382 size_t cnt{0};
383 GLUtesselator *tobj = gluNewTess();
384
385 gluTessCallback(tobj, GLU_TESS_VERTEX, (_GLUfuncptr)&shpsvertexCallback);
386 gluTessCallback(tobj, GLU_TESS_BEGIN, (_GLUfuncptr)&shpsbeginCallback);
387 gluTessCallback(tobj, GLU_TESS_END, (_GLUfuncptr)&shpsendCallback);
388 gluTessCallback(tobj, GLU_TESS_COMBINE, (_GLUfuncptr)&shpscombineCallback);
389 gluTessCallback(tobj, GLU_TESS_ERROR, (_GLUfuncptr)&shpserrorCallback);
390
391 gluTessNormal(tobj, 0, 0, 1);
392 gluTessProperty(tobj, GLU_TESS_WINDING_RULE, GLU_TESS_WINDING_NONZERO);
393
394 gluTessBeginPolygon(tobj, NULL);
395 gluTessBeginContour(tobj);
396 for (auto &point : ring.getPoints()) {
397 AddPointToTessList(point, vp, tobj, idl);
398 cnt++;
399 }
400 gluTessEndContour(tobj);
401 gluTessEndPolygon(tobj);
402 gluDeleteTess(tobj);
403
404 for (auto ver : g_vertexesshp) delete ver;
405 g_vertexesshp.clear();
406 }
407 float_2Dpt *polyv = new float_2Dpt[g_pvshp.size()];
408 int cnt = 0;
409 for (auto pt : g_pvshp) {
410 polyv[cnt++] = pt;
411 }
412 size_t polycnt = g_pvshp.size();
413 g_pvshp.clear();
414
415 GLuint vbo = 0;
416
417 // Build the shader viewport transform matrix
418 mat4x4 m, mvp;
419 mat4x4_identity(m);
420 mat4x4_scale_aniso(mvp, m, 2.0 / (float)vp.pix_width,
421 2.0 / (float)vp.pix_height, 1.0);
422 mat4x4_translate_in_place(mvp, -vp.pix_width / 2, vp.pix_height / 2, 0);
423
424 float *pvt = new float[2 * (polycnt)];
425 for (size_t i = 0; i < polycnt; i++) {
426 float_2Dpt *pc = polyv + i;
427 // wxPoint2DDouble q(pc->y, pc->x);// = vp.GetDoublePixFromLL(pc->y, pc->x);
428 wxPoint2DDouble q = vp.GetDoublePixFromLL(pc->y, pc->x);
429
430 pvt[i * 2] = q.m_x;
431 pvt[(i * 2) + 1] = q.m_y;
432 }
433
434 GLShaderProgram *shader = pcolor_tri_shader_program[pnt.m_canvasIndex];
435 shader->Bind();
436
437 float colorv[4];
438 colorv[0] = _color.Red() / float(256);
439 colorv[1] = _color.Green() / float(256);
440 colorv[2] = _color.Blue() / float(256);
441 colorv[3] = 1.0;
442 shader->SetUniform4fv("color", colorv);
443
444 shader->SetAttributePointerf("position", pvt);
445
446 glDrawArrays(GL_TRIANGLES, 0, polycnt);
447
448 delete[] pvt;
449 delete[] polyv;
450
451 glDeleteBuffers(1, &vbo);
452 shader->UnBind();
453#endif
454}
455
456void ShapeBaseChart::DrawPolygonFilled(ocpnDC &pnt, ViewPort &vp) {
457 if (!_is_usable) {
458 return;
459 }
460 if (!_reader && !_loading) {
461 _loading = true;
462 _loaded = std::async(std::launch::async, [&]() {
463 bool ret = LoadSHP();
464 _loading = false;
465 return ret;
466 });
467 }
468 if (_loading) {
469 if (_loaded.wait_for(std::chrono::milliseconds(0)) ==
470 std::future_status::ready) {
471 _is_usable = _loaded.get();
472 } else {
473 return; // not yet loaded
474 }
475 }
476
477 int pmod = _dmod;
478 LLBBox bbox = vp.GetBBox();
479
480 // Calculate the starting latitude for the tiles grid
481 // Using floor() aligns with the top-left corner tile model
482 int lat_start = floor(bbox.GetMinLat());
483 if (lat_start < 0)
484 lat_start = lat_start - (pmod + (lat_start % pmod));
485 else
486 lat_start = lat_start - (lat_start % pmod);
487
488 // Calculate the starting longitude for the tiles grid
489 // Using floor() aligns with the top-left corner tile model
490 int lon_start = floor(bbox.GetMinLon());
491 if (lon_start < 0)
492 lon_start = lon_start - (pmod + (lon_start % pmod));
493 else
494 lon_start = lon_start - (lon_start % pmod);
495
496 if (_is_tiled) {
497 for (int i = lat_start; i < ceil(bbox.GetMaxLat()) + pmod; i += pmod) {
498 for (int j = lon_start; j < ceil(bbox.GetMaxLon()) + pmod; j += pmod) {
499 int lon{j};
500 if (j < -180) {
501 lon = j + 360;
502 } else if (j >= 180) {
503 lon = j - 360;
504 }
505 for (auto fid : _tiles[LatLonKey(i, lon)]) {
506 auto const &feature = _reader->getFeature(fid);
507 if (pnt.GetDC()) {
508 DoDrawPolygonFilled(pnt, vp,
509 feature); // Parallelize using std::async?
510 } else {
511 DoDrawPolygonFilledGL(pnt, vp,
512 feature); // Parallelize using std::async?
513 }
514 }
515 }
516 }
517 } else {
518 for (auto const &feature : *_reader) {
519 if (pnt.GetDC()) {
520 DoDrawPolygonFilled(pnt, vp,
521 feature); // Parallelize using std::async?
522 } else {
523 DoDrawPolygonFilledGL(pnt, vp,
524 feature); // Parallelize using std::async?
525 }
526 }
527 }
528}
529
530bool ShapeBaseChart::CrossesLand(double &lat1, double &lon1, double &lat2,
531 double &lon2) {
532 if (!_reader && !_loading) {
533 _loading = true;
534 _loaded = std::async(std::launch::async, [&]() {
535 bool ret = LoadSHP();
536 _loading = false;
537 return ret;
538 });
539 }
540 if (_loading) {
541 if (_loaded.wait_for(std::chrono::milliseconds(0)) ==
542 std::future_status::ready) {
543 _is_usable = _loaded.get();
544 } else {
545 // Chart not yet loaded. Assume no land crossing.
546 return false;
547 }
548 }
549
550 double norm_lon1 = lon1;
551 double norm_lon2 = lon2;
552
553 while (norm_lon1 < -180) norm_lon1 += 360;
554 while (norm_lon1 > 180) norm_lon1 -= 360;
555 while (norm_lon2 < -180) norm_lon2 += 360;
556 while (norm_lon2 > 180) norm_lon2 -= 360;
557
558 // Create normalized coordinate pairs
559 auto A = std::make_pair(lat1, norm_lon1);
560 auto B = std::make_pair(lat2, norm_lon2);
561
562 if (_is_tiled) {
563 // Calculate grid-aligned min/max coordinates based on _dmod
564 double minLat = std::min(lat1, lat2);
565 double maxLat = std::max(lat1, lat2);
566 double minLon = std::min(norm_lon1, norm_lon2);
567 double maxLon = std::max(norm_lon1, norm_lon2);
568
569 // Calculate grid-aligned indices based on the top-left corner model
570 // For latitudes: ceil(lat) gives the top edge of the tile containing that
571 // latitude For longitudes: floor(lon) gives the left edge of the tile
572 // containing that longitude
573 int latmax = ceil(maxLat / _dmod) * _dmod;
574 int latmin = floor(minLat / _dmod) * _dmod;
575 int lonmin = floor(minLon / _dmod) * _dmod;
576 int lonmax = ceil(maxLon / _dmod) * _dmod;
577
578 // Adjust the loop to iterate through all relevant tiles
579 for (int i = latmin; i <= latmax; i += _dmod) {
580 for (int j = lonmin; j < lonmax; j += _dmod) {
581 int lon = j;
582 // Normalize tile longitude to valid range
583 while (lon < -180) lon += 360;
584 while (lon >= 180) lon -= 360;
585
586 // Check if the tile exists in our map
587 auto tileIter = _tiles.find(LatLonKey(i, lon));
588 if (tileIter != _tiles.end()) {
589 for (auto fid : tileIter->second) {
590 auto const &feature = _reader->getFeature(fid);
591 if (PolygonLineIntersect(feature, A, B)) {
592 return true;
593 }
594 }
595 }
596 }
597 }
598 } else {
599 // Non-tiled case: check all features
600 for (auto const &feature : *_reader) {
601 if (PolygonLineIntersect(feature, A, B)) {
602 return true;
603 }
604 }
605 }
606
607 return false;
608}
609
611 if (_loading) {
612 _loading = false;
613 if (_loaded.valid()) {
614 _loaded.wait(); // Wait for async operation to complete
615 }
616 }
617}
618
619bool ShapeBaseChart::LineLineIntersect(const std::pair<double, double> &A,
620 const std::pair<double, double> &B,
621 const std::pair<double, double> &C,
622 const std::pair<double, double> &D) {
623 constexpr double EPSILON = 1e-9; // Tolerance for floating-point comparisons
624
625 // Fast bounding box check - early rejection for improved performance
626 double minABx = std::min(A.first, B.first);
627 double maxABx = std::max(A.first, B.first);
628 double minCDx = std::min(C.first, D.first);
629 double maxCDx = std::max(C.first, D.first);
630
631 if (maxABx < minCDx || maxCDx < minABx) return false;
632
633 double minABy = std::min(A.second, B.second);
634 double maxABy = std::max(A.second, B.second);
635 double minCDy = std::min(C.second, D.second);
636 double maxCDy = std::max(C.second, D.second);
637
638 if (maxABy < minCDy || maxCDy < minABy) return false;
639
640 // Line AB represented as a1x + b1y = c1
641 double a1 = B.second - A.second;
642 double b1 = A.first - B.first;
643 double c1 = a1 * A.first + b1 * A.second;
644
645 // Line CD represented as a2x + b2y = c2
646 double a2 = D.second - C.second;
647 double b2 = C.first - D.first;
648 double c2 = a2 * C.first + b2 * C.second;
649
650 double determinant = a1 * b2 - a2 * b1;
651
652 if (std::abs(determinant) < EPSILON) {
653 // Lines are parallel or collinear.
654 // Check if collinear by testing if points from one segment lie on the other
655 // line.
656 if (std::abs(a1 * C.first + b1 * C.second - c1) < EPSILON &&
657 std::abs(a1 * D.first + b1 * D.second - c1) < EPSILON) {
658 // Segments are collinear, check for overlap
659 return !(maxABx < minCDx || maxCDx < minABx || maxABy < minCDy ||
660 maxCDy < minABy);
661 }
662 return false; // Parallel but not collinear
663 }
664
665 // Calculate intersection point
666 double x = (b2 * c1 - b1 * c2) / determinant;
667 double y = (a1 * c2 - a2 * c1) / determinant;
668
669 // Check if intersection point lies on both segments
670 return (minABx <= x && x <= maxABx && minABy <= y && y <= maxABy &&
671 minCDx <= x && x <= maxCDx && minCDy <= y && y <= maxCDy);
672}
673
674bool ShapeBaseChart::PolygonLineIntersect(const shp::Feature &feature,
675 const std::pair<double, double> &A,
676 const std::pair<double, double> &B) {
677 auto geometry = feature.getGeometry();
678 if (!geometry) {
679 return false;
680 }
681
682 // Try to cast to polygon - if this fails, the geometry isn't a polygon
683 auto polygon = dynamic_cast<shp::Polygon *>(geometry);
684 if (!polygon) {
685 return false;
686 }
687
688 // Calculate line segment bounding box (lat, lon)
689 double minLat = std::min(A.first, B.first);
690 double maxLat = std::max(A.first, B.first);
691 double minLon = std::min(A.second, B.second);
692 double maxLon = std::max(A.second, B.second);
693
694 for (auto &ring : polygon->getRings()) {
695 const auto &points = ring.getPoints();
696 if (points.size() < 3) continue;
697
698 // Quick bounding box check for ring
699 double minRingLat = std::numeric_limits<double>::max();
700 double maxRingLat = std::numeric_limits<double>::lowest();
701 double minRingLon = std::numeric_limits<double>::max();
702 double maxRingLon = std::numeric_limits<double>::lowest();
703
704 // Pre-compute all point pairs to avoid repeated getX/getY calls
705 std::vector<std::pair<double, double>> ringPoints;
706 ringPoints.reserve(points.size());
707
708 for (const auto &point : points) {
709 double lat = point.getY();
710 double lon = point.getX();
711
712 minRingLat = std::min(minRingLat, lat);
713 maxRingLat = std::max(maxRingLat, lat);
714 minRingLon = std::min(minRingLon, lon);
715 maxRingLon = std::max(maxRingLon, lon);
716
717 ringPoints.emplace_back(lat, lon);
718 }
719
720 // Early reject if bounding boxes don't overlap
721 if (maxLat < minRingLat || minLat > maxRingLat || maxLon < minRingLon ||
722 minLon > maxRingLon) {
723 continue;
724 }
725
726 // Check each edge for intersection
727 size_t numPoints = ringPoints.size();
728 std::pair<double, double> previous = ringPoints.back();
729
730 for (size_t i = 0; i < numPoints; i++) {
731 const auto &current = ringPoints[i];
732
733 if (LineLineIntersect(A, B, previous, current)) {
734 return true;
735 }
736
737 previous = current;
738 }
739 }
740
741 return false;
742}
743
744void ShapeBaseChartSet::RenderViewOnDC(ocpnDC &dc, ViewPort &vp) {
745 if (IsUsable()) {
746 ShapeBaseChart &chart = SelectBaseMap(vp.chart_scale);
747 chart.SetColor(land_color);
748 chart.RenderViewOnDC(dc, vp);
749 }
750}
Wrapper class for OpenGL shader programs.
Definition shaders.h:57
A latitude/longitude key for 1x1 or 10x10 degree grid tiles.
Provides platform-specific support utilities for OpenCPN.
bool IsUsable()
Checks if the chart set contains at least one usable chart.
Represents a basemap chart based on shapefile data.
bool CrossesLand(double &lat1, double &lon1, double &lat2, double &lon2)
Determines if a line segment between two geographical points intersects any land mass represented in ...
void CancelLoading()
Cancel the chart loading operation.
int _dmod
Tile size in degrees.
bool LoadSHP()
Loads the shapefile data into memory.
ViewPort - Core geographic projection and coordinate transformation engine.
Definition viewport.h:81
int pix_height
Height of the viewport in physical pixels.
Definition viewport.h:258
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 chart_scale
Chart scale denominator (e.g., 50000 for a 1:50000 scale).
Definition viewport.h:244
Device context class that can use either wxDC or OpenGL for drawing.
Definition ocpndc.h:64
Enhanced logging interface on top of wx/log.h.