From bae86bf9d59b5adf139b9e8d8d5b0fda3ec19749 Mon Sep 17 00:00:00 2001 From: Gonzalo Pesquero Date: Thu, 26 Dec 2024 14:32:09 +0100 Subject: [PATCH 1/3] [routing] Enable get distance and time to route segment Signed-off-by: Gonzalo Pesquero --- map/routing_manager.cpp | 40 ++++++++++++++++++++++++++++++++++--- map/routing_manager.hpp | 1 + map/routing_mark.hpp | 2 ++ routing/route.cpp | 24 +++++++++++++++++++++- routing/route.hpp | 7 ++++++- routing/routing_session.hpp | 6 ++++++ 6 files changed, 75 insertions(+), 5 deletions(-) diff --git a/map/routing_manager.cpp b/map/routing_manager.cpp index fab7dd21e1..55bd20ee79 100644 --- a/map/routing_manager.cpp +++ b/map/routing_manager.cpp @@ -22,6 +22,7 @@ #include "indexer/map_style_reader.hpp" #include "platform/country_file.hpp" +#include "platform/distance.hpp" #include "platform/platform.hpp" #include "platform/socket.hpp" @@ -820,9 +821,42 @@ bool RoutingManager::IsMyPosition(RouteMarkType type, size_t intermediateIndex) vector RoutingManager::GetRoutePoints() const { vector result; - RoutePointsLayout routePoints(*m_bmManager); - for (auto const & p : routePoints.GetRoutePoints()) - result.push_back(p->GetMarkData()); + RoutePointsLayout routePointsLayout(*m_bmManager); + + auto const & routePoints = routePointsLayout.GetRoutePoints(); + + size_t subrouteCount = m_routingSession.GetSubrouteCount(); + //std::vector subroutes = m_routingSession.GetSubroutes(); + + // Copy subroute time & distance to mark data only if subroute and route point sizes match. + bool bCopy = (routePoints.size() == subrouteCount + 1); + + for(int point = 0; point < (int)routePoints.size(); point++) + { + auto markData = routePoints[point]->GetMarkData(); + + if (bCopy) + { + if (point == 0) + { + // This is the first starting point. Time & distance are 0. + markData.m_timeSec = 0; + markData.m_distance = ""; + } + else + { + // Get subroute time & distance pair. + auto timeDistancePair = m_routingSession.GetSubrouteTotalTimeAndDistance(point - 1); + markData.m_timeSec = timeDistancePair.first; + markData.m_distance = platform::Distance::CreateFormatted(timeDistancePair.second).ToString(); + } + + LOG(LINFO, ("point", point, "timeSec", markData.m_timeSec, "distance", markData.m_distance)); + } + + result.push_back(markData); + } + return result; } diff --git a/map/routing_manager.hpp b/map/routing_manager.hpp index 0141d16ab0..5d5333c1cf 100644 --- a/map/routing_manager.hpp +++ b/map/routing_manager.hpp @@ -125,6 +125,7 @@ public: routing::RoutingSession const & RoutingSession() const { return m_routingSession; } routing::RoutingSession & RoutingSession() { return m_routingSession; } + //std::vector GetSubroutes() { return m_routingSession.GetSubroutes(); } void SetRouter(routing::RouterType type); routing::RouterType GetRouter() const { return m_currentRouterType; } bool IsRoutingActive() const { return m_routingSession.IsActive(); } diff --git a/map/routing_mark.hpp b/map/routing_mark.hpp index a77764fefb..8d4ea9e61c 100644 --- a/map/routing_mark.hpp +++ b/map/routing_mark.hpp @@ -24,6 +24,8 @@ struct RouteMarkData bool m_isPassed = false; bool m_replaceWithMyPositionAfterRestart = false; m2::PointD m_position; + long m_timeSec = 0; + std::string m_distance; }; class RouteMarkPoint : public UserMark diff --git a/routing/route.cpp b/routing/route.cpp index 3bfd7d096f..53587c66b1 100644 --- a/routing/route.cpp +++ b/routing/route.cpp @@ -65,6 +65,13 @@ double Route::GetTotalDistanceMeters() const return m_poly.GetTotalDistanceMeters(); } +double Route::GetTotalDistanceToSegmentMeters(size_t segIdx) const +{ + if (!IsValid()) + return 0.0; + return m_routeSegments[segIdx].GetDistFromBeginningMeters(); +} + double Route::GetCurrentDistanceFromBeginMeters() const { if (!IsValid()) @@ -160,6 +167,13 @@ double Route::GetCurrentTimeToSegmentSec(size_t segIdx) const return endTimeSec - passedTimeSec; } +double Route::GetTotalTimeToSegmentSec(size_t segIdx) const +{ + if (!IsValid()) + return 0.0; + return m_routeSegments[segIdx].GetTimeFromBeginningSec(); +} + void Route::GetCurrentSpeedLimit(SpeedInUnits & speedLimit) const { if (!IsValid()) @@ -452,6 +466,14 @@ Route::SubrouteSettings const Route::GetSubrouteSettings(size_t segmentIdx) cons return SubrouteSettings(m_routingSettings, m_router, m_subrouteUid); } +std::pair Route::GetSubrouteTotalTimeAndDistance(size_t subrouteIdx) const +{ + size_t endSegmentIdx = m_subrouteAttrs.at(subrouteIdx).GetEndSegmentIdx() - 1; + long timeSec = GetTotalTimeToSegmentSec(endSegmentIdx); + double distanceMeters = GetTotalDistanceToSegmentMeters(endSegmentIdx); + return std::pair(timeSec, distanceMeters); +} + bool Route::IsSubroutePassed(size_t subrouteIdx) const { size_t const endSegmentIdx = GetSubrouteAttrs(subrouteIdx).GetEndSegmentIdx(); @@ -532,7 +554,7 @@ std::string Route::DebugPrintTurns() const { auto const & turn = m_routeSegments[i].GetTurn(); - // Always print first elemenst as Start. + // Always print first element as Start. if (i == 0 || !turn.IsTurnNone()) { res += DebugPrint(mercator::ToLatLon(m_routeSegments[i].GetJunction())); diff --git a/routing/route.hpp b/routing/route.hpp index adb4e143d3..fb3bcb5a0a 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -353,7 +353,10 @@ public: /// \brief estimated time to reach segment. double GetCurrentTimeToSegmentSec(size_t segIdx) const; - /// \brief estimated time to the nearest turn. + /// \brief total route time to reach segment. + double GetTotalTimeToSegmentSec(size_t segIdx) const; + + /// \brief estimated time to the nearest turn. double GetCurrentTimeToNearestTurnSec() const; FollowedPolyline const & GetFollowedPolyline() const { return m_poly; } @@ -363,11 +366,13 @@ public: size_t GetCurrentSubrouteIdx() const { return m_currentSubrouteIdx; } std::vector const & GetSubroutes() const { return m_subrouteAttrs; } + std::pair GetSubrouteTotalTimeAndDistance(size_t subrouteIdx) const; std::vector const & GetSegDistanceMeters() const { return m_poly.GetSegDistanceMeters(); } bool IsValid() const { return m_poly.IsValid(); } double GetTotalDistanceMeters() const; + double GetTotalDistanceToSegmentMeters(size_t segIdx) const; double GetCurrentDistanceFromBeginMeters() const; double GetCurrentDistanceToEndMeters() const; double GetMercatorDistanceFromBegin() const; diff --git a/routing/routing_session.hpp b/routing/routing_session.hpp index 21041c24b1..8bbfbae6e5 100644 --- a/routing/routing_session.hpp +++ b/routing/routing_session.hpp @@ -174,6 +174,12 @@ public: double GetCompletionPercent() const; + size_t GetSubrouteCount() const { return m_route->GetSubrouteCount(); } + std::pair GetSubrouteTotalTimeAndDistance(size_t subrouteIdx) const + { + return m_route->GetSubrouteTotalTimeAndDistance(subrouteIdx); + } + private: struct DoReadyCallback { -- 2.45.3 From 88b138f7ef66fd91e2d9533980f3cd1a080b24f4 Mon Sep 17 00:00:00 2001 From: Gonzalo Pesquero Date: Thu, 26 Dec 2024 16:18:27 +0100 Subject: [PATCH 2/3] [android] Add route point and finish icons Signed-off-by: Gonzalo Pesquero --- .../src/main/res/drawable/route_point_01.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_02.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_03.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_04.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_05.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_06.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_07.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_08.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_09.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_10.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_11.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_12.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_13.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_14.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_15.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_16.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_17.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_18.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_19.xml | 19 +++++++++++++ .../src/main/res/drawable/route_point_20.xml | 19 +++++++++++++ .../main/res/drawable/route_point_finish.png | Bin 0 -> 7416 bytes .../main/res/drawable/route_point_start.png | Bin 0 -> 1826 bytes android/app/src/main/res/values/arrays.xml | 25 ++++++++++++++++++ 23 files changed, 405 insertions(+) create mode 100644 android/app/src/main/res/drawable/route_point_01.xml create mode 100644 android/app/src/main/res/drawable/route_point_02.xml create mode 100644 android/app/src/main/res/drawable/route_point_03.xml create mode 100644 android/app/src/main/res/drawable/route_point_04.xml create mode 100644 android/app/src/main/res/drawable/route_point_05.xml create mode 100644 android/app/src/main/res/drawable/route_point_06.xml create mode 100644 android/app/src/main/res/drawable/route_point_07.xml create mode 100644 android/app/src/main/res/drawable/route_point_08.xml create mode 100644 android/app/src/main/res/drawable/route_point_09.xml create mode 100644 android/app/src/main/res/drawable/route_point_10.xml create mode 100644 android/app/src/main/res/drawable/route_point_11.xml create mode 100644 android/app/src/main/res/drawable/route_point_12.xml create mode 100644 android/app/src/main/res/drawable/route_point_13.xml create mode 100644 android/app/src/main/res/drawable/route_point_14.xml create mode 100644 android/app/src/main/res/drawable/route_point_15.xml create mode 100644 android/app/src/main/res/drawable/route_point_16.xml create mode 100644 android/app/src/main/res/drawable/route_point_17.xml create mode 100644 android/app/src/main/res/drawable/route_point_18.xml create mode 100644 android/app/src/main/res/drawable/route_point_19.xml create mode 100644 android/app/src/main/res/drawable/route_point_20.xml create mode 100644 android/app/src/main/res/drawable/route_point_finish.png create mode 100644 android/app/src/main/res/drawable/route_point_start.png create mode 100644 android/app/src/main/res/values/arrays.xml diff --git a/android/app/src/main/res/drawable/route_point_01.xml b/android/app/src/main/res/drawable/route_point_01.xml new file mode 100644 index 0000000000..f61bca62be --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_01.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_02.xml b/android/app/src/main/res/drawable/route_point_02.xml new file mode 100644 index 0000000000..d945fc7573 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_02.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_03.xml b/android/app/src/main/res/drawable/route_point_03.xml new file mode 100644 index 0000000000..1bd922d612 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_03.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_04.xml b/android/app/src/main/res/drawable/route_point_04.xml new file mode 100644 index 0000000000..5eb2839fac --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_04.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_05.xml b/android/app/src/main/res/drawable/route_point_05.xml new file mode 100644 index 0000000000..8c87e652f9 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_05.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_06.xml b/android/app/src/main/res/drawable/route_point_06.xml new file mode 100644 index 0000000000..f42b79c893 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_06.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_07.xml b/android/app/src/main/res/drawable/route_point_07.xml new file mode 100644 index 0000000000..003e621640 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_07.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_08.xml b/android/app/src/main/res/drawable/route_point_08.xml new file mode 100644 index 0000000000..0517fce861 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_08.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_09.xml b/android/app/src/main/res/drawable/route_point_09.xml new file mode 100644 index 0000000000..6cf968bf54 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_09.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_10.xml b/android/app/src/main/res/drawable/route_point_10.xml new file mode 100644 index 0000000000..9b0dd4ac00 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_10.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_11.xml b/android/app/src/main/res/drawable/route_point_11.xml new file mode 100644 index 0000000000..011e29d3a1 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_11.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_12.xml b/android/app/src/main/res/drawable/route_point_12.xml new file mode 100644 index 0000000000..dc49d8702a --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_12.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_13.xml b/android/app/src/main/res/drawable/route_point_13.xml new file mode 100644 index 0000000000..53a78161c6 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_13.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_14.xml b/android/app/src/main/res/drawable/route_point_14.xml new file mode 100644 index 0000000000..596445aa6b --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_14.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_15.xml b/android/app/src/main/res/drawable/route_point_15.xml new file mode 100644 index 0000000000..3d4285f9b3 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_15.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_16.xml b/android/app/src/main/res/drawable/route_point_16.xml new file mode 100644 index 0000000000..e98de508d7 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_16.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_17.xml b/android/app/src/main/res/drawable/route_point_17.xml new file mode 100644 index 0000000000..30836bb004 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_17.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_18.xml b/android/app/src/main/res/drawable/route_point_18.xml new file mode 100644 index 0000000000..ac0edc0523 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_18.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_19.xml b/android/app/src/main/res/drawable/route_point_19.xml new file mode 100644 index 0000000000..d2f55877df --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_19.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_20.xml b/android/app/src/main/res/drawable/route_point_20.xml new file mode 100644 index 0000000000..c9a25476c6 --- /dev/null +++ b/android/app/src/main/res/drawable/route_point_20.xml @@ -0,0 +1,19 @@ + + + + + diff --git a/android/app/src/main/res/drawable/route_point_finish.png b/android/app/src/main/res/drawable/route_point_finish.png new file mode 100644 index 0000000000000000000000000000000000000000..46b0adf6d63377546bd13cc01de05f1138373263 GIT binary patch literal 7416 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 QKmY&$07*qoM6N<$f@*7nI{*Lx literal 0 HcmV?d00001 diff --git a/android/app/src/main/res/values/arrays.xml b/android/app/src/main/res/values/arrays.xml new file mode 100644 index 0000000000..be609331b5 --- /dev/null +++ b/android/app/src/main/res/values/arrays.xml @@ -0,0 +1,25 @@ + + + + @drawable/route_point_01 + @drawable/route_point_02 + @drawable/route_point_03 + @drawable/route_point_04 + @drawable/route_point_05 + @drawable/route_point_06 + @drawable/route_point_07 + @drawable/route_point_08 + @drawable/route_point_09 + @drawable/route_point_10 + @drawable/route_point_11 + @drawable/route_point_12 + @drawable/route_point_13 + @drawable/route_point_14 + @drawable/route_point_15 + @drawable/route_point_16 + @drawable/route_point_17 + @drawable/route_point_18 + @drawable/route_point_19 + @drawable/route_point_20 + + \ No newline at end of file -- 2.45.3 From 19c91ebd6bc3dc77dd09feb450b1750445657a0a Mon Sep 17 00:00:00 2001 From: Gonzalo Pesquero Date: Thu, 26 Dec 2024 16:45:22 +0100 Subject: [PATCH 3/3] [android] Add route plan info Signed-off-by: Gonzalo Pesquero --- .../main/cpp/app/organicmaps/Framework.cpp | 10 +- .../organicmaps/routing/RouteMarkData.java | 9 +- .../organicmaps/routing/RoutePlanAdapter.java | 125 ++++++++++++ .../routing/RoutingBottomMenuController.java | 113 +++++------ .../java/app/organicmaps/util/DateUtils.java | 29 +++ .../app/organicmaps/util/StringUtils.java | 2 + .../java/app/organicmaps/util/UiUtils.java | 15 ++ .../app/organicmaps/widget/menu/NavMenu.java | 7 +- .../res/layout-land/altitude_chart_panel.xml | 188 ++++++++++++------ .../main/res/layout/altitude_chart_panel.xml | 175 +++++++++------- .../main/res/layout/route_plan_list_item.xml | 77 +++++++ .../layout/routing_bottom_panel_transit.xml | 25 ++- 12 files changed, 573 insertions(+), 202 deletions(-) create mode 100644 android/app/src/main/java/app/organicmaps/routing/RoutePlanAdapter.java create mode 100644 android/app/src/main/res/layout/route_plan_list_item.xml diff --git a/android/app/src/main/cpp/app/organicmaps/Framework.cpp b/android/app/src/main/cpp/app/organicmaps/Framework.cpp index d749e7a390..8fc342f8c1 100644 --- a/android/app/src/main/cpp/app/organicmaps/Framework.cpp +++ b/android/app/src/main/cpp/app/organicmaps/Framework.cpp @@ -1608,13 +1608,15 @@ Java_app_organicmaps_Framework_nativeGetRoutePoints(JNIEnv * env, jclass) // Java signature : RouteMarkData(String title, String subtitle, // @RoutePointInfo.RouteMarkType int pointType, // int intermediateIndex, boolean isVisible, boolean isMyPosition, - // boolean isPassed, double lat, double lon) + // boolean isPassed, double lat, double lon, long timeSec, + // String distanceString) static jmethodID const pointConstructor = jni::GetConstructorID(env, pointClazz, - "(Ljava/lang/String;Ljava/lang/String;IIZZZDD)V"); + "(Ljava/lang/String;Ljava/lang/String;IIZZZDDJLjava/lang/String;)V"); return jni::ToJavaArray(env, pointClazz, points, [&](JNIEnv * jEnv, RouteMarkData const & data) { jni::TScopedLocalRef const title(env, jni::ToJavaString(env, data.m_title)); jni::TScopedLocalRef const subtitle(env, jni::ToJavaString(env, data.m_subTitle)); + jni::TScopedLocalRef const distance(env, jni::ToJavaString(env, data.m_distance)); return env->NewObject(pointClazz, pointConstructor, title.get(), subtitle.get(), static_cast(data.m_pointType), @@ -1623,7 +1625,9 @@ Java_app_organicmaps_Framework_nativeGetRoutePoints(JNIEnv * env, jclass) static_cast(data.m_isMyPosition), static_cast(data.m_isPassed), mercator::YToLat(data.m_position.y), - mercator::XToLon(data.m_position.x)); + mercator::XToLon(data.m_position.x), + static_cast(data.m_timeSec), + distance.get()); }); } diff --git a/android/app/src/main/java/app/organicmaps/routing/RouteMarkData.java b/android/app/src/main/java/app/organicmaps/routing/RouteMarkData.java index 8004267f80..3471e17d09 100644 --- a/android/app/src/main/java/app/organicmaps/routing/RouteMarkData.java +++ b/android/app/src/main/java/app/organicmaps/routing/RouteMarkData.java @@ -3,6 +3,8 @@ package app.organicmaps.routing; import androidx.annotation.Keep; import androidx.annotation.Nullable; +import app.organicmaps.util.Distance; + /** * Represents RouteMarkData from core. */ @@ -23,11 +25,14 @@ public class RouteMarkData public final boolean mIsPassed; public final double mLat; public final double mLon; + public final long mTimeSec; + public final String mDistance; public RouteMarkData(@Nullable String title, @Nullable String subtitle, @RoutePointInfo.RouteMarkType int pointType, int intermediateIndex, boolean isVisible, boolean isMyPosition, - boolean isPassed, double lat, double lon) + boolean isPassed, double lat, double lon, long timeSec, + String distance) { mTitle = title; mSubtitle = subtitle; @@ -38,5 +43,7 @@ public class RouteMarkData mIsPassed = isPassed; mLat = lat; mLon = lon; + mTimeSec = timeSec; + mDistance = distance; } } diff --git a/android/app/src/main/java/app/organicmaps/routing/RoutePlanAdapter.java b/android/app/src/main/java/app/organicmaps/routing/RoutePlanAdapter.java new file mode 100644 index 0000000000..ed4fa25517 --- /dev/null +++ b/android/app/src/main/java/app/organicmaps/routing/RoutePlanAdapter.java @@ -0,0 +1,125 @@ +package app.organicmaps.routing; + +import android.content.Context; +import android.content.res.TypedArray; +import android.view.LayoutInflater; +import android.view.View; +import android.view.ViewGroup; +import android.widget.ImageView; +import android.widget.TextView; + +import androidx.annotation.NonNull; +import androidx.appcompat.content.res.AppCompatResources; +import androidx.recyclerview.widget.RecyclerView; + +import app.organicmaps.R; +import app.organicmaps.util.DateUtils; +import app.organicmaps.util.UiUtils; + +public class RoutePlanAdapter extends RecyclerView.Adapter +{ + Context mContext; + RouteMarkData[] mRouteMarkData; + + public RoutePlanAdapter(Context context, RouteMarkData[] routeMarkData) + { + mContext = context; + + // Copy all route points, except the first one (starting point). + mRouteMarkData = new RouteMarkData[routeMarkData.length - 1]; + System.arraycopy(routeMarkData, 1, mRouteMarkData, 0, routeMarkData.length - 1); + } + + @NonNull + @Override + public RoutePlanViewHolder onCreateViewHolder(@NonNull ViewGroup parent, int viewType) + { + View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.route_plan_list_item, + parent, false); + + return new RoutePlanViewHolder(view); + } + + @Override + public void onBindViewHolder(@NonNull RoutePlanViewHolder holder, int position) + { + int markPos = mRouteMarkData.length - 1 - position; + int iconId; + + if (mRouteMarkData[markPos].mPointType == 0) + { + // Start point. + iconId = R.drawable.route_point_start; + } + else if (mRouteMarkData[markPos].mPointType == 1) + { + // Intermediate stop. + TypedArray iconArray = mContext.getResources().obtainTypedArray(R.array.route_stop_icons); + iconId = iconArray.getResourceId(mRouteMarkData[markPos].mIntermediateIndex, + R.drawable.route_point_01); + } + else + { + // Finish point. + iconId = R.drawable.route_point_finish; + } + + holder.mImageViewIcon.setImageDrawable(AppCompatResources.getDrawable(mContext, iconId)); + + holder.mTextViewEta.setText(DateUtils.getEstimateTimeString(mContext, + mRouteMarkData[markPos].mTimeSec)); + + if (mRouteMarkData[markPos].mPointType == 0) + { + UiUtils.hide(holder.mTextViewSeparator1); + UiUtils.hide(holder.mTextViewTime); + UiUtils.hide(holder.mTextViewSeparator2); + UiUtils.hide(holder.mTextViewDistance); + } + else + { + holder.mTextViewTime.setText(DateUtils.getRemainingTimeString(mContext, + mRouteMarkData[markPos].mTimeSec)); + + holder.mTextViewDistance.setText(mRouteMarkData[markPos].mDistance); + } + } + + @Override + public int getItemCount() + { + return mRouteMarkData.length; + } + + static class RoutePlanViewHolder extends RecyclerView.ViewHolder + { + @NonNull + public final ImageView mImageViewIcon; + + @NonNull + public final TextView mTextViewEta; + + @NonNull + public final TextView mTextViewSeparator1; + + @NonNull + public final TextView mTextViewTime; + + @NonNull + public final TextView mTextViewSeparator2; + + @NonNull + public final TextView mTextViewDistance; + + RoutePlanViewHolder(@NonNull View itemView) + { + super(itemView); + mImageViewIcon = itemView.findViewById(R.id.icon); + mTextViewEta = itemView.findViewById(R.id.eta); + mTextViewSeparator1 = itemView.findViewById(R.id.separator1); + mTextViewTime = itemView.findViewById(R.id.time); + mTextViewSeparator2 = itemView.findViewById(R.id.separator2); + mTextViewDistance = itemView.findViewById(R.id.distance); + } + } +} diff --git a/android/app/src/main/java/app/organicmaps/routing/RoutingBottomMenuController.java b/android/app/src/main/java/app/organicmaps/routing/RoutingBottomMenuController.java index 748cb17855..0fc8d23c67 100644 --- a/android/app/src/main/java/app/organicmaps/routing/RoutingBottomMenuController.java +++ b/android/app/src/main/java/app/organicmaps/routing/RoutingBottomMenuController.java @@ -17,6 +17,8 @@ import android.text.style.ForegroundColorSpan; import android.text.style.StyleSpan; import android.text.style.TypefaceSpan; import android.view.View; +import android.view.ViewGroup; +import android.view.ViewTreeObserver; import android.widget.Button; import android.widget.ImageView; import android.widget.ScrollView; @@ -26,14 +28,17 @@ import androidx.annotation.IdRes; import androidx.annotation.NonNull; import androidx.annotation.Nullable; import androidx.core.content.ContextCompat; +import androidx.recyclerview.widget.LinearLayoutManager; import androidx.recyclerview.widget.RecyclerView; import app.organicmaps.Framework; import app.organicmaps.R; import app.organicmaps.bookmarks.data.DistanceAndAzimut; import app.organicmaps.location.LocationHelper; +import app.organicmaps.util.DateUtils; import app.organicmaps.util.Distance; import app.organicmaps.util.Graphics; +import app.organicmaps.util.StringUtils; import app.organicmaps.util.ThemeUtils; import app.organicmaps.util.UiUtils; import app.organicmaps.widget.recycler.DotDividerItemDecoration; @@ -50,7 +55,7 @@ final class RoutingBottomMenuController implements View.OnClickListener @NonNull private final Activity mContext; @NonNull - private final View mTimeElevationLine; + private final View mAltitudeLine; @NonNull private final View mAltitudeChartFrame; @NonNull @@ -62,13 +67,9 @@ final class RoutingBottomMenuController implements View.OnClickListener @NonNull private final ImageView mAltitudeChart; @NonNull - private final TextView mTime; + private final TextView mAltitudeDiffAscent; @NonNull - private final TextView mAltitudeDifference; - @NonNull - private final TextView mTimeVehicle; - @Nullable - private final TextView mArrival; + private final TextView mAltitudeDiffDescent; @NonNull private final View mActionFrame; @NonNull @@ -79,6 +80,8 @@ final class RoutingBottomMenuController implements View.OnClickListener private final ImageView mActionIcon; @NonNull private final DotDividerItemDecoration mTransitViewDecorator; + @NonNull + private final RecyclerView mRoutePlanList; @Nullable private final RoutingBottomMenuListener mListener; @@ -88,20 +91,19 @@ final class RoutingBottomMenuController implements View.OnClickListener @Nullable RoutingBottomMenuListener listener) { View altitudeChartFrame = getViewById(activity, frame, R.id.altitude_chart_panel); - View timeElevationLine = getViewById(activity, frame, R.id.time_elevation_line); + View altitudeLine = getViewById(activity, frame, R.id.altitude_line); View transitFrame = getViewById(activity, frame, R.id.transit_panel); TextView error = (TextView) getViewById(activity, frame, R.id.error); Button start = (Button) getViewById(activity, frame, R.id.start); ImageView altitudeChart = (ImageView) getViewById(activity, frame, R.id.altitude_chart); - TextView time = (TextView) getViewById(activity, frame, R.id.time); - TextView timeVehicle = (TextView) getViewById(activity, frame, R.id.time_vehicle); - TextView altitudeDifference = (TextView) getViewById(activity, frame, R.id.altitude_difference); - TextView arrival = (TextView) getViewById(activity, frame, R.id.arrival); + TextView altitudeDiffAscent = (TextView) getViewById(activity, frame, R.id.altitude_diff_ascent); + TextView altitudeDiffDescent = (TextView) getViewById(activity, frame, R.id.altitude_diff_descent); View actionFrame = getViewById(activity, frame, R.id.routing_action_frame); + RecyclerView routePlanList = (RecyclerView) getViewById(activity, frame, R.id.route_plan_list); - return new RoutingBottomMenuController(activity, altitudeChartFrame, timeElevationLine, transitFrame, - error, start, altitudeChart, time, altitudeDifference, - timeVehicle, arrival, actionFrame, listener); + return new RoutingBottomMenuController(activity, altitudeChartFrame, altitudeLine, transitFrame, + error, start, altitudeChart, altitudeDiffAscent, + altitudeDiffDescent, actionFrame, routePlanList, listener); } @NonNull @@ -114,29 +116,26 @@ final class RoutingBottomMenuController implements View.OnClickListener private RoutingBottomMenuController(@NonNull Activity context, @NonNull View altitudeChartFrame, - @NonNull View timeElevationLine, + @NonNull View altitudeLine, @NonNull View transitFrame, @NonNull TextView error, @NonNull Button start, @NonNull ImageView altitudeChart, - @NonNull TextView time, - @NonNull TextView altitudeDifference, - @NonNull TextView timeVehicle, - @Nullable TextView arrival, + @NonNull TextView altitudeDiffAscent, + @NonNull TextView altitudeDiffDescent, @NonNull View actionFrame, + @NonNull RecyclerView routePlanList, @Nullable RoutingBottomMenuListener listener) { mContext = context; mAltitudeChartFrame = altitudeChartFrame; - mTimeElevationLine = timeElevationLine; + mAltitudeLine = altitudeLine; mTransitFrame = transitFrame; mError = error; mStart = start; mAltitudeChart = altitudeChart; - mTime = time; - mAltitudeDifference = altitudeDifference; - mTimeVehicle = timeVehicle; - mArrival = arrival; + mAltitudeDiffAscent = altitudeDiffAscent; + mAltitudeDiffDescent = altitudeDiffDescent; mActionFrame = actionFrame; mActionMessage = actionFrame.findViewById(R.id.tv__message); mActionButton = actionFrame.findViewById(R.id.btn__my_position_use); @@ -149,13 +148,16 @@ final class RoutingBottomMenuController implements View.OnClickListener int dividerRes = ThemeUtils.getResource(mContext, R.attr.transitStepDivider); Drawable dividerDrawable = ContextCompat.getDrawable(mContext, dividerRes); Resources res = mContext.getResources(); - mTransitViewDecorator = new DotDividerItemDecoration(dividerDrawable, res.getDimensionPixelSize(R.dimen.margin_base), + mTransitViewDecorator = new DotDividerItemDecoration(dividerDrawable, + res.getDimensionPixelSize(R.dimen.margin_base), res.getDimensionPixelSize(R.dimen.margin_half)); + mRoutePlanList = routePlanList; + mRoutePlanList.setLayoutManager(new LinearLayoutManager(mContext)); } void showAltitudeChartAndRoutingDetails() { - UiUtils.hide(mError, mActionFrame, mAltitudeChart, mTimeElevationLine, mTransitFrame); + UiUtils.hide(mError, mActionFrame, mAltitudeChart, mAltitudeLine, mTransitFrame); if (!RoutingController.get().isVehicleRouterType() && !RoutingController.get().isRulerRouterType()) showRouteAltitudeChart(); @@ -185,13 +187,19 @@ final class RoutingBottomMenuController implements View.OnClickListener scrollToBottom(rv); + TextView estimatedTimeOfArrivalView = mTransitFrame.findViewById(R.id.estimated_time_of_arrival); + estimatedTimeOfArrivalView.setText(DateUtils.getEstimateTimeString(mContext, info.getTotalTime())); + UiUtils.show(mTransitFrame, estimatedTimeOfArrivalView); + + UiUtils.show(mTransitFrame, R.id.dot1); + TextView totalTimeView = mTransitFrame.findViewById(R.id.total_time); totalTimeView.setText(RoutingController.formatRoutingTime(mContext, info.getTotalTime(), R.dimen.text_size_routing_number)); - View dotView = mTransitFrame.findViewById(R.id.dot); + View dot2View = mTransitFrame.findViewById(R.id.dot2); View pedestrianIcon = mTransitFrame.findViewById(R.id.pedestrian_icon); TextView distanceView = mTransitFrame.findViewById(R.id.total_distance); - UiUtils.showIf(info.getTotalPedestrianTimeInSec() > 0, dotView, pedestrianIcon, distanceView); + UiUtils.showIf(info.getTotalPedestrianTimeInSec() > 0, dot2View, pedestrianIcon, distanceView); distanceView.setText(info.getTotalPedestrianDistance() + " " + info.getTotalPedestrianDistanceUnits()); } @@ -216,13 +224,15 @@ final class RoutingBottomMenuController implements View.OnClickListener scrollToBottom(rv); } else - UiUtils.hide(rv); // Show only distance between start and finish + UiUtils.hide(rv); // Show only distance between start and finish. TextView totalTimeView = mTransitFrame.findViewById(R.id.total_time); totalTimeView.setText(mContext.getString(R.string.placepage_distance) + ": " + totalLength.mDistanceStr + " " + totalLength.getUnitsStr(mContext)); - UiUtils.hide(mTransitFrame, R.id.dot); + UiUtils.hide(mTransitFrame, R.id.estimated_time_of_arrival); + UiUtils.hide(mTransitFrame, R.id.dot1); + UiUtils.hide(mTransitFrame, R.id.dot2); UiUtils.hide(mTransitFrame, R.id.pedestrian_icon); UiUtils.hide(mTransitFrame, R.id.total_distance); } @@ -330,11 +340,11 @@ final class RoutingBottomMenuController implements View.OnClickListener { if (RoutingController.get().isVehicleRouterType()) { - UiUtils.hide(mTimeElevationLine, mAltitudeChart); + UiUtils.hide(mAltitudeLine, mAltitudeChart); return; } - UiUtils.hide(mTimeVehicle); + UiUtils.show(mAltitudeLine, mAltitudeChart); int chartWidth = UiUtils.dimen(mContext, R.dimen.altitude_chart_image_width); int chartHeight = UiUtils.dimen(mContext, R.dimen.altitude_chart_image_height); @@ -344,39 +354,23 @@ final class RoutingBottomMenuController implements View.OnClickListener { mAltitudeChart.setImageBitmap(bm); UiUtils.show(mAltitudeChart); - final String unit = limits.isMetricUnits ? mAltitudeDifference.getResources().getString(R.string.m) : mAltitudeDifference.getResources().getString(R.string.ft); - mAltitudeDifference.setText("↗ " + limits.totalAscentString + " " + unit + - " ↘ " + limits.totalDescentString + " " + unit); - UiUtils.show(mAltitudeDifference); + final String unit = limits.isMetricUnits ? mAltitudeDiffAscent.getResources().getString(R.string.m) : + mAltitudeDiffAscent.getResources().getString(R.string.ft); + mAltitudeDiffAscent.setText(limits.totalAscentString + StringUtils.kNarrowNonBreakingSpace + unit); + mAltitudeDiffDescent.setText(limits.totalDescentString + StringUtils.kNarrowNonBreakingSpace + unit); + + UiUtils.show(mAltitudeDiffAscent, mAltitudeDiffDescent); } } private void showRoutingDetails() { + // Show route plan info. + mRoutePlanList.setAdapter(new RoutePlanAdapter(mContext, Framework.nativeGetRoutePoints())); + final RoutingInfo rinfo = RoutingController.get().getCachedRoutingInfo(); if (rinfo == null) - { - UiUtils.hide(mTimeElevationLine, mTimeVehicle); - return; - } - - Spanned spanned = makeSpannedRoutingDetails(mContext, rinfo); - if (RoutingController.get().isVehicleRouterType()) - { - UiUtils.show(mTimeVehicle); - mTimeVehicle.setText(spanned); - } - else - { - UiUtils.show(mTimeElevationLine); - mTime.setText(spanned); - } - - if (mArrival != null) - { - String arrivalTime = RoutingController.formatArrivalTime(rinfo.totalTimeInSeconds); - mArrival.setText(arrivalTime); - } + UiUtils.hide(mAltitudeLine); } // Scroll RecyclerView to bottom using parent ScrollView. @@ -389,7 +383,6 @@ final class RoutingBottomMenuController implements View.OnClickListener @NonNull private static Spanned makeSpannedRoutingDetails(@NonNull Context context, @NonNull RoutingInfo routingInfo) - { CharSequence time = RoutingController.formatRoutingTime(context, routingInfo.totalTimeInSeconds, diff --git a/android/app/src/main/java/app/organicmaps/util/DateUtils.java b/android/app/src/main/java/app/organicmaps/util/DateUtils.java index 82a8c47752..d4aebff79d 100644 --- a/android/app/src/main/java/app/organicmaps/util/DateUtils.java +++ b/android/app/src/main/java/app/organicmaps/util/DateUtils.java @@ -6,7 +6,12 @@ import androidx.annotation.Keep; import androidx.annotation.NonNull; import java.text.DateFormat; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; import java.util.Locale; +import java.util.concurrent.TimeUnit; + +import app.organicmaps.R; public final class DateUtils { @@ -27,4 +32,28 @@ public final class DateUtils { return android.text.format.DateFormat.is24HourFormat(context); } + + public static String getEstimateTimeString(@NonNull Context context, long seconds) + { + final String format = android.text.format.DateFormat.is24HourFormat(context)? "HH:mm" : "h:mm a"; + final LocalTime localTime = LocalTime.now().plusSeconds(seconds); + return localTime.format(DateTimeFormatter.ofPattern(format)); + } + + public static String getRemainingTimeString(@NonNull Context context, long seconds) + { + final long hours = TimeUnit.SECONDS.toHours(seconds); + final long minutes = TimeUnit.SECONDS.toMinutes(seconds) % 60; + + String timeString = ""; + + if (hours != 0) + timeString = String.valueOf(hours) + StringUtils.kNarrowNonBreakingSpace + + context.getResources().getString(R.string.hour) + " "; + + timeString += String.valueOf(minutes) + StringUtils.kNarrowNonBreakingSpace + + context.getResources().getString(R.string.minute); + + return timeString; + } } diff --git a/android/app/src/main/java/app/organicmaps/util/StringUtils.java b/android/app/src/main/java/app/organicmaps/util/StringUtils.java index d82b80a092..2c6f29089e 100644 --- a/android/app/src/main/java/app/organicmaps/util/StringUtils.java +++ b/android/app/src/main/java/app/organicmaps/util/StringUtils.java @@ -14,6 +14,8 @@ import java.util.Locale; public class StringUtils { + final public static String kNarrowNonBreakingSpace = "\u202F"; + public static String formatUsingUsLocale(String pattern, Object... args) { return String.format(Locale.US, pattern, args); diff --git a/android/app/src/main/java/app/organicmaps/util/UiUtils.java b/android/app/src/main/java/app/organicmaps/util/UiUtils.java index 3bd2609007..8fc4d69723 100644 --- a/android/app/src/main/java/app/organicmaps/util/UiUtils.java +++ b/android/app/src/main/java/app/organicmaps/util/UiUtils.java @@ -114,6 +114,21 @@ public final class UiUtils v.setVisibility(View.VISIBLE); } + public static void show(View frame, @IdRes int viewId) + { + View view = frame.findViewById(viewId); + if (view == null) + return; + + show(view); + } + + public static void show(View frame, @IdRes int... viewIds) + { + for (final int id : viewIds) + show(frame, id); + } + public static void invisible(View... views) { for (final View v : views) diff --git a/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java b/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java index 9e2b8612b7..c856b0cbc7 100644 --- a/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java +++ b/android/app/src/main/java/app/organicmaps/widget/menu/NavMenu.java @@ -16,6 +16,7 @@ import app.organicmaps.R; import app.organicmaps.location.LocationHelper; import app.organicmaps.routing.RoutingInfo; import app.organicmaps.sound.TtsPlayer; +import app.organicmaps.util.DateUtils; import app.organicmaps.util.Graphics; import app.organicmaps.util.StringUtils; import app.organicmaps.util.ThemeUtils; @@ -200,10 +201,8 @@ public class NavMenu private void updateTimeEstimate(int seconds) { - final String format = android.text.format.DateFormat.is24HourFormat(mTimeMinuteValue.getContext()) - ? "HH:mm" : "h:mm a"; - final LocalTime localTime = LocalTime.now().plusSeconds(seconds); - mTimeEstimate.setText(localTime.format(DateTimeFormatter.ofPattern(format))); + mTimeEstimate.setText(DateUtils.getEstimateTimeString(mTimeMinuteValue.getContext(), + seconds)); } private void updateSpeedView(@NonNull RoutingInfo info) diff --git a/android/app/src/main/res/layout-land/altitude_chart_panel.xml b/android/app/src/main/res/layout-land/altitude_chart_panel.xml index d1080a5e53..56c6576bd9 100644 --- a/android/app/src/main/res/layout-land/altitude_chart_panel.xml +++ b/android/app/src/main/res/layout-land/altitude_chart_panel.xml @@ -1,77 +1,139 @@ - + + + + + + + + + - + android:id="@+id/altitude_line" + android:layout_marginStart="6dp" + android:layout_weight="1" + android:layout_width="0dp" + android:layout_height="match_parent" + android:orientation="horizontal" + tools:visibility="visible"> + + + + + + + + + + + + + tools:text="1043 m"/> + + + + + - +