[WIP] Add User Route saving functionality

Enables the creation of user routes to be loaded at any time.
Overloads previous methods that handled the saved route points and
creates a generalised verion capable of using any file path.
New function use these generalised methods to handle user routes.
Adds a test to the map_tests folder (in progress)

Signed-off-by: Fábio Gomes <gabriel.gomes@tecnico.ulisboa.pt>
This commit is contained in:
Fábio Gomes 2024-05-23 12:15:39 +01:00
parent 6e93e5b7a5
commit 150eaa1c6b
4 changed files with 168 additions and 27 deletions

View file

@ -16,6 +16,7 @@ set(SRC
power_manager_tests.cpp
search_api_tests.cpp
transliteration_test.cpp
user_routes_test.cpp
working_time_tests.cpp
)

View file

@ -0,0 +1,42 @@
#include "testing/testing.hpp"
#include "map/routing_manager.hpp"
#include <string>
#include <vector>
namespace user_routes_test
{
using namespace std;
#define RM_CALLBACKS { \
static_cast<RoutingManager::Callbacks::DataSourceGetterFn>(nullptr), \
static_cast<RoutingManager::Callbacks::CountryInfoGetterFn>(nullptr), \
static_cast<RoutingManager::Callbacks::CountryParentNameGetterFn>(nullptr), \
static_cast<RoutingManager::Callbacks::GetStringsBundleFn>(nullptr), \
static_cast<RoutingManager::Callbacks::PowerManagerGetter>(nullptr) \
}
class TestDelegate : public RoutingManager::Delegate
{
void OnRouteFollow(routing::RouterType type) override
{
// Empty
}
void RegisterCountryFilesOnRoute(std::shared_ptr<routing::NumMwmIds> ptr) const override
{
// Empty
}
};
UNIT_TEST(user_routes_test)
{
TestDelegate d = TestDelegate();
TestDelegate & dRef = d;
RoutingManager rManager(RM_CALLBACKS, dRef);
TEST(rManager.getUserRoutes().empty(),("Found User Routes before test start"));
}
}

View file

