75#include <wx/datetime.h>
77#include <wx/jsonval.h>
79#include <wx/progdlg.h>
88#include "model/nav_object_database.h"
102#if defined(__UNIX__) && !defined(__WXOSX__)
106 void Reset() { clock_gettime(CLOCK_REALTIME, &tp); }
110 clock_gettime(CLOCK_REALTIME, &tp_end);
111 return (tp_end.tv_sec - tp.tv_sec) * 1.e3 +
112 (tp_end.tv_nsec - tp.tv_nsec) / 1.e6;
120TrackPoint::TrackPoint(
double lat,
double lon, wxString ts)
121 : m_lat(lat), m_lon(lon), m_GPXTrkSegNo(1) {
125TrackPoint::TrackPoint(
double lat,
double lon, wxDateTime dt)
126 : m_lat(lat), m_lon(lon), m_GPXTrkSegNo(1) {
132 : m_lat(orig->m_lat), m_lon(orig->m_lon), m_GPXTrkSegNo(1) {
136TrackPoint::~TrackPoint() {}
139 wxDateTime CreateTimeX;
147 ts = dt.FormatISODate().Append(
"T").Append(dt.FormatISOTime()).Append(
"Z");
154 m_stimestring = ts.mb_str();
163double _distance2(vector2D &a, vector2D &b) {
164 return (a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y);
166double _distance(vector2D &a, vector2D &b) {
return sqrt(_distance2(a, b)); }
172 m_width = WIDTH_UNDEFINED;
173 m_style = wxPENSTYLE_INVALID;
175 m_GUID = pWayPointMan->CreateGUID(NULL);
176 m_bIsInLayer =
false;
179 m_TrackHyperlinkList =
new HyperlinkList;
180 m_HighlightedTrackPoint = -1;
184 for (
size_t i = 0; i < TrackPoints.size(); i++)
delete TrackPoints[i];
186 delete m_TrackHyperlinkList;
189#define TIMER_TRACK1 778
192EVT_TIMER(TIMER_TRACK1, ActiveTrack::OnTimerTrack)
196 m_TimerTrack.SetOwner(
this, TIMER_TRACK1);
200 SetPrecision(g_nTrackPrecision);
202 m_prev_time = wxInvalidDateTime;
203 m_lastStoredTP = NULL;
205 wxDateTime now = wxDateTime::Now();
207 trackPointState = firstPoint;
208 m_lastStoredTP = NULL;
212 m_CurrentTrackSeg = 0;
216ActiveTrack::~ActiveTrack() { Stop(); }
218void ActiveTrack::SetPrecision(
int prec) {
220 switch (m_nPrecision) {
222 m_allowedMaxAngle = 10;
223 m_allowedMaxXTE = 0.008;
225 m_minTrackpoint_delta = .004;
229 m_allowedMaxAngle = 10;
230 m_allowedMaxXTE = 0.004;
232 m_minTrackpoint_delta = .002;
236 m_allowedMaxAngle = 10;
237 m_allowedMaxXTE = 0.0015;
239 m_minTrackpoint_delta = .001;
245void ActiveTrack::Start() {
248 m_TimerTrack.Start(1000, wxTIMER_CONTINUOUS);
253void ActiveTrack::Stop(
bool do_add_point) {
260 delta = DistGreatCircle(
gLat,
gLon, m_lastStoredTP->m_lat,
261 m_lastStoredTP->m_lon);
263 if (delta > m_minTrackpoint_delta) AddPointNow(
true);
272Track *ActiveTrack::DoExtendDaily() {
273 Track *pExtendTrack = NULL;
280 if (!ptrack->m_bIsInLayer && ptrack->m_GUID != m_GUID) {
282 if (ptrack->GetName().StartsWith((
"AIS")))
continue;
284 TrackPoint *track_node = ptrack->GetLastPoint();
290 pExtendPoint = track_node;
291 pExtendTrack = ptrack;
296 if (pExtendTrack && pExtendTrack->GetPoint(0)
298 .FromTimezone(wxDateTime::GMT0)
300 wxDateTime::GMT0))) {
303 pSelect->DeleteAllSelectableTrackSegments(pExtendTrack);
304 wxString suffix =
"";
305 if (GetName().IsNull()) {
306 suffix = pExtendTrack->GetName();
307 if (suffix.IsNull()) suffix = wxDateTime::Today().FormatISODate();
309 pExtendTrack->Clone(
this, begin, GetnPoints(), suffix);
310 pSelect->AddAllSelectableTrackSegments(pExtendTrack);
311 pSelect->DeleteAllSelectableTrackSegments(
this);
315 if (GetName().IsNull()) SetName(wxDateTime::Today().FormatISODate());
320void Track::Clone(
Track *psourcetrack,
int start_nPoint,
int end_nPoint,
321 const wxString &suffix) {
322 if (psourcetrack->m_bIsInLayer)
return;
324 m_TrackNameString = psourcetrack->m_TrackNameString + suffix;
325 m_TrackStartString = psourcetrack->m_TrackStartString;
326 m_TrackEndString = psourcetrack->m_TrackEndString;
328 bool b_splitting = GetnPoints() == 0;
332 startTrkSegNo = psourcetrack->GetPoint(start_nPoint)->m_GPXTrkSegNo;
334 startTrkSegNo = GetLastPoint()->m_GPXTrkSegNo;
337 for (i = start_nPoint; i <= end_nPoint; i++) {
338 TrackPoint *psourcepoint = psourcetrack->GetPoint(i);
342 AddPoint(ptargetpoint);
347void ActiveTrack::AdjustCurrentTrackPoint(
TrackPoint *prototype) {
349 *m_lastStoredTP = *prototype;
354void ActiveTrack::OnTimerTrack(wxTimerEvent &event) {
359 m_prev_dist = DistGreatCircle(
gLat,
gLon, m_lastStoredTP->m_lat,
360 m_lastStoredTP->m_lon);
364 bool b_addpoint =
false;
366 if ((m_TrackTimerSec > 0.) && ((
double)m_track_run >= m_TrackTimerSec) &&
367 (m_prev_dist > m_minTrackpoint_delta)) {
375 if ((trackPointState == firstPoint) && !g_bTrackDaily) {
376 wxDateTime now = wxDateTime::Now();
377 if (TrackPoints.empty()) TrackPoints.front()->SetCreateTime(now.ToUTC());
380 m_TimerTrack.Start(1000, wxTIMER_CONTINUOUS);
383void ActiveTrack::AddPointNow(
bool do_add_point) {
384 wxDateTime now = wxDateTime::Now();
386 if (m_prev_dist < 0.0005)
387 if (!do_add_point)
return;
389 if (m_prev_time.IsValid())
390 if (m_prev_time == now)
391 if (!do_add_point)
return;
400 if (g_trackFilterMax) {
401 if (trackPointState == potentialPoint) {
402 double distToLastGpsPoint = DistLoxodrome(
403 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon,
gLat,
gLon);
404 if (distToLastGpsPoint > g_trackFilterMax)
return;
412 switch (trackPointState) {
414 TrackPoint *pTrackPoint = AddNewPoint(gpsPoint, now.ToUTC());
415 m_lastStoredTP = pTrackPoint;
416 trackPointState = secondPoint;
417 do_add_point =
false;
422 skipPoints.push_back(pPoint);
423 skipTimes.push_back(now.ToUTC());
424 trackPointState = potentialPoint;
427 case potentialPoint: {
428 if (gpsPoint == skipPoints[skipPoints.size() - 1])
break;
430 unsigned int xteMaxIndex = 0;
435 for (
unsigned int i = 0; i < skipPoints.size(); i++) {
436 double xte = GetXTE(m_lastStoredTP->m_lat, m_lastStoredTP->m_lon,
gLat,
437 gLon, skipPoints[i].lat, skipPoints[i].lon);
443 if (xteMax > m_allowedMaxXTE) {
445 AddNewPoint(skipPoints[xteMaxIndex], skipTimes[xteMaxIndex]);
446 pSelect->AddSelectableTrackSegment(
447 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, pTrackPoint->m_lat,
448 pTrackPoint->m_lon, m_lastStoredTP, pTrackPoint,
this);
450 m_prevFixedTP = m_fixedTP;
451 m_fixedTP = m_removeTP;
452 m_removeTP = m_lastStoredTP;
453 m_lastStoredTP = pTrackPoint;
454 for (
unsigned int i = 0; i <= xteMaxIndex; i++) {
455 skipPoints.pop_front();
456 skipTimes.pop_front();
463 if (GetnPoints() > 2) {
465 DistGreatCircle(m_fixedTP->m_lat, m_fixedTP->m_lon,
466 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon);
467 double xte = GetXTE(m_fixedTP, m_lastStoredTP, m_removeTP);
468 if (xte < m_allowedMaxXTE / wxMax(1.0, 2.0 - dist * 2.0)) {
469 TrackPoints.pop_back();
470 TrackPoints.pop_back();
471 TrackPoints.push_back(m_lastStoredTP);
472 pSelect->DeletePointSelectableTrackSegments(m_removeTP);
473 pSelect->AddSelectableTrackSegment(
474 m_fixedTP->m_lat, m_fixedTP->m_lon, m_lastStoredTP->m_lat,
475 m_lastStoredTP->m_lon, m_fixedTP, m_lastStoredTP,
this);
477 m_removeTP = m_fixedTP;
478 m_fixedTP = m_prevFixedTP;
483 skipPoints.push_back(gpsPoint);
484 skipTimes.push_back(now.ToUTC());
491 TrackPoint *pTrackPoint = AddNewPoint(gpsPoint, now.ToUTC());
492 pSelect->AddSelectableTrackSegment(
493 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, pTrackPoint->m_lat,
494 pTrackPoint->m_lon, m_lastStoredTP, pTrackPoint,
this);
500void Track::ClearHighlights() { m_HighlightedTrackPoint = -1; }
503 if (nWhichPoint < (
int)TrackPoints.size())
504 return TrackPoints[nWhichPoint];
510 if (TrackPoints.empty())
return NULL;
512 return TrackPoints.back();
515static double heading_diff(
double x) {
516 if (x > 180)
return 360 - x;
517 if (x < -180)
return -360 + x;
525double Track::ComputeScale(
int left,
int right) {
526 const double z = WGS84_semimajor_axis_meters * mercator_k0;
527 const double mult = DEGREE * z;
532 double lata = TrackPoints[left]->m_lat, lona = TrackPoints[left]->m_lon;
533 double latb = TrackPoints[right]->m_lat, lonb = TrackPoints[right]->m_lon;
535 double bx = heading_diff(lonb - lona), by = latb - lata;
537 double lengthSquared = bx * bx + by * by;
541 if (lengthSquared > 3)
return INFINITY;
543 if (lengthSquared == 0.0) {
544 for (
int i = left + 1; i < right; i++) {
545 double lat = TrackPoints[i]->m_lat, lon = TrackPoints[i]->m_lon;
547 double vx = heading_diff(lon - lona);
548 double vy = lat - lata;
549 double dist = vx * vx + vy * vy;
551 if (dist > max_dist) max_dist = dist;
554 double invLengthSquared = 1 / lengthSquared;
555 for (
int i = left + 1; i < right; i++) {
556 double lat = TrackPoints[i]->m_lat, lon = TrackPoints[i]->m_lon;
558 double vx = heading_diff(lon - lona);
559 double vy = lat - lata;
560 double t = (vx * bx + vy * by) * invLengthSquared;
564 dist = vx * vx + vy * vy;
566 double wx = heading_diff(lona - lon);
567 double wy = lata - lat;
568 dist = wx * wx + wy * wy;
570 double projx = vx - t * bx;
571 double projy = vy - t * by;
572 dist = projx * projx + projy * projy;
575 if (dist > max_dist) max_dist = dist;
579 return max_dist * mult * mult;
586 TrackPoints.push_back(pNewPoint);
591void Track::Finalize() {
592 if (SubTracks.size())
597 int n = TrackPoints.size() - 1;
600 std::vector<SubTrack> new_level;
603 for (
int i = 0; i < n; i++) {
604 new_level[i].m_box.SetFromSegment(
605 TrackPoints[i]->m_lat, TrackPoints[i]->m_lon,
606 TrackPoints[i + 1]->m_lat, TrackPoints[i + 1]->m_lon);
607 new_level[i].m_scale = 0;
610 for (
int i = 0; i < n; i++) {
612 new_level[i].m_box = SubTracks[level - 1][p].m_box;
613 if (p + 1 < (
int)SubTracks[level - 1].size())
614 new_level[i].m_box.Expand(SubTracks[level - 1][p + 1].m_box);
616 int left = i << level;
617 int right = wxMin(left + (1 << level), TrackPoints.size() - 1);
618 new_level[i].m_scale = ComputeScale(left, right);
621 SubTracks.push_back(new_level);
623 if (n > 1 && n & 1) n++;
632void Track::InsertSubTracks(LLBBox &box,
int level,
int pos) {
633 if (level == (
int)SubTracks.size()) {
634 std::vector<SubTrack> new_level;
635 if (level > 0) box.Expand(SubTracks[level - 1][0].m_box);
637 new_level[pos].m_box = box;
638 SubTracks.push_back(new_level);
639 }
else if (pos < (
int)SubTracks[level].size())
640 SubTracks[level][pos].m_box.Expand(box);
642 SubTracks[level].push_back(
SubTrack());
643 SubTracks[level][pos].m_box = box;
647 SubTracks[level][pos].m_scale = 0;
649 int left = pos << level;
650 int right = wxMin(left + (1 << level), TrackPoints.size() - 1);
651 SubTracks[level][pos].m_scale = ComputeScale(left, right);
654 if (pos > 0) InsertSubTracks(box, level + 1, pos >> 1);
663void Track::AddPointFinalized(
TrackPoint *pNewPoint) {
664 TrackPoints.push_back(pNewPoint);
666 int pos = TrackPoints.size() - 1;
670 box.SetFromSegment(TrackPoints[pos - 1]->m_lat, TrackPoints[pos - 1]->m_lon,
671 TrackPoints[pos]->m_lat, TrackPoints[pos]->m_lon);
672 InsertSubTracks(box, 0, pos - 1);
676TrackPoint *Track::AddNewPoint(vector2D point, wxDateTime time) {
679 AddPointFinalized(tPoint);
684 NavObj_dB::GetInstance().AddTrackPoint(
this, tPoint);
688 v[
"lat"] = tPoint->m_lat;
689 v[
"lon"] = tPoint->m_lon;
690 v[
"Track_ID"] = m_GUID;
691 std::string msg_id(
"OCPN_TRK_POINT_ADDED");
692 JsonEvent::getInstance().Notify(msg_id, std::make_shared<wxJSONValue>(v));
697void Track::DouglasPeuckerReducer(std::vector<TrackPoint *> &list,
698 std::vector<bool> &keeplist,
int from,
int to,
700 keeplist[from] =
true;
703 int maxdistIndex = -1;
706 for (
int i = from + 1; i < to; i++) {
707 double dist = 1852.0 * GetXTE(list[from], list[to], list[i]);
709 if (dist > maxdist) {
715 if (maxdist > delta) {
716 DouglasPeuckerReducer(list, keeplist, from, maxdistIndex, delta);
717 DouglasPeuckerReducer(list, keeplist, maxdistIndex, to, delta);
721double Track::Length() {
724 for (
size_t i = 0; i < TrackPoints.size(); i++) {
727 const double offsetLat = 1e-6;
728 const double deltaLat = l->m_lat - t->m_lat;
729 if (fabs(deltaLat) > offsetLat)
730 total += DistGreatCircle(l->m_lat, l->m_lon, t->m_lat, t->m_lon);
732 total += DistGreatCircle(l->m_lat + copysign(offsetLat, deltaLat),
733 l->m_lon, t->m_lat, t->m_lon);
741int Track::Simplify(
double maxDelta) {
744 std::vector<TrackPoint *> pointlist;
745 std::vector<bool> keeplist;
747 ::wxBeginBusyCursor();
749 for (
size_t i = 0; i < TrackPoints.size(); i++) {
752 pointlist.push_back(trackpoint);
753 keeplist.push_back(
false);
756 DouglasPeuckerReducer(pointlist, keeplist, 0, pointlist.size() - 1, maxDelta);
758 pSelect->DeleteAllSelectableTrackSegments(
this);
762 for (
size_t i = 0; i < pointlist.size(); i++) {
764 TrackPoints.push_back(pointlist[i]);
772 pSelect->AddAllSelectableTrackSegments(
this);
779Route *Track::RouteFromTrack(wxGenericProgressDialog *pprog) {
788 wxString icon =
"xmblue";
789 if (g_TrackDeltaDistance >= 0.1) icon =
"diamond";
793 int nPoints = TrackPoints.size();
794 bool isProminent =
true;
795 double delta_dist = 0.;
796 double delta_hdg, xte;
797 double leg_speed = 0.1;
799 leg_speed = g_PlanSpeed;
803 pWP_dst =
new RoutePoint(pWP_src->m_lat, pWP_src->m_lon, icon,
"",
"");
804 route->AddPoint(pWP_dst);
808 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon, pWP_dst);
812 for (
size_t i = 1; i < TrackPoints.size();) {
815 pWP_dst->m_lat = pWP_prev->m_lat;
816 pWP_dst->m_lon = pWP_prev->m_lon;
823 DistanceBearingMercator(prp->m_lat, prp->m_lon, pWP_prev->m_lat,
824 pWP_prev->m_lon, &delta_hdg, &delta_dist);
826 if ((delta_dist > (leg_speed * 6.0)) && !prp_OK) {
827 int delta_inserts = floor(delta_dist / (leg_speed * 4.0));
828 delta_dist = delta_dist / (delta_inserts + 1);
832 while (delta_inserts--) {
833 ll_gc_ll(pWP_prev->m_lat, pWP_prev->m_lon, delta_hdg, delta_dist, &tlat,
835 pWP_dst =
new RoutePoint(tlat, tlon, icon,
"",
"");
836 route->AddPoint(pWP_dst);
838 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon,
841 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
842 pWP_dst->m_lat, pWP_dst->m_lon,
843 pWP_prev, pWP_dst, route);
856 if (delta_dist >= (leg_speed * 4.0)) isProminent =
true;
857 if (!prp_OK) prp_OK = prp;
859 while (prpnodeX < TrackPoints.size()) {
862 xte = GetXTE(pWP_src, prpX, prp);
863 if (isProminent || (xte > g_TrackDeltaDistance)) {
864 pWP_dst =
new RoutePoint(prp_OK->m_lat, prp_OK->m_lon, icon,
"",
"");
866 route->AddPoint(pWP_dst);
869 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon,
872 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
873 pWP_dst->m_lat, pWP_dst->m_lon,
874 pWP_prev, pWP_dst, route);
878 prpnodeX = TrackPoints.size();
882 if (prpnodeX != TrackPoints.size()) prpnodeX--;
883 if (back_ic-- <= 0) {
884 prpnodeX = TrackPoints.size();
892 DistanceBearingMercator(prp->m_lat, prp->m_lon, pWP_prev->m_lat,
893 pWP_prev->m_lon, NULL, &delta_dist);
895 if (!((delta_dist > (g_TrackDeltaDistance)) && !prp_OK)) {
899 int iProg = (i * 100) / nPoints;
900 if (pprog && (iProg > dProg)) {
902 pprog->Update(dProg);
907 if (delta_dist >= g_TrackDeltaDistance) {
908 pWP_dst =
new RoutePoint(TrackPoints.back()->m_lat,
909 TrackPoints.back()->m_lon, icon,
"",
"");
910 route->AddPoint(pWP_dst);
914 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon, pWP_dst);
916 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
917 pWP_dst->m_lat, pWP_dst->m_lon, pWP_prev,
928double Track::GetXTE(
double fm1Lat,
double fm1Lon,
double fm2Lat,
double fm2Lon,
929 double toLat,
double toLon) {
935 double brg1, dist1, brg2, dist2;
936 DistanceBearingMercator(toLat, toLon, fm1Lat, fm1Lon, &brg1, &dist1);
937 w.x = dist1 * sin(brg1 * PI / 180.);
938 w.y = dist1 * cos(brg1 * PI / 180.);
940 DistanceBearingMercator(toLat, toLon, fm2Lat, fm2Lon, &brg2, &dist2);
941 v.x = dist2 * sin(brg2 * PI / 180.);
942 v.y = dist2 * cos(brg2 * PI / 180.);
947 const double lengthSquared = _distance2(v, w);
948 if (lengthSquared == 0.0) {
950 return _distance(p, v);
960 double t = vDotProduct(&a, &b) / lengthSquared;
963 return _distance(p, v);
965 return _distance(p, w);
966 vector2D projection = v + t * (w - v);
967 return _distance(p, projection);
971 if (!fm1 || !fm2 || !to)
return 0.0;
972 if (fm1 == to)
return 0.0;
973 if (fm2 == to)
return 0.0;
974 return GetXTE(fm1->m_lat, fm1->m_lon, fm2->m_lat, fm2->m_lon, to->m_lat,
979wxString Track::GetIsoDateTime(
const wxString label_for_invalid_date)
const {
982 if ((
int)TrackPoints.size() > 0) rp = TrackPoints[0];
986 name = label_for_invalid_date;
990wxString Track::GetDateTime(
const wxString label_for_invalid_date)
const {
993 if ((
int)TrackPoints.size() > 0) rp = TrackPoints[0];
995 name = ocpn::toUsrDateTimeFormat(rp->
GetCreateTime().FromUTC());
997 name = label_for_invalid_date;
Represents an active track that is currently being recorded.
Represents a waypoint or mark within the navigation system.
bool m_bShowName
Flag indicating if the waypoint name should be shown.
Represents a navigational route in the navigation system.
wxString m_RouteStartString
Name or description of the route's starting point.
bool m_bDeleteOnArrival
Flag indicating whether the route should be deleted once navigation reaches the end.
wxString m_RouteEndString
Name or description of the route's ending point.
wxString m_RouteNameString
User-assigned name for the route.
Represents a single point in a track.
wxDateTime GetCreateTime(void)
Retrieves the creation timestamp of a track point as a wxDateTime object.
void SetCreateTime(wxDateTime dt)
Sets the creation timestamp for a track point.
Represents a track, which is a series of connected track points.
The JSON value class implementation.
Global variables stored in configuration file.
JSON event definition used in internal communications to/from plugins.
MySQL based storage for routes, tracks, etc.
const wxChar * ParseGPXDateTime(wxDateTime &dt, const wxChar *datetime)
This function parses a string containing a GPX time representation and returns a wxDateTime containin...
Navigation Utility Functions without GUI dependencies.
PlugIn Object Definition/API.
double gLat
Vessel's current latitude in decimal degrees.
double gLon
Vessel's current longitude in decimal degrees.
Position, course, speed, etc.
Select * pSelect
Global instance.
Selected route, segment, waypoint, etc.
ActiveTrack * g_pActiveTrack
global instance
std::vector< Track * > g_TrackList
Global instance.
Recorded track abstraction.
std::vector< Track * > g_TrackList
Global instance.