From 058f1b476dabdab1b79fa16527b09aef96ec4e56 Mon Sep 17 00:00:00 2001 From: Olga Khlopkova Date: Fri, 21 Aug 2020 11:38:15 +0300 Subject: [PATCH] [drape_frontend] New transit version in transit scheme renderer. --- drape_frontend/transit_scheme_builder.cpp | 1194 ++++++++++++++++----- drape_frontend/transit_scheme_builder.hpp | 119 +- transit/transit_entities.hpp | 4 + 3 files changed, 1027 insertions(+), 290 deletions(-) diff --git a/drape_frontend/transit_scheme_builder.cpp b/drape_frontend/transit_scheme_builder.cpp index 8c55e93bdc..93b6c0c3b1 100644 --- a/drape_frontend/transit_scheme_builder.cpp +++ b/drape_frontend/transit_scheme_builder.cpp @@ -19,8 +19,11 @@ #include "drape/render_bucket.hpp" #include "drape/utils/vertex_decl.hpp" +#include "base/assert.hpp" #include "base/string_utils.hpp" +#include + using namespace std; namespace df @@ -56,7 +59,7 @@ std::string const kTransitTransferOuterColor = "TransitTransferOuterMarker"; std::string const kTransitTransferInnerColor = "TransitTransferInnerMarker"; std::string const kTransitStopInnerColor = "TransitStopInnerMarker"; -float const kTransitMarkTextSize = 11.0f; +float constexpr kTransitMarkTextSize = 11.0f; struct TransitStaticVertex { @@ -82,6 +85,7 @@ struct SchemeSegment glsl::vec2 m_leftNormal; glsl::vec2 m_rightNormal; }; + using TGeometryBuffer = std::vector; dp::BindingInfo const & GetTransitStaticBindingInfo() @@ -124,7 +128,8 @@ void GenerateLineCaps(ref_ptr context, } dp::AttributeProvider provider(1 /* stream count */, static_cast(geometry.size())); - provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), make_ref(geometry.data())); + provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), + make_ref(geometry.data())); auto state = CreateRenderState(gpu::Program::TransitCircle, DepthLayer::TransitSchemeLayer); batcher.InsertTriangleList(context, state, make_ref(&provider)); } @@ -132,9 +137,7 @@ void GenerateLineCaps(ref_ptr context, struct TitleInfo { TitleInfo() = default; - explicit TitleInfo(std::string const & text) - : m_text(text) - {} + explicit TitleInfo(std::string const & text) : m_text(text) {} std::string m_text; size_t m_rowsCount = 0; @@ -143,14 +146,15 @@ struct TitleInfo dp::Anchor m_anchor = dp::Left; }; -std::vector PlaceTitles(StopNodeParams const & stopParams, float textSize, - ref_ptr textures) +std::vector GetTitles(StopNodeParamsSubway const & stopParams) { std::vector titles; + for (auto const & stopInfo : stopParams.m_stopsInfo) { if (stopInfo.second.m_name.empty()) continue; + bool isUnique = true; for (auto const & title : titles) { @@ -164,78 +168,110 @@ std::vector PlaceTitles(StopNodeParams const & stopParams, float text titles.emplace_back(stopInfo.second.m_name); } - if (titles.size() > 1) - { - auto const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); - - size_t summaryRowsCount = 0; - for (auto & name : titles) - { - df::StraightTextLayout layout(strings::MakeUniString(name.m_text), textSize, - false /* isSdf */, textures, dp::Left, false /* forceNoWrap */); - name.m_pixelSize = layout.GetPixelSize() + m2::PointF(4.0f * vs, 4.0f * vs); - name.m_rowsCount = layout.GetRowsCount(); - summaryRowsCount += layout.GetRowsCount(); - } - - auto const rightRowsCount = summaryRowsCount > 3 ? (summaryRowsCount + 1) / 2 : summaryRowsCount; - float rightHeight = 0.0f; - float leftHeight = 0.0f; - size_t rowsCount = 0; - size_t rightTitlesCount = 0; - for (size_t i = 0; i < titles.size(); ++i) - { - if (rowsCount < rightRowsCount) - { - rightHeight += titles[i].m_pixelSize.y; - ++rightTitlesCount; - } - else - { - leftHeight += titles[i].m_pixelSize.y; - } - rowsCount += titles[i].m_rowsCount; - } - - float currentOffset = -rightHeight / 2.0f; - for (size_t i = 0; i < rightTitlesCount; ++i) - { - titles[i].m_anchor = dp::Left; - titles[i].m_offset.y = currentOffset + titles[i].m_pixelSize.y / 2.0f; - currentOffset += titles[i].m_pixelSize.y; - } - - currentOffset = -leftHeight / 2.0f; - for (size_t i = rightTitlesCount; i < titles.size(); ++i) - { - titles[i].m_anchor = dp::Right; - titles[i].m_offset.y = currentOffset + titles[i].m_pixelSize.y / 2.0f; - currentOffset += titles[i].m_pixelSize.y; - } - } return titles; } -vector GetTransitMarkerSizes(float markerScale, float maxRouteWidth) +std::vector GetTitles(StopNodeParamsPT const & stopParams) +{ + std::vector titles; + + for (auto const & stopInfo : stopParams.m_stopsInfo) + { + if (stopInfo.second.m_name.empty()) + continue; + + bool isUnique = true; + + for (auto const & title : titles) + { + if (title.m_text == stopInfo.second.m_name) + { + isUnique = false; + break; + } + } + + if (isUnique) + titles.emplace_back(stopInfo.second.m_name); + } + + return titles; +} + +void PlaceTitles(std::vector & titles, float textSize, + ref_ptr textures) +{ + if (titles.size() < 2) + return; + + auto const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); + + size_t summaryRowsCount = 0; + for (auto & name : titles) + { + df::StraightTextLayout layout(strings::MakeUniString(name.m_text), textSize, false /* isSdf */, + textures, dp::Left, false /* forceNoWrap */); + name.m_pixelSize = layout.GetPixelSize() + m2::PointF(4.0f * vs, 4.0f * vs); + name.m_rowsCount = layout.GetRowsCount(); + summaryRowsCount += layout.GetRowsCount(); + } + + auto const rightRowsCount = summaryRowsCount > 3 ? (summaryRowsCount + 1) / 2 : summaryRowsCount; + float rightHeight = 0.0f; + float leftHeight = 0.0f; + size_t rowsCount = 0; + size_t rightTitlesCount = 0; + for (size_t i = 0; i < titles.size(); ++i) + { + if (rowsCount < rightRowsCount) + { + rightHeight += titles[i].m_pixelSize.y; + ++rightTitlesCount; + } + else + { + leftHeight += titles[i].m_pixelSize.y; + } + rowsCount += titles[i].m_rowsCount; + } + + float currentOffset = -rightHeight / 2.0f; + for (size_t i = 0; i < rightTitlesCount; ++i) + { + titles[i].m_anchor = dp::Left; + titles[i].m_offset.y = currentOffset + titles[i].m_pixelSize.y / 2.0f; + currentOffset += titles[i].m_pixelSize.y; + } + + currentOffset = -leftHeight / 2.0f; + for (size_t i = rightTitlesCount; i < titles.size(); ++i) + { + titles[i].m_anchor = dp::Right; + titles[i].m_offset.y = currentOffset + titles[i].m_pixelSize.y / 2.0f; + currentOffset += titles[i].m_pixelSize.y; + } +} + +std::vector GetTransitMarkerSizes(float markerScale, float maxRouteWidth) { auto const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); - vector markerSizes; + std::vector markerSizes; markerSizes.reserve(df::kTransitLinesWidthInPixel.size()); + for (auto const halfWidth : df::kTransitLinesWidthInPixel) { float const d = 2.0f * std::min(halfWidth * vs, maxRouteWidth * 0.5f) * markerScale; - markerSizes.push_back(m2::PointF(d, d)); + markerSizes.emplace_back(d, d); } + return markerSizes; } -uint32_t GetRouteId(routing::transit::LineId lineId) -{ - return static_cast(lineId >> 4); -} +uint32_t GetRouteId(routing::transit::LineId lineId) { return static_cast(lineId >> 4); } -void FillStopParams(TransitDisplayInfo const & transitDisplayInfo, MwmSet::MwmId const & mwmId, - routing::transit::Stop const & stop, StopNodeParams & stopParams) +void FillStopParamsSubway(TransitDisplayInfo const & transitDisplayInfo, + MwmSet::MwmId const & mwmId, routing::transit::Stop const & stop, + StopNodeParamsSubway & stopParams) { FeatureID featureId; std::string title; @@ -255,6 +291,38 @@ void FillStopParams(TransitDisplayInfo const & transitDisplayInfo, MwmSet::MwmId } } +void FillStopParamsPT(TransitDisplayInfo const & transitDisplayInfo, MwmSet::MwmId const & mwmId, + ::transit::experimental::Stop const & stop, ::transit::IdSet const & lineIds, + StopNodeParamsPT & stopParams) +{ + FeatureID featureId; + std::string title; + + if (stop.GetFeatureId() != routing::transit::kInvalidFeatureId) + { + featureId = FeatureID(mwmId, stop.GetFeatureId()); + auto const itFeature = transitDisplayInfo.m_features.find(featureId); + CHECK(itFeature != transitDisplayInfo.m_features.end(), (featureId)); + + title = itFeature->second.m_title; + } + + stopParams.m_isTransfer = false; + stopParams.m_pivot = stop.GetPoint(); + + for (auto lineId : lineIds) + { + auto const itLine = transitDisplayInfo.m_linesPT.find(lineId); + CHECK(itLine != transitDisplayInfo.m_linesPT.end(), (lineId)); + + ::transit::TransitId const routeId = itLine->second.GetRouteId(); + StopInfo & info = stopParams.m_stopsInfo[routeId]; + info.m_featureId = featureId; + info.m_name = title; + info.m_lines.insert(lineId); + } +} + bool FindLongerPath(routing::transit::StopId stop1Id, routing::transit::StopId stop2Id, std::vector const & sameStops, size_t & stop1Ind, size_t & stop2Ind) @@ -286,22 +354,40 @@ void TransitSchemeBuilder::UpdateSchemes(ref_ptr context, TransitDisplayInfos const & transitDisplayInfos, ref_ptr textures) { - for (auto const & mwmInfo : transitDisplayInfos) + for (auto const & [mwmId, transitDisplayInfoPtr] : transitDisplayInfos) { - if (!mwmInfo.second) + if (!transitDisplayInfoPtr) continue; - auto const & mwmId = mwmInfo.first; - auto const & transitDisplayInfo = *mwmInfo.second.get(); + auto const & transitDisplayInfo = *transitDisplayInfoPtr.get(); MwmSchemeData & scheme = m_schemes[mwmId]; + scheme.m_transitVersion = transitDisplayInfo.m_transitVersion; - CollectStops(transitDisplayInfo, mwmId, scheme); - CollectLines(transitDisplayInfo, scheme); - CollectShapes(transitDisplayInfo, scheme); + if (scheme.m_transitVersion == ::transit::TransitVersion::OnlySubway) + { + CollectStopsSubway(transitDisplayInfo, mwmId, scheme); + CollectLinesSubway(transitDisplayInfo, scheme); + CollectShapesSubway(transitDisplayInfo, scheme); - PrepareScheme(scheme); - BuildScheme(context, mwmId, textures); + PrepareSchemeSubway(scheme); + BuildScheme(context, mwmId, textures); + } + else if (scheme.m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + { + LinesDataPT const & linesData = CollectLinesPT(transitDisplayInfo, scheme); + + CollectStopsPT(transitDisplayInfo, linesData, mwmId, scheme); + CollectShapesPT(transitDisplayInfo, scheme); + + PrepareSchemePT(transitDisplayInfo, linesData, scheme); + BuildScheme(context, mwmId, textures); + } + else + { + LOG(LERROR, (scheme.m_transitVersion)); + UNREACHABLE(); + } } } @@ -328,27 +414,90 @@ void TransitSchemeBuilder::BuildScheme(ref_ptr context, { if (m_schemes.find(mwmId) == m_schemes.end()) return; + ++m_recacheId; GenerateShapes(context, mwmId); GenerateStops(context, mwmId, textures); } -void TransitSchemeBuilder::CollectStops(TransitDisplayInfo const & transitDisplayInfo, - MwmSet::MwmId const & mwmId, MwmSchemeData & scheme) +void TransitSchemeBuilder::GenerateLinesSubway(MwmSchemeData const & scheme, dp::Batcher & batcher, + ref_ptr context) { + for (auto const & shape : scheme.m_shapesSubway) + { + size_t const linesCount = + shape.second.m_forwardLines.size() + shape.second.m_backwardLines.size(); + float shapeOffset = + -static_cast(linesCount / 2) * 2.0f - static_cast(linesCount % 2) + 1.0f; + size_t constexpr shapeOffsetIncrement = 2.0f; + + std::vector> coloredLines; + + for (auto lineId : shape.second.m_forwardLines) + { + auto const & lineColor = scheme.m_linesSubway.at(lineId).m_color; + auto const colorName = df::GetTransitColorName(lineColor); + auto const color = GetColorConstant(colorName); + coloredLines.emplace_back(color, lineId); + } + + for (auto it = shape.second.m_backwardLines.rbegin(); it != shape.second.m_backwardLines.rend(); + ++it) + { + auto const & lineColor = scheme.m_linesSubway.at(*it).m_color; + auto const colorName = df::GetTransitColorName(lineColor); + auto const color = GetColorConstant(colorName); + coloredLines.emplace_back(color, *it); + } + + for (auto const & coloredLine : coloredLines) + { + auto const & colorConst = coloredLine.first; + auto const & lineId = coloredLine.second; + auto const depth = scheme.m_linesSubway.at(lineId).m_depth; + + GenerateLine(context, shape.second.m_polyline, scheme.m_pivot, colorConst, shapeOffset, + kTransitLineHalfWidth, depth, batcher); + + shapeOffset += shapeOffsetIncrement; + } + } +} + +void TransitSchemeBuilder::GenerateLinesPT(MwmSchemeData const & scheme, dp::Batcher & batcher, + ref_ptr context) +{ + for (auto const & data : scheme.m_routeSegmentsPT) + { + dp::Color const color = GetColorConstant(df::GetTransitColorName(data.m_color)); + + for (auto const & routeData : data.m_routeShapes) + { + float const offset = static_cast(routeData.m_order); + GenerateLine(context, routeData.m_polyline, scheme.m_pivot, color, offset, + kTransitLineHalfWidth, data.m_depth, batcher); + } + } +} + +void TransitSchemeBuilder::CollectStopsSubway(TransitDisplayInfo const & transitDisplayInfo, + MwmSet::MwmId const & mwmId, MwmSchemeData & scheme) +{ + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + for (auto const & stopInfo : transitDisplayInfo.m_stopsSubway) { routing::transit::Stop const & stop = stopInfo.second; if (stop.GetTransferId() != routing::transit::kInvalidTransferId) continue; - auto & stopNode = scheme.m_stops[stop.GetId()]; - FillStopParams(transitDisplayInfo, mwmId, stop, stopNode); + auto & stopNode = scheme.m_stopsSubway[stop.GetId()]; + FillStopParamsSubway(transitDisplayInfo, mwmId, stop, stopNode); } for (auto const & stopInfo : transitDisplayInfo.m_transfersSubway) { routing::transit::Transfer const & transfer = stopInfo.second; - auto & stopNode = scheme.m_transfers[transfer.GetId()]; + auto & stopNode = scheme.m_transfersSubway[transfer.GetId()]; for (auto stopId : transfer.GetStopIds()) { @@ -358,15 +507,66 @@ void TransitSchemeBuilder::CollectStops(TransitDisplayInfo const & transitDispla continue; } routing::transit::Stop const & stop = transitDisplayInfo.m_stopsSubway.at(stopId); - FillStopParams(transitDisplayInfo, mwmId, stop, stopNode); + FillStopParamsSubway(transitDisplayInfo, mwmId, stop, stopNode); } stopNode.m_isTransfer = true; stopNode.m_pivot = transfer.GetPoint(); } } -void TransitSchemeBuilder::CollectLines(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme) +void TransitSchemeBuilder::CollectStopsPT(TransitDisplayInfo const & transitDisplayInfo, + LinesDataPT const & linesData, + MwmSet::MwmId const & mwmId, MwmSchemeData & scheme) { + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::AllPublicTransport, + ()); + + for (auto const & [stopId, lineIds] : linesData.m_stopToLineIds) + { + auto const itStop = transitDisplayInfo.m_stopsPT.find(stopId); + CHECK(itStop != transitDisplayInfo.m_stopsPT.end(), (stopId)); + + ::transit::experimental::Stop const & stop = itStop->second; + + if (!stop.GetTransferIds().empty()) + continue; + + FillStopParamsPT(transitDisplayInfo, mwmId, stop, lineIds, scheme.m_stopsPT[stopId]); + + scheme.m_stopsPT[stopId].m_isTerminalStop = + (linesData.m_terminalStops.find(stopId) != linesData.m_terminalStops.end()); + } + + for (auto const & transferInfo : transitDisplayInfo.m_transfersPT) + { + ::transit::experimental::Transfer const & transfer = transferInfo.second; + auto & transferNode = scheme.m_transfersPT[transfer.GetId()]; + + for (auto stopId : transfer.GetStopIds()) + { + auto itId = linesData.m_stopToLineIds.find(stopId); + if (itId == linesData.m_stopToLineIds.end()) + continue; + + auto it = transitDisplayInfo.m_stopsPT.find(stopId); + if (it == transitDisplayInfo.m_stopsPT.end()) + continue; + + ::transit::experimental::Stop const & stop = it->second; + FillStopParamsPT(transitDisplayInfo, mwmId, stop, itId->second, transferNode); + transferNode.m_stopsInfo.emplace(stopId, StopInfo()); + } + + transferNode.m_isTransfer = true; + transferNode.m_pivot = transfer.GetPoint(); + } +} + +void TransitSchemeBuilder::CollectLinesSubway(TransitDisplayInfo const & transitDisplayInfo, + MwmSchemeData & scheme) +{ + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + std::multimap linesLengths; for (auto const & line : transitDisplayInfo.m_linesSubway) { @@ -381,15 +581,65 @@ void TransitSchemeBuilder::CollectLines(TransitDisplayInfo const & transitDispla for (auto const & pair : linesLengths) { auto const lineId = pair.second; - scheme.m_lines[lineId] = + + scheme.m_linesSubway[lineId] = LineParams(transitDisplayInfo.m_linesSubway.at(lineId).GetColor(), depth); + depth += kDepthPerLine; } } -void TransitSchemeBuilder::CollectShapes(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme) +LinesDataPT TransitSchemeBuilder::CollectLinesPT(TransitDisplayInfo const & transitDisplayInfo, + MwmSchemeData & scheme) { + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::AllPublicTransport, + ()); + + LinesDataPT linesData; + + std::set<::transit::TransitId> notTerminal; + + for (auto const & [lineId, line] : transitDisplayInfo.m_linesPT) + { + auto const itRoute = transitDisplayInfo.m_routesPT.find(line.GetRouteId()); + CHECK(itRoute != transitDisplayInfo.m_routesPT.end(), (line.GetRouteId())); + + auto const & lineType = itRoute->second.GetType(); + + // We skip types that are not mentioned for displaying on the layer - buses, ferries, etc. + if (transit::kSubwayLayerTypes.find(lineType) == transit::kSubwayLayerTypes.end()) + continue; + + for (auto stopId : line.GetStopIds()) + linesData.m_stopToLineIds[stopId].insert(lineId); + + auto const & firstStop = line.GetStopIds().front(); + auto const & lastStop = line.GetStopIds().back(); + + if (!linesData.m_terminalStops.insert(firstStop).second) + notTerminal.insert(firstStop); + + if (!linesData.m_terminalStops.insert(lastStop).second) + notTerminal.insert(lastStop); + + auto & schemeLine = scheme.m_linesPT[lineId]; + schemeLine = LineParams(itRoute->second.GetColor(), kBaseLineDepth); + schemeLine.m_stopIds = line.GetStopIds(); + } + + for (auto id : notTerminal) + linesData.m_terminalStops.erase(id); + + return linesData; +} + +void TransitSchemeBuilder::CollectShapesSubway(TransitDisplayInfo const & transitDisplayInfo, + MwmSchemeData & scheme) +{ + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::OnlySubway, ()); + std::map> roads; + for (auto const & line : transitDisplayInfo.m_linesSubway) { auto const lineId = line.second.GetId(); @@ -411,10 +661,59 @@ void TransitSchemeBuilder::CollectShapes(TransitDisplayInfo const & transitDispl } } -void TransitSchemeBuilder::FindShapes(routing::transit::StopId stop1Id, routing::transit::StopId stop2Id, +void TransitSchemeBuilder::CollectShapesPT(TransitDisplayInfo const & transitDisplayInfo, + MwmSchemeData & scheme) +{ + CHECK_EQUAL(transitDisplayInfo.m_transitVersion, ::transit::TransitVersion::AllPublicTransport, + ()); + + float curDepth = kBaseLineDepth; + std::unordered_map routeColorToDepth; + + for (auto const & [lineId, metaData] : transitDisplayInfo.m_linesMetadataPT) + { + auto const itLine = transitDisplayInfo.m_linesPT.find(lineId); + CHECK(itLine != transitDisplayInfo.m_linesPT.end(), (lineId)); + + auto const & lineData = itLine->second; + ::transit::TransitId const routeId = lineData.GetRouteId(); + auto const & shapeLink = lineData.GetShapeLink(); + + auto const itRoute = transitDisplayInfo.m_routesPT.find(routeId); + CHECK(itRoute != transitDisplayInfo.m_routesPT.end(), (routeId)); + + RouteData routeData; + routeData.m_color = itRoute->second.GetColor(); + auto [itDepth, insertedDepth] = routeColorToDepth.emplace(routeData.m_color, curDepth); + if (insertedDepth) + curDepth += kDepthPerLine; + + routeData.m_depth = itDepth->second; + + auto const itShape = transitDisplayInfo.m_shapesPT.find(shapeLink.m_shapeId); + CHECK(itShape != transitDisplayInfo.m_shapesPT.end(), (shapeLink.m_shapeId)); + + auto const & shape = itShape->second.GetPolyline(); + + for (auto const & part : metaData.GetLineSegmentsOrder()) + { + RouteSegment rs; + rs.m_polyline = + ::transit::GetPolylinePart(shape, part.m_segment.m_startIdx, part.m_segment.m_endIdx); + rs.m_order = part.m_order; + routeData.m_routeShapes.push_back(rs); + } + + scheme.m_routeSegmentsPT.push_back(routeData); + } +} + +void TransitSchemeBuilder::FindShapes(routing::transit::StopId stop1Id, + routing::transit::StopId stop2Id, routing::transit::LineId lineId, std::vector const & sameLines, - TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme) + TransitDisplayInfo const & transitDisplayInfo, + MwmSchemeData & scheme) { bool shapeAdded = false; @@ -478,15 +777,15 @@ void TransitSchemeBuilder::AddShape(TransitDisplayInfo const & transitDisplayInf if (it == transitDisplayInfo.m_shapesSubway.end()) return; - auto const itScheme = scheme.m_shapes.find(shapeId); - if (itScheme == scheme.m_shapes.end()) + auto const itScheme = scheme.m_shapesSubway.find(shapeId); + if (itScheme == scheme.m_shapesSubway.end()) { auto const & polyline = transitDisplayInfo.m_shapesSubway.at(it->first).GetPolyline(); if (isForward) - scheme.m_shapes[shapeId].m_forwardLines.push_back(lineId); + scheme.m_shapesSubway[shapeId].m_forwardLines.push_back(lineId); else - scheme.m_shapes[shapeId].m_backwardLines.push_back(lineId); - scheme.m_shapes[shapeId].m_polyline = polyline; + scheme.m_shapesSubway[shapeId].m_backwardLines.push_back(lineId); + scheme.m_shapesSubway[shapeId].m_polyline = polyline; } else { @@ -508,17 +807,22 @@ void TransitSchemeBuilder::AddShape(TransitDisplayInfo const & transitDisplayInf } } -void TransitSchemeBuilder::PrepareScheme(MwmSchemeData & scheme) +void TransitSchemeBuilder::PrepareSchemeSubway(MwmSchemeData & scheme) { m2::RectD boundingRect; - for (auto const & shape : scheme.m_shapes) + + for (auto const & shape : scheme.m_shapesSubway) { auto const stop1 = shape.first.GetStop1Id(); auto const stop2 = shape.first.GetStop2Id(); - StopNodeParams & params1 = (scheme.m_stops.find(stop1) == scheme.m_stops.end()) ? scheme.m_transfers[stop1] - : scheme.m_stops[stop1]; - StopNodeParams & params2 = (scheme.m_stops.find(stop2) == scheme.m_stops.end()) ? scheme.m_transfers[stop2] - : scheme.m_stops[stop2]; + StopNodeParamsSubway & params1 = + (scheme.m_stopsSubway.find(stop1) == scheme.m_stopsSubway.end()) + ? scheme.m_transfersSubway[stop1] + : scheme.m_stopsSubway[stop1]; + StopNodeParamsSubway & params2 = + (scheme.m_stopsSubway.find(stop2) == scheme.m_stopsSubway.end()) + ? scheme.m_transfersSubway[stop2] + : scheme.m_stopsSubway[stop2]; auto const linesCount = shape.second.m_forwardLines.size() + shape.second.m_backwardLines.size(); @@ -536,68 +840,204 @@ void TransitSchemeBuilder::PrepareScheme(MwmSchemeData & scheme) for (auto const & pt : shape.second.m_polyline) boundingRect.Add(pt); } + scheme.m_pivot = boundingRect.Center(); } -void TransitSchemeBuilder::GenerateShapes(ref_ptr context, MwmSet::MwmId const & mwmId) +void UpdateShapeInfos(std::vector & shapeInfos, m2::PointD const & newDir, + std::string const & color) +{ + static double constexpr eps = 1e-1; + + auto const newDirReverse = -newDir; + + for (ShapeInfoPT & info : shapeInfos) + { + if (base::AlmostEqualAbs(info.m_direction, newDir, eps) || + base::AlmostEqualAbs(info.m_direction, newDirReverse, eps)) + { + info.m_colors.insert(color); + return; + } + } + + shapeInfos.push_back(ShapeInfoPT(newDir, {color})); +} + +void UpdateShapeInfos(std::vector & shapeInfos, m2::PointD const & newDir, + std::set const & colors) +{ + for (auto const & color : colors) + { + UpdateShapeInfos(shapeInfos, newDir, color); + } +} + +void TransitSchemeBuilder::PrepareSchemePT(TransitDisplayInfo const & transitDisplayInfo, + LinesDataPT const & lineData, MwmSchemeData & scheme) +{ + m2::RectD boundingRect; + + for (auto const & [lineId, lineData] : scheme.m_linesPT) + { + if (transitDisplayInfo.m_linesMetadataPT.find(lineId) == + transitDisplayInfo.m_linesMetadataPT.end()) + continue; + + auto const & color = lineData.m_color; + + CHECK(!lineData.m_color.empty(), ()); + + for (size_t i = 0; i < lineData.m_stopIds.size() - 1; ++i) + { + ::transit::TransitId stop1Id = lineData.m_stopIds[i]; + ::transit::TransitId stop2Id = lineData.m_stopIds[i + 1]; + + StopNodeParamsPT & params1 = (scheme.m_stopsPT.find(stop1Id) == scheme.m_stopsPT.end()) + ? scheme.m_transfersPT[stop1Id] + : scheme.m_stopsPT[stop1Id]; + + StopNodeParamsPT & params2 = (scheme.m_stopsPT.find(stop2Id) == scheme.m_stopsPT.end()) + ? scheme.m_transfersPT[stop2Id] + : scheme.m_stopsPT[stop2Id]; + + m2::PointD dir1; + m2::PointD dir2; + + auto it = transitDisplayInfo.m_edgesPT.find(::transit::EdgeId(stop1Id, stop2Id, lineId)); + + if (it == transitDisplayInfo.m_edgesPT.end()) + { + dir1 = (params2.m_pivot - params1.m_pivot).Normalize(); + dir2 = -dir1; + } + else + { + ::transit::ShapeLink const & shapeLink = it->second.m_shapeLink; + auto const itShape = transitDisplayInfo.m_shapesPT.find(shapeLink.m_shapeId); + CHECK(itShape != transitDisplayInfo.m_shapesPT.end(), (shapeLink.m_shapeId)); + + auto const & polyline = itShape->second.GetPolyline(); + + auto const [startIndex, endIndex] = + std::minmax(shapeLink.m_startIndex, shapeLink.m_endIndex); + + dir1 = (polyline[startIndex + 1] - polyline[startIndex]).Normalize(); + dir2 = (polyline[endIndex] - polyline[endIndex - 1]).Normalize(); + + if (shapeLink.m_startIndex > shapeLink.m_endIndex) + std::swap(dir1, dir2); + + for (size_t j = shapeLink.m_startIndex; j <= shapeLink.m_endIndex; ++j) + boundingRect.Add(polyline[j]); + } + + UpdateShapeInfos(params1.m_shapeInfoOut, dir1, color); + UpdateShapeInfos(params2.m_shapeInfoIn, dir2, color); + } + } + + for (auto & [transferId, transferData] : scheme.m_transfersPT) + { + if (!transferData.m_isTransfer) + continue; + + auto const transferIt = transitDisplayInfo.m_transfersPT.find(transferId); + CHECK(transferIt != transitDisplayInfo.m_transfersPT.end(), (transferId)); + + auto const & transfer = transferIt->second; + + for (::transit::TransitId stopId : transfer.GetStopIds()) + { + auto it = scheme.m_stopsPT.find(stopId); + if (it == scheme.m_stopsPT.end()) + { + it = scheme.m_transfersPT.find(stopId); + if (it == scheme.m_transfersPT.end()) + continue; + } + + for (auto const & info : it->second.m_shapeInfoIn) + UpdateShapeInfos(transferData.m_shapeInfoIn, info.m_direction, info.m_colors); + + for (auto const & info : it->second.m_shapeInfoOut) + UpdateShapeInfos(transferData.m_shapeInfoOut, info.m_direction, info.m_colors); + } + } + + scheme.m_pivot = boundingRect.Center(); +} + +void TransitSchemeBuilder::GenerateShapes(ref_ptr context, + MwmSet::MwmId const & mwmId) { MwmSchemeData const & scheme = m_schemes[mwmId]; - uint32_t const kBatchSize = 65000; + uint32_t constexpr kBatchSize = 65000; dp::Batcher batcher(kBatchSize, kBatchSize); batcher.SetBatcherHash(static_cast(BatcherBucket::Transit)); { - dp::SessionGuard guard(context, batcher, [this, &mwmId, &scheme](dp::RenderState const & state, - drape_ptr && b) + dp::SessionGuard guard( + context, batcher, + [this, &mwmId, &scheme](dp::RenderState const & state, drape_ptr && b) { + TransitRenderData::Type type = TransitRenderData::Type::Lines; + if (state.GetProgram() == gpu::Program::TransitCircle) + type = TransitRenderData::Type::LinesCaps; + TransitRenderData renderData(type, state, m_recacheId, mwmId, scheme.m_pivot, + std::move(b)); + m_flushRenderDataFn(std::move(renderData)); + }); + + if (scheme.m_transitVersion == ::transit::TransitVersion::OnlySubway) + GenerateLinesSubway(scheme, batcher, context); + else if (scheme.m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + GenerateLinesPT(scheme, batcher, context); + else { - TransitRenderData::Type type = TransitRenderData::Type::Lines; - if (state.GetProgram() == gpu::Program::TransitCircle) - type = TransitRenderData::Type::LinesCaps; - TransitRenderData renderData(type, state, m_recacheId, mwmId, scheme.m_pivot, std::move(b)); - m_flushRenderDataFn(std::move(renderData)); - }); - - for (auto const & shape : scheme.m_shapes) - { - auto const linesCount = shape.second.m_forwardLines.size() + shape.second.m_backwardLines.size(); - auto shapeOffset = -static_cast(linesCount / 2) * 2.0f - 1.0f * static_cast(linesCount % 2) + 1.0f; - auto const shapeOffsetIncrement = 2.0f; - - std::vector> coloredLines; - for (auto lineId : shape.second.m_forwardLines) - { - auto const colorName = df::GetTransitColorName(scheme.m_lines.at(lineId).m_color); - auto const color = GetColorConstant(colorName); - coloredLines.push_back(std::make_pair(color, lineId)); - } - for (auto it = shape.second.m_backwardLines.rbegin(); it != shape.second.m_backwardLines.rend(); ++it) - { - auto const colorName = df::GetTransitColorName(scheme.m_lines.at(*it).m_color); - auto const color = GetColorConstant(colorName); - coloredLines.push_back(std::make_pair(color, *it)); - } - - for (auto const & coloredLine : coloredLines) - { - auto const & colorConst = coloredLine.first; - auto const & lineId = coloredLine.second; - auto const depth = scheme.m_lines.at(lineId).m_depth; - - GenerateLine(context, shape.second.m_polyline, scheme.m_pivot, colorConst, shapeOffset, - kTransitLineHalfWidth, depth, batcher); - shapeOffset += shapeOffsetIncrement; - } + LOG(LERROR, (scheme.m_transitVersion)); + UNREACHABLE(); } } } -void TransitSchemeBuilder::GenerateStops(ref_ptr context, MwmSet::MwmId const & mwmId, +template +void TransitSchemeBuilder::GenerateLocationsWithTitles(ref_ptr context, + ref_ptr textures, + dp::Batcher & batcher, F && flusher, + MwmSchemeData const & scheme, + S const & stops, T const & transfers, + L const & lines) +{ + dp::SessionGuard guard(context, batcher, flusher); + + float constexpr kStopScale = 2.5f; + float constexpr kTransferScale = 3.0f; + + std::vector const transferMarkerSizes = GetTransitMarkerSizes(kTransferScale, 1000); + std::vector const stopMarkerSizes = GetTransitMarkerSizes(kStopScale, 1000); + + for (auto const & stop : stops) + { + GenerateStop(context, stop.second, scheme.m_pivot, lines, batcher); + GenerateTitles(context, stop.second, scheme.m_pivot, stopMarkerSizes, textures, batcher); + } + + for (auto const & transfer : transfers) + { + GenerateTransfer(context, transfer.second, scheme.m_pivot, batcher); + GenerateTitles(context, transfer.second, scheme.m_pivot, transferMarkerSizes, textures, + batcher); + } +} + +void TransitSchemeBuilder::GenerateStops(ref_ptr context, + MwmSet::MwmId const & mwmId, ref_ptr textures) { MwmSchemeData const & scheme = m_schemes[mwmId]; - auto const flusher = [this, &mwmId, &scheme](dp::RenderState const & state, drape_ptr && b) - { + auto const flusher = [this, &mwmId, &scheme](dp::RenderState const & state, + drape_ptr && b) { TransitRenderData::Type type = TransitRenderData::Type::Stubs; if (state.GetProgram() == gpu::Program::TransitMarker) type = TransitRenderData::Type::Markers; @@ -608,86 +1048,227 @@ void TransitSchemeBuilder::GenerateStops(ref_ptr context, M m_flushRenderDataFn(std::move(renderData)); }; - uint32_t const kBatchSize = 5000; + uint32_t constexpr kBatchSize = 5000; dp::Batcher batcher(kBatchSize, kBatchSize); + batcher.SetBatcherHash(static_cast(BatcherBucket::Transit)); + if (scheme.m_transitVersion == ::transit::TransitVersion::OnlySubway) { - dp::SessionGuard guard(context, batcher, flusher); - - float const kStopScale = 2.5f; - float const kTransferScale = 3.0f; - std::vector const transferMarkerSizes = GetTransitMarkerSizes(kTransferScale, 1000); - std::vector const stopMarkerSizes = GetTransitMarkerSizes(kStopScale, 1000); - - for (auto const & stop : scheme.m_stops) - { - GenerateStop(context, stop.second, scheme.m_pivot, scheme.m_lines, batcher); - GenerateTitles(context, stop.second, scheme.m_pivot, stopMarkerSizes, textures, batcher); - } - for (auto const & transfer : scheme.m_transfers) - { - GenerateTransfer(context, transfer.second, scheme.m_pivot, batcher); - GenerateTitles(context, transfer.second, scheme.m_pivot, transferMarkerSizes, textures, batcher); - } + GenerateLocationsWithTitles(context, textures, batcher, flusher, scheme, scheme.m_stopsSubway, + scheme.m_transfersSubway, scheme.m_linesSubway); + } + else if (scheme.m_transitVersion == ::transit::TransitVersion::AllPublicTransport) + { + GenerateLocationsWithTitles(context, textures, batcher, flusher, scheme, scheme.m_stopsPT, + scheme.m_transfersPT, scheme.m_linesPT); + } + else + { + LOG(LERROR, (scheme.m_transitVersion)); + UNREACHABLE(); } } -void TransitSchemeBuilder::GenerateTransfer(ref_ptr context, - StopNodeParams const & params, m2::PointD const & pivot, - dp::Batcher & batcher) +void TransitSchemeBuilder::GenerateMarker(ref_ptr context, + m2::PointD const & pt, m2::PointD widthDir, + float linesCountWidth, float linesCountHeight, + float scaleWidth, float scaleHeight, float depth, + dp::Color const & color, dp::Batcher & batcher) { - m2::PointD const pt = MapShape::ConvertToLocal(params.m_pivot, pivot, kShapeCoordScalar); + using TV = TransitStaticVertex; - size_t maxLinesCount = 0; - m2::PointD dir; - for (auto const & shapeInfo : params.m_shapesInfo) - { - if (shapeInfo.second.m_linesCount > maxLinesCount) - { - dir = shapeInfo.second.m_direction; - maxLinesCount = shapeInfo.second.m_linesCount; - } - } + scaleWidth = (scaleWidth - 1.0f) / linesCountWidth + 1.0f; + scaleHeight = (scaleHeight - 1.0f) / linesCountHeight + 1.0f; - float const kInnerScale = 1.0f; - float const kOuterScale = 1.5f; - auto const outerColor = GetColorConstant(kTransitTransferOuterColor); - auto const innerColor = GetColorConstant(kTransitTransferInnerColor); + widthDir.y = -widthDir.y; + m2::PointD heightDir = widthDir.Ort(); - float const widthLinesCount = maxLinesCount > 3 ? 1.6f : 1.0f; - float const innerScale = maxLinesCount == 1 ? 1.4f : kInnerScale; - float const outerScale = maxLinesCount == 1 ? 1.9f : kOuterScale; + auto const v1 = + -widthDir * scaleWidth * linesCountWidth - heightDir * scaleHeight * linesCountHeight; + auto const v2 = + widthDir * scaleWidth * linesCountWidth - heightDir * scaleHeight * linesCountHeight; + auto const v3 = + widthDir * scaleWidth * linesCountWidth + heightDir * scaleHeight * linesCountHeight; + auto const v4 = + -widthDir * scaleWidth * linesCountWidth + heightDir * scaleHeight * linesCountHeight; - GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, outerScale, outerScale, - kOuterMarkerDepth, outerColor, batcher); + glsl::vec3 const pos(pt.x, pt.y, depth); + auto const colorVal = + glsl::vec4(color.GetRedF(), color.GetGreenF(), color.GetBlueF(), 1.0f /* alpha */); - GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, innerScale, innerScale, - kInnerMarkerDepth, innerColor, batcher); + TGeometryBuffer geometry; + geometry.reserve(6); + geometry.emplace_back(pos, TV::TNormal(v1.x, v1.y, -linesCountWidth, -linesCountHeight), + colorVal); + geometry.emplace_back(pos, TV::TNormal(v2.x, v2.y, linesCountWidth, -linesCountHeight), colorVal); + geometry.emplace_back(pos, TV::TNormal(v3.x, v3.y, linesCountWidth, linesCountHeight), colorVal); + geometry.emplace_back(pos, TV::TNormal(v1.x, v1.y, -linesCountWidth, -linesCountHeight), + colorVal); + geometry.emplace_back(pos, TV::TNormal(v3.x, v3.y, linesCountWidth, linesCountHeight), colorVal); + geometry.emplace_back(pos, TV::TNormal(v4.x, v4.y, -linesCountWidth, linesCountHeight), colorVal); + + dp::AttributeProvider provider(1 /* stream count */, static_cast(geometry.size())); + provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), + make_ref(geometry.data())); + auto state = CreateRenderState(gpu::Program::TransitMarker, DepthLayer::TransitSchemeLayer); + batcher.InsertTriangleList(context, state, make_ref(&provider)); } -void TransitSchemeBuilder::GenerateStop(ref_ptr context, StopNodeParams const & params, - m2::PointD const & pivot, std::map const & lines, dp::Batcher & batcher) +void TransitSchemeBuilder::GenerateLine(ref_ptr context, + std::vector const & path, + m2::PointD const & pivot, dp::Color const & colorConst, + float lineOffset, float halfWidth, float depth, + dp::Batcher & batcher) { - bool const severalRoads = params.m_stopsInfo.size() > 1; + using TV = TransitStaticVertex; + + TGeometryBuffer geometry; + auto const color = glsl::vec4(colorConst.GetRedF(), colorConst.GetGreenF(), colorConst.GetBlueF(), + 1.0f /* alpha */); + size_t const kAverageSize = path.size() * 6; + size_t const kAverageCapSize = 12; + geometry.reserve(kAverageSize + kAverageCapSize * 2); + + std::vector segments; + segments.reserve(path.size() - 1); + + for (size_t i = 1; i < path.size(); ++i) + { + if (path[i].EqualDxDy(path[i - 1], 1.0e-5)) + continue; + + SchemeSegment segment; + segment.m_p1 = glsl::ToVec2(MapShape::ConvertToLocal(path[i - 1], pivot, kShapeCoordScalar)); + segment.m_p2 = glsl::ToVec2(MapShape::ConvertToLocal(path[i], pivot, kShapeCoordScalar)); + CalculateTangentAndNormals(segment.m_p1, segment.m_p2, segment.m_tangent, segment.m_leftNormal, + segment.m_rightNormal); + + auto const startPivot = glsl::vec3(segment.m_p1, depth); + auto const endPivot = glsl::vec3(segment.m_p2, depth); + auto const offset = lineOffset * segment.m_rightNormal; + + geometry.emplace_back(startPivot, + TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), + color); + geometry.emplace_back( + startPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); + geometry.emplace_back( + endPivot, TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), color); + geometry.emplace_back( + endPivot, TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), color); + geometry.emplace_back( + startPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); + geometry.emplace_back( + endPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); + + segments.emplace_back(std::move(segment)); + } + + dp::AttributeProvider provider(1 /* stream count */, static_cast(geometry.size())); + provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), + make_ref(geometry.data())); + auto state = CreateRenderState(gpu::Program::Transit, DepthLayer::TransitSchemeLayer); + batcher.InsertTriangleList(context, state, make_ref(&provider)); + + GenerateLineCaps(context, segments, color, lineOffset, halfWidth, depth, batcher); +} + +void TransitSchemeBuilder::GenerateStop( + ref_ptr context, StopNodeParamsSubway const & stopParams, + m2::PointD const & pivot, std::map const & lines, + dp::Batcher & batcher) +{ + bool const severalRoads = stopParams.m_stopsInfo.size() > 1; + if (severalRoads) { - GenerateTransfer(context, params, pivot, batcher); + GenerateTransfer(context, stopParams, pivot, batcher); return; } float const kInnerScale = 0.8f; float const kOuterScale = 2.0f; - auto const lineId = *params.m_stopsInfo.begin()->second.m_lines.begin(); + auto const lineId = *stopParams.m_stopsInfo.begin()->second.m_lines.begin(); auto const colorName = df::GetTransitColorName(lines.at(lineId).m_color); auto const outerColor = GetColorConstant(colorName); auto const innerColor = GetColorConstant(kTransitStopInnerColor); - m2::PointD const pt = MapShape::ConvertToLocal(params.m_pivot, pivot, kShapeCoordScalar); + m2::PointD const pt = MapShape::ConvertToLocal(stopParams.m_pivot, pivot, kShapeCoordScalar); - m2::PointD dir = params.m_shapesInfo.begin()->second.m_direction; + m2::PointD dir = stopParams.m_shapesInfo.begin()->second.m_direction; + + GenerateMarker(context, pt, dir, 1.0f, 1.0f, kOuterScale, kOuterScale, kOuterMarkerDepth, + outerColor, batcher); + + GenerateMarker(context, pt, dir, 1.0f, 1.0f, kInnerScale, kInnerScale, kInnerMarkerDepth, + innerColor, batcher); +} + +std::pair GetFittingDirectionAndSize( + std::vector const & shapeInfos) +{ + if (shapeInfos.empty()) + return {m2::PointD::Zero(), 0}; + + size_t idxMax = 0; + size_t max = shapeInfos[0].m_colors.size(); + + for (size_t i = 1; i < shapeInfos.size(); ++i) + { + if (shapeInfos[i].m_colors.size() > max) + { + max = shapeInfos[i].m_colors.size(); + idxMax = i; + } + } + + return {shapeInfos[idxMax].m_direction, max}; +} + +std::pair GetFittingDirectionAndSize( + std::vector const & shapeInfosIn, std::vector const & shapeInfosOut) +{ + auto const dirSizeIn = GetFittingDirectionAndSize(shapeInfosIn); + auto const dirSizeOut = GetFittingDirectionAndSize(shapeInfosOut); + + if (dirSizeIn.second > dirSizeOut.second) + return dirSizeIn; + + return dirSizeOut; +} + +void TransitSchemeBuilder::GenerateStop( + ref_ptr context, StopNodeParamsPT const & stopParams, + m2::PointD const & pivot, std::map const & lines, + dp::Batcher & batcher) +{ + auto const & [dir, linesCount] = + GetFittingDirectionAndSize(stopParams.m_shapeInfoIn, stopParams.m_shapeInfoOut); + bool const severalRoads = + std::max(stopParams.m_shapeInfoIn.size(), stopParams.m_shapeInfoOut.size()) > 1; + + if (linesCount > 1 || severalRoads) + { + GenerateTransfer(context, stopParams, pivot, batcher); + return; + } + + float const kInnerScale = 0.8f; + float const kOuterScale = 2.0f; + + ::transit::TransitId lineId = *stopParams.m_stopsInfo.begin()->second.m_lines.begin(); + + auto const itColor = lines.find(lineId); + CHECK(itColor != lines.end(), (lineId)); + + std::string const colorName = df::GetTransitColorName(itColor->second.m_color); + auto const outerColor = GetColorConstant(colorName); + + auto const innerColor = GetColorConstant(kTransitStopInnerColor); + + m2::PointD const pt = MapShape::ConvertToLocal(stopParams.m_pivot, pivot, kShapeCoordScalar); GenerateMarker(context, pt, dir, 1.0f, 1.0f, kOuterScale, kOuterScale, kOuterMarkerDepth, outerColor, batcher); @@ -697,7 +1278,7 @@ void TransitSchemeBuilder::GenerateStop(ref_ptr context, St } void TransitSchemeBuilder::GenerateTitles(ref_ptr context, - StopNodeParams const & stopParams, + StopNodeParamsSubway const & stopParams, m2::PointD const & pivot, std::vector const & markerSizes, ref_ptr textures, @@ -705,13 +1286,16 @@ void TransitSchemeBuilder::GenerateTitles(ref_ptr context, { auto const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); - auto const titles = PlaceTitles(stopParams, kTransitMarkTextSize * vs, textures); + std::vector titles = GetTitles(stopParams); if (titles.empty()) return; + PlaceTitles(titles, kTransitMarkTextSize * vs, textures); + auto const featureId = stopParams.m_stopsInfo.begin()->second.m_featureId; - auto priority = static_cast(stopParams.m_isTransfer ? Priority::TransferMin : Priority::StopMin); + auto priority = + static_cast(stopParams.m_isTransfer ? Priority::TransferMin : Priority::StopMin); priority += static_cast(stopParams.m_stopsInfo.size()); auto minVisibleScale = stopParams.m_isTransfer ? kTransferMinZoomLevel : kStopMinZoomLevel; @@ -723,8 +1307,10 @@ void TransitSchemeBuilder::GenerateTitles(ref_ptr context, priority += kFinalStationPriorityInc; } - ASSERT_LESS_OR_EQUAL(priority, static_cast(stopParams.m_isTransfer ? Priority::TransferMax - : Priority::StopMax), ()); + ASSERT_LESS_OR_EQUAL( + priority, + static_cast(stopParams.m_isTransfer ? Priority::TransferMax : Priority::StopMax), + ()); std::vector symbolSizes; symbolSizes.reserve(markerSizes.size()); @@ -753,8 +1339,9 @@ void TransitSchemeBuilder::GenerateTitles(ref_ptr context, textParams.m_startOverlayRank = dp::OverlayRank0; textParams.m_minVisibleScale = minVisibleScale; - TextShape(stopParams.m_pivot, textParams, TileKey(), symbolSizes, - title.m_offset, dp::Center, kTransitOverlayIndex).Draw(context, &batcher, textures); + TextShape(stopParams.m_pivot, textParams, TileKey(), symbolSizes, title.m_offset, dp::Center, + kTransitOverlayIndex) + .Draw(context, &batcher, textures); } df::ColoredSymbolViewParams colorParams; @@ -768,94 +1355,153 @@ void TransitSchemeBuilder::GenerateTitles(ref_ptr context, colorParams.m_specialPriority = static_cast(Priority::Stub); colorParams.m_startOverlayRank = dp::OverlayRank0; - ColoredSymbolShape(stopParams.m_pivot, colorParams, TileKey(), - kTransitStubOverlayIndex, markerSizes).Draw(context, &batcher, textures); + ColoredSymbolShape(stopParams.m_pivot, colorParams, TileKey(), kTransitStubOverlayIndex, + markerSizes) + .Draw(context, &batcher, textures); } -void TransitSchemeBuilder::GenerateMarker(ref_ptr context, - m2::PointD const & pt, m2::PointD widthDir, - float linesCountWidth, float linesCountHeight, - float scaleWidth, float scaleHeight, float depth, - dp::Color const & color, dp::Batcher & batcher) +void TransitSchemeBuilder::GenerateTitles(ref_ptr context, + StopNodeParamsPT const & stopParams, + m2::PointD const & pivot, + std::vector const & markerSizes, + ref_ptr textures, + dp::Batcher & batcher) { - using TV = TransitStaticVertex; + auto const vs = static_cast(df::VisualParams::Instance().GetVisualScale()); - scaleWidth = (scaleWidth - 1.0f) / linesCountWidth + 1.0f; - scaleHeight = (scaleHeight - 1.0f) / linesCountHeight + 1.0f; + std::vector titles = GetTitles(stopParams); + if (titles.empty()) + return; - widthDir.y = -widthDir.y; - m2::PointD heightDir = widthDir.Ort(); + PlaceTitles(titles, kTransitMarkTextSize * vs, textures); - auto const v1 = -widthDir * scaleWidth * linesCountWidth - heightDir * scaleHeight * linesCountHeight; - auto const v2 = widthDir * scaleWidth * linesCountWidth - heightDir * scaleHeight * linesCountHeight; - auto const v3 = widthDir * scaleWidth * linesCountWidth + heightDir * scaleHeight * linesCountHeight; - auto const v4 = -widthDir * scaleWidth * linesCountWidth + heightDir * scaleHeight * linesCountHeight; + auto const featureId = stopParams.m_stopsInfo.begin()->second.m_featureId; - glsl::vec3 const pos(pt.x, pt.y, depth); - auto const colorVal = glsl::vec4(color.GetRedF(), color.GetGreenF(), color.GetBlueF(), 1.0f /* alpha */ ); + auto priority = + static_cast(stopParams.m_isTransfer ? Priority::TransferMin : Priority::StopMin); + priority += static_cast(stopParams.m_stopsInfo.size()); - TGeometryBuffer geometry; - geometry.reserve(6); - geometry.emplace_back(pos, TV::TNormal(v1.x, v1.y, -linesCountWidth, -linesCountHeight), colorVal); - geometry.emplace_back(pos, TV::TNormal(v2.x, v2.y, linesCountWidth, -linesCountHeight), colorVal); - geometry.emplace_back(pos, TV::TNormal(v3.x, v3.y, linesCountWidth, linesCountHeight), colorVal); - geometry.emplace_back(pos, TV::TNormal(v1.x, v1.y, -linesCountWidth, -linesCountHeight), colorVal); - geometry.emplace_back(pos, TV::TNormal(v3.x, v3.y, linesCountWidth, linesCountHeight), colorVal); - geometry.emplace_back(pos, TV::TNormal(v4.x, v4.y, -linesCountWidth, linesCountHeight), colorVal); + auto minVisibleScale = stopParams.m_isTransfer ? kTransferMinZoomLevel : kStopMinZoomLevel; - dp::AttributeProvider provider(1 /* stream count */, static_cast(geometry.size())); - provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), make_ref(geometry.data())); - auto state = CreateRenderState(gpu::Program::TransitMarker, DepthLayer::TransitSchemeLayer); - batcher.InsertTriangleList(context, state, make_ref(&provider)); -} - -void TransitSchemeBuilder::GenerateLine(ref_ptr context, - std::vector const & path, - m2::PointD const & pivot, dp::Color const & colorConst, - float lineOffset, float halfWidth, float depth, - dp::Batcher & batcher) -{ - using TV = TransitStaticVertex; - - TGeometryBuffer geometry; - auto const color = glsl::vec4(colorConst.GetRedF(), colorConst.GetGreenF(), colorConst.GetBlueF(), 1.0f /* alpha */); - size_t const kAverageSize = path.size() * 6; - size_t const kAverageCapSize = 12; - geometry.reserve(kAverageSize + kAverageCapSize * 2); - - std::vector segments; - segments.reserve(path.size() - 1); - - for (size_t i = 1; i < path.size(); ++i) + if (stopParams.m_isTerminalStop) { - if (path[i].EqualDxDy(path[i - 1], 1.0e-5)) - continue; - - SchemeSegment segment; - segment.m_p1 = glsl::ToVec2(MapShape::ConvertToLocal(path[i - 1], pivot, kShapeCoordScalar)); - segment.m_p2 = glsl::ToVec2(MapShape::ConvertToLocal(path[i], pivot, kShapeCoordScalar)); - CalculateTangentAndNormals(segment.m_p1, segment.m_p2, segment.m_tangent, - segment.m_leftNormal, segment.m_rightNormal); - - auto const startPivot = glsl::vec3(segment.m_p1, depth); - auto const endPivot = glsl::vec3(segment.m_p2, depth); - auto const offset = lineOffset * segment.m_rightNormal; - - geometry.emplace_back(startPivot, TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), color); - geometry.emplace_back(startPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); - geometry.emplace_back(endPivot, TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), color); - geometry.emplace_back(endPivot, TV::TNormal(segment.m_rightNormal * halfWidth - offset, -halfWidth, 0.0), color); - geometry.emplace_back(startPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); - geometry.emplace_back(endPivot, TV::TNormal(segment.m_leftNormal * halfWidth - offset, halfWidth, 0.0), color); - - segments.emplace_back(std::move(segment)); + minVisibleScale = std::min(minVisibleScale, kFinalStationMinZoomLevel); + priority += kFinalStationPriorityInc; } - dp::AttributeProvider provider(1 /* stream count */, static_cast(geometry.size())); - provider.InitStream(0 /* stream index */, GetTransitStaticBindingInfo(), make_ref(geometry.data())); - auto state = CreateRenderState(gpu::Program::Transit, DepthLayer::TransitSchemeLayer); - batcher.InsertTriangleList(context, state, make_ref(&provider)); + ASSERT_LESS_OR_EQUAL( + priority, + static_cast(stopParams.m_isTransfer ? Priority::TransferMax : Priority::StopMax), + ()); - GenerateLineCaps(context, segments, color, lineOffset, halfWidth, depth, batcher); + std::vector symbolSizes; + symbolSizes.reserve(markerSizes.size()); + for (auto const & sz : markerSizes) + symbolSizes.push_back(sz * 1.1f); + + dp::TitleDecl titleDecl; + titleDecl.m_primaryOptional = true; + titleDecl.m_primaryTextFont.m_color = df::GetColorConstant(kTransitMarkText); + titleDecl.m_primaryTextFont.m_outlineColor = df::GetColorConstant(kTransitMarkTextOutline); + titleDecl.m_primaryTextFont.m_size = kTransitMarkTextSize * vs; + titleDecl.m_anchor = dp::Left; + + for (auto const & title : titles) + { + TextViewParams textParams; + textParams.m_featureId = featureId; + textParams.m_tileCenter = pivot; + textParams.m_titleDecl = titleDecl; + textParams.m_titleDecl.m_primaryText = title.m_text; + textParams.m_titleDecl.m_anchor = title.m_anchor; + textParams.m_depthTestEnabled = false; + textParams.m_depthLayer = DepthLayer::TransitSchemeLayer; + textParams.m_specialDisplacement = SpecialDisplacement::SpecialModeUserMark; + textParams.m_specialPriority = priority; + textParams.m_startOverlayRank = dp::OverlayRank0; + textParams.m_minVisibleScale = minVisibleScale; + + TextShape(stopParams.m_pivot, textParams, TileKey(), symbolSizes, title.m_offset, dp::Center, + kTransitOverlayIndex) + .Draw(context, &batcher, textures); + } + + df::ColoredSymbolViewParams colorParams; + colorParams.m_radiusInPixels = markerSizes.front().x * 0.5f; + colorParams.m_color = dp::Color::Transparent(); + colorParams.m_featureId = featureId; + colorParams.m_tileCenter = pivot; + colorParams.m_depthTestEnabled = false; + colorParams.m_depthLayer = DepthLayer::TransitSchemeLayer; + colorParams.m_specialDisplacement = SpecialDisplacement::SpecialModeUserMark; + colorParams.m_specialPriority = static_cast(Priority::Stub); + colorParams.m_startOverlayRank = dp::OverlayRank0; + + ColoredSymbolShape(stopParams.m_pivot, colorParams, TileKey(), kTransitStubOverlayIndex, + markerSizes) + .Draw(context, &batcher, textures); +} + +void TransitSchemeBuilder::GenerateTransfer(ref_ptr context, + StopNodeParamsSubway const & stopParams, + m2::PointD const & pivot, dp::Batcher & batcher) +{ + m2::PointD const pt = MapShape::ConvertToLocal(stopParams.m_pivot, pivot, kShapeCoordScalar); + + size_t maxLinesCount = 0; + m2::PointD dir; + + for (auto const & shapeInfo : stopParams.m_shapesInfo) + { + if (shapeInfo.second.m_linesCount > maxLinesCount) + { + dir = shapeInfo.second.m_direction; + maxLinesCount = shapeInfo.second.m_linesCount; + } + } + + float const kInnerScale = 1.0f; + float const kOuterScale = 1.5f; + + auto const outerColor = GetColorConstant(kTransitTransferOuterColor); + auto const innerColor = GetColorConstant(kTransitTransferInnerColor); + + float const widthLinesCount = maxLinesCount > 3 ? 1.6f : 1.0f; + float const innerScale = maxLinesCount == 1 ? 1.4f : kInnerScale; + float const outerScale = maxLinesCount == 1 ? 1.9f : kOuterScale; + + GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, outerScale, outerScale, + kOuterMarkerDepth, outerColor, batcher); + + GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, innerScale, innerScale, + kInnerMarkerDepth, innerColor, batcher); +} + +void TransitSchemeBuilder::GenerateTransfer(ref_ptr context, + StopNodeParamsPT const & stopParams, + m2::PointD const & pivot, dp::Batcher & batcher) +{ + m2::PointD const pt = MapShape::ConvertToLocal(stopParams.m_pivot, pivot, kShapeCoordScalar); + + auto [dir, maxLinesCount] = + GetFittingDirectionAndSize(stopParams.m_shapeInfoIn, stopParams.m_shapeInfoOut); + + CHECK_GREATER(maxLinesCount, 0, ()); + + float const kInnerScale = 1.0f; + float const kOuterScale = 1.5f; + + auto const outerColor = GetColorConstant(kTransitTransferOuterColor); + auto const innerColor = GetColorConstant(kTransitTransferInnerColor); + + float const widthLinesCount = maxLinesCount > 3 ? 1.6f : 1.0f; + float const innerScale = maxLinesCount == 1 ? 1.4f : kInnerScale; + float const outerScale = maxLinesCount == 1 ? 1.9f : kOuterScale; + + GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, outerScale, outerScale, + kOuterMarkerDepth, outerColor, batcher); + + GenerateMarker(context, pt, dir, widthLinesCount, maxLinesCount, innerScale, innerScale, + kInnerMarkerDepth, innerColor, batcher); } } // namespace df diff --git a/drape_frontend/transit_scheme_builder.hpp b/drape_frontend/transit_scheme_builder.hpp index 44f8938306..d7c11321de 100644 --- a/drape_frontend/transit_scheme_builder.hpp +++ b/drape_frontend/transit_scheme_builder.hpp @@ -6,6 +6,7 @@ #include "drape/texture_manager.hpp" #include "transit/transit_display_info.hpp" +#include "transit/transit_version.hpp" #include #include @@ -56,7 +57,8 @@ struct LineParams : m_color(color), m_depth(depth) {} std::string m_color; - float m_depth; + float m_depth = 0.0; + std::vector<::transit::TransitId> m_stopIds; }; struct ShapeParams @@ -66,12 +68,25 @@ struct ShapeParams std::vector m_polyline; }; -struct ShapeInfo +struct ShapeInfoSubway { m2::PointD m_direction; size_t m_linesCount; }; +struct ShapeInfoPT +{ + ShapeInfoPT() = default; + + ShapeInfoPT(m2::PointD const & dir, std::set const & colors) + : m_direction(dir), m_colors(colors) + { + } + + m2::PointD m_direction; + std::set m_colors; +}; + struct StopInfo { StopInfo() = default; @@ -85,14 +100,47 @@ struct StopInfo std::set m_lines; }; -struct StopNodeParams +struct StopNodeParamsSubway { bool m_isTransfer = false; m2::PointD m_pivot; - std::map m_shapesInfo; + std::map m_shapesInfo; std::map m_stopsInfo; }; +struct StopNodeParamsPT +{ + bool m_isTransfer = false; + bool m_isTerminalStop = false; + m2::PointD m_pivot; + // Two types of line shapes going through the stop: inbound and outbound. + std::vector m_shapeInfoIn; + std::vector m_shapeInfoOut; + // Route id to StopInfo mapping. + std::map<::transit::TransitId, StopInfo> m_stopsInfo; +}; + +using IdToIdSet = std::unordered_map<::transit::TransitId, ::transit::IdSet>; + +struct RouteSegment +{ + int m_order = 0; + std::vector m_polyline; +}; + +struct RouteData +{ + std::string m_color; + float m_depth = 0; + std::vector m_routeShapes; +}; + +struct LinesDataPT +{ + IdToIdSet m_stopToLineIds; + std::set<::transit::TransitId> m_terminalStops; +}; + class TransitSchemeBuilder { public: @@ -127,21 +175,45 @@ private: { m2::PointD m_pivot; - std::map m_lines; - std::map m_shapes; - std::map m_stops; - std::map m_transfers; + ::transit::TransitVersion m_transitVersion; + + std::map m_linesSubway; + std::map m_shapesSubway; + std::map m_stopsSubway; + std::map m_transfersSubway; + + std::map<::transit::TransitId, LineParams> m_linesPT; + std::vector m_routeSegmentsPT; + std::map<::transit::TransitId, StopNodeParamsPT> m_stopsPT; + std::map<::transit::TransitId, StopNodeParamsPT> m_transfersPT; }; void BuildScheme(ref_ptr context, MwmSet::MwmId const & mwmId, ref_ptr textures); - void CollectStops(TransitDisplayInfo const & transitDisplayInfo, - MwmSet::MwmId const & mwmId, MwmSchemeData & scheme); + void GenerateLinesSubway(MwmSchemeData const & scheme, dp::Batcher & batcher, + ref_ptr context); - void CollectLines(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); + void GenerateLinesPT(MwmSchemeData const & scheme, dp::Batcher & batcher, + ref_ptr context); + + template + void GenerateLocationsWithTitles(ref_ptr context, + ref_ptr textures, dp::Batcher & batcher, + F && flusher, MwmSchemeData const & scheme, S const & stops, + T const & transfers, L const & lines); + + void CollectStopsSubway(TransitDisplayInfo const & transitDisplayInfo, + MwmSet::MwmId const & mwmId, MwmSchemeData & scheme); + void CollectStopsPT(TransitDisplayInfo const & transitDisplayInfo, LinesDataPT const & linesData, + MwmSet::MwmId const & mwmId, MwmSchemeData & scheme); + + void CollectLinesSubway(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); + LinesDataPT CollectLinesPT(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); + + void CollectShapesSubway(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); + void CollectShapesPT(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); - void CollectShapes(TransitDisplayInfo const & transitDisplayInfo, MwmSchemeData & scheme); void FindShapes(routing::transit::StopId stop1Id, routing::transit::StopId stop2Id, routing::transit::LineId lineId, std::vector const & sameLines, @@ -149,7 +221,9 @@ private: void AddShape(TransitDisplayInfo const & transitDisplayInfo, routing::transit::StopId stop1Id, routing::transit::StopId stop2Id, routing::transit::LineId lineId, MwmSchemeData & scheme); - void PrepareScheme(MwmSchemeData & scheme); + void PrepareSchemeSubway(MwmSchemeData & scheme); + void PrepareSchemePT(TransitDisplayInfo const & transitDisplayInfo, LinesDataPT const & lineData, + MwmSchemeData & scheme); void GenerateShapes(ref_ptr context, MwmSet::MwmId const & mwmId); @@ -161,15 +235,28 @@ private: float scaleWidth, float scaleHeight, float depth, dp::Color const & color, dp::Batcher & batcher); - void GenerateTransfer(ref_ptr context, StopNodeParams const & params, + void GenerateTransfer(ref_ptr context, + StopNodeParamsSubway const & stopParams, m2::PointD const & pivot, + dp::Batcher & batcher); + + void GenerateTransfer(ref_ptr context, StopNodeParamsPT const & stopParams, m2::PointD const & pivot, dp::Batcher & batcher); - void GenerateStop(ref_ptr context, StopNodeParams const & params, + void GenerateStop(ref_ptr context, StopNodeParamsSubway const & stopParams, m2::PointD const & pivot, std::map const & lines, dp::Batcher & batcher); - void GenerateTitles(ref_ptr context, StopNodeParams const & params, + void GenerateStop(ref_ptr context, StopNodeParamsPT const & stopParams, + m2::PointD const & pivot, + std::map const & lines, + dp::Batcher & batcher); + + void GenerateTitles(ref_ptr context, StopNodeParamsSubway const & stopParams, + m2::PointD const & pivot, std::vector const & markerSizes, + ref_ptr textures, dp::Batcher & batcher); + + void GenerateTitles(ref_ptr context, StopNodeParamsPT const & stopParams, m2::PointD const & pivot, std::vector const & markerSizes, ref_ptr textures, dp::Batcher & batcher); diff --git a/transit/transit_entities.hpp b/transit/transit_entities.hpp index 3fa98bd124..6fa210eaa1 100644 --- a/transit/transit_entities.hpp +++ b/transit/transit_entities.hpp @@ -42,6 +42,10 @@ inline std::string const kEdgesTransferFile = "edges_transfer" + kTransitFileExt inline std::string const kTransfersFile = "transfers" + kTransitFileExtension; inline std::string const kGatesFile = "gates" + kTransitFileExtension; +// Route types shown on the subway layer. +static std::unordered_set const kSubwayLayerTypes{"subway", "train", "light_rail", + "monorail"}; + // Unique id for transit entities. It is generated by gtfs_converter and is persistent between // re-runs. Generated based on the unique string hash of the GTFS entity. Lies in the interval // |routing::FakeFeatureIds::IsTransitFeature()|. If the GTFS entity is renamed or the new GTFS