33#include <wx/arrimpl.cpp>
35#include <wx/encconv.h>
36#include <wx/progdlg.h>
38#include <wx/tokenzr.h>
39#include <wx/evtloop.h>
46#include "LOD_reduce.h"
54#define UINT32 unsigned int
58#include "androidUTIL.h"
63static int s_dbVersion;
69static ChartFamilyEnum GetChartFamily(
int charttype) {
74 cf = CHART_FAMILY_RASTER;
77 cf = CHART_FAMILY_RASTER;
80 cf = CHART_FAMILY_VECTOR;
83 cf = CHART_FAMILY_VECTOR;
85 case CHART_TYPE_CM93COMP:
86 cf = CHART_FAMILY_VECTOR;
88 case CHART_TYPE_DUMMY:
89 cf = CHART_FAMILY_RASTER;
91 case CHART_TYPE_UNKNOWN:
92 cf = CHART_FAMILY_UNKNOWN;
95 cf = CHART_FAMILY_UNKNOWN;
105void ChartTableHeader::Read(wxInputStream &is) {
109void ChartTableHeader::Write(wxOutputStream &os) {
111 sprintf(vb,
"V%03d", DB_VERSION_CURRENT);
113 memcpy(dbVersion, vb, 4);
117bool ChartTableHeader::CheckValid() {
119 sprintf(vb,
"V%03d", DB_VERSION_CURRENT);
120 if (strncmp(vb, dbVersion,
sizeof(dbVersion))) {
123 memcpy(vbo, dbVersion, 4);
125 msg.Append(wxString(vbo, wxConvUTF8));
126 msg.Prepend(
" Warning: found incorrect chart db version: ");
132 sprintf(vb,
"V%03d", DB_VERSION_PREVIOUS);
133 if (strncmp(vb, dbVersion,
sizeof(dbVersion)))
137 " Scheduling db upgrade to current db version on "
138 "Options->Charts page visit...");
145 memcpy(vbo, dbVersion, 4);
147 msg.Append(wxString(vbo, wxConvUTF8));
148 msg.Prepend(
"Loading chart db version: ");
159void ChartTableEntry::SetScale(
int scale) {
163 if (Scale >= 1000) rounding = 5 * pow(10, log10(Scale) - 2);
166ChartTableEntry::ChartTableEntry(
ChartBase &theChart, wxString &utf8Path) {
169 char *pt = (
char *)malloc(strlen(utf8Path.mb_str(wxConvUTF8)) + 1);
170 strcpy(pt, utf8Path.mb_str(wxConvUTF8));
173 SetScale(theChart.GetNativeScale());
175 ChartType = theChart.GetChartType();
176 ChartFamily = theChart.GetChartFamily();
178 Skew = theChart.GetChartSkew();
179 ProjectionType = theChart.GetChartProjectionType();
181 wxDateTime ed = theChart.GetEditionDate();
182 if (theChart.GetEditionDate().IsValid())
183 edition_date = theChart.GetEditionDate().GetTicks();
185 wxFileName fn(theChart.GetFullPath());
186 if (fn.GetModificationTime().IsValid())
187 file_date = fn.GetModificationTime().GetTicks();
189 m_pfilename =
new wxString;
190 *m_pfilename = fn.GetFullName();
191 m_psFullPath =
new wxString;
192 *m_psFullPath = utf8Path;
193 m_fullSystemPath = utf8Path;
194 m_FullPath = std::string(pFullPath);
197 m_fullSystemPath = wxString(utf8Path.mb_str(wxConvUTF8));
201 theChart.GetChartExtent(&ext);
207 m_bbox.Set(LatMin, LonMin, LatMax, LonMax);
212 double scale_max_zoom = Scale / 4;
214 double display_ppm = 1 / .00025;
215 double meters_per_pixel_max_scale = scale_max_zoom / display_ppm;
216 double LOD_meters = meters_per_pixel_max_scale * LOD_pixels;
221 if (theChart.GetCOVREntries() == 1) {
222 nPlyEntries = theChart.GetCOVRTablePoints(0);
224 if (nPlyEntries > 5 && (LOD_meters > .01)) {
225 std::vector<int> index_keep{0, nPlyEntries - 1, 1, nPlyEntries - 2};
227 double *DPbuffer = (
double *)malloc(2 * nPlyEntries *
sizeof(
double));
229 double *pfed = DPbuffer;
232 for (
int i = 0; i < nPlyEntries; i++) {
238 DouglasPeucker(DPbuffer, 1, nPlyEntries - 2, LOD_meters / (1852 * 60),
244 for (
unsigned int i = 0; i < index_keep.size(); i++) {
245 DPbuffer[2 * index_keep[i]] += 2000.;
248 float *pf = (
float *)malloc(2 * index_keep.size() *
sizeof(float));
251 for (
int i = 0; i < nPlyEntries; i++) {
252 if (DPbuffer[2 * i] > 1000.) {
253 *pfe++ = DPbuffer[2 * i] - 2000.;
254 *pfe++ = DPbuffer[(2 * i) + 1];
259 nPlyEntries = index_keep.size();
262 float *pf = (
float *)malloc(2 * nPlyEntries *
sizeof(
float));
267 for (
int i = 0; i < nPlyEntries; i++) {
279 float *pf1 = (
float *)malloc(2 * 4 *
sizeof(
float));
283 theChart.GetChartExtent(&fext);
299 nAuxPlyEntries = theChart.GetCOVREntries();
300 wxASSERT(nAuxPlyEntries);
301 float **pfp = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
303 int *pip = (
int *)malloc(nAuxPlyEntries *
sizeof(
int));
305 for (
int j = 0; j < nAuxPlyEntries; j++) {
306 int nPE = theChart.GetCOVRTablePoints(j);
308 if (nPE > 5 && (LOD_meters > .01)) {
309 std::vector<int> index_keep{0, nPE - 1, 1, nPE - 2};
311 double *DPbuffer = (
double *)malloc(2 * nPE *
sizeof(
double));
313 double *pfed = DPbuffer;
316 for (
int i = 0; i < nPE; i++) {
322 DouglasPeucker(DPbuffer, 1, nPE - 2, LOD_meters / (1852 * 60),
328 for (
unsigned int i = 0; i < index_keep.size(); i++) {
329 DPbuffer[2 * index_keep[i]] += 2000.;
332 float *pf = (
float *)malloc(2 * index_keep.size() *
sizeof(float));
335 for (
int i = 0; i < nPE; i++) {
336 if (DPbuffer[2 * i] > 1000.) {
337 *pfe1++ = DPbuffer[2 * i] - 2000.;
338 *pfe1++ = DPbuffer[(2 * i) + 1];
343 pip[j] = index_keep.size();
347 (
float *)malloc(theChart.GetCOVRTablePoints(j) * 2 *
sizeof(float));
348 memcpy(pf_entry, theChart.GetCOVRTableHead(j),
349 theChart.GetCOVRTablePoints(j) * 2 *
sizeof(
float));
351 pip[j] = theChart.GetCOVRTablePoints(j);
361 nNoCovrPlyEntries = theChart.GetNoCOVREntries();
362 if (nNoCovrPlyEntries == 0)
return;
364 float **pfpnc = (
float **)malloc(nNoCovrPlyEntries *
sizeof(
float *));
365 float **pft0nc = pfpnc;
366 int *pipnc = (
int *)malloc(nNoCovrPlyEntries *
sizeof(
int));
368 for (
int j = 0; j < nNoCovrPlyEntries; j++) {
370 (
float *)malloc(theChart.GetNoCOVRTablePoints(j) * 2 *
sizeof(float));
371 memcpy(pf_entry, theChart.GetNoCOVRTableHead(j),
372 theChart.GetNoCOVRTablePoints(j) * 2 *
sizeof(
float));
373 pft0nc[j] = pf_entry;
374 pipnc[j] = theChart.GetNoCOVRTablePoints(j);
377 pNoCovrPlyTable = pfpnc;
378 pNoCovrCntTable = pipnc;
383ChartTableEntry::~ChartTableEntry() {
387 for (
int i = 0; i < nAuxPlyEntries; i++) free(pAuxPlyTable[i]);
391 if (nNoCovrPlyEntries) {
392 for (
int i = 0; i < nNoCovrPlyEntries; i++) free(pNoCovrPlyTable[i]);
393 free(pNoCovrPlyTable);
394 free(pNoCovrCntTable);
406 wxDateTime mine(edition_date);
407 wxDateTime theirs(cte.edition_date);
409 if (!mine.IsValid() || !theirs.IsValid())
412 return (mine.IsEarlierThan(theirs));
416 wxDateTime mine(edition_date);
417 wxDateTime theirs(cte.edition_date);
419 if (!mine.IsValid() || !theirs.IsValid())
422 return (mine.IsEqualTo(theirs));
427static int convertChartType(
int charttype) {
430 if (s_dbVersion == 14) {
433 return CHART_TYPE_KAP;
435 return CHART_TYPE_GEO;
437 return CHART_TYPE_S57;
439 return CHART_TYPE_CM93;
441 return CHART_TYPE_CM93COMP;
443 return CHART_TYPE_UNKNOWN;
445 return CHART_TYPE_DONTCARE;
447 return CHART_TYPE_DUMMY;
449 return CHART_TYPE_UNKNOWN;
455static int convertChartFamily(
int charttype,
int chartfamily) {
456 if (s_dbVersion < 18) {
460 return CHART_FAMILY_RASTER;
463 case CHART_TYPE_CM93:
464 case CHART_TYPE_CM93COMP:
465 return CHART_FAMILY_VECTOR;
468 return CHART_FAMILY_UNKNOWN;
474bool ChartTableEntry::Read(
const ChartDatabase *pDb, wxInputStream &is) {
475 char path[4096], *cp;
481 int db_version = pD->GetVersion();
483 if (db_version == 18) {
485 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
486 pFullPath = (
char *)malloc(cp - path + 1);
487 strncpy(pFullPath, path, cp - path + 1);
488 wxLogVerbose(
" Chart %s", pFullPath);
491 m_pfilename =
new wxString;
492 wxString fullfilename(pFullPath, wxConvUTF8);
493 wxFileName fn(fullfilename);
494 *m_pfilename = fn.GetFullName();
495 m_psFullPath =
new wxString;
496 *m_psFullPath = fullfilename;
497 m_fullSystemPath = fullfilename;
498 m_FullPath = std::string(pFullPath);
501 m_fullSystemPath = wxString(fullfilename.mb_str(wxConvUTF8));
508 EntryOffset = cte.EntryOffset;
509 ChartType = cte.ChartType;
510 ChartFamily = cte.ChartFamily;
516 m_bbox.Set(LatMin, LonMin, LatMax, LonMax);
519 ProjectionType = cte.ProjectionType;
522 edition_date = cte.edition_date;
523 file_date = cte.file_date;
525 nPlyEntries = cte.nPlyEntries;
526 nAuxPlyEntries = cte.nAuxPlyEntries;
528 nNoCovrPlyEntries = cte.nNoCovrPlyEntries;
533 int npeSize = nPlyEntries * 2 *
sizeof(float);
534 pPlyTable = (
float *)malloc(npeSize);
535 is.Read(pPlyTable, npeSize);
538 if (nAuxPlyEntries) {
539 int napeSize = nAuxPlyEntries *
sizeof(int);
540 pAuxPlyTable = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
541 pAuxCntTable = (
int *)malloc(napeSize);
542 is.Read(pAuxCntTable, napeSize);
544 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
546 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
547 pAuxPlyTable[nAuxPlyEntry] = (
float *)malloc(nfSize);
548 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
552 if (nNoCovrPlyEntries) {
553 int napeSize = nNoCovrPlyEntries *
sizeof(int);
554 pNoCovrCntTable = (
int *)malloc(napeSize);
555 is.Read(pNoCovrCntTable, napeSize);
557 pNoCovrPlyTable = (
float **)malloc(nNoCovrPlyEntries *
sizeof(
float *));
558 for (
int i = 0; i < nNoCovrPlyEntries; i++) {
559 int nfSize = pNoCovrCntTable[i] * 2 *
sizeof(float);
560 pNoCovrPlyTable[i] = (
float *)malloc(nfSize);
561 is.Read(pNoCovrPlyTable[i], nfSize);
566 else if (db_version == 17) {
568 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
569 pFullPath = (
char *)malloc(cp - path + 1);
570 strncpy(pFullPath, path, cp - path + 1);
571 wxLogVerbose(
" Chart %s", pFullPath);
574 m_pfilename =
new wxString;
575 wxString fullfilename(pFullPath, wxConvUTF8);
576 wxFileName fn(fullfilename);
577 *m_pfilename = fn.GetFullName();
578 m_psFullPath =
new wxString;
579 *m_psFullPath = fullfilename;
580 m_FullPath = std::string(pFullPath);
587 EntryOffset = cte.EntryOffset;
588 ChartType = cte.ChartType;
594 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
597 ProjectionType = cte.ProjectionType;
600 edition_date = cte.edition_date;
601 file_date = cte.file_date;
603 nPlyEntries = cte.nPlyEntries;
604 nAuxPlyEntries = cte.nAuxPlyEntries;
606 nNoCovrPlyEntries = cte.nNoCovrPlyEntries;
611 int npeSize = nPlyEntries * 2 *
sizeof(float);
612 pPlyTable = (
float *)malloc(npeSize);
613 is.Read(pPlyTable, npeSize);
616 if (nAuxPlyEntries) {
617 int napeSize = nAuxPlyEntries *
sizeof(int);
618 pAuxPlyTable = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
619 pAuxCntTable = (
int *)malloc(napeSize);
620 is.Read(pAuxCntTable, napeSize);
622 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
624 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
625 pAuxPlyTable[nAuxPlyEntry] = (
float *)malloc(nfSize);
626 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
630 if (nNoCovrPlyEntries) {
631 int napeSize = nNoCovrPlyEntries *
sizeof(int);
632 pNoCovrCntTable = (
int *)malloc(napeSize);
633 is.Read(pNoCovrCntTable, napeSize);
635 pNoCovrPlyTable = (
float **)malloc(nNoCovrPlyEntries *
sizeof(
float *));
636 for (
int i = 0; i < nNoCovrPlyEntries; i++) {
637 int nfSize = pNoCovrCntTable[i] * 2 *
sizeof(float);
638 pNoCovrPlyTable[i] = (
float *)malloc(nfSize);
639 is.Read(pNoCovrPlyTable[i], nfSize);
644 else if (db_version == 16) {
646 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
648 pFullPath = (
char *)malloc(cp - path + 1);
649 strncpy(pFullPath, path, cp - path + 1);
650 wxLogVerbose(
" Chart %s", pFullPath);
653 m_pfilename =
new wxString;
654 wxString fullfilename(pFullPath, wxConvUTF8);
655 wxFileName fn(fullfilename);
656 *m_pfilename = fn.GetFullName();
657 m_psFullPath =
new wxString;
658 *m_psFullPath = fullfilename;
659 m_FullPath = std::string(pFullPath);
666 EntryOffset = cte.EntryOffset;
667 ChartType = cte.ChartType;
673 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
676 ProjectionType = cte.ProjectionType;
679 edition_date = cte.edition_date;
680 file_date = cte.file_date;
682 nPlyEntries = cte.nPlyEntries;
683 nAuxPlyEntries = cte.nAuxPlyEntries;
688 int npeSize = nPlyEntries * 2 *
sizeof(float);
689 pPlyTable = (
float *)malloc(npeSize);
690 is.Read(pPlyTable, npeSize);
693 if (nAuxPlyEntries) {
694 int napeSize = nAuxPlyEntries *
sizeof(int);
695 pAuxPlyTable = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
696 pAuxCntTable = (
int *)malloc(napeSize);
697 is.Read(pAuxCntTable, napeSize);
699 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
701 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
702 pAuxPlyTable[nAuxPlyEntry] = (
float *)malloc(nfSize);
703 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
708 else if (db_version == 15) {
710 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
712 pFullPath = (
char *)malloc(cp - path + 1);
713 strncpy(pFullPath, path, cp - path + 1);
714 wxLogVerbose(
" Chart %s", pFullPath);
721 EntryOffset = cte.EntryOffset;
722 ChartType = cte.ChartType;
728 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
731 edition_date = cte.edition_date;
732 file_date = cte.file_date;
734 nPlyEntries = cte.nPlyEntries;
735 nAuxPlyEntries = cte.nAuxPlyEntries;
740 int npeSize = nPlyEntries * 2 *
sizeof(float);
741 pPlyTable = (
float *)malloc(npeSize);
742 is.Read(pPlyTable, npeSize);
745 if (nAuxPlyEntries) {
746 int napeSize = nAuxPlyEntries *
sizeof(int);
747 pAuxPlyTable = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
748 pAuxCntTable = (
int *)malloc(napeSize);
749 is.Read(pAuxCntTable, napeSize);
751 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
753 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
754 pAuxPlyTable[nAuxPlyEntry] = (
float *)malloc(nfSize);
755 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
758 }
else if (db_version == 14) {
760 for (cp = path; (*cp = (char)is.GetC()) != 0; cp++);
761 pFullPath = (
char *)malloc(cp - path + 1);
762 strncpy(pFullPath, path, cp - path + 1);
763 wxLogVerbose(
" Chart %s", pFullPath);
770 EntryOffset = cte.EntryOffset;
771 ChartType = cte.ChartType;
777 m_bbox.Set(LatMin, LatMax, LonMin, LonMax);
780 edition_date = cte.edition_date;
782 nPlyEntries = cte.nPlyEntries;
783 nAuxPlyEntries = cte.nAuxPlyEntries;
787 int npeSize = nPlyEntries * 2 *
sizeof(float);
788 pPlyTable = (
float *)malloc(npeSize);
789 is.Read(pPlyTable, npeSize);
792 if (nAuxPlyEntries) {
793 int napeSize = nAuxPlyEntries *
sizeof(int);
794 pAuxPlyTable = (
float **)malloc(nAuxPlyEntries *
sizeof(
float *));
795 pAuxCntTable = (
int *)malloc(napeSize);
796 is.Read(pAuxCntTable, napeSize);
798 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries;
800 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
801 pAuxPlyTable[nAuxPlyEntry] = (
float *)malloc(nfSize);
802 is.Read(pAuxPlyTable[nAuxPlyEntry], nfSize);
806 ChartFamily = convertChartFamily(ChartType, ChartFamily);
807 ChartType = convertChartType(ChartType);
814bool ChartTableEntry::Write(
const ChartDatabase *pDb, wxOutputStream &os) {
815 os.Write(pFullPath, strlen(pFullPath) + 1);
822 cte.EntryOffset = EntryOffset;
823 cte.ChartType = ChartType;
824 cte.ChartFamily = ChartFamily;
831 cte.edition_date = edition_date;
832 cte.file_date = file_date;
834 cte.nPlyEntries = nPlyEntries;
835 cte.nAuxPlyEntries = nAuxPlyEntries;
838 cte.ProjectionType = ProjectionType;
842 cte.nNoCovrPlyEntries = nNoCovrPlyEntries;
845 wxLogVerbose(
" Wrote Chart %s", pFullPath);
849 int npeSize = nPlyEntries * 2 *
sizeof(float);
850 os.Write(pPlyTable, npeSize);
853 if (nAuxPlyEntries) {
854 int napeSize = nAuxPlyEntries *
sizeof(int);
855 os.Write(pAuxCntTable, napeSize);
857 for (
int nAuxPlyEntry = 0; nAuxPlyEntry < nAuxPlyEntries; nAuxPlyEntry++) {
858 int nfSize = pAuxCntTable[nAuxPlyEntry] * 2 *
sizeof(float);
859 os.Write(pAuxPlyTable[nAuxPlyEntry], nfSize);
863 if (nNoCovrPlyEntries) {
864 int ncSize = nNoCovrPlyEntries *
sizeof(int);
865 os.Write(pNoCovrCntTable, ncSize);
867 for (
int i = 0; i < nNoCovrPlyEntries; i++) {
868 int nctSize = pNoCovrCntTable[i] * 2 *
sizeof(float);
869 os.Write(pNoCovrPlyTable[i], nctSize);
878void ChartTableEntry::Clear() {
885 pNoCovrCntTable = NULL;
886 pNoCovrPlyTable = NULL;
888 nNoCovrPlyEntries = 0;
898void ChartTableEntry::Disable() {
902 LatMax += (float)1000.;
903 LatMin += (float)1000.;
906void ChartTableEntry::ReEnable() {
908 LatMax -= (float)1000.;
909 LatMin -= (float)1000.;
912bool ChartTableEntry::IsBasemap()
const {
913 wxFileName fn(GetFullPath());
914 return (fn.GetPath().Lower().Contains(
"basemap"));
917std::vector<float> ChartTableEntry::GetReducedPlyPoints() {
918 if (m_reducedPlyPoints.size())
return m_reducedPlyPoints;
921 float LOD_meters = 1;
923 float plylat, plylon;
924 const int nPoints = GetnPlyEntries();
926 float *fpo = GetpPlyTable();
928 double *ppd =
new double[nPoints * 2];
929 double *ppsm =
new double[nPoints * 2];
932 for (
int i = 0; i < nPoints; i++) {
934 plylon = fpo[i * 2 + 1];
937 toSM(plylat, plylon, fpo[0], fpo[1], &x, &y);
945 std::vector<int> index_keep;
947 index_keep.push_back(0);
948 index_keep.push_back(nPoints - 1);
949 index_keep.push_back(1);
950 index_keep.push_back(nPoints - 2);
952 DouglasPeuckerM(ppsm, 1, nPoints - 2, LOD_meters, &index_keep);
955 index_keep.resize(nPoints);
956 for (
int i = 0; i < nPoints; i++) index_keep[i] = i;
960 for (
int ip = 0; ip < nPoints; ip++) {
964 for (
unsigned int j = 0; j < index_keep.size(); j++) {
965 if (index_keep[j] == ip) {
966 m_reducedPlyPoints.push_back(x);
967 m_reducedPlyPoints.push_back(y);
976 int nprr = m_reducedPlyPoints.size() / 2;
978 return m_reducedPlyPoints;
981std::vector<float> ChartTableEntry::GetReducedAuxPlyPoints(
int iTable) {
983 if (!m_reducedAuxPlyPointsVector.size()) {
984 std::vector<float> vec;
985 for (
int i = 0; i < GetnAuxPlyEntries(); i++) {
986 m_reducedAuxPlyPointsVector.push_back(vec);
990 std::vector<float> vec;
993 if ((
unsigned int)iTable >= m_reducedAuxPlyPointsVector.size())
return vec;
995 if (m_reducedAuxPlyPointsVector.at(iTable).size())
996 return m_reducedAuxPlyPointsVector.at(iTable);
999 float LOD_meters = 1.0;
1001 const int nPoints = GetAuxCntTableEntry(iTable);
1002 float *fpo = GetpAuxPlyTableEntry(iTable);
1004 double *ppd =
new double[nPoints * 2];
1005 double *ppsm =
new double[nPoints * 2];
1007 double *npsm = ppsm;
1008 float plylat, plylon;
1010 for (
int i = 0; i < nPoints; i++) {
1011 plylat = fpo[i * 2];
1012 plylon = fpo[i * 2 + 1];
1015 toSM(plylat, plylon, fpo[0], fpo[1], &x, &y);
1023 std::vector<int> index_keep;
1025 index_keep.push_back(0);
1026 index_keep.push_back(nPoints - 1);
1027 index_keep.push_back(1);
1028 index_keep.push_back(nPoints - 2);
1030 DouglasPeuckerM(ppsm, 1, nPoints - 2, LOD_meters, &index_keep);
1033 index_keep.resize(nPoints);
1034 for (
int i = 0; i < nPoints; i++) index_keep[i] = i;
1037 int nnn = index_keep.size();
1040 for (
int ip = 0; ip < nPoints; ip++) {
1044 for (
unsigned int j = 0; j < index_keep.size(); j++) {
1045 if (index_keep[j] == ip) {
1056 m_reducedAuxPlyPointsVector[iTable] = vec;
1058 int nprr = vec.size() / 2;
1066const int ID_DBS_PROGRESS_UPDATE = wxNewId();
1068WX_DEFINE_OBJARRAY(ChartTable);
1070ChartDatabase::ChartDatabase() {
1074 m_ChartTableEntryDummy.Clear();
1076 Bind(wxEVT_OCPN_CHARTTABLEENTRYTHREAD, &ChartDatabase::OnEvtThread,
this);
1077 Bind(wxEVT_COMMAND_MENU_SELECTED, &ChartDatabase::OnDBSProgressUpdate,
this,
1078 ID_DBS_PROGRESS_UPDATE);
1080 UpdateChartClassDescriptorArray();
1083void ChartDatabase::OnDBSProgressUpdate(wxCommandEvent &evt) {
1084 int value = evt.GetInt();
1087 m_pprog->Update(value);
1093 if (in_event)
int yyp = 4;
1097 auto ticket =
event.GetTicket();
1105 if (m_pprog && (m_jobsRemaining > 1)) {
1107 double ratio = 100. * (double)m_progcount / m_ticketcount;
1109 if (((m_progcount % m_nFileProgressQuantum) == 0)) {
1110 if (val != m_progint) {
1116 wxEventLoopBase *loop = wxEventLoopBase::GetActive();
1119 wxEventLoopActivator noLoop(
nullptr);
1120 m_pprog->Update(val);
1123 m_pprog->Update(val);
1130 if (!ticket->m_ticket_type) {
1131 bool collision_found =
false;
1132 if (ticket->b_thread_safe) {
1133 wxFileName fn(ticket->m_ChartPath);
1134 ChartCollisionsHashMap::iterator it;
1135 for (it = m_full_collision_map.begin(); it != m_full_collision_map.end();
1137 if (it->first.IsSameAs(fn.GetFullName())) {
1141 collision_found =
true;
1145 if (!collision_found) {
1146 m_full_collision_map[fn.GetFullName()] = 1;
1151 m_ticket_vector.push_back(ticket);
1154 int remaining = --m_jobsRemaining;
1156 if (remaining == 0) {
1158 m_pool_deferred.Shutdown();
1159 size_t a = m_ticket_vector.size();
1161 if (m_deferred_ticket_vector.size()) {
1165 for (
auto &ticket_d : m_deferred_ticket_vector) {
1166 m_pool_deferred.Push(ticket_d);
1169 m_deferred_ticket_vector.clear();
1174 m_ticketcount = m_jobsRemaining;
1175 m_nFileProgressQuantum = 1;
1176 if (m_pprog) m_pprog->Update(0, _(
"Processing charts."));
1179 if (m_jobsRemaining) {
1180 const int workerCount = 1;
1181 if (m_pool_deferred.GetWorkerCount() < workerCount) {
1182 int threads_needed = workerCount - m_pool_deferred.GetWorkerCount();
1183 for (
int i = 0; i < threads_needed; ++i) {
1185 m_pool_deferred.AddWorker();
1200 for (
auto &ticket_valid : m_ticket_vector) {
1201 if (ticket_valid->m_chart_table_entry)
1202 active_chartTable.push_back(ticket_valid->m_chart_table_entry);
1204 size_t c = m_ticket_vector.size();
1205 size_t d = active_chartTable.size();
1208 m_ticket_vector.clear();
1210 FinalizeChartUpdate();
1216void ChartDatabase::FinalizeChartUpdate() {
1219 active_chartTable.erase(
1220 std::remove_if(active_chartTable.begin(), active_chartTable.end(),
1221 [](
const auto &cte) { return !cte->GetbValid(); }),
1222 active_chartTable.end());
1224 size_t d = active_chartTable.size();
1227 active_chartTable_pathindex.clear();
1229 for (
auto cte : active_chartTable) {
1230 active_chartTable_pathindex[cte->GetFullSystemPath()] = i;
1231 cte->SetEntryOffset(i);
1235 m_nentries = active_chartTable.size();
1241 ChartData->SaveBinary(ChartListFileName);
1242 wxLogMessage(
"Finished chart database Update");
1245 if (m_pprog) m_pprog->Destroy();
1250 if (gWorldMapLocation.empty()) {
1254 gWorldMapLocation = gDefaultWorldMapLocation;
1255 m_gshhg_chart_loc = wxEmptyString;
1259 if (gWorldMapLocation != m_gshhg_chart_loc) {
1268 gshhsCrossesLandReset();
1281bool ChartDatabase::ScrubGroupArray() {
1286 bool b_change =
false;
1287 unsigned int igroup = 0;
1288 while (igroup < g_pGroupArray->GetCount()) {
1289 bool b_chart_in_element =
false;
1292 for (
unsigned int j = 0; j < pGroup->m_element_array.size(); j++) {
1293 const wxString &element_root = pGroup->m_element_array[j].m_element_name;
1295 for (
unsigned int ic = 0;
1296 ic < (
unsigned int)
ChartData->GetChartTableEntries(); ic++) {
1297 auto &cte =
ChartData->GetChartTableEntry(ic);
1298 wxString chart_full_path = cte.GetFullSystemPath();
1300 if (chart_full_path.StartsWith(element_root)) {
1301 b_chart_in_element =
true;
1307 if (!b_chart_in_element) {
1308 wxString test_string =
"GSHH";
1309 if (element_root.Upper().Contains(test_string))
1310 b_chart_in_element =
true;
1313 if (!b_chart_in_element)
1315 pGroup->m_element_array.erase(pGroup->m_element_array.begin() + j);
1327void ChartDatabase::UpdateChartClassDescriptorArray() {
1328 if (m_ChartClassDescriptorArray.empty()) {
1329 m_ChartClassDescriptorArray.push_back(
1331 m_ChartClassDescriptorArray.push_back(
1333 m_ChartClassDescriptorArray.push_back(
1335 m_ChartClassDescriptorArray.push_back(
1338 "cm93compchart",
"00300000.a",
nullptr, BUILTIN_DESCRIPTOR));
1340 "ChartMbTiles",
"*.mbtiles",
nullptr, BUILTIN_DESCRIPTOR));
1345 m_ChartClassDescriptorArray.erase(
1346 std::remove_if(m_ChartClassDescriptorArray.begin(),
1347 m_ChartClassDescriptorArray.end(),
1349 return cd.m_descriptor_type == PLUGIN_DESCRIPTOR;
1351 m_ChartClassDescriptorArray.end());
1353 wxArrayString array =
g_pi_manager->GetPlugInChartClassNameArray();
1354 for (
unsigned int j = 0; j < array.GetCount(); j++) {
1357 wxString class_name = array[j];
1360 wxString mask = cpiw->GetFileSearchMask();
1363 auto plugin_t =
g_pi_manager->GetProvidingPlugin(class_name);
1366 m_ChartClassDescriptorArray.push_back(
1375 if (index < GetChartTableEntries())
1376 return *active_chartTable[index];
1378 return m_ChartTableEntryDummy;
1381bool ChartDatabase::CompareChartDirArray(ArrayOfCDI &test_array) {
1385 if (test_array.GetCount() != m_dir_array.GetCount())
return false;
1388 unsigned int nfound_outer = 0;
1390 for (
unsigned int i = 0; i < test_array.GetCount(); i++) {
1392 bfound_inner =
false;
1393 for (
unsigned int j = 0; j < m_dir_array.GetCount(); j++) {
1396 if (p.fullpath.IsSameAs(q.fullpath)) {
1397 bfound_inner =
true;
1401 if (bfound_inner) nfound_outer++;
1404 return (nfound_outer == test_array.GetCount());
1407wxString ChartDatabase::GetMagicNumberCached(wxString dir) {
1408 for (
unsigned int j = 0; j < m_dir_array.GetCount(); j++) {
1410 if (dir.IsSameAs(q.fullpath))
return q.magic_number;
1416bool ChartDatabase::Read(
const wxString &filePath) {
1422 wxFileName file(filePath);
1423 if (!file.FileExists())
return false;
1425 m_DBFileName = filePath;
1427 wxFFileInputStream ifs(filePath);
1428 if (!ifs.Ok())
return false;
1432 if (!cth.CheckValid())
return false;
1436 memcpy(vbo, cth.GetDBVersionString(), 4);
1438 m_dbversion = atoi(&vbo[1]);
1439 s_dbVersion = m_dbversion;
1441 wxLogVerbose(
"Chartdb:Reading %d directory entries, %d table entries",
1442 cth.GetDirEntries(), cth.GetTableEntries());
1443 wxLogMessage(
"Chartdb: Chart directory list follows");
1444 if (0 == cth.GetDirEntries()) wxLogMessage(
" Nil");
1447 for (
int iDir = 0; iDir < cth.GetDirEntries(); iDir++) {
1450 ifs.Read(&dirlen,
sizeof(
int));
1451 while (dirlen > 0) {
1453 int alen = dirlen > 1023 ? 1023 : dirlen;
1454 if (ifs.Read(&dirbuf, alen).Eof())
goto read_error;
1457 dir.Append(wxString(dirbuf, wxConvUTF8));
1460 msg.Printf(
" Chart directory #%d: ", iDir);
1463 m_chartDirs.Add(dir);
1466 entries = cth.GetTableEntries();
1468 active_chartTable_pathindex.clear();
1469 while (entries-- && entry.Read(
this, ifs)) {
1470 active_chartTable_pathindex[entry.GetFullSystemPath()] = ind++;
1471 auto sharedPtr = std::make_shared<ChartTableEntry>(entry);
1472 active_chartTable.push_back(sharedPtr);
1477 entry.SetAvailable(
true);
1479 m_nentries = active_chartTable.size();
1484 m_nentries = active_chartTable.size();
1490bool ChartDatabase::Write(
const wxString &filePath) {
1491 wxFileName file(filePath);
1493 file.GetPath(wxPATH_GET_SEPARATOR | wxPATH_GET_VOLUME, wxPATH_NATIVE));
1495 if (!dir.DirExists() && !dir.Mkdir())
return false;
1497 wxFFileOutputStream ofs(filePath);
1498 if (!ofs.Ok())
return false;
1503 for (
int iDir = 0; iDir < cth.GetDirEntries(); iDir++) {
1504 wxString dir1 = m_chartDirs[iDir];
1505 int dirlen = dir1.length();
1507 strncpy(s, dir1.mb_str(wxConvUTF8), 199);
1510 ofs.Write(&dirlen,
sizeof(
int));
1512 ofs.Write(s, dirlen);
1515 for (UINT32 iTable = 0; iTable < active_chartTable.size(); iTable++) {
1516 auto &cte = GetChartTableEntry(iTable);
1517 cte.Write(
this, ofs);
1521 m_dbversion = DB_VERSION_CURRENT;
1527wxString SplitPath(wxString s, wxString tkd,
int nchar,
int offset,
1533 wxStringTokenizer tkz(s, tkd);
1534 while (tkz.HasMoreTokens()) {
1535 wxString token = tkz.GetNextToken();
1536 if ((rlen + (
int)token.Len() + 1) < nchar) {
1539 rlen += token.Len() + 1;
1543 for (
int i = 0; i < offset; i++) {
1548 rlen = offset + token.Len() + 1;
1552 if (pn_split) *pn_split = ncr;
1554 return r.Mid(0, r.Len() - 1);
1557wxString ChartDatabase::GetFullChartInfo(
ChartBase *pc,
int dbIndex,
1558 int *char_width,
int *line_count) {
1561 unsigned int max_width = 0;
1563 unsigned int target_width = 60;
1569 line = _(
" Name: ");
1570 wxString longline = pc->GetName();
1573 if (longline.Find(
' ') != wxNOT_FOUND)
1578 if (longline.Len() > target_width) {
1579 line += SplitPath(pc->GetName(), tkz, target_width, 12, &ncr);
1580 max_width = wxMax(max_width, target_width + 4);
1584 max_width = wxMax(max_width, line.Len() + 4);
1593 line.Printf(
" %s: 1:%d", _(
"Scale"), pc->GetNativeScale());
1595 line.Printf(
" %s: 1:%d", _(
"Scale"), cte.GetScale());
1598 max_width = wxMax(max_width, line.Len());
1602 wxDateTime ed = pc->GetEditionDate();
1604 line = _(
" Updated: ") + ed.FormatISODate() +
"\n";
1605 max_width = wxMax(max_width, line.Len());
1610 line = _(
" Source Edition: ") + pc->GetSE() +
"\n";
1611 max_width = wxMax(max_width, line.Len());
1617 line = _(
" Depth Units: ") + pc->GetDepthUnits() +
"\n";
1618 max_width = wxMax(max_width, line.Len());
1622 line = _(
" Soundings: ") + pc->GetSoundingsDatum() +
"\n";
1623 max_width = wxMax(max_width, line.Len());
1627 line = _(
" Datum: ") + pc->GetDatumString() +
"\n";
1628 max_width = wxMax(max_width, line.Len());
1633 line = _(
" Projection: ");
1634 if (PROJECTION_UNKNOWN == cte.GetChartProjectionType())
1635 line += _(
"Unknown");
1636 else if (PROJECTION_MERCATOR == cte.GetChartProjectionType())
1637 line += _(
"Mercator");
1638 else if (PROJECTION_TRANSVERSE_MERCATOR == cte.GetChartProjectionType())
1639 line += _(
"Transverse Mercator");
1640 else if (PROJECTION_POLYCONIC == cte.GetChartProjectionType())
1641 line += _(
"Polyconic");
1642 else if (PROJECTION_WEB_MERCATOR == cte.GetChartProjectionType())
1643 line += _(
"Web Mercator (EPSG:3857)");
1645 max_width = wxMax(max_width, line.Len());
1654 if (pc && pc->GetExtraInfo().Len()) {
1655 line += pc->GetExtraInfo();
1657 max_width = wxMax(max_width, line.Len());
1664 line += pc->GetID();
1666 max_width = wxMax(max_width, line.Len());
1671 line = _(
" ChartFile: ");
1672 wxString longline = *(cte.GetpsFullPath());
1673 if (longline.Len() > target_width) {
1674 line += SplitPath(longline,
"/,\\", target_width, 15, &ncr);
1675 max_width = wxMax(max_width, target_width + 4);
1679 max_width = wxMax(max_width, line.Len() + 4);
1685 if (line_count) *line_count = lc;
1687 if (char_width) *char_width = max_width;
1697 wxGenericProgressDialog *pprog) {
1698 m_dir_array = dir_array;
1702 m_chartDirs.Clear();
1703 active_chartTable.clear();
1704 active_chartTable_pathindex.clear();
1706 Update(dir_array,
true, pprog);
1711 m_dbversion = DB_VERSION_CURRENT;
1723 virtual wxDirTraverseResult OnFile(
const wxString &filename)
override {
1724 wxFileName fn(filename);
1725 wxFileName dir(fn.GetPath());
1726 if (fn.GetFullName().Matches(
"poly-*-1.dat") &&
1727 dir.GetFullName().IsSameAs(
"GSHHG",
false)) {
1728 parent_dir = fn.GetPath(wxPATH_GET_VOLUME | wxPATH_GET_SEPARATOR);
1731 return wxDIR_CONTINUE;
1733 virtual wxDirTraverseResult OnDir(
const wxString &dirname)
override {
1735 return wxDIR_CONTINUE;
1737 wxString GetGshhsDir()
const {
return parent_dir; }
1740 wxString parent_dir;
1747wxString findGshhgDirectory(
const wxString &directory) {
1748 wxDir dir(directory);
1749 if (!dir.IsOpened()) {
1750 return wxEmptyString;
1753 dir.Traverse(traverser, wxEmptyString, wxDIR_FILES | wxDIR_DIRS);
1754 return traverser.GetGshhsDir();
1757bool ChartDatabase::UpdateChartDatabaseInplace(ArrayOfCDI &DirArray,
1759 wxGenericProgressDialog *_prog) {
1765 wxLogMessage(
"Starting chart database Update...");
1769 m_gshhg_chart_loc = gWorldMapLocation;
1770 gWorldMapLocation = wxEmptyString;
1772 Update(DirArray, b_force, m_pprog);
1783 wxGenericProgressDialog *pprog) {
1785 m_ticket_vector.clear();
1786 m_full_collision_map.clear();
1787 m_jobsRemaining = 0;
1788 m_deferred_ticket_vector.clear();
1789 SetChartDirArray(dir_array);
1796 for (
unsigned int i = 0; i < active_chartTable.size(); i++) {
1797 auto &cte = GetChartTableEntry(i);
1798 cte.SetValid(
false);
1801 m_chartDirs.Clear();
1803 if (bForce) active_chartTable.clear();
1805 bool lbForce = bForce;
1808 if (s_dbVersion != DB_VERSION_CURRENT) {
1809 active_chartTable.clear();
1811 s_dbVersion = DB_VERSION_CURRENT;
1812 m_dbversion = DB_VERSION_CURRENT;
1817 for (
unsigned int j = 0; j < dir_array.GetCount(); j++) {
1824 if (!androidIsDirWritable(dir_info.fullpath))
continue;
1830 wxString gshhg_dir = findGshhgDirectory(dir_info.fullpath);
1831 if (!gshhg_dir.empty()) {
1836 wxLogMessage(
"Updating GSHHG directory: %s", gshhg_dir.c_str());
1837 gWorldMapLocation = gshhg_dir;
1842 if (dir_info.fullpath.Find(
"OSMSHP") != wxNOT_FOUND) {
1843 if (!wxDir::FindFirst(dir_info.fullpath,
"basemap_*.shp").empty()) {
1844 wxLogMessage(
"Updating OSMSHP directory: %s", dir_info.fullpath);
1845 gWorldShapefileLocation =
1846 dir_info.fullpath + wxFileName::GetPathSeparator();
1851 TraverseDirAndAddCharts(dir_info, pprog, dir_magic, lbForce);
1854 dir_info.magic_number = dir_magic;
1855 dir_array.RemoveAt(j);
1856 dir_array.Insert(dir_info, j);
1858 m_chartDirs.Add(dir_info.fullpath);
1862 if (m_chartDirs.IsEmpty() || !m_jobsRemaining) {
1863 if (!m_deferred_ticket_vector.size()) {
1864 FinalizeChartUpdate();
1869 auto ticket_deferred = m_deferred_ticket_vector.back();
1870 ticket_deferred->m_ticket_type = 1;
1872 m_jobsRemaining = 1;
1873 evt->SetTicket(ticket_deferred);
1874 wxQueueEvent(
this, evt);
1881 m_ticketcount = m_jobsRemaining;
1882 m_nFileProgressQuantum = wxMax(m_ticketcount / 10, 2);
1883 if (pprog) pprog->Update(0, _(
"Processing charts."));
1886 if (m_jobsRemaining) {
1887 const int workerCount = 4;
1888 if (m_pool.GetWorkerCount() < workerCount) {
1889 int threads_needed = workerCount - m_pool.GetWorkerCount();
1890 for (
int i = 0; i < threads_needed; ++i) {
1904int ChartDatabase::FinddbIndex(wxString PathToFind) {
1905 if (active_chartTable_pathindex.find(PathToFind) !=
1906 active_chartTable_pathindex.end())
1907 return active_chartTable_pathindex[PathToFind];
1916int ChartDatabase::DisableChart(wxString &PathToDisable) {
1917 int index = FinddbIndex(PathToDisable);
1919 auto &entry = GetChartTableEntry(index);
1933int ChartDatabase::TraverseDirAndAddCharts(
ChartDirInfo &dir_info,
1934 wxGenericProgressDialog *pprog,
1935 wxString &dir_magic,
bool bForce) {
1937 wxString dir_path = dir_info.fullpath;
1939 dir_path = wxString(dir_info.fullpath.mb_str(wxConvUTF8));
1942 wxString old_magic = dir_info.magic_number;
1943 wxString new_magic = old_magic;
1944 dir_magic = old_magic;
1948 bool b_skipDetectDirChange =
false;
1949 bool b_dirchange =
false;
1952 if (!wxDir::Exists(dir_path))
return 0;
1958 bool b_cm93 = Check_CM93_Structure(dir_path);
1960 b_skipDetectDirChange =
true;
1966 if (!b_skipDetectDirChange)
1967 b_dirchange = DetectDirChange(dir_path, dir_info.fullpath, old_magic,
1970 if (!bForce && !b_dirchange) {
1971 wxString msg(
" No change detected on directory ");
1972 msg.Append(dir_path);
1978 wxFileName fn_dir(dir_path,
"stuff");
1979 unsigned int dir_path_count = fn_dir.GetDirCount();
1981 if (pprog) pprog->SetTitle(_(
"OpenCPN Chart Scan...."));
1983 int nEntries = active_chartTable.size();
1985 for (
int ic = 0; ic < nEntries; ic++) {
1986 auto &cte = GetChartTableEntry(ic);
1987 wxFileName fn(cte.GetFullSystemPath());
1989 while (fn.GetDirCount() >= dir_path_count) {
1990 if (fn.GetPath() == dir_path) {
1991 auto &cte_a = GetChartTableEntry(ic);
1992 cte_a.SetValid(
true);
2008 dir_magic = new_magic;
2011 for (
auto &cd : m_ChartClassDescriptorArray) {
2012 nAdd += SearchDirAndAddCharts(dir_info.fullpath, cd, pprog);
2018bool ChartDatabase::DetectDirChange(
const wxString &dir_path,
2019 const wxString &prog_label,
2020 const wxString &magic, wxString &new_magic,
2021 wxGenericProgressDialog *pprog) {
2022 if (pprog) pprog->SetTitle(_(
"OpenCPN Directory Scan...."));
2025 long long unsigned int nmagic;
2026 wxULongLong nacc = 0;
2028 magic.ToULongLong(&nmagic, 10);
2031 wxArrayString FileList;
2032 wxDir dir(dir_path);
2033 int n_files = dir.GetAllFiles(dir_path, &FileList);
2039 if (pprog) pprog->Update(0, prog_label);
2043 for (
int ifile = 0; ifile < n_files; ifile++) {
2044 wxFileName file(FileList[ifile]);
2049 wxString fileNameNative = file.GetFullPath();
2050 wxScopedCharBuffer fileNameUTF8 = fileNameNative.ToUTF8();
2051 hash.Update(fileNameUTF8.data(), fileNameUTF8.length());
2054 wxULongLong size = file.GetSize();
2055 wxULongLong fileSize = ((size != wxInvalidSize) ? size : 0);
2056 hash.Update(&fileSize, (
sizeof fileSize));
2059 wxDateTime t = file.GetModificationTime();
2060 wxULongLong fileTime = t.GetTicks();
2061 hash.Update(&fileTime, (
sizeof fileTime));
2065 hash.Receive(&nacc);
2068 new_magic = nacc.ToString();
2071 if (new_magic != magic)
2077bool ChartDatabase::IsChartDirUsed(
const wxString &theDir) {
2078 wxString dir(theDir);
2079 if (dir.Last() ==
'/' || dir.Last() == wxFileName::GetPathSeparator())
2083 for (UINT32 i = 0; i < active_chartTable.size(); i++) {
2084 auto &cte_u = GetChartTableEntry(i);
2085 if (cte_u.GetpsFullPath()->Matches(dir))
return true;
2094bool ChartDatabase::Check_CM93_Structure(wxString dir_name) {
2097 wxRegEx test(
"[0-9]+");
2099 wxDir dirt(dir_name);
2102 if (dirt.IsOpened())
2103 wxLogMessage(
"check_cm93 opened dir OK: " + dir_name);
2105 wxLogMessage(
"check_cm93 NOT OPENED OK: " + dir_name);
2106 wxLogMessage(
"check_cm93 returns false." + dir_name);
2110 bool b_maybe_found_cm93 =
false;
2111 bool b_cont = dirt.GetFirst(&candidate);
2114 if (test.Matches(candidate) && (candidate.Len() == 8)) {
2115 b_maybe_found_cm93 =
true;
2119 b_cont = dirt.GetNext(&candidate);
2122 if (b_maybe_found_cm93) {
2123 wxString dir_next = dir_name;
2125 dir_next += candidate;
2126 if (wxDir::Exists(dir_next)) {
2127 wxDir dir_n(dir_next);
2128 if (dirt.IsOpened()) {
2129 wxString candidate_n;
2131 wxRegEx test_n(
"^[A-Ga-g]");
2132 bool b_probably_found_cm93 =
false;
2133 bool b_cont_n = dir_n.IsOpened() && dir_n.GetFirst(&candidate_n);
2135 if (test_n.Matches(candidate_n) && (candidate_n.Len() == 1)) {
2136 b_probably_found_cm93 =
true;
2139 b_cont_n = dir_n.GetNext(&candidate_n);
2142 if (b_probably_found_cm93)
2147 wxString dir_luk = dir_next;
2149 dir_luk += candidate_n;
2150 if (wxDir::Exists(dir_luk))
return true;
2263int ChartDatabase::SearchDirAndAddCharts(wxString &dir_name_base,
2265 wxGenericProgressDialog *pprog) {
2266 wxString msg(
"Searching directory: ");
2267 msg += dir_name_base;
2269 msg += chart_desc.m_search_mask;
2272 wxString dir_name = dir_name_base;
2275 dir_name = wxString(dir_name_base.mb_str(wxConvUTF8));
2278 if (!wxDir::Exists(dir_name))
return 0;
2280 wxString filespec = chart_desc.m_search_mask.Upper();
2281 wxString lowerFileSpec = chart_desc.m_search_mask.Lower();
2282 wxString filespecXZ = filespec +
".xz";
2283 wxString lowerFileSpecXZ = lowerFileSpec +
".xz";
2287 wxArrayString FileList;
2288 int gaf_flags = wxDIR_DEFAULT;
2295 bool b_found_cm93 =
false;
2296 bool b_cm93 = Check_CM93_Structure(dir_name);
2298 if (filespec !=
"00300000.A")
2301 filespec = dir_name;
2302 b_found_cm93 =
true;
2306 if (!b_found_cm93) {
2307 wxDir dir(dir_name);
2308 dir.GetAllFiles(dir_name, &FileList, filespec, gaf_flags);
2311 if (!FileList.GetCount()) {
2312 wxArrayString afl = androidTraverseDir(dir_name, filespec);
2313 for (wxArrayString::const_iterator item = afl.begin(); item != afl.end();
2315 FileList.Add(*item);
2320 if (filespec != lowerFileSpec) {
2322 wxArrayString lowerFileList;
2323 dir.GetAllFiles(dir_name, &lowerFileList, lowerFileSpec, gaf_flags);
2326 if (!lowerFileList.GetCount()) {
2327 wxArrayString afl = androidTraverseDir(dir_name, lowerFileSpec);
2328 for (wxArrayString::const_iterator item = afl.begin();
2329 item != afl.end(); item++)
2330 lowerFileList.Add(*item);
2334 for (wxArrayString::const_iterator item = lowerFileList.begin();
2335 item != lowerFileList.end(); item++)
2336 FileList.Add(*item);
2342 dir.GetAllFiles(dir_name, &FileList, filespecXZ, gaf_flags);
2343 dir.GetAllFiles(dir_name, &FileList, lowerFileSpecXZ, gaf_flags);
2349 wxString dir_plus = dir_name;
2350 dir_plus += wxFileName::GetPathSeparator();
2351 FileList.Add(dir_plus);
2354 int nFile = FileList.GetCount();
2356 if (!nFile)
return false;
2365 if (pprog) pprog->SetTitle(_(
"OpenCPN Chart Add...."));
2369 ChartCollisionsHashMap collision_map;
2378 std::vector<std::shared_ptr<ChartTableEntryJobTicket>> ticket_vector;
2380 for (
int ifile = 0; ifile < nFile; ifile++) {
2381 wxFileName file(FileList[ifile]);
2382 wxString full_path = file.GetFullPath();
2383 wxString file_name = file.GetFullName();
2384 wxString utf8_path = full_path;
2392 wxFileName fnbase(dir_name_base);
2393 int nDirs = fnbase.GetDirCount();
2395 wxFileName file_target(FileList[ifile]);
2397 for (
int i = 0; i < nDirs + 1;
2399 file_target.RemoveDir(0);
2401 wxString leftover_path = file_target.GetFullPath();
2403 dir_name_base + leftover_path;
2409 if (!file_name.Matches(lowerFileSpec) && !file_name.Matches(filespec) &&
2410 !file_name.Matches(lowerFileSpecXZ) && !file_name.Matches(filespecXZ) &&
2417 bool bAddFinal =
true;
2433 if (b_found_cm93) collision =
false;
2436 pEntry = &active_chartTable[collision_ptr->second];
2437 table_file_name = pEntry->GetFullSystemPath();
2439 bthis_dir_in_dB && full_name.IsSameAs(table_file_name);
2443 if (file_path_is_same) {
2447 time_t t_oldFile = pEntry->GetFileTime();
2448 time_t t_newFile = file.GetModificationTime().GetTicks();
2450 if (t_newFile <= t_oldFile) {
2451 file_time_is_same =
true;
2453 pEntry->SetValid(
true);
2456 pEntry->SetValid(
false);
2463 bool collision_found =
false;
2464 ChartCollisionsHashMap::iterator it;
2465 for (it = collision_map.begin(); it != collision_map.end(); ++it) {
2466 if (it->first.IsSameAs(file_name)) {
2470 collision_found =
true;
2475 if (!collision_found) {
2477 auto ticket = std::make_shared<ChartTableEntryJobTicket>();
2478 ticket->m_ChartPath = full_path;
2479 ticket->m_ChartPathUTF8 = full_path;
2480 ticket->chart_desc = chart_desc;
2482 ticket_vector.push_back(ticket);
2483 collision_map[file_name] = ifile;
2490 bool is_kap =
false;
2491 if (chart_desc.m_descriptor_type == PLUGIN_DESCRIPTOR || is_kap) {
2497 if (!provider_plugin118 || is_kap) {
2501 for (
auto &ticket : ticket_vector) {
2502 ticket->b_thread_safe =
false;
2503 ticket->m_provider_type = 1;
2504 ticket->provider_class_name = chart_desc.m_class_name;
2505 m_deferred_ticket_vector.push_back(ticket);
2514 for (
auto &ticket : ticket_vector) {
2515 m_pool.Push(ticket);
2523bool ChartDatabase::AddChart(wxString &chartfilename,
2525 wxGenericProgressDialog *pprog,
int isearch,
2526 bool bthis_dir_in_dB) {
2528 wxFileName file(chartfilename);
2529 wxString full_name = file.GetFullPath();
2530 wxString file_name = file.GetFullName();
2540 pprog->Update(wxMin((m_pdifile * 100) / m_pdnFile, 100), full_name);
2543 bool bAddFinal =
true;
2545 wxString msg_fn(full_name);
2546 msg_fn.Replace(
"%",
"%%");
2548 pnewChart = CreateChartTableEntry(full_name, full_name, chart_desc);
2551 wxLogMessage(wxString::Format(
2552 " CreateChartTableEntry() failed for file: %s", msg_fn.c_str()));
2557 int nEntry = active_chartTable.size();
2558 for (
int i = 0; i < nEntry; i++) {
2559 auto &cte_a = GetChartTableEntry(i);
2560 wxString *ptable_file_name = cte_a.GetpsFullPath();
2564 if (bthis_dir_in_dB && full_name.IsSameAs(*ptable_file_name)) {
2568 auto &cte_search = GetChartTableEntry(isearch);
2569 time_t t_oldFile = cte_search.GetFileTime();
2570 time_t t_newFile = file.GetModificationTime().GetTicks();
2572 if (t_newFile <= t_oldFile) {
2574 cte_search.SetValid(
true);
2577 cte_search.SetValid(
false);
2579 wxString::Format(
" Replacing older chart file of same path: %s",
2589 wxFileName table_file(*ptable_file_name);
2591 if (table_file.GetFullName() == file_name) {
2596 if (pnewChart->IsEarlierThan(active_chartTable[isearch])) {
2598 if (table_file.IsFileReadable()) {
2599 active_chartTable[isearch].SetValid(
true);
2601 wxLogMessage(wxString::Format(
2602 " Retaining newer chart file of same name: %s",
2605 }
else if (pnewChart->IsEqualTo(active_chartTable[isearch])) {
2615 active_chartTable[isearch].SetValid(
false);
2618 wxString::Format(
" Replacing older chart file of same name: %s",
2628 if (nEntry == isearch) isearch = 0;
2633 if (0 == b_add_msg) {
2635 wxString::Format(
" Adding chart file: %s", msg_fn.c_str()));
2637 std::shared_ptr<ChartTableEntry> sharedPtr(pnewChart);
2638 active_chartTable.push_back(sharedPtr);
2647 m_nentries = active_chartTable.size();
2652bool ChartDatabase::AddSingleChart(wxString &ChartFullPath,
2653 bool b_force_full_search) {
2655 wxFileName fn(ChartFullPath);
2656 wxString ext = fn.GetExt();
2658 wxString ext_upper = ext.MakeUpper();
2659 wxString ext_lower = ext.MakeLower();
2660 wxString dir_name = fn.GetPath();
2666 for (
auto &cd : m_ChartClassDescriptorArray) {
2667 if (cd.m_descriptor_type == PLUGIN_DESCRIPTOR) {
2668 if (cd.m_search_mask == ext_upper) {
2672 if (cd.m_search_mask == ext_lower) {
2681 bool b_recurse =
true;
2682 if (!b_force_full_search) b_recurse = IsChartDirUsed(dir_name);
2684 bool rv = AddChart(ChartFullPath, desc, NULL, 0, b_recurse);
2688 for (
unsigned int i = 0; i < active_chartTable.size(); i++) {
2689 auto &cte_r = GetChartTableEntry(i);
2690 if (!cte_r.GetbValid()) {
2698 for (
unsigned int i = 0; i < active_chartTable.size(); i++) {
2699 auto &cte_ef = GetChartTableEntry(i);
2700 cte_ef.SetEntryOffset(i);
2705 DetectDirChange(dir_name,
"",
"", new_magic, 0);
2708 bool bcfound =
false;
2709 ArrayOfCDI NewChartDirArray;
2711 ArrayOfCDI ChartDirArray = GetChartDirArray();
2712 for (
unsigned int i = 0; i < ChartDirArray.GetCount(); i++) {
2718 if (newcdi.fullpath == dir_name) {
2719 newcdi.magic_number = new_magic;
2723 NewChartDirArray.Add(newcdi);
2728 cdi.fullpath = dir_name;
2729 cdi.magic_number = new_magic;
2730 NewChartDirArray.Add(cdi);
2734 SetChartDirArray(NewChartDirArray);
2737 m_chartDirs.Clear();
2739 for (
unsigned int i = 0; i < GetChartDirArray().GetCount(); i++) {
2741 m_chartDirs.Add(cdi.fullpath);
2744 m_nentries = active_chartTable.size();
2746 ChartData->UpdateChartDatabaseInplace(NewChartDirArray,
false,
nullptr);
2750bool ChartDatabase::RemoveSingleChart(wxString &ChartFullPath) {
2754 for (
unsigned int i = 0; i < active_chartTable.size(); i++) {
2755 auto &cte = GetChartTableEntry(i);
2756 if (ChartFullPath.IsSameAs(cte.GetFullSystemPath())) {
2758 std::swap(active_chartTable[i], active_chartTable.back());
2759 active_chartTable.pop_back();
2765 for (
unsigned int i = 0; i < active_chartTable.size(); i++) {
2766 auto &pcte = GetChartTableEntry(i);
2767 pcte.SetEntryOffset(i);
2771 wxFileName fn(ChartFullPath);
2772 wxString fd = fn.GetPath();
2773 if (!IsChartDirUsed(fd)) {
2775 ArrayOfCDI NewChartDirArray;
2777 ArrayOfCDI ChartDirArray = GetChartDirArray();
2778 for (
unsigned int i = 0; i < ChartDirArray.GetCount(); i++) {
2783 if (newcdi.fullpath != fd) NewChartDirArray.Add(newcdi);
2786 SetChartDirArray(NewChartDirArray);
2790 m_chartDirs.Clear();
2791 for (
unsigned int i = 0; i < GetChartDirArray().GetCount(); i++) {
2793 m_chartDirs.Add(cdi.fullpath);
2796 m_nentries = active_chartTable.size();
2797 ChartData->UpdateChartDatabaseInplace(m_dir_array,
false,
nullptr);
2806ChartBase *ChartDatabase::GetChart(
const wxChar *theFilePath,
2817 const wxString &filePath, wxString &utf8Path,
2819 wxString msg_fn(filePath);
2820 msg_fn.Replace(
"%",
"%%");
2821 wxLogMessage(wxString::Format(
"Loading chart data for %s", msg_fn.c_str()));
2823 ChartBase *pch = GetChart(filePath, chart_desc);
2826 wxString::Format(
" ...creation failed for %s", msg_fn.c_str()));
2830 InitReturn rc = pch->Init(filePath, HEADER_ONLY);
2831 if (rc != INIT_OK) {
2834 wxString::Format(
" ...initialization failed for %s", msg_fn.c_str()));
2839 ret_val->SetValid(
true);
2846bool ChartDatabase::GetCentroidOfLargestScaleChart(
double *clat,
double *clon,
2847 ChartFamilyEnum family) {
2849 int cur_max_scale = 0;
2851 int nEntry = active_chartTable.size();
2853 for (
int i = 0; i < nEntry; i++) {
2854 auto &cte_fam = GetChartTableEntry(i);
2855 if (GetChartFamily(cte_fam.GetChartType()) == family) {
2856 if (cte_fam.GetScale() > cur_max_scale) {
2857 cur_max_scale = cte_fam.GetScale();
2863 if (cur_max_i == -1)
2866 auto &cte_sel = GetChartTableEntry(cur_max_i);
2867 *clat = (cte_sel.GetLatMax() + cte_sel.GetLatMin()) / 2.;
2868 *clon = (cte_sel.GetLonMin() + cte_sel.GetLonMax()) / 2.;
2876int ChartDatabase::GetDBChartProj(
int dbIndex) {
2877 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2878 auto &cte = GetChartTableEntry(dbIndex);
2879 return cte.GetChartProjectionType();
2881 return PROJECTION_UNKNOWN;
2887int ChartDatabase::GetDBChartFamily(
int dbIndex) {
2888 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2889 auto &cte = GetChartTableEntry(dbIndex);
2890 return cte.GetChartFamily();
2892 return CHART_FAMILY_UNKNOWN;
2898wxString ChartDatabase::GetDBChartFileName(
int dbIndex) {
2899 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2900 auto &cte = GetChartTableEntry(dbIndex);
2901 return wxString(cte.GetFullSystemPath());
2909int ChartDatabase::GetDBChartType(
int dbIndex) {
2910 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2911 auto &cte = GetChartTableEntry(dbIndex);
2912 return cte.GetChartType();
2920float ChartDatabase::GetDBChartSkew(
int dbIndex) {
2921 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2922 auto &cte = GetChartTableEntry(dbIndex);
2923 return cte.GetChartSkew();
2931int ChartDatabase::GetDBChartScale(
int dbIndex) {
2932 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2933 auto &cte = GetChartTableEntry(dbIndex);
2934 return cte.GetScale();
2942bool ChartDatabase::GetDBBoundingBox(
int dbIndex, LLBBox &box) {
2943 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2944 auto &entry = GetChartTableEntry(dbIndex);
2945 box.Set(entry.GetLatMin(), entry.GetLonMin(), entry.GetLatMax(),
2952const LLBBox &ChartDatabase::GetDBBoundingBox(
int dbIndex) {
2953 if ((bValid) && (dbIndex >= 0)) {
2954 auto &entry = GetChartTableEntry(dbIndex);
2955 return entry.GetBBox();
2957 return m_dummy_bbox;
2964int ChartDatabase::GetDBPlyPoint(
int dbIndex,
int plyindex,
float *lat,
2966 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2968 if (entry.GetnPlyEntries()) {
2969 float *fp = entry.GetpPlyTable();
2971 if (lat) *lat = *fp;
2973 if (lon) *lon = *fp;
2975 return entry.GetnPlyEntries();
2983int ChartDatabase::GetDBAuxPlyPoint(
int dbIndex,
int plyindex,
int ply,
2984 float *lat,
float *lon) {
2985 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
2987 if (entry.GetnAuxPlyEntries()) {
2988 float *fp = entry.GetpAuxPlyTableEntry(ply);
2991 if (lat) *lat = *fp;
2993 if (lon) *lon = *fp;
2996 return entry.GetAuxCntTableEntry(ply);
3001int ChartDatabase::GetnAuxPlyEntries(
int dbIndex) {
3002 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
3004 return entry.GetnAuxPlyEntries();
3012std::vector<float> ChartDatabase::GetReducedPlyPoints(
int dbIndex) {
3013 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
3015 return cte.GetReducedPlyPoints();
3018 std::vector<float> dummy;
3025std::vector<float> ChartDatabase::GetReducedAuxPlyPoints(
int dbIndex,
3027 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
3028 auto &cte = GetChartTableEntry(dbIndex);
3029 return cte.GetReducedAuxPlyPoints(iTable);
3032 std::vector<float> dummy;
3036bool ChartDatabase::IsChartAvailable(
int dbIndex) {
3037 if ((bValid) && (dbIndex >= 0) && (dbIndex < (
int)active_chartTable.size())) {
3038 auto &cte = GetChartTableEntry(dbIndex);
3041 if (cte.GetChartType() != CHART_TYPE_PLUGIN)
return true;
3043 wxString *path = cte.GetpsFullPath();
3044 wxFileName fn(*path);
3045 wxString ext = fn.GetExt();
3047 wxString ext_upper = ext.MakeUpper();
3048 wxString ext_lower = ext.MakeLower();
3053 for (
auto &cd : m_ChartClassDescriptorArray) {
3054 if (cd.m_descriptor_type == PLUGIN_DESCRIPTOR) {
3055 wxString search_mask = cd.m_search_mask;
3057 if (search_mask == ext_upper) {
3060 if (search_mask == ext_lower) {
3063 if (path->Matches(search_mask)) {
3073void ChartDatabase::ApplyGroupArray(ChartGroupArray *pGroupArray) {
3074 wxString separator(wxFileName::GetPathSeparator());
3076 for (
unsigned int ic = 0; ic < active_chartTable.size(); ic++) {
3077 auto &cte = GetChartTableEntry(ic);
3079 cte.ClearGroupArray();
3081 wxString *chart_full_path = cte.GetpsFullPath();
3083 for (
unsigned int igroup = 0; igroup < pGroupArray->GetCount(); igroup++) {
3084 ChartGroup *pGroup = pGroupArray->Item(igroup);
3085 for (
const auto &elem : pGroup->m_element_array) {
3086 wxString element_root = elem.m_element_name;
3092 if (!chart_full_path->IsSameAs(element_root))
3093 element_root.Append(
3095 if (chart_full_path->StartsWith(element_root)) {
3097 for (
unsigned int k = 0; k < elem.m_missing_name_array.size(); k++) {
3098 const wxString &missing_item = elem.m_missing_name_array[k];
3099 if (chart_full_path->StartsWith(missing_item)) {
3100 if (chart_full_path->IsSameAs(
3106 if (wxDir::Exists(missing_item))
3115 if (b_add) cte.AddIntToGroupArray(igroup + 1);
General chart base definitions.
ChartDB * ChartData
Global instance.
Define threaded chart database classes.
ChartGroupArray * g_pGroupArray
Global instance.
Basic chart info storage.
ChartGroupArray * g_pGroupArray
Global instance.
Base class for all chart types.
Manages a database of charts, including reading, writing, and querying chart information.
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.
Wrapper class for plugin-based charts.
void Notify() override
Notify all listeners, no data supplied.
Class for computing hash of arbitrary length.
EventVar options_on_finalize_chartdbs
Notified when chartdbs async operations complete, to finalize settings.
EventVar on_finalize_chartdbs
Notified when chartdbs async operations complete, to reload charts.
Base class for OpenCPN plugins.
Hash of arbitrary length.
Misc GUI event vars, a singleton.
PlugInManager * g_pi_manager
Global instance.
PlugInManager and helper classes – Mostly gui parts (dialogs) and plugin API stuff.
ShapeBaseChartSet gShapeBasemap
global instance
Represents an entry in the chart table, containing information about a single chart.