diff --git a/routing/cross_mwm_connector.hpp b/routing/cross_mwm_connector.hpp index 0cfcec2a52..8779fb3ac3 100644 --- a/routing/cross_mwm_connector.hpp +++ b/routing/cross_mwm_connector.hpp @@ -159,6 +159,7 @@ public: ASSERT_NOT_EQUAL(transition.m_exitIdx, connector::kFakeIndex, ()); for (size_t enterIdx = 0; enterIdx < m_enters.size(); ++enterIdx) { + // @todo(t.yan) fix weight for ingoing edges https://jira.mail.ru/browse/MAPSME-5953 auto const weight = GetWeight(enterIdx, base::asserted_cast(transition.m_exitIdx)); AddEdge(m_enters[enterIdx], weight, edges); } diff --git a/routing/cross_mwm_graph.hpp b/routing/cross_mwm_graph.hpp index c45540ffa0..017b5a649e 100644 --- a/routing/cross_mwm_graph.hpp +++ b/routing/cross_mwm_graph.hpp @@ -93,17 +93,13 @@ public: void Clear(); - // Applies |fn| to each nontransit transition. We never need to apply function to both transit - // and nontransit transitions. - // We may need to implement ForEachTransitTransition() or |isTransit| flag here when we'll have - // leaps for transit. - template - void ForEachTransition(NumMwmId numMwmId, bool isEnter, Fn && fn) + // \returns nontransit transitions for mwm with id |numMwmId|. + // Should be used with CrossMwmIndexGraph only. + // @todo(bykoianko): rewrite comment and check after CrossMwmOsrm removal. + std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) { - if (CrossMwmSectionExists(numMwmId)) - m_crossMwmIndexGraph.ForEachTransition(numMwmId, isEnter, std::forward(fn)); - else - m_crossMwmOsrmGraph.ForEachTransition(numMwmId, isEnter, std::forward(fn)); + CHECK(CrossMwmSectionExists(numMwmId), ("Should be used in LeapsOnly mode only. LeapsOnly mode requires CrossMwmIndexGraph.")); + return m_crossMwmIndexGraph.GetTransitions(numMwmId, isEnter); } private: diff --git a/routing/cross_mwm_index_graph.hpp b/routing/cross_mwm_index_graph.hpp index 6fcb430643..dd3844d9ca 100644 --- a/routing/cross_mwm_index_graph.hpp +++ b/routing/cross_mwm_index_graph.hpp @@ -148,12 +148,10 @@ public: GetCrossMwmConnectorWithTransitions(numMwmId); } - template - void ForEachTransition(NumMwmId numMwmId, bool isEnter, Fn && fn) + std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) { - auto const & connectors = GetCrossMwmConnectorWithTransitions(numMwmId); - for (Segment const & t : (isEnter ? connectors.GetEnters() : connectors.GetExits())) - fn(t); + auto const & connector = GetCrossMwmConnectorWithTransitions(numMwmId); + return isEnter ? connector.GetEnters() : connector.GetExits(); } private: diff --git a/routing/cross_mwm_osrm_graph.hpp b/routing/cross_mwm_osrm_graph.hpp index cc38e063e0..6a215bedc2 100644 --- a/routing/cross_mwm_osrm_graph.hpp +++ b/routing/cross_mwm_osrm_graph.hpp @@ -32,14 +32,6 @@ public: void Clear(); TransitionPoints GetTransitionPoints(Segment const & s, bool isOutgoing); - template - void ForEachTransition(NumMwmId numMwmId, bool isEnter, Fn && fn) - { - auto const & ts = GetSegmentMaps(numMwmId); - for (auto const & kv : isEnter ? ts.m_ingoing : ts.m_outgoing) - fn(kv.first); - } - private: struct TransitionSegments { diff --git a/routing/fake_edges_container.hpp b/routing/fake_edges_container.hpp index 50d8e7a649..6957ff30e7 100644 --- a/routing/fake_edges_container.hpp +++ b/routing/fake_edges_container.hpp @@ -18,8 +18,7 @@ class FakeEdgesContainer final public: FakeEdgesContainer(IndexGraphStarter && starter) - : m_finishId(starter.m_finishId) - , m_finishPassThroughAllowed(starter.m_finishPassThroughAllowed) + : m_finish(starter.m_finish) , m_fake(std::move(starter.m_fake)) { } @@ -33,10 +32,8 @@ public: } private: - // Finish segment id. - uint32_t m_finishId; - // Finish segment is located in a pass-through/non-pass-through area. - bool m_finishPassThroughAllowed; + // Finish ending. + IndexGraphStarter::Ending m_finish; FakeGraph m_fake; }; } // namespace routing diff --git a/routing/fake_graph.hpp b/routing/fake_graph.hpp index 42cb9431c8..d3ba504a26 100644 --- a/routing/fake_graph.hpp +++ b/routing/fake_graph.hpp @@ -132,8 +132,6 @@ public: return true; } - std::map const & GetFakeToReal() const { return m_fakeToReal; } - private: // Key is fake segment, value is set of outgoing fake segments. std::map> m_outgoing; diff --git a/routing/index_graph_starter.cpp b/routing/index_graph_starter.cpp index e093841d80..97f597acac 100644 --- a/routing/index_graph_starter.cpp +++ b/routing/index_graph_starter.cpp @@ -4,6 +4,7 @@ #include "geometry/mercator.hpp" +#include #include namespace @@ -41,12 +42,10 @@ IndexGraphStarter::IndexGraphStarter(FakeEnding const & startEnding, FakeEnding const & finishEnding, uint32_t fakeNumerationStart, bool strictForward, WorldGraph & graph) : m_graph(graph) - , m_startId(fakeNumerationStart) - , m_startPassThroughAllowed(EndingPassThroughAllowed(startEnding)) - , m_finishPassThroughAllowed(EndingPassThroughAllowed(finishEnding)) { + m_start.m_id = fakeNumerationStart; AddStart(startEnding, finishEnding, strictForward, fakeNumerationStart); - m_finishId = fakeNumerationStart; + m_finish.m_id = fakeNumerationStart; AddFinish(finishEnding, startEnding, fakeNumerationStart); auto const startPoint = GetPoint(GetStartSegment(), false /* front */); auto const finishPoint = GetPoint(GetFinishSegment(), true /* front */); @@ -55,8 +54,7 @@ IndexGraphStarter::IndexGraphStarter(FakeEnding const & startEnding, void IndexGraphStarter::Append(FakeEdgesContainer const & container) { - m_finishId = container.m_finishId; - m_finishPassThroughAllowed = container.m_finishPassThroughAllowed; + m_finish = container.m_finish; m_fake.Append(container.m_fake); // It's important to calculate distance after m_fake.Append() because @@ -115,18 +113,20 @@ m2::PointD const & IndexGraphStarter::GetPoint(Segment const & segment, bool fro set IndexGraphStarter::GetMwms() const { set mwms; - for (auto const & kv : m_fake.GetFakeToReal()) - mwms.insert(kv.second.GetMwmId()); + for (auto const & s : m_start.m_real) + mwms.insert(s.GetMwmId()); + for (auto const & s : m_finish.m_real) + mwms.insert(s.GetMwmId()); return mwms; } -bool IndexGraphStarter::CheckLength(RouteWeight const & weight) const +bool IndexGraphStarter::CheckLength(RouteWeight const & weight) { // We allow 1 pass-through/non-pass-through crossing per ending located in // non-pass-through zone to allow user to leave this zone. int const nonPassThroughCrossAllowed = - (m_startPassThroughAllowed ? 0 : 1) + (m_finishPassThroughAllowed ? 0 : 1); + (StartPassThroughAllowed() ? 0 : 1) + (FinishPassThroughAllowed() ? 0 : 1); return weight.GetNonPassThroughCross() <= nonPassThroughCrossAllowed && m_graph.CheckLength(weight, m_startToFinishDistanceM); @@ -136,6 +136,46 @@ void IndexGraphStarter::GetEdgesList(Segment const & segment, bool isOutgoing, vector & edges) const { edges.clear(); + + // If mode is LeapsOnly we need to connect start/finish segment to transitions. + if (m_graph.GetMode() == WorldGraph::Mode::LeapsOnly) + { + m2::PointD const & segmentPoint = GetPoint(segment, true /* front */); + if ((segment == GetStartSegment() && isOutgoing) || (segment == GetFinishSegment() && !isOutgoing)) + { + // Note. If |isOutgoing| == true it's necessary to add edges which connect the start with all + // exits of its mwm. So |isEnter| below should be set to false. + // If |isOutgoing| == false all enters of the finish mwm should be connected with the finish + // point. So |isEnter| below should be set to true. + auto const mwms = GetEndingMwms(isOutgoing ? m_start : m_finish); + for (auto const & mwm : mwms) + { + for (auto const & s : m_graph.GetTransitions(mwm, !isOutgoing /* isEnter */)) + { + // @todo(t.yan) fix weight for ingoing edges https://jira.mail.ru/browse/MAPSME-5953 + edges.emplace_back(s, m_graph.CalcLeapWeight(segmentPoint, GetPoint(s, isOutgoing))); + } + } + return; + } + // Edge from finish mwm enter to finish. + if (GetSegmentLocation(segment) == SegmentLocation::FinishMwm && isOutgoing) + { + edges.emplace_back(GetFinishSegment(), m_graph.CalcLeapWeight( + segmentPoint, GetPoint(GetFinishSegment(), true /* front */ ))); + return; + } + // Ingoing edge from start mwm exit to start. + if (GetSegmentLocation(segment) == SegmentLocation::StartMwm && !isOutgoing) + { + // @todo(t.yan) fix weight for ingoing edges https://jira.mail.ru/browse/MAPSME-5953 + edges.emplace_back( + GetStartSegment(), + m_graph.CalcLeapWeight(segmentPoint, GetPoint(GetStartSegment(), true /* front */))); + return; + } + } + if (IsFakeSegment(segment)) { Segment real; @@ -188,18 +228,10 @@ RouteWeight IndexGraphStarter::CalcRouteSegmentWeight(vector const & ro return CalcSegmentWeight(route[segmentIndex]); } -bool IndexGraphStarter::IsLeap(NumMwmId mwmId) const +bool IndexGraphStarter::IsLeap(Segment const & segment) const { - if (mwmId == kFakeNumMwmId) - return false; - - for (auto const & kv : m_fake.GetFakeToReal()) - { - if (kv.second.GetMwmId() == mwmId) - return false; - } - - return m_graph.LeapIsAllowed(mwmId); + return !IsFakeSegment(segment) && GetSegmentLocation(segment) == SegmentLocation::OtherMwm && + m_graph.LeapIsAllowed(segment.GetMwmId()); } void IndexGraphStarter::AddEnding(FakeEnding const & thisEnding, FakeEnding const & otherEnding, @@ -223,6 +255,11 @@ void IndexGraphStarter::AddEnding(FakeEnding const & thisEnding, FakeEnding cons m_fake.AddStandaloneVertex(fakeSegment, fakeVertex); for (auto const & projection : thisEnding.m_projections) { + if (isStart) + m_start.m_real.insert(projection.m_segment); + else + m_finish.m_real.insert(projection.m_segment); + // Add projection edges auto const projectionSegment = GetFakeSegmentAndIncr(fakeNumerationStart); FakeVertex projectionVertex(isStart ? thisEnding.m_originJunction : projection.m_junction, @@ -308,16 +345,52 @@ void IndexGraphStarter::AddFakeEdges(Segment const & segment, vector & edges) const { - bool const isEnding = !m_fake.GetFake(segment).empty(); - m_graph.GetEdgeList(segment, isOutgoing, IsLeap(segment.GetMwmId()), isEnding, edges); + m_graph.GetEdgeList(segment, isOutgoing, IsLeap(segment), edges); } -bool IndexGraphStarter::EndingPassThroughAllowed(FakeEnding const & ending) +bool IndexGraphStarter::EndingPassThroughAllowed(Ending const & ending) { - return any_of(ending.m_projections.cbegin(), ending.m_projections.cend(), - [this](Projection const & projection) { - return m_graph.IsPassThroughAllowed(projection.m_segment.GetMwmId(), - projection.m_segment.GetFeatureId()); + return any_of(ending.m_real.cbegin(), ending.m_real.cend(), + [this](Segment const & s) { + return m_graph.IsPassThroughAllowed(s.GetMwmId(), + s.GetFeatureId()); }); } + +bool IndexGraphStarter::StartPassThroughAllowed() +{ + return EndingPassThroughAllowed(m_start); +} + +bool IndexGraphStarter::FinishPassThroughAllowed() +{ + return EndingPassThroughAllowed(m_finish); +} + +set IndexGraphStarter::GetEndingMwms(Ending const & ending) const +{ + set res; + for (auto const & s : ending.m_real) + res.insert(s.GetMwmId()); + return res; +} + +IndexGraphStarter::SegmentLocation IndexGraphStarter::GetSegmentLocation(Segment const & segment) const +{ + CHECK(!IsFakeSegment(segment),()); + auto const mwmId = segment.GetMwmId(); + auto containsSegment = [mwmId](Segment const & s) { return s.GetMwmId() == mwmId; }; + + bool const sameMwmWithStart = + any_of(m_start.m_real.cbegin(), m_start.m_real.end(), containsSegment); + bool const sameMwmWithFinish = + any_of(m_finish.m_real.cbegin(), m_finish.m_real.end(), containsSegment); + if (sameMwmWithStart && sameMwmWithFinish) + return SegmentLocation::StartFinishMwm; + if (sameMwmWithStart) + return SegmentLocation::StartMwm; + if (sameMwmWithFinish) + return SegmentLocation::FinishMwm; + return SegmentLocation::OtherMwm; +} } // namespace routing diff --git a/routing/index_graph_starter.hpp b/routing/index_graph_starter.hpp index b01c3699a3..45c58aadec 100644 --- a/routing/index_graph_starter.hpp +++ b/routing/index_graph_starter.hpp @@ -55,8 +55,8 @@ public: WorldGraph & GetGraph() const { return m_graph; } Junction const & GetStartJunction() const; Junction const & GetFinishJunction() const; - Segment GetStartSegment() const { return GetFakeSegment(m_startId); } - Segment GetFinishSegment() const { return GetFakeSegment(m_finishId); } + Segment GetStartSegment() const { return GetFakeSegment(m_start.m_id); } + Segment GetFinishSegment() const { return GetFakeSegment(m_finish.m_id); } // If segment is real returns true and does not modify segment. // If segment is part of real converts it to real and returns true. // Otherwise returns false and does not modify segment. @@ -76,7 +76,7 @@ public: // Checks whether |weight| meets non-pass-through crossing restrictions according to placement of // start and finish in pass-through/non-pass-through area and number of non-pass-through crosses. - bool CheckLength(RouteWeight const & weight) const; + bool CheckLength(RouteWeight const & weight); void GetEdgesList(Segment const & segment, bool isOutgoing, std::vector & edges) const; @@ -100,9 +100,26 @@ public: RouteWeight CalcSegmentWeight(Segment const & segment) const; RouteWeight CalcRouteSegmentWeight(std::vector const & route, size_t segmentIndex) const; - bool IsLeap(NumMwmId mwmId) const; + bool IsLeap(Segment const & segment) const; private: + enum class SegmentLocation + { + StartFinishMwm, // Segment is located in the same mwm with start and finish of the route. + StartMwm, // Segment is located in the same mwm with start but not finish of the route. + FinishMwm, // Segment is located in the same mwm with finish but not start of the route. + OtherMwm, // Neither start nor finish are located in the segment's mwm. + }; + + // Start or finish ending information. + struct Ending + { + // Fake segment id. + uint32_t m_id = 0; + // Real segments connected to the ending. + std::set m_real; + }; + static Segment GetFakeSegment(uint32_t segmentIdx) { // We currently ignore |isForward| and use FakeGraph to get ingoing/outgoing. @@ -134,18 +151,21 @@ private: std::vector & edges) const; // Checks whether ending belongs to pass-through or non-pass-through zone. - bool EndingPassThroughAllowed(FakeEnding const & ending); + bool EndingPassThroughAllowed(Ending const & ending); + // Start segment is located in a pass-through/non-pass-through area. + bool StartPassThroughAllowed(); + // Finish segment is located in a pass-through/non-pass-through area. + bool FinishPassThroughAllowed(); + + std::set GetEndingMwms(Ending const & ending) const; + SegmentLocation GetSegmentLocation(Segment const & segment) const; static uint32_t constexpr kFakeFeatureId = FakeFeatureIds::kIndexGraphStarterId; WorldGraph & m_graph; // Start segment id - uint32_t m_startId; + Ending m_start; // Finish segment id - uint32_t m_finishId; - // Start segment is located in a pass-through/non-pass-through area. - bool m_startPassThroughAllowed; - // Finish segment is located in a pass-through/non-pass-through area. - bool m_finishPassThroughAllowed; + Ending m_finish; double m_startToFinishDistanceM; FakeGraph m_fake; }; diff --git a/routing/index_router.cpp b/routing/index_router.cpp index 2520d86020..b243199c75 100644 --- a/routing/index_router.cpp +++ b/routing/index_router.cpp @@ -159,7 +159,7 @@ bool IsDeadEnd(Segment const & segment, bool isOutgoing, WorldGraph & worldGraph // If |isOutgoing| == false it's the finish. So ingoing edges are looked for. auto const getOutgoingEdgesFn = [isOutgoing](WorldGraph & graph, Segment const & u, vector & edges) { - graph.GetEdgeList(u, isOutgoing, false /* isLeap */, false /* isEnding */, edges); + graph.GetEdgeList(u, isOutgoing, false /* isLeap */, edges); }; return !CheckGraphConnectivity(segment, kDeadEndTestLimit, worldGraph, @@ -752,13 +752,11 @@ IRouter::ResultCode IndexRouter::ProcessLeaps(vector const & input, output.reserve(input.size()); auto const isNotLeap = [&](Segment const & s) { - auto real = s; - return (prevMode == WorldGraph::Mode::LeapsOnly && !starter.ConvertToReal(real)) || - (prevMode != WorldGraph::Mode::LeapsOnly && !starter.IsLeap(s.GetMwmId())); + return prevMode != WorldGraph::Mode::LeapsOnly && !starter.IsLeap(s); }; auto const isEndingToExitLeap = [&](Segment const & s) { - return prevMode == WorldGraph::Mode::LeapsOnly && !starter.IsLeap(s.GetMwmId()); + return prevMode == WorldGraph::Mode::LeapsOnly && !starter.IsLeap(s); }; auto const sameMwm = [](NumMwmId prev, NumMwmId current) { @@ -773,7 +771,7 @@ IRouter::ResultCode IndexRouter::ProcessLeaps(vector const & input, NumMwmId prevMwmId = kFakeNumMwmId; for (size_t i = 0; i < input.size(); ++i) { - Segment current = input[i]; + auto const & current = input[i]; auto const currentMwmId = current.GetMwmId(); if (isNotLeap(current)) @@ -792,13 +790,7 @@ IRouter::ResultCode IndexRouter::ProcessLeaps(vector const & input, ++i; CHECK_LESS(i, input.size(), ()); - Segment next = input[i]; - - CHECK(starter.ConvertToReal(current), ()); - CHECK(starter.ConvertToReal(next), ()); - CHECK_EQUAL( - current.GetMwmId(), next.GetMwmId(), - ("Different mwm ids for leap enter and exit, i:", i, "size of input:", input.size())); + auto const & next = input[i]; IRouter::ResultCode result = IRouter::InternalError; RoutingResult routingResult; @@ -807,31 +799,34 @@ IRouter::ResultCode IndexRouter::ProcessLeaps(vector const & input, // In case of leaps from the start to its mwm transition and from finish to mwm transition // route calculation should be made on the world graph (WorldGraph::Mode::NoLeaps). worldGraph.SetMode(WorldGraph::Mode::NoLeaps); + result = + FindPath(current, next, delegate, starter, {} /* onVisitedVertexCallback */, + [&starter](RouteWeight const & weight) { return starter.CheckLength(weight); }, + routingResult); } else { - CHECK(starter.IsLeap(current.GetMwmId()), ()); + CHECK(!IndexGraphStarter::IsFakeSegment(current), ()); + CHECK(!IndexGraphStarter::IsFakeSegment(next), ()); + CHECK_EQUAL( + current.GetMwmId(), next.GetMwmId(), + ("Different mwm ids for leap enter and exit, i:", i, "size of input:", input.size())); + + CHECK(starter.IsLeap(current), ()); // Single mwm route. worldGraph.SetMode(WorldGraph::Mode::SingleMwm); // It's not start-to-mwm-exit leap, we already have its first segment in previous mwm. dropFirstSegment = true; + result = + FindPath(current, next, delegate, worldGraph, {} /* onVisitedVertexCallback */, + [&starter](RouteWeight const & weight) { return starter.CheckLength(weight); }, + routingResult); } - result = - FindPath(current, next, delegate, worldGraph, {} /* onVisitedVertexCallback */, - [&starter](RouteWeight const & weight) { return starter.CheckLength(weight); }, - routingResult); - if (result != IRouter::NoError) return result; CHECK(!routingResult.m_path.empty(), ()); - - // Start and finish segments may be changed by starter.ConvertToReal. It was necessary to use - // them in worldGraph but we need to reset them to original values. - routingResult.m_path.front() = input[i - 1]; - routingResult.m_path.back() = input[i]; - output.insert( output.end(), dropFirstSegment ? routingResult.m_path.cbegin() + 1 : routingResult.m_path.cbegin(), diff --git a/routing/single_vehicle_world_graph.cpp b/routing/single_vehicle_world_graph.cpp index 15e651c8fe..6e091ab808 100644 --- a/routing/single_vehicle_world_graph.cpp +++ b/routing/single_vehicle_world_graph.cpp @@ -15,30 +15,10 @@ SingleVehicleWorldGraph::SingleVehicleWorldGraph(unique_ptr cross CHECK(m_estimator, ()); } -void SingleVehicleWorldGraph::GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, - bool isEnding, vector & edges) +void SingleVehicleWorldGraph::GetEdgeList(Segment const & segment, bool isOutgoing, + bool isLeap, vector & edges) { - // If mode is LeapsOnly and |isEnding| == true we need to connect segment to transitions. - // If |isOutgoing| == true connects |segment| with all exits of mwm. - // If |isOutgoing| == false connects all enters to mwm with |segment|. - if (m_mode == Mode::LeapsOnly && isEnding) - { - edges.clear(); - m2::PointD const & segmentPoint = GetPoint(segment, true /* front */); - - // Note. If |isOutgoing| == true it's necessary to add edges which connect the start with all - // exits of its mwm. So |isEnter| below should be set to false. - // If |isOutgoing| == false all enters of the finish mwm should be connected with the finish - // point. So |isEnter| below should be set to true. - m_crossMwmGraph->ForEachTransition( - segment.GetMwmId(), !isOutgoing /* isEnter */, [&](Segment const & transition) { - edges.emplace_back(transition, RouteWeight(m_estimator->CalcLeapWeight( - segmentPoint, GetPoint(transition, isOutgoing)))); - }); - return; - } - - if (m_mode != Mode::NoLeaps && m_mode != Mode::SingleMwm && (isLeap || m_mode == Mode::LeapsOnly)) + if ((m_mode == Mode::LeapsIfPossible && isLeap) || m_mode == Mode::LeapsOnly) { CHECK(m_crossMwmGraph, ()); if (m_crossMwmGraph->IsTransition(segment, isOutgoing)) @@ -80,14 +60,14 @@ void SingleVehicleWorldGraph::GetOutgoingEdgesList(Segment const & segment, vector & edges) { edges.clear(); - GetEdgeList(segment, true /* isOutgoing */, false /* isLeap */, false /* isEnding */, edges); + GetEdgeList(segment, true /* isOutgoing */, false /* isLeap */, edges); } void SingleVehicleWorldGraph::GetIngoingEdgesList(Segment const & segment, vector & edges) { edges.clear(); - GetEdgeList(segment, false /* isOutgoing */, false /* isLeap */, false /* isEnding */, edges); + GetEdgeList(segment, false /* isOutgoing */, false /* isLeap */, edges); } RouteWeight SingleVehicleWorldGraph::HeuristicCostEstimate(Segment const & from, Segment const & to) @@ -107,6 +87,12 @@ RouteWeight SingleVehicleWorldGraph::CalcSegmentWeight(Segment const & segment) segment, GetRoadGeometry(segment.GetMwmId(), segment.GetFeatureId()))); } +RouteWeight SingleVehicleWorldGraph::CalcLeapWeight(m2::PointD const & from, + m2::PointD const & to) const +{ + return RouteWeight(m_estimator->CalcLeapWeight(from, to)); +} + RouteWeight SingleVehicleWorldGraph::CalcOffroadWeight(m2::PointD const & from, m2::PointD const & to) const { @@ -118,6 +104,11 @@ bool SingleVehicleWorldGraph::LeapIsAllowed(NumMwmId mwmId) const return m_estimator->LeapIsAllowed(mwmId); } +vector const & SingleVehicleWorldGraph::GetTransitions(NumMwmId numMwmId, bool isEnter) +{ + return m_crossMwmGraph->GetTransitions(numMwmId, isEnter); +} + unique_ptr SingleVehicleWorldGraph::GetTransitInfo(Segment const &) { return {}; } RoadGeometry const & SingleVehicleWorldGraph::GetRoadGeometry(NumMwmId mwmId, uint32_t featureId) @@ -138,6 +129,7 @@ void SingleVehicleWorldGraph::GetTwins(Segment const & segment, bool isOutgoing, // in different mwms. But if we have mwms with different versions and feature // was moved in one of them we can have nonzero weight here. double const weight = m_estimator->CalcHeuristic(from, to); + // @todo(t.yan) fix weight for ingoing edges https://jira.mail.ru/browse/MAPSME-5953 edges.emplace_back(twin, RouteWeight(weight)); } } diff --git a/routing/single_vehicle_world_graph.hpp b/routing/single_vehicle_world_graph.hpp index ab5365cba8..f4898ba4ae 100644 --- a/routing/single_vehicle_world_graph.hpp +++ b/routing/single_vehicle_world_graph.hpp @@ -27,7 +27,7 @@ public: std::shared_ptr estimator); // WorldGraph overrides: - void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, bool isEnding, + void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, std::vector & edges) override; bool CheckLength(RouteWeight const &, double) const override { return true; } Junction const & GetJunction(Segment const & segment, bool front) override; @@ -42,8 +42,10 @@ public: RouteWeight HeuristicCostEstimate(Segment const & from, Segment const & to) override; RouteWeight HeuristicCostEstimate(m2::PointD const & from, m2::PointD const & to) override; RouteWeight CalcSegmentWeight(Segment const & segment) override; + RouteWeight CalcLeapWeight(m2::PointD const & from, m2::PointD const & to) const override; RouteWeight CalcOffroadWeight(m2::PointD const & from, m2::PointD const & to) const override; bool LeapIsAllowed(NumMwmId mwmId) const override; + std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) override; std::unique_ptr GetTransitInfo(Segment const & segment) override; // This method should be used for tests only diff --git a/routing/transit_world_graph.cpp b/routing/transit_world_graph.cpp index 6101b2a638..5f14c27449 100644 --- a/routing/transit_world_graph.cpp +++ b/routing/transit_world_graph.cpp @@ -25,8 +25,8 @@ TransitWorldGraph::TransitWorldGraph(unique_ptr crossMwmGraph, CHECK(m_estimator, ()); } -void TransitWorldGraph::GetEdgeList(Segment const & segment, bool isOutgoing, bool /* isLeap */, - bool /* isEnding */, vector & edges) +void TransitWorldGraph::GetEdgeList(Segment const & segment, bool isOutgoing, + bool /* isLeap */, vector & edges) { auto & transitGraph = GetTransitGraph(segment.GetMwmId()); @@ -102,13 +102,13 @@ void TransitWorldGraph::ClearCachedGraphs() void TransitWorldGraph::GetOutgoingEdgesList(Segment const & segment, vector & edges) { edges.clear(); - GetEdgeList(segment, true /* isOutgoing */, false /* isLeap */, false /* isEnding */, edges); + GetEdgeList(segment, true /* isOutgoing */, false /* isLeap */, edges); } void TransitWorldGraph::GetIngoingEdgesList(Segment const & segment, vector & edges) { edges.clear(); - GetEdgeList(segment, false /* isOutgoing */, false /* isLeap */, false /* isEnding */, edges); + GetEdgeList(segment, false /* isOutgoing */, false /* isLeap */, edges); } RouteWeight TransitWorldGraph::HeuristicCostEstimate(Segment const & from, Segment const & to) @@ -133,6 +133,11 @@ RouteWeight TransitWorldGraph::CalcSegmentWeight(Segment const & segment) segment, GetRealRoadGeometry(segment.GetMwmId(), segment.GetFeatureId()))); } +RouteWeight TransitWorldGraph::CalcLeapWeight(m2::PointD const & from, m2::PointD const & to) const +{ + return RouteWeight(m_estimator->CalcLeapWeight(from, to)); +} + RouteWeight TransitWorldGraph::CalcOffroadWeight(m2::PointD const & from, m2::PointD const & to) const { @@ -141,6 +146,11 @@ RouteWeight TransitWorldGraph::CalcOffroadWeight(m2::PointD const & from, bool TransitWorldGraph::LeapIsAllowed(NumMwmId /* mwmId */) const { return false; } +vector const & TransitWorldGraph::GetTransitions(NumMwmId numMwmId, bool isEnter) +{ + return kEmptyTransitions; +} + unique_ptr TransitWorldGraph::GetTransitInfo(Segment const & segment) { if (!TransitGraph::IsTransitSegment(segment)) @@ -176,6 +186,7 @@ void TransitWorldGraph::GetTwins(Segment const & segment, bool isOutgoing, // in different mwms. But if we have mwms with different versions and feature // was moved in one of them we can have nonzero weight here. double const weight = m_estimator->CalcHeuristic(from, to); + // @todo(t.yan) fix weight for ingoing edges https://jira.mail.ru/browse/MAPSME-5953 edges.emplace_back(twin, RouteWeight(weight)); } } diff --git a/routing/transit_world_graph.hpp b/routing/transit_world_graph.hpp index 25e6ef97dc..241daaddc0 100644 --- a/routing/transit_world_graph.hpp +++ b/routing/transit_world_graph.hpp @@ -30,7 +30,7 @@ public: std::shared_ptr estimator); // WorldGraph overrides: - void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, bool isEnding, + void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, std::vector & edges) override; bool CheckLength(RouteWeight const & weight, double startToFinishDistanceM) const override { @@ -51,8 +51,10 @@ public: RouteWeight HeuristicCostEstimate(Segment const & from, Segment const & to) override; RouteWeight HeuristicCostEstimate(m2::PointD const & from, m2::PointD const & to) override; RouteWeight CalcSegmentWeight(Segment const & segment) override; + RouteWeight CalcLeapWeight(m2::PointD const & from, m2::PointD const & to) const override; RouteWeight CalcOffroadWeight(m2::PointD const & from, m2::PointD const & to) const override; bool LeapIsAllowed(NumMwmId mwmId) const override; + std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) override; std::unique_ptr GetTransitInfo(Segment const & segment) override; private: @@ -75,5 +77,6 @@ private: std::shared_ptr m_estimator; std::vector m_twins; Mode m_mode = Mode::NoLeaps; + std::vector const kEmptyTransitions = {}; }; } // namespace routing diff --git a/routing/world_graph.hpp b/routing/world_graph.hpp index 280a812974..4beac6ef5e 100644 --- a/routing/world_graph.hpp +++ b/routing/world_graph.hpp @@ -39,9 +39,7 @@ public: virtual ~WorldGraph() = default; - // |isEnding| == true iff |segment| is first or last segment of the route. Needed because first and - // last segments may need special processing. - virtual void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, bool isEnding, + virtual void GetEdgeList(Segment const & segment, bool isOutgoing, bool isLeap, std::vector & edges) = 0; // Checks whether path length meets restrictions. Restrictions may depend on the distance from @@ -67,10 +65,14 @@ public: virtual RouteWeight HeuristicCostEstimate(Segment const & from, Segment const & to) = 0; virtual RouteWeight HeuristicCostEstimate(m2::PointD const & from, m2::PointD const & to) = 0; virtual RouteWeight CalcSegmentWeight(Segment const & segment) = 0; + virtual RouteWeight CalcLeapWeight(m2::PointD const & from, m2::PointD const & to) const = 0; virtual RouteWeight CalcOffroadWeight(m2::PointD const & from, m2::PointD const & to) const = 0; virtual bool LeapIsAllowed(NumMwmId mwmId) const = 0; - // Returns transit-specific information for segment. For nontransit segments returns nullptr. + /// \returns transitions for mwm with id |numMwmId|. + virtual std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) = 0; + + /// \returns transit-specific information for segment. For nontransit segments returns nullptr. virtual std::unique_ptr GetTransitInfo(Segment const & segment) = 0; };