[search_marks] dummies for booking improvements

This commit is contained in:
Arsentiy Milchakov 2020-07-17 16:39:49 +03:00 committed by Anatoliy V. Tomilov
parent 45e26661fa
commit 6739a1a74d
6 changed files with 155 additions and 51 deletions

View file

@ -16,27 +16,30 @@ namespace booking
{
namespace
{
void Sort(search::Results && results, std::vector<Extras> && extras,
std::vector<FeatureID> & sortedFeatures, std::vector<booking::Extras> & sortedExtras)
void SortTransform(search::Results && results, std::vector<Extras> && extras,
std::vector<FeatureID> & sortedFeatures, std::vector<std::string> & sortedPrices)
{
if (!extras.empty())
{
CHECK_EQUAL(results.GetCount(), extras.size(), ());
std::vector<std::pair<FeatureID, booking::Extras>> featuresWithExtras;
featuresWithExtras.reserve(results.GetCount());
std::vector<std::pair<FeatureID, std::string>> featuresWithPrices;
featuresWithPrices.reserve(results.GetCount());
for (size_t i = 0; i < results.GetCount(); ++i)
featuresWithExtras.emplace_back(std::move(results[i].GetFeatureID()), std::move(extras[i]));
{
auto pricesFormatted = FormatPrice(extras[i].m_price, extras[i].m_currency);
featuresWithPrices.emplace_back(std::move(results[i].GetFeatureID()), std::move(pricesFormatted));
}
std::sort(featuresWithExtras.begin(), featuresWithExtras.end(),
base::LessBy(&std::pair<FeatureID, booking::Extras>::first));
std::sort(featuresWithPrices.begin(), featuresWithPrices.end(),
base::LessBy(&std::pair<FeatureID, std::string>::first));
sortedFeatures.reserve(featuresWithExtras.size());
sortedExtras.reserve(featuresWithExtras.size());
for (auto & item : featuresWithExtras)
sortedFeatures.reserve(featuresWithPrices.size());
sortedPrices.reserve(featuresWithPrices.size());
for (auto & item : featuresWithPrices)
{
sortedFeatures.emplace_back(std::move(item.first));
sortedExtras.emplace_back(std::move(item.second));
sortedPrices.emplace_back(std::move(item.second));
}
}
else
@ -74,16 +77,18 @@ filter::TasksInternal MakeInternalTasks(filter::Tasks const & filterTasks,
if (results.GetCount() == 0)
return;
if (!inViewport)
extras.clear();
std::vector<FeatureID> sortedFeatures;
std::vector<booking::Extras> sortedExtras;
Sort(std::move(results), std::move(extras), sortedFeatures, sortedExtras);
std::vector<std::string> sortedPrices;
SortTransform(std::move(results), std::move(extras), sortedFeatures, sortedPrices);
if (inViewport)
{
// TODO(a): add price formatting for array.
// auto const pricesFormatted = FormatPrices(sortedExtras);
GetPlatform().RunTask(Platform::Thread::Gui, [&searchMarks, type, sortedFeatures]()
GetPlatform().RunTask(Platform::Thread::Gui, [&searchMarks, type, sortedFeatures,
sortedPrices = std::move(sortedPrices)]() mutable
{
switch (type)
{
@ -92,13 +97,10 @@ filter::TasksInternal MakeInternalTasks(filter::Tasks const & filterTasks,
break;
case Type::Availability:
searchMarks.SetPreparingState(sortedFeatures, false /* isPreparing */);
searchMarks.SetPrices(sortedFeatures, std::move(sortedPrices));
break;
}
});
// TODO(a): to add SetPrices method into search marks.
// if (!pricesFormatted.empty())
// m_searchMarks.SetPrices(sortedFeatures, pricesFormatted)
}
cb(apiParams, sortedFeatures);
}
@ -134,10 +136,13 @@ filter::TasksRawInternal MakeInternalTasks(filter::Tasks const & filterTasks,
if (sortedFeatures.empty())
return;
// TODO(a): add price formatting for array.
// auto const pricesFormatted = FormatPrices(extras);
std::vector<std::string> sortedPrices;
sortedPrices.reserve(extras.size());
for (size_t i = 0; i < extras.size(); ++i)
sortedPrices.emplace_back(FormatPrice(extras[i].m_price, extras[i].m_currency));
GetPlatform().RunTask(Platform::Thread::Gui, [&searchMarks, type, sortedFeatures]()
GetPlatform().RunTask(Platform::Thread::Gui, [&searchMarks, type, sortedFeatures,
sortedPrices = std::move(sortedPrices)]() mutable
{
switch (type)
{
@ -146,11 +151,9 @@ filter::TasksRawInternal MakeInternalTasks(filter::Tasks const & filterTasks,
break;
case Type::Availability:
searchMarks.SetPreparingState(sortedFeatures, false /* isPreparing */);
searchMarks.SetPrices(sortedFeatures, std::move(sortedPrices));
break;
}
// TODO(a): to add SetPrices method into search marks.
// if (!pricesFormatted.empty())
// m_searchMarks.SetPrices(sortedFeatures, pricesFormatted)
});
cb(apiParams, sortedFeatures);
}
@ -161,4 +164,10 @@ filter::TasksRawInternal MakeInternalTasks(filter::Tasks const & filterTasks,
return tasksInternal;
}
std::string FormatPrice(double price, std::string const & currency)
{
// TODO(a): add price formatting.
return std::to_string(static_cast<uint32_t>(price)) + " " + currency;
}
} // namespace booking

View file

@ -10,4 +10,5 @@ filter::TasksInternal MakeInternalTasks(filter::Tasks const & filterTasks,
SearchMarks & searchMarks, bool inViewport);
filter::TasksRawInternal MakeInternalTasks(filter::Tasks const & filterTasks,
SearchMarks & searchMarks);
std::string FormatPrice(double price, std::string const & currency);
} // namespace booking

View file

@ -1052,6 +1052,8 @@ void Framework::FillApiMarkInfo(ApiMarkPoint const & api, place_page::Info & inf
void Framework::FillSearchResultInfo(SearchMarkPoint const & smp, place_page::Info & info) const
{
info.SetIsSearchMark(true);
if (smp.GetFeatureID().IsValid())
FillFeatureInfo(smp.GetFeatureID(), info);
else
@ -2423,6 +2425,13 @@ void Framework::ActivateMapSelection(std::optional<place_page::Info> const & inf
else
GetBookmarkManager().OnTrackDeselected();
// TODO(a): to replace dummies with correct values.
if (m_currentPlacePageInfo->GetSponsoredType() == place_page::SponsoredType::Booking &&
info->IsSearchMark())
{
m_searchMarks.OnSelected(m_currentPlacePageInfo->GetID(), false /* "Dummy" */, "Dummy reason");
}
CHECK_NOT_EQUAL(info->GetSelectedObject(), df::SelectionShape::OBJECT_EMPTY, ("Empty selections are impossible."));
if (m_drapeEngine != nullptr)
{
@ -2447,11 +2456,18 @@ void Framework::DeactivateMapSelection(bool notifyUI)
if (notifyUI && m_onPlacePageClose)
m_onPlacePageClose(!somethingWasAlreadySelected);
m_currentPlacePageInfo = {};
if (somethingWasAlreadySelected)
{
if (m_currentPlacePageInfo->GetSponsoredType() == place_page::SponsoredType::Booking &&
m_currentPlacePageInfo->IsSearchMark())
{
m_searchMarks.OnDeselected(m_currentPlacePageInfo->GetID());
}
GetBookmarkManager().OnTrackDeselected();
m_currentPlacePageInfo = {};
if (m_drapeEngine != nullptr)
m_drapeEngine->DeselectObject();
}
@ -3907,11 +3923,10 @@ void Framework::ShowViewportSearchResults(search::Results::ConstIter begin,
if (found && !filterResult.m_extras.empty())
{
// auto const index = std::distance(features.cbegin(), it);
// TODO(a): to implement FormatPrice and SetPrice methods.
// auto const price = FormatPrice(filterResult.m_extras[index].m_price,
// filterResult.m_extras[index].m_currency);
// mark.SetPrice(price);
auto const index = std::distance(features.cbegin(), it);
auto price = booking::FormatPrice(filterResult.m_extras[index].m_price,
filterResult.m_extras[index].m_currency);
mark.SetPrice(std::move(price));
}
}
};

View file

@ -146,6 +146,7 @@ public:
/// Place traits
bool IsFeature() const { return m_featureID.IsValid(); }
bool IsBookmark() const { return m_markGroupId != kml::kInvalidMarkGroupId && m_markId != kml::kInvalidMarkId; }
bool IsSearchMark() const { return m_isSearchMark; }
bool IsTrack() const { return m_trackId != kml::kInvalidTrackId; }
bool IsMyPosition() const { return m_selectedObject == df::SelectionShape::ESelectedObject::OBJECT_MY_POSITION; }
bool IsRoutePoint() const { return m_isRoutePoint; }
@ -210,6 +211,9 @@ public:
void SetTrackId(kml::TrackId trackId) { m_trackId = trackId; };
kml::TrackId GetTrackId() const { return m_trackId; };
/// SearchMark
void SetIsSearchMark(bool isSearchMark) { m_isSearchMark = isSearchMark; };
/// Guide
void SetIsGuide(bool isGuide) { m_isGuide = isGuide; }
@ -359,6 +363,9 @@ private:
/// If not invalid, track is bound to this place page.
kml::TrackId m_trackId = kml::kInvalidTrackId;
/// SearchMark
kml::MarkId m_isSearchMark = false;
/// Guide
bool m_isGuide = false;

View file

@ -492,9 +492,11 @@ void SearchMarkPoint::SetPricing(int pricing)
SetAttributeValue(m_pricing, pricing);
}
void SearchMarkPoint::SetPrice(std::string const & price)
void SearchMarkPoint::SetPrice(std::string && price)
{
SetAttributeValue(m_price, price);
// Dummy.
// TODO: uncomment when drape will be ready.
// SetAttributeValue(m_price, std::move(price));
}
void SearchMarkPoint::SetSale(bool hasSale)
@ -502,6 +504,21 @@ void SearchMarkPoint::SetSale(bool hasSale)
SetAttributeValue(m_hasSale, hasSale);
}
void SearchMarkPoint::SetUsed(bool isUsed)
{
SetAttributeValue(m_isUsed, isUsed);
}
void SearchMarkPoint::SetAvailable(bool isAvailable)
{
SetAttributeValue(m_isAvailable, isAvailable);
}
void SearchMarkPoint::SetReason(std::string && reason)
{
SetAttributeValue(m_reason, std::move(reason));
}
bool SearchMarkPoint::IsBookingSpecialMark() const
{
return (SMT(m_type) == SearchMarkType::Booking) && !m_isPreparing;
@ -612,33 +629,75 @@ std::optional<m2::PointD> SearchMarks::GetSize(std::string const & symbolName)
void SearchMarks::SetPreparingState(std::vector<FeatureID> const & features, bool isPreparing)
{
FilterAndProcessMarks(features, [isPreparing](SearchMarkPoint * mark)
ProcessMarks([&features, isPreparing](SearchMarkPoint * mark)
{
mark->SetPreparing(isPreparing);
ASSERT(std::is_sorted(features.begin(), features.end()), ());
if (std::binary_search(features.cbegin(), features.cend(), mark->GetFeatureID()))
mark->SetPreparing(isPreparing);
});
}
void SearchMarks::SetSales(std::vector<FeatureID> const & features, bool hasSale)
{
FilterAndProcessMarks(features, [hasSale](SearchMarkPoint * mark)
ProcessMarks([&features, hasSale](SearchMarkPoint * mark)
{
mark->SetSale(hasSale);
ASSERT(std::is_sorted(features.begin(), features.end()), ());
if (std::binary_search(features.cbegin(), features.cend(), mark->GetFeatureID()))
mark->SetSale(hasSale);
});
}
void SearchMarks::FilterAndProcessMarks(std::vector<FeatureID> const & features,
std::function<void(SearchMarkPoint *)> && processor)
void SearchMarks::SetPrices(std::vector<FeatureID> const & features, std::vector<std::string> && prices)
{
ProcessMarks([&features, &prices](SearchMarkPoint * mark)
{
ASSERT(std::is_sorted(features.begin(), features.end()), ());
auto const it = std::lower_bound(features.cbegin(), features.cend(), mark->GetFeatureID());
if (it == features.cend())
return;
auto const index = std::distance(features.cbegin(), it);
ASSERT_LESS(index, prices.size(), ());
mark->SetPrice(std::move(prices[index]));
});
}
void SearchMarks::OnSelected(FeatureID const &featureId, bool isAvailable, std::string && reason)
{
ProcessMarks([&featureId, isAvailable, &reason](SearchMarkPoint * mark)
{
if (featureId != mark->GetFeatureID())
return;
mark->SetUsed(true);
mark->SetAvailable(isAvailable);
mark->SetReason(std::move(reason));
});
}
void SearchMarks::OnDeselected(FeatureID const & featureId)
{
ProcessMarks([&featureId](SearchMarkPoint * mark)
{
if (featureId != mark->GetFeatureID())
return;
mark->SetAvailable(true);
mark->SetReason({});
});
}
void SearchMarks::ProcessMarks(std::function<void(SearchMarkPoint *)> && processor)
{
if (m_bmManager == nullptr || processor == nullptr)
return;
ASSERT(std::is_sorted(features.begin(), features.end()), ());
auto editSession = m_bmManager->GetEditSession();
for (auto markId : m_bmManager->GetUserMarkIds(UserMark::Type::SEARCH))
{
auto * mark = editSession.GetMarkForEdit<SearchMarkPoint>(markId);
if (std::binary_search(features.begin(), features.end(), mark->GetFeatureID()))
processor(mark);
processor(mark);
}
}

View file

@ -14,6 +14,7 @@
#include <map>
#include <optional>
#include <string>
#include <utility>
#include <vector>
class BookmarkManager;
@ -46,17 +47,21 @@ public:
void SetPreparing(bool isPreparing);
void SetRating(float rating);
void SetPricing(int pricing);
void SetPrice(std::string const & price);
void SetPrice(std::string && price);
void SetSale(bool hasSale);
void SetUsed(bool isUsed);
void SetAvailable(bool isAvailable);
void SetReason(std::string && reason);
protected:
template<typename T> void SetAttributeValue(T & dst, T const & src)
template <typename T, typename U>
void SetAttributeValue(T & dst, U && src)
{
if (dst == src)
return;
SetDirty();
dst = src;
dst = std::forward<U>(src);
}
bool IsBookingSpecialMark() const;
@ -76,6 +81,9 @@ protected:
bool m_hasSale = false;
dp::TitleDecl m_titleDecl;
dp::TitleDecl m_ugcTitleDecl;
bool m_isUsed = false;
bool m_isAvailable = true;
std::string m_reason;
};
class SearchMarks
@ -94,12 +102,17 @@ public:
// NOTE: Vector of features must be sorted.
void SetSales(std::vector<FeatureID> const & features, bool hasSale);
// NOTE: Vector of features must be sorted.
void SetPrices(std::vector<FeatureID> const & features, std::vector<std::string> && prices);
void OnSelected(FeatureID const & featureId, bool isAvailable, std::string && reason);
void OnDeselected(FeatureID const & featureId);
static bool HaveSizes() { return !m_searchMarksSizes.empty(); };
static std::optional<m2::PointD> GetSize(std::string const & symbolName);
private:
void FilterAndProcessMarks(std::vector<FeatureID> const & features,
std::function<void(SearchMarkPoint *)> && processor);
void ProcessMarks(std::function<void(SearchMarkPoint *)> && processor);
BookmarkManager * m_bmManager;
df::DrapeEngineSafePtr m_drapeEngine;