OpenCPN Partial API docs
Loading...
Searching...
No Matches
shapefile_basemap.h
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#ifndef SHAPEFILE_BASEMAP_H
29#define SHAPEFILE_BASEMAP_H
30
31#include <functional>
32#include <vector>
33#include <map>
34#include <thread>
35#include <future>
36#include "ShapefileReader.hpp"
37#include "poly_math.h"
38#include "ocpndc.h"
39
40#if (defined(OCPN_GHC_FILESYSTEM) || \
41 (defined(__clang_major__) && (__clang_major__ < 15)))
42// MacOS 1.13
43#include <ghc/filesystem.hpp>
44namespace fs = ghc::filesystem;
45#else
46#include <filesystem>
47#include <utility>
48namespace fs = std::filesystem;
49#endif
50
63class LatLonKey {
64public:
72 LatLonKey(int lat, int lon) {
73 this->lat = lat;
74 this->lon = lon;
75 }
81 int lat;
87 int lon;
88
89 bool operator<(const LatLonKey &k) const {
90 if (this->lat < k.lat) {
91 return this->lon < k.lon;
92 }
93 return this->lat < k.lat;
94 }
95
96 bool operator==(const LatLonKey &other) const {
97 return (lat == other.lat && lon == other.lon);
98 }
99};
100
101template <>
102struct std::hash<LatLonKey> {
103 std::size_t operator()(const LatLonKey &k) const {
104 return 360 * k.lat + k.lon;
105 }
106};
107
109enum Quality {
111 crude,
113 low,
115 medium,
117 high,
120 full
121};
122
123typedef std::vector<wxRealPoint> contour;
124typedef std::vector<contour> contour_list;
125
135public:
136 ShapeBaseChart() = delete;
137 ShapeBaseChart(const std::string &filename, const size_t &min_scale,
138 const wxColor &color = *wxBLACK)
139 : _dmod(1),
140 _loading(false),
141 _is_usable(false),
142 _is_tiled(false),
143 _min_scale(min_scale),
144 _filename(filename),
145 _reader(nullptr),
146 _color(color) {
147 _is_usable = fs::exists(filename);
148 }
149
151 this->_filename = t._filename;
152 this->_is_usable = t._is_usable;
153 this->_is_tiled = t._is_tiled;
154 this->_min_scale = t._min_scale;
155 this->_reader = nullptr;
156 this->_color = t._color;
157 this->_dmod = t._dmod;
158 this->_loading = t._loading;
159 }
161 CancelLoading(); // Ensure async operation is done before cleanup.
162 delete _reader;
163 }
164
165 void SetColor(wxColor color) { _color = color; }
166
173 int _dmod;
174
185 bool LoadSHP();
192 bool IsUsable() { return _is_usable && !_loading; }
193 size_t MinScale() { return _min_scale; }
194 void RenderViewOnDC(ocpnDC &dc, ViewPort &vp) { DrawPolygonFilled(dc, vp); }
195 static const std::string ConstructPath(const std::string &dir,
196 const std::string &quality_suffix) {
197 return std::string(dir + fs::path::preferred_separator + "basemap_" +
198 quality_suffix + ".shp");
199 }
200
218 bool CrossesLand(double &lat1, double &lon1, double &lat2, double &lon2);
219
221 void CancelLoading();
222
223private:
224 std::future<bool> _loaded;
225 bool _loading;
226 bool _is_usable;
232 bool _is_tiled;
238 size_t _min_scale;
239 void DoDrawPolygonFilled(ocpnDC &pnt, ViewPort &vp,
240 const shp::Feature &feature);
241 void DoDrawPolygonFilledGL(ocpnDC &pnt, ViewPort &vp,
242 const shp::Feature &feature);
243 void DrawPolygonFilled(ocpnDC &pnt, ViewPort &vp);
244 void AddPointToTessList(shp::Point &point, ViewPort &vp, GLUtesselator *tobj,
245 bool idl);
246
251 std::string _filename;
258 shp::ShapefileReader *_reader;
265 std::unordered_map<LatLonKey, std::vector<size_t>> _tiles;
272 wxColor _color;
273
296 bool LineLineIntersect(const std::pair<double, double> &A,
297 const std::pair<double, double> &B,
298 const std::pair<double, double> &C,
299 const std::pair<double, double> &D);
300
319 bool PolygonLineIntersect(const shp::Feature &feature,
320 const std::pair<double, double> &A,
321 const std::pair<double, double> &B);
322};
323
330public:
332 ~ShapeBaseChartSet() { Cleanup(); }
333 static wxPoint2DDouble GetDoublePixFromLL(ViewPort &vp, double lat,
334 double lon);
335
336 void SetBasemapLandColor(wxColor color);
337 wxColor GetBasemapLandColor();
338
339 void RenderViewOnDC(ocpnDC &dc, ViewPort &vp);
340
341 ShapeBaseChart &SelectBaseMap(const size_t &scale);
347 bool IsUsable() {
348 return _basemap_map.size() > 0 && LowestQualityBaseMap().IsUsable();
349 }
350
362 bool CrossesLand(double lat1, double lon1, double lat2, double lon2) {
363 if (IsUsable()) {
364 return HighestQualityBaseMap().CrossesLand(lat1, lon1, lat2, lon2);
365 }
366 return false;
367 }
368
369 void Cleanup() {
370 for (auto &pair : _basemap_map) {
371 pair.second.CancelLoading();
372 }
373 _basemap_map.clear();
374 _loaded = false;
375 }
376 void Reset();
377
378private:
379 void LoadBasemaps(const std::string &dir);
380 void DrawPolygonFilled(ocpnDC &pnt, ViewPort &vp, wxColor const &color);
381 void DrawPolygonFilledGL(ocpnDC &pnt, int *pvc, ViewPort &vp,
382 wxColor const &color, bool idl);
390 ShapeBaseChart &LowestQualityBaseMap();
398 ShapeBaseChart &HighestQualityBaseMap();
399
400 bool _loaded;
406 wxColor land_color;
407
408 std::map<Quality, ShapeBaseChart> _basemap_map;
409};
410
411#endif
A latitude/longitude key for 1x1 or 10x10 degree grid tiles.
int lat
Integer latitude value representing the northern (top) boundary of a latitude band.
LatLonKey(int lat, int lon)
Constructor for creating a LatLonKey.
int lon
Integer longitude value representing the western (left) boundary of a longitude band.
Manages a set of ShapeBaseChart objects at different resolutions.
bool CrossesLand(double lat1, double lon1, double lat2, double lon2)
Determines if a line segment between two geographical points crosses any land mass.
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.
bool IsUsable()
Determines if the chart is ready to be used for rendering or spatial queries.
ViewPort - Core geographic projection and coordinate transformation engine.
Definition viewport.h:81
Device context class that can use either wxDC or OpenGL for drawing.
Definition ocpndc.h:64