80#include <wx/datetime.h>
82#include <wx/jsonval.h>
84#include <wx/progdlg.h>
88#include "model/track.h"
91#include "model/georef.h"
92#include "model/json_event.h"
93#include "model/nav_object_database.h"
94#include "model/navutil_base.h"
95#include "model/own_ship.h"
96#include "model/routeman.h"
97#include "model/select.h"
101std::vector<Track *> g_TrackList;
106#if defined(__UNIX__) && \
111 void Reset() { clock_gettime(CLOCK_REALTIME, &tp); }
115 clock_gettime(CLOCK_REALTIME, &tp_end);
116 return (tp_end.tv_sec - tp.tv_sec) * 1.e3 +
117 (tp_end.tv_nsec - tp.tv_nsec) / 1.e6;
125TrackPoint::TrackPoint(
double lat,
double lon, wxString ts)
126 : m_lat(lat), m_lon(lon), m_GPXTrkSegNo(1) {
130TrackPoint::TrackPoint(
double lat,
double lon, wxDateTime dt)
131 : m_lat(lat), m_lon(lon), m_GPXTrkSegNo(1) {
137 : m_lat(orig->m_lat), m_lon(orig->m_lon), m_GPXTrkSegNo(1) {
141TrackPoint::~TrackPoint() {}
144 wxDateTime CreateTimeX;
145 ParseGPXDateTime(CreateTimeX, wxString(m_stimestring.c_str()));
152 ts = dt.FormatISODate()
154 .Append(dt.FormatISOTime())
162 m_stimestring = ts.mb_str();
171double _distance2(vector2D &a, vector2D &b) {
172 return (a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y);
174double _distance(vector2D &a, vector2D &b) {
return sqrt(_distance2(a, b)); }
180 m_width = WIDTH_UNDEFINED;
181 m_style = wxPENSTYLE_INVALID;
183 m_GUID = pWayPointMan->CreateGUID(NULL);
184 m_bIsInLayer =
false;
187 m_TrackHyperlinkList =
new HyperlinkList;
188 m_HighlightedTrackPoint = -1;
192 for (
size_t i = 0; i < TrackPoints.size(); i++)
delete TrackPoints[i];
194 delete m_TrackHyperlinkList;
197#define TIMER_TRACK1 778
200EVT_TIMER(TIMER_TRACK1, ActiveTrack::OnTimerTrack)
204 m_TimerTrack.SetOwner(
this, TIMER_TRACK1);
208 SetPrecision(g_nTrackPrecision);
210 m_prev_time = wxInvalidDateTime;
211 m_lastStoredTP = NULL;
213 wxDateTime now = wxDateTime::Now();
215 trackPointState = firstPoint;
216 m_lastStoredTP = NULL;
220 m_CurrentTrackSeg = 0;
224ActiveTrack::~ActiveTrack() { Stop(); }
226void ActiveTrack::SetPrecision(
int prec) {
228 switch (m_nPrecision) {
230 m_allowedMaxAngle = 10;
231 m_allowedMaxXTE = 0.008;
233 m_minTrackpoint_delta = .004;
237 m_allowedMaxAngle = 10;
238 m_allowedMaxXTE = 0.004;
240 m_minTrackpoint_delta = .002;
244 m_allowedMaxAngle = 10;
245 m_allowedMaxXTE = 0.0015;
247 m_minTrackpoint_delta = .001;
253void ActiveTrack::Start(
void) {
256 m_TimerTrack.Start(1000, wxTIMER_CONTINUOUS);
261void ActiveTrack::Stop(
bool do_add_point) {
268 delta = DistGreatCircle(gLat, gLon, m_lastStoredTP->m_lat,
269 m_lastStoredTP->m_lon);
271 if (delta > m_minTrackpoint_delta) AddPointNow(
true);
280Track *ActiveTrack::DoExtendDaily() {
281 Track *pExtendTrack = NULL;
287 for (
Track *ptrack : g_TrackList) {
288 if (!ptrack->m_bIsInLayer && ptrack->m_GUID != m_GUID) {
289 TrackPoint *track_node = ptrack->GetLastPoint();
295 pExtendPoint = track_node;
296 pExtendTrack = ptrack;
301 if (pExtendTrack && pExtendTrack->GetPoint(0)
303 .FromTimezone(wxDateTime::GMT0)
305 wxDateTime::GMT0))) {
308 pSelect->DeleteAllSelectableTrackSegments(pExtendTrack);
309 wxString suffix = _T(
"");
310 if (GetName().IsNull()) {
311 suffix = pExtendTrack->GetName();
312 if (suffix.IsNull()) suffix = wxDateTime::Today().FormatISODate();
314 pExtendTrack->Clone(
this, begin, GetnPoints(), suffix);
315 pSelect->AddAllSelectableTrackSegments(pExtendTrack);
316 pSelect->DeleteAllSelectableTrackSegments(
this);
320 if (GetName().IsNull()) SetName(wxDateTime::Today().FormatISODate());
325void Track::Clone(
Track *psourcetrack,
int start_nPoint,
int end_nPoint,
326 const wxString &suffix) {
327 if (psourcetrack->m_bIsInLayer)
return;
329 m_TrackNameString = psourcetrack->m_TrackNameString + suffix;
330 m_TrackStartString = psourcetrack->m_TrackStartString;
331 m_TrackEndString = psourcetrack->m_TrackEndString;
333 bool b_splitting = GetnPoints() == 0;
337 startTrkSegNo = psourcetrack->GetPoint(start_nPoint)->m_GPXTrkSegNo;
339 startTrkSegNo = GetLastPoint()->m_GPXTrkSegNo;
342 for (i = start_nPoint; i <= end_nPoint; i++) {
343 TrackPoint *psourcepoint = psourcetrack->GetPoint(i);
347 AddPoint(ptargetpoint);
352void ActiveTrack::AdjustCurrentTrackPoint(
TrackPoint *prototype) {
354 *m_lastStoredTP = *prototype;
359void ActiveTrack::OnTimerTrack(wxTimerEvent &event) {
364 m_prev_dist = DistGreatCircle(gLat, gLon, m_lastStoredTP->m_lat,
365 m_lastStoredTP->m_lon);
369 bool b_addpoint =
false;
371 if ((m_TrackTimerSec > 0.) && ((
double)m_track_run >= m_TrackTimerSec) &&
372 (m_prev_dist > m_minTrackpoint_delta)) {
380 if ((trackPointState == firstPoint) && !g_bTrackDaily) {
381 wxDateTime now = wxDateTime::Now();
382 if (TrackPoints.empty()) TrackPoints.front()->SetCreateTime(now.ToUTC());
385 m_TimerTrack.Start(1000, wxTIMER_CONTINUOUS);
388void ActiveTrack::AddPointNow(
bool do_add_point) {
389 wxDateTime now = wxDateTime::Now();
391 if (m_prev_dist < 0.0005)
392 if (!do_add_point)
return;
394 if (m_prev_time.IsValid())
395 if (m_prev_time == now)
396 if (!do_add_point)
return;
398 vector2D gpsPoint(gLon, gLat);
405 if (g_trackFilterMax) {
406 if (trackPointState == potentialPoint) {
407 double distToLastGpsPoint = DistLoxodrome(
408 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, gLat, gLon);
409 if (distToLastGpsPoint > g_trackFilterMax)
return;
417 switch (trackPointState) {
419 TrackPoint *pTrackPoint = AddNewPoint(gpsPoint, now.ToUTC());
420 m_lastStoredTP = pTrackPoint;
421 trackPointState = secondPoint;
422 do_add_point =
false;
426 vector2D pPoint(gLon, gLat);
427 skipPoints.push_back(pPoint);
428 skipTimes.push_back(now.ToUTC());
429 trackPointState = potentialPoint;
432 case potentialPoint: {
433 if (gpsPoint == skipPoints[skipPoints.size() - 1])
break;
435 unsigned int xteMaxIndex = 0;
440 for (
unsigned int i = 0; i < skipPoints.size(); i++) {
441 double xte = GetXTE(m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, gLat,
442 gLon, skipPoints[i].lat, skipPoints[i].lon);
448 if (xteMax > m_allowedMaxXTE) {
450 AddNewPoint(skipPoints[xteMaxIndex], skipTimes[xteMaxIndex]);
451 pSelect->AddSelectableTrackSegment(
452 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, pTrackPoint->m_lat,
453 pTrackPoint->m_lon, m_lastStoredTP, pTrackPoint,
this);
455 m_prevFixedTP = m_fixedTP;
456 m_fixedTP = m_removeTP;
457 m_removeTP = m_lastStoredTP;
458 m_lastStoredTP = pTrackPoint;
459 for (
unsigned int i = 0; i <= xteMaxIndex; i++) {
460 skipPoints.pop_front();
461 skipTimes.pop_front();
468 if (GetnPoints() > 2) {
470 DistGreatCircle(m_fixedTP->m_lat, m_fixedTP->m_lon,
471 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon);
472 double xte = GetXTE(m_fixedTP, m_lastStoredTP, m_removeTP);
473 if (xte < m_allowedMaxXTE / wxMax(1.0, 2.0 - dist * 2.0)) {
474 TrackPoints.pop_back();
475 TrackPoints.pop_back();
476 TrackPoints.push_back(m_lastStoredTP);
477 pSelect->DeletePointSelectableTrackSegments(m_removeTP);
478 pSelect->AddSelectableTrackSegment(
479 m_fixedTP->m_lat, m_fixedTP->m_lon, m_lastStoredTP->m_lat,
480 m_lastStoredTP->m_lon, m_fixedTP, m_lastStoredTP,
this);
482 m_removeTP = m_fixedTP;
483 m_fixedTP = m_prevFixedTP;
488 skipPoints.push_back(gpsPoint);
489 skipTimes.push_back(now.ToUTC());
496 TrackPoint *pTrackPoint = AddNewPoint(gpsPoint, now.ToUTC());
497 pSelect->AddSelectableTrackSegment(
498 m_lastStoredTP->m_lat, m_lastStoredTP->m_lon, pTrackPoint->m_lat,
499 pTrackPoint->m_lon, m_lastStoredTP, pTrackPoint,
this);
505void Track::ClearHighlights() { m_HighlightedTrackPoint = -1; }
508 if (nWhichPoint < (
int)TrackPoints.size())
509 return TrackPoints[nWhichPoint];
515 if (TrackPoints.empty())
return NULL;
517 return TrackPoints.back();
520static double heading_diff(
double x) {
521 if (x > 180)
return 360 - x;
522 if (x < -180)
return -360 + x;
530double Track::ComputeScale(
int left,
int right) {
531 const double z = WGS84_semimajor_axis_meters * mercator_k0;
532 const double mult = DEGREE * z;
537 double lata = TrackPoints[left]->m_lat, lona = TrackPoints[left]->m_lon;
538 double latb = TrackPoints[right]->m_lat, lonb = TrackPoints[right]->m_lon;
540 double bx = heading_diff(lonb - lona), by = latb - lata;
542 double lengthSquared = bx * bx + by * by;
546 if (lengthSquared > 3)
return INFINITY;
548 if (lengthSquared == 0.0) {
549 for (
int i = left + 1; i < right; i++) {
550 double lat = TrackPoints[i]->m_lat, lon = TrackPoints[i]->m_lon;
552 double vx = heading_diff(lon - lona);
553 double vy = lat - lata;
554 double dist = vx * vx + vy * vy;
556 if (dist > max_dist) max_dist = dist;
559 double invLengthSquared = 1 / lengthSquared;
560 for (
int i = left + 1; i < right; i++) {
561 double lat = TrackPoints[i]->m_lat, lon = TrackPoints[i]->m_lon;
563 double vx = heading_diff(lon - lona);
564 double vy = lat - lata;
565 double t = (vx * bx + vy * by) * invLengthSquared;
569 dist = vx * vx + vy * vy;
571 double wx = heading_diff(lona - lon);
572 double wy = lata - lat;
573 dist = wx * wx + wy * wy;
575 double projx = vx - t * bx;
576 double projy = vy - t * by;
577 dist = projx * projx + projy * projy;
580 if (dist > max_dist) max_dist = dist;
584 return max_dist * mult * mult;
591 TrackPoints.push_back(pNewPoint);
596void Track::Finalize() {
597 if (SubTracks.size())
602 int n = TrackPoints.size() - 1;
605 std::vector<SubTrack> new_level;
608 for (
int i = 0; i < n; i++) {
609 new_level[i].m_box.SetFromSegment(
610 TrackPoints[i]->m_lat, TrackPoints[i]->m_lon,
611 TrackPoints[i + 1]->m_lat, TrackPoints[i + 1]->m_lon);
612 new_level[i].m_scale = 0;
615 for (
int i = 0; i < n; i++) {
617 new_level[i].m_box = SubTracks[level - 1][p].m_box;
618 if (p + 1 < (
int)SubTracks[level - 1].size())
619 new_level[i].m_box.Expand(SubTracks[level - 1][p + 1].m_box);
621 int left = i << level;
622 int right = wxMin(left + (1 << level), TrackPoints.size() - 1);
623 new_level[i].m_scale = ComputeScale(left, right);
626 SubTracks.push_back(new_level);
628 if (n > 1 && n & 1) n++;
637void Track::InsertSubTracks(LLBBox &box,
int level,
int pos) {
638 if (level == (
int)SubTracks.size()) {
639 std::vector<SubTrack> new_level;
640 if (level > 0) box.Expand(SubTracks[level - 1][0].m_box);
642 new_level[pos].m_box = box;
643 SubTracks.push_back(new_level);
644 }
else if (pos < (
int)SubTracks[level].size())
645 SubTracks[level][pos].m_box.Expand(box);
647 SubTracks[level].push_back(
SubTrack());
648 SubTracks[level][pos].m_box = box;
652 SubTracks[level][pos].m_scale = 0;
654 int left = pos << level;
655 int right = wxMin(left + (1 << level), TrackPoints.size() - 1);
656 SubTracks[level][pos].m_scale = ComputeScale(left, right);
659 if (pos > 0) InsertSubTracks(box, level + 1, pos >> 1);
668void Track::AddPointFinalized(
TrackPoint *pNewPoint) {
669 TrackPoints.push_back(pNewPoint);
671 int pos = TrackPoints.size() - 1;
675 box.SetFromSegment(TrackPoints[pos - 1]->m_lat, TrackPoints[pos - 1]->m_lon,
676 TrackPoints[pos]->m_lat, TrackPoints[pos]->m_lon);
677 InsertSubTracks(box, 0, pos - 1);
681TrackPoint *Track::AddNewPoint(vector2D point, wxDateTime time) {
684 AddPointFinalized(tPoint);
689 NavObj_dB::GetInstance().AddTrackPoint(
this, tPoint);
693 v[
"lat"] = tPoint->m_lat;
694 v[
"lon"] = tPoint->m_lon;
695 v[
"Track_ID"] = m_GUID;
696 std::string msg_id(
"OCPN_TRK_POINT_ADDED");
697 JsonEvent::getInstance().Notify(msg_id, std::make_shared<wxJSONValue>(v));
702void Track::DouglasPeuckerReducer(std::vector<TrackPoint *> &list,
703 std::vector<bool> &keeplist,
int from,
int to,
705 keeplist[from] =
true;
708 int maxdistIndex = -1;
711 for (
int i = from + 1; i < to; i++) {
712 double dist = 1852.0 * GetXTE(list[from], list[to], list[i]);
714 if (dist > maxdist) {
720 if (maxdist > delta) {
721 DouglasPeuckerReducer(list, keeplist, from, maxdistIndex, delta);
722 DouglasPeuckerReducer(list, keeplist, maxdistIndex, to, delta);
726double Track::Length() {
729 for (
size_t i = 0; i < TrackPoints.size(); i++) {
732 const double offsetLat = 1e-6;
733 const double deltaLat = l->m_lat - t->m_lat;
734 if (fabs(deltaLat) > offsetLat)
735 total += DistGreatCircle(l->m_lat, l->m_lon, t->m_lat, t->m_lon);
737 total += DistGreatCircle(l->m_lat + copysign(offsetLat, deltaLat),
738 l->m_lon, t->m_lat, t->m_lon);
746int Track::Simplify(
double maxDelta) {
749 std::vector<TrackPoint *> pointlist;
750 std::vector<bool> keeplist;
752 ::wxBeginBusyCursor();
754 for (
size_t i = 0; i < TrackPoints.size(); i++) {
757 pointlist.push_back(trackpoint);
758 keeplist.push_back(
false);
761 DouglasPeuckerReducer(pointlist, keeplist, 0, pointlist.size() - 1, maxDelta);
763 pSelect->DeleteAllSelectableTrackSegments(
this);
767 for (
size_t i = 0; i < pointlist.size(); i++) {
769 TrackPoints.push_back(pointlist[i]);
777 pSelect->AddAllSelectableTrackSegments(
this);
784Route *Track::RouteFromTrack(wxGenericProgressDialog *pprog) {
793 wxString icon = _T(
"xmblue");
794 if (g_TrackDeltaDistance >= 0.1) icon = _T(
"diamond");
798 int nPoints = TrackPoints.size();
799 bool isProminent =
true;
800 double delta_dist = 0.;
801 double delta_hdg, xte;
802 double leg_speed = 0.1;
804 leg_speed = g_PlanSpeed;
808 pWP_dst =
new RoutePoint(pWP_src->m_lat, pWP_src->m_lon, icon, _T (
"" ),
810 route->AddPoint(pWP_dst);
814 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon, pWP_dst);
818 for (
size_t i = 1; i < TrackPoints.size();) {
821 pWP_dst->m_lat = pWP_prev->m_lat;
822 pWP_dst->m_lon = pWP_prev->m_lon;
829 DistanceBearingMercator(prp->m_lat, prp->m_lon, pWP_prev->m_lat,
830 pWP_prev->m_lon, &delta_hdg, &delta_dist);
832 if ((delta_dist > (leg_speed * 6.0)) && !prp_OK) {
833 int delta_inserts = floor(delta_dist / (leg_speed * 4.0));
834 delta_dist = delta_dist / (delta_inserts + 1);
838 while (delta_inserts--) {
839 ll_gc_ll(pWP_prev->m_lat, pWP_prev->m_lon, delta_hdg, delta_dist, &tlat,
841 pWP_dst =
new RoutePoint(tlat, tlon, icon, _T (
"" ), wxEmptyString);
842 route->AddPoint(pWP_dst);
844 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon,
847 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
848 pWP_dst->m_lat, pWP_dst->m_lon,
849 pWP_prev, pWP_dst, route);
862 if (delta_dist >= (leg_speed * 4.0)) isProminent =
true;
863 if (!prp_OK) prp_OK = prp;
865 while (prpnodeX < TrackPoints.size()) {
868 xte = GetXTE(pWP_src, prpX, prp);
869 if (isProminent || (xte > g_TrackDeltaDistance)) {
870 pWP_dst =
new RoutePoint(prp_OK->m_lat, prp_OK->m_lon, icon, _T (
"" ),
873 route->AddPoint(pWP_dst);
876 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon,
879 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
880 pWP_dst->m_lat, pWP_dst->m_lon,
881 pWP_prev, pWP_dst, route);
885 prpnodeX = TrackPoints.size();
889 if (prpnodeX != TrackPoints.size()) prpnodeX--;
890 if (back_ic-- <= 0) {
891 prpnodeX = TrackPoints.size();
899 DistanceBearingMercator(prp->m_lat, prp->m_lon, pWP_prev->m_lat,
900 pWP_prev->m_lon, NULL, &delta_dist);
902 if (!((delta_dist > (g_TrackDeltaDistance)) && !prp_OK)) {
906 int iProg = (i * 100) / nPoints;
907 if (pprog && (iProg > dProg)) {
909 pprog->Update(dProg);
914 if (delta_dist >= g_TrackDeltaDistance) {
916 new RoutePoint(TrackPoints.back()->m_lat, TrackPoints.back()->m_lon,
917 icon, _T (
"" ), wxEmptyString);
918 route->AddPoint(pWP_dst);
922 pSelect->AddSelectableRoutePoint(pWP_dst->m_lat, pWP_dst->m_lon, pWP_dst);
924 pSelect->AddSelectableRouteSegment(pWP_prev->m_lat, pWP_prev->m_lon,
925 pWP_dst->m_lat, pWP_dst->m_lon, pWP_prev,
936double Track::GetXTE(
double fm1Lat,
double fm1Lon,
double fm2Lat,
double fm2Lon,
937 double toLat,
double toLon) {
943 double brg1, dist1, brg2, dist2;
944 DistanceBearingMercator(toLat, toLon, fm1Lat, fm1Lon, &brg1, &dist1);
945 w.x = dist1 * sin(brg1 * PI / 180.);
946 w.y = dist1 * cos(brg1 * PI / 180.);
948 DistanceBearingMercator(toLat, toLon, fm2Lat, fm2Lon, &brg2, &dist2);
949 v.x = dist2 * sin(brg2 * PI / 180.);
950 v.y = dist2 * cos(brg2 * PI / 180.);
955 const double lengthSquared = _distance2(v, w);
956 if (lengthSquared == 0.0) {
958 return _distance(p, v);
968 double t = vDotProduct(&a, &b) / lengthSquared;
971 return _distance(p, v);
973 return _distance(p, w);
974 vector2D projection = v + t * (w - v);
975 return _distance(p, projection);
979 if (!fm1 || !fm2 || !to)
return 0.0;
980 if (fm1 == to)
return 0.0;
981 if (fm2 == to)
return 0.0;
982 return GetXTE(fm1->m_lat, fm1->m_lon, fm2->m_lat, fm2->m_lon, to->m_lat,
987wxString Track::GetIsoDateTime(
const wxString label_for_invalid_date)
const {
990 if ((
int)TrackPoints.size() > 0) rp = TrackPoints[0];
994 name = label_for_invalid_date;
998wxString Track::GetDateTime(
const wxString label_for_invalid_date)
const {
1001 if ((
int)TrackPoints.size() > 0) rp = TrackPoints[0];
1003 name = ocpn::toUsrDateTimeFormat(rp->
GetCreateTime().FromUTC());
1005 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.
PlugIn Object Definition/API.