From 442ee4a443983cbbd35a7869f0a7abd8cc916362 Mon Sep 17 00:00:00 2001 From: tatiana-yan Date: Fri, 5 Mar 2021 11:02:05 +0300 Subject: [PATCH] [transit] Fix projection for backward trip. --- transit/world_feed/feed_helpers.cpp | 77 +++++++----- transit/world_feed/feed_helpers.hpp | 8 +- transit/world_feed/world_feed.cpp | 117 +++++++++--------- .../world_feed_tests/world_feed_tests.cpp | 105 +++++++++++++--- 4 files changed, 204 insertions(+), 103 deletions(-) diff --git a/transit/world_feed/feed_helpers.cpp b/transit/world_feed/feed_helpers.cpp index f88299d7ad..b5bf224c08 100644 --- a/transit/world_feed/feed_helpers.cpp +++ b/transit/world_feed/feed_helpers.cpp @@ -23,7 +23,7 @@ struct ProjectionData // Distance from point to its projection. double m_distFromPoint = 0.0; // Distance from start point on polyline to the projection. - double m_distFromStart = 0.0; + double m_distFromEnding = 0.0; // Point on polyline almost equal to the projection can already exist, so we don't need to // insert projection. Or we insert it to the polyline. bool m_needsInsertion = false; @@ -31,14 +31,14 @@ struct ProjectionData // Returns true if |p1| is much closer to start then |p2| (parameter |distDeltaStart|) and its // distance to projections to polyline |m_distFromPoint| is comparable. -bool CloserToStartAndOnSimilarDistToLine(ProjectionData const & p1, ProjectionData const & p2) +bool CloserToEndingAndOnSimilarDistToLine(ProjectionData const & p1, ProjectionData const & p2) { // Delta between two points distances from start point on polyline. double constexpr distDeltaStart = 100.0; // Delta between two points distances from their corresponding projections to polyline. double constexpr distDeltaProj = 90.0; - return (p1.m_distFromStart + distDeltaStart < p2.m_distFromStart && + return (p1.m_distFromEnding + distDeltaStart < p2.m_distFromEnding && std::abs(p2.m_distFromPoint - p1.m_distFromPoint) <= distDeltaProj); } } // namespace @@ -55,25 +55,27 @@ ProjectionToShape ProjectStopOnTrack(m2::PointD const & stopPoint, m2::PointD co } ProjectionData GetProjection(std::vector const & polyline, size_t index, - ProjectionToShape const & proj) + Direction direction, ProjectionToShape const & proj) { ProjectionData projData; projData.m_distFromPoint = proj.m_dist; projData.m_proj = proj.m_point; + auto const next = direction == Direction::Forward ? index + 1 : index - 1; + if (base::AlmostEqualAbs(proj.m_point, polyline[index], kEps)) { projData.m_indexOnShape = index; projData.m_needsInsertion = false; } - else if (base::AlmostEqualAbs(proj.m_point, polyline[index + 1], kEps)) + else if (base::AlmostEqualAbs(proj.m_point, polyline[next], kEps)) { - projData.m_indexOnShape = index + 1; + projData.m_indexOnShape = next; projData.m_needsInsertion = false; } else { - projData.m_indexOnShape = index + 1; + projData.m_indexOnShape = direction == Direction::Forward ? next : index; projData.m_needsInsertion = true; } @@ -81,24 +83,36 @@ ProjectionData GetProjection(std::vector const & polyline, size_t in } void FillProjections(std::vector & polyline, size_t startIndex, size_t endIndex, - m2::PointD const & point, double distStopsM, + m2::PointD const & point, double distStopsM, Direction direction, std::vector & projections) { double distTravelledM = 0.0; // Stop can't be further from its projection to line then |maxDistFromStopM|. double constexpr maxDistFromStopM = 1000; - for (size_t i = startIndex; i < endIndex; ++i) - { - if (i > startIndex) - distTravelledM += mercator::DistanceOnEarth(polyline[i - 1], polyline[i]); + size_t const from = direction == Direction::Forward ? startIndex : endIndex; + auto const endCriteria = [&](size_t i) { + return direction == Direction::Forward ? i < endIndex : i > startIndex; + }; + auto const move = [direction](size_t & i) { direction == Direction::Forward ? ++i : --i; }; - auto proj = GetProjection(polyline, i, ProjectStopOnTrack(point, polyline[i], polyline[i + 1])); - proj.m_distFromStart = distTravelledM + mercator::DistanceOnEarth(polyline[i], proj.m_proj); + for (size_t i = from; endCriteria(i); move(i)) + { + auto const current = i; + auto const prev = direction == Direction::Forward ? i - 1 : i + 1; + auto const next = direction == Direction::Forward ? i + 1 : i - 1; + + if (i != from) + distTravelledM += mercator::DistanceOnEarth(polyline[prev], polyline[current]); + + auto proj = GetProjection(polyline, current, direction, + ProjectStopOnTrack(point, polyline[current], polyline[next])); + proj.m_distFromEnding = + distTravelledM + mercator::DistanceOnEarth(polyline[current], proj.m_proj); // The distance on the polyline between the projections of stops must not be less than the // shortest possible distance between the stops themselves. - if (proj.m_distFromStart < distStopsM) + if (proj.m_distFromEnding < distStopsM) continue; if (proj.m_distFromPoint < maxDistFromStopM) @@ -108,7 +122,7 @@ void FillProjections(std::vector & polyline, size_t startIndex, size std::pair PrepareNearestPointOnTrack(m2::PointD const & point, std::optional const & prevPoint, - size_t startIndex, + size_t prevIndex, Direction direction, std::vector & polyline) { // We skip 70% of the distance in a straight line between two stops for preventing incorrect @@ -118,9 +132,12 @@ std::pair PrepareNearestPointOnTrack(m2::PointD const & point, std::vector projections; // Reserve space for points on polyline which are relatively close to the shape. // Approximately 1/4 of all points on shape. - projections.reserve((polyline.size() - startIndex) / 4); + auto const size = direction == Direction::Forward ? polyline.size() - prevIndex : prevIndex; + projections.reserve(size / 4); - FillProjections(polyline, startIndex, polyline.size() - 1, point, distStopsM, projections); + auto const startIndex = direction == Direction::Forward ? prevIndex : 0; + auto const endIndex = direction == Direction::Forward ? polyline.size() - 1 : prevIndex; + FillProjections(polyline, startIndex, endIndex, point, distStopsM, direction, projections); if (projections.empty()) return {polyline.size() + 1, false}; @@ -128,24 +145,24 @@ std::pair PrepareNearestPointOnTrack(m2::PointD const & point, // We find the most fitting projection of the stop to the polyline. For two different projections // with approximately equal distances to the stop the most preferable is the one that is closer // to the beginning of the polyline segment. - auto proj = - std::min_element(projections.begin(), projections.end(), - [](ProjectionData const & p1, ProjectionData const & p2) { - if (CloserToStartAndOnSimilarDistToLine(p1, p2)) - return true; + auto const cmp = [](ProjectionData const & p1, ProjectionData const & p2) { + if (CloserToEndingAndOnSimilarDistToLine(p1, p2)) + return true; - if (CloserToStartAndOnSimilarDistToLine(p2, p1)) - return false; + if (CloserToEndingAndOnSimilarDistToLine(p2, p1)) + return false; - if (base::AlmostEqualAbs(p1.m_distFromPoint, p2.m_distFromPoint, kEps)) - return p1.m_distFromStart < p2.m_distFromStart; + if (base::AlmostEqualAbs(p1.m_distFromPoint, p2.m_distFromPoint, kEps)) + return p1.m_distFromEnding < p2.m_distFromEnding; - return p1.m_distFromPoint < p2.m_distFromPoint; - }); + return p1.m_distFromPoint < p2.m_distFromPoint; + }; + + auto proj = std::min_element(projections.begin(), projections.end(), cmp); // This case is possible not only for the first stop on the shape. We try to resolve situation // when two stops are projected to the same point on the shape. - if (proj->m_indexOnShape == startIndex) + if (proj->m_indexOnShape == prevIndex) { proj = std::min_element(projections.begin(), projections.end(), [](ProjectionData const & p1, ProjectionData const & p2) { diff --git a/transit/world_feed/feed_helpers.hpp b/transit/world_feed/feed_helpers.hpp index 339bad6610..f9e0be4b5b 100644 --- a/transit/world_feed/feed_helpers.hpp +++ b/transit/world_feed/feed_helpers.hpp @@ -20,6 +20,12 @@ struct ProjectionToShape double m_dist; }; +enum class Direction +{ + Forward, + Backward +}; + /// \returns |stopPoint| projection to the track segment [|point1|, |point2|] and /// distance from the |stopPoint| to its projection. ProjectionToShape ProjectStopOnTrack(m2::PointD const & stopPoint, m2::PointD const & point1, @@ -30,7 +36,7 @@ ProjectionToShape ProjectStopOnTrack(m2::PointD const & stopPoint, m2::PointD co /// the |polyline| and the flag is set to true. std::pair PrepareNearestPointOnTrack(m2::PointD const & point, std::optional const & prevPoint, - size_t startIndex, + size_t prevIndex, Direction direction, std::vector & polyline); /// \returns true if we should not skip routes with this GTFS |routeType|. diff --git a/transit/world_feed/world_feed.cpp b/transit/world_feed/world_feed.cpp index 678512c993..0a516a7a6f 100644 --- a/transit/world_feed/world_feed.cpp +++ b/transit/world_feed/world_feed.cpp @@ -211,15 +211,9 @@ struct StopOnShape size_t m_index = 0; }; -enum class Direction -{ - Forward, - Backward -}; - std::optional GetStopIndex( std::unordered_map> const & stopIndexes, - transit::TransitId id, size_t fromIndex, Direction direction) + transit::TransitId id, size_t fromIndex, transit::Direction direction) { auto it = stopIndexes.find(id); CHECK(it != stopIndexes.end(), (id)); @@ -227,9 +221,11 @@ std::optional GetStopIndex( std::optional bestIdx; for (auto const & index : it->second) { - if (direction == Direction::Forward && index >= fromIndex && (!bestIdx || index < bestIdx)) + if (direction == transit::Direction::Forward && index >= fromIndex && + (!bestIdx || index < bestIdx)) bestIdx = index; - if (direction == Direction::Backward && index <= fromIndex && (!bestIdx || index > bestIdx)) + if (direction == transit::Direction::Backward && index <= fromIndex && + (!bestIdx || index > bestIdx)) bestIdx = index; } return bestIdx; @@ -237,7 +233,8 @@ std::optional GetStopIndex( std::optional> GetStopPairOnShape( std::unordered_map> const & stopIndexes, - transit::StopsOnLines const & stopsOnLines, size_t index, size_t fromIndex, Direction direction) + transit::StopsOnLines const & stopsOnLines, size_t index, size_t fromIndex, + transit::Direction direction) { auto const & stopIds = stopsOnLines.m_stopSeq; @@ -277,14 +274,14 @@ Link::Link(transit::TransitId lineId, transit::TransitId shapeId, size_t shapeSi { } -Direction GetDirection( +transit::Direction GetDirection( transit::StopsOnLines const & stopsOnLines, std::unordered_map> const & stopIndexes) { auto const & stopIds = stopsOnLines.m_stopSeq; if (stopIds.size() <= 1 || !stopsOnLines.m_isValid) - return Direction::Forward; + return transit::Direction::Forward; for (size_t i = 0; i < stopIds.size() - 1; ++i) { @@ -300,10 +297,10 @@ Direction GetDirection( auto const index2 = indexes2->second[0]; if (index2 == index1) continue; - return index2 > index1 ? Direction::Forward : Direction::Backward; + return index2 > index1 ? transit::Direction::Forward : transit::Direction::Backward; } - return Direction::Forward; + return transit::Direction::Forward; } } // namespace @@ -949,61 +946,67 @@ bool WorldFeed::ProjectStopsToShape( { IdList const & stopIds = stopsOnLines.m_stopSeq; TransitId const shapeId = itShape->first; - auto & shape = itShape->second.m_points; - std::optional prevPoint = std::nullopt; - for (size_t i = 0; i < stopIds.size(); ++i) - { - auto const & stopId = stopIds[i]; - auto const itStop = m_stops.m_data.find(stopId); - CHECK(itStop != m_stops.m_data.end(), (stopId)); - auto const & stop = itStop->second; - - size_t const startIdx = i == 0 ? 0 : stopsToIndexes[stopIds[i - 1]].back(); - auto const [curIdx, pointInserted] = - PrepareNearestPointOnTrack(stop.m_point, prevPoint, startIdx, shape); - - if (curIdx > shape.size()) + auto const tryProject = [&](Direction direction) { + auto shape = itShape->second.m_points; + std::optional prevPoint = std::nullopt; + for (size_t i = 0; i < stopIds.size(); ++i) { - CHECK(!itShape->second.m_lineIds.empty(), (shapeId)); - TransitId const lineId = *stopsOnLines.m_lines.begin(); + auto const & stopId = stopIds[i]; + auto const itStop = m_stops.m_data.find(stopId); + CHECK(itStop != m_stops.m_data.end(), (stopId)); + auto const & stop = itStop->second; - LOG(LWARNING, - ("Error projecting stops to the shape. GTFS trip id", m_lines.m_data[lineId].m_gtfsTripId, - "shapeId", shapeId, "stopId", stopId, "i", i, "start index on shape", startIdx, - "trips count", stopsOnLines.m_lines.size())); - return false; - } + size_t const prevIdx = i == 0 ? direction == Direction::Forward ? 0 : shape.size() + : stopsToIndexes[stopIds[i - 1]].back(); + auto const [curIdx, pointInserted] = + PrepareNearestPointOnTrack(stop.m_point, prevPoint, prevIdx, direction, shape); - prevPoint = std::optional(stop.m_point); - - if (pointInserted) - { - for (auto & indexesList : stopsToIndexes) + if (curIdx > shape.size()) { - for (auto & stopIndex : indexesList.second) + CHECK(!itShape->second.m_lineIds.empty(), (shapeId)); + TransitId const lineId = *stopsOnLines.m_lines.begin(); + + LOG(LWARNING, + ("Error projecting stops to the shape. GTFS trip id", + m_lines.m_data[lineId].m_gtfsTripId, "shapeId", shapeId, "stopId", stopId, "i", i, + "previous index on shape", prevIdx, "trips count", stopsOnLines.m_lines.size())); + return false; + } + + prevPoint = std::optional(stop.m_point); + + if (pointInserted) + { + for (auto & indexesList : stopsToIndexes) { - if (stopIndex >= curIdx) - ++stopIndex; + for (auto & stopIndex : indexesList.second) + { + if (stopIndex >= curIdx) + ++stopIndex; + } + } + + for (auto const & lineId : m_shapes.m_data[shapeId].m_lineIds) + { + auto & line = m_lines.m_data[lineId]; + + if (line.m_shapeLink.m_startIndex >= curIdx) + ++line.m_shapeLink.m_startIndex; + + if (line.m_shapeLink.m_endIndex >= curIdx) + ++line.m_shapeLink.m_endIndex; } } - for (auto const & lineId : m_shapes.m_data[shapeId].m_lineIds) - { - auto & line = m_lines.m_data[lineId]; - - if (line.m_shapeLink.m_startIndex >= curIdx) - ++line.m_shapeLink.m_startIndex; - - if (line.m_shapeLink.m_endIndex >= curIdx) - ++line.m_shapeLink.m_endIndex; - } + stopsToIndexes[stopId].push_back(curIdx); } - stopsToIndexes[stopId].push_back(curIdx); - } + itShape->second.m_points = shape; + return true; + }; - return true; + return tryProject(Direction::Forward) || tryProject(Direction::Backward); } std::unordered_map> WorldFeed::GetStopsForShapeMatching() diff --git a/transit/world_feed/world_feed_tests/world_feed_tests.cpp b/transit/world_feed/world_feed_tests/world_feed_tests.cpp index fc6812fdb1..a2a76d4ee6 100644 --- a/transit/world_feed/world_feed_tests/world_feed_tests.cpp +++ b/transit/world_feed/world_feed_tests/world_feed_tests.cpp @@ -211,7 +211,8 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_Simple) // Test that point_A is projected between two existing polyline points and the new point is // added in the place of its projection. TestPlanFact(1 /* planIndex */, true /* planInsert */, - PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* prevIndex */, + Direction::Forward, shape)); TEST_EQUAL(shape.size(), 5, ()); TEST_EQUAL(shape[1 /* expectedIndex */], m2::PointD(point_A.x, y), ()); @@ -220,17 +221,35 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_Simple) // Expected point projection index is the same. // But this projection is not inserted (it is already present). TestPlanFact(1 /* planIndex */, false /* planInsert */, - PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* prevIndex */, + Direction::Forward, shape)); // So the shape size remains the same. TEST_EQUAL(shape.size(), 5, ()); // Test that point_B insertion leads to addition of the new projection to the shape. TestPlanFact(4, true, - PrepareNearestPointOnTrack(point_B, std::nullopt, 1 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_B, std::nullopt, 1 /* prevIndex */, + Direction::Forward, shape)); // Test that point_C insertion does not lead to the addition of the new projection. TestPlanFact(5, false, - PrepareNearestPointOnTrack(point_C, std::nullopt, 4 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_C, std::nullopt, 4 /* prevIndex */, + Direction::Forward, shape)); + + // Test point_C projection in backward direction. + TestPlanFact(5 /* planIndex */, false /* planInsert */, + PrepareNearestPointOnTrack(point_C, std::nullopt, shape.size() /* prevIndex */, + Direction::Backward, shape)); + + // Test point_B projection in backward direction. + TestPlanFact(4 /* planIndex */, false /* planInsert */, + PrepareNearestPointOnTrack(point_B, std::nullopt, 5 /* prevIndex */, + Direction::Backward, shape)); + + // Test point_A projection in backward direction. + TestPlanFact(1 /* planIndex */, false /* planInsert */, + PrepareNearestPointOnTrack(point_A, std::nullopt, 4 /* prevIndex */, + Direction::Backward, shape)); } // Stop is on approximately the same distance from the segment (0, 1) and segment (1, 2). @@ -251,7 +270,8 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_DifferentStartIndexes) { auto shape = referenceShape; TestPlanFact(1, true, - PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* prevIndex */, + Direction::Forward, shape)); TEST_EQUAL(shape.size(), 4, ()); TEST_EQUAL(shape[1 /* expectedIndex */], m2::PointD(0.001, point_A.y), ()); } @@ -260,7 +280,8 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_DifferentStartIndexes) { auto shape = referenceShape; TestPlanFact(2, true, - PrepareNearestPointOnTrack(point_A, std::nullopt, 1 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 1 /* prevIndex */, + Direction::Forward, shape)); TEST_EQUAL(shape.size(), 4, ()); TEST_EQUAL(shape[2 /* expectedIndex */], m2::PointD(point_A.x, 0.002), ()); } @@ -285,7 +306,8 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_MaxDistance) {0.010, 0.0031}, {0.005, 0.0031}, {0.001, 0.0031}}; m2::PointD const point_A{0.0028, 0.0029}; TestPlanFact(1, true, - PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* prevIndex */, + Direction::Forward, shape)); } // Complex shape with multiple points on it and multiple stops for projection. @@ -293,7 +315,7 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_MaxDistance) // +-----+ // C* / \ // /+\ / \ *D -// + / \*/ \ +// + / \+/ \ // / + // / | *E // + +-----+ @@ -309,7 +331,8 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_MaxDistance) // UNIT_TEST(Transit_GTFS_ProjectStopToLine_NearCircle) { - std::vector shape{ + double constexpr kEps = 1e-5; + std::vector const initialShape{ {0.003, 0.001}, {0.003, 0.0015}, {0.0025, 0.002}, {0.002, 0.0025}, {0.001, 0.0025}, {0.001, 0.0035}, {0.0015, 0.0045}, {0.0025, 0.005}, {0.0035, 0.0045}, {0.004, 0.0055}, {0.0055, 0.0055}, {0.0065, 0.0045}, {0.0065, 0.0035}, {0.0075, 0.0035}, {0.0075, 0.0025}, @@ -321,18 +344,70 @@ UNIT_TEST(Transit_GTFS_ProjectStopToLine_NearCircle) m2::PointD const point_D{0.0063, 0.005}; m2::PointD const point_E{0.008, 0.004}; m2::PointD const point_F{0.0047, 0.0005}; + + // Forward + auto shape = initialShape; TestPlanFact(2, true, - PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_A, std::nullopt, 0 /* prevIndex */, + Direction::Forward, shape)); + auto const coordA = shape[2]; + TestPlanFact(3, false, - PrepareNearestPointOnTrack(point_B, std::nullopt, 2 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_B, std::nullopt, 2 /* prevIndex */, + Direction::Forward, shape)); + auto const coordB = shape[3]; + TestPlanFact(10, true, - PrepareNearestPointOnTrack(point_C, std::nullopt, 3 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_C, std::nullopt, 3 /* prevIndex */, + Direction::Forward, shape)); + auto const coordC = shape[10]; + TestPlanFact(12, false, - PrepareNearestPointOnTrack(point_D, std::nullopt, 10 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_D, std::nullopt, 10 /* prevIndex */, + Direction::Forward, shape)); + auto const coordD = shape[12]; + TestPlanFact(14, true, - PrepareNearestPointOnTrack(point_E, std::nullopt, 12 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_E, std::nullopt, 12 /* prevIndex */, + Direction::Forward, shape)); + auto const coordE = shape[14]; + TestPlanFact(20, true, - PrepareNearestPointOnTrack(point_F, std::nullopt, 14 /* startIndex */, shape)); + PrepareNearestPointOnTrack(point_F, std::nullopt, 14 /* prevIndex */, + Direction::Forward, shape)); + + // Backward processing of reversed shape + shape = initialShape; + reverse(shape.begin(), shape.end()); + TestPlanFact(17, true, + PrepareNearestPointOnTrack(point_A, std::nullopt, shape.size() /* prevIndex */, + Direction::Backward, shape)); + TEST(base::AlmostEqualAbs(coordA, shape[17], kEps), (coordA, shape[17])); + + TestPlanFact(16, false, + PrepareNearestPointOnTrack(point_B, std::nullopt, 17 /* prevIndex */, + Direction::Backward, shape)); + TEST(base::AlmostEqualAbs(coordB, shape[16], kEps), (coordA, shape[17])); + + TestPlanFact(10, true, + PrepareNearestPointOnTrack(point_C, std::nullopt, 16 /* prevIndex */, + Direction::Backward, shape)); + TEST(base::AlmostEqualAbs(coordC, shape[10], kEps), (coordA, shape[17])); + + TestPlanFact(8, false, + PrepareNearestPointOnTrack(point_D, std::nullopt, 10 /* prevIndex */, + Direction::Backward, shape)); + TEST(base::AlmostEqualAbs(coordD, shape[8], kEps), (coordA, shape[17])); + + TestPlanFact(7, true, + PrepareNearestPointOnTrack(point_E, std::nullopt, 8 /* prevIndex */, + Direction::Backward, shape)); + TEST(base::AlmostEqualAbs(coordE, shape[7], kEps), (coordA, shape[17])); + + // point_F has different position because we do not insert before point 0. + TestPlanFact(2, true, + PrepareNearestPointOnTrack(point_F, std::nullopt, 7 /* prevIndex */, + Direction::Backward, shape)); } UNIT_TEST(Transit_ColorPicker)