diff --git a/drape_frontend/apply_feature_functors.cpp b/drape_frontend/apply_feature_functors.cpp index 3d7f2c35bb..5140c338fc 100644 --- a/drape_frontend/apply_feature_functors.cpp +++ b/drape_frontend/apply_feature_functors.cpp @@ -404,12 +404,12 @@ BaseApplyFeature::BaseApplyFeature(TileKey const & tileKey, TInsertShapeFn const void BaseApplyFeature::ExtractCaptionParams(CaptionDefProto const * primaryProto, CaptionDefProto const * secondaryProto, - float depth, TextViewParams & params) const + double depth, TextViewParams & params) const { dp::FontDecl decl; CaptionDefProtoToFontDecl(primaryProto, decl); - params.m_depth = depth; + params.m_depth = static_cast(depth); params.m_featureId = m_id; auto & titleDecl = params.m_titleDecl; @@ -463,23 +463,20 @@ void ApplyPointFeature::ProcessPointRule(Stylist::TRuleWrapper const & rule) if (!m_hasPoint) return; - drule::BaseRule const * pRule = rule.first; - auto const depth = static_cast(rule.second); - - SymbolRuleProto const * symRule = pRule->GetSymbol(); + SymbolRuleProto const * symRule = rule.m_rule->GetSymbol(); if (symRule != nullptr) { - m_symbolDepth = depth; + m_symbolDepth = rule.m_depth; m_symbolRule = symRule; } - bool const isNode = (pRule->GetType() & drule::node) != 0; - CaptionDefProto const * capRule = pRule->GetCaption(0); + bool const isNode = (rule.m_rule->GetType() & drule::node) != 0; + CaptionDefProto const * capRule = rule.m_rule->GetCaption(0); if (capRule && isNode) { TextViewParams params; params.m_tileCenter = m_tileRect.Center(); - ExtractCaptionParams(capRule, pRule->GetCaption(1), depth, params); + ExtractCaptionParams(capRule, rule.m_rule->GetCaption(1), rule.m_depth, params); params.m_depthLayer = m_depthLayer; params.m_depthTestEnabled = m_depthLayer != DepthLayer::NavigationLayer && m_depthLayer != DepthLayer::OverlayLayer; @@ -575,12 +572,11 @@ void ApplyPointFeature::Finish(ref_ptr texMng) ApplyAreaFeature::ApplyAreaFeature(TileKey const & tileKey, TInsertShapeFn const & insertShape, FeatureID const & id, double currentScaleGtoP, bool isBuilding, bool skipAreaGeometry, float minPosZ, float posZ, int minVisibleScale, - uint8_t rank, CaptionDescription const & captions, bool hatchingArea) + uint8_t rank, CaptionDescription const & captions) : TBase(tileKey, insertShape, id, minVisibleScale, rank, captions, posZ, DepthLayer::OverlayLayer) , m_minPosZ(minPosZ) , m_isBuilding(isBuilding) , m_skipAreaGeometry(skipAreaGeometry) - , m_hatchingArea(hatchingArea) , m_currentScaleGtoP(currentScaleGtoP) {} @@ -743,21 +739,18 @@ void ApplyAreaFeature::CalculateBuildingOutline(bool calculateNormals, BuildingO void ApplyAreaFeature::ProcessAreaRule(Stylist::TRuleWrapper const & rule) { - drule::BaseRule const * pRule = rule.first; - auto const depth = static_cast(rule.second); - - AreaRuleProto const * areaRule = pRule->GetArea(); + AreaRuleProto const * areaRule = rule.m_rule->GetArea(); if (areaRule && !m_triangles.empty()) { AreaViewParams params; params.m_tileCenter = m_tileRect.Center(); - params.m_depth = depth; + params.m_depth = static_cast(rule.m_depth); params.m_color = ToDrapeColor(areaRule->color()); params.m_minVisibleScale = m_minVisibleScale; params.m_rank = m_rank; params.m_minPosZ = m_minPosZ; params.m_posZ = m_posZ; - params.m_hatching = m_hatchingArea; + params.m_hatching = rule.m_hatching; params.m_baseGtoPScale = static_cast(m_currentScaleGtoP); BuildingOutline outline; @@ -837,10 +830,8 @@ bool ApplyLineFeatureGeometry::HasGeometry() const void ApplyLineFeatureGeometry::ProcessLineRule(Stylist::TRuleWrapper const & rule) { ASSERT(HasGeometry(), ()); - drule::BaseRule const * pRule = rule.first; - float const depth = static_cast(rule.second); - LineDefProto const * pLineRule = pRule->GetLine(); + LineDefProto const * pLineRule = rule.m_rule->GetLine(); if (pLineRule == nullptr) return; @@ -879,7 +870,7 @@ void ApplyLineFeatureGeometry::ProcessLineRule(Stylist::TRuleWrapper const & rul PathSymProto const & symRule = pLineRule->pathsym(); PathSymbolViewParams params; params.m_tileCenter = m_tileRect.Center(); - params.m_depth = depth; + params.m_depth = static_cast(rule.m_depth); params.m_minVisibleScale = m_minVisibleScale; params.m_rank = m_rank; params.m_symbolName = symRule.name(); @@ -896,7 +887,7 @@ void ApplyLineFeatureGeometry::ProcessLineRule(Stylist::TRuleWrapper const & rul LineViewParams params; params.m_tileCenter = m_tileRect.Center(); Extract(pLineRule, params); - params.m_depth = depth; + params.m_depth = static_cast(rule.m_depth); params.m_minVisibleScale = m_minVisibleScale; params.m_rank = m_rank; params.m_baseGtoPScale = m_currentScaleGtoP; @@ -934,18 +925,17 @@ void ApplyLineFeatureAdditional::ProcessLineRule(Stylist::TRuleWrapper const & r if (m_clippedSplines.empty()) return; - drule::BaseRule const * pRule = rule.first; - m_depth = static_cast(rule.second); + m_depth = static_cast(rule.m_depth); - ShieldRuleProto const * pShieldRule = pRule->GetShield(); + ShieldRuleProto const * pShieldRule = rule.m_rule->GetShield(); if (pShieldRule != nullptr) m_shieldRule = pShieldRule; - bool const isWay = (pRule->GetType() & drule::way) != 0; + bool const isWay = (rule.m_rule->GetType() & drule::way) != 0; if (!isWay) return; - CaptionDefProto const * pCaptionRule = pRule->GetCaption(0); + CaptionDefProto const * pCaptionRule = rule.m_rule->GetCaption(0); if (pCaptionRule != nullptr && pCaptionRule->height() > 2 && !m_captions.GetMainText().empty()) m_captionRule = pCaptionRule; } diff --git a/drape_frontend/apply_feature_functors.hpp b/drape_frontend/apply_feature_functors.hpp index 314812b7f2..9d10ab39ce 100644 --- a/drape_frontend/apply_feature_functors.hpp +++ b/drape_frontend/apply_feature_functors.hpp @@ -49,7 +49,7 @@ public: protected: void ExtractCaptionParams(CaptionDefProto const * primaryProto, CaptionDefProto const * secondaryProto, - float depth, TextViewParams & params) const; + double depth, TextViewParams & params) const; std::string ExtractHotelInfo() const; TInsertShapeFn m_insertShape; @@ -99,7 +99,7 @@ public: ApplyAreaFeature(TileKey const & tileKey, TInsertShapeFn const & insertShape, FeatureID const & id, double currentScaleGtoP, bool isBuilding, bool skipAreaGeometry, float minPosZ, float posZ, int minVisibleScale, - uint8_t rank, CaptionDescription const & captions, bool hatchingArea); + uint8_t rank, CaptionDescription const & captions); using TBase::operator (); @@ -150,7 +150,6 @@ private: float const m_minPosZ; bool const m_isBuilding; bool const m_skipAreaGeometry; - bool const m_hatchingArea; double const m_currentScaleGtoP; }; diff --git a/drape_frontend/rule_drawer.cpp b/drape_frontend/rule_drawer.cpp index e324a0e089..813bd352d6 100644 --- a/drape_frontend/rule_drawer.cpp +++ b/drape_frontend/rule_drawer.cpp @@ -262,8 +262,6 @@ void RuleDrawer::ProcessAreaStyle(FeatureType & f, Stylist const & s, m2::PointD featureCenter; - bool hatchingArea = false; - float areaHeight = 0.0f; float areaMinHeight = 0.0f; if (is3dBuilding) @@ -280,11 +278,6 @@ void RuleDrawer::ProcessAreaStyle(FeatureType & f, Stylist const & s, rectMercator = mercator::MetersToXY(lon, lat, minHeightInMeters); areaMinHeight = static_cast((rectMercator.SizeX() + rectMercator.SizeY()) * 0.5); } - else - { - // Put here in case if we have hatching and 3D-building at the same time? :) - hatchingArea = s.m_isHatchingArea; - } bool applyPointStyle = s.m_pointStyleExists; if (applyPointStyle) @@ -301,7 +294,7 @@ void RuleDrawer::ProcessAreaStyle(FeatureType & f, Stylist const & s, m_currentScaleGtoP, isBuilding, m_context->Is3dBuildingsEnabled() && isBuildingOutline, areaMinHeight, areaHeight, minVisibleScale, f.GetRank(), - s.GetCaptionDescription(), hatchingArea); + s.GetCaptionDescription()); f.ForEachTriangle(apply, zoomLevel); if (applyPointStyle) apply(featureCenter, true /* hasArea */); @@ -428,7 +421,7 @@ void RuleDrawer::ProcessPointStyle(FeatureType & f, Stylist const & s, if (CheckCancelled()) return; - s.ForEachRule(bind(&ApplyPointFeature::ProcessPointRule, &apply, _1)); + s.ForEachRule(std::bind(&ApplyPointFeature::ProcessPointRule, &apply, _1)); apply.Finish(m_context->GetTextureManager()); } diff --git a/drape_frontend/stylist.cpp b/drape_frontend/stylist.cpp index 5be9d239c8..fbf395ecaf 100644 --- a/drape_frontend/stylist.cpp +++ b/drape_frontend/stylist.cpp @@ -138,7 +138,7 @@ private: if (lineRule != nullptr && (lineRule->width() < 1e-5 && !lineRule->has_pathsym())) return; - m_rules.emplace_back(std::make_pair(dRule, depth)); + m_rules.push_back({ dRule, depth, key.m_hatching }); } void Init() @@ -274,22 +274,39 @@ bool InitStylist(FeatureType & f, int8_t deviceLang, int const zoomLevel, bool b !ftypes::IsBuildingChecker::Instance()(types)) return false; + Classificator const & cl = classif(); + auto const & hatchingChecker = IsHatchingTerritoryChecker::Instance(); + auto const geomType = types.GetGeomType(); + drule::KeysT keys; - feature::GetDrawRule(types, zoomLevel, keys); + size_t idx = 0; + for (uint32_t t : types) + { + cl.GetObject(t)->GetSuitable(zoomLevel, geomType, keys); + + if (hatchingChecker(t)) + { + while (idx < keys.size()) + { + if (keys[idx].m_type == drule::area) + keys[idx].m_hatching = true; + ++idx; + } + } + else + idx = keys.size(); + } feature::FilterRulesByRuntimeSelector(f, zoomLevel, keys); if (keys.empty()) return false; - s.m_isHatchingArea = IsHatchingTerritoryChecker::Instance()(types); - drule::MakeUnique(keys); - s.m_isCoastline = types.Has(classif().GetCoastType()); - auto const mainGeomType = types.GetGeomType(); + s.m_isCoastline = types.Has(cl.GetCoastType()); - switch (mainGeomType) + switch (geomType) { case feature::GeomType::Point: s.m_pointStyleExists = true; @@ -305,11 +322,11 @@ bool InitStylist(FeatureType & f, int8_t deviceLang, int const zoomLevel, bool b return false; } - Aggregator aggregator(f, mainGeomType, zoomLevel, keys.size()); + Aggregator aggregator(f, geomType, zoomLevel, keys.size()); aggregator.AggregateKeys(keys); CaptionDescription & descr = s.GetCaptionDescriptionImpl(); - descr.Init(f, deviceLang, zoomLevel, mainGeomType, aggregator.m_mainTextType, aggregator.m_auxCaptionFound); + descr.Init(f, deviceLang, zoomLevel, geomType, aggregator.m_mainTextType, aggregator.m_auxCaptionFound); aggregator.AggregateStyleFlags(keys, descr.IsNameExists()); @@ -337,8 +354,8 @@ double GetFeaturePriority(FeatureType & f, int const zoomLevel) double maxPriority = kMinPriority; for (auto const & rule : aggregator.m_rules) { - if (rule.second > maxPriority) - maxPriority = rule.second; + if (rule.m_depth > maxPriority) + maxPriority = rule.m_depth; } return maxPriority; diff --git a/drape_frontend/stylist.hpp b/drape_frontend/stylist.hpp index 3cab59a2fe..249a25ccd6 100644 --- a/drape_frontend/stylist.hpp +++ b/drape_frontend/stylist.hpp @@ -55,12 +55,17 @@ public: bool m_areaStyleExists = false; bool m_lineStyleExists = false; bool m_pointStyleExists = false; - bool m_isHatchingArea = false; public: CaptionDescription const & GetCaptionDescription() const; - using TRuleWrapper = std::pair; + struct TRuleWrapper + { + drule::BaseRule const * m_rule; + double m_depth; + bool m_hatching; + }; + template void ForEachRule(ToDo && toDo) const { for (auto const & r : m_rules) diff --git a/indexer/drawing_rule_def.cpp b/indexer/drawing_rule_def.cpp index af76c65b71..f79f045a58 100644 --- a/indexer/drawing_rule_def.cpp +++ b/indexer/drawing_rule_def.cpp @@ -1,12 +1,12 @@ #include "indexer/drawing_rule_def.hpp" #include -#include - -using namespace std; +#include namespace drule { +using namespace std; + namespace { struct less_key @@ -29,7 +29,14 @@ struct equal_key if (r1.m_type == drule::line) return (r1 == r2); else - return (r1.m_type == r2.m_type); + { + if (r1.m_type == r2.m_type) + { + // Keep several area styles if bigger one (r1) is hatching. + return !(r1.m_type == drule::area && r1.m_hatching); + } + return false; + } } }; } // namespace diff --git a/indexer/drawing_rule_def.hpp b/indexer/drawing_rule_def.hpp index ef03251b93..6f34e25a57 100644 --- a/indexer/drawing_rule_def.hpp +++ b/indexer/drawing_rule_def.hpp @@ -8,12 +8,13 @@ namespace drule class Key { public: - int m_scale; - int m_type; - int m_index; - int m_priority; + int m_scale = -1; + int m_type = -1; + int m_index = -1; + int m_priority = -1; + bool m_hatching = false; - Key() : m_scale(-1), m_type(-1), m_index(-1), m_priority(-1) {} + Key() = default; Key(int s, int t, int i) : m_scale(s), m_type(t), m_index(i), m_priority(-1) {} bool operator==(Key const & r) const diff --git a/map/transit/transit_reader.cpp b/map/transit/transit_reader.cpp index 45ab31a51f..e46019fbcd 100644 --- a/map/transit/transit_reader.cpp +++ b/map/transit/transit_reader.cpp @@ -170,7 +170,7 @@ void ReadTransitTask::Do() { stylist.ForEachRule([&](df::Stylist::TRuleWrapper const & rule) { - auto const * symRule = rule.first->GetSymbol(); + auto const * symRule = rule.m_rule->GetSymbol(); if (symRule != nullptr) featureInfo.m_gateSymbolName = symRule->name(); });