[drape] [booking] Add special badge (so called "chip") expanding for price text near the hotel's rating symbol MAPSME-14322

This commit is contained in:
Anatoliy Tomilov 2020-07-28 20:19:42 +05:00 committed by Tatiana Yan
parent e4c83b0540
commit e7aa4e0c0b
9 changed files with 260 additions and 136 deletions

View file

@ -935,7 +935,7 @@ drape_ptr<UserMarkRenderParams> DrapeEngine::GenerateMarkRenderInfo(UserPointMar
renderInfo->m_pixelOffset = mark->GetPixelOffset();
renderInfo->m_titleDecl = mark->GetTitleDecl();
renderInfo->m_symbolNames = mark->GetSymbolNames();
renderInfo->m_badgeNames = mark->GetBadgeNames();
renderInfo->m_badgeInfo = mark->GetBadgeInfo();
renderInfo->m_coloredSymbols = mark->GetColoredSymbols();
renderInfo->m_symbolSizes = mark->GetSymbolSizes();
renderInfo->m_symbolOffsets = mark->GetSymbolOffsets();

View file

@ -1,4 +1,5 @@
#include "drape_frontend/poi_symbol_shape.hpp"
#include "drape_frontend/color_constants.hpp"
#include "shaders/programs.hpp"
@ -9,6 +10,7 @@
#include "drape/utils/vertex_decl.hpp"
#include <utility>
#include <vector>
namespace
{
@ -53,21 +55,44 @@ void Batch<SV>(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Batcher> batche
dp::TextureManager::SymbolRegion const & symbolRegion,
dp::TextureManager::ColorRegion const & colorRegion)
{
m2::PointF const pixelSize = symbolRegion.GetPixelSize();
m2::PointF const halfSize(pixelSize.x * 0.5f, pixelSize.y * 0.5f);
m2::PointF const symbolRegionSize = symbolRegion.GetPixelSize();
m2::PointF pixelSize = symbolRegionSize;
if (pixelSize.x < params.m_pixelWidth)
pixelSize.x = params.m_pixelWidth;
m2::PointF const halfSize = pixelSize * 0.5f;
m2::RectF const & texRect = symbolRegion.GetTexRect();
SV vertexes[] =
std::vector<SV> vertexes;
vertexes.reserve(params.m_pixelWidth ? 8 : 4);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.maxY()));
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.minY()));
if (symbolRegionSize.x < params.m_pixelWidth)
{
SV{ position, ShiftNormal(glsl::vec2(-halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.maxY()) },
SV{ position, ShiftNormal(glsl::vec2(-halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.minY()) },
SV{ position, ShiftNormal(glsl::vec2(halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.maxY()) },
SV{ position, ShiftNormal(glsl::vec2(halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.minY()) },
};
float const stretchHalfWidth = (params.m_pixelWidth - symbolRegionSize.x) * 0.5f;
float const midTexU = (texRect.minX() + texRect.maxX()) * 0.5f;
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-stretchHalfWidth, halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.maxY()));
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-stretchHalfWidth, -halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.minY()));
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(stretchHalfWidth, halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.maxY()));
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(stretchHalfWidth, -halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.minY()));
}
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.maxY()));
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.minY()));
auto state = df::CreateRenderState(gpu::Program::Texturing, params.m_depthLayer);
state.SetProgram3d(gpu::Program::TexturingBillboard);
@ -76,8 +101,8 @@ void Batch<SV>(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Batcher> batche
state.SetTextureFilter(dp::TextureFilter::Nearest);
state.SetTextureIndex(symbolRegion.GetTextureIndex());
dp::AttributeProvider provider(1 /* streamCount */, ARRAY_SIZE(vertexes));
provider.InitStream(0 /* streamIndex */, SV::GetBindingInfo(), make_ref(vertexes));
dp::AttributeProvider provider(1 /* streamCount */, vertexes.size());
provider.InitStream(0 /* streamIndex */, SV::GetBindingInfo(), make_ref(vertexes.data()));
batcher->InsertTriangleStrip(context, state, make_ref(&provider), move(handle));
}
@ -88,22 +113,45 @@ void Batch<MV>(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Batcher> batche
dp::TextureManager::SymbolRegion const & symbolRegion,
dp::TextureManager::ColorRegion const & colorRegion)
{
m2::PointF const pixelSize = symbolRegion.GetPixelSize();
m2::PointF const halfSize(pixelSize.x * 0.5f, pixelSize.y * 0.5f);
m2::PointF const symbolRegionSize = symbolRegion.GetPixelSize();
m2::PointF pixelSize = symbolRegionSize;
if (pixelSize.x < params.m_pixelWidth)
pixelSize.x = params.m_pixelWidth;
m2::PointF const halfSize = pixelSize * 0.5f;
m2::RectF const & texRect = symbolRegion.GetTexRect();
glsl::vec2 const maskColorCoords = glsl::ToVec2(colorRegion.GetTexRect().Center());
MV vertexes[] =
std::vector<MV> vertexes;
vertexes.reserve(params.m_pixelWidth ? 8 : 4);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.maxY()), maskColorCoords);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.minY()), maskColorCoords);
if (symbolRegionSize.x < params.m_pixelWidth)
{
MV{ position, ShiftNormal(glsl::vec2(-halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.maxY()), maskColorCoords },
MV{ position, ShiftNormal(glsl::vec2(-halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.minX(), texRect.minY()), maskColorCoords },
MV{ position, ShiftNormal(glsl::vec2(halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.maxY()), maskColorCoords },
MV{ position, ShiftNormal(glsl::vec2(halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.minY()), maskColorCoords },
};
float const stretchHalfWidth = (params.m_pixelWidth - symbolRegionSize.x) * 0.5f;
float const midTexU = (texRect.minX() + texRect.maxX()) * 0.5f;
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-stretchHalfWidth, halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.maxY()), maskColorCoords);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(-stretchHalfWidth, -halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.minY()), maskColorCoords);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(stretchHalfWidth, halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.maxY()), maskColorCoords);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(stretchHalfWidth, -halfSize.y), params, pixelSize),
glsl::vec2(midTexU, texRect.minY()), maskColorCoords);
}
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(halfSize.x, halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.maxY()), maskColorCoords);
vertexes.emplace_back(position,
ShiftNormal(glsl::vec2(halfSize.x, -halfSize.y), params, pixelSize),
glsl::vec2(texRect.maxX(), texRect.minY()), maskColorCoords);
auto state = df::CreateRenderState(gpu::Program::MaskedTexturing, params.m_depthLayer);
state.SetProgram3d(gpu::Program::MaskedTexturingBillboard);
@ -113,8 +161,8 @@ void Batch<MV>(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Batcher> batche
state.SetTextureFilter(dp::TextureFilter::Nearest);
state.SetTextureIndex(symbolRegion.GetTextureIndex());
dp::AttributeProvider provider(1 /* streamCount */, ARRAY_SIZE(vertexes));
provider.InitStream(0 /* streamIndex */, MV::GetBindingInfo(), make_ref(vertexes));
dp::AttributeProvider provider(1 /* streamCount */, vertexes.size());
provider.InitStream(0 /* streamIndex */, MV::GetBindingInfo(), make_ref(vertexes.data()));
batcher->InsertTriangleStrip(context, state, make_ref(&provider), move(handle));
}
} // namespace