@ -50,6 +50,8 @@ double const kRouteScaleMultiplier = 1.5;
string const kRoutePointsFile = "route_points.dat";
string const kUserRoutesFolder = "user_routes/";
uint32_t constexpr kInvalidTransactionId = 0;
void FillTurnsDistancesForRendering(vector<RouteSegment> const & segments,
@ -109,7 +111,7 @@ RouteMarkData GetLastPassedPoint(BookmarkManager * bmManager, vector<RouteMarkDa
return data;
}
void SerializeRoutePoint(json_t * node, RouteMarkData const & data)
void SerializeRoutePoint(json_t * node, RouteMarkData const & data, bool const keepReplaceWithMyPositionAfterRestart)
{
ASSERT(node != nullptr, ());
ToJSONObject(*node, "type", static_cast<int>(data.m_pointType));
@ -117,7 +119,8 @@ void SerializeRoutePoint(json_t * node, RouteMarkData const & data)
ToJSONObject(*node, "subtitle", data.m_subTitle);
ToJSONObject(*node, "x", data.m_position.x);
ToJSONObject(*node, "y", data.m_position.y);
ToJSONObject(*node, "replaceWithMyPosition", data.m_replaceWithMyPositionAfterRestart);
ToJSONObject(*node, "replaceWithMyPosition",
keepReplaceWithMyPositionAfterRestart ? data.m_replaceWithMyPositionAfterRestart : false);
}
RouteMarkData DeserializeRoutePoint(json_t * node)
@ -140,18 +143,18 @@ RouteMarkData DeserializeRoutePoint(json_t * node)
return data;
}
string SerializeRoutePoints(vector<RouteMarkData> const & points)
string SerializeRoutePoints(vector<RouteMarkData> const & points, bool const keepReplaceWithMyPositionAfterRestart)
{
ASSERT_GREATER_OR_EQUAL(points.size(), 2, ());
auto pointsNode = base::NewJSONArray();
for (auto const & p : points)
{
auto pointNode = base::NewJSONObject();
SerializeRoutePoint(pointNode.get(), p);
SerializeRoutePoint(pointNode.get(), p, keepReplaceWithMyPositionAfterRestart);
json_array_append_new(pointsNode.get(), pointNode.release());
}
unique_ptr<char, JSONFreeDeleter> buffer(
json_dumps(pointsNode.get(), JSON_COMPACT));
json_dumps(pointsNode.get(), JSON_COMPACT | JSON_ENSURE_ASCII));
return string(buffer.get());
}
@ -1356,31 +1359,56 @@ void RoutingManager::CancelRoutePointsTransaction(uint32_t transactionId)
routePoints.AddRoutePoint(std::move(markData));
}
bool RoutingManager::HasSavedUserRoute(string const fileName) const
{
return HasSavedRoutePoints(kUserRoutesFolder + fileName + ".dat");
}
bool RoutingManager::HasSavedRoutePoints() const
{
auto const fileName = GetPlatform().SettingsPathForFile(kRoutePointsFile);
return GetPlatform().IsFileExistsByFullPath(fileName);
return HasSavedRoutePoints(kRoutePointsFile);
}
bool RoutingManager::HasSavedRoutePoints(string const fileName) const
{
auto const filePath = GetPlatform().SettingsPathForFile(fileName);
return GetPlatform().IsFileExistsByFullPath(filePath);
}
void RoutingManager::LoadUserRoutePoints(LoadRouteHandler const & handler, string const fileName)
{
LoadRoutePoints(handler, kUserRoutesFolder + fileName + ".dat", false);
}
void RoutingManager::LoadRoutePoints(LoadRouteHandler const & handler)
{
GetPlatform().RunTask(Platform::Thread::File, [this, handler]()
LoadRoutePoints(handler, kRoutePointsFile, true);
}
void RoutingManager::LoadRoutePoints(LoadRouteHandler const & handler, string const & fileName, bool const deleteAfterLoading)
{
GetPlatform().RunTask(Platform::Thread::File, [this, handler, fileName, deleteAfterLoading]()
{
if (!HasSavedRoutePoints())
if (!HasSavedRoutePoints(fileName))
{
if (handler)
handler(false /* success */);
return;
}
// Delete file after loading.
auto const fileName = GetPlatform().SettingsPathForFile(kRoutePointsFile);
SCOPE_GUARD(routePointsFileGuard, bind(&FileWriter::DeleteFileX, cref(fileName)));
auto const filePath = GetPlatform().SettingsPathForFile(fileName);
// define a lambda to delete the file based on deleteAfterLoading
auto conditionalDelete = [&]()
{
if (deleteAfterLoading) FileWriter::DeleteFileX(filePath);
};
SCOPE_GUARD(routePointsFileGuard, conditionalDelete);
string data;
try
{
ReaderPtr<Reader>(GetPlatform().GetReader(fileName)).ReadAsString(data);
ReaderPtr<Reader>(GetPlatform().GetReader(filePath)).ReadAsString(data);
}
catch (RootException const & ex)
{
@ -1437,22 +1465,30 @@ void RoutingManager::LoadRoutePoints(LoadRouteHandler const & handler)
});
}
void RoutingManager::SaveRoutePoints()
void RoutingManager::SaveUserRoutePoints(string const fileName) {
SaveRoutePoints(kUserRoutesFolder + fileName + ".dat", false);
}
void RoutingManager::SaveRoutePoints() {
SaveRoutePoints(kRoutePointsFile, true);
}
void RoutingManager::SaveRoutePoints(string const fileName, bool const keepReplaceWithMyPositionAfterRestart)
{
auto points = GetRoutePointsToSave();
if (points.empty())
{
DeleteSavedRoutePoints();
DeleteSavedRoutePoints(fileName);
return;
}
GetPlatform().RunTask(Platform::Thread::File, [points = std::move(points)]()
GetPlatform().RunTask(Platform::Thread::File, [points = std::move(points), fileName, keepReplaceWithMyPositionAfterRestart]()
{
try
{
auto const fileName = GetPlatform().SettingsPathForFile(kRoutePointsFile);
FileWriter writer(fileName);
string const pointsData = SerializeRoutePoints(points);
auto const filePath = GetPlatform().SettingsPathForFile(fileName);
FileWriter writer(filePath);
string const pointsData = SerializeRoutePoints(points, keepReplaceWithMyPositionAfterRestart);
writer.Write(pointsData.c_str(), pointsData.length());
}
catch (RootException const & ex)
@ -1501,18 +1537,60 @@ void RoutingManager::OnExtrapolatedLocationUpdate(location::GpsInfo const & info
routeMatchingInfo);
}
void RoutingManager::DeleteUserRoute(string const fileName)
{
DeleteSavedRoutePoints(kUserRoutesFolder + fileName + ".dat");
}
void RoutingManager::DeleteSavedRoutePoints()
{
if (!HasSavedRoutePoints())
DeleteSavedRoutePoints(kRoutePointsFile);
}
void RoutingManager::DeleteSavedRoutePoints(string const fileName)
{
if (!HasSavedRoutePoints(fileName))
return;
GetPlatform().RunTask(Platform::Thread::File, []()
GetPlatform().RunTask(Platform::Thread::File, [fileName = fileName]()
{
auto const fileName = GetPlatform().SettingsPathForFile(kRoutePointsFile);
FileWriter::DeleteFileX(fileName);
auto const filePath = GetPlatform().SettingsPathForFile(fileName);
FileWriter::DeleteFileX(filePath);
});
}
void RoutingManager::RenameUserRoute(string const oldFileName, string const newFileName)
{
if (!HasSavedRoutePoints(kUserRoutesFolder + oldFileName + ".dat"))
return;
if (HasSavedRoutePoints(kUserRoutesFolder + newFileName + ".dat"))
return;
GetPlatform().RunTask(Platform::Thread::File, [oldFileName = kUserRoutesFolder + oldFileName + ".dat" ,
newFileName = kUserRoutesFolder + newFileName + ".dat"]()
{
auto const oldPath = GetPlatform().SettingsPathForFile(oldFileName);
auto const newPath = GetPlatform().SettingsPathForFile(newFileName);
base::RenameFileX(oldPath, newPath);
});
}
vector<string> RoutingManager::getUserRoutes()
{
vector<string> routeNames;
GetPlatform().GetFilesByExt(GetPlatform().SettingsPathForFile(kUserRoutesFolder), ".dat", routeNames);
for(auto name : routeNames)
{
size_t idx = name.rfind(".dat");
if (idx == string::npos)
continue;
name.erase(idx, 4); // erase file extension from the string
}
return routeNames/*TODO find out how this return value is able to be interpreted in java*/;
}
void RoutingManager::UpdatePreviewMode()
{
SetSubroutesVisibility(false /* visible */);

View file

@ -304,16 +304,36 @@ public:
void CancelRoutePointsTransaction(uint32_t transactionId);
static uint32_t InvalidRoutePointsTransactionId();
/// \returns true if there are route points saved in file and false otherwise.
/// \returns true if there is a user route points saved with the given name and false otherwise.
bool HasSavedUserRoute(std::string fileName) const;
/// \returns true if there are route points saved in the default file and false otherwise.
bool HasSavedRoutePoints() const;
/// \brief It loads road points from file and delete file after loading.
/// \returns true if there are route points saved in file and false otherwise.
bool HasSavedRoutePoints(std::string fileName) const;
/// The result of the loading will be sent via SafeCallback.
using LoadRouteHandler = platform::SafeCallback<void(bool success)>;
/// \brief It loads road points from file in a user routes folder and keeps file after loading.
void LoadUserRoutePoints(LoadRouteHandler const & handler, std::string fileName);
/// \brief It loads road points from file and delete file after loading.
void LoadRoutePoints(LoadRouteHandler const & handler);
/// \brief It saves route points to file.
/// \brief It loads road points from file and can be set to delete file after loading.
void LoadRoutePoints(LoadRouteHandler const & handler, std::string const& fileName, bool deleteAfterLoading);
/// \brief It saves route points to file in a user routes folder
void SaveUserRoutePoints(std::string fileName);
/// \brief It saves route points to default file.
void SaveRoutePoints();
/// \brief It deletes file with saved route points if it exists.
/// \brief It saves route points to file with the given name.
void SaveRoutePoints(std::string fileName, bool keepReplaceWithMyPositionAfterRestart);
/// \brief It deletes the user route with the user route with the given name if it exists.
void DeleteUserRoute(std::string fileName);
/// \brief It deletes the default file with saved route points if it exists.
void DeleteSavedRoutePoints();
/// \brief It deletes file with saved route points if it exists.
void DeleteSavedRoutePoints(std::string fileName);
/// \brief It renames a user route
void RenameUserRoute(std::string oldFileName, std::string newFileName);
/// \returns names of the files in the user routes folder without the .dat file extension.
std::vector<std::string> getUserRoutes();
void UpdatePreviewMode();
void CancelPreviewMode();