OpenCPN Partial API docs
Loading...
Searching...
No Matches
chartdbs.cpp
1/**************************************************************************
2 *
3 * Project: ChartManager
4 * Purpose: Basic Chart Info Storage
5 * Author: David Register, Mark A Sikes
6 *
7 ***************************************************************************
8 * Copyright (C) 2010 by David S. Register *
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 * This program is distributed in the hope that it will be useful, *
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
18 * GNU General Public License for more details. *
19 * *
20 * You should have received a copy of the GNU General Public License *
21 * along with this program; if not, write to the *
22 * Free Software Foundation, Inc., *
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
24 **************************************************************************/
25
26#include <wx/wxprec.h>
27
28#ifndef WX_PRECOMP
29#include <wx/wx.h>
30#endif
31
32#include <wx/arrimpl.cpp>
33#include <wx/encconv.h>
34#include <wx/regex.h>
35#include <wx/progdlg.h>
36#include <wx/tokenzr.h>
37#include <wx/dir.h>
38
39#include "chartdbs.h"
40#include "chartbase.h"
41#include "pluginmanager.h"
42#include "mbtiles.h"
43#include "mygeom.h" // For DouglasPeucker();
44#include "FlexHash.h"
45#include "LOD_reduce.h"
46#include "shapefile_basemap.h"
47
48#ifndef UINT32
49#define UINT32 unsigned int
50#endif
51
52#ifdef __OCPN__ANDROID__
53#include "androidUTIL.h"
54#endif
55
56extern PlugInManager *g_pi_manager;
57extern wxString gWorldMapLocation;
58extern wxString gWorldShapefileLocation;
59extern ShapeBaseChartSet gShapeBasemap;
60
61static int s_dbVersion; // Database version currently in use at runtime
62 // Needed for ChartTableEntry::GetChartType() only
63 // TODO This can go away at opencpn Version 1.3.8 and
64 // above....
66
67static ChartFamilyEnum GetChartFamily(int charttype) {
68 ChartFamilyEnum cf;
69
70 switch (charttype) {
71 case CHART_TYPE_KAP:
72 cf = CHART_FAMILY_RASTER;
73 break;
74 case CHART_TYPE_GEO:
75 cf = CHART_FAMILY_RASTER;
76 break;
77 case CHART_TYPE_S57:
78 cf = CHART_FAMILY_VECTOR;
79 break;
80 case CHART_TYPE_CM93:
81 cf = CHART_FAMILY_VECTOR;
82 break;
83 case CHART_TYPE_CM93COMP:
84 cf = CHART_FAMILY_VECTOR;
85 break;
86 case CHART_TYPE_DUMMY:
87 cf = CHART_FAMILY_RASTER;
88 break;
89 case CHART_TYPE_UNKNOWN:
90 cf = CHART_FAMILY_UNKNOWN;
91 break;
92 default:
93 cf = CHART_FAMILY_UNKNOWN;
94 break;
95 }
96 return cf;
97}
98
100// ChartTableHeader
102
103void ChartTableHeader::Read(wxInputStream &is) {
104 is.Read(this, sizeof(ChartTableHeader));
105}
106
107void ChartTableHeader::Write(wxOutputStream &os) {
108 char vb[5];
109 sprintf(vb, "V%03d", DB_VERSION_CURRENT);
110
111 memcpy(dbVersion, vb, 4);
112 os.Write(this, sizeof(ChartTableHeader));
113}
114
115bool ChartTableHeader::CheckValid() {
116 char vb[5];
117 sprintf(vb, "V%03d", DB_VERSION_CURRENT);
118 if (strncmp(vb, dbVersion, sizeof(dbVersion))) {
119 wxString msg;
120 char vbo[5];
121 memcpy(vbo, dbVersion, 4);
122 vbo[4] = 0;
123 msg.Append(wxString(vbo, wxConvUTF8));
124 msg.Prepend(wxT(" Warning: found incorrect chart db version: "));
125 wxLogMessage(msg);
126
127 // return false; // no match....
128
129 // Try previous version....
130 sprintf(vb, "V%03d", DB_VERSION_PREVIOUS);
131 if (strncmp(vb, dbVersion, sizeof(dbVersion)))
132 return false;
133 else {
134 wxLogMessage(
135 _T(" Scheduling db upgrade to current db version on ")
136 _T("Options->Charts page visit..."));
137 return true;
138 }
139
140 } else {
141 wxString msg;
142 char vbo[5];
143 memcpy(vbo, dbVersion, 4);
144 vbo[4] = 0;
145 msg.Append(wxString(vbo, wxConvUTF8));
146 msg.Prepend(wxT("Loading chart db version: "));
147 wxLogMessage(msg);
148 }
149
150 return true;
151}
152
154// ChartTableEntry
156
157void ChartTableEntry::SetScale(int scale) {
158 Scale = scale;
159 rounding = 0;
160 // XXX find the right rounding
161 if (Scale >= 1000) rounding = 5 * pow(10, log10(Scale) - 2);
162}
163
164ChartTableEntry::ChartTableEntry(ChartBase &theChart, wxString &utf8Path) {
165 Clear();
166
167 char *pt = (char *)malloc(strlen(utf8Path.mb_str(wxConvUTF8)) + 1);
168 strcpy(pt, utf8Path.mb_str(wxConvUTF8));
169 pFullPath = pt;
170
171 SetScale(theChart.GetNativeScale());
172
173 ChartType = theChart.GetChartType();
174 ChartFamily = theChart.GetChartFamily();
175
176 Skew = theChart.GetChartSkew();
177 ProjectionType = theChart.GetChartProjectionType();
178
179 wxDateTime ed = theChart.GetEditionDate();
180 if (theChart.GetEditionDate().IsValid())
181 edition_date = theChart.GetEditionDate().GetTicks();
182
183 wxFileName fn(theChart.GetFullPath());
184 if (fn.GetModificationTime().IsValid())
185 file_date = fn.GetModificationTime().GetTicks();
186
187 m_pfilename = new wxString; // create and populate helper members
188 *m_pfilename = fn.GetFullName();
189 m_psFullPath = new wxString;
190 *m_psFullPath = utf8Path;
191 m_fullSystemPath = utf8Path;
192
193#ifdef __OCPN__ANDROID__
194 m_fullSystemPath = wxString(utf8Path.mb_str(wxConvUTF8));
195#endif
196
197 Extent ext;
198 theChart.GetChartExtent(&ext);
199 LatMax = ext.NLAT;
200 LatMin = ext.SLAT;
201 LonMin = ext.WLON;
202 LonMax = ext.ELON;
203
204 m_bbox.Set(LatMin, LonMin, LatMax, LonMax);
205
206 // Fill in the PLY information
207 // LOD calculation
208 int LOD_pixels = 1;
209 double scale_max_zoom = Scale / 4;
210
211 double display_ppm = 1 / .00025; // nominal for most LCD displays
212 double meters_per_pixel_max_scale = scale_max_zoom / display_ppm;
213 double LOD_meters = meters_per_pixel_max_scale * LOD_pixels;
214
215 // double LOD_meters = 5;
216
217 // If COVR table has only one entry, us it for the primary Ply Table
218 if (theChart.GetCOVREntries() == 1) {
219 nPlyEntries = theChart.GetCOVRTablePoints(0);
220
221 if (nPlyEntries > 5 && (LOD_meters > .01)) {
222 std::vector<int> index_keep{0, nPlyEntries - 1, 1, nPlyEntries - 2};
223
224 double *DPbuffer = (double *)malloc(2 * nPlyEntries * sizeof(double));
225
226 double *pfed = DPbuffer;
227 Plypoint *ppp = (Plypoint *)theChart.GetCOVRTableHead(0);
228
229 for (int i = 0; i < nPlyEntries; i++) {
230 *pfed++ = ppp->ltp;
231 *pfed++ = ppp->lnp;
232 ppp++;
233 }
234
235 DouglasPeucker(DPbuffer, 1, nPlyEntries - 2, LOD_meters / (1852 * 60),
236 &index_keep);
237 // printf("DB DP Reduction: %d/%d\n", index_keep.size(),
238 // nPlyEntries);
239
240 // Mark the keepers by adding a simple constant to ltp
241 for (unsigned int i = 0; i < index_keep.size(); i++) {
242 DPbuffer[2 * index_keep[i]] += 2000.;
243 }
244
245 float *pf = (float *)malloc(2 * index_keep.size() * sizeof(float));
246 float *pfe = pf;
247
248 for (int i = 0; i < nPlyEntries; i++) {
249 if (DPbuffer[2 * i] > 1000.) {
250 *pfe++ = DPbuffer[2 * i] - 2000.;
251 *pfe++ = DPbuffer[(2 * i) + 1];
252 }
253 }
254
255 pPlyTable = pf;
256 nPlyEntries = index_keep.size();
257 free(DPbuffer);
258 } else {
259 float *pf = (float *)malloc(2 * nPlyEntries * sizeof(float));
260 pPlyTable = pf;
261 float *pfe = pf;
262 Plypoint *ppp = (Plypoint *)theChart.GetCOVRTableHead(0);
263
264 for (int i = 0; i < nPlyEntries; i++) {
265 *pfe++ = ppp->ltp;
266 *pfe++ = ppp->lnp;
267 ppp++;
268 }
269 }
270 }
271 // Else create a rectangular primary Ply Table from the chart extents
272 // and create AuxPly table from the COVR tables
273 else {
274 // Create new artificial Ply table from chart extents
275 nPlyEntries = 4;
276 float *pf1 = (float *)malloc(2 * 4 * sizeof(float));
277 pPlyTable = pf1;
278 float *pfe = pf1;
279 Extent fext;
280 theChart.GetChartExtent(&fext);
281
282 *pfe++ = fext.NLAT; // LatMax;
283 *pfe++ = fext.WLON; // LonMin;
284
285 *pfe++ = fext.NLAT; // LatMax;
286 *pfe++ = fext.ELON; // LonMax;
287
288 *pfe++ = fext.SLAT; // LatMin;
289 *pfe++ = fext.ELON; // LonMax;
290
291 *pfe++ = fext.SLAT; // LatMin;
292 *pfe++ = fext.WLON; // LonMin;
293
294 // Fill in the structure for pAuxPlyTable
295
296 nAuxPlyEntries = theChart.GetCOVREntries();
297 wxASSERT(nAuxPlyEntries);
298 float **pfp = (float **)malloc(nAuxPlyEntries * sizeof(float *));
299 float **pft0 = pfp;
300 int *pip = (int *)malloc(nAuxPlyEntries * sizeof(int));
301
302 for (int j = 0; j < nAuxPlyEntries; j++) {
303 int nPE = theChart.GetCOVRTablePoints(j);
304
305 if (nPE > 5 && (LOD_meters > .01)) {
306 std::vector<int> index_keep{0, nPE - 1, 1, nPE - 2};
307
308 double *DPbuffer = (double *)malloc(2 * nPE * sizeof(double));
309
310 double *pfed = DPbuffer;
311 Plypoint *ppp = (Plypoint *)theChart.GetCOVRTableHead(j);
312
313 for (int i = 0; i < nPE; i++) {
314 *pfed++ = ppp->ltp;
315 *pfed++ = ppp->lnp;
316 ppp++;
317 }
318
319 DouglasPeucker(DPbuffer, 1, nPE - 2, LOD_meters / (1852 * 60),
320 &index_keep);
321 // printf("DBa DP Reduction: %d/%d\n",
322 // index_keep.size(), nPE);
323
324 // Mark the keepers by adding a simple constant to ltp
325 for (unsigned int i = 0; i < index_keep.size(); i++) {
326 DPbuffer[2 * index_keep[i]] += 2000.;
327 }
328
329 float *pf = (float *)malloc(2 * index_keep.size() * sizeof(float));
330 float *pfe = pf;
331
332 for (int i = 0; i < nPE; i++) {
333 if (DPbuffer[2 * i] > 1000.) {
334 *pfe++ = DPbuffer[2 * i] - 2000.;
335 *pfe++ = DPbuffer[(2 * i) + 1];
336 }
337 }
338
339 pft0[j] = pf;
340 pip[j] = index_keep.size();
341 free(DPbuffer);
342 } else {
343 float *pf_entry =
344 (float *)malloc(theChart.GetCOVRTablePoints(j) * 2 * sizeof(float));
345 memcpy(pf_entry, theChart.GetCOVRTableHead(j),
346 theChart.GetCOVRTablePoints(j) * 2 * sizeof(float));
347 pft0[j] = pf_entry;
348 pip[j] = theChart.GetCOVRTablePoints(j);
349 }
350 }
351
352 pAuxPlyTable = pfp;
353 pAuxCntTable = pip;
354 }
355
356 // Get and populate the NoCovr tables
357
358 nNoCovrPlyEntries = theChart.GetNoCOVREntries();
359 if (nNoCovrPlyEntries == 0) return;
360
361 float **pfpnc = (float **)malloc(nNoCovrPlyEntries * sizeof(float *));
362 float **pft0nc = pfpnc;
363 int *pipnc = (int *)malloc(nNoCovrPlyEntries * sizeof(int));
364
365 for (int j = 0; j < nNoCovrPlyEntries; j++) {
366 float *pf_entry =
367 (float *)malloc(theChart.GetNoCOVRTablePoints(j) * 2 * sizeof(float));
368 memcpy(pf_entry, theChart.GetNoCOVRTableHead(j),
369 theChart.GetNoCOVRTablePoints(j) * 2 * sizeof(float));
370 pft0nc[j] = pf_entry;
371 pipnc[j] = theChart.GetNoCOVRTablePoints(j);
372 }
373
374 pNoCovrPlyTable = pfpnc;
375 pNoCovrCntTable = pipnc;
376}
377
379
380ChartTableEntry::~ChartTableEntry() {
381 free(pFullPath);
382 free(pPlyTable);
383
384 for (int i = 0; i < nAuxPlyEntries; i++) free(pAuxPlyTable[i]);
385 free(pAuxPlyTable);
386 free(pAuxCntTable);
387
388 if (nNoCovrPlyEntries) {
389 for (int i = 0; i < nNoCovrPlyEntries; i++) free(pNoCovrPlyTable[i]);
390 free(pNoCovrPlyTable);
391 free(pNoCovrCntTable);
392 }
393
394 delete m_pfilename;
395 delete m_psFullPath;
396}
397
399
401
402bool ChartTableEntry::IsEarlierThan(const ChartTableEntry &cte) const {
403 wxDateTime mine(edition_date);
404 wxDateTime theirs(cte.edition_date);
405
406 if (!mine.IsValid() || !theirs.IsValid())
407 return false; // will have the effect of keeping all questionable charts
408
409 return (mine.IsEarlierThan(theirs));
410}
411
412bool ChartTableEntry::IsEqualTo(const ChartTableEntry &cte) const {
413 wxDateTime mine(edition_date);
414 wxDateTime theirs(cte.edition_date);
415
416 if (!mine.IsValid() || !theirs.IsValid())
417 return true; // will have the effect of keeping all questionable charts
418
419 return (mine.IsEqualTo(theirs));
420}
421
423
424static int convertChartType(int charttype) {
425 // Hackeroo here....
426 // dB version 14 had different ChartType Enum, patch it here
427 if (s_dbVersion == 14) {
428 switch (charttype) {
429 case 0:
430 return CHART_TYPE_KAP;
431 case 1:
432 return CHART_TYPE_GEO;
433 case 2:
434 return CHART_TYPE_S57;
435 case 3:
436 return CHART_TYPE_CM93;
437 case 4:
438 return CHART_TYPE_CM93COMP;
439 case 5:
440 return CHART_TYPE_UNKNOWN;
441 case 6:
442 return CHART_TYPE_DONTCARE;
443 case 7:
444 return CHART_TYPE_DUMMY;
445 default:
446 return CHART_TYPE_UNKNOWN;
447 }
448 }
449 return charttype;
450}
451
452static int convertChartFamily(int charttype, int chartfamily) {
453 if (s_dbVersion < 18) {
454 switch (charttype) {
455 case CHART_TYPE_KAP:
456 case CHART_TYPE_GEO:
457 return CHART_FAMILY_RASTER;
458
459 case CHART_TYPE_S57:
460 case CHART_TYPE_CM93:
461 case CHART_TYPE_CM93COMP:
462 return CHART_FAMILY_VECTOR;
463
464 default:
465 return CHART_FAMILY_UNKNOWN;
466 }
467 }
468 return chartfamily;
469}
470
471bool ChartTableEntry::Read(const ChartDatabase *pDb, wxInputStream &is) {
472 char path[4096], *cp;
473
474 Clear();
475
476 // Allow reading of current db format, and maybe others
477 ChartDatabase *pD = (ChartDatabase *)pDb;
478 int db_version = pD->GetVersion();
479
480 if (db_version == 18) {
481 // Read the path first
482 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
483 pFullPath = (char *)malloc(cp - path + 1);
484 strncpy(pFullPath, path, cp - path + 1);
485 wxLogVerbose(_T(" Chart %s"), pFullPath);
486
487 // Create and populate the helper members
488 m_pfilename = new wxString;
489 wxString fullfilename(pFullPath, wxConvUTF8);
490 wxFileName fn(fullfilename);
491 *m_pfilename = fn.GetFullName();
492 m_psFullPath = new wxString;
493 *m_psFullPath = fullfilename;
494 m_fullSystemPath = fullfilename;
495
496#ifdef __OCPN__ANDROID__
497 m_fullSystemPath = wxString(fullfilename.mb_str(wxConvUTF8));
498#endif
499 // Read the table entry
501 is.Read(&cte, sizeof(ChartTableEntry_onDisk_18));
502
503 // Transcribe the elements....
504 EntryOffset = cte.EntryOffset;
505 ChartType = cte.ChartType;
506 ChartFamily = cte.ChartFamily;
507 LatMax = cte.LatMax;
508 LatMin = cte.LatMin;
509 LonMax = cte.LonMax;
510 LonMin = cte.LonMin;
511
512 m_bbox.Set(LatMin, LonMin, LatMax, LonMax);
513
514 Skew = cte.skew;
515 ProjectionType = cte.ProjectionType;
516
517 SetScale(cte.Scale);
518 edition_date = cte.edition_date;
519 file_date = cte.file_date;
520
521 nPlyEntries = cte.nPlyEntries;
522 nAuxPlyEntries = cte.nAuxPlyEntries;
523
524 nNoCovrPlyEntries = cte.nNoCovrPlyEntries;
525
526 bValid = cte.bValid;
527
528 if (nPlyEntries) {
529 int npeSize = nPlyEntries * 2 * sizeof(float);
530 pPlyTable = (float *)malloc(npeSize);
531 is.Read(pPlyTable, npeSize);
532 }
533
534 if (nAuxPlyEntries) {
535 int napeSize = nAuxPlyEntries * sizeof(int);
536 pAuxPlyTable = (float **)malloc(nAuxPlyEntries * sizeof(float *));
537 pAuxCntTable = (int *)malloc(napeSize);
538 is.Read(pAuxCntTable, napeSize);
539
540 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
541 nAuxPlyEntry++) {
542 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
543 pAuxPlyTable[nAuxPlyEntry] = (float *)malloc(nfSize);
544 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
545 }
546 }
547
548 if (nNoCovrPlyEntries) {
549 int napeSize = nNoCovrPlyEntries * sizeof(int);
550 pNoCovrCntTable = (int *)malloc(napeSize);
551 is.Read(pNoCovrCntTable, napeSize);
552
553 pNoCovrPlyTable = (float **)malloc(nNoCovrPlyEntries * sizeof(float *));
554 for (int i = 0; i < nNoCovrPlyEntries; i++) {
555 int nfSize = pNoCovrCntTable[i] * 2 * sizeof(float);
556 pNoCovrPlyTable[i] = (float *)malloc(nfSize);
557 is.Read(pNoCovrPlyTable[i], nfSize);
558 }
559 }
560 }
561
562 else if (db_version == 17) {
563 // Read the path first
564 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
565 pFullPath = (char *)malloc(cp - path + 1);
566 strncpy(pFullPath, path, cp - path + 1);
567 wxLogVerbose(_T(" Chart %s"), pFullPath);
568
569 // Create and populate the helper members
570 m_pfilename = new wxString;
571 wxString fullfilename(pFullPath, wxConvUTF8);
572 wxFileName fn(fullfilename);
573 *m_pfilename = fn.GetFullName();
574 m_psFullPath = new wxString;
575 *m_psFullPath = fullfilename;
576
577 // Read the table entry
579 is.Read(&cte, sizeof(ChartTableEntry_onDisk_17));
580
581 // Transcribe the elements....
582 EntryOffset = cte.EntryOffset;
583 ChartType = cte.ChartType;
584 LatMax = cte.LatMax;
585 LatMin = cte.LatMin;
586 LonMax = cte.LonMax;
587 LonMin = cte.LonMin;
588
589 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
590
591 Skew = cte.skew;
592 ProjectionType = cte.ProjectionType;
593
594 SetScale(cte.Scale);
595 edition_date = cte.edition_date;
596 file_date = cte.file_date;
597
598 nPlyEntries = cte.nPlyEntries;
599 nAuxPlyEntries = cte.nAuxPlyEntries;
600
601 nNoCovrPlyEntries = cte.nNoCovrPlyEntries;
602
603 bValid = cte.bValid;
604
605 if (nPlyEntries) {
606 int npeSize = nPlyEntries * 2 * sizeof(float);
607 pPlyTable = (float *)malloc(npeSize);
608 is.Read(pPlyTable, npeSize);
609 }
610
611 if (nAuxPlyEntries) {
612 int napeSize = nAuxPlyEntries * sizeof(int);
613 pAuxPlyTable = (float **)malloc(nAuxPlyEntries * sizeof(float *));
614 pAuxCntTable = (int *)malloc(napeSize);
615 is.Read(pAuxCntTable, napeSize);
616
617 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
618 nAuxPlyEntry++) {
619 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
620 pAuxPlyTable[nAuxPlyEntry] = (float *)malloc(nfSize);
621 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
622 }
623 }
624
625 if (nNoCovrPlyEntries) {
626 int napeSize = nNoCovrPlyEntries * sizeof(int);
627 pNoCovrCntTable = (int *)malloc(napeSize);
628 is.Read(pNoCovrCntTable, napeSize);
629
630 pNoCovrPlyTable = (float **)malloc(nNoCovrPlyEntries * sizeof(float *));
631 for (int i = 0; i < nNoCovrPlyEntries; i++) {
632 int nfSize = pNoCovrCntTable[i] * 2 * sizeof(float);
633 pNoCovrPlyTable[i] = (float *)malloc(nfSize);
634 is.Read(pNoCovrPlyTable[i], nfSize);
635 }
636 }
637 }
638
639 else if (db_version == 16) {
640 // Read the path first
641 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
642 // TODO: optimize prepended dir
643 pFullPath = (char *)malloc(cp - path + 1);
644 strncpy(pFullPath, path, cp - path + 1);
645 wxLogVerbose(_T(" Chart %s"), pFullPath);
646
647 // Create and populate the helper members
648 m_pfilename = new wxString;
649 wxString fullfilename(pFullPath, wxConvUTF8);
650 wxFileName fn(fullfilename);
651 *m_pfilename = fn.GetFullName();
652 m_psFullPath = new wxString;
653 *m_psFullPath = fullfilename;
654
655 // Read the table entry
657 is.Read(&cte, sizeof(ChartTableEntry_onDisk_16));
658
659 // Transcribe the elements....
660 EntryOffset = cte.EntryOffset;
661 ChartType = cte.ChartType;
662 LatMax = cte.LatMax;
663 LatMin = cte.LatMin;
664 LonMax = cte.LonMax;
665 LonMin = cte.LonMin;
666
667 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
668
669 Skew = cte.skew;
670 ProjectionType = cte.ProjectionType;
671
672 SetScale(cte.Scale);
673 edition_date = cte.edition_date;
674 file_date = cte.file_date;
675
676 nPlyEntries = cte.nPlyEntries;
677 nAuxPlyEntries = cte.nAuxPlyEntries;
678
679 bValid = cte.bValid;
680
681 if (nPlyEntries) {
682 int npeSize = nPlyEntries * 2 * sizeof(float);
683 pPlyTable = (float *)malloc(npeSize);
684 is.Read(pPlyTable, npeSize);
685 }
686
687 if (nAuxPlyEntries) {
688 int napeSize = nAuxPlyEntries * sizeof(int);
689 pAuxPlyTable = (float **)malloc(nAuxPlyEntries * sizeof(float *));
690 pAuxCntTable = (int *)malloc(napeSize);
691 is.Read(pAuxCntTable, napeSize);
692
693 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
694 nAuxPlyEntry++) {
695 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
696 pAuxPlyTable[nAuxPlyEntry] = (float *)malloc(nfSize);
697 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
698 }
699 }
700 }
701
702 else if (db_version == 15) {
703 // Read the path first
704 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
705 // TODO: optimize prepended dir
706 pFullPath = (char *)malloc(cp - path + 1);
707 strncpy(pFullPath, path, cp - path + 1);
708 wxLogVerbose(_T(" Chart %s"), pFullPath);
709
710 // Read the table entry
712 is.Read(&cte, sizeof(ChartTableEntry_onDisk_15));
713
714 // Transcribe the elements....
715 EntryOffset = cte.EntryOffset;
716 ChartType = cte.ChartType;
717 LatMax = cte.LatMax;
718 LatMin = cte.LatMin;
719 LonMax = cte.LonMax;
720 LonMin = cte.LonMin;
721
722 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
723
724 SetScale(cte.Scale);
725 edition_date = cte.edition_date;
726 file_date = cte.file_date;
727
728 nPlyEntries = cte.nPlyEntries;
729 nAuxPlyEntries = cte.nAuxPlyEntries;
730
731 bValid = cte.bValid;
732
733 if (nPlyEntries) {
734 int npeSize = nPlyEntries * 2 * sizeof(float);
735 pPlyTable = (float *)malloc(npeSize);
736 is.Read(pPlyTable, npeSize);
737 }
738
739 if (nAuxPlyEntries) {
740 int napeSize = nAuxPlyEntries * sizeof(int);
741 pAuxPlyTable = (float **)malloc(nAuxPlyEntries * sizeof(float *));
742 pAuxCntTable = (int *)malloc(napeSize);
743 is.Read(pAuxCntTable, napeSize);
744
745 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
746 nAuxPlyEntry++) {
747 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
748 pAuxPlyTable[nAuxPlyEntry] = (float *)malloc(nfSize);
749 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
750 }
751 }
752 } else if (db_version == 14) {
753 // Read the path first
754 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
755 pFullPath = (char *)malloc(cp - path + 1);
756 strncpy(pFullPath, path, cp - path + 1);
757 wxLogVerbose(_T(" Chart %s"), pFullPath);
758
759 // Read the table entry
761 is.Read(&cte, sizeof(ChartTableEntry_onDisk_14));
762
763 // Transcribe the elements....
764 EntryOffset = cte.EntryOffset;
765 ChartType = cte.ChartType;
766 LatMax = cte.LatMax;
767 LatMin = cte.LatMin;
768 LonMax = cte.LonMax;
769 LonMin = cte.LonMin;
770
771 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
772
773 SetScale(cte.Scale);
774 edition_date = cte.edition_date;
775 file_date = 0; // file_date does not exist in V14;
776 nPlyEntries = cte.nPlyEntries;
777 nAuxPlyEntries = cte.nAuxPlyEntries;
778 bValid = cte.bValid;
779
780 if (nPlyEntries) {
781 int npeSize = nPlyEntries * 2 * sizeof(float);
782 pPlyTable = (float *)malloc(npeSize);
783 is.Read(pPlyTable, npeSize);
784 }
785
786 if (nAuxPlyEntries) {
787 int napeSize = nAuxPlyEntries * sizeof(int);
788 pAuxPlyTable = (float **)malloc(nAuxPlyEntries * sizeof(float *));
789 pAuxCntTable = (int *)malloc(napeSize);
790 is.Read(pAuxCntTable, napeSize);
791
792 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
793 nAuxPlyEntry++) {
794 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
795 pAuxPlyTable[nAuxPlyEntry] = (float *)malloc(nfSize);
796 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
797 }
798 }
799 }
800 ChartFamily = convertChartFamily(ChartType, ChartFamily);
801 ChartType = convertChartType(ChartType);
802
803 return true;
804}
805
807
808bool ChartTableEntry::Write(const ChartDatabase *pDb, wxOutputStream &os) {
809 os.Write(pFullPath, strlen(pFullPath) + 1);
810
811 // Write the current version type only
812 // Create an on_disk table entry
814
815 // Transcribe the elements....
816 cte.EntryOffset = EntryOffset;
817 cte.ChartType = ChartType;
818 cte.ChartFamily = ChartFamily;
819 cte.LatMax = LatMax;
820 cte.LatMin = LatMin;
821 cte.LonMax = LonMax;
822 cte.LonMin = LonMin;
823
824 cte.Scale = Scale;
825 cte.edition_date = edition_date;
826 cte.file_date = file_date;
827
828 cte.nPlyEntries = nPlyEntries;
829 cte.nAuxPlyEntries = nAuxPlyEntries;
830
831 cte.skew = Skew;
832 cte.ProjectionType = ProjectionType;
833
834 cte.bValid = bValid;
835
836 cte.nNoCovrPlyEntries = nNoCovrPlyEntries;
837
838 os.Write(&cte, sizeof(ChartTableEntry_onDisk_18));
839 wxLogVerbose(_T(" Wrote Chart %s"), pFullPath);
840
841 // Write out the tables
842 if (nPlyEntries) {
843 int npeSize = nPlyEntries * 2 * sizeof(float);
844 os.Write(pPlyTable, npeSize);
845 }
846
847 if (nAuxPlyEntries) {
848 int napeSize = nAuxPlyEntries * sizeof(int);
849 os.Write(pAuxCntTable, napeSize);
850
851 for (int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries; nAuxPlyEntry++) {
852 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 * sizeof(float);
853 os.Write(pAuxPlyTable[nAuxPlyEntry], nfSize);
854 }
855 }
856
857 if (nNoCovrPlyEntries) {
858 int ncSize = nNoCovrPlyEntries * sizeof(int);
859 os.Write(pNoCovrCntTable, ncSize);
860
861 for (int i = 0; i < nNoCovrPlyEntries; i++) {
862 int nctSize = pNoCovrCntTable[i] * 2 * sizeof(float);
863 os.Write(pNoCovrPlyTable[i], nctSize);
864 }
865 }
866
867 return true;
868}
869
871
872void ChartTableEntry::Clear() {
873 // memset(this, 0, sizeof(ChartTableEntry));
874
875 pFullPath = NULL;
876 pPlyTable = NULL;
877 pAuxPlyTable = NULL;
878 pAuxCntTable = NULL;
879 bValid = false;
880 ;
881 pNoCovrCntTable = NULL;
882 pNoCovrPlyTable = NULL;
883
884 nNoCovrPlyEntries = 0;
885 nAuxPlyEntries = 0;
886
887 m_pfilename = NULL; // a helper member, not on disk
888 m_psFullPath = NULL;
889}
890
892
893void ChartTableEntry::Disable() {
894 // Mark this chart in the database, so that it will not be seen during this
895 // run How? By setting the chart bounding box to an absurd value
896 // TODO... Fix this heinous hack
897 LatMax += (float)1000.;
898 LatMin += (float)1000.;
899}
900
901void ChartTableEntry::ReEnable() {
902 if (LatMax > 90.) {
903 LatMax -= (float)1000.;
904 LatMin -= (float)1000.;
905 }
906}
907
908std::vector<float> ChartTableEntry::GetReducedPlyPoints() {
909 if (m_reducedPlyPoints.size()) return m_reducedPlyPoints;
910
911 // Reduce the LOD of the chart outline PlyPoints
912 float LOD_meters = 1;
913
914 float plylat, plylon;
915 const int nPoints = GetnPlyEntries();
916
917 float *fpo = GetpPlyTable();
918
919 double *ppd = new double[nPoints * 2];
920 double *ppsm = new double[nPoints * 2];
921 double *npr = ppd;
922 double *npsm = ppsm;
923 for (int i = 0; i < nPoints; i++) {
924 plylat = fpo[i * 2];
925 plylon = fpo[i * 2 + 1];
926
927 double x, y;
928 toSM(plylat, plylon, fpo[0], fpo[1], &x, &y);
929
930 *npr++ = plylon;
931 *npr++ = plylat;
932 *npsm++ = x;
933 *npsm++ = y;
934 }
935
936 std::vector<int> index_keep;
937 if (nPoints > 10) {
938 index_keep.push_back(0);
939 index_keep.push_back(nPoints - 1);
940 index_keep.push_back(1);
941 index_keep.push_back(nPoints - 2);
942
943 DouglasPeuckerM(ppsm, 1, nPoints - 2, LOD_meters, &index_keep);
944
945 } else {
946 index_keep.resize(nPoints);
947 for (int i = 0; i < nPoints; i++) index_keep[i] = i;
948 }
949
950 double *ppr = ppd;
951 for (int ip = 0; ip < nPoints; ip++) {
952 double x = *ppr++;
953 double y = *ppr++;
954
955 for (unsigned int j = 0; j < index_keep.size(); j++) {
956 if (index_keep[j] == ip) {
957 m_reducedPlyPoints.push_back(x);
958 m_reducedPlyPoints.push_back(y);
959 break;
960 }
961 }
962 }
963
964 delete[] ppd;
965 delete[] ppsm;
966
967 int nprr = m_reducedPlyPoints.size() / 2;
968
969 return m_reducedPlyPoints;
970}
971
972std::vector<float> ChartTableEntry::GetReducedAuxPlyPoints(int iTable) {
973 // Maybe need to initialize the vector
974 if (!m_reducedAuxPlyPointsVector.size()) {
975 std::vector<float> vec;
976 for (int i = 0; i < GetnAuxPlyEntries(); i++) {
977 m_reducedAuxPlyPointsVector.push_back(vec);
978 }
979 }
980
981 std::vector<float> vec;
982
983 // Invalid parameter
984 if ((unsigned int)iTable >= m_reducedAuxPlyPointsVector.size()) return vec;
985
986 if (m_reducedAuxPlyPointsVector.at(iTable).size())
987 return m_reducedAuxPlyPointsVector.at(iTable);
988
989 // Reduce the LOD of the chart outline PlyPoints
990 float LOD_meters = 1.0;
991
992 const int nPoints = GetAuxCntTableEntry(iTable);
993 float *fpo = GetpAuxPlyTableEntry(iTable);
994
995 double *ppd = new double[nPoints * 2];
996 double *ppsm = new double[nPoints * 2];
997 double *npr = ppd;
998 double *npsm = ppsm;
999 float plylat, plylon;
1000
1001 for (int i = 0; i < nPoints; i++) {
1002 plylat = fpo[i * 2];
1003 plylon = fpo[i * 2 + 1];
1004
1005 double x, y;
1006 toSM(plylat, plylon, fpo[0], fpo[1], &x, &y);
1007
1008 *npr++ = plylon;
1009 *npr++ = plylat;
1010 *npsm++ = x;
1011 *npsm++ = y;
1012 }
1013
1014 std::vector<int> index_keep;
1015 if (nPoints > 10) {
1016 index_keep.push_back(0);
1017 index_keep.push_back(nPoints - 1);
1018 index_keep.push_back(1);
1019 index_keep.push_back(nPoints - 2);
1020
1021 DouglasPeuckerM(ppsm, 1, nPoints - 2, LOD_meters, &index_keep);
1022
1023 } else {
1024 index_keep.resize(nPoints);
1025 for (int i = 0; i < nPoints; i++) index_keep[i] = i;
1026 }
1027
1028 int nnn = index_keep.size();
1029
1030 double *ppr = ppd;
1031 for (int ip = 0; ip < nPoints; ip++) {
1032 double x = *ppr++;
1033 double y = *ppr++;
1034
1035 for (unsigned int j = 0; j < index_keep.size(); j++) {
1036 if (index_keep[j] == ip) {
1037 vec.push_back(x);
1038 vec.push_back(y);
1039 break;
1040 }
1041 }
1042 }
1043
1044 delete[] ppd;
1045 delete[] ppsm;
1046
1047 m_reducedAuxPlyPointsVector[iTable] = vec;
1048
1049 int nprr = vec.size() / 2;
1050
1051 return vec;
1052}
1053
1055// ChartDatabase
1057
1058WX_DEFINE_OBJARRAY(ChartTable);
1059
1060ChartDatabase::ChartDatabase() {
1061 bValid = false;
1062 m_b_busy = false;
1063
1064 m_ChartTableEntryDummy.Clear();
1065
1066 UpdateChartClassDescriptorArray();
1067}
1068
1069void ChartDatabase::UpdateChartClassDescriptorArray(void) {
1070 if (m_ChartClassDescriptorArray.empty()) {
1071 m_ChartClassDescriptorArray.push_back(
1072 ChartClassDescriptor(_T("ChartKAP"), _T("*.kap"), BUILTIN_DESCRIPTOR));
1073 m_ChartClassDescriptorArray.push_back(
1074 ChartClassDescriptor(_T("ChartGEO"), _T("*.geo"), BUILTIN_DESCRIPTOR));
1075 m_ChartClassDescriptorArray.push_back(
1076 ChartClassDescriptor(_T("s57chart"), _T("*.000"), BUILTIN_DESCRIPTOR));
1077 m_ChartClassDescriptorArray.push_back(
1078 ChartClassDescriptor(_T("s57chart"), _T("*.s57"), BUILTIN_DESCRIPTOR));
1079 m_ChartClassDescriptorArray.push_back(ChartClassDescriptor(
1080 _T("cm93compchart"), _T("00300000.a"), BUILTIN_DESCRIPTOR));
1081 m_ChartClassDescriptorArray.push_back(ChartClassDescriptor(
1082 _T("ChartMbTiles"), _T("*.mbtiles"), BUILTIN_DESCRIPTOR));
1083 }
1084 // If the PlugIn Manager exists, get the array of dynamically loadable
1085 // chart class names
1086 if (g_pi_manager) {
1087 m_ChartClassDescriptorArray.erase(
1088 std::remove_if(m_ChartClassDescriptorArray.begin(),
1089 m_ChartClassDescriptorArray.end(),
1090 [](const ChartClassDescriptor &cd) {
1091 return cd.m_descriptor_type == PLUGIN_DESCRIPTOR;
1092 }),
1093 m_ChartClassDescriptorArray.end());
1094
1095 wxArrayString array = g_pi_manager->GetPlugInChartClassNameArray();
1096 for (unsigned int j = 0; j < array.GetCount(); j++) {
1097 // Instantiate a blank chart to retrieve the directory search mask for
1098 // this chart type
1099 wxString class_name = array[j];
1100 ChartPlugInWrapper *cpiw = new ChartPlugInWrapper(class_name);
1101 if (cpiw) {
1102 wxString mask = cpiw->GetFileSearchMask();
1103
1104 // Create a new descriptor and add it to the database
1105 m_ChartClassDescriptorArray.push_back(
1106 ChartClassDescriptor(class_name, mask, PLUGIN_DESCRIPTOR));
1107 delete cpiw;
1108 }
1109 }
1110 }
1111}
1112
1113const ChartTableEntry &ChartDatabase::GetChartTableEntry(int index) const {
1114 if (index < GetChartTableEntries())
1115 return active_chartTable[index];
1116 else
1117 return m_ChartTableEntryDummy;
1118}
1119
1120ChartTableEntry *ChartDatabase::GetpChartTableEntry(int index) const {
1121 if (index < GetChartTableEntries())
1122 return &active_chartTable[index];
1123 else
1124 return (ChartTableEntry *)&m_ChartTableEntryDummy;
1125}
1126
1127bool ChartDatabase::CompareChartDirArray(ArrayOfCDI &test_array) {
1128 // Compare the parameter "test_array" with this.m_dir_array
1129 // Return true if functionally identical (order does not signify).
1130
1131 if (test_array.GetCount() != m_dir_array.GetCount()) return false;
1132
1133 bool bfound_inner;
1134 unsigned int nfound_outer = 0;
1135
1136 for (unsigned int i = 0; i < test_array.GetCount(); i++) {
1137 ChartDirInfo p = test_array[i];
1138 bfound_inner = false;
1139 for (unsigned int j = 0; j < m_dir_array.GetCount(); j++) {
1140 ChartDirInfo q = m_dir_array[j];
1141
1142 if (p.fullpath.IsSameAs(q.fullpath)) {
1143 bfound_inner = true;
1144 break;
1145 }
1146 }
1147 if (bfound_inner) nfound_outer++;
1148 }
1149
1150 return (nfound_outer == test_array.GetCount());
1151}
1152
1153wxString ChartDatabase::GetMagicNumberCached(wxString dir) {
1154 for (unsigned int j = 0; j < m_dir_array.GetCount(); j++) {
1155 ChartDirInfo q = m_dir_array[j];
1156 if (dir.IsSameAs(q.fullpath)) return q.magic_number;
1157 }
1158
1159 return _T("");
1160}
1161
1162bool ChartDatabase::Read(const wxString &filePath) {
1163 ChartTableEntry entry;
1164 int entries;
1165
1166 bValid = false;
1167
1168 wxFileName file(filePath);
1169 if (!file.FileExists()) return false;
1170
1171 m_DBFileName = filePath;
1172
1173 wxFFileInputStream ifs(filePath);
1174 if (!ifs.Ok()) return false;
1175
1176 ChartTableHeader cth;
1177 cth.Read(ifs);
1178 if (!cth.CheckValid()) return false;
1179
1180 // Capture the version number
1181 char vbo[5];
1182 memcpy(vbo, cth.GetDBVersionString(), 4);
1183 vbo[4] = 0;
1184 m_dbversion = atoi(&vbo[1]);
1185 s_dbVersion = m_dbversion; // save the static copy
1186
1187 wxLogVerbose(wxT("Chartdb:Reading %d directory entries, %d table entries"),
1188 cth.GetDirEntries(), cth.GetTableEntries());
1189 wxLogMessage(_T("Chartdb: Chart directory list follows"));
1190 if (0 == cth.GetDirEntries()) wxLogMessage(_T(" Nil"));
1191
1192 int ind = 0;
1193 for (int iDir = 0; iDir < cth.GetDirEntries(); iDir++) {
1194 wxString dir;
1195 int dirlen;
1196 ifs.Read(&dirlen, sizeof(int));
1197 while (dirlen > 0) {
1198 char dirbuf[1024];
1199 int alen = dirlen > 1023 ? 1023 : dirlen;
1200 if (ifs.Read(&dirbuf, alen).Eof()) goto read_error;
1201 dirbuf[alen] = 0;
1202 dirlen -= alen;
1203 dir.Append(wxString(dirbuf, wxConvUTF8));
1204 }
1205 wxString msg;
1206 msg.Printf(wxT(" Chart directory #%d: "), iDir);
1207 msg.Append(dir);
1208 wxLogMessage(msg);
1209 m_chartDirs.Add(dir);
1210 }
1211
1212 entries = cth.GetTableEntries();
1213 active_chartTable.Alloc(entries);
1214 active_chartTable_pathindex.clear();
1215 while (entries-- && entry.Read(this, ifs)) {
1216 active_chartTable_pathindex[entry.GetFullSystemPath()] = ind++;
1217 active_chartTable.Add(entry);
1218 }
1219
1220 entry.Clear();
1221 bValid = true;
1222 entry.SetAvailable(true);
1223
1224 m_nentries = active_chartTable.GetCount();
1225 return true;
1226
1227read_error:
1228 bValid = false;
1229 m_nentries = active_chartTable.GetCount();
1230 return false;
1231}
1232
1234
1235bool ChartDatabase::Write(const wxString &filePath) {
1236 wxFileName file(filePath);
1237 wxFileName dir(
1238 file.GetPath(wxPATH_GET_SEPARATOR | wxPATH_GET_VOLUME, wxPATH_NATIVE));
1239
1240 if (!dir.DirExists() && !dir.Mkdir()) return false;
1241
1242 wxFFileOutputStream ofs(filePath);
1243 if (!ofs.Ok()) return false;
1244
1245 ChartTableHeader cth(m_chartDirs.GetCount(), active_chartTable.GetCount());
1246 cth.Write(ofs);
1247
1248 for (int iDir = 0; iDir < cth.GetDirEntries(); iDir++) {
1249 wxString &dir = m_chartDirs[iDir];
1250 int dirlen = dir.length();
1251 char s[200];
1252 strncpy(s, dir.mb_str(wxConvUTF8), 199);
1253 s[199] = 0;
1254 dirlen = strlen(s);
1255 ofs.Write(&dirlen, sizeof(int));
1256 // ofs.Write(dir.fn_str(), dirlen);
1257 ofs.Write(s, dirlen);
1258 }
1259
1260 for (UINT32 iTable = 0; iTable < active_chartTable.size(); iTable++)
1261 active_chartTable[iTable].Write(this, ofs);
1262
1263 // Explicitly set the version
1264 m_dbversion = DB_VERSION_CURRENT;
1265
1266 return true;
1267}
1268
1270wxString SplitPath(wxString s, wxString tkd, int nchar, int offset,
1271 int *pn_split) {
1272 wxString r;
1273 int ncr = 0;
1274
1275 int rlen = offset;
1276 wxStringTokenizer tkz(s, tkd);
1277 while (tkz.HasMoreTokens()) {
1278 wxString token = tkz.GetNextToken();
1279 if ((rlen + (int)token.Len() + 1) < nchar) {
1280 r += token;
1281 r += tkd[0];
1282 rlen += token.Len() + 1;
1283 } else {
1284 r += _T("\n");
1285 ncr++;
1286 for (int i = 0; i < offset; i++) {
1287 r += _T(" ");
1288 }
1289 r += token;
1290 r += tkd[0];
1291 rlen = offset + token.Len() + 1;
1292 }
1293 }
1294
1295 if (pn_split) *pn_split = ncr;
1296
1297 return r.Mid(0, r.Len() - 1); // strip the last separator char
1298}
1299
1300wxString ChartDatabase::GetFullChartInfo(ChartBase *pc, int dbIndex,
1301 int *char_width, int *line_count) {
1302 wxString r;
1303 int lc = 0;
1304 unsigned int max_width = 0;
1305 int ncr;
1306 unsigned int target_width = 60;
1307
1308 const ChartTableEntry &cte = GetChartTableEntry(dbIndex);
1309 if (1) // TODO why can't this be cte.GetbValid()?
1310 {
1311 wxString line;
1312 line = _(" ChartFile: ");
1313 wxString longline = *(cte.GetpsFullPath());
1314
1315 if (longline.Len() > target_width) {
1316 line += SplitPath(longline, _T("/,\\"), target_width, 15, &ncr);
1317 max_width = wxMax(max_width, target_width + 4);
1318 lc += ncr;
1319 } else {
1320 line += longline;
1321 max_width = wxMax(max_width, line.Len() + 4);
1322 }
1323
1324 r += line;
1325 r += _T("\n");
1326 lc++;
1327
1328 line.Empty();
1329 if (pc) {
1330 line = _(" Name: ");
1331 wxString longline = pc->GetName();
1332
1333 wxString tkz;
1334 if (longline.Find(' ') != wxNOT_FOUND) // assume a proper name
1335 tkz = _T(" ");
1336 else
1337 tkz = _T("/,\\"); // else a file name
1338
1339 if (longline.Len() > target_width) {
1340 line += SplitPath(pc->GetName(), tkz, target_width, 12, &ncr);
1341 max_width = wxMax(max_width, target_width + 4);
1342 lc += ncr;
1343 } else {
1344 line += longline;
1345 max_width = wxMax(max_width, line.Len() + 4);
1346 }
1347 }
1348
1349 line += _T("\n");
1350 r += line;
1351 lc++;
1352
1353 if (pc) // chart is loaded and available
1354 line.Printf(_T(" %s: 1:%d"), _("Scale"), pc->GetNativeScale());
1355 else
1356 line.Printf(_T(" %s: 1:%d"), _("Scale"), cte.GetScale());
1357
1358 line += _T("\n");
1359 max_width = wxMax(max_width, line.Len());
1360 r += line;
1361 lc++;
1362
1363 if (pc) {
1364 line.Empty();
1365 line = _(" ID: ");
1366 line += pc->GetID();
1367 line += _T("\n");
1368 max_width = wxMax(max_width, line.Len());
1369 r += line;
1370 lc++;
1371
1372 line.Empty();
1373 line = _(" Depth Units: ");
1374 line += pc->GetDepthUnits();
1375 line += _T("\n");
1376 max_width = wxMax(max_width, line.Len());
1377 r += line;
1378 lc++;
1379
1380 line.Empty();
1381 line = _(" Soundings: ");
1382 line += pc->GetSoundingsDatum();
1383 line += _T("\n");
1384 max_width = wxMax(max_width, line.Len());
1385 r += line;
1386 lc++;
1387
1388 line.Empty();
1389 line = _(" Datum: ");
1390 line += pc->GetDatumString();
1391 line += _T("\n");
1392 max_width = wxMax(max_width, line.Len());
1393 r += line;
1394 lc++;
1395 }
1396
1397 line = _(" Projection: ");
1398 if (PROJECTION_UNKNOWN == cte.GetChartProjectionType())
1399 line += _("Unknown");
1400 else if (PROJECTION_MERCATOR == cte.GetChartProjectionType())
1401 line += _("Mercator");
1402 else if (PROJECTION_TRANSVERSE_MERCATOR == cte.GetChartProjectionType())
1403 line += _("Transverse Mercator");
1404 else if (PROJECTION_POLYCONIC == cte.GetChartProjectionType())
1405 line += _("Polyconic");
1406 else if (PROJECTION_WEB_MERCATOR == cte.GetChartProjectionType())
1407 line += _("Web Mercator (EPSG:3857)");
1408 line += _T("\n");
1409 max_width = wxMax(max_width, line.Len());
1410 r += line;
1411 lc++;
1412
1413 line.Empty();
1414 if (pc) {
1415 line = _(" Source Edition: ");
1416 line += pc->GetSE();
1417 line += _T("\n");
1418 max_width = wxMax(max_width, line.Len());
1419 r += line;
1420 lc++;
1421
1422 line.Empty();
1423 wxDateTime ed = pc->GetEditionDate();
1424 if (ed.IsValid()) {
1425 line = _(" Updated: ");
1426 line += ed.FormatISODate();
1427 line += _T("\n");
1428 max_width = wxMax(max_width, line.Len());
1429 r += line;
1430 }
1431 lc++;
1432 }
1433
1434 line.Empty();
1435 if (pc && pc->GetExtraInfo().Len()) {
1436 line += pc->GetExtraInfo();
1437 line += _T("\n");
1438 max_width = wxMax(max_width, line.Len());
1439 r += line;
1440 lc++;
1441 }
1442 }
1443
1444 if (line_count) *line_count = lc;
1445
1446 if (char_width) *char_width = max_width;
1447
1448 return r;
1449}
1450
1451// ----------------------------------------------------------------------------
1452// Create Chart Table Database by directory search
1453// resulting in valid pChartTable in (this)
1454// ----------------------------------------------------------------------------
1455bool ChartDatabase::Create(ArrayOfCDI &dir_array,
1456 wxGenericProgressDialog *pprog) {
1457 m_dir_array = dir_array;
1458
1459 bValid = false;
1460
1461 m_chartDirs.Clear();
1462 active_chartTable.Clear();
1463 active_chartTable_pathindex.clear();
1464
1465 Update(dir_array, true, pprog); // force the update the reload everything
1466
1467 bValid = true;
1468
1469 // Explicitly set the version
1470 m_dbversion = DB_VERSION_CURRENT;
1471
1472 return true;
1473}
1474
1475/*
1476 * Traverse a directory recursively and find the GSHHG directory
1477 * that contains GSHHG data files.
1478 */
1479class GshhsTraverser : public wxDirTraverser {
1480public:
1481 GshhsTraverser() {}
1482 virtual wxDirTraverseResult OnFile(const wxString &filename) override {
1483 wxFileName fn(filename);
1484 wxFileName dir(fn.GetPath());
1485 if (fn.GetFullName().Matches(_T("poly-*-1.dat")) &&
1486 dir.GetFullName().IsSameAs(_T("GSHHG"), false)) {
1487 parent_dir = fn.GetPath(wxPATH_GET_VOLUME | wxPATH_GET_SEPARATOR);
1488 return wxDIR_STOP;
1489 }
1490 return wxDIR_CONTINUE;
1491 }
1492 virtual wxDirTraverseResult OnDir(const wxString &dirname) override {
1493 // Always recurse into directories.
1494 return wxDIR_CONTINUE;
1495 }
1496 wxString GetGshhsDir() const { return parent_dir; }
1497
1498private:
1499 wxString parent_dir;
1500};
1501
1502/*
1503 * Find and return the full path a of directory containing GSHHG data files.
1504 * Search recursively starting from directory.
1505 */
1506wxString findGshhgDirectory(const wxString &directory) {
1507 wxDir dir(directory);
1508 if (!dir.IsOpened()) {
1509 return wxEmptyString;
1510 }
1511 GshhsTraverser traverser;
1512 dir.Traverse(traverser, wxEmptyString, wxDIR_FILES | wxDIR_DIRS);
1513 return traverser.GetGshhsDir();
1514}
1515
1516// ----------------------------------------------------------------------------
1517// Update existing ChartTable Database by directory search
1518// resulting in valid pChartTable in (this)
1519// ----------------------------------------------------------------------------
1520bool ChartDatabase::Update(ArrayOfCDI &dir_array, bool bForce,
1521 wxGenericProgressDialog *pprog) {
1522 m_dir_array = dir_array;
1523
1524 bValid = false; // database is not useable right now...
1525 m_b_busy = true;
1526
1527 // Mark all charts provisionally invalid
1528 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++)
1529 active_chartTable[i].SetValid(false);
1530
1531 m_chartDirs.Clear();
1532
1533 if (bForce) active_chartTable.Clear();
1534
1535 bool lbForce = bForce;
1536
1537 // Do a dB Version upgrade if the current one is obsolete
1538 if (s_dbVersion != DB_VERSION_CURRENT) {
1539 active_chartTable.Clear();
1540 lbForce = true;
1541 s_dbVersion = DB_VERSION_CURRENT; // Update the static indicator
1542 m_dbversion = DB_VERSION_CURRENT; // and the member
1543 }
1544
1545 // Get the new charts
1546
1547 for (unsigned int j = 0; j < dir_array.GetCount(); j++) {
1548 ChartDirInfo dir_info = dir_array[j];
1549
1550 // On Android, with SDK >= 30, traversal of a folder that is
1551 // on within the "scoped storage" domain is very slow.
1552 // Aviod it....
1553#ifdef __OCPN__ANDROID__
1554 if (!androidIsDirWritable(dir_info.fullpath)) continue;
1555#endif
1556
1557 wxString dir_magic;
1558 // Recursively search for a directory that contains GSHHG files starting
1559 // from dir_info.
1560 wxString gshhg_dir = findGshhgDirectory(dir_info.fullpath);
1561 if (!gshhg_dir.empty()) {
1562 // If some polygons exist in the directory, set it as the one to use for
1563 // GSHHG
1564 // TODO: We should probably compare the version and maybe resolutions
1565 // available with what is currently used...
1566 wxLogMessage("Updating GSHHG directory: %s", gshhg_dir.c_str());
1567 gWorldMapLocation = gshhg_dir;
1568 }
1569 if (dir_info.fullpath.Find(_T("OSMSHP")) != wxNOT_FOUND) {
1570 if (!wxDir::FindFirst(dir_info.fullpath, "basemap_*.shp").empty()) {
1571 gWorldShapefileLocation =
1572 dir_info.fullpath + wxFileName::GetPathSeparator();
1573 gShapeBasemap.Reset();
1574 }
1575 }
1576
1577 TraverseDirAndAddCharts(dir_info, pprog, dir_magic, lbForce);
1578
1579 // Update the dir_list entry, even if the magic values are the same
1580
1581 dir_info.magic_number = dir_magic;
1582 dir_array.RemoveAt(j);
1583 dir_array.Insert(dir_info, j);
1584
1585 m_chartDirs.Add(dir_info.fullpath);
1586 } // for
1587
1588 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++) {
1589 if (!active_chartTable[i].GetbValid()) {
1590 active_chartTable.RemoveAt(i);
1591 i--; // entry is gone, recheck this index for next entry
1592 }
1593 }
1594
1595 // And once more, setting the Entry index field
1596 active_chartTable_pathindex.clear();
1597 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++) {
1598 active_chartTable_pathindex[active_chartTable[i].GetFullSystemPath()] = i;
1599 active_chartTable[i].SetEntryOffset(i);
1600 }
1601
1602 m_nentries = active_chartTable.GetCount();
1603
1604 bValid = true;
1605 m_b_busy = false;
1606 return true;
1607}
1608
1609//-------------------------------------------------------------------
1610// Find Chart dbIndex
1611//-------------------------------------------------------------------
1612
1613int ChartDatabase::FinddbIndex(wxString PathToFind) {
1614#if 0
1615 // Find the chart
1616 for(unsigned int i=0 ; i<active_chartTable.GetCount() ; i++)
1617 {
1618 if(active_chartTable[i].GetpsFullPath()->IsSameAs(PathToFind))
1619 {
1620 return i;
1621 }
1622 }
1623#else
1624 if (active_chartTable_pathindex.find(PathToFind) !=
1625 active_chartTable_pathindex.end())
1626 return active_chartTable_pathindex[PathToFind];
1627#endif
1628
1629 return -1;
1630}
1631
1632//-------------------------------------------------------------------
1633// Disable Chart
1634//-------------------------------------------------------------------
1635
1636int ChartDatabase::DisableChart(wxString &PathToDisable) {
1637 int index = FinddbIndex(PathToDisable);
1638 if (index != -1) {
1639 ChartTableEntry *pentry = &active_chartTable[index];
1640 pentry->Disable();
1641 return 1;
1642 }
1643 return 0;
1644}
1645
1646// ----------------------------------------------------------------------------
1647// Traverse the given directory looking for charts
1648// If bupdate is true, also search the existing database for a name match.
1649// If target chart is already in database, mark the entry valid and skip
1650// additional processing
1651// ----------------------------------------------------------------------------
1652
1653int ChartDatabase::TraverseDirAndAddCharts(ChartDirInfo &dir_info,
1654 wxGenericProgressDialog *pprog,
1655 wxString &dir_magic, bool bForce) {
1656 // Extract the true dir name and magic number from the compound string
1657 wxString dir_path = dir_info.fullpath;
1658#ifdef __OCPN__ANDROID__
1659 dir_path = wxString(dir_info.fullpath.mb_str(wxConvUTF8));
1660#endif
1661
1662 wxString old_magic = dir_info.magic_number;
1663 wxString new_magic = old_magic;
1664 dir_magic = old_magic; // provisionally the same
1665
1666 int nAdd = 0;
1667
1668 bool b_skipDetectDirChange = false;
1669 bool b_dirchange = false;
1670
1671 // Does this directory actually exist?
1672 if (!wxDir::Exists(dir_path)) return 0;
1673
1674 // Check to see if this is a cm93 directory root
1675 // If so, skip the DetectDirChange since it may be very slow
1676 // and give no information
1677 // Assume a change has happened, and process accordingly
1678 bool b_cm93 = Check_CM93_Structure(dir_path);
1679 if (b_cm93) {
1680 b_skipDetectDirChange = true;
1681 b_dirchange = true;
1682 }
1683
1684 // Quick scan the directory to see if it has changed
1685 // If not, there is no need to scan again.....
1686 if (!b_skipDetectDirChange)
1687 b_dirchange = DetectDirChange(dir_path, dir_info.fullpath, old_magic,
1688 new_magic, pprog);
1689
1690 if (!bForce && !b_dirchange) {
1691 wxString msg(_T(" No change detected on directory "));
1692 msg.Append(dir_path);
1693 wxLogMessage(msg);
1694
1695 // Traverse the database, and mark as valid all charts coming from this
1696 // dir, or anywhere in its tree
1697
1698 wxFileName fn_dir(dir_path, _T("stuff"));
1699 unsigned int dir_path_count = fn_dir.GetDirCount();
1700
1701 if (pprog) pprog->SetTitle(_("OpenCPN Chart Scan...."));
1702
1703 int nEntries = active_chartTable.GetCount();
1704
1705 for (int ic = 0; ic < nEntries; ic++) {
1706 wxFileName fn(active_chartTable[ic].GetFullSystemPath());
1707
1708 while (fn.GetDirCount() >= dir_path_count) {
1709 if (fn.GetPath() == dir_path) {
1710 active_chartTable[ic].SetValid(true);
1711 // if(pprog)
1712 // pprog->Update((ic * 100)
1713 // /nEntries, fn.GetFullPath());
1714
1715 break;
1716 }
1717 fn.RemoveLastDir();
1718 }
1719 }
1720
1721 return 0;
1722 }
1723
1724 // There presumably was a change in the directory contents. Return the new
1725 // magic number
1726 dir_magic = new_magic;
1727
1728 // Look for all possible defined chart classes
1729 for (auto &cd : m_ChartClassDescriptorArray) {
1730 nAdd += SearchDirAndAddCharts(dir_info.fullpath, cd, pprog);
1731 }
1732
1733 return nAdd;
1734}
1735
1736bool ChartDatabase::DetectDirChange(const wxString &dir_path,
1737 const wxString &prog_label,
1738 const wxString &magic, wxString &new_magic,
1739 wxGenericProgressDialog *pprog) {
1740 if (pprog) pprog->SetTitle(_("OpenCPN Directory Scan...."));
1741
1742 // parse the magic number
1743 long long unsigned int nmagic;
1744 wxULongLong nacc = 0;
1745
1746 magic.ToULongLong(&nmagic, 10);
1747
1748 // Get an arraystring of all files
1749 wxArrayString FileList;
1750 wxDir dir(dir_path);
1751 int n_files = dir.GetAllFiles(dir_path, &FileList);
1752 FileList.Sort(); // Ensure persistent order of items being hashed.
1753
1754 FlexHash hash(sizeof nacc);
1755 hash.Reset();
1756
1757 // Traverse the list of files, getting their interesting stuff to add to
1758 // accumulator
1759 for (int ifile = 0; ifile < n_files; ifile++) {
1760 if (pprog && (ifile % (n_files / 60 + 1)) == 0)
1761 pprog->Update(wxMin((ifile * 100) / n_files, 100), prog_label);
1762
1763 wxFileName file(FileList[ifile]);
1764
1765 // NOTE. Do not ever try to optimize this code by combining `wxString`
1766 // calls. Otherwise `fileNameUTF8` will point to a stale buffer overwritten
1767 // by garbage.
1768 wxString fileNameNative = file.GetFullPath();
1769 wxScopedCharBuffer fileNameUTF8 = fileNameNative.ToUTF8();
1770 hash.Update(fileNameUTF8.data(), fileNameUTF8.length());
1771
1772 // File Size;
1773 wxULongLong size = file.GetSize();
1774 wxULongLong fileSize = ((size != wxInvalidSize) ? size : 0);
1775 hash.Update(&fileSize, (sizeof fileSize));
1776
1777 // Mod time, in ticks
1778 wxDateTime t = file.GetModificationTime();
1779 wxULongLong fileTime = t.GetTicks();
1780 hash.Update(&fileTime, (sizeof fileTime));
1781 }
1782
1783 hash.Finish();
1784 hash.Receive(&nacc);
1785
1786 // Return the calculated magic number
1787 new_magic = nacc.ToString();
1788
1789 // And do the test
1790 if (new_magic != magic)
1791 return true;
1792 else
1793 return false;
1794}
1795
1796bool ChartDatabase::IsChartDirUsed(const wxString &theDir) {
1797 wxString dir(theDir);
1798 if (dir.Last() == '/' || dir.Last() == wxFileName::GetPathSeparator())
1799 dir.RemoveLast();
1800
1801 dir.Append(wxT("*"));
1802 for (UINT32 i = 0; i < active_chartTable.GetCount(); i++) {
1803 if (active_chartTable[i].GetpsFullPath()->Matches(dir)) return true;
1804 }
1805 return false;
1806}
1807
1808//-----------------------------------------------------------------------------
1809// Validate a given directory as a cm93 root database
1810// If it appears to be a cm93 database, then return true
1811//-----------------------------------------------------------------------------
1812bool ChartDatabase::Check_CM93_Structure(wxString dir_name) {
1813 wxString filespec;
1814
1815 wxRegEx test(_T("[0-9]+"));
1816
1817 wxDir dirt(dir_name);
1818 wxString candidate;
1819
1820 if (dirt.IsOpened())
1821 wxLogMessage(_T("check_cm93 opened dir OK: ") + dir_name);
1822 else {
1823 wxLogMessage(_T("check_cm93 NOT OPENED OK: ") + dir_name);
1824 wxLogMessage(_T("check_cm93 returns false.") + dir_name);
1825 return false;
1826 }
1827
1828 bool b_maybe_found_cm93 = false;
1829 bool b_cont = dirt.GetFirst(&candidate);
1830
1831 while (b_cont) {
1832 if (test.Matches(candidate) && (candidate.Len() == 8)) {
1833 b_maybe_found_cm93 = true;
1834 break;
1835 }
1836
1837 b_cont = dirt.GetNext(&candidate);
1838 }
1839
1840 if (b_maybe_found_cm93) {
1841 wxString dir_next = dir_name;
1842 dir_next += _T("/");
1843 dir_next += candidate;
1844 if (wxDir::Exists(dir_next)) {
1845 wxDir dir_n(dir_next);
1846 if (dirt.IsOpened()) {
1847 wxString candidate_n;
1848
1849 wxRegEx test_n(_T("^[A-Ga-g]"));
1850 bool b_probably_found_cm93 = false;
1851 bool b_cont_n = dir_n.IsOpened() && dir_n.GetFirst(&candidate_n);
1852 while (b_cont_n) {
1853 if (test_n.Matches(candidate_n) && (candidate_n.Len() == 1)) {
1854 b_probably_found_cm93 = true;
1855 break;
1856 }
1857 b_cont_n = dir_n.GetNext(&candidate_n);
1858 }
1859
1860 if (b_probably_found_cm93) // found a directory that looks
1861 // like {dir_name}/12345678/A
1862 // probably cm93
1863 {
1864 // make sure the dir exists
1865 wxString dir_luk = dir_next;
1866 dir_luk += _T("/");
1867 dir_luk += candidate_n;
1868 if (wxDir::Exists(dir_luk)) return true;
1869 }
1870 }
1871 }
1872 }
1873
1874 return false;
1875}
1876
1877/*
1878//-----------------------------------------------------------------------------
1879// Validate a given directory as a cm93 root database
1880// If it appears to be a cm93 database, then return the name of an existing cell
1881file
1882// File name will be unique with respect to member element m_cm93_filename_array
1883// If not cm93, return empty string
1884//-----------------------------------------------------------------------------
1885wxString ChartDatabase::Get_CM93_FileName(wxString dir_name)
1886{
1887 wxString filespec;
1888
1889 wxRegEx test(_T("[0-9]+"));
1890
1891 wxDir dirt(dir_name);
1892 wxString candidate;
1893
1894 bool b_maybe_found_cm93 = false;
1895 bool b_cont = dirt.GetFirst(&candidate);
1896
1897 while(b_cont)
1898 {
1899 if(test.Matches(candidate)&& (candidate.Len() == 8))
1900 {
1901 b_maybe_found_cm93 = true;
1902 break;
1903 }
1904
1905 b_cont = dirt.GetNext(&candidate);
1906
1907 }
1908
1909 if(b_maybe_found_cm93)
1910 {
1911 wxString dir_next = dir_name;
1912 dir_next += _T("/");
1913 dir_next += candidate;
1914 if(wxDir::Exists(dir_next))
1915 {
1916 wxDir dir_n(dir_next);
1917 wxString candidate_n;
1918
1919 wxRegEx test_n(_T("^[A-Ga-g]"));
1920 bool b_probably_found_cm93 = false;
1921 bool b_cont_n = dir_n.GetFirst(&candidate_n);
1922 while(b_cont_n)
1923 {
1924 if(test_n.Matches(candidate_n) && (candidate_n.Len() ==
19251))
1926 {
1927 b_probably_found_cm93 = true;
1928 break;
1929 }
1930 b_cont_n = dir_n.GetNext(&candidate_n);
1931 }
1932
1933 if(b_probably_found_cm93) // found a directory that
1934looks like {dir_name}/12345678/A probably cm93 { // and we want to try and
1935shorten the recursive search
1936 // make sure the dir exists
1937 wxString dir_luk = dir_next;
1938 dir_luk += _T("/");
1939 dir_luk += candidate_n;
1940 if(wxDir::Exists(dir_luk))
1941 {
1942 wxString msg(_T("Found probable CM93 database in
1943")); msg += dir_name; wxLogMessage(msg);
1944
1945 wxString dir_name_plus = dir_luk; // be very
1946specific about the dir_name,
1947
1948 wxDir dir_get(dir_name_plus);
1949 wxString one_file;
1950 dir_get.GetFirst(&one_file);
1951
1952 // We must return a unique file name, i.e. one
1953that has not bee seen
1954 // before in this invocation of chart dir
1955scans. bool find_unique = false; while(!find_unique)
1956 {
1957 find_unique = true;
1958 for(unsigned int ifile=0; ifile <
1959m_cm93_filename_array.GetCount(); ifile++)
1960 {
1961 if(m_cm93_filename_array[ifile] ==
1962one_file) find_unique = false;
1963 }
1964 if(!find_unique)
1965 dir_get.GetNext(&one_file);
1966 }
1967
1968 m_cm93_filename_array.Add(one_file);
1969
1970 filespec = one_file;
1971 }
1972
1973 }
1974 }
1975 }
1976
1977 return filespec;
1978}
1979*/
1980
1981// ----------------------------------------------------------------------------
1982// Populate Chart Table by directory search for specified file type
1983// If bupdate flag is true, search the Chart Table for matching chart.
1984// if target chart is already in table, mark it valid and skip chart processing
1985// ----------------------------------------------------------------------------
1986
1987WX_DECLARE_STRING_HASH_MAP(int, ChartCollisionsHashMap);
1988
1989int ChartDatabase::SearchDirAndAddCharts(wxString &dir_name_base,
1990 ChartClassDescriptor &chart_desc,
1991 wxGenericProgressDialog *pprog) {
1992 wxString msg(_T("Searching directory: "));
1993 msg += dir_name_base;
1994 msg += _T(" for ");
1995 msg += chart_desc.m_search_mask;
1996 wxLogMessage(msg);
1997
1998 wxString dir_name = dir_name_base;
1999
2000#ifdef __OCPN__ANDROID__
2001 dir_name = wxString(dir_name_base.mb_str(wxConvUTF8)); // android
2002#endif
2003
2004 if (!wxDir::Exists(dir_name)) return 0;
2005
2006 wxString filespec = chart_desc.m_search_mask.Upper();
2007 wxString lowerFileSpec = chart_desc.m_search_mask.Lower();
2008 wxString filespecXZ = filespec + _T(".xz");
2009 wxString lowerFileSpecXZ = lowerFileSpec + _T(".xz");
2010 wxString filename;
2011
2012 // Count the files
2013 wxArrayString FileList;
2014 int gaf_flags = wxDIR_DEFAULT; // as default, recurse into subdirs
2015
2016 // Here is an optimization for MSW/cm93 especially
2017 // If this directory seems to be a cm93, and we are not explicitely looking
2018 // for cm93, then abort Otherwise, we will be looking thru entire cm93 tree
2019 // for non-existent .KAP files, etc.
2020
2021 bool b_found_cm93 = false;
2022 bool b_cm93 = Check_CM93_Structure(dir_name);
2023 if (b_cm93) {
2024 if (filespec != _T("00300000.A"))
2025 return false;
2026 else {
2027 filespec = dir_name;
2028 b_found_cm93 = true;
2029 }
2030 }
2031
2032 if (!b_found_cm93) {
2033 wxDir dir(dir_name);
2034 dir.GetAllFiles(dir_name, &FileList, filespec, gaf_flags);
2035
2036#ifdef __OCPN__ANDROID__
2037 if (!FileList.GetCount()) {
2038 wxArrayString afl = androidTraverseDir(dir_name, filespec);
2039 for (wxArrayString::const_iterator item = afl.begin(); item != afl.end();
2040 item++)
2041 FileList.Add(*item);
2042 }
2043#endif
2044
2045#ifndef __WXMSW__
2046 if (filespec != lowerFileSpec) {
2047 // add lowercase filespec files too
2048 wxArrayString lowerFileList;
2049 dir.GetAllFiles(dir_name, &lowerFileList, lowerFileSpec, gaf_flags);
2050
2051#ifdef __OCPN__ANDROID__
2052 if (!lowerFileList.GetCount()) {
2053 wxArrayString afl = androidTraverseDir(dir_name, lowerFileSpec);
2054 for (wxArrayString::const_iterator item = afl.begin();
2055 item != afl.end(); item++)
2056 lowerFileList.Add(*item);
2057 }
2058#endif
2059
2060 for (wxArrayString::const_iterator item = lowerFileList.begin();
2061 item != lowerFileList.end(); item++)
2062 FileList.Add(*item);
2063 }
2064#endif
2065
2066#ifdef OCPN_USE_LZMA
2067 // add xz compressed files;
2068 dir.GetAllFiles(dir_name, &FileList, filespecXZ, gaf_flags);
2069 dir.GetAllFiles(dir_name, &FileList, lowerFileSpecXZ, gaf_flags);
2070#endif
2071
2072 FileList.Sort(); // Sorted processing order makes the progress bar more
2073 // meaningful to the user.
2074 } else { // This is a cm93 dataset, specified as yada/yada/cm93
2075 wxString dir_plus = dir_name;
2076 dir_plus += wxFileName::GetPathSeparator();
2077 FileList.Add(dir_plus);
2078 }
2079
2080 int nFile = FileList.GetCount();
2081
2082 if (!nFile) return false;
2083
2084 int nDirEntry = 0;
2085
2086 // Check to see if there are any charts in the DB which refer to this
2087 // directory If none at all, there is no need to scan the DB for fullpath
2088 // match of each potential addition and bthis_dir_in_dB is false.
2089 bool bthis_dir_in_dB = IsChartDirUsed(dir_name);
2090
2091 if (pprog) pprog->SetTitle(_("OpenCPN Chart Add...."));
2092
2093 // build a hash table based on filename (without directory prefix) of
2094 // the chart to fast to detect identical charts
2095 ChartCollisionsHashMap collision_map;
2096 int nEntry = active_chartTable.GetCount();
2097 for (int i = 0; i < nEntry; i++) {
2098 wxString table_file_name = active_chartTable[i].GetFullSystemPath();
2099 wxFileName table_file(table_file_name);
2100 collision_map[table_file.GetFullName()] = i;
2101 }
2102
2103 int nFileProgressQuantum = wxMax(nFile / 100, 2);
2104 double rFileProgressRatio = 100.0 / wxMax(nFile, 1);
2105
2106 for (int ifile = 0; ifile < nFile; ifile++) {
2107 wxFileName file(FileList[ifile]);
2108 wxString full_name = file.GetFullPath();
2109 wxString file_name = file.GetFullName();
2110 wxString utf8_path = full_name;
2111
2112#ifdef __OCPN__ANDROID__
2113 // The full path (full_name) is the broken Android files system
2114 // interpretation, which does not display well onscreen. So, here we
2115 // reconstruct a full path spec in UTF-8 encoding for later use in string
2116 // displays. This utf-8 string will be used to construct the chart database
2117 // entry if required.
2118 wxFileName fnbase(dir_name_base);
2119 int nDirs = fnbase.GetDirCount();
2120
2121 wxFileName file_target(FileList[ifile]);
2122
2123 for (int i = 0; i < nDirs + 1;
2124 i++) // strip off the erroneous intial directories
2125 file_target.RemoveDir(0);
2126
2127 wxString leftover_path = file_target.GetFullPath();
2128 utf8_path =
2129 dir_name_base + leftover_path; // reconstruct a fully utf-8 version
2130#endif
2131
2132 // Validate the file name again, considering MSW's semi-random treatment
2133 // of case....
2134 // TODO...something fishy here - may need to normalize saved name?
2135 if (!file_name.Matches(lowerFileSpec) && !file_name.Matches(filespec) &&
2136 !file_name.Matches(lowerFileSpecXZ) && !file_name.Matches(filespecXZ) &&
2137 !b_found_cm93) {
2138 // wxLogMessage(_T("FileSpec test failed for:") + file_name);
2139 continue;
2140 }
2141
2142 if (pprog && ((ifile % nFileProgressQuantum) == 0))
2143 pprog->Update(static_cast<int>(ifile * rFileProgressRatio), utf8_path);
2144
2145 ChartTableEntry *pnewChart = NULL;
2146 bool bAddFinal = true;
2147 int b_add_msg = 0;
2148
2149 // Check the collisions map looking for duplicates, and choosing the right
2150 // one.
2151 ChartCollisionsHashMap::const_iterator collision_ptr =
2152 collision_map.find(file_name);
2153 bool collision = (collision_ptr != collision_map.end());
2154 bool file_path_is_same = false;
2155 bool file_time_is_same = false;
2156 ChartTableEntry *pEntry = NULL;
2157 wxString table_file_name;
2158
2159 // Allow multiple cm93 chart sets #4217
2160 if (b_found_cm93) collision = false;
2161
2162 if (collision) {
2163 pEntry = &active_chartTable[collision_ptr->second];
2164 table_file_name = pEntry->GetFullSystemPath();
2165 file_path_is_same =
2166 bthis_dir_in_dB && full_name.IsSameAs(table_file_name);
2167
2168 // If the chart full file paths are exactly the same, select the newer
2169 // one.
2170 if (file_path_is_same) {
2171 b_add_msg++;
2172
2173 // Check the file modification time
2174 time_t t_oldFile = pEntry->GetFileTime();
2175 time_t t_newFile = file.GetModificationTime().GetTicks();
2176
2177 if (t_newFile <= t_oldFile) {
2178 file_time_is_same = true;
2179 bAddFinal = false;
2180 pEntry->SetValid(true);
2181 } else {
2182 bAddFinal = true;
2183 pEntry->SetValid(false);
2184 }
2185 }
2186 }
2187
2188 wxString msg_fn(full_name);
2189 msg_fn.Replace(_T("%"), _T("%%"));
2190 if (file_time_is_same) {
2191 // Produce the same output without actually calling
2192 // `CreateChartTableEntry()`.
2193 wxLogMessage(
2194 wxString::Format(_T("Loading chart data for %s"), msg_fn.c_str()));
2195 } else {
2196 pnewChart = CreateChartTableEntry(full_name, utf8_path, chart_desc);
2197 if (!pnewChart) {
2198 bAddFinal = false;
2199 wxLogMessage(wxString::Format(
2200 _T(" CreateChartTableEntry() failed for file: %s"),
2201 msg_fn.c_str()));
2202 }
2203 }
2204
2205 if (!collision || !pnewChart) {
2206 // Do nothing.
2207 } else if (file_path_is_same) {
2208 wxLogMessage(
2209 wxString::Format(_T(" Replacing older chart file of same path: %s"),
2210 msg_fn.c_str()));
2211 } else if (!file_time_is_same) {
2212 // Look at the chart file name (without directory prefix) for a further
2213 // check for duplicates This catches the case in which the "same" chart
2214 // is in different locations, and one may be newer than the other.
2215 b_add_msg++;
2216
2217 if (pnewChart->IsEarlierThan(*pEntry)) {
2218 wxFileName table_file(table_file_name);
2219 // Make sure the compare file actually exists
2220 if (table_file.IsFileReadable()) {
2221 pEntry->SetValid(true);
2222 bAddFinal = false;
2223 wxLogMessage(wxString::Format(
2224 _T(" Retaining newer chart file of same name: %s"),
2225 msg_fn.c_str()));
2226 }
2227 } else if (pnewChart->IsEqualTo(*pEntry)) {
2228 // The file names (without dir prefix) are identical,
2229 // and the mod times are identical
2230 // Prsume that this is intentional, in order to facilitate
2231 // having the same chart in multiple groups.
2232 // So, add this chart.
2233 bAddFinal = true;
2234 } else {
2235 pEntry->SetValid(false);
2236 bAddFinal = true;
2237 wxLogMessage(wxString::Format(
2238 _T(" Replacing older chart file of same name: %s"),
2239 msg_fn.c_str()));
2240 }
2241 }
2242
2243 if (bAddFinal) {
2244 if (0 == b_add_msg) {
2245 wxLogMessage(
2246 wxString::Format(_T(" Adding chart file: %s"), msg_fn.c_str()));
2247 }
2248 collision_map[file_name] = active_chartTable.GetCount();
2249 active_chartTable.Add(pnewChart);
2250 nDirEntry++;
2251 } else {
2252 if (pnewChart) delete pnewChart;
2253 // wxLogMessage(wxString::Format(_T(" Not adding
2254 // chart file: %s"), msg_fn.c_str()));
2255 }
2256 }
2257
2258 m_nentries = active_chartTable.GetCount();
2259
2260 return nDirEntry;
2261}
2262
2263bool ChartDatabase::AddChart(wxString &chartfilename,
2264 ChartClassDescriptor &chart_desc,
2265 wxGenericProgressDialog *pprog, int isearch,
2266 bool bthis_dir_in_dB) {
2267 bool rv = false;
2268 wxFileName file(chartfilename);
2269 wxString full_name = file.GetFullPath();
2270 wxString file_name = file.GetFullName();
2271
2272 // Validate the file name again, considering MSW's semi-random treatment of
2273 // case....
2274 // TODO...something fishy here - may need to normalize saved name?
2275 // if(!file_name.Matches(lowerFileSpec) && !file_name.Matches(filespec) &&
2276 // !b_found_cm93)
2277 // continue;
2278
2279 if (pprog)
2280 pprog->Update(wxMin((m_pdifile * 100) / m_pdnFile, 100), full_name);
2281
2282 ChartTableEntry *pnewChart = NULL;
2283 bool bAddFinal = true;
2284 int b_add_msg = 0;
2285 wxString msg_fn(full_name);
2286 msg_fn.Replace(_T("%"), _T("%%"));
2287
2288 pnewChart = CreateChartTableEntry(full_name, full_name, chart_desc);
2289 if (!pnewChart) {
2290 bAddFinal = false;
2291 wxLogMessage(wxString::Format(
2292 _T(" CreateChartTableEntry() failed for file: %s"), msg_fn.c_str()));
2293 return false;
2294 } else // traverse the existing database looking for duplicates, and choosing
2295 // the right one
2296 {
2297 int nEntry = active_chartTable.GetCount();
2298 for (int i = 0; i < nEntry; i++) {
2299 wxString *ptable_file_name = active_chartTable[isearch].GetpsFullPath();
2300
2301 // If the chart full file paths are exactly the same, select the newer
2302 // one
2303 if (bthis_dir_in_dB && full_name.IsSameAs(*ptable_file_name)) {
2304 b_add_msg++;
2305
2306 // Check the file modification time
2307 time_t t_oldFile = active_chartTable[isearch].GetFileTime();
2308 time_t t_newFile = file.GetModificationTime().GetTicks();
2309
2310 if (t_newFile <= t_oldFile) {
2311 bAddFinal = false;
2312 active_chartTable[isearch].SetValid(true);
2313 } else {
2314 bAddFinal = true;
2315 active_chartTable[isearch].SetValid(false);
2316 wxLogMessage(wxString::Format(
2317 _T(" Replacing older chart file of same path: %s"),
2318 msg_fn.c_str()));
2319 }
2320
2321 break;
2322 }
2323
2324 // Look at the chart file name (without directory prefix) for a further
2325 // check for duplicates This catches the case in which the "same" chart
2326 // is in different locations, and one may be newer than the other.
2327 wxFileName table_file(*ptable_file_name);
2328
2329 if (table_file.GetFullName() == file_name) {
2330 b_add_msg++;
2331
2332 if (pnewChart->IsEarlierThan(active_chartTable[isearch])) {
2333 // Make sure the compare file actually exists
2334 if (table_file.IsFileReadable()) {
2335 active_chartTable[isearch].SetValid(true);
2336 bAddFinal = false;
2337 wxLogMessage(wxString::Format(
2338 _T(" Retaining newer chart file of same name: %s"),
2339 msg_fn.c_str()));
2340 }
2341 } else if (pnewChart->IsEqualTo(active_chartTable[isearch])) {
2342 // The file names (without dir prefix) are identical,
2343 // and the mod times are identical
2344 // Prsume that this is intentional, in order to facilitate
2345 // having the same chart in multiple groups.
2346 // So, add this chart.
2347 bAddFinal = true;
2348 }
2349
2350 else {
2351 active_chartTable[isearch].SetValid(false);
2352 bAddFinal = true;
2353 wxLogMessage(wxString::Format(
2354 _T(" Replacing older chart file of same name: %s"),
2355 msg_fn.c_str()));
2356 }
2357
2358 break;
2359 }
2360
2361 // TODO Look at the chart ID as a further check against duplicates
2362
2363 isearch++;
2364 if (nEntry == isearch) isearch = 0;
2365 } // for
2366 }
2367
2368 if (bAddFinal) {
2369 if (0 == b_add_msg) {
2370 wxLogMessage(
2371 wxString::Format(_T(" Adding chart file: %s"), msg_fn.c_str()));
2372 }
2373
2374 active_chartTable.Add(pnewChart);
2375
2376 rv = true;
2377 } else {
2378 delete pnewChart;
2379 // wxLogMessage(wxString::Format(_T(" Not adding chart
2380 // file: %s"), msg_fn.c_str()));
2381 rv = false;
2382 }
2383
2384 m_nentries = active_chartTable.GetCount();
2385
2386 return rv;
2387}
2388
2389bool ChartDatabase::AddSingleChart(wxString &ChartFullPath,
2390 bool b_force_full_search) {
2391 // Find a relevant chart class descriptor
2392 wxFileName fn(ChartFullPath);
2393 wxString ext = fn.GetExt();
2394 ext.Prepend(_T("*."));
2395 wxString ext_upper = ext.MakeUpper();
2396 wxString ext_lower = ext.MakeLower();
2397 wxString dir_name = fn.GetPath();
2398
2399 // Search the array of chart class descriptors to find a match
2400 // bewteen the search mask and the the chart file extension
2401
2403 for (auto &cd : m_ChartClassDescriptorArray) {
2404 if (cd.m_descriptor_type == PLUGIN_DESCRIPTOR) {
2405 if (cd.m_search_mask == ext_upper) {
2406 desc = cd;
2407 break;
2408 }
2409 if (cd.m_search_mask == ext_lower) {
2410 desc = cd;
2411 break;
2412 }
2413 }
2414 }
2415
2416 // If we know that we need to do a full recursive search of the db,
2417 // then there is no need to verify it by doing a directory match
2418 bool b_recurse = true;
2419 if (!b_force_full_search) b_recurse = IsChartDirUsed(dir_name);
2420
2421 bool rv = AddChart(ChartFullPath, desc, NULL, 0, b_recurse);
2422
2423 // remove duplicates marked in AddChart()
2424
2425 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++) {
2426 if (!active_chartTable[i].GetbValid()) {
2427 active_chartTable.RemoveAt(i);
2428 i--; // entry is gone, recheck this index for next entry
2429 }
2430 }
2431
2432 // Update the Entry index fields
2433 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++)
2434 active_chartTable[i].SetEntryOffset(i);
2435
2436 // Get a new magic number
2437 wxString new_magic;
2438 DetectDirChange(dir_name, _T(""), _T(""), new_magic, 0);
2439
2440 // Update (clone) the CDI array
2441 bool bcfound = false;
2442 ArrayOfCDI NewChartDirArray;
2443
2444 ArrayOfCDI ChartDirArray = GetChartDirArray();
2445 for (unsigned int i = 0; i < ChartDirArray.GetCount(); i++) {
2446 ChartDirInfo cdi = ChartDirArray[i];
2447
2448 ChartDirInfo newcdi = cdi;
2449
2450 // If entry is found that matches this cell, clear the magic number.
2451 if (newcdi.fullpath == dir_name) {
2452 newcdi.magic_number = new_magic;
2453 bcfound = true;
2454 }
2455
2456 NewChartDirArray.Add(newcdi);
2457 }
2458
2459 if (!bcfound) {
2460 ChartDirInfo cdi;
2461 cdi.fullpath = dir_name;
2462 cdi.magic_number = new_magic;
2463 NewChartDirArray.Add(cdi);
2464 }
2465
2466 // Update the database master copy of the CDI array
2467 SetChartDirArray(NewChartDirArray);
2468
2469 // Update the list of chart dirs.
2470 m_chartDirs.Clear();
2471
2472 for (unsigned int i = 0; i < GetChartDirArray().GetCount(); i++) {
2473 ChartDirInfo cdi = GetChartDirArray()[i];
2474 m_chartDirs.Add(cdi.fullpath);
2475 }
2476
2477 m_nentries = active_chartTable.GetCount();
2478
2479 return rv;
2480}
2481
2482bool ChartDatabase::RemoveSingleChart(wxString &ChartFullPath) {
2483 bool rv = false;
2484
2485 // Walk the chart table, looking for the target
2486 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++) {
2487 if (ChartFullPath.IsSameAs(GetChartTableEntry(i).GetFullSystemPath())) {
2488 active_chartTable.RemoveAt(i);
2489 break;
2490 }
2491 }
2492
2493 // Update the EntryOffset fields for the array
2494 ChartTableEntry *pcte;
2495
2496 for (unsigned int i = 0; i < active_chartTable.GetCount(); i++) {
2497 pcte = GetpChartTableEntry(i);
2498 pcte->SetEntryOffset(i);
2499 }
2500
2501 // Check and update the dir array
2502 wxFileName fn(ChartFullPath);
2503 wxString fd = fn.GetPath();
2504 if (!IsChartDirUsed(fd)) {
2505 // Clone a new array, removing the unused directory,
2506 ArrayOfCDI NewChartDirArray;
2507
2508 ArrayOfCDI ChartDirArray = GetChartDirArray();
2509 for (unsigned int i = 0; i < ChartDirArray.GetCount(); i++) {
2510 ChartDirInfo cdi = ChartDirArray[i];
2511
2512 ChartDirInfo newcdi = cdi;
2513
2514 if (newcdi.fullpath != fd) NewChartDirArray.Add(newcdi);
2515 }
2516
2517 SetChartDirArray(NewChartDirArray);
2518 }
2519
2520 // Update the list of chart dirs.
2521 m_chartDirs.Clear();
2522 for (unsigned int i = 0; i < GetChartDirArray().GetCount(); i++) {
2523 ChartDirInfo cdi = GetChartDirArray()[i];
2524 m_chartDirs.Add(cdi.fullpath);
2525 }
2526
2527 m_nentries = active_chartTable.GetCount();
2528
2529 return rv;
2530}
2531
2533// Create a Chart object
2535
2536ChartBase *ChartDatabase::GetChart(const wxChar *theFilePath,
2537 ChartClassDescriptor &chart_desc) const {
2538 // TODO: support non-UI chart factory
2539 return NULL;
2540}
2541
2543// Create Chart Table entry by reading chart header info, etc.
2545
2546ChartTableEntry *ChartDatabase::CreateChartTableEntry(
2547 const wxString &filePath, wxString &utf8Path,
2548 ChartClassDescriptor &chart_desc) {
2549 wxString msg_fn(filePath);
2550 msg_fn.Replace(_T("%"), _T("%%"));
2551 wxLogMessage(
2552 wxString::Format(_T("Loading chart data for %s"), msg_fn.c_str()));
2553
2554 ChartBase *pch = GetChart(filePath, chart_desc);
2555 if (pch == NULL) {
2556 wxLogMessage(
2557 wxString::Format(_T(" ...creation failed for %s"), msg_fn.c_str()));
2558 return NULL;
2559 }
2560
2561 InitReturn rc = pch->Init(filePath, HEADER_ONLY);
2562 if (rc != INIT_OK) {
2563 delete pch;
2564 wxLogMessage(wxString::Format(_T(" ...initialization failed for %s"),
2565 msg_fn.c_str()));
2566 return NULL;
2567 }
2568
2569 ChartTableEntry *ret_val = new ChartTableEntry(*pch, utf8Path);
2570 ret_val->SetValid(true);
2571
2572 delete pch;
2573
2574 return ret_val;
2575}
2576
2577bool ChartDatabase::GetCentroidOfLargestScaleChart(double *clat, double *clon,
2578 ChartFamilyEnum family) {
2579 int cur_max_i = -1;
2580 int cur_max_scale = 0;
2581
2582 int nEntry = active_chartTable.GetCount();
2583
2584 for (int i = 0; i < nEntry; i++) {
2585 if (GetChartFamily(active_chartTable[i].GetChartType()) == family) {
2586 if (active_chartTable[i].GetScale() > cur_max_scale) {
2587 cur_max_scale = active_chartTable[i].GetScale();
2588 cur_max_i = i;
2589 }
2590 }
2591 }
2592
2593 if (cur_max_i == -1)
2594 return false; // nothing found
2595 else {
2596 *clat = (active_chartTable[cur_max_i].GetLatMax() +
2597 active_chartTable[cur_max_i].GetLatMin()) /
2598 2.;
2599 *clon = (active_chartTable[cur_max_i].GetLonMin() +
2600 active_chartTable[cur_max_i].GetLonMax()) /
2601 2.;
2602 }
2603 return true;
2604}
2605
2606//-------------------------------------------------------------------
2607// Get DBChart Projection
2608//-------------------------------------------------------------------
2609int ChartDatabase::GetDBChartProj(int dbIndex) {
2610 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size()))
2611 return active_chartTable[dbIndex].GetChartProjectionType();
2612 else
2613 return PROJECTION_UNKNOWN;
2614}
2615
2616//-------------------------------------------------------------------
2617// Get DBChart Family
2618//-------------------------------------------------------------------
2619int ChartDatabase::GetDBChartFamily(int dbIndex) {
2620 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size()))
2621 return active_chartTable[dbIndex].GetChartFamily();
2622 else
2623 return CHART_FAMILY_UNKNOWN;
2624}
2625
2626//-------------------------------------------------------------------
2627// Get DBChart FullFileName
2628//-------------------------------------------------------------------
2629wxString ChartDatabase::GetDBChartFileName(int dbIndex) {
2630 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2631 return wxString(active_chartTable[dbIndex].GetFullSystemPath());
2632 } else
2633 return _T("");
2634}
2635
2636//-------------------------------------------------------------------
2637// Get DBChart Type
2638//-------------------------------------------------------------------
2639int ChartDatabase::GetDBChartType(int dbIndex) {
2640 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size()))
2641 return active_chartTable[dbIndex].GetChartType();
2642 else
2643 return 0;
2644}
2645
2646//-------------------------------------------------------------------
2647// Get DBChart Skew
2648//-------------------------------------------------------------------
2649float ChartDatabase::GetDBChartSkew(int dbIndex) {
2650 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size()))
2651 return active_chartTable[dbIndex].GetChartSkew();
2652 else
2653 return 0.;
2654}
2655
2656//-------------------------------------------------------------------
2657// Get DBChart Scale
2658//-------------------------------------------------------------------
2659int ChartDatabase::GetDBChartScale(int dbIndex) {
2660 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size()))
2661 return active_chartTable[dbIndex].GetScale();
2662 else
2663 return 1;
2664}
2665
2666//-------------------------------------------------------------------
2667// Get Lat/Lon Bounding Box from db
2668//-------------------------------------------------------------------
2669bool ChartDatabase::GetDBBoundingBox(int dbIndex, LLBBox &box) {
2670 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2671 const ChartTableEntry &entry = GetChartTableEntry(dbIndex);
2672 box.Set(entry.GetLatMin(), entry.GetLonMin(), entry.GetLatMax(),
2673 entry.GetLonMax());
2674 }
2675
2676 return true;
2677}
2678
2679const LLBBox &ChartDatabase::GetDBBoundingBox(int dbIndex) {
2680 if ((bValid) && (dbIndex >= 0)) {
2681 const ChartTableEntry &entry = GetChartTableEntry(dbIndex);
2682 return entry.GetBBox();
2683 } else {
2684 return m_dummy_bbox;
2685 }
2686}
2687
2688//-------------------------------------------------------------------
2689// Get PlyPoint from Database
2690//-------------------------------------------------------------------
2691int ChartDatabase::GetDBPlyPoint(int dbIndex, int plyindex, float *lat,
2692 float *lon) {
2693 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2694 const ChartTableEntry &entry = GetChartTableEntry(dbIndex);
2695 if (entry.GetnPlyEntries()) {
2696 float *fp = entry.GetpPlyTable();
2697 fp += plyindex * 2;
2698 if (lat) *lat = *fp;
2699 fp++;
2700 if (lon) *lon = *fp;
2701 }
2702 return entry.GetnPlyEntries();
2703 } else
2704 return 0;
2705}
2706
2707//-------------------------------------------------------------------
2708// Get AuxPlyPoint from Database
2709//-------------------------------------------------------------------
2710int ChartDatabase::GetDBAuxPlyPoint(int dbIndex, int plyindex, int ply,
2711 float *lat, float *lon) {
2712 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2713 const ChartTableEntry &entry = GetChartTableEntry(dbIndex);
2714 if (entry.GetnAuxPlyEntries()) {
2715 float *fp = entry.GetpAuxPlyTableEntry(ply);
2716
2717 fp += plyindex * 2;
2718 if (lat) *lat = *fp;
2719 fp++;
2720 if (lon) *lon = *fp;
2721 }
2722
2723 return entry.GetAuxCntTableEntry(ply);
2724 } else
2725 return 0;
2726}
2727
2728int ChartDatabase::GetnAuxPlyEntries(int dbIndex) {
2729 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2730 const ChartTableEntry &entry = GetChartTableEntry(dbIndex);
2731 return entry.GetnAuxPlyEntries();
2732 } else
2733 return 0;
2734}
2735
2736//-------------------------------------------------------------------
2737// Get vector of reduced Plypoints
2738//-------------------------------------------------------------------
2739std::vector<float> ChartDatabase::GetReducedPlyPoints(int dbIndex) {
2740 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2741 ChartTableEntry *pentry = GetpChartTableEntry(dbIndex);
2742 if (pentry) return pentry->GetReducedPlyPoints();
2743 }
2744
2745 std::vector<float> dummy;
2746 return dummy;
2747}
2748
2749//-------------------------------------------------------------------
2750// Get vector of reduced AuxPlypoints
2751//-------------------------------------------------------------------
2752std::vector<float> ChartDatabase::GetReducedAuxPlyPoints(int dbIndex,
2753 int iTable) {
2754 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2755 ChartTableEntry *pentry = GetpChartTableEntry(dbIndex);
2756 if (pentry) return pentry->GetReducedAuxPlyPoints(iTable);
2757 }
2758
2759 std::vector<float> dummy;
2760 return dummy;
2761}
2762
2763bool ChartDatabase::IsChartAvailable(int dbIndex) {
2764 if ((bValid) && (dbIndex >= 0) && (dbIndex < (int)active_chartTable.size())) {
2765 ChartTableEntry *pentry = GetpChartTableEntry(dbIndex);
2766
2767 // If not PLugIn chart, assume always available
2768 if (pentry->GetChartType() != CHART_TYPE_PLUGIN) return true;
2769
2770 wxString *path = pentry->GetpsFullPath();
2771 wxFileName fn(*path);
2772 wxString ext = fn.GetExt();
2773 ext.Prepend(_T("*."));
2774 wxString ext_upper = ext.MakeUpper();
2775 wxString ext_lower = ext.MakeLower();
2776
2777 // Search the array of chart class descriptors to find a match
2778 // between the search mask and the the chart file extension
2779
2780 for (auto &cd : m_ChartClassDescriptorArray) {
2781 if (cd.m_descriptor_type == PLUGIN_DESCRIPTOR) {
2782 wxString search_mask = cd.m_search_mask;
2783
2784 if (search_mask == ext_upper) {
2785 return true;
2786 }
2787 if (search_mask == ext_lower) {
2788 return true;
2789 }
2790 if (path->Matches(search_mask)) {
2791 return true;
2792 }
2793 }
2794 }
2795 }
2796
2797 return false;
2798}
2799
2800void ChartDatabase::ApplyGroupArray(ChartGroupArray *pGroupArray) {
2801 wxString separator(wxFileName::GetPathSeparator());
2802
2803 for (unsigned int ic = 0; ic < active_chartTable.GetCount(); ic++) {
2804 ChartTableEntry *pcte = &active_chartTable[ic];
2805
2806 pcte->ClearGroupArray();
2807
2808 wxString *chart_full_path = pcte->GetpsFullPath();
2809
2810 for (unsigned int igroup = 0; igroup < pGroupArray->GetCount(); igroup++) {
2811 ChartGroup *pGroup = pGroupArray->Item(igroup);
2812 for (const auto &elem : pGroup->m_element_array) {
2813 wxString element_root = elem.m_element_name;
2814
2815 // The element may be a full single chart name
2816 // If so, add it
2817 // Otherwise, append a sep character so that similar paths are
2818 // distinguished. See FS#1060
2819 if (!chart_full_path->IsSameAs(element_root))
2820 element_root.Append(
2821 separator); // Prevent comingling similar looking path names
2822 if (chart_full_path->StartsWith(element_root)) {
2823 bool b_add = true;
2824 for (unsigned int k = 0; k < elem.m_missing_name_array.size(); k++) {
2825 const wxString &missing_item = elem.m_missing_name_array[k];
2826 if (chart_full_path->StartsWith(missing_item)) {
2827 if (chart_full_path->IsSameAs(
2828 missing_item)) // missing item is full chart name
2829 {
2830 b_add = false;
2831 break;
2832 } else {
2833 if (wxDir::Exists(missing_item)) // missing item is a dir
2834 {
2835 b_add = false;
2836 break;
2837 }
2838 }
2839 }
2840 }
2841
2842 if (b_add) pcte->AddIntToGroupArray(igroup + 1);
2843 }
2844 }
2845 }
2846 }
2847}
Base class for all chart types.
Definition chartbase.h:119
Manages a database of charts, including reading, writing, and querying chart information.
Definition chartdbs.h:308
bool Create(ArrayOfCDI &dir_array, wxGenericProgressDialog *pprog)
Creates a new chart database from a list of directories.
bool Update(ArrayOfCDI &dir_array, bool bForce, wxGenericProgressDialog *pprog)
Updates the chart database.
Represents a user-defined collection of logically related charts.
Definition chartdbs.h:464
Wrapper class for plugin-based charts.
Definition chartimg.h:392
A class for computing hash of arbitrary length.
Definition FlexHash.h:38
Manages a set of ShapeBaseChart objects at different resolutions.
Represents an entry in the chart table, containing information about a single chart.
Definition chartdbs.h:181