From d03e167bc0484fcf9837da2a84ff2bc220ad0203 Mon Sep 17 00:00:00 2001 From: vng Date: Fri, 21 Sep 2012 02:31:52 +0300 Subject: [PATCH] Framework::GetVisiblePOI is working for symbols now. --- indexer/feature.hpp | 13 +++++++ indexer/index.cpp | 12 +++++++ indexer/index.hpp | 24 +++++++++++-- map/address_finder.cpp | 75 ++++++++++++++++++++++++++++----------- map/draw_processor.cpp | 11 +++--- map/drawer_yg.cpp | 45 ++++++++++++++--------- map/drawer_yg.hpp | 15 +++++--- map/feature_vec_model.hpp | 6 ++-- map/framework.cpp | 48 +++++++++++++------------ map/framework.hpp | 9 +++-- qt/draw_widget.cpp | 22 ++++++++---- yg/overlay_element.cpp | 5 --- yg/overlay_element.hpp | 8 +++-- yg/overlay_renderer.cpp | 21 ++++++----- yg/overlay_renderer.hpp | 1 + 15 files changed, 215 insertions(+), 100 deletions(-) diff --git a/indexer/feature.hpp b/indexer/feature.hpp index 31faeb29b2..a9c3a3b761 100644 --- a/indexer/feature.hpp +++ b/indexer/feature.hpp @@ -148,9 +148,22 @@ class FeatureType : public FeatureBase { typedef FeatureBase base_type; + /// @name This params define unique id for feature. + //@{ + size_t m_mwmID; + uint32_t m_offset; + //@} + public: void Deserialize(feature::LoaderBase * pLoader, BufferT buffer); + inline void SetID(size_t mwmID, uint32_t offset) + { + m_mwmID = mwmID; + m_offset = offset; + } + inline pair GetID() const { return make_pair(m_mwmID, m_offset); } + /// @name Parse functions. Do simple dispatching to m_pLoader. //@{ void ParseHeader2() const; diff --git a/indexer/index.cpp b/indexer/index.cpp index 69de20d718..cd6989798d 100644 --- a/indexer/index.cpp +++ b/indexer/index.cpp @@ -120,3 +120,15 @@ void Index::UpdateMwmInfo(MwmId id) break; } } + +Index::FeaturesLoaderGuard::FeaturesLoaderGuard(Index const & parent, MwmId id) + : m_lock(parent, id), + /// @note This guard is suitable when mwm is loaded + m_vector(m_lock.GetValue()->m_cont, m_lock.GetValue()->GetHeader()) +{ +} + +void Index::FeaturesLoaderGuard::GetFeature(uint32_t offset, FeatureType & ft) +{ + m_vector.Get(offset, ft); +} diff --git a/indexer/index.hpp b/indexer/index.hpp index ff3afbd85f..669624f6e9 100644 --- a/indexer/index.hpp +++ b/indexer/index.hpp @@ -77,6 +77,16 @@ public: ForEachInIntervals(f, 2, m2::RectD::GetInfiniteRect(), scale); } + /// Guard for loading features from particular MWM by demand. + class FeaturesLoaderGuard + { + MwmLock m_lock; + FeaturesVector m_vector; + public: + FeaturesLoaderGuard(Index const & parent, MwmId id); + void GetFeature(uint32_t offset, FeatureType & ft); + }; + private: template @@ -85,16 +95,24 @@ private: FeaturesVector const & m_V; F & m_F; unordered_set & m_offsets; + MwmId m_mwmID; public: - ReadFeatureFunctor(FeaturesVector const & v, F & f, unordered_set & offsets) - : m_V(v), m_F(f), m_offsets(offsets) {} + ReadFeatureFunctor(FeaturesVector const & v, F & f, + unordered_set & offsets, MwmId mwmID) + : m_V(v), m_F(f), m_offsets(offsets), m_mwmID(mwmID) + { + } + void operator() (uint32_t offset) const { if (m_offsets.insert(offset).second) { FeatureType feature; + m_V.Get(offset, feature); + feature.SetID(m_mwmID, offset); + m_F(feature); } } @@ -125,7 +143,7 @@ private: // iterate through intervals unordered_set offsets; - ReadFeatureFunctor f1(fv, f, offsets); + ReadFeatureFunctor f1(fv, f, offsets, id); for (size_t i = 0; i < interval.size(); ++i) { index.ForEachInIntervalAndScale(f1, interval[i].first, interval[i].second, scale); diff --git a/map/address_finder.cpp b/map/address_finder.cpp index 5b096cdfb2..3337259333 100644 --- a/map/address_finder.cpp +++ b/map/address_finder.cpp @@ -425,6 +425,21 @@ namespace }; } +namespace +{ + /// Global instance for type checker. + /// @todo Possible need to add synhronization. + typedef DoGetAddressBase::TypeChecker CheckerT; + CheckerT * g_checker = 0; + + CheckerT & GetChecker() + { + if (g_checker == 0) + g_checker = new CheckerT(); + return *g_checker; + } +} + void Framework::GetAddressInfo(m2::PointD const & pt, AddressInfo & info) const { info.Clear(); @@ -436,39 +451,59 @@ void Framework::GetAddressInfo(m2::PointD const & pt, AddressInfo & info) const return; } - int scale = scales::GetUpperScale(); + int const scale = scales::GetUpperScale(); double addressR[] = { 15.0, // radius to search point POI's 100.0, // radius to search street names 5.0 // radius to search building numbers (POI's) }; - double const localityR = 20000.0; // radius to search localities - static DoGetAddressBase::TypeChecker checker; + // pass maximum value for all addressR + m2::RectD const rect = MercatorBounds::RectByCenterXYAndSizeInMeters(pt, addressR[1]); + DoGetAddressInfo getAddress(pt, scale, GetChecker(), addressR); - // first of all - get an address + m_model.ForEachFeature(rect, getAddress, scale); + + getAddress.FillAddress(GetSearchEngine(), info); + + GetLocality(pt, info); +} + +void Framework::GetAddressInfo(FeatureType const & ft, m2::PointD const & pt, AddressInfo & info) const +{ + info.Clear(); + + info.m_country = GetCountryName(pt); + if (info.m_country.empty()) { - // pass maximum value for all addressR - m2::RectD const rect = MercatorBounds::RectByCenterXYAndSizeInMeters(pt, addressR[1]); - DoGetAddressInfo getAddress(pt, scale, checker, addressR); - - m_model.ForEachFeature(rect, getAddress, scale); - - getAddress.FillAddress(GetSearchEngine(), info); + LOG(LINFO, ("Can't find region for point ", pt)); + return; } - // now - get the locality - { - scale = checker.GetLocalitySearchScale(); - LOG(LDEBUG, ("Locality scale = ", scale)); + double const inf = numeric_limits::max(); + double addressR[] = { inf, inf, inf }; - m2::RectD const rect = MercatorBounds::RectByCenterXYAndSizeInMeters(pt, localityR); - DoGetLocality getLocality(pt, scale, checker, rect); + DoGetAddressInfo getAddress(pt, scales::GetUpperScale(), GetChecker(), addressR); + getAddress(ft); + getAddress.FillAddress(GetSearchEngine(), info); - m_model.ForEachFeature(rect, getLocality, scale); + GetLocality(pt, info); +} - getLocality.FillLocality(info, *this); - } +void Framework::GetLocality(m2::PointD const & pt, AddressInfo & info) const +{ + CheckerT & checker = GetChecker(); + + int const scale = checker.GetLocalitySearchScale(); + LOG(LDEBUG, ("Locality scale = ", scale)); + + // radius to search localities + m2::RectD const rect = MercatorBounds::RectByCenterXYAndSizeInMeters(pt, 20000.0); + DoGetLocality getLocality(pt, scale, checker, rect); + + m_model.ForEachFeature(rect, getLocality, scale); + + getLocality.FillLocality(info, *this); } string Framework::AddressInfo::FormatAddress() const diff --git a/map/draw_processor.cpp b/map/draw_processor.cpp index 370ed2fb90..f34f342941 100644 --- a/map/draw_processor.cpp +++ b/map/draw_processor.cpp @@ -279,8 +279,11 @@ namespace fwork else m_hasNonCoast = true; - if (GetDrawer()->screen()->renderState()) - GetDrawer()->screen()->renderState()->m_isEmptyModelCurrent = IsEmptyDrawing(); + { + yg::gl::RenderState * pState = GetDrawer()->screen()->renderState().get(); + if (pState) + pState->m_isEmptyModelCurrent = IsEmptyDrawing(); + } // remove duplicating identical drawing keys PreProcessKeys(keys); @@ -324,7 +327,7 @@ namespace fwork intName.clear(); } - shared_ptr ptr(new di::DrawInfo( + scoped_ptr ptr(new di::DrawInfo( defaultName, intName, f.GetRoadNumber(), @@ -415,7 +418,7 @@ namespace fwork } if (isExist) - pDrawer->Draw(ptr.get(), rules.data(), count); + pDrawer->Draw(ptr.get(), rules.data(), count, f.GetID()); return true; } diff --git a/map/drawer_yg.cpp b/map/drawer_yg.cpp index a236cef229..adaa7dc603 100644 --- a/map/drawer_yg.cpp +++ b/map/drawer_yg.cpp @@ -109,12 +109,14 @@ void DrawerYG::onSize(int w, int h) m_pScreen->onSize(w, h); } -void DrawerYG::drawSymbol(m2::PointD const & pt, string const & symbolName, yg::EPosition pos, int depth) +void DrawerYG::drawSymbol(m2::PointD const & pt, string const & symbolName, + yg::EPosition pos, int depth) { - m_pScreen->drawSymbol(pt, symbolName.c_str(), pos, depth); + m_pScreen->drawSymbol(pt, symbolName, pos, depth); } -void DrawerYG::drawCircle(m2::PointD const & pt, rule_ptr_t pRule, yg::EPosition pos, int depth) +void DrawerYG::drawCircle(m2::PointD const & pt, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id) { yg::CircleInfo ci; ConvertStyle(pRule->GetCircle(), m_visualScale, ci); @@ -122,12 +124,21 @@ void DrawerYG::drawCircle(m2::PointD const & pt, rule_ptr_t pRule, yg::EPosition m_pScreen->drawCircle(pt, ci, pos, depth); } -void DrawerYG::drawSymbol(m2::PointD const & pt, rule_ptr_t pRule, yg::EPosition pos, int depth) +void DrawerYG::drawSymbol(m2::PointD const & pt, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id) { string name; ConvertStyle(pRule->GetSymbol(), name); - m_pScreen->drawSymbol(pt, name, pos, depth); + yg::SymbolElement::Params params; + params.m_depth = depth; + params.m_position = pos; + params.m_pivot = pt; + params.m_symbolName.swap(name); + params.m_userInfo.m_mwmID = id.first; + params.m_userInfo.m_offset = id.second; + + m_pScreen->drawSymbol(params); } void DrawerYG::drawPath(di::PathInfo const & info, di::DrawRule const * rules, size_t count) @@ -206,7 +217,8 @@ bool DrawerYG::filter_text_size(rule_ptr_t pRule) const } } -void DrawerYG::drawText(m2::PointD const & pt, di::DrawInfo const * pInfo, rule_ptr_t pRule, yg::EPosition pos, int depth) +void DrawerYG::drawText(m2::PointD const & pt, di::DrawInfo const * pInfo, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id) { yg::FontDesc font; ConvertStyle(pRule->GetCaption(0), m_visualScale, font); @@ -279,7 +291,8 @@ void DrawerYG::SetScale(int level) m_level = level; } -void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size_t count) +void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size_t count, + FeatureID const & id) { buffer_vector pathRules; @@ -306,8 +319,6 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size } } - bool isNumber = true; - for (size_t i = 0; i < count; ++i) { rule_ptr_t pRule = rules[i].m_rule; @@ -319,6 +330,8 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size if (!isCaption) { + double const sm = 2.0 * m_visualScale; + // draw area if (isArea) { @@ -330,7 +343,7 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size if (isFill) drawArea(i->m_path, pRule, depth); else if (hasSym) - drawSymbol(i->GetCenter() + m2::PointD(0, 2 * m_visualScale), pRule, yg::EPosUnder, depth); + drawSymbol(i->GetCenter() + m2::PointD(0.0, sm), pRule, yg::EPosUnder, depth, id); } } @@ -338,9 +351,9 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size if (!isPath && !isArea && ((pRule->GetType() & drule::node) != 0)) { if (hasSymbol) - drawSymbol(pInfo->m_point + m2::PointD(0, 2 * m_visualScale), pRule, yg::EPosUnder, depth); - if (isCircle) - drawCircle(pInfo->m_point + m2::PointD(0, 2 * m_visualScale), pRule, yg::EPosUnder, depth); + drawSymbol(pInfo->m_point + m2::PointD(0.0, sm), pRule, yg::EPosUnder, depth, id); + else if (isCircle) + drawCircle(pInfo->m_point + m2::PointD(0.0, sm), pRule, yg::EPosUnder, depth, id); } } else @@ -353,7 +366,7 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size if (isArea/* && isN*/) { for (list::const_iterator i = pInfo->m_areas.begin(); i != pInfo->m_areas.end(); ++i) - drawText(i->GetCenter(), pInfo, pRule, yg::EPosAbove, depth); + drawText(i->GetCenter(), pInfo, pRule, yg::EPosAbove, depth, id); } // draw way name @@ -366,13 +379,13 @@ void DrawerYG::Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size // draw point text isN = ((pRule->GetType() & drule::node) != 0); if (!isPath && !isArea && isN) - drawText(pInfo->m_point, pInfo, pRule, yg::EPosAbove, depth); + drawText(pInfo->m_point, pInfo, pRule, yg::EPosAbove, depth, id); } } } // draw road numbers - if (isNumber && isPath && !pInfo->m_road.empty() && m_level >= 12) + if (isPath && !pInfo->m_road.empty() && m_level >= 12) { for (list::const_iterator i = pInfo->m_pathes.begin(); i != pInfo->m_pathes.end(); ++i) drawPathNumber(*i, pInfo); diff --git a/map/drawer_yg.hpp b/map/drawer_yg.hpp index 445695fbad..a4d9b36358 100644 --- a/map/drawer_yg.hpp +++ b/map/drawer_yg.hpp @@ -78,13 +78,18 @@ class DrawerYG static void ClearSkinPage(size_t threadID, uint8_t pipelineID); + typedef pair FeatureID; + protected: - void drawSymbol(m2::PointD const & pt, rule_ptr_t pRule, yg::EPosition pos, int depth); - void drawCircle(m2::PointD const & pt, rule_ptr_t pRule, yg::EPosition pos, int depth); + void drawSymbol(m2::PointD const & pt, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id); + void drawCircle(m2::PointD const & pt, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id); void drawPath(di::PathInfo const & info, di::DrawRule const * rules, size_t count); void drawArea(vector const & pts, rule_ptr_t pRule, int depth); - void drawText(m2::PointD const & pt, di::DrawInfo const * pInfo, rule_ptr_t pRule, yg::EPosition pos, int depth); + void drawText(m2::PointD const & pt, di::DrawInfo const * pInfo, rule_ptr_t pRule, + yg::EPosition pos, int depth, FeatureID const & id); bool drawPathText(di::PathInfo const & info, di::DrawInfo const * pInfo, rule_ptr_t pRule, int depth); void drawPathNumber(di::PathInfo const & path, di::DrawInfo const * pInfo); @@ -116,7 +121,9 @@ public: double VisualScale() const; void SetScale(int level); - void Draw(di::DrawInfo const * pInfo, di::DrawRule const * rules, size_t count); + void Draw(di::DrawInfo const * pInfo, + di::DrawRule const * rules, size_t count, + FeatureID const & id); bool filter_text_size(rule_ptr_t pRule) const; uint8_t get_text_font_size(rule_ptr_t pRule) const; diff --git a/map/feature_vec_model.hpp b/map/feature_vec_model.hpp index 479a4619a5..bdb294ae87 100644 --- a/map/feature_vec_model.hpp +++ b/map/feature_vec_model.hpp @@ -26,9 +26,7 @@ namespace model private: m2::RectD m_rect; - typedef Index index_t; - - index_t m_multiIndex; + Index m_multiIndex; public: void InitClassificator(); @@ -75,7 +73,7 @@ namespace model } //@} - index_t const & GetIndex() const { return m_multiIndex; } + Index const & GetIndex() const { return m_multiIndex; } m2::RectD GetWorldRect() const; }; } diff --git a/map/framework.cpp b/map/framework.cpp index 83b30687cc..bdbeffb03d 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -1264,8 +1264,7 @@ shared_ptr const GetClosestToPivot(listpivot()); - + double const curDist = pxPoint.SquareLength((*it)->pivot()); if (curDist < dist) { dist = curDist; @@ -1276,42 +1275,47 @@ shared_ptr const GetClosestToPivot(list overlay = m_renderPolicy->GetOverlay(); + m2::PointD const pt = m_navigator.ShiftPoint(pxPoint); + double const halfSize = 12 * m_renderPolicy->VisualScale(); - double halfSize = 12 * m_renderPolicy->VisualScale(); - - list > candidates; + typedef yg::OverlayElement ElementT; + list > candidates; m2::RectD rect(pt.x - halfSize, pt.y - halfSize, pt.x + halfSize, pt.y + halfSize); + m_renderPolicy->GetOverlay()->selectOverlayElements(rect, candidates); - overlay->selectOverlayElements(rect, candidates); - - shared_ptr res; - - res = GetClosestToPivot(candidates, pt); - + shared_ptr res = GetClosestToPivot(candidates, pt); if (res) { - /// TODO : get featureID associated with res. - /// Obtain AddressInfo by this featureID. + ElementT::UserInfo const & ui = res->userInfo(); + if (ui.IsValid()) + { + Index::FeaturesLoaderGuard guard(m_model.GetIndex(), ui.m_mwmID); - return true; + FeatureType ft; + guard.GetFeature(ui.m_offset, ft); + + /// @todo Get correct center point for feature. + m2::PointD const center = (ft.GetFeatureType() == feature::GEOM_POINT ? + ft.GetCenter() : PtoG(pxPoint)); + + GetAddressInfo(ft, center, info); + + pxPivot = GtoP(center); + return true; + } } - else - return false; + + return false; } Navigator & Framework::GetNavigator() diff --git a/map/framework.hpp b/map/framework.hpp index 7956ab70c1..55470af39f 100644 --- a/map/framework.hpp +++ b/map/framework.hpp @@ -292,9 +292,12 @@ public: /// @param[in] pt Point in mercator coordinates. void GetAddressInfo(m2::PointD const & pt, AddressInfo & info) const; - bool GetVisiblePOI(m2::PointD const & pxPoint, - m2::PointD & pv, - AddressInfo & info); +private: + void GetAddressInfo(FeatureType const & ft, m2::PointD const & pt, AddressInfo & info) const; + void GetLocality(m2::PointD const & pt, AddressInfo & info) const; + +public: + bool GetVisiblePOI(m2::PointD const & pxPoint, m2::PointD & pxPivot, AddressInfo & info) const; virtual void BeginPaint(shared_ptr const & e); /// Function for calling from platform dependent-paint function. diff --git a/qt/draw_widget.cpp b/qt/draw_widget.cpp index 0695a5288c..93a6bae616 100644 --- a/qt/draw_widget.cpp +++ b/qt/draw_widget.cpp @@ -343,17 +343,25 @@ namespace qt QPoint const & pt = e->pos(); QMenu menu; - vector types; - m_framework->GetFeatureTypes(m2::PointD(pt.x(), pt.y()), types); + Framework::AddressInfo info; + m2::PointD dummy; + if (m_framework->GetVisiblePOI(m2::PointD(pt.x(), pt.y()), dummy, info)) + { + add_string(menu, "POI"); + } + else + { + vector types; + m_framework->GetFeatureTypes(m2::PointD(pt.x(), pt.y()), types); - for (size_t i = 0; i < types.size(); ++i) - add_string(menu, types[i]); + for (size_t i = 0; i < types.size(); ++i) + add_string(menu, types[i]); + + m_framework->GetAddressInfo(m_framework->PtoG(m2::PointD(pt.x(), pt.y())), info); + } (void)menu.addSeparator(); - Framework::AddressInfo info; - m_framework->GetAddressInfo(m_framework->PtoG(m2::PointD(pt.x(), pt.y())), info); - if (!info.m_name.empty()) add_string(menu, info.m_name); add_string(menu, info.FormatAddress()); diff --git a/yg/overlay_element.cpp b/yg/overlay_element.cpp index d8a1e0ab9d..4f4ede9d8a 100644 --- a/yg/overlay_element.cpp +++ b/yg/overlay_element.cpp @@ -10,11 +10,6 @@ namespace yg : m_pivot(), m_position(yg::EPosAboveRight), m_depth(yg::maxDepth) {} - OverlayElement::UserInfo::UserInfo() - : m_featureID0(0), - m_featureID1(0) - {} - OverlayElement::OverlayElement(Params const & p) : m_pivot(p.m_pivot), m_position(p.m_position), diff --git a/yg/overlay_element.hpp b/yg/overlay_element.hpp index 224ee24e5e..75a86e30a1 100644 --- a/yg/overlay_element.hpp +++ b/yg/overlay_element.hpp @@ -19,9 +19,11 @@ namespace yg struct UserInfo { - uint32_t m_featureID0; - uint32_t m_featureID1; - UserInfo(); + size_t m_mwmID; + uint32_t m_offset; + + UserInfo() : m_mwmID(size_t(-1)) {} + inline bool IsValid() const { return (m_mwmID != size_t(-1)); } }; private: diff --git a/yg/overlay_renderer.cpp b/yg/overlay_renderer.cpp index aa92a81a1b..a88dbeaae9 100644 --- a/yg/overlay_renderer.cpp +++ b/yg/overlay_renderer.cpp @@ -29,19 +29,11 @@ namespace yg { } - void OverlayRenderer::drawSymbol(m2::PointD const & pt, - string const & symbolName, - EPosition pos, - int depth) + void OverlayRenderer::drawSymbol(SymbolElement::Params & params) { if (!m_drawSymbols) return; - SymbolElement::Params params; - params.m_depth = depth; - params.m_position = pos; - params.m_pivot = pt; - params.m_symbolName = symbolName; params.m_skin = skin().get(); shared_ptr oe(new SymbolElement(params)); @@ -54,6 +46,17 @@ namespace yg m_overlay->processOverlayElement(oe); } + void OverlayRenderer::drawSymbol(m2::PointD const & pt, string const & symbolName, EPosition pos, int depth) + { + yg::SymbolElement::Params params; + params.m_depth = depth; + params.m_position = pos; + params.m_pivot = pt; + params.m_symbolName = symbolName; + + drawSymbol(params); + } + void OverlayRenderer::drawCircle(m2::PointD const & pt, yg::CircleInfo const & ci, EPosition pos, diff --git a/yg/overlay_renderer.hpp b/yg/overlay_renderer.hpp index 6e3a23d071..9241ace3b1 100644 --- a/yg/overlay_renderer.hpp +++ b/yg/overlay_renderer.hpp @@ -33,6 +33,7 @@ namespace yg OverlayRenderer(Params const & p); /// Drawing POI symbol + void drawSymbol(SymbolElement::Params & params); void drawSymbol(m2::PointD const & pt, string const & symbolName, EPosition pos, int depth); /// Drawing circle