diff --git a/routing/base/followed_polyline.cpp b/routing/base/followed_polyline.cpp index c51123290c..b4ea7f209e 100644 --- a/routing/base/followed_polyline.cpp +++ b/routing/base/followed_polyline.cpp @@ -128,37 +128,33 @@ Iter FollowedPolyline::GetBestProjection(m2::RectD const & posRect, return GetClosestProjectionInInterval(posRect, distFn, hoppingBorderIdx, m_nextCheckpointIndex); } - pair FollowedPolyline::GetBestMatchedProjection(m2::RectD const & posRect) const + FollowedPolyline::UpdatedProjection FollowedPolyline::GetBestMatchedProjection(m2::RectD const & posRect) 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; - tie(closestIter, nearestIsFake) = GetClosestMatchedProjectionInInterval(posRect, m_current.m_ind, hoppingBorderIdx); - if (closestIter.IsValid()) - return make_pair(closestIter, nearestIsFake); + auto res = GetClosestMatchedProjectionInInterval(posRect, m_current.m_ind, hoppingBorderIdx); + if (res.iter.IsValid()) + return UpdatedProjection{res.iter, res.closerToUnmatched}; // 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, hoppingBorderIdx, m_nextCheckpointIndex); } -pair FollowedPolyline::UpdateMatchedProjection(m2::RectD const & posRect) +FollowedPolyline::UpdatedProjectionInfo 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(); - tie(iter, nearestIsFake) = GetBestMatchedProjection(posRect); + auto res = GetBestMatchedProjection(posRect); - if (iter.IsValid()) - m_current = iter; - return make_pair(iter.IsValid(), nearestIsFake); + if (res.iter.IsValid()) + m_current = res.iter; + return UpdatedProjectionInfo{res.iter.IsValid(), res.closerToUnmatched}; } Iter FollowedPolyline::UpdateProjectionByPrediction(m2::RectD const & posRect, diff --git a/routing/base/followed_polyline.hpp b/routing/base/followed_polyline.hpp index a9800d40dc..d0ddad6219 100644 --- a/routing/base/followed_polyline.hpp +++ b/routing/base/followed_polyline.hpp @@ -8,7 +8,6 @@ #include #include -#include #include namespace routing @@ -70,6 +69,20 @@ public: bool IsValid() const { return m_ind != kInvalidIndex; } }; + struct UpdatedProjection + { + // Iterator to the projection point. + Iter iter; + // True if nearest point is on an unmatched segment. + bool closerToUnmatched; + }; + + struct UpdatedProjectionInfo + { + bool updatedProjection; + bool closerToFake; + }; + const Iter GetCurrentIter() const { return m_current; } double GetDistanceM(Iter const & it1, Iter const & it2) const; @@ -79,8 +92,8 @@ public: /// \brief Sets indexes of all unmatched segments on route. void SetUnmatchedSegmentIndexes(std::vector && unmatchedSegmentIndexes); - /// \brief Updates projection to the closest real segment if it's possible. - std::pair UpdateMatchedProjection(m2::RectD const & posRect); + /// \brief Updates projection to the closest matched segment if it's possible. + UpdatedProjectionInfo UpdateMatchedProjection(m2::RectD const & posRect); Iter UpdateProjection(m2::RectD const & posRect); @@ -125,8 +138,7 @@ public: return res; } - /// \returns pair of iterator (projection point) and bool (true if nearest point is on an unmatched segment). - std::pair GetClosestMatchedProjectionInInterval(m2::RectD const & posRect, + UpdatedProjection GetClosestMatchedProjectionInInterval(m2::RectD const & posRect, size_t startIdx, size_t endIdx) const { CHECK_LESS_OR_EQUAL(endIdx, m_segProj.size(), ()); @@ -163,7 +175,7 @@ public: minDistUnmatched = dp; } } - return std::make_pair(nearestIter, minDistUnmatched < minDist); + return UpdatedProjection{nearestIter, minDistUnmatched < minDist}; } private: @@ -174,7 +186,7 @@ private: template Iter GetBestProjection(m2::RectD const & posRect, DistanceFn const & distFn) const; - std::pair GetBestMatchedProjection(m2::RectD const & posRect) const; + UpdatedProjection GetBestMatchedProjection(m2::RectD const & posRect) const; void Update(); diff --git a/routing/route.cpp b/routing/route.cpp index 96305ca061..69530a2739 100644 --- a/routing/route.cpp +++ b/routing/route.cpp @@ -250,13 +250,13 @@ void Route::SetFakeSegmentsOnPolyline() m_poly.SetUnmatchedSegmentIndexes(move(fakeSegmentIndexes)); } -pair Route::MoveIteratorToReal(location::GpsInfo const & info) +Route::MovedIteratorInfo 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); + auto resUpdate = m_poly.UpdateMatchedProjection(rect); + return MovedIteratorInfo{resUpdate.updatedProjection, resUpdate.closerToFake}; } double Route::GetPolySegAngle(size_t ind) const diff --git a/routing/route.hpp b/routing/route.hpp index 472763ddb1..39b9ef9f13 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -24,7 +24,6 @@ #include #include #include -#include #include namespace location @@ -193,6 +192,14 @@ public: size_t m_endSegmentIdx = 0; }; + struct MovedIteratorInfo + { + // Indicator of setting the iterator to one of real segments + bool movedIterator; + // Indicator of the presence of the fake segment which is the nearest to the given point + bool closerToFake; + }; + /// \brief For every subroute some attributes are kept in the following structure. struct SubrouteSettings final { @@ -326,7 +333,7 @@ public: /// \returns 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); + MovedIteratorInfo MoveIteratorToReal(location::GpsInfo const & info); void MatchLocationToRoute(location::GpsInfo & location, location::RouteMatchingInfo & routeMatchingInfo) const; diff --git a/routing/routing_session.cpp b/routing/routing_session.cpp index c21efac67e..0670765617 100644 --- a/routing/routing_session.cpp +++ b/routing/routing_session.cpp @@ -291,11 +291,10 @@ SessionState RoutingSession::OnLocationPositionChanged(GpsInfo const & info) ASSERT(m_route->IsValid(), ()); m_turnNotificationsMgr.SetSpeedMetersPerSecond(info.m_speedMpS); - bool movedIterator = false; - bool closerToFake = false; - std::tie(movedIterator, closerToFake) = m_route->MoveIteratorToReal(info); - if (movedIterator) + auto iteratorAction = m_route->MoveIteratorToReal(info); + + if (iteratorAction.movedIterator) { m_moveAwayCounter = 0; m_lastDistance = 0.0; @@ -325,7 +324,7 @@ SessionState RoutingSession::OnLocationPositionChanged(GpsInfo const & info) return m_state; } - if(!closerToFake) + if(!iteratorAction.closerToFake) { // Distance from the last known projection on route // (check if we are moving far from the last known projection).