From 6b3ddfbd4e61659fe1053db0002d4e3a02d366b9 Mon Sep 17 00:00:00 2001 From: Daria Volvenkova Date: Tue, 24 Mar 2020 15:50:35 +0300 Subject: [PATCH] [bookmarks] Review fixes. --- map/track.cpp | 124 ++++++++++++++++++++++++++++---------------------- map/track.hpp | 4 +- 2 files changed, 70 insertions(+), 58 deletions(-) diff --git a/map/track.cpp b/map/track.cpp index 78b8d3d1ed..153ef818bf 100644 --- a/map/track.cpp +++ b/map/track.cpp @@ -7,6 +7,60 @@ #include +namespace +{ +bool GetTrackPoint(std::vector const & points, + std::vector const & lengths, double distanceInMeters, m2::PointD & pt) +{ + CHECK_GREATER_OR_EQUAL(distanceInMeters, 0.0, ()); + CHECK_EQUAL(points.size(), lengths.size(), ()); + + double const kEpsMeters = 1e-2; + if (base::AlmostEqualAbs(distanceInMeters, lengths.front(), kEpsMeters)) + { + pt = points.front().GetPoint(); + return true; + } + + if (base::AlmostEqualAbs(distanceInMeters, lengths.back(), kEpsMeters)) + { + pt = points.back().GetPoint(); + return true; + } + + auto const it = std::lower_bound(lengths.begin(), lengths.end(), distanceInMeters); + if (it == lengths.end()) + return false; + + auto const pointIndex = std::distance(lengths.begin(), it); + auto const length = *it; + + auto const segmentLength = length - lengths[pointIndex - 1]; + auto const k = (segmentLength - (length - distanceInMeters)) / segmentLength; + + auto const & pt1 = points[pointIndex - 1].GetPoint(); + auto const & pt2 = points[pointIndex].GetPoint(); + pt = pt1 + (pt2 - pt1) * k; + + return true; +} + +double GetLengthInMeters(std::vector const & points, size_t pointIndex) +{ + CHECK_LESS(pointIndex, points.size(), (pointIndex, points.size())); + + double length = 0.0; + for (size_t i = 1; i <= pointIndex; ++i) + { + auto const & pt1 = points[i - 1].GetPoint(); + auto const & pt2 = points[i].GetPoint(); + auto const segmentLength = mercator::DistanceOnEarth(pt1, pt2); + length += segmentLength; + } + return length; +} +} // namespace + Track::Track(kml::TrackData && data, bool interactive) : Base(data.m_id == kml::kInvalidTrackId ? UserMarkIdStorage::Instance().GetNextTrackId() : data.m_id) , m_data(std::move(data)) @@ -20,14 +74,13 @@ Track::Track(kml::TrackData && data, bool interactive) void Track::CacheDataForInteraction() { m_interactionData = InteractionData(); - GetLengthsImpl(m_interactionData->m_lengths); + m_interactionData->m_lengths = GetLengthsImpl(); m_interactionData->m_limitRect = GetLimitRectImpl(); } -void Track::GetLengthsImpl(std::vector & lengths) const +std::vector Track::GetLengthsImpl() const { - lengths.resize(m_data.m_pointsWithAltitudes.size()); - lengths[0] = 0.0; + std::vector lengths(m_data.m_pointsWithAltitudes.size(), 0.0); for (size_t i = 1; i < m_data.m_pointsWithAltitudes.size(); ++i) { auto const & pt1 = m_data.m_pointsWithAltitudes[i - 1].GetPoint(); @@ -35,6 +88,7 @@ void Track::GetLengthsImpl(std::vector & lengths) const auto const segmentLength = mercator::DistanceOnEarth(pt1, pt2); lengths[i] = lengths[i - 1] + segmentLength; } + return lengths; } m2::RectD Track::GetLimitRectImpl() const @@ -52,7 +106,8 @@ bool Track::HasAltitudes() const { if (pt.GetAltitude() == geometry::kInvalidAltitude) return false; - hasNonDefaultAltitude |= pt.GetAltitude() != geometry::kDefaultAltitudeMeters; + if (!hasNonDefaultAltitude && pt.GetAltitude() != geometry::kDefaultAltitudeMeters) + hasNonDefaultAltitude = true; } return hasNonDefaultAltitude; @@ -75,26 +130,18 @@ double Track::GetLengthMeters() const if (m_interactionData) return m_interactionData->m_lengths.back(); - return GetLengthMeters(m_data.m_pointsWithAltitudes.size() - 1); + return GetLengthInMeters(m_data.m_pointsWithAltitudes, m_data.m_pointsWithAltitudes.size() - 1); } double Track::GetLengthMeters(size_t pointIndex) const { - CHECK_LESS(pointIndex, m_data.m_pointsWithAltitudes.size(), - (pointIndex, m_data.m_pointsWithAltitudes.size())); - if (m_interactionData) - return m_interactionData->m_lengths[pointIndex]; - - double length = 0.0; - for (size_t i = 1; i <= pointIndex; ++i) { - auto const & pt1 = m_data.m_pointsWithAltitudes[i - 1].GetPoint(); - auto const & pt2 = m_data.m_pointsWithAltitudes[i].GetPoint(); - auto const segmentLength = mercator::DistanceOnEarth(pt1, pt2); - length += segmentLength; + CHECK_LESS(pointIndex, m_interactionData->m_lengths.size(), ()); + return m_interactionData->m_lengths[pointIndex]; } - return length; + + return GetLengthInMeters(m_data.m_pointsWithAltitudes, pointIndex); } bool Track::IsInteractive() const @@ -162,45 +209,12 @@ void Track::SetSelectionMarkId(kml::MarkId markId) bool Track::GetPoint(double distanceInMeters, m2::PointD & pt) const { - CHECK_GREATER_OR_EQUAL(distanceInMeters, 0.0, (distanceInMeters)); - if (m_interactionData) - return GetPointImpl(m_interactionData->m_lengths, distanceInMeters, pt); - - std::vector lengths; - GetLengthsImpl(lengths); - return GetPointImpl(lengths, distanceInMeters, pt); -} - -bool Track::GetPointImpl(std::vector const & lengths, double distanceInMeters, - m2::PointD & pt) const -{ - double const kEpsMeters = 1e-2; - if (base::AlmostEqualAbs(distanceInMeters, lengths.front(), kEpsMeters)) { - pt = m_data.m_pointsWithAltitudes.front().GetPoint(); - return true; + return GetTrackPoint(m_data.m_pointsWithAltitudes, m_interactionData->m_lengths, + distanceInMeters, pt); } - if (base::AlmostEqualAbs(distanceInMeters, lengths.back(), kEpsMeters)) - { - pt = m_data.m_pointsWithAltitudes.back().GetPoint(); - return true; - } - - auto const it = std::lower_bound(lengths.begin(), lengths.end(), distanceInMeters); - if (it == lengths.end()) - return false; - - auto const pointIndex = std::distance(lengths.begin(), it); - auto const length = *it; - - auto const segmentLength = length - lengths[pointIndex - 1]; - auto const k = (segmentLength - (length - distanceInMeters)) / segmentLength; - - auto const & pt1 = m_data.m_pointsWithAltitudes[pointIndex - 1].GetPoint(); - auto const & pt2 = m_data.m_pointsWithAltitudes[pointIndex].GetPoint(); - pt = pt1 + (pt2 - pt1) * k; - - return true; + auto const lengths = GetLengthsImpl(); + return GetTrackPoint(m_data.m_pointsWithAltitudes, lengths, distanceInMeters, pt); } diff --git a/map/track.hpp b/map/track.hpp index e8383d66d4..349393f27c 100644 --- a/map/track.hpp +++ b/map/track.hpp @@ -44,10 +44,8 @@ public: private: void CacheDataForInteraction(); bool HasAltitudes() const; - void GetLengthsImpl(std::vector & lengths) const; + std::vector GetLengthsImpl() const; m2::RectD GetLimitRectImpl() const; - bool GetPointImpl(std::vector const & lengths, double distanceInMeters, - m2::PointD & pt) const; kml::TrackData m_data; kml::MarkGroupId m_groupID = kml::kInvalidMarkGroupId;