forked from organicmaps/organicmaps
[ugc] HasUGCForPlace method is added
This commit is contained in:
parent
4fa50bb98d
commit
703971ad11
6 changed files with 88 additions and 13 deletions
|
@ -1,7 +1,5 @@
|
|||
#pragma once
|
||||
|
||||
#include "storage/index.hpp"
|
||||
|
||||
#include "geometry/mercator.hpp"
|
||||
#include "geometry/point2d.hpp"
|
||||
#include "geometry/tree4d.hpp"
|
||||
|
|
18
ugc/api.cpp
18
ugc/api.cpp
|
@ -1,5 +1,7 @@
|
|||
#include "ugc/api.hpp"
|
||||
|
||||
#include "indexer/feature_data.hpp"
|
||||
|
||||
#include "base/assert.hpp"
|
||||
|
||||
#include <utility>
|
||||
|
@ -38,6 +40,16 @@ void Api::GetUGCToSend(UGCJsonToSendCallback const & callback)
|
|||
m_thread.Push([callback, this] { GetUGCToSendImpl(callback); });
|
||||
}
|
||||
|
||||
void Api::HasUGCForPlace(feature::TypesHolder const & types, m2::PointD const & point,
|
||||
HasUGCForPlaceCallback const & callback)
|
||||
{
|
||||
m_thread.Push([this, editableTypes = types, point, callback]() mutable
|
||||
{
|
||||
editableTypes.SortBySpec();
|
||||
HasUGCForPlaceImpl(editableTypes.GetBestType(), point, callback);
|
||||
});
|
||||
}
|
||||
|
||||
void Api::SendingCompleted()
|
||||
{
|
||||
m_thread.Push([this] { SendingCompletedImpl(); });
|
||||
|
@ -80,6 +92,12 @@ void Api::GetUGCToSendImpl(UGCJsonToSendCallback const & callback)
|
|||
callback(move(json), m_storage.GetNumberOfUnsynchronized());
|
||||
}
|
||||
|
||||
void Api::HasUGCForPlaceImpl(uint32_t bestType, m2::PointD const & point,
|
||||
HasUGCForPlaceCallback const & callback) const
|
||||
{
|
||||
callback(m_storage.HasUGCForPlace(bestType, point));
|
||||
}
|
||||
|
||||
void Api::SendingCompletedImpl()
|
||||
{
|
||||
m_storage.MarkAllAsSynchronized();
|
||||
|
|
12
ugc/api.hpp
12
ugc/api.hpp
|
@ -6,6 +6,8 @@
|
|||
|
||||
#include "platform/safe_callback.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
|
||||
#include "base/worker_thread.hpp"
|
||||
|
||||
#include <functional>
|
||||
|
@ -13,6 +15,11 @@
|
|||
class DataSource;
|
||||
struct FeatureID;
|
||||
|
||||
namespace feature
|
||||
{
|
||||
class TypesHolder;
|
||||
}
|
||||
|
||||
namespace ugc
|
||||
{
|
||||
class Api
|
||||
|
@ -23,6 +30,7 @@ public:
|
|||
using UGCJsonToSendCallback = std::function<void(std::string && jsonStr, size_t numberOfUnsynchronized)>;
|
||||
using OnResultCallback = platform::SafeCallback<void(Storage::SettingResult const result)>;
|
||||
using NumberOfUnsynchronizedCallback = std::function<void(size_t number)>;
|
||||
using HasUGCForPlaceCallback = std::function<void(bool result)>;
|
||||
|
||||
Api(DataSource const & dataSource, NumberOfUnsynchronizedCallback const & callback);
|
||||
|
||||
|
@ -30,6 +38,8 @@ public:
|
|||
void SetUGCUpdate(FeatureID const & id, UGCUpdate const & ugc,
|
||||
OnResultCallback const & callback = nullptr);
|
||||
void GetUGCToSend(UGCJsonToSendCallback const & callback);
|
||||
void HasUGCForPlace(feature::TypesHolder const & types, m2::PointD const & point,
|
||||
HasUGCForPlaceCallback const & callback);
|
||||
void SendingCompleted();
|
||||
void SaveUGCOnDisk();
|
||||
|
||||
|
@ -39,6 +49,8 @@ private:
|
|||
void GetUGCImpl(FeatureID const & id, UGCCallbackUnsafe const & callback);
|
||||
Storage::SettingResult SetUGCUpdateImpl(FeatureID const & id, UGCUpdate const & ugc);
|
||||
void GetUGCToSendImpl(UGCJsonToSendCallback const & callback);
|
||||
void HasUGCForPlaceImpl(uint32_t bestType, m2::PointD const & point,
|
||||
HasUGCForPlaceCallback const & callback) const;
|
||||
void SendingCompletedImpl();
|
||||
void SaveUGCOnDiskImpl();
|
||||
|
||||
|
|
|
@ -7,6 +7,7 @@
|
|||
|
||||
#include "indexer/classificator.hpp"
|
||||
#include "indexer/feature_algo.hpp"
|
||||
#include "indexer/feature_data.hpp"
|
||||
#include "indexer/feature_decl.hpp"
|
||||
#include "indexer/ftraits.hpp"
|
||||
|
||||
|
@ -155,17 +156,7 @@ UGCUpdate Storage::GetUGCUpdate(FeatureID const & id) const
|
|||
if (m_indexes.empty())
|
||||
return {};
|
||||
|
||||
auto const feature = GetFeature(id);
|
||||
auto const mercator = feature::GetCenter(*feature);
|
||||
feature::TypesHolder th(*feature);
|
||||
th.SortBySpec();
|
||||
auto const & c = classif();
|
||||
auto const type = c.GetIndexForType(th.GetBestType());
|
||||
|
||||
auto const index = find_if(
|
||||
m_indexes.begin(), m_indexes.end(), [type, &mercator](UpdateIndex const & index) -> bool {
|
||||
return type == index.m_type && mercator == index.m_mercator && !index.m_deleted;
|
||||
});
|
||||
auto const index = FindIndex(id);
|
||||
|
||||
if (index == m_indexes.end())
|
||||
return {};
|
||||
|
@ -291,6 +282,30 @@ void Storage::Migrate(string const & indexFilePath)
|
|||
}
|
||||
}
|
||||
|
||||
UpdateIndexes::const_iterator Storage::FindIndex(FeatureID const & id) const
|
||||
{
|
||||
auto const feature = GetFeature(id);
|
||||
auto const mercator = feature::GetCenter(*feature);
|
||||
feature::TypesHolder th(*feature);
|
||||
th.SortBySpec();
|
||||
|
||||
return FindIndex(th.GetBestType(), mercator);
|
||||
}
|
||||
|
||||
UpdateIndexes::const_iterator Storage::FindIndex(uint32_t bestType,
|
||||
m2::PointD const & point) const
|
||||
{
|
||||
auto const & c = classif();
|
||||
auto const typeIndex = c.GetIndexForType(bestType);
|
||||
|
||||
return find_if(
|
||||
m_indexes.begin(), m_indexes.end(), [typeIndex, &point](UpdateIndex const & index) -> bool {
|
||||
// We are use 1e-5 eps because of points in mwm have this accuracy.
|
||||
return typeIndex == index.m_type && point.EqualDxDy(index.m_mercator, 1e-5 /* eps */) &&
|
||||
!index.m_deleted;
|
||||
});
|
||||
}
|
||||
|
||||
bool Storage::SaveIndex(std::string const & pathToTargetFile /* = "" */) const
|
||||
{
|
||||
if (m_indexes.empty())
|
||||
|
@ -442,6 +457,11 @@ size_t Storage::GetNumberOfUnsynchronized() const
|
|||
return numberOfUnsynchronized;
|
||||
}
|
||||
|
||||
bool Storage::HasUGCForPlace(uint32_t bestType, m2::PointD const & point) const
|
||||
{
|
||||
return FindIndex(bestType, point) != m_indexes.end();
|
||||
}
|
||||
|
||||
void Storage::MarkAllAsSynchronized()
|
||||
{
|
||||
if (m_indexes.empty())
|
||||
|
|
|
@ -2,6 +2,8 @@
|
|||
|
||||
#include "ugc/types.hpp"
|
||||
|
||||
#include "geometry/point2d.hpp"
|
||||
|
||||
#include "base/thread_checker.hpp"
|
||||
|
||||
#include <memory>
|
||||
|
@ -11,6 +13,11 @@ class DataSource;
|
|||
class FeatureType;
|
||||
struct FeatureID;
|
||||
|
||||
namespace feature
|
||||
{
|
||||
class TypesHolder;
|
||||
}
|
||||
|
||||
namespace ugc
|
||||
{
|
||||
class Storage
|
||||
|
@ -34,6 +41,7 @@ public:
|
|||
void Defragmentation();
|
||||
void Load();
|
||||
size_t GetNumberOfUnsynchronized() const;
|
||||
bool HasUGCForPlace(uint32_t bestType, m2::PointD const & point) const;
|
||||
|
||||
/// Testing
|
||||
UpdateIndexes & GetIndexesForTesting() { return m_indexes; }
|
||||
|
@ -46,6 +54,8 @@ private:
|
|||
uint64_t UGCSizeAtIndex(size_t const indexPosition) const;
|
||||
std::unique_ptr<FeatureType> GetFeature(FeatureID const & id) const;
|
||||
void Migrate(std::string const & indexFilePath);
|
||||
UpdateIndexes::const_iterator FindIndex(FeatureID const & id) const;
|
||||
UpdateIndexes::const_iterator FindIndex(uint32_t bestType, m2::PointD const & point) const;
|
||||
|
||||
DataSource const & m_dataSource;
|
||||
UpdateIndexes m_indexes;
|
||||
|
|
|
@ -604,3 +604,20 @@ UNIT_TEST(UGC_TooOldDataVersionsForMigration)
|
|||
TEST(DeleteUGCFile(IndexVersion::V0), ());
|
||||
}
|
||||
|
||||
UNIT_CLASS_TEST(StorageTest, UGC_HasUGCForPlace)
|
||||
{
|
||||
auto & builder = MwmBuilder::Builder();
|
||||
m2::PointD const point(1.0, 1.0);
|
||||
builder.Build({TestCafe(point)});
|
||||
auto const id = builder.FeatureIdForCafeAtPoint(point);
|
||||
auto const original = MakeTestUGCUpdate(Time(chrono::hours(24 * 300)));
|
||||
Storage storage(builder.GetDataSource());
|
||||
storage.Load();
|
||||
TEST_EQUAL(storage.SetUGCUpdate(id, original), Storage::SettingResult::Success, ());
|
||||
auto const actual = storage.GetUGCUpdate(id);
|
||||
TEST_EQUAL(original, actual, ());
|
||||
|
||||
auto const & c = classif();
|
||||
auto const cafeType = c.GetTypeByReadableObjectName("amenity-cafe");
|
||||
TEST(storage.HasUGCForPlace(cafeType, point), ());
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue