diff --git a/map/routing_manager.cpp b/map/routing_manager.cpp index 5a1ebf3fd6..7010a9763e 100644 --- a/map/routing_manager.cpp +++ b/map/routing_manager.cpp @@ -12,6 +12,7 @@ #include "routing/index_router.hpp" #include "routing/online_absent_fetcher.hpp" #include "routing/route.hpp" +#include "routing/routing_callbacks.hpp" #include "routing/routing_helpers.hpp" #include "routing/speed_camera.hpp" @@ -1054,14 +1055,14 @@ void RoutingManager::CheckLocationForRouting(location::GpsInfo const & info) if (!IsRoutingActive()) return; - RoutingSession::State const state = m_routingSession.OnLocationPositionChanged(info); - if (state == RoutingSession::RouteNeedRebuild) + SessionState const state = m_routingSession.OnLocationPositionChanged(info); + if (state == SessionState::RouteNeedRebuild) { m_routingSession.RebuildRoute( MercatorBounds::FromLatLon(info.m_latitude, info.m_longitude), [this](Route const & route, RouterResultCode code) { OnRebuildRouteReady(route, code); }, nullptr /* needMoreMapsCallback */, nullptr /* removeRouteCallback */, 0 /* timeoutSec */, - RoutingSession::State::RouteRebuilding, true /* adjustToPrevRoute */); + SessionState::RouteRebuilding, true /* adjustToPrevRoute */); } } diff --git a/routing/routing_callbacks.hpp b/routing/routing_callbacks.hpp index fb30605652..0acca47441 100644 --- a/routing/routing_callbacks.hpp +++ b/routing/routing_callbacks.hpp @@ -40,6 +40,34 @@ enum class RouterResultCode HasWarnings = 16, }; +enum class SessionState +{ + RoutingNotActive, + RouteBuilding, // we requested a route and wait when it will be builded + RouteNotReady, // routing was not build + RouteNotStarted, // route is builded but the user isn't on it + OnRoute, // user follows the route + RouteNeedRebuild, // user left the route + RouteFinished, // destination point is reached but the session isn't closed + RouteNoFollowing, // route is built but following mode has been disabled + RouteRebuilding, // we requested a route rebuild and wait when it will be rebuilded +}; + +/* + * RoutingNotActive -> RouteBuilding // start route building + * RouteBuilding -> RouteNotReady // waiting for route in case of building a new route + * RouteBuilding -> RouteNotStarted // route is built in case of building a new route + * RouteRebuilding -> RouteNotReady // waiting for route in case of rebuilding + * RouteRebuilding -> RouteNotStarted // route is built in case of rebuilding + * RouteNotStarted -> OnRoute // user started following the route + * RouteNotStarted -> RouteNeedRebuild // user doesn't like the route. + * OnRoute -> RouteNeedRebuild // user moves away from route - need to rebuild + * OnRoute -> RouteNoFollowing // following mode was disabled. Router doesn't track position. + * OnRoute -> RouteFinished // user reached the end of route + * RouteNeedRebuild -> RouteNotReady // start rebuild route + * RouteFinished -> RouteNotReady // start new route + */ + using CheckpointCallback = std::function; using NeedMoreMapsCallback = std::function const &)>; using PointCheckCallback = std::function; @@ -48,6 +76,7 @@ using ReadyCallback = std::function; using ReadyCallbackOwnership = std::function, RouterResultCode)>; using RemoveRouteCallback = std::function; using RouteCallback = std::function; +using RouteChangeStateCallback = std::function; using RoutingStatisticsCallback = std::function const &)>; using SpeedCameraShowCallback = std::function; using SpeedCameraClearCallback = std::function; diff --git a/routing/routing_session.cpp b/routing/routing_session.cpp index 87e7ed837e..552f17190d 100644 --- a/routing/routing_session.cpp +++ b/routing/routing_session.cpp @@ -53,7 +53,7 @@ void FormatDistance(double dist, string & value, string & suffix) RoutingSession::RoutingSession() : m_router(nullptr) , m_route(make_shared(string() /* router */, 0 /* route id */)) - , m_state(RoutingNotActive) + , m_state(SessionState::RoutingNotActive) , m_isFollowing(false) , m_speedCameraManager(m_turnNotificationsMgr) , m_routingSettings(GetRoutingSettings(VehicleType::Car)) @@ -88,14 +88,14 @@ void RoutingSession::BuildRoute(Checkpoints const & checkpoints, m_routingRebuildCount = -1; // -1 for the first rebuild. RebuildRoute(checkpoints.GetStart(), m_buildReadyCallback, m_needMoreMapsCallback, - m_removeRouteCallback, timeoutSec, RouteBuilding, false /* adjust */); + m_removeRouteCallback, timeoutSec, SessionState::RouteBuilding, false /* adjust */); } void RoutingSession::RebuildRoute(m2::PointD const & startPoint, ReadyCallback const & readyCallback, NeedMoreMapsCallback const & needMoreMapsCallback, RemoveRouteCallback const & removeRouteCallback, - uint32_t timeoutSec, State routeRebuildingState, + uint32_t timeoutSec, SessionState routeRebuildingState, bool adjustToPrevRoute) { CHECK_THREAD_CHECKER(m_threadChecker, ()); @@ -136,7 +136,7 @@ void RoutingSession::DoReadyCallback::operator()(shared_ptr route, Router void RoutingSession::RemoveRoute() { CHECK_THREAD_CHECKER(m_threadChecker, ()); - SetState(RoutingNotActive); + SetState(SessionState::RoutingNotActive); m_lastDistance = 0.0; m_moveAwayCounter = 0; m_turnNotificationsMgr.Reset(); @@ -156,17 +156,17 @@ void RoutingSession::RebuildRouteOnTrafficUpdate() switch (m_state) { - case RoutingNotActive: - case RouteNotReady: - case RouteFinished: return; + case SessionState::RoutingNotActive: + case SessionState::RouteNotReady: + case SessionState::RouteFinished: return; - case RouteBuilding: - case RouteNotStarted: - case RouteNoFollowing: - case RouteRebuilding: startPoint = m_checkpoints.GetPointFrom(); + case SessionState::RouteBuilding: + case SessionState::RouteNotStarted: + case SessionState::RouteNoFollowing: + case SessionState::RouteRebuilding: startPoint = m_checkpoints.GetPointFrom(); - case OnRoute: - case RouteNeedRebuild: break; + case SessionState::OnRoute: + case SessionState::RouteNeedRebuild: break; } // Cancel current route building. @@ -175,67 +175,68 @@ void RoutingSession::RebuildRouteOnTrafficUpdate() RebuildRoute(startPoint, m_rebuildReadyCallback, nullptr /* needMoreMapsCallback */, nullptr /* removeRouteCallback */, 0 /* timeoutSec */, - routing::RoutingSession::State::RouteRebuilding, false /* adjustToPrevRoute */); + SessionState::RouteRebuilding, false /* adjustToPrevRoute */); } bool RoutingSession::IsActive() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return (m_state != RoutingNotActive); + return (m_state != SessionState::RoutingNotActive); } bool RoutingSession::IsNavigable() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return (m_state == RouteNotStarted || m_state == OnRoute || m_state == RouteFinished); + return (m_state == SessionState::RouteNotStarted || m_state == SessionState::OnRoute || + m_state == SessionState::RouteFinished); } bool RoutingSession::IsBuilt() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return (IsNavigable() || m_state == RouteNeedRebuild); + return (IsNavigable() || m_state == SessionState::RouteNeedRebuild); } bool RoutingSession::IsBuilding() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return (m_state == RouteBuilding || m_state == RouteRebuilding); + return (m_state == SessionState::RouteBuilding || m_state == SessionState::RouteRebuilding); } bool RoutingSession::IsBuildingOnly() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return m_state == RouteBuilding; + return m_state == SessionState::RouteBuilding; } bool RoutingSession::IsRebuildingOnly() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return m_state == RouteRebuilding; + return m_state == SessionState::RouteRebuilding; } bool RoutingSession::IsNotReady() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return m_state == RouteNotReady; + return m_state == SessionState::RouteNotReady; } bool RoutingSession::IsFinished() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return m_state == RouteFinished; + return m_state == SessionState::RouteFinished; } bool RoutingSession::IsNoFollowing() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return m_state == RouteNoFollowing; + return m_state == SessionState::RouteNoFollowing; } bool RoutingSession::IsOnRoute() const { CHECK_THREAD_CHECKER(m_threadChecker, ()); - return (m_state == OnRoute); + return (m_state == SessionState::OnRoute); } bool RoutingSession::IsFollowing() const @@ -257,15 +258,15 @@ void RoutingSession::Reset() m_lastCompletionPercent = 0; } -RoutingSession::State RoutingSession::OnLocationPositionChanged(GpsInfo const & info) +SessionState RoutingSession::OnLocationPositionChanged(GpsInfo const & info) { CHECK_THREAD_CHECKER(m_threadChecker, ()); - ASSERT(m_state != RoutingNotActive, ()); + ASSERT(m_state != SessionState::RoutingNotActive, ()); ASSERT(m_router != nullptr, ()); - if (m_state == RouteNeedRebuild || m_state == RouteFinished - || m_state == RouteBuilding || m_state == RouteRebuilding - || m_state == RouteNotReady || m_state == RouteNoFollowing) + if (m_state == SessionState::RouteNeedRebuild || m_state == SessionState::RouteFinished + || m_state == SessionState::RouteBuilding || m_state == SessionState::RouteRebuilding + || m_state == SessionState::RouteNotReady || m_state == SessionState::RouteNoFollowing) return m_state; ASSERT(m_route, ()); @@ -283,7 +284,7 @@ RoutingSession::State RoutingSession::OnLocationPositionChanged(GpsInfo const & if (m_checkpoints.IsFinished()) { m_passedDistanceOnRouteMeters += m_route->GetTotalDistanceMeters(); - SetState(RouteFinished); + SetState(SessionState::RouteFinished); alohalytics::TStringMap params = {{"router", m_route->GetRouterId()}, {"passedDistance", strings::to_string(m_passedDistanceOnRouteMeters)}, @@ -292,7 +293,7 @@ RoutingSession::State RoutingSession::OnLocationPositionChanged(GpsInfo const & } else { - SetState(OnRoute); + SetState(SessionState::OnRoute); m_speedCameraManager.OnLocationPositionChanged(info); } @@ -316,7 +317,7 @@ RoutingSession::State RoutingSession::OnLocationPositionChanged(GpsInfo const & if (m_moveAwayCounter > kOnRouteMissedCount) { m_passedDistanceOnRouteMeters += m_route->GetCurrentDistanceFromBeginMeters(); - SetState(RouteNeedRebuild); + SetState(SessionState::RouteNeedRebuild); alohalytics::TStringMap params = { {"router", m_route->GetRouterId()}, {"percent", strings::to_string(GetCompletionPercent())}, @@ -467,16 +468,16 @@ void RoutingSession::AssignRoute(shared_ptr route, RouterResultCode e) if (e != RouterResultCode::Cancelled) { if (route->IsValid()) - SetState(RouteNotStarted); + SetState(SessionState::RouteNotStarted); else - SetState(RoutingNotActive); + SetState(SessionState::RoutingNotActive); if (e != RouterResultCode::NoError) - SetState(RouteNotReady); + SetState(SessionState::RouteNotReady); } else { - SetState(RoutingNotActive); + SetState(SessionState::RoutingNotActive); } ASSERT(m_route, ()); @@ -523,10 +524,10 @@ traffic::SpeedGroup RoutingSession::MatchTraffic( bool RoutingSession::DisableFollowMode() { CHECK_THREAD_CHECKER(m_threadChecker, ()); - LOG(LINFO, ("Routing disables a following mode. State: ", m_state)); - if (m_state == RouteNotStarted || m_state == OnRoute) + LOG(LINFO, ("Routing disables a following mode. SessionState: ", m_state)); + if (m_state == SessionState::RouteNotStarted || m_state == SessionState::OnRoute) { - SetState(RouteNoFollowing); + SetState(SessionState::RouteNoFollowing); m_isFollowing = false; return true; } @@ -536,10 +537,10 @@ bool RoutingSession::DisableFollowMode() bool RoutingSession::EnableFollowMode() { CHECK_THREAD_CHECKER(m_threadChecker, ()); - LOG(LINFO, ("Routing enables a following mode. State: ", m_state)); - if (m_state == RouteNotStarted || m_state == OnRoute) + LOG(LINFO, ("Routing enables a following mode. SessionState: ", m_state)); + if (m_state == SessionState::RouteNotStarted || m_state == SessionState::OnRoute) { - SetState(OnRoute); + SetState(SessionState::OnRoute); m_isFollowing = true; } return m_isFollowing; @@ -737,19 +738,19 @@ void RoutingSession::SetLocaleWithJsonForTesting(std::string const & json, std:: m_turnNotificationsMgr.SetLocaleWithJsonForTesting(json, locale); } -string DebugPrint(RoutingSession::State state) +string DebugPrint(SessionState state) { switch (state) { - case RoutingSession::RoutingNotActive: return "RoutingNotActive"; - case RoutingSession::RouteBuilding: return "RouteBuilding"; - case RoutingSession::RouteNotReady: return "RouteNotReady"; - case RoutingSession::RouteNotStarted: return "RouteNotStarted"; - case RoutingSession::OnRoute: return "OnRoute"; - case RoutingSession::RouteNeedRebuild: return "RouteNeedRebuild"; - case RoutingSession::RouteFinished: return "RouteFinished"; - case RoutingSession::RouteNoFollowing: return "RouteNoFollowing"; - case RoutingSession::RouteRebuilding: return "RouteRebuilding"; + case SessionState::RoutingNotActive: return "RoutingNotActive"; + case SessionState::RouteBuilding: return "RouteBuilding"; + case SessionState::RouteNotReady: return "RouteNotReady"; + case SessionState::RouteNotStarted: return "RouteNotStarted"; + case SessionState::OnRoute: return "OnRoute"; + case SessionState::RouteNeedRebuild: return "RouteNeedRebuild"; + case SessionState::RouteFinished: return "RouteFinished"; + case SessionState::RouteNoFollowing: return "RouteNoFollowing"; + case SessionState::RouteRebuilding: return "RouteRebuilding"; } UNREACHABLE(); } diff --git a/routing/routing_session.hpp b/routing/routing_session.hpp index fcbd4a3312..35d154ad39 100644 --- a/routing/routing_session.hpp +++ b/routing/routing_session.hpp @@ -45,34 +45,6 @@ class RoutingSession : public traffic::TrafficObserver, public traffic::TrafficC friend struct UnitClass_AsyncGuiThreadTestWithRoutingSession_TestFollowRoutePercentTest; public: - enum State - { - RoutingNotActive, - RouteBuilding, // we requested a route and wait when it will be builded - RouteNotReady, // routing was not build - RouteNotStarted, // route is builded but the user isn't on it - OnRoute, // user follows the route - RouteNeedRebuild, // user left the route - RouteFinished, // destination point is reached but the session isn't closed - RouteNoFollowing, // route is built but following mode has been disabled - RouteRebuilding, // we requested a route rebuild and wait when it will be rebuilded - }; - - /* - * RoutingNotActive -> RouteBuilding // start route building - * RouteBuilding -> RouteNotReady // waiting for route in case of building a new route - * RouteBuilding -> RouteNotStarted // route is builded in case of building a new route - * RouteRebuilding -> RouteNotReady // waiting for route in case of rebuilding - * RouteRebuilding -> RouteNotStarted // route is builded in case of rebuilding - * RouteNotStarted -> OnRoute // user started following the route - * RouteNotStarted -> RouteNeedRebuild // user doesn't like the route. - * OnRoute -> RouteNeedRebuild // user moves away from route - need to rebuild - * OnRoute -> RouteNoFollowing // following mode was disabled. Router doesn't track position. - * OnRoute -> RouteFinished // user reached the end of route - * RouteNeedRebuild -> RouteNotReady // start rebuild route - * RouteFinished -> RouteNotReady // start new route - */ - RoutingSession(); void Init(RoutingStatisticsCallback const & routingStatisticsFn, @@ -88,7 +60,7 @@ public: void RebuildRoute(m2::PointD const & startPoint, ReadyCallback const & readyCallback, NeedMoreMapsCallback const & needMoreMapsCallback, RemoveRouteCallback const & removeRouteCallback, uint32_t timeoutSec, - State routeRebuildingState, bool adjustToPrevRoute); + SessionState routeRebuildingState, bool adjustToPrevRoute); m2::PointD GetStartPoint() const; m2::PointD GetEndPoint() const; @@ -108,7 +80,7 @@ public: bool IsFollowing() const; void Reset(); - inline void SetState(State state) { m_state = state; } + inline void SetState(SessionState state) { m_state = state; } /// \returns true if altitude information along |m_route| is available and /// false otherwise. @@ -122,7 +94,7 @@ public: bool GetRouteAltitudesAndDistancesM(std::vector & routeSegDistanceM, feature::TAltitudes & routeAltitudesM) const; - State OnLocationPositionChanged(location::GpsInfo const & info); + SessionState OnLocationPositionChanged(location::GpsInfo const & info); void GetRouteFollowingInfo(location::FollowingInfo & info) const; void MatchLocationToRoute(location::GpsInfo & location, @@ -212,7 +184,7 @@ private: private: std::unique_ptr m_router; std::shared_ptr m_route; - State m_state; + SessionState m_state; bool m_isFollowing; Checkpoints m_checkpoints; @@ -253,5 +225,5 @@ private: void FormatDistance(double dist, std::string & value, std::string & suffix); -std::string DebugPrint(RoutingSession::State state); +std::string DebugPrint(SessionState state); } // namespace routing diff --git a/routing/routing_tests/routing_session_test.cpp b/routing/routing_tests/routing_session_test.cpp index 13b141fcae..de22389dc5 100644 --- a/routing/routing_tests/routing_session_test.cpp +++ b/routing/routing_tests/routing_session_test.cpp @@ -164,11 +164,11 @@ UNIT_CLASS_TEST(AsyncGuiThreadTestWithRoutingSession, TestRouteRebuildingMovingA info.m_verticalAccuracy = 0.01; info.m_longitude = 0.; info.m_latitude = 1.; - RoutingSession::State code; + SessionState code; while (info.m_latitude < kTestRoute.back().y) { code = m_session->OnLocationPositionChanged(info); - TEST_EQUAL(code, RoutingSession::State::OnRoute, ()); + TEST_EQUAL(code, SessionState::OnRoute, ()); info.m_latitude += 0.01; } @@ -186,13 +186,13 @@ UNIT_CLASS_TEST(AsyncGuiThreadTestWithRoutingSession, TestRouteRebuildingMovingA GetPlatform().RunTask(Platform::Thread::Gui, [&checkTimedSignal, &info, this]() { info.m_longitude = 0.; info.m_latitude = 1.; - RoutingSession::State code = RoutingSession::State::RoutingNotActive; + SessionState code = SessionState::RoutingNotActive; for (size_t i = 0; i < 10; ++i) { code = m_session->OnLocationPositionChanged(info); info.m_latitude -= 0.1; } - TEST_EQUAL(code, RoutingSession::State::RouteNeedRebuild, ()); + TEST_EQUAL(code, SessionState::RouteNeedRebuild, ()); checkTimedSignal.Signal(); }); TEST(checkTimedSignal.WaitUntil(steady_clock::now() + kRouteBuildingMaxDuration), @@ -231,13 +231,13 @@ UNIT_CLASS_TEST(AsyncGuiThreadTestWithRoutingSession, TestRouteRebuildingMovingT GetPlatform().RunTask(Platform::Thread::Gui, [&checkTimedSignalAway, &info, this]() { info.m_longitude = 0.0; info.m_latitude = 0.0; - RoutingSession::State code = RoutingSession::State::RoutingNotActive; + SessionState code = SessionState::RoutingNotActive; for (size_t i = 0; i < 8; ++i) { code = m_session->OnLocationPositionChanged(info); info.m_latitude += 0.1; } - TEST_EQUAL(code, RoutingSession::State::RouteNeedRebuild, ()); + TEST_EQUAL(code, SessionState::RouteNeedRebuild, ()); checkTimedSignalAway.Signal(); }); TEST(checkTimedSignalAway.WaitUntil(steady_clock::now() + kRouteBuildingMaxDuration), @@ -304,20 +304,20 @@ UNIT_CLASS_TEST(AsyncGuiThreadTestWithRoutingSession, TestFollowRouteFlagPersist TEST(m_session->IsFollowing(), ()); info.m_longitude = 0.; info.m_latitude = 1.; - RoutingSession::State code; + SessionState code; for (size_t i = 0; i < 10; ++i) { code = m_session->OnLocationPositionChanged(info); info.m_latitude -= 0.1; } - TEST_EQUAL(code, RoutingSession::State::RouteNeedRebuild, ()); + TEST_EQUAL(code, SessionState::RouteNeedRebuild, ()); TEST(m_session->IsFollowing(), ()); m_session->RebuildRoute( kTestRoute.front(), [&rebuildTimedSignal](Route const &, RouterResultCode) { rebuildTimedSignal.Signal(); }, nullptr /* needMoreMapsCallback */, nullptr /* removeRouteCallback */, 0, - RoutingSession::State::RouteBuilding, false /* adjust */); + SessionState::RouteBuilding, false /* adjust */); }); TEST(rebuildTimedSignal.WaitUntil(steady_clock::now() + kRouteBuildingMaxDuration), ("Route was not built."));