Directions refactoring + Show speed limit #2796

Merged
AntonM030481 merged 5 commits from RouteSegments-refactoring-and-speed-limit into master 2022-06-26 21:35:51 +00:00
AntonM030481 commented 2022-06-21 13:37:55 +00:00 (Migrated from github.com)

Now RouteSegments generation starts at DirectionsEngine::Generate().
But this is just preparation, when segments are not created,
but turns, street names, any new staff like speed limits are created
with indexes when it should be applied later.
Later in FillSegmentInfo all this info is unpacked to RouteSegments
and populated by segments and junctions.
This approach is overcomplicated and makes difficult adding new info,
like speed limits, by creating more and more data to pack and unpack.
Also extra memory allocations are done for intermediate steps.
Several todo were found mentioning desirability of changes.
For this reason it was decided to create RouteSegments
at DirectionsEngine::Generate() with all available info
with minimal post-processing at later stages.

Now RouteSegments generation starts at DirectionsEngine::Generate(). But this is just preparation, when segments are not created, but turns, street names, any new staff like speed limits are created with indexes when it should be applied later. Later in FillSegmentInfo all this info is unpacked to RouteSegments and populated by segments and junctions. This approach is overcomplicated and makes difficult adding new info, like speed limits, by creating more and more data to pack and unpack. Also extra memory allocations are done for intermediate steps. Several todo were found mentioning desirability of changes. For this reason it was decided to create RouteSegments at DirectionsEngine::Generate() with all available info with minimal post-processing at later stages.
AntonM030481 commented 2022-06-21 13:45:27 +00:00 (Migrated from github.com)
#952
vng (Migrated from github.com) reviewed 2022-06-22 16:03:06 +00:00
vng (Migrated from github.com) commented 2022-06-22 14:19:01 +00:00

Add size_t const segmentsCount = routeSegments.size(); var

Add size_t const segmentsCount = routeSegments.size(); var
vng (Migrated from github.com) commented 2022-06-22 14:20:26 +00:00

I have doubts about the speed of this approach .. Please, comment for now and place todo.

I have doubts about the speed of this approach .. Please, comment for now and place todo.
@ -86,0 +86,4 @@
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
std::string m_speedLimitUnitsSuffix;
vng (Migrated from github.com) commented 2022-06-22 14:11:07 +00:00

Can we join this into one string?