View file

@ -7,6 +7,7 @@
#include "drape/stipple_pen_resource.hpp"
#include "indexer/feature_decl.hpp"
#include "geometry/point2d.hpp"
#include <cstdint>
@ -58,6 +59,11 @@ struct PoiSymbolViewParams : CommonOverlayViewParams
bool m_obsoleteInEditor = false;
dp::Anchor m_anchor = dp::Center;
m2::PointF m_offset = m2::PointF(0.0f, 0.0f);
// If greater then actual width of the symbol then symbol splitted at horz center and
// extended horizontally using texture coordinates taken at split line.
// If less or equal, then ignored.
float m_pixelWidth = 0.0f;
};
struct AreaViewParams : CommonViewParams

View file

@ -66,6 +66,19 @@ void AlignVertical(float halfHeight, dp::Anchor anchor, glsl::vec2 & up, glsl::v
dp::Bottom, up, down);
}
TextLayout MakePrimaryTextLayout(dp::TitleDecl const & titleDecl,
ref_ptr<dp::TextureManager> textures)
{
dp::FontDecl const & fontDecl = titleDecl.m_primaryTextFont;
auto const vs = static_cast<float>(df::VisualParams::Instance().GetVisualScale());
bool const isSdf = fontDecl.m_outlineColor != dp::Color::Transparent() ||
df::VisualParams::Instance().IsSdfPrefered();
TextLayout textLayout;
textLayout.Init(strings::MakeUniString(titleDecl.m_primaryText), fontDecl.m_size * vs, isSdf,
textures);
return textLayout;
}
struct UserPointVertex : public gpu::BaseVertex
{
using TTexCoord = glsl::vec4;
@ -135,13 +148,7 @@ void GenerateColoredSymbolShapes(ref_ptr<dp::GraphicsContext> context, ref_ptr<d
if (isTextBg)
{
auto const & titleDecl = renderInfo.m_titleDecl->at(0);
dp::FontDecl const & fontDecl = titleDecl.m_primaryTextFont;
auto isSdf = df::VisualParams::Instance().IsSdfPrefered();
isSdf = fontDecl.m_outlineColor != dp::Color::Transparent() ? true : isSdf;
auto const vs = static_cast<float>(df::VisualParams::Instance().GetVisualScale());
TextLayout textLayout;
textLayout.Init(strings::MakeUniString(titleDecl.m_primaryText), fontDecl.m_size * vs, isSdf, textures);
auto textLayout = MakePrimaryTextLayout(titleDecl, textures);
sizeInc.x = textLayout.GetPixelLength();
sizeInc.y = textLayout.GetPixelHeight();
@ -222,6 +229,29 @@ void GeneratePoiSymbolShape(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Te
m2::PointF const & symbolOffset, dp::Batcher & batcher)
{
PoiSymbolViewParams params(renderInfo.m_featureId);
params.m_offset = symbolOffset;
if (renderInfo.m_badgeInfo->m_badgeTitleIndex)
{
size_t const badgeTitleIndex = *renderInfo.m_badgeInfo->m_badgeTitleIndex;
CHECK_LESS(badgeTitleIndex, renderInfo.m_titleDecl->size(), ());
dp::TitleDecl const & titleDecl = (*renderInfo.m_titleDecl)[badgeTitleIndex];
TextLayout textLayout = MakePrimaryTextLayout(titleDecl, textures);
float const textWidth = textLayout.GetPixelLength();
dp::TextureManager::SymbolRegion region;
textures->GetSymbolRegion(symbolName, region);
float const pixelHalfWidth = 0.5f * region.GetPixelSize().x;
float constexpr kBadgeSpecialMarginsAdjustmentMultplier = 4.0f;
float const badgeSpecialMarginsAdjustment =
kBadgeSpecialMarginsAdjustmentMultplier * titleDecl.m_primaryOffset.x;
params.m_pixelWidth = 3.0f * pixelHalfWidth + textWidth + badgeSpecialMarginsAdjustment;
params.m_offset.x += 0.5f * (pixelHalfWidth + textWidth + badgeSpecialMarginsAdjustment);
}
params.m_tileCenter = tileCenter;
params.m_depthTestEnabled = renderInfo.m_depthTestEnabled;
params.m_depth = renderInfo.m_depth;
@ -231,7 +261,6 @@ void GeneratePoiSymbolShape(ref_ptr<dp::GraphicsContext> context, ref_ptr<dp::Te
params.m_specialPriority = renderInfo.m_priority;
params.m_symbolName = symbolName;
params.m_anchor = renderInfo.m_anchor;
params.m_offset = symbolOffset;
bool const hasColoredOverlay = renderInfo.m_coloredSymbols != nullptr && renderInfo.m_coloredSymbols->m_needOverlay;
params.m_startOverlayRank = hasColoredOverlay ? dp::OverlayRank1 : dp::OverlayRank0;
@ -482,11 +511,12 @@ void CacheUserMarks(ref_ptr<dp::GraphicsContext> context, TileKey const & tileKe
if (renderInfo.m_titleDecl != nullptr)
GenerateTextShapes(context, textures, renderInfo, tileKey, tileCenter, symbolOffset, symbolSize, batcher);
if (renderInfo.m_badgeNames != nullptr)
if (renderInfo.m_badgeInfo != nullptr)
{
ASSERT(!renderInfo.m_symbolIsPOI || renderInfo.m_symbolNames == nullptr,
("Multiple POI shapes in an usermark are not supported yet"));
auto const badgeName = GetSymbolNameForZoomLevel(make_ref(renderInfo.m_badgeNames), tileKey);
auto const badgeName =
GetSymbolNameForZoomLevel(make_ref(&renderInfo.m_badgeInfo->m_zoomInfo), tileKey);
if (!badgeName.empty())
{
// TODO: Badges use symbol offset. Refactor and create own "offset"-method for badges.

View file

@ -26,7 +26,7 @@ struct UserMarkRenderParams
drape_ptr<UserPointMark::TitlesInfo> m_titleDecl;
drape_ptr<UserPointMark::SymbolSizes> m_symbolSizes;
drape_ptr<UserPointMark::SymbolOffsets> m_symbolOffsets;
drape_ptr<UserPointMark::SymbolNameZoomInfo> m_badgeNames;
drape_ptr<UserPointMark::BageInfo> m_badgeInfo;
df::ColorConstant m_color;
bool m_symbolIsPOI = false;
bool m_hasTitlePriority = false;

View file

@ -38,6 +38,10 @@ struct IDCollections
class UserPointMark
{
public:
using SymbolNameZoomInfo = std::map<int, std::string>;
using TitlesInfo = std::vector<dp::TitleDecl>;
using SymbolSizes = std::vector<m2::PointF>;
using SymbolOffsets = std::vector<m2::PointF>;
struct ColoredSymbolZoomInfo
{
std::map<int, df::ColoredSymbolViewParams> m_zoomInfo;
@ -45,10 +49,11 @@ public:
bool m_needOverlay = true;
bool m_addTextSize = false;
};
using SymbolNameZoomInfo = std::map<int, std::string>;
using TitlesInfo = std::vector<dp::TitleDecl>;
using SymbolSizes = std::vector<m2::PointF>;
using SymbolOffsets = std::vector<m2::PointF>;
struct BageInfo
{
SymbolNameZoomInfo m_zoomInfo;
std::optional<size_t> m_badgeTitleIndex;
};
static float constexpr kInvalidDepth = dp::kMinDepth - 1.0;
@ -70,7 +75,7 @@ public:
virtual bool IsVisible() const = 0;
virtual drape_ptr<TitlesInfo> GetTitleDecl() const = 0;
virtual drape_ptr<SymbolNameZoomInfo> GetSymbolNames() const = 0;
virtual drape_ptr<SymbolNameZoomInfo> GetBadgeNames() const = 0;
virtual drape_ptr<BageInfo> GetBadgeInfo() const = 0;
virtual drape_ptr<ColoredSymbolZoomInfo> GetColoredSymbols() const = 0;
virtual drape_ptr<SymbolSizes> GetSymbolSizes() const = 0;
virtual drape_ptr<SymbolOffsets> GetSymbolOffsets() const = 0;

View file

@ -1,20 +1,23 @@
#include "map/search_mark.hpp"
#include "map/bookmark_manager.hpp"
#include "map/place_page_info.hpp"
#include "drape_frontend/drape_engine.hpp"
#include "drape_frontend/visual_params.hpp"
#include "indexer/scales.hpp"
#include "platform/platform.hpp"
#include "base/stl_helpers.hpp"
#include "defines.hpp"
#include <algorithm>
#include <array>
#include <limits>
#include "defines.hpp"
namespace
{
enum class SearchMarkType
@ -101,53 +104,26 @@ std::array<std::string, static_cast<size_t>(SearchMarkType::Count)> const kSymbo
"non-found-search-result", // NotFound.
};
std::string const kSaleBadgeName = "search-badge-booking-sale";
std::string const kPriceChips = "price-chips";
std::string const kPriceChipsSelected = "price-chips-selected";
std::string const kPriceChipsDiscount = "price-chips-discount";
std::string const kPriceChipsSelectedDiscount = "price-chips-selected-discount";
std::string const kUGCRatingBadgeName = "search-badge-rating";
std::string const kRatedDefaultSearchIcon = "rated-default-search-result";
std::string const kLocalAdsRatedDefaultSearchIcon = "local_ads-rated-default-search-result";
float const kRatingThreshold = 6.0f;
float const kMetricThreshold = 0.38f;
int constexpr kUGCBadgeMinZoomLevel = 10;
inline bool HasRating(float rating)
// Offset for price text relative to symbol and adjustment of price chip size
// for better margins of price text into the chip
float constexpr kPriceSpecialOffset = -4.0f;
std::string GetBookingSmallIcon(bool hasLocalAds)
{
return rating > kInvalidRatingValue;
}
float CalculateAggregateMetric(float rating, int pricing)
{
float const p1 = base::Clamp(rating, 0.0f, 10.0f) / 10.0f;
float const p2 = (3 - base::Clamp(pricing, 0, 3)) / 3.0f;
float const s = p1 + p2;
if (fabs(s) < 1e-5)
return 0.0f;
return 2.0f * p1 * p2 / s;
}
std::string GetBookingBadgeName(int pricing)
{
if (pricing == 0)
return {};
return std::string("search-badge-booking-ext-") + strings::to_string(pricing);
}
bool NeedShowBookingBadge(float rating, int pricing)
{
if (pricing == 0)
return false;
auto const metric = CalculateAggregateMetric(rating, pricing);
return metric >= kMetricThreshold;
}
std::string GetBookingSmallIcon(SearchMarkType type, bool hasLocalAds)
{
if (type != SearchMarkType::Booking)
return {};
return hasLocalAds ? "search-adv" : "coloredmark-default-s";
return hasLocalAds ? "local_ads-coloredmark-default-s" : "coloredmark-default-s";
}
std::string GetSymbol(SearchMarkType searchMarkType, bool hasLocalAds, bool isRated)
@ -160,12 +136,12 @@ std::string GetSymbol(SearchMarkType searchMarkType, bool hasLocalAds, bool isRa
if (!hasLocalAds)
return kSymbols[index];
if (searchMarkType == SearchMarkType::Booking)
return "searchbookingadv-default-l";
return "local_ads-" + kSymbols[index];
}
constexpr int GetGoodRatingZoomLevel() { return 1; }
constexpr int GetBadRatingZoomLevel() { return scales::GetUpperScale(); }
bool HasLocalAdsVariant(SearchMarkType searchMarkType)
{
if (searchMarkType == SearchMarkType::NotFound)
@ -274,15 +250,20 @@ SearchMarkType GetSearchMarkType(uint32_t type)
SearchMarkPoint::SearchMarkPoint(m2::PointD const & ptOrg)
: UserMark(ptOrg, UserMark::Type::SEARCH)
{
double const fs = df::VisualParams::Instance().GetFontScale();
m_titleDecl.m_anchor = dp::Center;
m_titleDecl.m_primaryTextFont.m_color = df::GetColorConstant("RatingText");
m_titleDecl.m_primaryTextFont.m_size =
static_cast<float>(12.0 / df::VisualParams::Instance().GetFontScale());
m_titleDecl.m_primaryTextFont.m_size = static_cast<float>(10.0 / fs);
m_ugcTitleDecl.m_anchor = dp::LeftTop;
m_ugcTitleDecl.m_primaryTextFont.m_color = df::GetColorConstant("UGCRatingText");
m_ugcTitleDecl.m_primaryTextFont.m_size =
static_cast<float>(14.0 / df::VisualParams::Instance().GetFontScale());
m_ugcTitleDecl.m_primaryTextFont.m_size = static_cast<float>(10.0 / fs);
m_badgeTitleDecl.m_anchor = dp::Left;
m_badgeTitleDecl.m_primaryTextFont.m_color = df::GetColorConstant("HotelPriceText");
m_badgeTitleDecl.m_primaryTextFont.m_size = static_cast<float>(10.0 / fs);
m_badgeTitleDecl.m_primaryOffset.x = kPriceSpecialOffset;
}
std::string SearchMarkPoint::GetSymbolName() const
@ -291,15 +272,15 @@ std::string SearchMarkPoint::GetSymbolName() const
if (m_type >= static_cast<uint8_t>(SearchMarkType::Count))
{
ASSERT(false, ("Unknown search mark symbol."));
name = GetSymbol(SearchMarkType::Default, false /* hasLocalAds */, HasRating(m_rating));
name = GetSymbol(SearchMarkType::Default, false /* hasLocalAds */, HasRating());
}
else if (m_isPreparing)
{
name = GetPreparingSymbol(SMT(m_type), m_hasLocalAds, HasRating(m_rating));
name = GetPreparingSymbol(SMT(m_type), m_hasLocalAds, HasRating());
}
else
{
name = GetSymbol(SMT(m_type), m_hasLocalAds, HasRating(m_rating));
name = GetSymbol(SMT(m_type), m_hasLocalAds, HasRating());
}
return name;
}
@ -310,39 +291,48 @@ drape_ptr<df::UserPointMark::SymbolNameZoomInfo> SearchMarkPoint::GetSymbolNames
auto symbol = make_unique_dp<SymbolNameZoomInfo>();
if (IsBookingSpecialMark())
{
symbol->insert(std::make_pair(
1 /* zoomLevel */,
m_rating < kRatingThreshold ? GetBookingSmallIcon(SMT(m_type), m_hasLocalAds) : name));
symbol->insert(std::make_pair(17 /* zoomLevel */, name));
if (HasGoodRating())
{
symbol->emplace(GetGoodRatingZoomLevel(), name);
}
else
{
symbol->emplace(GetGoodRatingZoomLevel(), GetBookingSmallIcon(m_hasLocalAds));
symbol->emplace(GetBadRatingZoomLevel(), name);
}
}
else
{
symbol->insert(std::make_pair(1 /* zoomLevel */, name));
symbol->emplace(1 /* zoomLevel */, name);
}
return symbol;
}
drape_ptr<df::UserPointMark::SymbolNameZoomInfo> SearchMarkPoint::GetBadgeNames() const
drape_ptr<df::UserPointMark::BageInfo> SearchMarkPoint::GetBadgeInfo() const
{
if (!HasRating())
return nullptr;
auto const badgeName = GetBadgeName();
if (badgeName.empty())
return nullptr;
if (IsBookingSpecialMark())
if (IsBookingSpecialMark() && (HasPrice() || HasPricing()))
{
auto symbol = make_unique_dp<SymbolNameZoomInfo>();
if (NeedShowBookingBadge(m_rating, m_pricing) && m_rating >= kRatingThreshold)
symbol->insert(std::make_pair(10 /* zoomLevel */, badgeName));
auto badgeInfo = make_unique_dp<BageInfo>();
badgeInfo->m_badgeTitleIndex = 1;
if (HasGoodRating())
badgeInfo->m_zoomInfo.emplace(GetGoodRatingZoomLevel(), badgeName);
else
symbol->insert(std::make_pair(17 /* zoomLevel */, badgeName));
return symbol;
badgeInfo->m_zoomInfo.emplace(GetBadRatingZoomLevel(), badgeName);
return badgeInfo;
}
if (IsUGCMark())
{
auto symbol = make_unique_dp<SymbolNameZoomInfo>();
symbol->insert(std::make_pair(kUGCBadgeMinZoomLevel /* zoomLevel */, badgeName));
return symbol;
auto badgeInfo = make_unique_dp<BageInfo>();
badgeInfo->m_zoomInfo.emplace(kUGCBadgeMinZoomLevel, badgeName);
return badgeInfo;
}
return nullptr;
@ -386,7 +376,7 @@ df::ColorConstant SearchMarkPoint::GetColorConstant() const
if (SMT(m_type) == SearchMarkType::Booking && m_hasLocalAds)
return "RatingLocalAds";
if (!HasRating(m_rating))
if (!HasRating())
return "RatingNone";
if (m_rating < 2.0f)
return "RatingHorrible";
@ -401,13 +391,14 @@ df::ColorConstant SearchMarkPoint::GetColorConstant() const
drape_ptr<df::UserPointMark::TitlesInfo> SearchMarkPoint::GetTitleDecl() const
{
auto const isUGC = IsUGCMark();
if ((!IsBookingSpecialMark() && !isUGC) || !HasRating(m_rating))
if (!HasRating())
return nullptr;
auto titles = make_unique_dp<TitlesInfo>();
if (isUGC)
drape_ptr<TitlesInfo> titles;
if (IsUGCMark())
{
titles = make_unique_dp<TitlesInfo>();
auto titleDecl = m_ugcTitleDecl;
auto const sz = SearchMarks::GetSize(GetSymbolName());
if (!sz)
@ -421,25 +412,40 @@ drape_ptr<df::UserPointMark::TitlesInfo> SearchMarkPoint::GetTitleDecl() const
}
else
{
titles = make_unique_dp<TitlesInfo>();
titles->push_back(m_titleDecl);
if (!m_price.empty())
if (IsBookingSpecialMark())
{
auto priceDecl = m_titleDecl;
priceDecl.m_primaryText = m_price;
priceDecl.m_anchor = dp::Anchor::Left;
titles->push_back(priceDecl);
if (HasPrice())
{
dp::TitleDecl badgeTitleDecl = m_badgeTitleDecl;
badgeTitleDecl.m_primaryText = m_price;
titles->push_back(badgeTitleDecl);
}
else if (HasPricing())
{
dp::TitleDecl badgeTitleDecl = m_badgeTitleDecl;
badgeTitleDecl.m_primaryText.assign(static_cast<size_t>(m_pricing), '$');
titles->push_back(badgeTitleDecl);
}
}
}
return titles;
}
int SearchMarkPoint::GetMinTitleZoom() const
{
if (IsBookingSpecialMark() && m_rating < kRatingThreshold)
return 17;
if (IsBookingSpecialMark() && (HasPrice() || HasPricing()))
{
if (HasGoodRating())
return GetGoodRatingZoomLevel();
else
return GetBadRatingZoomLevel();
}
if (IsUGCMark())
return kUGCBadgeMinZoomLevel;
return 1;
return GetGoodRatingZoomLevel();
}
df::DepthLayer SearchMarkPoint::GetDepthLayer() const
@ -494,9 +500,7 @@ void SearchMarkPoint::SetPricing(int pricing)
void SearchMarkPoint::SetPrice(std::string && price)
{
// Dummy.
// TODO: uncomment when drape will be ready.
// SetAttributeValue(m_price, std::move(price));
SetAttributeValue(m_price, std::move(price));
}
void SearchMarkPoint::SetSale(bool hasSale)
@ -524,19 +528,41 @@ bool SearchMarkPoint::IsBookingSpecialMark() const
return (SMT(m_type) == SearchMarkType::Booking) && !m_isPreparing;
}
bool SearchMarkPoint::IsUGCMark() const
bool SearchMarkPoint::HasGoodRating() const { return m_rating >= kRatingThreshold; }
bool SearchMarkPoint::HasPrice() const { return !m_price.empty(); }
bool SearchMarkPoint::HasPricing() const { return m_pricing > 0; }
bool SearchMarkPoint::HasRating() const { return m_rating > kInvalidRatingValue; }
bool SearchMarkPoint::IsUGCMark() const { return SMT(m_type) != SearchMarkType::Booking; }
bool SearchMarkPoint::IsSelected() const
{
return (SMT(m_type) != SearchMarkType::Booking) && HasRating(m_rating);
return false; // TODO(tomilov): add sane condition
}
bool SearchMarkPoint::HasSale() const { return m_hasSale; }
std::string SearchMarkPoint::GetBadgeName() const
{
if (!HasRating())
return {};
std::string badgeName;
if (IsBookingSpecialMark())
badgeName = m_hasSale ? kSaleBadgeName : GetBookingBadgeName(m_pricing);
{
if (IsSelected())
badgeName = HasSale() ? kPriceChipsSelectedDiscount : kPriceChipsSelected;
else
badgeName = HasSale() ? kPriceChipsDiscount : kPriceChips;
}
else if (IsUGCMark())
{
badgeName = kUGCRatingBadgeName;
}
if (badgeName.empty() || !SearchMarks::GetSize(badgeName))
return {};
@ -570,9 +596,11 @@ void SearchMarks::SetDrapeEngine(ref_ptr<df::DrapeEngine> engine)
}
}
for (int pricing = 1; pricing <= 3; pricing++)
symbols.emplace_back(GetBookingBadgeName(pricing));
symbols.emplace_back(kSaleBadgeName);
symbols.emplace_back(kPriceChips);
symbols.emplace_back(kPriceChipsSelected);
symbols.emplace_back(kPriceChipsDiscount);
symbols.emplace_back(kPriceChipsSelectedDiscount);
symbols.emplace_back(kUGCRatingBadgeName);
base::SortUnique(symbols);

View file

@ -30,7 +30,7 @@ public:
drape_ptr<TitlesInfo> GetTitleDecl() const override;
int GetMinTitleZoom() const override;
df::DepthLayer GetDepthLayer() const override;
drape_ptr<SymbolNameZoomInfo> GetBadgeNames() const override;
drape_ptr<BageInfo> GetBadgeInfo() const override;
drape_ptr<SymbolOffsets> GetSymbolOffsets() const override;
bool GetDepthTestEnabled() const override { return false; }
bool IsMarkAboveText() const override;
@ -66,7 +66,13 @@ protected:
}
bool IsBookingSpecialMark() const;
bool HasGoodRating() const;
bool HasPrice() const;
bool HasPricing() const;
bool HasRating() const;
bool IsUGCMark() const;
bool IsSelected() const;
bool HasSale() const;
std::string GetSymbolName() const;
std::string GetBadgeName() const;
@ -81,6 +87,7 @@ protected:
std::string m_price;
bool m_hasSale = false;
dp::TitleDecl m_titleDecl;
dp::TitleDecl m_badgeTitleDecl;
dp::TitleDecl m_ugcTitleDecl;
bool m_isVisited = false;
bool m_isAvailable = true;

View file

@ -78,7 +78,7 @@ public:
df::DepthLayer GetDepthLayer() const override { return df::DepthLayer::UserMarkLayer; }
drape_ptr<TitlesInfo> GetTitleDecl() const override { return nullptr; }
drape_ptr<ColoredSymbolZoomInfo> GetColoredSymbols() const override { return nullptr; }
drape_ptr<SymbolNameZoomInfo> GetBadgeNames() const override { return nullptr; }
drape_ptr<BageInfo> GetBadgeInfo() const override { return nullptr; }
drape_ptr<SymbolSizes> GetSymbolSizes() const override { return nullptr; }
drape_ptr<SymbolOffsets> GetSymbolOffsets() const override { return nullptr; }
uint16_t GetPriority() const override { return static_cast<uint16_t >(Priority::Default); }