diff --git a/drape_frontend/transit_scheme_builder.cpp b/drape_frontend/transit_scheme_builder.cpp index d4e2eaf8d4..71a3ff0a29 100644 --- a/drape_frontend/transit_scheme_builder.cpp +++ b/drape_frontend/transit_scheme_builder.cpp @@ -336,7 +336,7 @@ void TransitSchemeBuilder::BuildScheme(ref_ptr context, void TransitSchemeBuilder::CollectStops(TransitDisplayInfo const & transitDisplayInfo, MwmSet::MwmId const & mwmId, MwmSchemeData & scheme) { - for (auto const & stopInfo : transitDisplayInfo.m_stops) + for (auto const & stopInfo : transitDisplayInfo.m_stopsSubway) { routing::transit::Stop const & stop = stopInfo.second; if (stop.GetTransferId() != routing::transit::kInvalidTransferId) @@ -345,19 +345,19 @@ void TransitSchemeBuilder::CollectStops(TransitDisplayInfo const & transitDispla FillStopParams(transitDisplayInfo, mwmId, stop, stopNode); } - for (auto const & stopInfo : transitDisplayInfo.m_transfers) + for (auto const & stopInfo : transitDisplayInfo.m_transfersSubway) { routing::transit::Transfer const & transfer = stopInfo.second; auto & stopNode = scheme.m_transfers[transfer.GetId()]; for (auto stopId : transfer.GetStopIds()) { - if (transitDisplayInfo.m_stops.find(stopId) == transitDisplayInfo.m_stops.end()) + if (transitDisplayInfo.m_stopsSubway.find(stopId) == transitDisplayInfo.m_stopsSubway.end()) { LOG(LWARNING, ("Invalid stop", stopId, "in transfer", transfer.GetId())); continue; } - routing::transit::Stop const & stop = transitDisplayInfo.m_stops.at(stopId); + routing::transit::Stop const & stop = transitDisplayInfo.m_stopsSubway.at(stopId); FillStopParams(transitDisplayInfo, mwmId, stop, stopNode); } stopNode.m_isTransfer = true; @@ -368,7 +368,7 @@ void TransitSchemeBuilder::CollectStops(TransitDisplayInfo const & transitDispla void TransitSchemeBuilder::CollectLines(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme) { std::multimap linesLengths; - for (auto const & line : transitDisplayInfo.m_lines) + for (auto const & line : transitDisplayInfo.m_linesSubway) { auto const lineId = line.second.GetId(); size_t stopsCount = 0; @@ -381,7 +381,7 @@ void TransitSchemeBuilder::CollectLines(TransitDisplayInfo const & transitDispla for (auto const & pair : linesLengths) { auto const lineId = pair.second; - scheme.m_lines[lineId] = LineParams(transitDisplayInfo.m_lines.at(lineId).GetColor(), depth); + scheme.m_lines[lineId] = LineParams(transitDisplayInfo.m_linesSubway.at(lineId).GetColor(), depth); depth += kDepthPerLine; } } @@ -389,14 +389,14 @@ void TransitSchemeBuilder::CollectLines(TransitDisplayInfo const & transitDispla void TransitSchemeBuilder::CollectShapes(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme) { std::map> roads; - for (auto const & line : transitDisplayInfo.m_lines) + for (auto const & line : transitDisplayInfo.m_linesSubway) { auto const lineId = line.second.GetId(); auto const roadId = GetRouteId(lineId); roads[roadId].push_back(lineId); } - for (auto const & line : transitDisplayInfo.m_lines) + for (auto const & line : transitDisplayInfo.m_linesSubway) { auto const lineId = line.second.GetId(); auto const roadId = GetRouteId(lineId); @@ -422,7 +422,7 @@ void TransitSchemeBuilder::FindShapes(routing::transit::StopId stop1Id, routing: if (sameLineId == lineId) continue; - auto const & sameLine = transitDisplayInfo.m_lines.at(sameLineId); + auto const & sameLine = transitDisplayInfo.m_linesSubway.at(sameLineId); auto const & sameStopsRanges = sameLine.GetStopIds(); for (auto const & sameStops : sameStopsRanges) { @@ -452,11 +452,11 @@ void TransitSchemeBuilder::AddShape(TransitDisplayInfo const & transitDisplayInf routing::transit::LineId lineId, MwmSchemeData & scheme) { - auto const stop1It = transitDisplayInfo.m_stops.find(stop1Id); - ASSERT(stop1It != transitDisplayInfo.m_stops.end(), ()); + auto const stop1It = transitDisplayInfo.m_stopsSubway.find(stop1Id); + ASSERT(stop1It != transitDisplayInfo.m_stopsSubway.end(), ()); - auto const stop2It = transitDisplayInfo.m_stops.find(stop2Id); - ASSERT(stop2It != transitDisplayInfo.m_stops.end(), ()); + auto const stop2It = transitDisplayInfo.m_stopsSubway.find(stop2Id); + ASSERT(stop2It != transitDisplayInfo.m_stopsSubway.end(), ()); auto const transfer1Id = stop1It->second.GetTransferId(); auto const transfer2Id = stop2It->second.GetTransferId(); @@ -465,24 +465,24 @@ void TransitSchemeBuilder::AddShape(TransitDisplayInfo const & transitDisplayInf : stop1Id, transfer2Id != routing::transit::kInvalidTransferId ? transfer2Id : stop2Id); - auto it = transitDisplayInfo.m_shapes.find(shapeId); + auto it = transitDisplayInfo.m_shapesSubway.find(shapeId); bool isForward = true; - if (it == transitDisplayInfo.m_shapes.end()) + if (it == transitDisplayInfo.m_shapesSubway.end()) { isForward = false; shapeId = routing::transit::ShapeId(shapeId.GetStop2Id(), shapeId.GetStop1Id()); - it = transitDisplayInfo.m_shapes.find(shapeId); + it = transitDisplayInfo.m_shapesSubway.find(shapeId); } - if (it == transitDisplayInfo.m_shapes.end()) + if (it == transitDisplayInfo.m_shapesSubway.end()) return; - ASSERT(it != transitDisplayInfo.m_shapes.end(), ()); + ASSERT(it != transitDisplayInfo.m_shapesSubway.end(), ()); auto const itScheme = scheme.m_shapes.find(shapeId); if (itScheme == scheme.m_shapes.end()) { - auto const & polyline = transitDisplayInfo.m_shapes.at(it->first).GetPolyline(); + auto const & polyline = transitDisplayInfo.m_shapesSubway.at(it->first).GetPolyline(); if (isForward) scheme.m_shapes[shapeId].m_forwardLines.push_back(lineId); else diff --git a/routing/routing_tests/index_graph_tools.cpp b/routing/routing_tests/index_graph_tools.cpp index 3bf088649b..c25f645a31 100644 --- a/routing/routing_tests/index_graph_tools.cpp +++ b/routing/routing_tests/index_graph_tools.cpp @@ -2,12 +2,12 @@ #include "testing/testing.hpp" -#include "routing/geometry.hpp" - #include "routing/base/routing_result.hpp" - +#include "routing/geometry.hpp" #include "routing/routing_helpers.hpp" +#include "transit/transit_version.hpp" + #include "base/assert.hpp" #include "base/math.hpp" @@ -453,8 +453,9 @@ unique_ptr BuildWorldGraph(unique_ptr geo auto indexGraph = make_unique(make_shared(move(geometryLoader)), estimator); indexGraph->Import(joints); - auto transitGraph = make_unique(kTestNumMwmId, estimator); - TransitGraph::GateEndings gateEndings; + auto transitGraph = + make_unique(::transit::TransitVersion::OnlySubway, kTestNumMwmId, estimator); + TransitGraph::Endings gateEndings; MakeGateEndings(transitData.GetGates(), kTestNumMwmId, *indexGraph, gateEndings); transitGraph->Fill(transitData, gateEndings); diff --git a/routing/transit_graph.cpp b/routing/transit_graph.cpp index 8ffdee6035..0af1b45dcf 100644 --- a/routing/transit_graph.cpp +++ b/routing/transit_graph.cpp @@ -12,6 +12,9 @@ namespace routing namespace { using namespace std; +// TODO(o.khlopkova) Replace this constant with implementation of intervals calculation on the +// gtfs converter step. +size_t constexpr kDefaultIntervalS = 60 * 60; // 1 hour. Segment GetReverseSegment(Segment const & segment) { @@ -40,11 +43,14 @@ bool TransitGraph::IsTransitSegment(Segment const & segment) return IsTransitFeature(segment.GetFeatureId()); } -TransitGraph::TransitGraph(NumMwmId numMwmId, shared_ptr estimator) - : m_mwmId(numMwmId), m_estimator(estimator) +TransitGraph::TransitGraph(::transit::TransitVersion transitVersion, NumMwmId numMwmId, + shared_ptr estimator) + : m_transitVersion(transitVersion), m_mwmId(numMwmId), m_estimator(estimator) { } +::transit::TransitVersion TransitGraph::GetTransitVersion() const { return m_transitVersion; } + LatLonWithAltitude const & TransitGraph::GetJunction(Segment const & segment, bool front) const { @@ -57,18 +63,48 @@ RouteWeight TransitGraph::CalcSegmentWeight(Segment const & segment, EdgeEstimator::Purpose purpose) const { CHECK(IsTransitSegment(segment), ("Nontransit segment passed to TransitGraph.")); - if (IsGate(segment)) - { - auto const weight = GetGate(segment).GetWeight(); - return RouteWeight(weight /* weight */, 0 /* numPassThroughChanges */, 0 /* numAccessChanges */, - 0 /* numAccessConditionalPenalties */, weight /* transitTime */); - } - if (IsEdge(segment)) + if (m_transitVersion == ::transit::TransitVersion::OnlySubway) { - auto const weight = GetEdge(segment).GetWeight(); - return RouteWeight(weight /* weight */, 0 /* numPassThroughChanges */, 0 /* numAccessChanges */, - 0 /* numAccessConditionalPenalties */, weight /* transitTime */); + if (IsGate(segment)) + { + auto const weight = GetGate(segment).GetWeight(); + return RouteWeight(weight, 0 /* numPassThroughChanges */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + weight /* transitTime */); + } + + if (IsEdge(segment)) + { + auto const weight = GetEdge(segment).GetWeight(); + return RouteWeight(weight, 0 /* numPassThroughChanges */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + weight /* transitTime */); + } + } + else if (m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + { + if (IsGate(segment)) + { + // TODO (o.khlopkova) Manage different weights for different stops linked to the gate. + auto const weight = kDefaultIntervalS; + return RouteWeight(weight /* weight */, 0 /* numPassThroughChanges */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + weight /* transitTime */); + } + + if (IsEdge(segment)) + { + auto weight = GetEdgePT(segment).GetWeight(); + CHECK_NOT_EQUAL(weight, 0, (segment)); + return RouteWeight(weight /* weight */, 0 /* numPassThroughChanges */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + weight /* transitTime */); + } + } + else + { + UNREACHABLE(); } return RouteWeight( @@ -78,30 +114,67 @@ RouteWeight TransitGraph::CalcSegmentWeight(Segment const & segment, RouteWeight TransitGraph::GetTransferPenalty(Segment const & from, Segment const & to) const { - // We need to wait transport and apply additional penalty only if we change to transit::Edge. - if (!IsEdge(to)) - return GetAStarWeightZero(); + if (m_transitVersion == ::transit::TransitVersion::OnlySubway) + { + // We need to wait transport and apply additional penalty only if we change to transit::Edge. + if (!IsEdge(to)) + return GetAStarWeightZero(); - auto const & edgeTo = GetEdge(to); + auto const & edgeTo = GetEdge(to); - // We are changing to transfer and do not need to apply extra penalty here. We'll do it while - // changing from transfer. - if (edgeTo.GetTransfer()) - return GetAStarWeightZero(); + // We are changing to transfer and do not need to apply extra penalty here. We'll do it while + // changing from transfer. + if (edgeTo.GetTransfer()) + return GetAStarWeightZero(); - auto const lineIdTo = edgeTo.GetLineId(); + auto const lineIdTo = edgeTo.GetLineId(); - if (IsEdge(from) && GetEdge(from).GetLineId() == lineIdTo) - return GetAStarWeightZero(); + if (IsEdge(from) && GetEdge(from).GetLineId() == lineIdTo) + return GetAStarWeightZero(); - // We need to apply extra penalty when: - // 1. |from| is gate, |to| is edge - // 2. |from| is transfer, |to| is edge - // 3. |from| is edge, |to| is edge from another line directly connected to |from|. - auto const it = m_transferPenalties.find(lineIdTo); - CHECK(it != m_transferPenalties.cend(), ("Segment", to, "belongs to unknown line:", lineIdTo)); - return RouteWeight(it->second /* weight */, 0 /* nonPassThrougCross */, 0 /* numAccessChanges */, - 0 /* numAccessConditionalPenalties */, it->second /* transitTime */); + // We need to apply extra penalty when: + // 1. |from| is gate, |to| is edge + // 2. |from| is transfer, |to| is edge + // 3. |from| is edge, |to| is edge from another line directly connected to |from|. + auto const it = m_transferPenalties.find(lineIdTo); + CHECK(it != m_transferPenalties.cend(), ("Segment", to, "belongs to unknown line:", lineIdTo)); + return RouteWeight(it->second /* weight */, 0 /* nonPassThrougCross */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + it->second /* transitTime */); + } + else if (m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + { + // We need to wait transport and apply additional penalty only if we change to transit::Edge. + if (!IsEdge(to)) + return GetAStarWeightZero(); + + auto const & edgeTo = GetEdgePT(to); + + // We are changing to transfer and do not need to apply extra penalty here. We'll do it while + // changing from transfer. + if (edgeTo.IsTransfer()) + return GetAStarWeightZero(); + + auto const lineIdTo = edgeTo.GetLineId(); + + if (IsEdge(from) && GetEdgePT(from).GetLineId() == lineIdTo) + return GetAStarWeightZero(); + + // We need to apply extra penalty when: + // 1. |from| is gate, |to| is edge + // 2. |from| is transfer, |to| is edge + // 3. |from| is edge, |to| is edge from another line directly connected to |from|. + auto const it = m_transferPenaltiesPT.find(lineIdTo); + CHECK(it != m_transferPenaltiesPT.end(), ("Segment", to, "belongs to unknown line:", lineIdTo)); + CHECK(!it->second.empty(), ()); + size_t const headwayS = it->second.front().m_headwayS; + + return RouteWeight(static_cast(headwayS) /* weight */, 0 /* nonPassThroughCross */, + 0 /* numAccessChanges */, 0 /* numAccessConditionalPenalties */, + headwayS /* transitTime */); + } + + UNREACHABLE(); } void TransitGraph::GetTransitEdges(Segment const & segment, bool isOutgoing, @@ -127,8 +200,92 @@ bool TransitGraph::FindReal(Segment const & fake, Segment & real) const return m_fake.FindReal(fake, real); } -void TransitGraph::Fill(transit::GraphData const & transitData, GateEndings const & gateEndings) +void TransitGraph::Fill(::transit::experimental::TransitData const & transitData, + Endings const & stopEndings, Endings const & gateEndings) { + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + + for (auto const & line : transitData.GetLines()) + { + m_transferPenaltiesPT[line.GetId()].reserve(line.GetIntervals().size()); + + for (auto const & interval : line.GetIntervals()) + { + m_transferPenaltiesPT[line.GetId()].emplace_back(interval.m_headwayS / 2, + interval.m_timeIntervals); + } + + // TODO(o.khlopkova) Decide what to do with lines with empty intervals. + if (m_transferPenaltiesPT[line.GetId()].empty()) + { + m_transferPenaltiesPT[line.GetId()].emplace_back(kDefaultIntervalS / 2, + osmoh::OpeningHours("24/7")); + } + } + + map stopCoords; + + for (auto const & stop : transitData.GetStops()) + { + stopCoords[stop.GetId()] = + LatLonWithAltitude(mercator::ToLatLon(stop.GetPoint()), geometry::kDefaultAltitudeMeters); + } + + StopToSegmentsMap stopToBack; + StopToSegmentsMap stopToFront; + StopToSegmentsMap outgoing; + StopToSegmentsMap ingoing; + + // It's important to add transit edges first to ensure fake segment id for particular edge is edge + // order in mwm. We use edge fake segments in cross-mwm section and they should be stable. + auto const & edges = transitData.GetEdges(); + CHECK_EQUAL(m_fake.GetSize(), 0, ()); + for (size_t i = 0; i < edges.size(); ++i) + { + auto const & edge = edges[i]; + CHECK_NOT_EQUAL(edge.GetWeight(), transit::kInvalidWeight, ("Edge should have valid weight.")); + auto const edgeSegment = AddEdge(edge, stopCoords, stopToBack, stopToFront); + // Checks fake feature ids have consecutive numeration starting from + // FakeFeatureIds::kTransitGraphFeaturesStart. + CHECK_EQUAL(edgeSegment.GetFeatureId(), i + FakeFeatureIds::kTransitGraphFeaturesStart, ()); + outgoing[edge.GetStop1Id()].insert(edgeSegment); + ingoing[edge.GetStop2Id()].insert(edgeSegment); + } + CHECK_EQUAL(m_fake.GetSize(), edges.size(), ()); + + for (auto const & gate : transitData.GetGates()) + { + CHECK(!gate.GetStopsWithWeight().empty(), ("Gate should have valid weight.")); + + // Gate ending may have empty projections vector. It means gate is not connected to roads. + auto const it = gateEndings.find(gate.GetOsmId()); + if (it != gateEndings.end()) + { + if (gate.IsEntrance()) + AddGate(gate, it->second, stopCoords, true /* isEnter */, stopToBack, stopToFront); + if (gate.IsExit()) + AddGate(gate, it->second, stopCoords, false /* isEnter */, stopToBack, stopToFront); + } + } + + for (auto const & stop : transitData.GetStops()) + { + // Stop ending may have empty projections vector. It means stop is not connected to roads. + auto const it = stopEndings.find(stop.GetId()); + if (it != stopEndings.end()) + { + AddStop(stop, it->second, stopCoords, stopToBack, stopToFront); + } + } + + AddConnections(outgoing, stopToBack, stopToFront, true /* isOutgoing */); + AddConnections(ingoing, stopToBack, stopToFront, false /* isOutgoing */); +} + +void TransitGraph::Fill(transit::GraphData const & transitData, Endings const & gateEndings) +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + // Line has information about transit interval. // We assume arrival time has uniform distribution with min value |0| and max value |line.GetInterval()|. // Expected value of time to wait transport for particular line is |line.GetInterval() / 2|. @@ -183,12 +340,20 @@ void TransitGraph::Fill(transit::GraphData const & transitData, GateEndings cons bool TransitGraph::IsGate(Segment const & segment) const { - return m_segmentToGate.count(segment) > 0; + if (m_transitVersion == ::transit::TransitVersion::OnlySubway) + return m_segmentToGate.count(segment) > 0; + else if (m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + return m_segmentToGatePT.count(segment) > 0; + UNREACHABLE(); } bool TransitGraph::IsEdge(Segment const & segment) const { - return m_segmentToEdge.count(segment) > 0; + if (m_transitVersion == ::transit::TransitVersion::OnlySubway) + return m_segmentToEdge.count(segment) > 0; + else if (m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + return m_segmentToEdgePT.count(segment) > 0; + UNREACHABLE(); } transit::Gate const & TransitGraph::GetGate(Segment const & segment) const @@ -198,6 +363,14 @@ transit::Gate const & TransitGraph::GetGate(Segment const & segment) const return it->second; } +::transit::experimental::Gate const & TransitGraph::GetGatePT(Segment const & segment) const +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + auto const it = m_segmentToGatePT.find(segment); + CHECK(it != m_segmentToGatePT.cend(), ("Unknown transit segment.")); + return it->second; +} + transit::Edge const & TransitGraph::GetEdge(Segment const & segment) const { auto const it = m_segmentToEdge.find(segment); @@ -205,6 +378,14 @@ transit::Edge const & TransitGraph::GetEdge(Segment const & segment) const return it->second; } +::transit::experimental::Edge const & TransitGraph::GetEdgePT(Segment const & segment) const +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + auto const it = m_segmentToEdgePT.find(segment); + CHECK(it != m_segmentToEdgePT.cend(), ("Unknown transit segment", segment)); + return it->second; +} + Segment TransitGraph::GetTransitSegment(uint32_t featureId) const { CHECK(IsTransitFeature(featureId), ("Feature id is out of transit id interval.")); @@ -229,6 +410,8 @@ void TransitGraph::AddGate(transit::Gate const & gate, FakeEnding const & ending bool isEnter, StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront) { + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + Segment const dummy = Segment(); for (auto const & projection : ending.m_projections) { @@ -279,10 +462,130 @@ void TransitGraph::AddGate(transit::Gate const & gate, FakeEnding const & ending } } +void TransitGraph::AddGate(::transit::experimental::Gate const & gate, FakeEnding const & ending, + std::map const & stopCoords, + bool isEnter, StopToSegmentsMap & stopToBack, + StopToSegmentsMap & stopToFront) +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + + Segment const dummy = Segment(); + for (auto const & projection : ending.m_projections) + { + // Add projection edges + auto const projectionSegment = GetNewTransitSegment(); + FakeVertex projectionVertex( + projection.m_segment.GetMwmId(), isEnter ? projection.m_junction : ending.m_originJunction, + isEnter ? ending.m_originJunction : projection.m_junction, FakeVertex::Type::PureFake); + m_fake.AddStandaloneVertex(projectionSegment, projectionVertex); + + // Add fake parts of real + FakeVertex forwardPartOfReal( + projection.m_segment.GetMwmId(), isEnter ? projection.m_segmentBack : projection.m_junction, + isEnter ? projection.m_junction : projection.m_segmentFront, FakeVertex::Type::PartOfReal); + auto const fakeForwardSegment = GetNewTransitSegment(); + m_fake.AddVertex(projectionSegment, fakeForwardSegment, forwardPartOfReal, + !isEnter /* isOutgoing */, true /* isPartOfReal */, projection.m_segment); + + if (!projection.m_isOneWay) + { + FakeVertex backwardPartOfReal(projection.m_segment.GetMwmId(), + isEnter ? projection.m_segmentFront : projection.m_junction, + isEnter ? projection.m_junction : projection.m_segmentBack, + FakeVertex::Type::PartOfReal); + auto const fakeBackwardSegment = GetNewTransitSegment(); + m_fake.AddVertex(projectionSegment, fakeBackwardSegment, backwardPartOfReal, + !isEnter /* isOutgoing */, true /* isPartOfReal */, + GetReverseSegment(projection.m_segment)); + } + + // Connect gate to stops + for (auto const & timeFromGateToStop : gate.GetStopsWithWeight()) + { + auto const stopId = timeFromGateToStop.m_stopId; + auto const gateSegment = GetNewTransitSegment(); + auto const stopIt = stopCoords.find(stopId); + CHECK(stopIt != stopCoords.end(), ("Stop", stopId, "does not exist.")); + FakeVertex gateVertex( + projection.m_segment.GetMwmId(), isEnter ? ending.m_originJunction : stopIt->second, + isEnter ? stopIt->second : ending.m_originJunction, FakeVertex::Type::PureFake); + m_fake.AddVertex(projectionSegment, gateSegment, gateVertex, isEnter /* isOutgoing */, + false /* isPartOfReal */, dummy /* realSegment */); + m_segmentToGatePT[gateSegment] = gate; + if (isEnter) + stopToFront[stopId].insert(gateSegment); + else + stopToBack[stopId].insert(gateSegment); + } + } +} + +void TransitGraph::AddStop(::transit::experimental::Stop const & stop, FakeEnding const & ending, + std::map const & stopCoords, + StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront) +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + + Segment const dummy = Segment(); + for (bool isEnter : {true, false}) + { + for (auto const & projection : ending.m_projections) + { + // Add projection edges + auto const projectionSegment = GetNewTransitSegment(); + FakeVertex projectionVertex(projection.m_segment.GetMwmId(), + isEnter ? projection.m_junction : ending.m_originJunction, + isEnter ? ending.m_originJunction : projection.m_junction, + FakeVertex::Type::PureFake); + m_fake.AddStandaloneVertex(projectionSegment, projectionVertex); + + // Add fake parts of real + FakeVertex forwardPartOfReal(projection.m_segment.GetMwmId(), + isEnter ? projection.m_segmentBack : projection.m_junction, + isEnter ? projection.m_junction : projection.m_segmentFront, + FakeVertex::Type::PartOfReal); + auto const fakeForwardSegment = GetNewTransitSegment(); + m_fake.AddVertex(projectionSegment, fakeForwardSegment, forwardPartOfReal, + !isEnter /* isOutgoing */, true /* isPartOfReal */, projection.m_segment); + + if (!projection.m_isOneWay) + { + FakeVertex backwardPartOfReal(projection.m_segment.GetMwmId(), + isEnter ? projection.m_segmentFront : projection.m_junction, + isEnter ? projection.m_junction : projection.m_segmentBack, + FakeVertex::Type::PartOfReal); + auto const fakeBackwardSegment = GetNewTransitSegment(); + m_fake.AddVertex(projectionSegment, fakeBackwardSegment, backwardPartOfReal, + !isEnter /* isOutgoing */, true /* isPartOfReal */, + GetReverseSegment(projection.m_segment)); + } + + // Connect stop to graph. + + auto const stopId = stop.GetId(); + auto const stopSegment = GetNewTransitSegment(); + auto const stopIt = stopCoords.find(stopId); + CHECK(stopIt != stopCoords.end(), ("Stop", stopId, "does not exist.")); + FakeVertex stopVertex( + projection.m_segment.GetMwmId(), isEnter ? ending.m_originJunction : stopIt->second, + isEnter ? stopIt->second : ending.m_originJunction, FakeVertex::Type::PureFake); + m_fake.AddVertex(projectionSegment, stopSegment, stopVertex, isEnter /* isOutgoing */, + false /* isPartOfReal */, dummy /* realSegment */); + m_segmentToStopPT[stopSegment] = stop; + if (isEnter) + stopToFront[stopId].insert(stopSegment); + else + stopToBack[stopId].insert(stopSegment); + } + } +} + Segment TransitGraph::AddEdge(transit::Edge const & edge, map const & stopCoords, StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront) { + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + auto const edgeSegment = GetNewTransitSegment(); auto const stopFromId = edge.GetStop1Id(); auto const stopToId = edge.GetStop2Id(); @@ -295,6 +598,24 @@ Segment TransitGraph::AddEdge(transit::Edge const & edge, return edgeSegment; } +Segment TransitGraph::AddEdge(::transit::experimental::Edge const & edge, + std::map const & stopCoords, + StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront) +{ + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + + auto const edgeSegment = GetNewTransitSegment(); + auto const stopFromId = edge.GetStop1Id(); + auto const stopToId = edge.GetStop2Id(); + FakeVertex edgeVertex(m_mwmId, GetStopJunction(stopCoords, stopFromId), + GetStopJunction(stopCoords, stopToId), FakeVertex::Type::PureFake); + m_fake.AddStandaloneVertex(edgeSegment, edgeVertex); + m_segmentToEdgePT[edgeSegment] = edge; + stopToBack[stopFromId].insert(edgeSegment); + stopToFront[stopToId].insert(edgeSegment); + return edgeSegment; +} + void TransitGraph::AddConnections(StopToSegmentsMap const & connections, StopToSegmentsMap const & stopToBack, StopToSegmentsMap const & stopToFront, bool isOutgoing) @@ -316,8 +637,8 @@ void TransitGraph::AddConnections(StopToSegmentsMap const & connections, } } -void MakeGateEndings(vector const & gates, NumMwmId mwmId, - IndexGraph & indexGraph, TransitGraph::GateEndings & gateEndings) +void MakeGateEndings(vector const & gates, NumMwmId mwmId, IndexGraph & indexGraph, + TransitGraph::Endings & gateEndings) { for (auto const & gate : gates) { @@ -330,4 +651,32 @@ void MakeGateEndings(vector const & gates, NumMwmId mwmId, gateEndings.emplace(gate.GetOsmId(), MakeFakeEnding({real}, gate.GetPoint(), indexGraph)); } } + +void MakeGateEndings(std::vector<::transit::experimental::Gate> const & gates, NumMwmId mwmId, + IndexGraph & indexGraph, TransitGraph::Endings & gateEndings) +{ + for (auto const & gate : gates) + { + for (auto const & gateSegment : gate.GetBestPedestrianSegments()) + { + Segment const real(mwmId, gateSegment.GetFeatureId(), gateSegment.GetSegmentIdx(), + gateSegment.IsForward()); + gateEndings.emplace(gate.GetOsmId(), MakeFakeEnding(real, gate.GetPoint(), indexGraph)); + } + } +} + +void MakeStopEndings(std::vector<::transit::experimental::Stop> const & stops, NumMwmId mwmId, + IndexGraph & indexGraph, TransitGraph::Endings & stopEndings) +{ + for (auto const & stop : stops) + { + for (auto const & stopSegment : stop.GetBestPedestrianSegments()) + { + Segment const real(mwmId, stopSegment.GetFeatureId(), stopSegment.GetSegmentIdx(), + stopSegment.IsForward()); + stopEndings.emplace(stop.GetId(), MakeFakeEnding(real, stop.GetPoint(), indexGraph)); + } + } +} } // namespace routing diff --git a/routing/transit_graph.hpp b/routing/transit_graph.hpp index a0ce305350..ea57d3f863 100644 --- a/routing/transit_graph.hpp +++ b/routing/transit_graph.hpp @@ -9,8 +9,11 @@ #include "routing/route_weight.hpp" #include "routing/segment.hpp" +#include "transit/experimental/transit_data.hpp" +#include "transit/experimental/transit_types_experimental.hpp" #include "transit/transit_graph_data.hpp" #include "transit/transit_types.hpp" +#include "transit/transit_version.hpp" #include "routing_common/num_mwm_id.hpp" @@ -27,12 +30,17 @@ class IndexGraph; class TransitGraph final { public: - using GateEndings = std::map; + // Fake endings for gates (in subway transit version) or for gates and stops (in public transport + // version. + using Endings = std::map; static bool IsTransitFeature(uint32_t featureId); static bool IsTransitSegment(Segment const & segment); - TransitGraph(NumMwmId numMwmId, std::shared_ptr estimator); + TransitGraph(::transit::TransitVersion transitVersion, NumMwmId numMwmId, + std::shared_ptr estimator); + + ::transit::TransitVersion GetTransitVersion() const; LatLonWithAltitude const & GetJunction(Segment const & segment, bool front) const; RouteWeight CalcSegmentWeight(Segment const & segment, EdgeEstimator::Purpose purpose) const; @@ -42,12 +50,16 @@ public: std::set const & GetFake(Segment const & real) const; bool FindReal(Segment const & fake, Segment & real) const; - void Fill(transit::GraphData const & transitData, GateEndings const & gateEndings); + void Fill(::transit::experimental::TransitData const & transitData, Endings const & stopEndings, + Endings const & gateEndings); + void Fill(transit::GraphData const & transitData, Endings const & gateEndings); bool IsGate(Segment const & segment) const; bool IsEdge(Segment const & segment) const; transit::Gate const & GetGate(Segment const & segment) const; + ::transit::experimental::Gate const & GetGatePT(Segment const & segment) const; transit::Edge const & GetEdge(Segment const & segment) const; + ::transit::experimental::Edge const & GetEdgePT(Segment const & segment) const; private: using StopToSegmentsMap = std::map>; @@ -60,23 +72,52 @@ private: void AddGate(transit::Gate const & gate, FakeEnding const & ending, std::map const & stopCoords, bool isEnter, StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront); + + void AddGate(::transit::experimental::Gate const & gate, FakeEnding const & ending, + std::map const & stopCoords, bool isEnter, + StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront); + + void AddStop(::transit::experimental::Stop const & stop, FakeEnding const & ending, + std::map const & stopCoords, + StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront); + // Adds transit edge to fake graph, returns corresponding transit segment. Also adds gate to // temporary stopToBack, stopToFront maps used while TransitGraph::Fill. Segment AddEdge(transit::Edge const & edge, std::map const & stopCoords, StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront); + + Segment AddEdge(::transit::experimental::Edge const & edge, + std::map const & stopCoords, + StopToSegmentsMap & stopToBack, StopToSegmentsMap & stopToFront); + // Adds connections to fake graph. void AddConnections(StopToSegmentsMap const & connections, StopToSegmentsMap const & stopToBack, StopToSegmentsMap const & stopToFront, bool isOutgoing); + ::transit::TransitVersion const m_transitVersion; NumMwmId const m_mwmId = kFakeNumMwmId; std::shared_ptr m_estimator; FakeGraph m_fake; + + // Fields for working with OnlySubway version of transit. std::map m_segmentToEdge; std::map m_segmentToGate; std::map m_transferPenalties; + + // Fields for working with Public transport version of transit. + std::map m_segmentToEdgePT; + std::map m_segmentToGatePT; + std::map m_segmentToStopPT; + std::map<::transit::TransitId, std::vector<::transit::LineInterval>> m_transferPenaltiesPT; }; void MakeGateEndings(std::vector const & gates, NumMwmId mwmId, - IndexGraph & indexGraph, TransitGraph::GateEndings & gateEndings); + IndexGraph & indexGraph, TransitGraph::Endings & gateEndings); + +void MakeGateEndings(std::vector<::transit::experimental::Gate> const & gates, NumMwmId mwmId, + IndexGraph & indexGraph, TransitGraph::Endings & gateEndings); + +void MakeStopEndings(std::vector<::transit::experimental::Stop> const & stops, NumMwmId mwmId, + IndexGraph & indexGraph, TransitGraph::Endings & gateEndings); } // namespace routing diff --git a/routing/transit_graph_loader.cpp b/routing/transit_graph_loader.cpp index b8013647ac..222230280f 100644 --- a/routing/transit_graph_loader.cpp +++ b/routing/transit_graph_loader.cpp @@ -3,6 +3,7 @@ #include "routing/fake_ending.hpp" #include "routing/routing_exceptions.hpp" +#include "transit/experimental/transit_data.hpp" #include "transit/transit_graph_data.hpp" #include "transit/transit_serdes.hpp" #include "transit/transit_types.hpp" @@ -75,31 +76,65 @@ unique_ptr TransitGraphLoaderImpl::CreateTransitGraph(NumMwmId num MYTHROW(RoutingException, ("Can't get mwm handle for", file)); base::Timer timer; - auto graph = make_unique(numMwmId, m_estimator); + MwmValue const & mwmValue = *handle.GetValue(); + + // By default we return empty transit graph with version OnlySubway. if (!mwmValue.m_cont.IsExist(TRANSIT_FILE_TAG)) - return graph; + return make_unique(::transit::TransitVersion::OnlySubway, numMwmId, m_estimator); try { FilesContainerR::TReader reader(mwmValue.m_cont.GetReader(TRANSIT_FILE_TAG)); - transit::GraphData transitData; - transitData.DeserializeForRouting(*reader.GetPtr()); + auto const transitHeaderVersion = ::transit::GetVersion(*reader.GetPtr()); - TransitGraph::GateEndings gateEndings; - MakeGateEndings(transitData.GetGates(), numMwmId, indexGraph, gateEndings); + if (transitHeaderVersion == ::transit::TransitVersion::OnlySubway) + { + auto graph = + make_unique(::transit::TransitVersion::OnlySubway, numMwmId, m_estimator); - graph->Fill(transitData, gateEndings); + transit::GraphData transitData; + transitData.DeserializeForRouting(*reader.GetPtr()); + + TransitGraph::Endings gateEndings; + MakeGateEndings(transitData.GetGates(), numMwmId, indexGraph, gateEndings); + + graph->Fill(transitData, gateEndings); + + LOG(LINFO, (TRANSIT_FILE_TAG, "section, version", transitHeaderVersion, "for", file.GetName(), + "loaded in", timer.ElapsedSeconds(), "seconds")); + + return graph; + } + else if (transitHeaderVersion == ::transit::TransitVersion::AllPublicTransport) + { + auto graph = make_unique(::transit::TransitVersion::AllPublicTransport, + numMwmId, m_estimator); + + ::transit::experimental::TransitData transitData; + transitData.DeserializeForRouting(*reader.GetPtr()); + + TransitGraph::Endings gateEndings; + MakeGateEndings(transitData.GetGates(), numMwmId, indexGraph, gateEndings); + + TransitGraph::Endings stopEndings; + MakeStopEndings(transitData.GetStops(), numMwmId, indexGraph, stopEndings); + + graph->Fill(transitData, stopEndings, gateEndings); + + LOG(LINFO, (TRANSIT_FILE_TAG, "section, version", transitHeaderVersion, "for", file.GetName(), + "loaded in", timer.ElapsedSeconds(), "seconds")); + + return graph; + } + + UNREACHABLE(); } catch (Reader::OpenException const & e) { LOG(LERROR, ("Error while reading", TRANSIT_FILE_TAG, "section.", e.Msg())); throw; } - - LOG(LINFO, (TRANSIT_FILE_TAG, "section for", file.GetName(), "loaded in", - timer.ElapsedSeconds(), "seconds")); - return graph; } // static diff --git a/routing/transit_info.hpp b/routing/transit_info.hpp index 03a0389e02..13d1b470a3 100644 --- a/routing/transit_info.hpp +++ b/routing/transit_info.hpp @@ -1,6 +1,9 @@ #pragma once +#include "transit/experimental/transit_types_experimental.hpp" +#include "transit/transit_entities.hpp" #include "transit/transit_types.hpp" +#include "transit/transit_version.hpp" #include "base/assert.hpp" @@ -20,10 +23,10 @@ public: Transfer }; - struct Edge + struct EdgeSubway { - Edge() = default; - explicit Edge(transit::Edge const & edge) + EdgeSubway() = default; + explicit EdgeSubway(transit::Edge const & edge) : m_lineId(edge.GetLineId()) , m_stop1Id(edge.GetStop1Id()) , m_stop2Id(edge.GetStop2Id()) @@ -38,18 +41,46 @@ public: std::vector m_shapeIds; }; - struct Gate + struct EdgePT { - Gate() = default; - explicit Gate(transit::Gate const & gate) : m_featureId(gate.GetFeatureId()) {} + EdgePT() = default; + explicit EdgePT(::transit::experimental::Edge const & edge) + : m_lineId(edge.GetLineId()) + , m_stop1Id(edge.GetStop1Id()) + , m_stop2Id(edge.GetStop2Id()) + , m_shapeLink(edge.GetShapeLink()) + { + ASSERT(!edge.IsTransfer(), ()); + } + + ::transit::TransitId m_lineId = ::transit::kInvalidTransitId; + ::transit::TransitId m_stop1Id = ::transit::kInvalidTransitId; + ::transit::TransitId m_stop2Id = ::transit::kInvalidTransitId; + ::transit::ShapeLink m_shapeLink; + }; + + struct GateSubway + { + GateSubway() = default; + explicit GateSubway(transit::Gate const & gate) : m_featureId(gate.GetFeatureId()) {} transit::FeatureId m_featureId = transit::kInvalidFeatureId; }; - struct Transfer + struct GatePT { - Transfer() = default; - explicit Transfer(transit::Edge const & edge) + GatePT() = default; + explicit GatePT(::transit::experimental::Gate const & gate) : m_featureId(gate.GetFeatureId()) + { + } + + ::transit::experimental::FeatureId m_featureId = ::transit::experimental::kInvalidFeatureId; + }; + + struct TransferSubway + { + TransferSubway() = default; + explicit TransferSubway(transit::Edge const & edge) : m_stop1Id(edge.GetStop1Id()), m_stop2Id(edge.GetStop2Id()) { ASSERT(edge.GetTransfer(), ()); @@ -59,47 +90,110 @@ public: transit::StopId m_stop2Id = transit::kInvalidStopId; }; + struct TransferPT + { + TransferPT() = default; + explicit TransferPT(::transit::experimental::Edge const & edge) + : m_stop1Id(edge.GetStop1Id()), m_stop2Id(edge.GetStop2Id()) + { + ASSERT(edge.IsTransfer(), ()); + } + + ::transit::TransitId m_stop1Id = ::transit::kInvalidTransitId; + ::transit::TransitId m_stop2Id = ::transit::kInvalidTransitId; + }; + explicit TransitInfo(transit::Gate const & gate) - : m_type(Type::Gate), m_edge(), m_gate(gate), m_transfer() + : m_transitVersion(::transit::TransitVersion::OnlySubway) + , m_type(Type::Gate) + , m_edgeSubway() + , m_gateSubway(gate) + , m_transferSubway() + { + } + + explicit TransitInfo(::transit::experimental::Gate const & gate) + : m_transitVersion(::transit::TransitVersion::AllPublicTransport) + , m_type(Type::Gate) + , m_gatePT(gate) { } explicit TransitInfo(transit::Edge const & edge) - : m_type(edge.GetTransfer() ? Type::Transfer : Type::Edge) - , m_edge(edge.GetTransfer() ? Edge() : Edge(edge)) - , m_gate() - , m_transfer(edge.GetTransfer() ? Transfer(edge) : Transfer()) + : m_transitVersion(::transit::TransitVersion::OnlySubway) + , m_type(edge.GetTransfer() ? Type::Transfer : Type::Edge) + , m_edgeSubway(edge.GetTransfer() ? EdgeSubway() : EdgeSubway(edge)) + , m_gateSubway() + , m_transferSubway(edge.GetTransfer() ? TransferSubway(edge) : TransferSubway()) + { + } + + explicit TransitInfo(::transit::experimental::Edge const & edge) + : m_transitVersion(::transit::TransitVersion::AllPublicTransport) + , m_type(edge.IsTransfer() ? Type::Transfer : Type::Edge) + , m_edgePT(edge.IsTransfer() ? EdgePT() : EdgePT(edge)) + , m_gatePT() + , m_transferPT(edge.IsTransfer() ? TransferPT(edge) : TransferPT()) { } Type GetType() const { return m_type; } - Edge const & GetEdge() const + ::transit::TransitVersion GetVersion() const { return m_transitVersion; } + + EdgeSubway const & GetEdgeSubway() const { ASSERT_EQUAL(m_type, Type::Edge, ()); - return m_edge; + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + return m_edgeSubway; } - Gate const & GetGate() const + GateSubway const & GetGateSubway() const { ASSERT_EQUAL(m_type, Type::Gate, ()); - return m_gate; + return m_gateSubway; } - Transfer const & GetTransfer() const + TransferSubway const & GetTransferSubway() const { ASSERT_EQUAL(m_type, Type::Transfer, ()); - return m_transfer; + return m_transferSubway; + } + + EdgePT const & GetEdgePT() const + { + ASSERT_EQUAL(m_type, Type::Edge, ()); + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + return m_edgePT; + } + + GatePT const & GetGatePT() const + { + CHECK_EQUAL(m_type, Type::Gate, ()); + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + return m_gatePT; + } + + TransferPT const & GetTransferPT() const + { + ASSERT_EQUAL(m_type, Type::Transfer, ()); + CHECK_EQUAL(m_transitVersion, ::transit::TransitVersion::AllPublicTransport, ()); + return m_transferPT; } private: + ::transit::TransitVersion const m_transitVersion; Type const m_type; // Valid for m_type == Type::Edge only. - Edge const m_edge; + EdgeSubway const m_edgeSubway; // Valid for m_type == Type::Gate only. - Gate const m_gate; + GateSubway const m_gateSubway; // Valid for m_type == Type::Transfer only. - Transfer const m_transfer; + TransferSubway const m_transferSubway; + + EdgePT const m_edgePT; + GatePT const m_gatePT; + TransferPT const m_transferPT; }; class TransitInfoWrapper final diff --git a/routing/transit_world_graph.cpp b/routing/transit_world_graph.cpp index 2781cffead..ce1196fe93 100644 --- a/routing/transit_world_graph.cpp +++ b/routing/transit_world_graph.cpp @@ -177,11 +177,26 @@ unique_ptr TransitWorldGraph::GetTransitInfo(Segment const & segmen return {}; auto & transitGraph = GetTransitGraph(segment.GetMwmId()); - if (transitGraph.IsGate(segment)) - return make_unique(transitGraph.GetGate(segment)); + if (transitGraph.GetTransitVersion() == ::transit::TransitVersion::OnlySubway) + { + if (transitGraph.IsGate(segment)) + return make_unique(transitGraph.GetGate(segment)); - if (transitGraph.IsEdge(segment)) - return make_unique(transitGraph.GetEdge(segment)); + if (transitGraph.IsEdge(segment)) + return make_unique(transitGraph.GetEdge(segment)); + } + else if (transitGraph.GetTransitVersion() == ::transit::TransitVersion::AllPublicTransport) + { + if (transitGraph.IsGate(segment)) + return make_unique(transitGraph.GetGatePT(segment)); + + if (transitGraph.IsEdge(segment)) + return make_unique(transitGraph.GetEdgePT(segment)); + } + else + { + UNREACHABLE(); + } // Fake segment between pedestrian feature and gate. return {}; diff --git a/routing/transit_world_graph.hpp b/routing/transit_world_graph.hpp index ff2d4f2a45..fd7d1dd982 100644 --- a/routing/transit_world_graph.hpp +++ b/routing/transit_world_graph.hpp @@ -40,6 +40,7 @@ public: void GetEdgeList(astar::VertexData const & vertexData, bool isOutgoing, bool useRoutingOptions, bool useAccessConditional, std::vector & edges) override; + // Dummy method which shouldn't be called. void GetEdgeList(astar::VertexData const & parentVertexData, Segment const & segment, bool isOutgoing, bool useAccessConditional, std::vector & edges, diff --git a/transit/transit_display_info.hpp b/transit/transit_display_info.hpp index f5eba0ab37..4cb8547a6f 100644 --- a/transit/transit_display_info.hpp +++ b/transit/transit_display_info.hpp @@ -1,6 +1,8 @@ #pragma once +#include "transit/experimental/transit_types_experimental.hpp" #include "transit/transit_types.hpp" +#include "transit/transit_version.hpp" #include "indexer/feature_decl.hpp" @@ -24,14 +26,31 @@ using TransitShapesInfo = std::map; using TransitNetworksInfo = std::map; +using TransitStopsInfoPT = std::map<::transit::TransitId, ::transit::experimental::Stop>; +using TransitTransfersInfoPT = std::map<::transit::TransitId, ::transit::experimental::Transfer>; +using TransitShapesInfoPT = std::map<::transit::TransitId, ::transit::experimental::Shape>; +using TransitLinesInfoPT = std::map<::transit::TransitId, ::transit::experimental::Line>; +using TransitRoutesInfoPT = std::map<::transit::TransitId, ::transit::experimental::Route>; +using TransitNetworksInfoPT = std::map<::transit::TransitId, ::transit::experimental::Network>; + struct TransitDisplayInfo { - TransitNetworksInfo m_networks; - TransitLinesInfo m_lines; - TransitStopsInfo m_stops; - TransitTransfersInfo m_transfers; - TransitShapesInfo m_shapes; + ::transit::TransitVersion m_transitVersion; + TransitFeaturesInfo m_features; + + TransitNetworksInfo m_networksSubway; + TransitLinesInfo m_linesSubway; + TransitStopsInfo m_stopsSubway; + TransitTransfersInfo m_transfersSubway; + TransitShapesInfo m_shapesSubway; + + TransitNetworksInfoPT m_networksPT; + TransitLinesInfoPT m_linesPT; + TransitRoutesInfoPT m_routesPT; + TransitStopsInfoPT m_stopsPT; + TransitTransfersInfoPT m_transfersPT; + TransitShapesInfoPT m_shapesPT; }; using TransitDisplayInfos = std::map>;