Can we join this into one string?
@ -29,1 +23,4 @@
};
void RouteSegmentsFrom(std::vector<Segment> const & segments, std::vector<m2::PointD> const & path,
std::vector<turns::TurnItem> const & turns, std::vector<RouteSegment::RoadNameInfo> const & names,
vng (Migrated from github.com) commented 2022-06-22 16:03:01 +00:00

Add tools.cpp and move implementation?

Add tools.cpp and move implementation?
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 12:48:43 +00:00
@ -86,0 +86,4 @@
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
std::string m_speedLimitUnitsSuffix;
AntonM030481 (Migrated from github.com) commented 2022-06-23 12:48:43 +00:00

I was done according to this:
1703c99f75/routing/following_info.hpp (L57-L58)

I was done according to this: https://github.com/AntonM030481/organicmaps/blob/1703c99f7586ea40d7f43e8e371ba64a0ba41885/routing/following_info.hpp#L57-L58
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 12:54:05 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 12:54:04 +00:00

Can you clarify why?

Can you clarify why?
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 13:11:50 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 13:11:50 +00:00

Just leave comment with todo?
Or disable all new code by comments?

Just leave comment with todo? Or disable all new code by comments?
vng (Migrated from github.com) reviewed 2022-06-23 14:44:01 +00:00
vng (Migrated from github.com) commented 2022-06-23 14:44:00 +00:00

Avoid duplicates, routeSegments.size() used ~5 times below.

Avoid duplicates, routeSegments.size() used ~5 times below.
vng (Migrated from github.com) reviewed 2022-06-23 14:46:15 +00:00
@ -86,0 +86,4 @@
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
std::string m_speedLimitUnitsSuffix;
vng (Migrated from github.com) commented 2022-06-23 14:46:14 +00:00

Seems like we can store one string and modify FormatDistance accordingly.
Also fix the mentioned case, please.

Seems like we can store one string and modify FormatDistance accordingly. Also fix the mentioned case, please.
vng (Migrated from github.com) reviewed 2022-06-23 14:47:13 +00:00
vng (Migrated from github.com) commented 2022-06-23 14:47:13 +00:00

Can we disable only this code and keep pathSegment.m_maxSpeed empty (like its not stored)?

Can we disable only this code and keep pathSegment.m_maxSpeed empty (like its not stored)?
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 15:09:55 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 15:09:55 +00:00

Got it.
I will replace it by
size_t const kInvalidEnter = routeSegments.size();
It will make logic easier to understand.

Got it. I will replace it by `size_t const kInvalidEnter = routeSegments.size();` It will make logic easier to understand.
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 15:21:12 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 15:21:11 +00:00

Sure we can.

Sure we can.
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 15:21:52 +00:00
@ -29,1 +23,4 @@
};
void RouteSegmentsFrom(std::vector<Segment> const & segments, std::vector<m2::PointD> const & path,
std::vector<turns::TurnItem> const & turns, std::vector<RouteSegment::RoadNameInfo> const & names,
AntonM030481 (Migrated from github.com) commented 2022-06-23 15:21:52 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 15:23:58 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 15:23:58 +00:00

We can cache result of
routing::LoadMaxspeeds(handle);
If your concern is about performance.
Most likely it will be the same for all segments, right?

We can cache result of routing::LoadMaxspeeds(handle); If your concern is about performance. Most likely it will be the same for all segments, right?
vng (Migrated from github.com) reviewed 2022-06-23 16:03:36 +00:00
vng (Migrated from github.com) commented 2022-06-23 16:03:36 +00:00

Leave todo and I will think about the optimal strategy. We already have speeds cache in GeometryLoader (GeometryLoaderImpl), just need to access it here somehow.

Leave todo and I will think about the optimal strategy. We already have speeds cache in GeometryLoader (GeometryLoaderImpl), just need to access it here somehow.
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 16:15:55 +00:00
@ -86,0 +86,4 @@
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
std::string m_speedLimitUnitsSuffix;
AntonM030481 (Migrated from github.com) commented 2022-06-23 16:15:54 +00:00

The reason for separation is that in GUI values and units are shown now (and can be shown in the future) in different ways (fonts and positions).
image
In fact units for CURRENT speed already exist,
but current speed is calculated and stored separately (at device specific level),
so I decided to provide units here for asserted comparison.

The reason for separation is that in GUI values and units are shown now (and can be shown in the future) in different ways (fonts and positions). ![image](https://user-images.githubusercontent.com/85622715/175339379-bb900cf2-ad65-42a0-b0a9-a6cfc18c1ac8.png) In fact units for CURRENT speed already exist, but current speed is calculated and stored separately (at device specific level), so I decided to provide units here for asserted comparison.
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 16:19:19 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-23 16:19:18 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-23 16:38:56 +00:00
@ -29,1 +23,4 @@
};
void RouteSegmentsFrom(std::vector<Segment> const & segments, std::vector<m2::PointD> const & path,
std::vector<turns::TurnItem> const & turns, std::vector<RouteSegment::RoadNameInfo> const & names,
AntonM030481 (Migrated from github.com) commented 2022-06-23 16:38:56 +00:00

Can you please add to XCode project
routing/routing_tests/tools.cpp

Can you please add to XCode project routing/routing_tests/tools.cpp
vng (Migrated from github.com) reviewed 2022-06-23 17:09:58 +00:00
@ -86,0 +86,4 @@
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
std::string m_speedLimitUnitsSuffix;
vng (Migrated from github.com) commented 2022-06-23 17:09:58 +00:00

Hm, well ok, keep as is.

Hm, well ok, keep as is.
vng (Migrated from github.com) approved these changes 2022-06-23 17:20:20 +00:00
vng (Migrated from github.com) left a comment

Added minor comments.

Added minor comments.
@ -33,4 +29,1 @@
uint32_t turn_index = base::asserted_cast<uint32_t>(junctions.size() - 1);
turnsDir.emplace_back(TurnItem(turn_index, CarDirection::ReachedYourDestination));
double constexpr kMergeDistMeters = 15.0;
vng (Migrated from github.com) commented 2022-06-23 17:16:50 +00:00

We use 'k' prefix for the compile time constants, but here 'invalidEnter' is better.

We use 'k' prefix for the compile time constants, but here 'invalidEnter' is better.
vng (Migrated from github.com) commented 2022-06-23 17:17:05 +00:00

2 spaces for tab

2 spaces for tab
vng (Migrated from github.com) commented 2022-06-23 17:17:31 +00:00

; is redundant.

; is redundant.
vng (Migrated from github.com) commented 2022-06-23 17:18:18 +00:00

Well, we don't use this notation. Please, move ");" to the end of prev line.
And seems like RouteSegment() is not needed for emplace_back (but not sure).

Well, we don't use this notation. Please, move ");" to the end of prev line. And seems like RouteSegment() is not needed for emplace_back (but not sure).
AntonM030481 (Migrated from github.com) reviewed 2022-06-24 07:07:39 +00:00
@ -33,4 +29,1 @@
uint32_t turn_index = base::asserted_cast<uint32_t>(junctions.size() - 1);
turnsDir.emplace_back(TurnItem(turn_index, CarDirection::ReachedYourDestination));
double constexpr kMergeDistMeters = 15.0;
AntonM030481 (Migrated from github.com) commented 2022-06-24 07:07:39 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-24 07:08:51 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-24 07:08:51 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-24 07:23:21 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-24 07:23:21 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-24 07:24:12 +00:00
AntonM030481 (Migrated from github.com) commented 2022-06-24 07:24:11 +00:00

OMG, it's really not necessary!
Changed here and in routing/directions_engine.cpp

OMG, it's really not necessary! Changed here and in routing/directions_engine.cpp
biodranik (Migrated from github.com) reviewed 2022-06-26 22:18:08 +00:00
biodranik (Migrated from github.com) left a comment

@vng you're too fast )

@vng you're too fast )
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
biodranik (Migrated from github.com) commented 2022-06-25 05:53:53 +00:00
    // Immidiate update of the position in Route. m_framework.OnLocationUpdate is too late.
```suggestion // Immidiate update of the position in Route. m_framework.OnLocationUpdate is too late. ```
biodranik (Migrated from github.com) commented 2022-06-25 05:54:42 +00:00

Why updating it in m_framework.OnLocationUpdate is too late? It would be great to have a more detailed explanation here.

Why updating it in m_framework.OnLocationUpdate is too late? It would be great to have a more detailed explanation here.
biodranik (Migrated from github.com) commented 2022-06-25 05:55:26 +00:00

Also, should the update in m_framework.OnLocationUpdate be removed, because we already do it here?

Also, should the update in `m_framework.OnLocationUpdate` be removed, because we already do it here?
@ -35,4 +29,3 @@
double constexpr kMergeDistMeters = 15.0;
// For turns that are not EnterRoundAbout/ExitRoundAbout exitNum is always equal to zero.
// If a turn is EnterRoundAbout exitNum is a number of turns between two junctions:
biodranik (Migrated from github.com) commented 2022-06-25 05:58:09 +00:00

Generally, I would prefer to leave std:: and remove using namespace std; from the code, at least from the namespace level (using it locally in function bodies is ok). It could make surprises with Unity builds.

Generally, I would prefer to leave std:: and remove `using namespace std;` from the code, at least from the namespace level (using it locally in function bodies is ok). It could make surprises with [Unity builds](https://cmake.org/cmake/help/latest/prop_tgt/UNITY_BUILD.html#).
@ -85,1 +85,4 @@
//@}
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
biodranik (Migrated from github.com) commented 2022-06-26 22:08:13 +00:00

Should it have a default constructor = "0"?

Should it have a default constructor = "0"?
@ -117,3 +118,4 @@
m_poly.GetDistFromCurPointToRoutePointMeters() / curSegSpeedMPerS);
}
void Route::GetCurrentSpeedLimit(SpeedInUnits & speedLimit) const
biodranik (Migrated from github.com) commented 2022-06-26 22:10:23 +00:00

Why not returning/checking the value? What if IsValid() == false?

Why not returning/checking the value? What if IsValid() == false?
@ -0,0 +10,4 @@
m_session->SetOnNewTurnCallback([this]() { ++m_onNewTurnCallbackCounter; });
}
void RouteSegmentsFrom(std::vector<Segment> const & segments, std::vector<m2::PointD> const & path,
biodranik (Migrated from github.com) commented 2022-06-26 22:15:52 +00:00

Why not use 4 separate but simple and clear functions instead of this monster one?

Why not use 4 separate but simple and clear functions instead of this monster one?
@ -216,3 +237,4 @@
}
}
UNIT_TEST(TestIsLaneWayConformedTurnDirection)
biodranik (Migrated from github.com) commented 2022-06-26 22:17:00 +00:00

