56#include "glChartCanvas.h"
66#define __CALL_CONVENTION
68#define __CALL_CONVENTION
73extern wxString gWorldMapLocation;
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"
83 " fragColor = color;\n"
84 " gl_Position = MVMatrix * TransformMatrix * vec4(position, 0.0, 1.0);\n"
87static const GLchar *fragment_shader_source =
88 "precision lowp float;\n"
89 "varying vec4 fragColor;\n"
91 " gl_FragColor = fragColor;\n"
94static const GLfloat vertices2[] = {
95 0.0f, 0.5f, 0.5f, -0.5f, -0.5f, -0.5f,
98static const GLfloat vertices3[] = {
99 0.0f, 0.5f, 0.0f, 0.5f, -0.5f, 0.0f, -0.5f, -0.5f, 0.0f,
102enum Consts { INFOLOG_LEN = 512 };
103GLchar infoLog[INFOLOG_LEN];
104GLint fragment_shader;
109extern GLint color_tri_shader_program;
114GSHHSChart::GSHHSChart() {
116 land = wxColor(250, 250, 250);
117 water = wxColor(0, 0, 0);
120GSHHSChart::~GSHHSChart() {
121 if (reader)
delete reader;
124void GSHHSChart::SetColorScheme(ColorScheme scheme) {
125 land = wxColor(170, 175, 80);
126 water = wxColor(170, 195, 240);
131 case GLOBAL_COLOR_SCHEME_DUSK:
134 case GLOBAL_COLOR_SCHEME_NIGHT:
141 land.Set(land.Red() * dim, land.Green() * dim, land.Blue() * dim);
142 water.Set(water.Red() * dim, water.Green() * dim, water.Blue() * dim);
145void GSHHSChart::SetColorsDirect(wxColour newLand, wxColour newWater) {
150void GSHHSChart::Reset() {
151 if (reader)
delete reader;
155int GSHHSChart::GetMinAvailableQuality() {
157 return reader->GetMinAvailableQuality();
160int GSHHSChart::GetMaxAvailableQuality() {
162 return reader->GetMaxAvailableQuality();
168 if (reader->GetPolyVersion() < 210 || reader->GetPolyVersion() > 240) {
170 _T(
"GSHHS World chart files have wrong version. Found %d, expected ")
172 reader->GetPolyVersion());
175 _T(
"Background world map loaded from GSHHS datafiles found in: ") +
180 reader->drawContinents(dc, vp, water, land);
186GshhsPolyCell::GshhsPolyCell(FILE *fpoly_,
int x0_,
int y0_,
193 for (
int i = 0; i < 6; i++) polyv[i] = NULL;
197 for (
int i = 0; i < GSSH_SUBM * GSSH_SUBM; i++) high_res_map[i] = NULL;
200GshhsPolyCell::~GshhsPolyCell() {
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];
207void GshhsPolyCell::ClearPolyV() {
208 for (
int i = 0; i < 6; i++) {
214void GshhsPolyCell::ReadPoly(contour_list &poly) {
217 int32_t num_vertices, num_contours;
219 if (fread(&num_contours,
sizeof num_contours, 1, fpoly) != 1)
goto fail;
221 for (
int c = 0; c < num_contours; c++) {
223 if (fread(&value,
sizeof value, 1, fpoly) !=
225 fread(&value,
sizeof value, 1, fpoly) != 1)
228 num_vertices = value;
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)
236 tmp_contour.push_back(wxRealPoint(X * GSHHS_SCL, Y * GSHHS_SCL));
238 poly.push_back(tmp_contour);
243 wxLogMessage(_T(
"gshhs ReadPoly failed"));
246void GshhsPolyCell::ReadPolygonFile() {
252 tab_data = (x0cell / header->pasx) * (180 / header->pasy) +
253 (y0cell + 90) / header->pasy;
255 if (fread(&pos_data,
sizeof(
int), 1, fpoly) != 1)
goto fail;
257 fseek(fpoly, pos_data, SEEK_SET);
267 wxLogMessage(_T(
"gshhs ReadPolygon failed"));
270wxPoint2DDouble GetDoublePixFromLL(
ViewPort &vp,
double lat,
double lon) {
272 p.m_x -= vp.rv_rect.x, p.m_y -= vp.rv_rect.y;
276void GshhsPolyCell::DrawPolygonFilled(
ocpnDC &pnt, contour_list *p,
double dx,
277 ViewPort &vp, wxColor
const &color) {
291 for (c = 0; c < p->size(); c++) {
292 if (!p->at(c).size())
continue;
294 wxPoint *poly_pt =
new wxPoint[p->at(c).size()];
296 contour &cp = p->at(c);
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)) {
307 x = q.m_x, y = q.m_y;
309 if (v == 0 || x != x_old || y != y_old) {
310 poly_pt[pointCount].x = x;
311 poly_pt[pointCount].y = y;
318 if (pointCount > 1) pnt.DrawPolygonTessellated(pointCount, poly_pt, 0, 0);
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;
345void __CALL_CONVENTION gshhscombineCallback(GLdouble coords[3],
346 GLdouble *vertex_data[4],
348 GLdouble **dataOut) {
351 vertex =
new GLvertex();
352 g_vertexes.push_back(vertex);
354 vertex->info.x = coords[0];
355 vertex->info.y = coords[1];
357 *dataOut = vertex->data;
360void __CALL_CONVENTION gshhsvertexCallback(GLvoid *arg) {
362 vertex = (GLvertex *)arg;
364 p.y = vertex->info.x;
365 p.x = vertex->info.y;
368 if (g_type != GL_TRIANGLES) {
370 g_pv.push_back(g_p1);
371 g_pv.push_back(g_p2);
374 if (g_type == GL_TRIANGLE_STRIP)
385void __CALL_CONVENTION gshhserrorCallback(GLenum errorCode) {
386 const GLubyte *estring;
387 estring = gluErrorString(errorCode);
391void __CALL_CONVENTION gshhsbeginCallback(GLenum type) {
394 case GL_TRIANGLE_STRIP:
395 case GL_TRIANGLE_FAN:
399 printf(
"tess unhandled begin type: %d\n", type);
405void __CALL_CONVENTION gshhsendCallback() {}
407void GshhsPolyCell::DrawPolygonFilledGL(
ocpnDC &pnt, contour_list *p,
408 float_2Dpt **pv,
int *pvc,
ViewPort &vp,
409 wxColor
const &color,
bool idl) {
416 for (
unsigned int c = 0; c < p->size(); c++) {
417 if (!p->at(c).size())
continue;
419 contour &cp = p->at(c);
421 GLUtesselator *tobj = gluNewTess();
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);
430 gluTessNormal(tobj, 0, 0, 1);
431 gluTessProperty(tobj, GLU_TESS_WINDING_RULE, GLU_TESS_WINDING_NONZERO);
433 gluTessBeginPolygon(tobj, NULL);
434 gluTessBeginContour(tobj);
436 for (
unsigned int v = 0; v < p->at(c).size(); v++) {
437 wxRealPoint &ccp = cp.at(v);
439 if (v == 0 || ccp != cp.at(v - 1)) {
440 GLvertex *vertex =
new GLvertex();
441 g_vertexes.push_back(vertex);
444 if (glChartCanvas::HasNormalizedViewPort(vp))
445 q = GetDoublePixFromLL(vp, ccp.y, ccp.x);
447 q.m_x = ccp.y, q.m_y = ccp.x;
449 if (vp.m_projection_type != PROJECTION_POLAR) {
453 if (idl && ccp.x == 180) {
454 if (vp.m_projection_type != PROJECTION_MERCATOR &&
455 vp.m_projection_type != PROJECTION_EQUIRECTANGULAR)
460 vertex->info.x = q.m_x;
461 vertex->info.y = q.m_y;
463 gluTessVertex(tobj, (GLdouble *)vertex, (GLdouble *)vertex);
467 gluTessEndContour(tobj);
468 gluTessEndPolygon(tobj);
471 for (std::list<GLvertex *>::iterator it = g_vertexes.begin();
472 it != g_vertexes.end(); it++)
477 *pv =
new float_2Dpt[g_pv.size()];
479 for (std::list<float_2Dpt>::iterator it = g_pv.begin(); it != g_pv.end();
487#if defined(USE_ANDROID_GLES2) || defined(ocpnUSE_GLSL)
491 if (!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);
498 glGetShaderInfoLog(vertex_shader, INFOLOG_LEN, NULL, infoLog);
499 printf(
"ERROR::SHADER::VERTEX::COMPILATION_FAILED\n%s\n", infoLog);
503 if (!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);
510 glGetShaderInfoLog(fragment_shader, INFOLOG_LEN, NULL, infoLog);
511 printf(
"ERROR::SHADER::FRAGMENT::COMPILATION_FAILED\n%s\n", infoLog);
515 if (!shader_program) {
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);
523 glGetProgramInfoLog(shader_program, INFOLOG_LEN, NULL, infoLog);
524 printf(
"ERROR::SHADER::PROGRAM::LINKING_FAILED\n%s\n", infoLog);
533 mat4x4_scale_aniso(mvp, m, 2.0 / (
float)vp.
pix_width,
537 if (glChartCanvas::HasNormalizedViewPort(vp)) {
539 GLint pos = glGetAttribLocation(color_tri_shader_program,
"position");
540 glVertexAttribPointer(pos, 2, GL_FLOAT, GL_FALSE, 2 *
sizeof(
float), *pv);
541 glEnableVertexAttribArray(pos);
547 GLint tmatloc = glGetUniformLocation(color_tri_shader_program,
"TransformMatrix");
548 glUniformMatrix4fv(tmatloc, 1, GL_FALSE, (
const GLfloat*)m);
550 glUseProgram(color_tri_shader_program);
551 glDrawArrays(GL_TRIANGLES, 0, *pvc);
554 float *pvt =
new float[2 * (*pvc)];
555 for (
int i = 0; i < *pvc; i++) {
556 float_2Dpt *pc = *pv + i;
559 pvt[(i * 2) + 1] = q.m_y;
562 GLShaderProgram *shader = pcolor_tri_shader_program[pnt.m_canvasIndex];
566 colorv[0] = color.Red() / float(256);
567 colorv[1] = color.Green() / float(256);
568 colorv[2] = color.Blue() / float(256);
570 shader->SetUniform4fv(
"color", colorv);
572 shader->SetAttributePointerf(
"position", pvt);
574 glDrawArrays(GL_TRIANGLES, 0, *pvc);
577 glDeleteBuffers(1, &vbo);
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);
591void GshhsPolyCell::drawMapPlain(
ocpnDC &pnt,
double dx,
ViewPort &vp,
592 wxColor seaColor, wxColor landColor,
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)) {
601 40058986 * NORM_FACTOR;
603 glTranslated(dx > 0 ? ts : -ts, 0, 0);
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);
612#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
613 if (dx) glPopMatrix();
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);
627void GshhsPolyCell::DrawPolygonContour(
ocpnDC &pnt, contour_list *p,
double dx,
629 double x1, y1, x2, y2;
630 double long_max, lat_max, long_min, lat_min;
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);
639 for (
unsigned int i = 0; i < p->size(); i++) {
640 if (!p->at(i).size())
continue;
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;
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);
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;
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);
672#define DRAW_POLY_CONTOUR(POLY) \
673 if (POLY) DrawPolygonContour(pnt, POLY, dx, vp);
675void GshhsPolyCell::drawSeaBorderLines(
ocpnDC &pnt,
double dx,
ViewPort &vp) {
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)
686GshhsPolyReader::GshhsPolyReader(
int quality) {
689 for (
int i = 0; i < 360; i++) {
690 for (
int j = 0; j < 180; j++) {
691 allCells[i][j] = NULL;
695 polyHeader.version = -1;
696 InitializeLoadQuality(quality);
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;
716int GshhsPolyReader::ReadPolyVersion() {
717 wxString fname = GshhsReader::getFileName_Land(0);
718 if (fpoly) fclose(fpoly);
719 fpoly = fopen(fname.mb_str(),
"rb");
722 if (!fpoly)
return 0;
724 readPolygonFileHeader(fpoly, &polyHeader);
726 return polyHeader.version;
729void GshhsPolyReader::InitializeLoadQuality(
732 if (currentQuality != quality) {
733 currentQuality = quality;
735 wxString fname = GshhsReader::getFileName_Land(quality);
737 if (fpoly) fclose(fpoly);
739 fpoly = fopen(fname.mb_str(),
"rb");
740 if (fpoly) readPolygonFileHeader(fpoly, &polyHeader);
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;
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,
756 double x3 = line2.m_p1.x, y3 = line2.m_p1.y, x4 = line2.m_p2.x,
761 double ax = x2 - x1, ay = y2 - y1;
762 double bx = x3 - x4, by = y3 - y4;
763 double cx = x1 - x3, cy = y1 - y3;
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)
770 if (fabs((x1 * ay - ax * y1) * by - (x3 * by - bx * y3) * ay) > INTER_LIMIT)
776 const double reciprocal = 1 / denominator;
777 const double na = (by * cx - bx * cy) * reciprocal;
779 if (na < -INTER_LIMIT || na > 1 + INTER_LIMIT)
return false;
781 const double nb = (ax * cy - ay * cx) * reciprocal;
782 if (nb < -INTER_LIMIT || nb > 1 + INTER_LIMIT)
return false;
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;
791 int clonmin, clonmax, clatmax, clatmin;
792 clonmin = (int)floor(GSSH_SUBM * wxMin(x1, x2));
793 clonmax = (int)ceil(GSSH_SUBM * wxMax(x1, x2));
796 clonmin += GSSH_SUBM * 360;
797 clonmax += GSSH_SUBM * 360;
800 if (clonmax - clonmin > GSSH_SUBM * 180) {
801 clonmin = (int)floor(GSSH_SUBM * wxMax(x1, x2)) - GSSH_SUBM * 360;
802 clonmax = (int)ceil(GSSH_SUBM * wxMin(x1, x2));
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);
813 int clon, clonx, clat;
814 for (clon = clonmin; clon < clonmax; clon++) {
816 while (clonx < 0) clonx += GSSH_SUBM * 360;
817 while (clonx >= GSSH_SUBM * 360) clonx -= GSSH_SUBM * 360;
819 wxASSERT(clonx >= 0 && clonx < GSSH_SUBM * 360);
821 if (clonx < GSSH_SUBM * 180) {
822 if (x1 > 180) x1 -= 360;
823 if (x2 > 180) x2 -= 360;
825 if (x1 < 180) x1 += 360;
826 if (x2 < 180) x2 += 360;
829 wxLineF rtrajectWorld(x1, y1, x2, y2);
831 for (clat = clatmin; clat < clatmax; clat++) {
832 int cloni = clonx / GSSH_SUBM,
833 clati = (GSSH_SUBM * 90 + clat) / GSSH_SUBM;
839 cel =
new GshhsPolyCell(fpoly, cloni, clati - 90, &polyHeader);
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);
852 contour_list &poly1 = cel->getPoly1();
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;
865 int lstatex = lx < minlon ? -1 : lx > maxlon ? 1 : 0;
866 int lstatey = ly < minlat ? -1 : ly > maxlat ? 1 : 0;
868 for (
unsigned int pj = 0; pj < c.size(); pj++) {
869 double clon = c[pj].x, clat = c[pj].y;
874 if (lx == clon && ly == clat)
continue;
876 int statex = clon < minlon ? -1 : clon > maxlon ? 1 : 0;
877 int statey = clat < minlat ? -1 : clat > maxlat ? 1 : 0;
879 if ((!statex || lstatex != statex) &&
880 (!statey || lstatey != statey))
881 high_res_map->push_back(
wxLineF(lx, ly, clon, clat));
883 lx = clon, ly = clat;
884 lstatex = statex, lstatey = statey;
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;
900void GshhsPolyReader::readPolygonFileHeader(FILE *polyfile,
902 fseek(polyfile, 0, SEEK_SET);
904 wxLogMessage(_T(
"gshhs ReadPolygonFileHeader failed"));
908void GshhsPolyReader::drawGshhsPolyMapPlain(
ocpnDC &pnt,
ViewPort &vp,
909 wxColor
const &seaColor,
910 wxColor
const &landColor) {
913 pnt.SetPen(wxNullPen);
915 int clonmin, clonmax, clatmax, clatmin;
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;
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();
938#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
939 glEnableClientState(GL_VERTEX_ARRAY);
943 if (glChartCanvas::HasNormalizedViewPort(vp)) {
945 glChartCanvas::MultMatrixViewPort(vp);
946 nvp = glChartCanvas::NormalizedViewPort(vp);
951 for (clon = clonmin; clon < clonmax; clon++) {
953 while (clonx < 0) clonx += 360;
954 while (clonx >= 360) clonx -= 360;
956 for (clat = clatmin; clat < clatmax; clat++) {
957 if (clonx >= 0 && clonx <= 359 && clat >= -90 && clat <= 89) {
958 if (allCells[clonx][clat + 90] == NULL) {
961 allCells[clonx][clat + 90] = cel;
963 cel = allCells[clonx][clat + 90];
968 if (vp.m_projection_type != PROJECTION_MERCATOR &&
969 vp.m_projection_type != PROJECTION_EQUIRECTANGULAR)
971 else if (pnt.GetDC())
979 if (vp.
clon - clonn > 180)
981 else if (vp.
clon - clonn < -180)
987 cel->drawMapPlain(pnt, dx, nvp, seaColor, landColor, idl);
993#if !defined(USE_ANDROID_GLES2) && !defined(ocpnUSE_GLSL)
995 if (glChartCanvas::HasNormalizedViewPort(vp)) glPopMatrix();
996 glDisableClientState(GL_VERTEX_ARRAY);
1003void GshhsPolyReader::drawGshhsPolyMapSeaBorders(
ocpnDC &pnt,
ViewPort &vp) {
1005 int clonmin, clonmax, clatmax, clatmin;
1006 LLBBox bbox = vp.GetBBox();
1007 clonmin = bbox.GetMinLon(), clonmax = bbox.GetMaxLon(),
1008 clatmin = bbox.GetMinLat(), clatmax = bbox.GetMaxLat();
1010 int dx, clon, clonx, clat;
1013 for (clon = clonmin; clon < clonmax; clon++) {
1015 while (clonx < 0) clonx += 360;
1016 while (clonx >= 360) clonx -= 360;
1018 for (clat = clatmin; clat < clatmax; clat++) {
1019 if (clonx >= 0 && clonx <= 359 && clat >= -90 && clat <= 89) {
1020 if (allCells[clonx][clat + 90] == NULL) {
1023 allCells[clonx][clat + 90] = cel;
1025 cel = allCells[clonx][clat + 90];
1028 cel->drawSeaBorderLines(pnt, dx, vp);
1034int GshhsPolygon::readInt4() {
1037 unsigned char buf[4];
1040 unsigned char in[4];
1043 nb += fread(&in, 1, 4, file);
1057int GshhsPolygon::readInt2() {
1060 unsigned char buf[4];
1064 nb += fread(&v.buf[0], 2, 1, file);
1069 return v.buf[1] << 8 | v.buf[0];
1072GshhsPolygon::GshhsPolygon(FILE *file_) {
1078 west = readInt4() * 1e-6;
1079 east = readInt4() * 1e-6;
1080 south = readInt4() * 1e-6;
1081 north = readInt4() * 1e-6;
1084 if (((flag >> 8) & 255) >= 7) {
1085 areaFull = readInt4();
1086 container = readInt4();
1087 ancestor = readInt4();
1089 greenwich = (flag >> 16) & 1;
1090 antarctic = (west == 0 && east == 360);
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;
1100 lsPoints.insert(lsPoints.begin(),
new GshhsPoint(360, y));
1101 lsPoints.insert(lsPoints.begin(),
new GshhsPoint(360, -90));
1106 greenwich = (flag >> 16) & 1;
1107 antarctic = (west == 0 && east == 360);
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;
1122GshhsPolygon::~GshhsPolygon() {
1123 std::vector<GshhsPoint *>::iterator itp;
1124 for (itp = lsPoints.begin(); itp != lsPoints.end(); itp++) {
1133GshhsReader::GshhsReader() {
1134 maxQualityAvailable = -1;
1135 minQualityAvailable = -1;
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;
1146 if (maxQualityAvailable < 0) {
1148 _T(
"Unable to initialize background world map. No GSHHS datafiles ")
1150 msg += gWorldMapLocation;
1163 for (
int qual = 0; qual < 5; qual++) {
1164 lsPoly_boundaries[qual] =
new std::vector<GshhsPolygon *>;
1165 lsPoly_rivers[qual] =
new std::vector<GshhsPolygon *>;
1172int GshhsReader::ReadPolyVersion() {
1173 return gshhsPoly_reader->ReadPolyVersion();
1178GshhsReader::~GshhsReader() {
1180 delete gshhsPoly_reader;
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++) {
1192 for (itp = lsPoly_rivers[qual]->begin(); itp != lsPoly_rivers[qual]->end();
1198 lsPoly_boundaries[qual]->clear();
1199 lsPoly_rivers[qual]->clear();
1200 delete lsPoly_boundaries[qual];
1201 delete lsPoly_rivers[qual];
1206wxString GshhsReader::getNameExtension(
int quality) {
1231wxString GshhsReader::getFileName_Land(
int quality) {
1232 wxString ext = GshhsReader::getNameExtension(quality);
1234 gWorldMapLocation + wxString::Format(_T(
"poly-%c-1.dat"), ext.GetChar(0));
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));
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));
1253bool GshhsReader::gshhsFilesExists(
int quality) {
1254 if (!wxFile::Access(GshhsReader::getFileName_Land(quality), wxFile::read))
1266void GshhsReader::LoadQuality(
int newQuality)
1268 if (quality == newQuality)
return;
1270 wxStopWatch perftimer;
1274 quality = newQuality;
1277 else if (quality > 4)
1280 gshhsPoly_reader->InitializeLoadQuality(quality);
1282 if( lsPoly_boundaries[quality]->size() == 0 ) {
1283 fname = getFileName_boundaries( quality );
1284 file = fopen( fname.mb_str(),
"rb" );
1286 if( file != NULL ) {
1293 if( poly->getLevel() < 2 )
1294 lsPoly_boundaries[quality]->push_back( poly );
1302 if( lsPoly_rivers[quality]->size() == 0 ) {
1303 fname = getFileName_rivers( quality );
1304 file = fopen( fname.mb_str(),
"rb" );
1305 if( file != NULL ) {
1311 lsPoly_rivers[quality]->push_back( poly );
1319 wxLogMessage(_T(
"Loading World Chart Q=%d in %ld ms."), quality,
1324std::vector<GshhsPolygon *> &GshhsReader::getList_boundaries() {
1325 return *lsPoly_boundaries[quality];
1328std::vector<GshhsPolygon *> &GshhsReader::getList_rivers() {
1329 return *lsPoly_rivers[quality];
1334int GshhsReader::GSHHS_scaledPoints(
GshhsPolygon *pol, wxPoint *pts,
1337 box.Set(pol->south, pol->west + declon, pol->north, pol->east + declon);
1338 if (vp.GetBBox().IntersectOut(box))
return 0;
1342 wxPoint2DDouble p1 = GetDoublePixFromLL(vp, pol->west + declon, pol->north);
1343 wxPoint2DDouble p2 = GetDoublePixFromLL(vp, pol->east + declon, pol->south);
1345 if (p1.m_x == p2.m_x && p1.m_y == p2.m_y)
return 0;
1348 std::vector<GshhsPoint *>::iterator itp;
1349 int xx, yy, oxx = 0, oyy = 0;
1352 for (itp = (pol->lsPoints).begin(); itp != (pol->lsPoints).end(); itp++) {
1353 x = (*itp)->lon + declon;
1355 wxPoint2DDouble p = GetDoublePixFromLL(vp, y, x);
1356 xx = p.m_x, yy = p.m_y;
1357 if (j == 0 || (oxx != xx || oyy != yy)) {
1370void GshhsReader::GsshDrawLines(
ocpnDC &pnt, std::vector<GshhsPolygon *> &lst,
1372 std::vector<GshhsPolygon *>::iterator iter;
1374 wxPoint *pts = NULL;
1379 pts =
new wxPoint[nbmax];
1382 for (i = 0, iter = lst.begin(); iter != lst.end(); iter++, i++) {
1385 if (nbmax < pol->n + 2) {
1388 pts =
new wxPoint[nbmax];
1392 nbp = GSHHS_scaledPoints(pol, pts, 0, vp);
1394 if (pol->isAntarctic()) {
1397 pnt.DrawLines(nbp, pts);
1400 pnt.DrawLines(nbp, pts);
1402 pnt.
DrawLine(pts[0].x, pts[0].y, pts[nbp - 1].x, pts[nbp - 1].y);
1406 nbp = GSHHS_scaledPoints(pol, pts, -360, vp);
1408 if (pol->isAntarctic()) {
1411 pnt.DrawLines(nbp, pts);
1414 pnt.DrawLines(nbp, pts);
1416 pnt.
DrawLine(pts[0].x, pts[0].y, pts[nbp - 1].x, pts[nbp - 1].y);
1425 wxColor
const &seaColor,
1426 wxColor
const &landColor) {
1427 LoadQuality(selectBestQuality(vp));
1428 gshhsPoly_reader->drawGshhsPolyMapPlain(pnt, vp, seaColor, landColor);
1433 pnt.SetBrush(*wxTRANSPARENT_BRUSH);
1434 gshhsPoly_reader->drawGshhsPolyMapSeaBorders(pnt, vp);
1439 pnt.SetBrush(*wxTRANSPARENT_BRUSH);
1442 wxPen *pen = wxThePenList->FindOrCreatePen(*wxBLACK, 1, wxPENSTYLE_DOT);
1445 wxPen *pen = wxThePenList->FindOrCreatePen(wxColor(0, 0, 0, 80), 2,
1446 wxPENSTYLE_LONG_DASH);
1449 GsshDrawLines(pnt, getList_boundaries(), vp,
false);
1454 GsshDrawLines(pnt, getList_rivers(), vp,
false);
1458int GshhsReader::selectBestQuality(
ViewPort &vp) {
1459 int bestQuality = 0;
1461 if (vp.
chart_scale < 500000 && qualityAvailable[4])
1463 else if (vp.
chart_scale < 2000000 && qualityAvailable[3])
1465 else if (vp.
chart_scale < 8000000 && qualityAvailable[2])
1467 else if (vp.
chart_scale < 20000000 && qualityAvailable[1])
1469 else if (qualityAvailable[0])
1472 while (!qualityAvailable[bestQuality] &&
1478 while (!qualityAvailable[bestQuality]) {
1480 if (bestQuality < 0)
break;
1496void gshhsCrossesLandInit() {
1497 wxLogMessage(
"GSHHSChart::gshhsCrossesLandInit()");
1498 if (!gshhs_reader) {
1502 int bestQuality = 4;
1503 while (!gshhs_reader->qualityAvailable[bestQuality] && bestQuality > 0)
1505 gshhs_reader->LoadQuality(bestQuality);
1506 wxLogMessage(
"GSHHG: Loaded quality %d for land crossing detection.",
1510void gshhsCrossesLandReset() {
1511 wxLogMessage(
"GSHHSChart::gshhsCrossesLandReset()");
1512 if (gshhs_reader)
delete gshhs_reader;
1513 gshhs_reader = NULL;
1514 gshhsCrossesLandInit();
1517bool gshhsCrossesLand(
double lat1,
double lon1,
double lat2,
double lon2) {
1518 if (!gshhs_reader) {
1519 gshhsCrossesLandInit();
1521 if (lon1 < 0) lon1 += 360;
1522 if (lon2 < 0) lon2 += 360;
1524 wxLineF trajectWorld(lon1, lat1, lon2, lat2);
1525 return gshhs_reader->crossing1(trajectWorld);
Wrapper class for OpenGL shader programs.
ViewPort - Core geographic projection and coordinate transformation engine.
int pix_height
Height of the viewport in physical pixels.
int pix_width
Width of the viewport in physical pixels.
wxPoint2DDouble GetDoublePixFromLL(double lat, double lon)
Convert latitude and longitude on the ViewPort to physical pixel coordinates with double precision.
double clon
Center longitude of the viewport in degrees.
double clat
Center latitude of the viewport in degrees.
double chart_scale
Chart scale denominator (e.g., 50000 for a 1:50000 scale).
Device context class that can use either wxDC or OpenGL for drawing.
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.