diff --git a/routing/base/followed_polyline.cpp b/routing/base/followed_polyline.cpp index 7b668f126d..266ed981f3 100644 --- a/routing/base/followed_polyline.cpp +++ b/routing/base/followed_polyline.cpp @@ -126,6 +126,44 @@ Iter FollowedPolyline::GetBestProjection(m2::RectD const & posRect, return GetClosestProjectionInInterval(posRect, distFn, hoppingBorderIdx, m_nextCheckpointIndex); } + template + std::pair FollowedPolyline::GetBestMatchedProjection(m2::RectD const & posRect, + DistanceFn const & distFn) const + { + CHECK_EQUAL(m_segProj.size() + 1, m_poly.GetSize(), ()); + // At first trying to find a projection to two closest route segments of route which is close + // enough to |posRect| center. If m_current is right before intermediate point we can get closestIter + // right after intermediate point (in next subroute). + size_t const hoppingBorderIdx = min(m_segProj.size(), m_current.m_ind + 3); + Iter closestIter; + bool nearestIsFake = false; + std::tie(closestIter, nearestIsFake) = GetClosestMatchedProjectionInInterval(posRect, distFn, m_current.m_ind, + hoppingBorderIdx); + if (closestIter.IsValid()) + return std::make_pair(closestIter, nearestIsFake); + + // If a projection to the 3 closest route segments is not found tries to find projection to other route + // segments of current subroute. + return GetClosestMatchedProjectionInInterval(posRect, distFn, hoppingBorderIdx, m_nextCheckpointIndex); + } + +std::pair FollowedPolyline::UpdateMatchedProjection(m2::RectD const & posRect) +{ + ASSERT(m_current.IsValid(), ()); + ASSERT_LESS(m_current.m_ind, m_poly.GetSize() - 1, ()); + + Iter iter; + bool nearestIsFake = false; + m2::PointD const currPos = posRect.Center(); + std::tie(iter, nearestIsFake) = GetBestMatchedProjection(posRect, [&](Iter const &it) { + return MercatorBounds::DistanceOnEarth(it.m_pt, currPos); + }); + + if (iter.IsValid()) + m_current = iter; + return std::make_pair(iter.IsValid(), nearestIsFake); +} + Iter FollowedPolyline::UpdateProjectionByPrediction(m2::RectD const & posRect, double predictDistance) { @@ -163,6 +201,11 @@ Iter FollowedPolyline::UpdateProjection(m2::RectD const & posRect) return res; } +void FollowedPolyline::SetUnmatchedSegmentIndexes(std::vector const & unmatchedSegmentIndexes) +{ + m_unmatchedSegmentIndexes = unmatchedSegmentIndexes; +} + double FollowedPolyline::GetDistFromCurPointToRoutePointMerc() const { if (!m_current.IsValid()) diff --git a/routing/base/followed_polyline.hpp b/routing/base/followed_polyline.hpp index f753f3f97b..6967bec355 100644 --- a/routing/base/followed_polyline.hpp +++ b/routing/base/followed_polyline.hpp @@ -72,6 +72,13 @@ public: double GetDistanceM(Iter const & it1, Iter const & it2) const; Iter UpdateProjectionByPrediction(m2::RectD const & posRect, double predictDistance); + + /// \brief set indexes of all unmatched segments on route + void SetUnmatchedSegmentIndexes(std::vector const & unmatchedSegmentIndexes); + + /// \brief Updates projection to the closest real segment if possible + std::pair UpdateMatchedProjection(m2::RectD const & posRect); + Iter UpdateProjection(m2::RectD const & posRect); Iter Begin() const; @@ -115,6 +122,48 @@ public: return res; } + /// \returns pair of iterator (projection point) and bool (true if nearest point is on unmatched segment) + template + std::pair GetClosestMatchedProjectionInInterval(m2::RectD const & posRect, DistanceFn const & distFn, + size_t startIdx, size_t endIdx) const + { + CHECK_LESS_OR_EQUAL(endIdx, m_segProj.size(), ()); + CHECK_LESS_OR_EQUAL(startIdx, endIdx, ()); + Iter nearestIter; + double minDist = std::numeric_limits::max(); + double minDistUnmatched = minDist; + + m2::PointD const currPos = posRect.Center(); + + for (size_t i = startIdx; i < endIdx; ++i) + { + m2::PointD const & pt = m_segProj[i].ClosestPointTo(currPos); + + if (!posRect.IsPointInside(pt)) + continue; + + Iter it(pt, i); + double const dp = distFn(it); + if(dp >= minDistUnmatched && dp >= minDist) + continue; + + if (std::find(m_unmatchedSegmentIndexes.begin(), m_unmatchedSegmentIndexes.end(), it.m_ind) == m_unmatchedSegmentIndexes.end()) + { + if(minDist > dp) // overwrite best match for matched segment + { + minDist = dp; + nearestIter = it; + } + } + else + { + if(minDistUnmatched > dp) // overwrite best match for unmatched segment + minDistUnmatched = dp; + } + } + return std::make_pair(nearestIter, minDistUnmatched < minDist); + } + private: /// \returns iterator to the best projection of center of |posRect| to the |m_poly|. /// If there's a good projection of center of |posRect| to two closest segments of |m_poly| @@ -123,9 +172,14 @@ private: template Iter GetBestProjection(m2::RectD const & posRect, DistanceFn const & distFn) const; + template + std::pair GetBestMatchedProjection(m2::RectD const & posRect, + DistanceFn const & distFn) const; + void Update(); m2::PolylineD m_poly; + std::vector m_unmatchedSegmentIndexes; /// Iterator with the current position. Position sets with UpdateProjection methods. Iter m_current; diff --git a/routing/route.cpp b/routing/route.cpp index 23a35adae4..bcde2dc73c 100644 --- a/routing/route.cpp +++ b/routing/route.cpp @@ -247,6 +247,27 @@ bool Route::MoveIterator(location::GpsInfo const & info) return res.IsValid(); } +void Route::SetFakeSegmentsOnPolyline() +{ + std::vector fakeSegmentIndexes{}; + auto const & segs = GetRouteSegments(); + for(size_t i = 0; i < segs.size(); ++i) + { + if(!segs[i].GetSegment().IsRealSegment()) + fakeSegmentIndexes.push_back(i); + } + m_poly.SetUnmatchedSegmentIndexes(fakeSegmentIndexes); +} + +std::pair Route::MoveIteratorToReal(location::GpsInfo const & info) +{ + m2::RectD const rect = MercatorBounds::MetersToXY( + info.m_longitude, info.m_latitude, + max(m_routingSettings.m_matchingThresholdM, info.m_horizontalAccuracy)); + + return m_poly.UpdateMatchedProjection(rect); +} + double Route::GetPolySegAngle(size_t ind) const { size_t const polySz = m_poly.GetPolyline().GetSize(); diff --git a/routing/route.hpp b/routing/route.hpp index eae1194fd4..89c42c0568 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -256,7 +256,7 @@ public: if (s.GetJunction().GetAltitude() == feature::kInvalidAltitude) { m_haveAltitudes = false; - return; + break; } } } @@ -327,6 +327,10 @@ public: /// @return true If position was updated successfully (projection within gps error radius). bool MoveIterator(location::GpsInfo const & info); + /// @return pair of vals: first = true if position was updated successfully to real segment, + /// second = true if position is closer to the fake segment + std::pair MoveIteratorToReal(location::GpsInfo const & info); + void MatchLocationToRoute(location::GpsInfo & location, location::RouteMatchingInfo & routeMatchingInfo) const; /// Add country name if we have no country filename to make route @@ -392,7 +396,9 @@ public: private: friend std::string DebugPrint(Route const & r); - +public: + void SetFakeSegmentsOnPolyline(); +private: double GetPolySegAngle(size_t ind) const; void GetClosestTurn(size_t segIdx, turns::TurnItem & turn) const; size_t ConvertPointIdxToSegmentIdx(size_t pointIdx) const; diff --git a/routing/routing_session.cpp b/routing/routing_session.cpp index c3c943e4d2..633ae74a0e 100644 --- a/routing/routing_session.cpp +++ b/routing/routing_session.cpp @@ -289,8 +289,11 @@ SessionState RoutingSession::OnLocationPositionChanged(GpsInfo const & info) ASSERT(m_route->IsValid(), ()); m_turnNotificationsMgr.SetSpeedMetersPerSecond(info.m_speedMpS); + bool movedIterator, closerToFake; + m_route->SetFakeSegmentsOnPolyline(); + std::tie(movedIterator, closerToFake) = m_route->MoveIteratorToReal(info); - if (m_route->MoveIterator(info)) + if (movedIterator) { m_moveAwayCounter = 0; m_lastDistance = 0.0; @@ -316,8 +319,11 @@ SessionState RoutingSession::OnLocationPositionChanged(GpsInfo const & info) if (m_userCurrentPositionValid) m_lastGoodPosition = m_userCurrentPosition; + + return m_state; } - else + + if(!closerToFake) { // Distance from the last known projection on route // (check if we are moving far from the last known projection).