{} instead of empty containers.

{} instead of empty containers.
@ -261,1 +281,3 @@
{2, CarDirection::ReachedYourDestination}};
vector<turns::TurnItem> turns =
{{1, CarDirection::GoStraight},
{2, CarDirection::TurnLeft},
biodranik (Migrated from github.com) commented 2022-06-26 22:17:15 +00:00

{}

{}
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 08:39:04 +00:00
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
AntonM030481 (Migrated from github.com) commented 2022-06-27 08:39:04 +00:00

I need RoutingSession::OnLocationPositionChanged to be executed right now
to get updated FollowingInfo state. Previously you could see only old state,
so it prevented you from normal visual debug.

Framework::OnLocationUpdate calls RoutingSession::OnLocationPositionChanged later indirectly
via callbacks. BTW it is called 10+ times. I can't understand if it's normal.

@vng maybe you have any idea?
Just put breakpoint in master at RoutingSession::OnLocationPositionChanged
and check how many times it is called.

I need RoutingSession::OnLocationPositionChanged to be executed right now to get updated FollowingInfo state. Previously you could see only old state, so it prevented you from normal visual debug. Framework::OnLocationUpdate calls RoutingSession::OnLocationPositionChanged later indirectly via callbacks. BTW it is called 10+ times. I can't understand if it's normal. @vng maybe you have any idea? Just put breakpoint in master at RoutingSession::OnLocationPositionChanged and check how many times it is called.
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 08:51:47 +00:00
@ -35,4 +29,3 @@
double constexpr kMergeDistMeters = 15.0;
// For turns that are not EnterRoundAbout/ExitRoundAbout exitNum is always equal to zero.
// If a turn is EnterRoundAbout exitNum is a number of turns between two junctions:
AntonM030481 (Migrated from github.com) commented 2022-06-27 08:51:47 +00:00

Can you clarify this?
Currently almost all routing/*.cpp files have using namespace std; at global or namespace level.

Can you clarify this? Currently almost all routing/*.cpp files have `using namespace std;` at global or namespace level.
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 08:55:27 +00:00
@ -85,1 +85,4 @@
//@}
// Current speed limit. If no info about speed limit then m_speedLimit == 0.
std::string m_speedLimit;
AntonM030481 (Migrated from github.com) commented 2022-06-27 08:55:27 +00:00

Outdated comment. Changed to
// If no info about speed limit then m_speedLimit == "" and m_speedLimitUnitsSuffix == "".

Outdated comment. Changed to `// If no info about speed limit then m_speedLimit == "" and m_speedLimitUnitsSuffix == "".`
biodranik (Migrated from github.com) reviewed 2022-06-27 09:54:58 +00:00
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
biodranik (Migrated from github.com) commented 2022-06-27 09:54:57 +00:00

What do you mean "called 10+ times"? Can you please show some logs? That's not normal.

What do you mean "called 10+ times"? Can you please show some logs? That's not normal.
biodranik (Migrated from github.com) reviewed 2022-06-27 09:56:01 +00:00
@ -35,4 +29,3 @@
double constexpr kMergeDistMeters = 15.0;
// For turns that are not EnterRoundAbout/ExitRoundAbout exitNum is always equal to zero.
// If a turn is EnterRoundAbout exitNum is a number of turns between two junctions:
biodranik (Migrated from github.com) commented 2022-06-27 09:56:00 +00:00

Global level: Strictly NO. It may break the unity build.
Namespace level: better NO than YES.
Function level: YES if it makes the code cleaner.

Global level: Strictly NO. It may break the unity build. Namespace level: better NO than YES. Function level: YES if it makes the code cleaner.
vng (Migrated from github.com) reviewed 2022-06-27 12:55:12 +00:00
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
vng (Migrated from github.com) commented 2022-06-27 12:55:12 +00:00

Didn't check yet, but seems like this is the position extrapolation algorithm for the smooth location events.

Didn't check yet, but seems like this is the position extrapolation algorithm for the smooth location events.
vng (Migrated from github.com) reviewed 2022-06-27 12:55:24 +00:00
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
vng (Migrated from github.com) commented 2022-06-27 12:55:23 +00:00

And this is normal for usual use case :) But a bit frustrated for desktop debugging

And this is normal for usual use case :) But a bit frustrated for desktop debugging
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 13:34:50 +00:00
@ -35,4 +29,3 @@
double constexpr kMergeDistMeters = 15.0;
// For turns that are not EnterRoundAbout/ExitRoundAbout exitNum is always equal to zero.
// If a turn is EnterRoundAbout exitNum is a number of turns between two junctions:
AntonM030481 (Migrated from github.com) commented 2022-06-27 13:34:49 +00:00

So we need to fix all cases like?
4e75b5bea7/routing/async_router.cpp (L13)

So we need to fix all cases like? https://github.com/organicmaps/organicmaps/blob/4e75b5bea7312cc718a83f7f7fab4b3430f27496/routing/async_router.cpp#L13
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 15:36:14 +00:00
@ -467,21 +467,25 @@ void DrawWidget::SubmitFakeLocationPoint(m2::PointD const & pt)
if (m_framework.GetRoutingManager().IsRoutingActive())
{
// Immidiate update of of position in Route. m_framework.OnLocationUpdate is too late.
AntonM030481 (Migrated from github.com) commented 2022-06-27 15:36:13 +00:00

Will leave extended comment here.

Will leave extended comment here.
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 15:58:12 +00:00
@ -117,3 +118,4 @@
m_poly.GetDistFromCurPointToRoutePointMeters() / curSegSpeedMPerS);
}
void Route::GetCurrentSpeedLimit(SpeedInUnits & speedLimit) const
AntonM030481 (Migrated from github.com) commented 2022-06-27 15:58:11 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 16:03:51 +00:00
@ -216,3 +237,4 @@
}
}
UNIT_TEST(TestIsLaneWayConformedTurnDirection)
AntonM030481 (Migrated from github.com) commented 2022-06-27 16:03:51 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 16:04:01 +00:00
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 16:04:09 +00:00
@ -261,1 +281,3 @@
{2, CarDirection::ReachedYourDestination}};
vector<turns::TurnItem> turns =
{{1, CarDirection::GoStraight},
{2, CarDirection::TurnLeft},
AntonM030481 (Migrated from github.com) commented 2022-06-27 16:04:09 +00:00

OK

OK
AntonM030481 (Migrated from github.com) reviewed 2022-06-27 16:09:16 +00:00
@ -0,0 +10,4 @@
m_session->SetOnNewTurnCallback([this]() { ++m_onNewTurnCallbackCounter; });
}
void RouteSegmentsFrom(std::vector<Segment> const & segments, std::vector<m2::PointD> const & path,
AntonM030481 (Migrated from github.com) commented 2022-06-27 16:09:16 +00:00

I would prefer creation of test routeSegments in one line.

I would prefer creation of test routeSegments in one line.
vng (Migrated from github.com) reviewed 2022-06-27 17:35:52 +00:00
@ -35,4 +29,3 @@
double constexpr kMergeDistMeters = 15.0;
// For turns that are not EnterRoundAbout/ExitRoundAbout exitNum is always equal to zero.
// If a turn is EnterRoundAbout exitNum is a number of turns between two junctions:
vng (Migrated from github.com) commented 2022-06-27 17:35:52 +00:00

Well, no need to check all modules now, but fix least those you are modifying now. Put using under 'routing' namespace.

Well, no need to check all modules now, but fix least those you are modifying now. Put using under 'routing' namespace.
This repo is archived. You cannot comment on pull requests.
No reviewers
No labels
Accessibility
Accessibility
Address
Address
Android
Android
Android Auto
Android Auto
Android Automotive (AAOS)
Android Automotive (AAOS)
API
API
AppGallery
AppGallery
AppStore
AppStore
Battery and Performance
Battery and Performance
Blocker
Blocker
Bookmarks and Tracks
Bookmarks and Tracks
Borders
Borders
Bug
Bug
Build
Build
CarPlay
CarPlay
Classificator
Classificator
Community
Community
Core
Core
CrashReports
CrashReports
Cycling
Cycling
Desktop
Desktop
DevEx
DevEx
DevOps
DevOps
dev_sandbox
dev_sandbox
Directions
Directions
Documentation
Documentation
Downloader
Downloader
Drape
Drape
Driving
Driving
Duplicate
Duplicate
Editor
Editor
Elevation
Elevation
Enhancement
Enhancement
Epic
Epic
External Map Datasets
External Map Datasets
F-Droid
F-Droid
Fonts
Fonts
Frequently User Reported
Frequently User Reported
Fund
Fund
Generator
Generator
Good first issue
Good first issue
Google Play
Google Play
GPS
GPS
GSoC
GSoC
iCloud
iCloud
Icons
Icons
iOS
iOS
Legal
Legal
Linux Desktop
Linux Desktop
Linux packaging
Linux packaging
Linux Phone
Linux Phone
Mac OS
Mac OS
Map Data
Map Data
Metro
Metro
Navigation
Navigation
Need Feedback
Need Feedback
Night Mode
Night Mode
NLnet 2024-06-281
NLnet 2024-06-281
No Feature Parity
No Feature Parity
Opening Hours
Opening Hours
Outdoors
Outdoors
POI Info
POI Info
Privacy
Privacy
Public Transport
Public Transport
Raw Idea
Raw Idea
Refactoring
Refactoring
Regional
Regional
Regression
Regression
Releases
Releases
RoboTest
RoboTest
Route Planning
Route Planning
Routing
Routing
Ruler
Ruler
Search
Search
Security
Security
Styles
Styles
Tests
Tests
Track Recording
Track Recording
Translations
Translations
TTS
TTS
UI
UI
UX
UX
Walk Navigation
Walk Navigation
Watches
Watches
Web
Web
Wikipedia
Wikipedia
Windows
Windows
Won't fix
Won't fix
World Map
World Map
No milestone
No project
No assignees
1 participant
Due date
The due date is invalid or out of range. Please use the format "yyyy-mm-dd".

No due date set.

Dependencies

No dependencies set.

Reference: organicmaps/organicmaps-tmp#2796
No description provided.