OpenCPN Partial API docs
Loading...
Searching...
No Matches
gshhs.cpp
1/******************************************************************************
2 *
3 * Project: OpenCPN
4 * Purpose: GSHHS Chart Object (Global Self-consistent, Hierarchical,
5 *High-resolution Shoreline) Author: Jesper Weissglas for the OpenCPN port.
6 *
7 * Derived from http://www.zygrib.org/ and
8 *http://sourceforge.net/projects/qtvlm/ which has the original copyright:
9 * zUGrib: meteorologic GRIB file data viewer
10 * Copyright (C) 2008 - Jacques Zaninetti - http://www.zygrib.org
11 *
12 ***************************************************************************
13 * Copyright (C) 2012 by David S. Register *
14 * *
15 * This program is free software; you can redistribute it and/or modify *
16 * it under the terms of the GNU General Public License as published by *
17 * the Free Software Foundation; either version 2 of the License, or *
18 * (at your option) any later version. *
19 * *
20 * This program is distributed in the hope that it will be useful, *
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
23 * GNU General Public License for more details. *
24 * *
25 * You should have received a copy of the GNU General Public License *
26 * along with this program; if not, write to the *
27 * Free Software Foundation, Inc., *
28 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
29 ***************************************************************************
30 *
31 *
32 */
33
34#include <wx/wxprec.h>
35
36#ifndef WX_PRECOMP
37#include <wx/wx.h>
38#endif
39
40#include <wx/file.h>
41
42#include "dychart.h"
43
44// #if defined(__OCPN__ANDROID__)
45// #include <GLES2/gl2.h>
46// #elif defined(__WXQT__) || defined(__WXGTK__)
47// #include <GL/glew.h>
48// //#define GL_GLEXT_PROTOTYPES
49// //#include <GL/gl.h>
50// //#include <GL/glext.h>
51// #endif
52
53#include "ocpndc.h"
54
55#ifdef ocpnUSE_GL
56#include "glChartCanvas.h"
57#endif
58
59#include "gshhs.h"
60#include "chartbase.h" // for projections
61#ifdef ocpnUSE_GL
62#include "shaders.h"
63#endif
64
65#ifdef __WXMSW__
66#define __CALL_CONVENTION //__stdcall
67#else
68#define __CALL_CONVENTION
69#endif
70
71#include "linmath.h"
72
73extern wxString gWorldMapLocation;
74
75#if defined(USE_ANDROID_GLES2) || defined(ocpnUSE_GLSL)
76static const GLchar *vertex_shader_source =
77 "attribute vec2 position;\n"
78 "uniform mat4 MVMatrix;\n"
79 "uniform mat4 TransformMatrix;\n"
80 "uniform vec4 color;\n"
81 "varying vec4 fragColor;\n"
82 "void main() {\n"
83 " fragColor = color;\n"
84 " gl_Position = MVMatrix * TransformMatrix * vec4(position, 0.0, 1.0);\n"
85 "}\n";
86
87static const GLchar *fragment_shader_source =
88 "precision lowp float;\n"
89 "varying vec4 fragColor;\n"
90 "void main() {\n"
91 " gl_FragColor = fragColor;\n"
92 "}\n";
93
94static const GLfloat vertices2[] = {
95 0.0f, 0.5f, 0.5f, -0.5f, -0.5f, -0.5f,
96};
97
98static const GLfloat vertices3[] = {
99 0.0f, 0.5f, 0.0f, 0.5f, -0.5f, 0.0f, -0.5f, -0.5f, 0.0f,
100};
101
102enum Consts { INFOLOG_LEN = 512 };
103GLchar infoLog[INFOLOG_LEN];
104GLint fragment_shader;
105// GLint shader_program;
106GLint success;
107GLint vertex_shader;
108
109extern GLint color_tri_shader_program;
110#endif
111
112//-------------------------------------------------------------------------
113
114GSHHSChart::GSHHSChart() {
115 reader = NULL;
116 land = wxColor(250, 250, 250);
117 water = wxColor(0, 0, 0);
118}
119
120GSHHSChart::~GSHHSChart() {
121 if (reader) delete reader;
122}
123
124void GSHHSChart::SetColorScheme(ColorScheme scheme) {
125 land = wxColor(170, 175, 80);
126 water = wxColor(170, 195, 240);
127
128 float dim = 1.0;
129
130 switch (scheme) {
131 case GLOBAL_COLOR_SCHEME_DUSK:
132 dim = 0.5;
133 break;
134 case GLOBAL_COLOR_SCHEME_NIGHT:
135 dim = 0.25;
136 break;
137 default:
138 return;
139 }
140
141 land.Set(land.Red() * dim, land.Green() * dim, land.Blue() * dim);
142 water.Set(water.Red() * dim, water.Green() * dim, water.Blue() * dim);
143}
144
145void GSHHSChart::SetColorsDirect(wxColour newLand, wxColour newWater) {
146 land = newLand;
147 water = newWater;
148}
149
150void GSHHSChart::Reset() {
151 if (reader) delete reader;
152 reader = NULL;
153}
154
155int GSHHSChart::GetMinAvailableQuality() {
156 if (!reader) reader = new GshhsReader();
157 return reader->GetMinAvailableQuality();
158}
159
160int GSHHSChart::GetMaxAvailableQuality() {
161 if (!reader) reader = new GshhsReader();
162 return reader->GetMaxAvailableQuality();
163}
164
165void GSHHSChart::RenderViewOnDC(ocpnDC &dc, ViewPort &vp) {
166 if (!reader) {
167 reader = new GshhsReader();
168 if (reader->GetPolyVersion() < 210 || reader->GetPolyVersion() > 240) {
169 wxLogMessage(
170 _T("GSHHS World chart files have wrong version. Found %d, expected ")
171 _T("210-220."),
172 reader->GetPolyVersion());
173 } else {
174 wxLogMessage(
175 _T("Background world map loaded from GSHHS datafiles found in: ") +
176 gWorldMapLocation);
177 }
178 }
179
180 reader->drawContinents(dc, vp, water, land);
181
182 /* this is very inefficient since it draws the entire world*/
183 // reader->drawBoundaries( dc, vp );
184}
185
186GshhsPolyCell::GshhsPolyCell(FILE *fpoly_, int x0_, int y0_,
187 PolygonFileHeader *header_) {
188 header = header_;
189 fpoly = fpoly_;
190 x0cell = x0_;
191 y0cell = y0_;
192
193 for (int i = 0; i < 6; i++) polyv[i] = NULL;
194
195 ReadPolygonFile();
196
197 for (int i = 0; i < GSSH_SUBM * GSSH_SUBM; i++) high_res_map[i] = NULL;
198}
199
200GshhsPolyCell::~GshhsPolyCell() {
201 ClearPolyV();
202
203 for (int i = 0; i < GSSH_SUBM * GSSH_SUBM; i++) delete high_res_map[i];
204 for (int i = 0; i < 6; i++) delete[] polyv[i];
205}
206
207void GshhsPolyCell::ClearPolyV() {
208 for (int i = 0; i < 6; i++) {
209 delete[] polyv[i];
210 polyv[i] = NULL;
211 }
212}
213
214void GshhsPolyCell::ReadPoly(contour_list &poly) {
215 double X, Y;
216 contour tmp_contour;
217 int32_t num_vertices, num_contours;
218 poly.clear();
219 if (fread(&num_contours, sizeof num_contours, 1, fpoly) != 1) goto fail;
220
221 for (int c = 0; c < num_contours; c++) {
222 int32_t value;
223 if (fread(&value, sizeof value, 1, fpoly) !=
224 1 || /* discarding hole value */
225 fread(&value, sizeof value, 1, fpoly) != 1)
226 goto fail;
227
228 num_vertices = value;
229
230 tmp_contour.clear();
231 for (int v = 0; v < num_vertices; v++) {
232 if (fread(&X, sizeof X, 1, fpoly) != 1 ||
233 fread(&Y, sizeof Y, 1, fpoly) != 1)
234 goto fail;
235
236 tmp_contour.push_back(wxRealPoint(X * GSHHS_SCL, Y * GSHHS_SCL));
237 }
238 poly.push_back(tmp_contour);
239 }
240 return;
241
242fail:
243 wxLogMessage(_T("gshhs ReadPoly failed"));
244}
245
246void GshhsPolyCell::ReadPolygonFile() {
247 if (!fpoly) return;
248
249 int pos_data;
250 int tab_data;
251
252 tab_data = (x0cell / header->pasx) * (180 / header->pasy) +
253 (y0cell + 90) / header->pasy;
254 fseek(fpoly, sizeof(PolygonFileHeader) + tab_data * sizeof(int), SEEK_SET);
255 if (fread(&pos_data, sizeof(int), 1, fpoly) != 1) goto fail;
256
257 fseek(fpoly, pos_data, SEEK_SET);
258
259 ReadPoly(poly1);
260 ReadPoly(poly2);
261 ReadPoly(poly3);
262 ReadPoly(poly4);
263 ReadPoly(poly5);
264 return;
265
266fail:
267 wxLogMessage(_T("gshhs ReadPolygon failed"));
268}
269
270wxPoint2DDouble GetDoublePixFromLL(ViewPort &vp, double lat, double lon) {
271 wxPoint2DDouble p = vp.GetDoublePixFromLL(lat, lon);
272 p.m_x -= vp.rv_rect.x, p.m_y -= vp.rv_rect.y;
273 return p;
274}
275
276void GshhsPolyCell::DrawPolygonFilled(ocpnDC &pnt, contour_list *p, double dx,
277 ViewPort &vp, wxColor const &color) {
278 if (!p->size()) /* size of 0 is very common, and setting the brush is
279 actually quite slow, so exit early */
280 return;
281
282 int x, y;
283 unsigned int c, v;
284 int pointCount;
285
286 int x_old = 0;
287 int y_old = 0;
288
289 pnt.SetBrush(color);
290
291 for (c = 0; c < p->size(); c++) {
292 if (!p->at(c).size()) continue;
293
294 wxPoint *poly_pt = new wxPoint[p->at(c).size()];
295
296 contour &cp = p->at(c);
297 pointCount = 0;
298
299 for (v = 0; v < p->at(c).size(); v++) {
300 wxRealPoint &ccp = cp.at(v);
301 wxPoint2DDouble q = GetDoublePixFromLL(vp, ccp.y, ccp.x + dx);
302 if (std::isnan(q.m_x)) {
303 pointCount = 0;
304 break;
305 }
306
307 x = q.m_x, y = q.m_y;
308
309 if (v == 0 || x != x_old || y != y_old) {
310 poly_pt[pointCount].x = x;
311 poly_pt[pointCount].y = y;
312 pointCount++;
313 x_old = x;
314 y_old = y;
315 }
316 }
317
318 if (pointCount > 1) pnt.DrawPolygonTessellated(pointCount, poly_pt, 0, 0);
319
320 delete[] poly_pt;
321 }
322}
323
324#ifdef ocpnUSE_GL
325
326typedef union {
327 GLdouble data[6];
328 struct sGLvertex {
329 GLdouble x;
330 GLdouble y;
331 GLdouble z;
332 GLdouble r;
333 GLdouble g;
334 GLdouble b;
335 } info;
336} GLvertex;
337
338#include <list>
339
340static std::list<float_2Dpt> g_pv;
341static std::list<GLvertex *> g_vertexes;
342static int g_type, g_pos;
343static float_2Dpt g_p1, g_p2;
344
345void __CALL_CONVENTION gshhscombineCallback(GLdouble coords[3],
346 GLdouble *vertex_data[4],
347 GLfloat weight[4],
348 GLdouble **dataOut) {
349 GLvertex *vertex;
350
351 vertex = new GLvertex();
352 g_vertexes.push_back(vertex);
353
354 vertex->info.x = coords[0];
355 vertex->info.y = coords[1];
356
357 *dataOut = vertex->data;
358}
359
360void __CALL_CONVENTION gshhsvertexCallback(GLvoid *arg) {
361 GLvertex *vertex;
362 vertex = (GLvertex *)arg;
363 float_2Dpt p;
364 p.y = vertex->info.x;
365 p.x = vertex->info.y;
366
367 // convert strips and fans into triangles
368 if (g_type != GL_TRIANGLES) {
369 if (g_pos > 2) {
370 g_pv.push_back(g_p1);
371 g_pv.push_back(g_p2);
372 }
373
374 if (g_type == GL_TRIANGLE_STRIP)
375 g_p1 = g_p2;
376 else if (g_pos == 0)
377 g_p1 = p;
378 g_p2 = p;
379 }
380
381 g_pv.push_back(p);
382 g_pos++;
383}
384
385void __CALL_CONVENTION gshhserrorCallback(GLenum errorCode) {
386 const GLubyte *estring;
387 estring = gluErrorString(errorCode);
388 // wxLogMessage( _T("OpenGL Tessellation Error: %s"), estring );
389}
390
391void __CALL_CONVENTION gshhsbeginCallback(GLenum type) {
392 switch (type) {
393 case GL_TRIANGLES:
394 case GL_TRIANGLE_STRIP:
395 case GL_TRIANGLE_FAN:
396 g_type = type;
397 break;
398 default:
399 printf("tess unhandled begin type: %d\n", type);
400 }
401
402 g_pos = 0;
403}
404
405void __CALL_CONVENTION gshhsendCallback() {}
406
407void GshhsPolyCell::DrawPolygonFilledGL(ocpnDC &pnt, contour_list *p,
408 float_2Dpt **pv, int *pvc, ViewPort &vp,
409 wxColor const &color, bool idl) {
410 if (!p->size()) // size of 0 is very common, exit early
411 return;
412
413 // build the contour vertex array converted to normalized coordinates (if
414 // needed)
415 if (!*pv) {
416 for (unsigned int c = 0; c < p->size(); c++) {
417 if (!p->at(c).size()) continue;
418
419 contour &cp = p->at(c);
420
421 GLUtesselator *tobj = gluNewTess();
422
423 gluTessCallback(tobj, GLU_TESS_VERTEX, (_GLUfuncptr)&gshhsvertexCallback);
424 gluTessCallback(tobj, GLU_TESS_BEGIN, (_GLUfuncptr)&gshhsbeginCallback);
425 gluTessCallback(tobj, GLU_TESS_END, (_GLUfuncptr)&gshhsendCallback);
426 gluTessCallback(tobj, GLU_TESS_COMBINE,
427 (_GLUfuncptr)&gshhscombineCallback);
428 gluTessCallback(tobj, GLU_TESS_ERROR, (_GLUfuncptr)&gshhserrorCallback);
429
430 gluTessNormal(tobj, 0, 0, 1);
431 gluTessProperty(tobj, GLU_TESS_WINDING_RULE, GLU_TESS_WINDING_NONZERO);
432
433 gluTessBeginPolygon(tobj, NULL);
434 gluTessBeginContour(tobj);
435
436 for (unsigned int v = 0; v < p->at(c).size(); v++) {
437 wxRealPoint &ccp = cp.at(v);
438
439 if (v == 0 || ccp != cp.at(v - 1)) {
440 GLvertex *vertex = new GLvertex();
441 g_vertexes.push_back(vertex);
442
443 wxPoint2DDouble q;
444 if (glChartCanvas::HasNormalizedViewPort(vp))
445 q = GetDoublePixFromLL(vp, ccp.y, ccp.x);
446 else // tesselation directly from lat/lon
447 q.m_x = ccp.y, q.m_y = ccp.x;
448
449 if (vp.m_projection_type != PROJECTION_POLAR) {
450 // need to correctly pick +180 or -180 longitude for projections
451 // that have a discontiguous date line
452
453 if (idl && ccp.x == 180) {
454 if (vp.m_projection_type != PROJECTION_MERCATOR &&
455 vp.m_projection_type != PROJECTION_EQUIRECTANGULAR)
456 q.m_x -= 360; // lat/lon coordinates
457 }
458 }
459
460 vertex->info.x = q.m_x;
461 vertex->info.y = q.m_y;
462
463 gluTessVertex(tobj, (GLdouble *)vertex, (GLdouble *)vertex);
464 }
465 }
466
467 gluTessEndContour(tobj);
468 gluTessEndPolygon(tobj);
469 gluDeleteTess(tobj);
470
471 for (std::list<GLvertex *>::iterator it = g_vertexes.begin();
472 it != g_vertexes.end(); it++)
473 delete *it;
474 g_vertexes.clear();
475 }
476
477 *pv = new float_2Dpt[g_pv.size()];
478 int i = 0;
479 for (std::list<float_2Dpt>::iterator it = g_pv.begin(); it != g_pv.end();
480 it++)
481 (*pv)[i++] = *it;
482
483 *pvc = g_pv.size();
484 g_pv.clear();
485 }
486
487#if defined(USE_ANDROID_GLES2) || defined(ocpnUSE_GLSL)
488
489#if 0
490 // Are the shaders ready?
491 if (!vertex_shader) {
492 /* Vertex shader */
493 vertex_shader = glCreateShader(GL_VERTEX_SHADER);
494 glShaderSource(vertex_shader, 1, &vertex_shader_source, NULL);
495 glCompileShader(vertex_shader);
496 glGetShaderiv(vertex_shader, GL_COMPILE_STATUS, &success);
497 if (!success) {
498 glGetShaderInfoLog(vertex_shader, INFOLOG_LEN, NULL, infoLog);
499 printf("ERROR::SHADER::VERTEX::COMPILATION_FAILED\n%s\n", infoLog);
500 }
501 }
502
503 if (!fragment_shader) {
504 /* Fragment shader */
505 fragment_shader = glCreateShader(GL_FRAGMENT_SHADER);
506 glShaderSource(fragment_shader, 1, &fragment_shader_source, NULL);
507 glCompileShader(fragment_shader);
508 glGetShaderiv(fragment_shader, GL_COMPILE_STATUS, &success);
509 if (!success) {
510 glGetShaderInfoLog(fragment_shader, INFOLOG_LEN, NULL, infoLog);
511 printf("ERROR::SHADER::FRAGMENT::COMPILATION_FAILED\n%s\n", infoLog);
512 }
513 }
514
515 if (!shader_program) {
516 /* Link shaders */
517 shader_program = glCreateProgram();
518 glAttachShader(shader_program, vertex_shader);
519 glAttachShader(shader_program, fragment_shader);
520 glLinkProgram(shader_program);
521 glGetProgramiv(shader_program, GL_LINK_STATUS, &success);
522 if (!success) {
523 glGetProgramInfoLog(shader_program, INFOLOG_LEN, NULL, infoLog);
524 printf("ERROR::SHADER::PROGRAM::LINKING_FAILED\n%s\n", infoLog);
525 }
526 }
527#endif
528 GLuint vbo = 0;
529
530 // Build the shader viewport transform matrix
531 mat4x4 m, mvp;
532 mat4x4_identity(m);
533 mat4x4_scale_aniso(mvp, m, 2.0 / (float)vp.pix_width,
534 2.0 / (float)vp.pix_height, 1.0);
535 mat4x4_translate_in_place(mvp, -vp.pix_width / 2, vp.pix_height / 2, 0);
536
537 if (glChartCanvas::HasNormalizedViewPort(vp)) {
538#if 0
539 GLint pos = glGetAttribLocation(color_tri_shader_program, "position");
540 glVertexAttribPointer(pos, 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), *pv);
541 glEnableVertexAttribArray(pos);
542
543 // FIXME glUniformMatrix4fv( matloc, 1, GL_FALSE, (const
544 // GLfloat*)vp.vp_transform);
545 mat4x4 m;
546 mat4x4_identity(m);
547 GLint tmatloc = glGetUniformLocation(color_tri_shader_program, "TransformMatrix");
548 glUniformMatrix4fv(tmatloc, 1, GL_FALSE, (const GLfloat*)m);
549
550 glUseProgram(color_tri_shader_program);
551 glDrawArrays(GL_TRIANGLES, 0, *pvc);
552#endif
553 } else {
554 float *pvt = new float[2 * (*pvc)];
555 for (int i = 0; i < *pvc; i++) {
556 float_2Dpt *pc = *pv + i;
557 wxPoint2DDouble q = vp.GetDoublePixFromLL(pc->y, pc->x);
558 pvt[i * 2] = q.m_x;
559 pvt[(i * 2) + 1] = q.m_y;
560 }
561
562 GLShaderProgram *shader = pcolor_tri_shader_program[pnt.m_canvasIndex];
563 shader->Bind();
564
565 float colorv[4];
566 colorv[0] = color.Red() / float(256);
567 colorv[1] = color.Green() / float(256);
568 colorv[2] = color.Blue() / float(256);
569 colorv[3] = 1.0;
570 shader->SetUniform4fv("color", colorv);
571
572 shader->SetAttributePointerf("position", pvt);
573
574 glDrawArrays(GL_TRIANGLES, 0, *pvc);
575
576 delete[] pvt;
577 glDeleteBuffers(1, &vbo);
578 shader->UnBind();
579 }
580
581#else
582#endif
583}
584#endif // #ifdef ocpnUSE_GL
585
586#define DRAW_POLY_FILLED(POLY, COL) \
587 if (POLY) DrawPolygonFilled(pnt, POLY, dx, vp, COL);
588#define DRAW_POLY_FILLED_GL(NUM, COL) \
589 DrawPolygonFilledGL(pnt, &poly##NUM, &polyv[NUM], &polyc[NUM], vp, COL, idl);
590
591void GshhsPolyCell::drawMapPlain(ocpnDC &pnt, double dx, ViewPort &vp,
592 wxColor seaColor, wxColor landColor,
593 bool idl) {
594#ifdef ocpnUSE_GL
595 if (!pnt.GetDC()) { // opengl
596#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
597#define NORM_FACTOR 4096.0
598 if (dx && (vp.m_projection_type == PROJECTION_MERCATOR ||
599 vp.m_projection_type == PROJECTION_EQUIRECTANGULAR)) {
600 double ts =
601 40058986 * NORM_FACTOR; /* 360 degrees in normalized viewport */
602 glPushMatrix();
603 glTranslated(dx > 0 ? ts : -ts, 0, 0);
604 }
605#endif
606 DRAW_POLY_FILLED_GL(1, landColor);
607 DRAW_POLY_FILLED_GL(2, seaColor);
608 DRAW_POLY_FILLED_GL(3, landColor);
609 DRAW_POLY_FILLED_GL(4, seaColor);
610 DRAW_POLY_FILLED_GL(5, landColor);
611
612#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
613 if (dx) glPopMatrix();
614#endif
615
616 } else
617#endif
618 {
619 DRAW_POLY_FILLED(&poly1, landColor);
620 DRAW_POLY_FILLED(&poly2, seaColor);
621 DRAW_POLY_FILLED(&poly3, landColor);
622 DRAW_POLY_FILLED(&poly4, seaColor);
623 DRAW_POLY_FILLED(&poly5, landColor);
624 }
625}
626
627void GshhsPolyCell::DrawPolygonContour(ocpnDC &pnt, contour_list *p, double dx,
628 ViewPort &vp) {
629 double x1, y1, x2, y2;
630 double long_max, lat_max, long_min, lat_min;
631
632 long_min = (double)x0cell;
633 lat_min = (double)y0cell;
634 long_max = ((double)x0cell + (double)header->pasx);
635 lat_max = ((double)y0cell + (double)header->pasy);
636
637 // qWarning() << long_min << "," << lat_min << long_max << "," << lat_max;
638
639 for (unsigned int i = 0; i < p->size(); i++) {
640 if (!p->at(i).size()) continue;
641
642 unsigned int v;
643 for (v = 0; v < (p->at(i).size() - 1); v++) {
644 x1 = p->at(i).at(v).x;
645 y1 = p->at(i).at(v).y;
646 x2 = p->at(i).at(v + 1).x;
647 y2 = p->at(i).at(v + 1).y;
648
649 // Elimination des traits verticaux et horizontaux
650 if ((((x1 == x2) && ((x1 == long_min) || (x1 == long_max))) ||
651 ((y1 == y2) && ((y1 == lat_min) || (y1 == lat_max)))) == 0) {
652 wxPoint2DDouble AB = GetDoublePixFromLL(vp, x1 + dx, y1);
653 wxPoint2DDouble CD = GetDoublePixFromLL(vp, x2 + dx, y1);
654 pnt.DrawLine(AB.m_x, AB.m_y, CD.m_x, CD.m_y);
655 }
656 }
657
658 x1 = p->at(i).at(v).x;
659 y1 = p->at(i).at(v).y;
660 x2 = p->at(i).at(0).x;
661 y2 = p->at(i).at(0).y;
662
663 if ((((x1 == x2) && ((x1 == long_min) || (x1 == long_max))) ||
664 ((y1 == y2) && ((y1 == lat_min) || (y1 == lat_max)))) == 0) {
665 wxPoint2DDouble AB = GetDoublePixFromLL(vp, x1 + dx, y1);
666 wxPoint2DDouble CD = GetDoublePixFromLL(vp, x2 + dx, y1);
667 pnt.DrawLine(AB.m_x, AB.m_y, CD.m_x, CD.m_y);
668 }
669 }
670}
671
672#define DRAW_POLY_CONTOUR(POLY) \
673 if (POLY) DrawPolygonContour(pnt, POLY, dx, vp);
674
675void GshhsPolyCell::drawSeaBorderLines(ocpnDC &pnt, double dx, ViewPort &vp) {
676 coasts.clear();
677 DRAW_POLY_CONTOUR(&poly1)
678 DRAW_POLY_CONTOUR(&poly2)
679 DRAW_POLY_CONTOUR(&poly3)
680 DRAW_POLY_CONTOUR(&poly4)
681 DRAW_POLY_CONTOUR(&poly5)
682}
683
684//========================================================================
685
686GshhsPolyReader::GshhsPolyReader(int quality) {
687 fpoly = NULL;
688
689 for (int i = 0; i < 360; i++) {
690 for (int j = 0; j < 180; j++) {
691 allCells[i][j] = NULL;
692 }
693 }
694 currentQuality = -1;
695 polyHeader.version = -1;
696 InitializeLoadQuality(quality);
697}
698
699//-------------------------------------------------------------------------
700GshhsPolyReader::~GshhsPolyReader() {
701 for (int i = 0; i < 360; i++) {
702 for (int j = 0; j < 180; j++) {
703 if (allCells[i][j] != NULL) {
704 delete allCells[i][j];
705 allCells[i][j] = NULL;
706 }
707 }
708 }
709
710 if (fpoly) {
711 fclose(fpoly);
712 }
713}
714
715//-------------------------------------------------------------------------
716int GshhsPolyReader::ReadPolyVersion() {
717 wxString fname = GshhsReader::getFileName_Land(0);
718 if (fpoly) fclose(fpoly);
719 fpoly = fopen(fname.mb_str(), "rb");
720
721 /* init header */
722 if (!fpoly) return 0;
723
724 readPolygonFileHeader(fpoly, &polyHeader);
725
726 return polyHeader.version;
727}
728
729void GshhsPolyReader::InitializeLoadQuality(
730 int quality) // 5 levels: 0=low ... 4=full
731{
732 if (currentQuality != quality) {
733 currentQuality = quality;
734
735 wxString fname = GshhsReader::getFileName_Land(quality);
736
737 if (fpoly) fclose(fpoly);
738
739 fpoly = fopen(fname.mb_str(), "rb");
740 if (fpoly) readPolygonFileHeader(fpoly, &polyHeader);
741
742 for (int i = 0; i < 360; i++) {
743 for (int j = 0; j < 180; j++) {
744 if (allCells[i][j] != NULL) {
745 delete allCells[i][j];
746 allCells[i][j] = NULL;
747 }
748 }
749 }
750 }
751}
752
753static inline bool my_intersects(const wxLineF &line1, const wxLineF &line2) {
754 double x1 = line1.m_p1.x, y1 = line1.m_p1.y, x2 = line1.m_p2.x,
755 y2 = line1.m_p2.y;
756 double x3 = line2.m_p1.x, y3 = line2.m_p1.y, x4 = line2.m_p2.x,
757 y4 = line2.m_p2.y;
758
759 // implementation is based on Graphics Gems III's "Faster Line Segment
760 // Intersection"
761 double ax = x2 - x1, ay = y2 - y1;
762 double bx = x3 - x4, by = y3 - y4;
763 double cx = x1 - x3, cy = y1 - y3;
764
765#define INTER_LIMIT 1e-7
766 double denominator = ay * bx - ax * by;
767 if (denominator < 1e-10) {
768 if (fabs((y1 * ax - ay * x1) * bx - (y3 * bx - by * x3) * ax) > INTER_LIMIT)
769 return false; /* different intercepts, no intersection */
770 if (fabs((x1 * ay - ax * y1) * by - (x3 * by - bx * y3) * ay) > INTER_LIMIT)
771 return false; /* different intercepts, no intersection */
772
773 return true;
774 }
775
776 const double reciprocal = 1 / denominator;
777 const double na = (by * cx - bx * cy) * reciprocal;
778
779 if (na < -INTER_LIMIT || na > 1 + INTER_LIMIT) return false;
780
781 const double nb = (ax * cy - ay * cx) * reciprocal;
782 if (nb < -INTER_LIMIT || nb > 1 + INTER_LIMIT) return false;
783
784 return true;
785}
786
787bool GshhsPolyReader::crossing1(wxLineF trajectWorld) {
788 double x1 = trajectWorld.p1().x, y1 = trajectWorld.p1().y;
789 double x2 = trajectWorld.p2().x, y2 = trajectWorld.p2().y;
790
791 int clonmin, clonmax, clatmax, clatmin;
792 clonmin = (int)floor(GSSH_SUBM * wxMin(x1, x2));
793 clonmax = (int)ceil(GSSH_SUBM * wxMax(x1, x2));
794
795 if (clonmin < 0) {
796 clonmin += GSSH_SUBM * 360;
797 clonmax += GSSH_SUBM * 360;
798 }
799
800 if (clonmax - clonmin > GSSH_SUBM * 180) { /* dont go long way around world */
801 clonmin = (int)floor(GSSH_SUBM * wxMax(x1, x2)) - GSSH_SUBM * 360;
802 clonmax = (int)ceil(GSSH_SUBM * wxMin(x1, x2));
803 }
804
805 clatmin = (int)floor(GSSH_SUBM * wxMin(y1, y2));
806 clatmax = (int)ceil(GSSH_SUBM * wxMax(y1, y2));
807 wxASSERT(clatmin >= -GSSH_SUBM * 90 && clatmax <= GSSH_SUBM * 89);
808
809 // TODO: optimize by traversing only the cells the segment passes through,
810 // rather than all of the cells which fit in the bounding box,
811 // this may make a worthwhile difference for longer segments in some
812 // cases.
813 int clon, clonx, clat;
814 for (clon = clonmin; clon < clonmax; clon++) {
815 clonx = clon;
816 while (clonx < 0) clonx += GSSH_SUBM * 360;
817 while (clonx >= GSSH_SUBM * 360) clonx -= GSSH_SUBM * 360;
818
819 wxASSERT(clonx >= 0 && clonx < GSSH_SUBM * 360);
820
821 if (clonx < GSSH_SUBM * 180) {
822 if (x1 > 180) x1 -= 360;
823 if (x2 > 180) x2 -= 360;
824 } else {
825 if (x1 < 180) x1 += 360;
826 if (x2 < 180) x2 += 360;
827 }
828
829 wxLineF rtrajectWorld(x1, y1, x2, y2);
830
831 for (clat = clatmin; clat < clatmax; clat++) {
832 int cloni = clonx / GSSH_SUBM,
833 clati = (GSSH_SUBM * 90 + clat) / GSSH_SUBM;
834 GshhsPolyCell *&cel = allCells[cloni][clati];
835 if (!cel) {
836 mutex1.Lock();
837 if (!cel) {
838 /* load the needed cell from disk */
839 cel = new GshhsPolyCell(fpoly, cloni, clati - 90, &polyHeader);
840 wxASSERT(cel);
841 }
842 mutex1.Unlock();
843 }
844
845 int hash = GSSH_SUBM * (GSSH_SUBM * (90 - clati) + clat - cloni) + clonx;
846 std::vector<wxLineF> *&high_res_map = cel->high_res_map[hash];
847 wxASSERT(hash >= 0 && hash < GSSH_SUBM * GSSH_SUBM);
848 if (!high_res_map) {
849 mutex2.Lock();
850 if (!high_res_map) {
851 /* Build the needed sub cell of line segments from the cell */
852 contour_list &poly1 = cel->getPoly1();
853
854 double minlat = (double)clat / GSSH_SUBM,
855 maxlat = (double)(clat + 1) / GSSH_SUBM;
856 double minlon = (double)clonx / GSSH_SUBM,
857 maxlon = (double)(clonx + 1) / GSSH_SUBM;
858 high_res_map = new std::vector<wxLineF>;
859 for (unsigned int pi = 0; pi < poly1.size(); pi++) {
860 contour &c = poly1[pi];
861 double lx = c[c.size() - 1].x, ly = c[c.size() - 1].y;
862 /* must compute states because sometimes a
863 segment starts and ends outside our cell, but passes
864 through it so must be included */
865 int lstatex = lx < minlon ? -1 : lx > maxlon ? 1 : 0;
866 int lstatey = ly < minlat ? -1 : ly > maxlat ? 1 : 0;
867
868 for (unsigned int pj = 0; pj < c.size(); pj++) {
869 double clon = c[pj].x, clat = c[pj].y;
870 // gshhs data shouldn't, but sometimes contains zero segments
871 // which enlarges our table, but
872 // more importantly, the fast segment intersection test
873 // and doesn't correctly account for it
874 if (lx == clon && ly == clat) continue;
875
876 int statex = clon < minlon ? -1 : clon > maxlon ? 1 : 0;
877 int statey = clat < minlat ? -1 : clat > maxlat ? 1 : 0;
878
879 if ((!statex || lstatex != statex) &&
880 (!statey || lstatey != statey))
881 high_res_map->push_back(wxLineF(lx, ly, clon, clat));
882
883 lx = clon, ly = clat;
884 lstatex = statex, lstatey = statey;
885 }
886 }
887 }
888 mutex2.Unlock();
889 }
890
891 for (std::vector<wxLineF>::iterator it2 = high_res_map->begin();
892 it2 != high_res_map->end(); it2++)
893 if (my_intersects(rtrajectWorld, *it2)) return true;
894 }
895 }
896
897 return false;
898}
899
900void GshhsPolyReader::readPolygonFileHeader(FILE *polyfile,
901 PolygonFileHeader *header) {
902 fseek(polyfile, 0, SEEK_SET);
903 if (fread(header, sizeof(PolygonFileHeader), 1, polyfile) != 1)
904 wxLogMessage(_T("gshhs ReadPolygonFileHeader failed"));
905}
906
907//-------------------------------------------------------------------------
908void GshhsPolyReader::drawGshhsPolyMapPlain(ocpnDC &pnt, ViewPort &vp,
909 wxColor const &seaColor,
910 wxColor const &landColor) {
911 if (!fpoly) return;
912
913 pnt.SetPen(wxNullPen);
914
915 int clonmin, clonmax, clatmax, clatmin; // cellules visibles
916 LLBBox bbox = vp.GetBBox();
917 clonmin = bbox.GetMinLon(), clonmax = bbox.GetMaxLon(),
918 clatmin = bbox.GetMinLat(), clatmax = bbox.GetMaxLat();
919 if (clatmin <= 0) clatmin--;
920 if (clatmax >= 0) clatmax++;
921 if (clonmin <= 0) clonmin--;
922 if (clonmax >= 0) clonmax++;
923 int dx, clon, clonx, clat;
924 GshhsPolyCell *cel;
925
926 ViewPort nvp = vp;
927#ifdef ocpnUSE_GL
928 if (!pnt.GetDC()) { // opengl
929 // clear cached data when the projection changes
930 if (vp.m_projection_type != last_rendered_vp.m_projection_type ||
931 (last_rendered_vp.m_projection_type == PROJECTION_POLAR &&
932 last_rendered_vp.clat * vp.clat <= 0)) {
933 last_rendered_vp = vp;
934 for (int clon = 0; clon < 360; clon++)
935 for (int clat = 0; clat < 180; clat++)
936 if (allCells[clon][clat]) allCells[clon][clat]->ClearPolyV();
937 }
938#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
939 glEnableClientState(GL_VERTEX_ARRAY);
940
941 // use a viewport that allows the vertexes to be reused over many frames
942 // TODO fix for multicanvas
943 if (glChartCanvas::HasNormalizedViewPort(vp)) {
944 glPushMatrix();
945 glChartCanvas::MultMatrixViewPort(vp);
946 nvp = glChartCanvas::NormalizedViewPort(vp);
947 }
948#endif
949 }
950#endif
951 for (clon = clonmin; clon < clonmax; clon++) {
952 clonx = clon;
953 while (clonx < 0) clonx += 360;
954 while (clonx >= 360) clonx -= 360;
955
956 for (clat = clatmin; clat < clatmax; clat++) {
957 if (clonx >= 0 && clonx <= 359 && clat >= -90 && clat <= 89) {
958 if (allCells[clonx][clat + 90] == NULL) {
959 cel = new GshhsPolyCell(fpoly, clonx, clat, &polyHeader);
960 wxASSERT(cel);
961 allCells[clonx][clat + 90] = cel;
962 } else {
963 cel = allCells[clonx][clat + 90];
964 }
965 bool idl = false;
966
967 // only mercator needs the special idl fixes
968 if (vp.m_projection_type != PROJECTION_MERCATOR &&
969 vp.m_projection_type != PROJECTION_EQUIRECTANGULAR)
970 dx = 0;
971 else if (pnt.GetDC()) // dc
972 dx = clon - clonx;
973 else { // opengl
974 int clonn = clonx;
975 if (clonn >= 180) {
976 clonn -= 360;
977 idl = true;
978 }
979 if (vp.clon - clonn > 180)
980 dx = 1;
981 else if (vp.clon - clonn < -180)
982 dx = -1;
983 else
984 dx = 0;
985 }
986
987 cel->drawMapPlain(pnt, dx, nvp, seaColor, landColor, idl);
988 }
989 }
990 }
991
992#ifdef ocpnUSE_GL
993#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
994 if (!pnt.GetDC()) { // opengl
995 if (glChartCanvas::HasNormalizedViewPort(vp)) glPopMatrix();
996 glDisableClientState(GL_VERTEX_ARRAY);
997 }
998#endif
999#endif
1000}
1001
1002//-------------------------------------------------------------------------
1003void GshhsPolyReader::drawGshhsPolyMapSeaBorders(ocpnDC &pnt, ViewPort &vp) {
1004 if (!fpoly) return;
1005 int clonmin, clonmax, clatmax, clatmin; // cellules visibles
1006 LLBBox bbox = vp.GetBBox();
1007 clonmin = bbox.GetMinLon(), clonmax = bbox.GetMaxLon(),
1008 clatmin = bbox.GetMinLat(), clatmax = bbox.GetMaxLat();
1009
1010 int dx, clon, clonx, clat;
1011 GshhsPolyCell *cel;
1012
1013 for (clon = clonmin; clon < clonmax; clon++) {
1014 clonx = clon;
1015 while (clonx < 0) clonx += 360;
1016 while (clonx >= 360) clonx -= 360;
1017
1018 for (clat = clatmin; clat < clatmax; clat++) {
1019 if (clonx >= 0 && clonx <= 359 && clat >= -90 && clat <= 89) {
1020 if (allCells[clonx][clat + 90] == NULL) {
1021 cel = new GshhsPolyCell(fpoly, clonx, clat, &polyHeader);
1022 wxASSERT(cel);
1023 allCells[clonx][clat + 90] = cel;
1024 } else {
1025 cel = allCells[clonx][clat + 90];
1026 }
1027 dx = clon - clonx;
1028 cel->drawSeaBorderLines(pnt, dx, vp);
1029 }
1030 }
1031 }
1032}
1033
1034int GshhsPolygon::readInt4() {
1035 union {
1036 unsigned int n;
1037 unsigned char buf[4];
1038 } res;
1039
1040 unsigned char in[4];
1041
1042 int nb = 0;
1043 nb += fread(&in, 1, 4, file);
1044 res.buf[3] = in[0];
1045 res.buf[2] = in[1];
1046 res.buf[1] = in[2];
1047 res.buf[0] = in[3];
1048
1049 if (nb != 4) {
1050 ok = false;
1051 res.n = 0;
1052 }
1053
1054 return res.n;
1055}
1056
1057int GshhsPolygon::readInt2() {
1058 union {
1059 unsigned int n;
1060 unsigned char buf[4];
1061 } v;
1062
1063 int nb = 0;
1064 nb += fread(&v.buf[0], 2, 1, file);
1065 if (nb != 2) {
1066 ok = false;
1067 v.n = 0;
1068 }
1069 return v.buf[1] << 8 | v.buf[0];
1070}
1071
1072GshhsPolygon::GshhsPolygon(FILE *file_) {
1073 file = file_;
1074 ok = true;
1075 id = readInt4();
1076 n = readInt4();
1077 flag = readInt4();
1078 west = readInt4() * 1e-6;
1079 east = readInt4() * 1e-6;
1080 south = readInt4() * 1e-6;
1081 north = readInt4() * 1e-6;
1082 area = readInt4();
1083
1084 if (((flag >> 8) & 255) >= 7) { // GSHHS Release 2.0
1085 areaFull = readInt4();
1086 container = readInt4();
1087 ancestor = readInt4();
1088
1089 greenwich = (flag >> 16) & 1;
1090 antarctic = (west == 0 && east == 360);
1091 if (ok) {
1092 double x = 0, y = 0;
1093 for (int i = 0; i < n; i++) {
1094 x = readInt4() * 1e-6;
1095 if (greenwich && x > 270) x -= 360;
1096 y = readInt4() * 1e-6;
1097 lsPoints.push_back(new GshhsPoint(x, y));
1098 }
1099 if (antarctic) {
1100 lsPoints.insert(lsPoints.begin(), new GshhsPoint(360, y));
1101 lsPoints.insert(lsPoints.begin(), new GshhsPoint(360, -90));
1102 lsPoints.push_back(new GshhsPoint(0, -90));
1103 }
1104 }
1105 } else {
1106 greenwich = (flag >> 16) & 1;
1107 antarctic = (west == 0 && east == 360);
1108 if (ok) {
1109 for (int i = 0; i < n; i++) {
1110 double x = 0, y = 0;
1111 x = readInt4() * 1e-6;
1112 if (greenwich && x > 270) x -= 360;
1113 y = readInt4() * 1e-6;
1114 lsPoints.push_back(new GshhsPoint(x, y));
1115 }
1116 }
1117 }
1118}
1119
1120//--------------------------------------------------------
1121
1122GshhsPolygon::~GshhsPolygon() {
1123 std::vector<GshhsPoint *>::iterator itp;
1124 for (itp = lsPoints.begin(); itp != lsPoints.end(); itp++) {
1125 delete *itp;
1126 *itp = NULL;
1127 }
1128 lsPoints.clear();
1129}
1130
1131//==========================================================
1132
1133GshhsReader::GshhsReader() {
1134 maxQualityAvailable = -1;
1135 minQualityAvailable = -1;
1136
1137 for (int i = 0; i < 5; i++) {
1138 qualityAvailable[i] = false;
1139 if (GshhsReader::gshhsFilesExists(i)) {
1140 qualityAvailable[i] = true;
1141 if (minQualityAvailable < 0) minQualityAvailable = i;
1142 maxQualityAvailable = i;
1143 }
1144 }
1145
1146 if (maxQualityAvailable < 0) {
1147 wxString msg(
1148 _T("Unable to initialize background world map. No GSHHS datafiles ")
1149 _T("found in "));
1150 msg += gWorldMapLocation;
1151 wxLogMessage(msg);
1152 }
1153
1154 // int q = selectBestQuality( vp );
1155 // if( ! qualityAvailable[q] ) {
1156 // int q = maxQualityAvailable;
1157 // }
1158
1159 int q = 0;
1160
1161 gshhsPoly_reader = new GshhsPolyReader(q);
1162
1163 for (int qual = 0; qual < 5; qual++) {
1164 lsPoly_boundaries[qual] = new std::vector<GshhsPolygon *>;
1165 lsPoly_rivers[qual] = new std::vector<GshhsPolygon *>;
1166 }
1167
1168 quality = -1;
1169 LoadQuality(q);
1170}
1171
1172int GshhsReader::ReadPolyVersion() {
1173 return gshhsPoly_reader->ReadPolyVersion();
1174}
1175
1176//-------------------------------------------------------
1177
1178GshhsReader::~GshhsReader() {
1179 clearLists();
1180 delete gshhsPoly_reader;
1181}
1182
1183//-----------------------------------------------------------------------
1184void GshhsReader::clearLists() {
1185 std::vector<GshhsPolygon *>::iterator itp;
1186 for (int qual = 0; qual < 5; qual++) {
1187 for (itp = lsPoly_boundaries[qual]->begin();
1188 itp != lsPoly_boundaries[qual]->end(); itp++) {
1189 delete *itp;
1190 *itp = NULL;
1191 }
1192 for (itp = lsPoly_rivers[qual]->begin(); itp != lsPoly_rivers[qual]->end();
1193 itp++) {
1194 delete *itp;
1195 *itp = NULL;
1196 }
1197
1198 lsPoly_boundaries[qual]->clear();
1199 lsPoly_rivers[qual]->clear();
1200 delete lsPoly_boundaries[qual];
1201 delete lsPoly_rivers[qual];
1202 }
1203}
1204//-----------------------------------------------------------------------
1205
1206wxString GshhsReader::getNameExtension(int quality) {
1207 wxString ext;
1208 switch (quality) {
1209 case 0:
1210 ext = _T("c");
1211 break;
1212 case 1:
1213 ext = _T("l");
1214 break;
1215 case 2:
1216 ext = _T("i");
1217 break;
1218 case 3:
1219 ext = _T("h");
1220 break;
1221 case 4:
1222 ext = _T("f");
1223 break;
1224 default:
1225 ext = _T("l");
1226 break;
1227 }
1228 return ext;
1229}
1230
1231wxString GshhsReader::getFileName_Land(int quality) {
1232 wxString ext = GshhsReader::getNameExtension(quality);
1233 wxString fname =
1234 gWorldMapLocation + wxString::Format(_T("poly-%c-1.dat"), ext.GetChar(0));
1235 return fname;
1236}
1237
1238wxString GshhsReader::getFileName_boundaries(int quality) {
1239 wxString ext = GshhsReader::getNameExtension(quality);
1240 wxString fname = gWorldMapLocation +
1241 wxString::Format(_T("wdb_borders_%c.b"), ext.GetChar(0));
1242 return fname;
1243}
1244
1245wxString GshhsReader::getFileName_rivers(int quality) {
1246 wxString ext = GshhsReader::getNameExtension(quality);
1247 wxString fname = gWorldMapLocation +
1248 wxString::Format(_T("wdb_rivers_%c.b"), ext.GetChar(0));
1249 return fname;
1250}
1251
1252//-----------------------------------------------------------------------
1253bool GshhsReader::gshhsFilesExists(int quality) {
1254 if (!wxFile::Access(GshhsReader::getFileName_Land(quality), wxFile::read))
1255 return false;
1256 // Borders disabled anyway since the perf optimizations if( ! wxFile::Access(
1257 // GshhsReader::getFileName_boundaries( quality ), wxFile::read ) ) return
1258 // false; Rivers disabled anyway since the perf optimizations if( !
1259 // wxFile::Access( GshhsReader::getFileName_rivers( quality ), wxFile::read )
1260 // ) return false;
1261
1262 return true;
1263}
1264
1265//-----------------------------------------------------------------------
1266void GshhsReader::LoadQuality(int newQuality) // 5 levels: 0=low ... 4=full
1267{
1268 if (quality == newQuality) return;
1269
1270 wxStopWatch perftimer;
1271
1272 wxString fname;
1273
1274 quality = newQuality;
1275 if (quality < 0)
1276 quality = 0;
1277 else if (quality > 4)
1278 quality = 4;
1279
1280 gshhsPoly_reader->InitializeLoadQuality(quality);
1281#if 0 /* too slow to load the whole world at once */
1282 if( lsPoly_boundaries[quality]->size() == 0 ) {
1283 fname = getFileName_boundaries( quality );
1284 file = fopen( fname.mb_str(), "rb" );
1285
1286 if( file != NULL ) {
1287 ok = true;
1288 while( ok ) {
1289 GshhsPolygon *poly = new GshhsPolygon( file );
1290
1291 ok = poly->isOk();
1292 if( ok )
1293 if( poly->getLevel() < 2 )
1294 lsPoly_boundaries[quality]->push_back( poly );
1295 else delete poly;
1296 else delete poly;
1297 }
1298 fclose( file );
1299 }
1300 }
1301
1302 if( lsPoly_rivers[quality]->size() == 0 ) {
1303 fname = getFileName_rivers( quality );
1304 file = fopen( fname.mb_str(), "rb" );
1305 if( file != NULL ) {
1306 ok = true;
1307 while( ok ) {
1308 GshhsPolygon *poly = new GshhsPolygon( file );
1309 ok = poly->isOk();
1310 if( ok ) {
1311 lsPoly_rivers[quality]->push_back( poly );
1312 }
1313 else delete poly;
1314 }
1315 fclose( file );
1316 }
1317 }
1318#endif
1319 wxLogMessage(_T("Loading World Chart Q=%d in %ld ms."), quality,
1320 perftimer.Time());
1321}
1322
1323//-----------------------------------------------------------------------
1324std::vector<GshhsPolygon *> &GshhsReader::getList_boundaries() {
1325 return *lsPoly_boundaries[quality];
1326}
1327//-----------------------------------------------------------------------
1328std::vector<GshhsPolygon *> &GshhsReader::getList_rivers() {
1329 return *lsPoly_rivers[quality];
1330}
1331
1332//=====================================================================
1333
1334int GshhsReader::GSHHS_scaledPoints(GshhsPolygon *pol, wxPoint *pts,
1335 double declon, ViewPort &vp) {
1336 LLBBox box;
1337 box.Set(pol->south, pol->west + declon, pol->north, pol->east + declon);
1338 if (vp.GetBBox().IntersectOut(box)) return 0;
1339
1340 // Remove small polygons.
1341
1342 wxPoint2DDouble p1 = GetDoublePixFromLL(vp, pol->west + declon, pol->north);
1343 wxPoint2DDouble p2 = GetDoublePixFromLL(vp, pol->east + declon, pol->south);
1344
1345 if (p1.m_x == p2.m_x && p1.m_y == p2.m_y) return 0;
1346
1347 double x, y;
1348 std::vector<GshhsPoint *>::iterator itp;
1349 int xx, yy, oxx = 0, oyy = 0;
1350 int j = 0;
1351
1352 for (itp = (pol->lsPoints).begin(); itp != (pol->lsPoints).end(); itp++) {
1353 x = (*itp)->lon + declon;
1354 y = (*itp)->lat;
1355 wxPoint2DDouble p = GetDoublePixFromLL(vp, y, x);
1356 xx = p.m_x, yy = p.m_y;
1357 if (j == 0 || (oxx != xx || oyy != yy)) { // Remove close points
1358 oxx = xx;
1359 oyy = yy;
1360 pts[j].x = xx;
1361 pts[j].y = yy;
1362 j++;
1363 }
1364 }
1365
1366 return j;
1367}
1368
1369//-----------------------------------------------------------------------
1370void GshhsReader::GsshDrawLines(ocpnDC &pnt, std::vector<GshhsPolygon *> &lst,
1371 ViewPort &vp, bool isClosed) {
1372 std::vector<GshhsPolygon *>::iterator iter;
1373 GshhsPolygon *pol;
1374 wxPoint *pts = NULL;
1375 int i;
1376 int nbp;
1377
1378 int nbmax = 10000;
1379 pts = new wxPoint[nbmax];
1380 wxASSERT(pts);
1381
1382 for (i = 0, iter = lst.begin(); iter != lst.end(); iter++, i++) {
1383 pol = *iter;
1384
1385 if (nbmax < pol->n + 2) {
1386 nbmax = pol->n + 2;
1387 delete[] pts;
1388 pts = new wxPoint[nbmax];
1389 wxASSERT(pts);
1390 }
1391
1392 nbp = GSHHS_scaledPoints(pol, pts, 0, vp);
1393 if (nbp > 1) {
1394 if (pol->isAntarctic()) {
1395 pts++;
1396 nbp -= 2;
1397 pnt.DrawLines(nbp, pts);
1398 pts--;
1399 } else {
1400 pnt.DrawLines(nbp, pts);
1401 if (isClosed)
1402 pnt.DrawLine(pts[0].x, pts[0].y, pts[nbp - 1].x, pts[nbp - 1].y);
1403 }
1404 }
1405
1406 nbp = GSHHS_scaledPoints(pol, pts, -360, vp);
1407 if (nbp > 1) {
1408 if (pol->isAntarctic()) {
1409 pts++;
1410 nbp -= 2;
1411 pnt.DrawLines(nbp, pts);
1412 pts--;
1413 } else {
1414 pnt.DrawLines(nbp, pts);
1415 if (isClosed)
1416 pnt.DrawLine(pts[0].x, pts[0].y, pts[nbp - 1].x, pts[nbp - 1].y);
1417 }
1418 }
1419 }
1420 delete[] pts;
1421}
1422
1423//-----------------------------------------------------------------------
1424void GshhsReader::drawContinents(ocpnDC &pnt, ViewPort &vp,
1425 wxColor const &seaColor,
1426 wxColor const &landColor) {
1427 LoadQuality(selectBestQuality(vp));
1428 gshhsPoly_reader->drawGshhsPolyMapPlain(pnt, vp, seaColor, landColor);
1429}
1430
1431//-----------------------------------------------------------------------
1432void GshhsReader::drawSeaBorders(ocpnDC &pnt, ViewPort &vp) {
1433 pnt.SetBrush(*wxTRANSPARENT_BRUSH);
1434 gshhsPoly_reader->drawGshhsPolyMapSeaBorders(pnt, vp);
1435}
1436
1437//-----------------------------------------------------------------------
1438void GshhsReader::drawBoundaries(ocpnDC &pnt, ViewPort &vp) {
1439 pnt.SetBrush(*wxTRANSPARENT_BRUSH);
1440
1441 if (pnt.GetDC()) {
1442 wxPen *pen = wxThePenList->FindOrCreatePen(*wxBLACK, 1, wxPENSTYLE_DOT);
1443 pnt.SetPen(*pen);
1444 } else {
1445 wxPen *pen = wxThePenList->FindOrCreatePen(wxColor(0, 0, 0, 80), 2,
1446 wxPENSTYLE_LONG_DASH);
1447 pnt.SetPen(*pen);
1448 }
1449 GsshDrawLines(pnt, getList_boundaries(), vp, false);
1450}
1451
1452//-----------------------------------------------------------------------
1453void GshhsReader::drawRivers(ocpnDC &pnt, ViewPort &vp) {
1454 GsshDrawLines(pnt, getList_rivers(), vp, false);
1455}
1456
1457//-----------------------------------------------------------------------
1458int GshhsReader::selectBestQuality(ViewPort &vp) {
1459 int bestQuality = 0;
1460
1461 if (vp.chart_scale < 500000 && qualityAvailable[4])
1462 bestQuality = 4;
1463 else if (vp.chart_scale < 2000000 && qualityAvailable[3])
1464 bestQuality = 3;
1465 else if (vp.chart_scale < 8000000 && qualityAvailable[2])
1466 bestQuality = 2;
1467 else if (vp.chart_scale < 20000000 && qualityAvailable[1])
1468 bestQuality = 1;
1469 else if (qualityAvailable[0])
1470 bestQuality = 0;
1471 else
1472 while (!qualityAvailable[bestQuality] &&
1473 bestQuality <=
1474 4) // Find the worst quality actually available and use that
1475 // (normally we would use crude, but it is missing)
1476 bestQuality++;
1477
1478 while (!qualityAvailable[bestQuality]) {
1479 bestQuality--;
1480 if (bestQuality < 0) break;
1481 }
1482
1483 // if( bestQuality < 0 )
1484 // for( int i=0; i<5; i++ )
1485 // if( qualityAvailable[i] ) bestQuality = i;
1486
1487 return bestQuality;
1488}
1489
1490// A GshhsReader instance which is used to detect if a trajectory
1491// crosses land.
1492static GshhsReader *gshhs_reader = NULL;
1493
1494/* so plugins can determine if a line segment crosses land, must call from main
1495 thread once at startup to initialize array */
1496void gshhsCrossesLandInit() {
1497 wxLogMessage("GSHHSChart::gshhsCrossesLandInit()");
1498 if (!gshhs_reader) {
1499 gshhs_reader = new GshhsReader();
1500 }
1501 /* load best possible quality for crossing tests */
1502 int bestQuality = 4;
1503 while (!gshhs_reader->qualityAvailable[bestQuality] && bestQuality > 0)
1504 bestQuality--;
1505 gshhs_reader->LoadQuality(bestQuality);
1506 wxLogMessage("GSHHG: Loaded quality %d for land crossing detection.",
1507 bestQuality);
1508}
1509
1510void gshhsCrossesLandReset() {
1511 wxLogMessage("GSHHSChart::gshhsCrossesLandReset()");
1512 if (gshhs_reader) delete gshhs_reader;
1513 gshhs_reader = NULL;
1514 gshhsCrossesLandInit();
1515}
1516
1517bool gshhsCrossesLand(double lat1, double lon1, double lat2, double lon2) {
1518 if (!gshhs_reader) {
1519 gshhsCrossesLandInit();
1520 }
1521 if (lon1 < 0) lon1 += 360;
1522 if (lon2 < 0) lon2 += 360;
1523
1524 wxLineF trajectWorld(lon1, lat1, lon2, lat2);
1525 return gshhs_reader->crossing1(trajectWorld);
1526}
Wrapper class for OpenGL shader programs.
Definition shaders.h:57
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 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
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
void DrawLine(wxCoord x1, wxCoord y1, wxCoord x2, wxCoord y2, bool b_hiqual=true)
Draw a line between two points using either wxDC or OpenGL.
Definition ocpndc.cpp:476