diff --git a/android/jni/com/mapswithme/maps/Framework.cpp b/android/jni/com/mapswithme/maps/Framework.cpp index c4975bf104..c2cf95ad1d 100644 --- a/android/jni/com/mapswithme/maps/Framework.cpp +++ b/android/jni/com/mapswithme/maps/Framework.cpp @@ -1290,6 +1290,44 @@ Java_com_mapswithme_maps_Framework_nativeGetRoutePoints(JNIEnv * env, jclass) }); } +JNIEXPORT jobject JNICALL +Java_com_mapswithme_maps_Framework_nativeGetTransitRouteInfo(JNIEnv * env, jclass) +{ + auto const info = frm()->GetRoutingManager().GetTransitRouteInfo(); + + static jclass const transitStepClass = jni::GetGlobalClassRef(env, + "com/mapswithme/maps/routing/TransitStepInfo"); + // Java signature : TransitStepInfo(@TransitType int type, double distance, double time, + // @Nullable String number, int color) + static jmethodID const transitStepConstructor = jni::GetConstructorID(env, transitStepClass, + "(IDDLjava/lang/String;I)V"); + + jni::TScopedLocalRef const steps(env, jni::ToJavaArray(env, transitStepClass, + info.m_steps, + [&](JNIEnv * env, TransitStepInfo const & info) + { + jni::TScopedLocalRef const number(env, jni::ToJavaString(env, info.m_number)); + return env->NewObject(transitStepClass, transitStepConstructor, + static_cast(info.m_type), + static_cast(info.m_distance), + static_cast(info.m_time), + number.get(), + static_cast(info.m_color)); + })); + + static jclass const transitRouteInfoClass = jni::GetGlobalClassRef(env, + "com/mapswithme/maps/routing/TransitRouteInfo"); + // Java signature : TransitRouteInfo(double totalDistance, double totalTime, + // double totalPedestrianDistance, double totalPedestrianTime, + // TransitStepInfo[] steps) + static jmethodID const transitRouteInfoConstructor = jni::GetConstructorID(env, transitRouteInfoClass, + "(DDDD[Lcom/mapswithme/maps/routing/TransitStepInfo;)V"); + + return env->NewObject(transitRouteInfoClass, transitRouteInfoConstructor, + info.m_totalDistance, info.m_totalTime, info.m_totalPedestrianDistance, + info.m_totalPedestrianTime, steps.get()); +} + JNIEXPORT void JNICALL Java_com_mapswithme_maps_Framework_nativeRegisterMaps(JNIEnv * env, jclass) { diff --git a/android/src/com/mapswithme/maps/Framework.java b/android/src/com/mapswithme/maps/Framework.java index f14e2f311c..6eb7b4a909 100644 --- a/android/src/com/mapswithme/maps/Framework.java +++ b/android/src/com/mapswithme/maps/Framework.java @@ -19,6 +19,8 @@ import com.mapswithme.maps.location.LocationHelper; import com.mapswithme.maps.routing.RouteMarkData; import com.mapswithme.maps.routing.RoutePointInfo; import com.mapswithme.maps.routing.RoutingInfo; +import com.mapswithme.maps.routing.TransitRouteInfo; +import com.mapswithme.maps.routing.TransitStepInfo; import com.mapswithme.util.Constants; import java.lang.annotation.Retention; @@ -308,7 +310,8 @@ public class Framework public static native boolean nativeCouldAddIntermediatePoint(); @NonNull public static native RouteMarkData[] nativeGetRoutePoints(); - + @NonNull + public static native TransitRouteInfo nativeGetTransitRouteInfo(); /** * Registers all maps(.mwms). Adds them to the models, generates indexes and does all necessary stuff. */ diff --git a/android/src/com/mapswithme/maps/routing/RoutingController.java b/android/src/com/mapswithme/maps/routing/RoutingController.java index 5622d125c1..9e9ea0abed 100644 --- a/android/src/com/mapswithme/maps/routing/RoutingController.java +++ b/android/src/com/mapswithme/maps/routing/RoutingController.java @@ -106,6 +106,8 @@ public class RoutingController implements TaxiManager.TaxiListener private String[] mLastMissingMaps; @Nullable private RoutingInfo mCachedRoutingInfo; + @Nullable + private TransitRouteInfo mCachedTransitRouteInfo; private boolean mTaxiRequestHandled; private boolean mTaxiPlanning; private boolean mInternetConnected; @@ -134,6 +136,8 @@ public class RoutingController implements TaxiManager.TaxiListener || ResultCodesHelper.isMoreMapsNeeded(mLastResultCode)) { mCachedRoutingInfo = Framework.nativeGetRouteFollowingInfo(); + if (mLastRouterType == Framework.ROUTER_TYPE_TRANSIT) + mCachedTransitRouteInfo = Framework.nativeGetTransitRouteInfo(); setBuildState(BuildState.BUILT); mLastBuildProgress = 100; if (mContainer != null) diff --git a/android/src/com/mapswithme/maps/routing/TransitRouteInfo.java b/android/src/com/mapswithme/maps/routing/TransitRouteInfo.java new file mode 100644 index 0000000000..7ba64463c6 --- /dev/null +++ b/android/src/com/mapswithme/maps/routing/TransitRouteInfo.java @@ -0,0 +1,27 @@ +package com.mapswithme.maps.routing; + +import android.support.annotation.NonNull; + +/** + * Represents TransitRouteInfo from core. + */ +public class TransitRouteInfo +{ + public final double mTotalDistance; + public final double mTotalTime; + public final double mTotalPedestrianDistance; + public final double mTotalPedestrianTime; + @NonNull + public final TransitStepInfo[] mSteps; + + public TransitRouteInfo(double totalDistance, double totalTime, + double totalPedestrianDistance, double totalPedestrianTime, + @NonNull TransitStepInfo[] steps) + { + mTotalDistance = totalDistance; + mTotalTime = totalTime; + mTotalPedestrianDistance = totalPedestrianDistance; + mTotalPedestrianTime = totalPedestrianTime; + mSteps = steps; + } +} diff --git a/android/src/com/mapswithme/maps/routing/TransitStepInfo.java b/android/src/com/mapswithme/maps/routing/TransitStepInfo.java new file mode 100644 index 0000000000..5409d988ba --- /dev/null +++ b/android/src/com/mapswithme/maps/routing/TransitStepInfo.java @@ -0,0 +1,39 @@ +package com.mapswithme.maps.routing; + +import android.support.annotation.IntDef; +import android.support.annotation.Nullable; +import android.support.annotation.StringDef; + +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; + +/** + * Represents TransitStepInfo from core. + */ +public class TransitStepInfo +{ + public static final int TRANSIT_TYPE_PEDESTRIAN = 0; + public static final int TRANSIT_TYPE_SUBWAY = 1; + + @Retention(RetentionPolicy.SOURCE) + @IntDef({ TRANSIT_TYPE_PEDESTRIAN, TRANSIT_TYPE_SUBWAY }) + public @interface TransitType {} + + @TransitType + public final int mType; + public final double mDistance; + public final double mTime; + @Nullable + public final String mNumber; + public final int mColor; + + public TransitStepInfo(@TransitType int type, double distance, double time, + @Nullable String number, int color) + { + mType = type; + mDistance = distance; + mTime = time; + mNumber = number; + mColor = color; + } +} diff --git a/map/routing_manager.cpp b/map/routing_manager.cpp index 461e1dab57..6389eb3bfd 100644 --- a/map/routing_manager.cpp +++ b/map/routing_manager.cpp @@ -199,14 +199,16 @@ void AddTransitShapes(std::vector const & shapeIds, TransitSha subroute.AddStyle(style); } -string ColorToHexStr(dp::Color const & color) +uint32_t ColorToARGB(df::ColorConstant const & colorConstant) { - stringstream ss; - ss << nouppercase << hex << setfill('0'); - ss << setw(2) << static_cast(color.GetRed()) - << setw(2) << static_cast(color.GetGreen()) - << setw(2) << static_cast(color.GetBlue()); - return ss.str(); + auto const color = df::GetColorConstant(colorConstant); + return color.GetAlpha() << 24 | color.GetRed() << 16 | color.GetGreen() << 8 | color.GetBlue(); +} + +TransitType GetTransitType(string const & type) +{ + ASSERT_EQUAL(type, "subway", ()); + return TransitType::Subway; } void FillTransitStyleForRendering(vector const & segments, TransitReadManager & transitReadManager, @@ -231,7 +233,7 @@ void FillTransitStyleForRendering(vector const & segments, Transit TransitMarkInfo transitMarkInfo; double prevDistance = routeInfo.m_totalDistance; - double prevTime = routeInfo.m_totalDistance; + double prevTime = routeInfo.m_totalTime; bool pendingEntrance = false; @@ -242,7 +244,7 @@ void FillTransitStyleForRendering(vector const & segments, Transit if (!s.HasTransitInfo()) { - routeInfo.AddStep(TransitStepInfo(true /* isPedestrian */, distance, time)); + routeInfo.AddStep(TransitStepInfo(TransitType::Pedestrian, distance, time)); AddTransitPedestrianSegment(s.GetJunction().GetPoint(), subroute); lastColor = ""; @@ -261,9 +263,8 @@ void FillTransitStyleForRendering(vector const & segments, Transit auto const & line = displayInfo.m_lines.at(edge.m_lineId); auto const currentColor = df::GetTransitColorName(line.GetColor()); - string const hexColor = ColorToHexStr(df::GetColorConstant(currentColor)); - routeInfo.AddStep(TransitStepInfo(false /* isPedestrian */, distance, time, - line.GetType(), line.GetNumber(), hexColor)); + routeInfo.AddStep(TransitStepInfo(GetTransitType(line.GetType()), distance, time, + line.GetNumber(), ColorToARGB(currentColor))); auto const & stop1 = displayInfo.m_stops.at(edge.m_stop1Id); auto const & stop2 = displayInfo.m_stops.at(edge.m_stop2Id); @@ -360,7 +361,7 @@ void FillTransitStyleForRendering(vector const & segments, Transit auto const & gate = transitInfo.GetGate(); if (!lastColor.empty()) { - routeInfo.AddStep(TransitStepInfo(true /* isPedestrian */, distance, time)); + routeInfo.AddStep(TransitStepInfo(TransitType::Pedestrian, distance, time)); AddTransitGateSegment(s.GetJunction().GetPoint(), subroute); @@ -578,20 +579,23 @@ namespace marketing char const * const kRoutingCalculatingRoute = "Routing_CalculatingRoute"; } // namespace marketing -TransitStepInfo::TransitStepInfo(bool isPedestrian, double distance, double time, - std::string const & type, std::string const & number, std::string const & color) - : m_isPedestrian(isPedestrian) +TransitStepInfo::TransitStepInfo(TransitType type, double distance, double time, + std::string const & number, uint32_t color) + : m_type(type) , m_distance(distance) , m_time(time) - , m_type(type) , m_number(number) , m_color(color) {} bool TransitStepInfo::IsEqualType(TransitStepInfo const & ts) const { - return m_isPedestrian == ts.m_isPedestrian && - (m_isPedestrian || (m_type == ts.m_type && m_number == ts.m_number && m_color == ts.m_color)); + if (m_type != ts.m_type) + return false; + + if (m_type != TransitType::Pedestrian) + return m_number == ts.m_number && m_color == ts.m_color; + return true; } void TransitRouteInfo::AddStep(TransitStepInfo const & step) @@ -606,7 +610,7 @@ void TransitRouteInfo::AddStep(TransitStepInfo const & step) m_steps.push_back(step); } - if (step.m_isPedestrian) + if (step.m_type == TransitType::Pedestrian) { m_totalPedestrianDistance += step.m_distance; m_totalPedestrianTime += step.m_time; diff --git a/map/routing_manager.hpp b/map/routing_manager.hpp index d3a4317c71..0b3bae1619 100644 --- a/map/routing_manager.hpp +++ b/map/routing_manager.hpp @@ -50,19 +50,26 @@ struct RoutePointInfo m2::PointD m_position; }; +enum class TransitType: uint32_t +{ + // Do not change the order! + Pedestrian, + Subway +}; + struct TransitStepInfo { - TransitStepInfo(bool isPedestrian, double distance, double time, - std::string const & type = "", std::string const & number = "", std::string const & color = ""); + TransitStepInfo() = default; + TransitStepInfo(TransitType type, double distance, double time, + std::string const & number = "", uint32_t color = 0); bool IsEqualType(TransitStepInfo const & ts) const; - bool m_isPedestrian = false; + TransitType m_type = TransitType::Pedestrian; double m_distance = 0.0; double m_time = 0.0; - std::string m_type; std::string m_number; - std::string m_color; + uint32_t m_color; }; struct TransitRouteInfo diff --git a/routing/route.hpp b/routing/route.hpp index 55095d237b..16d2fd8524 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -81,11 +81,12 @@ private: Junction m_junction; /// Street name of |m_segment| if any. Otherwise |m_street| is empty. std::string m_street; - /// Distance from the route beginning to the farthest end of |m_segment| in meters. + /// Distance from the route (not the subroute) beginning to the farthest end of |m_segment| in meters. double m_distFromBeginningMeters = 0.0; - /// Distance from the route beginning to the farthest end of |m_segment| in mercator. + /// Distance from the route (not the subroute) beginning to the farthest end of |m_segment| in mercator. double m_distFromBeginningMerc = 0.0; - /// ETA from the route beginning in seconds to reach the farthest from the route beginning end of |m_segment|. + /// ETA from the route beginning (not the subroute) in seconds to reach the farthest from the route beginning + /// end of |m_segment|. double m_timeFromBeginningS = 0.0; traffic::SpeedGroup m_traffic = traffic::SpeedGroup::Unknown; /// Information needed to display transit segments properly.